Index: core/fpdfdoc/cpvt_generateap.cpp |
diff --git a/core/fpdfdoc/cpvt_generateap.cpp b/core/fpdfdoc/cpvt_generateap.cpp |
index aaf4b798588f511a3cdaa3ca8198d1ddbc229d46..9bde7e6b85f75456c2b32fecb3dcc35dcd759da0 100644 |
--- a/core/fpdfdoc/cpvt_generateap.cpp |
+++ b/core/fpdfdoc/cpvt_generateap.cpp |
@@ -452,6 +452,65 @@ bool GenerateWidgetAP(CPDF_Document* pDoc, |
return true; |
} |
+CFX_ByteString GetColorStringWithDefault(CPDF_Dictionary* pAnnotDict, |
+ const CPVT_Color& crDefaultColor, |
+ PaintOperation nOperation) { |
+ if (CPDF_Array* pColor = pAnnotDict->GetArrayBy("C")) { |
+ CPVT_Color color = CPVT_Color::ParseColor(*pColor); |
+ return CPVT_GenerateAP::GenerateColorAP(color, nOperation); |
+ } |
+ |
+ return CPVT_GenerateAP::GenerateColorAP(crDefaultColor, nOperation); |
+} |
+ |
+CPDF_Dictionary* GenerateExtGStateDict(const CPDF_Dictionary& pAnnotDict, |
+ const CFX_ByteString& sExtGSDictName, |
+ const CFX_ByteString& sBlendMode) { |
+ CPDF_Dictionary* pGSDict = new CPDF_Dictionary; |
+ pGSDict->SetAtString("Type", "ExtGState"); |
+ |
+ FX_FLOAT fOpacity = |
+ pAnnotDict.KeyExist("CA") ? pAnnotDict.GetNumberBy("CA") : 1; |
+ pGSDict->SetAtNumber("CA", fOpacity); |
+ pGSDict->SetAtNumber("ca", fOpacity); |
+ pGSDict->SetAtBoolean("AIS", false); |
+ pGSDict->SetAtString("BM", sBlendMode); |
+ |
+ CPDF_Dictionary* pExtGStateDict = new CPDF_Dictionary; |
+ pExtGStateDict->SetAt(sExtGSDictName, pGSDict); |
+ |
+ return pExtGStateDict; |
+} |
+ |
+// Takes ownership of |pExtGStateDict|. |
+void GenerateAndSetAPDict(CPDF_Document* pDoc, |
+ CPDF_Dictionary* pAnnotDict, |
+ const CFX_ByteTextBuf& sAppStream, |
+ CPDF_Dictionary* pExtGStateDict) { |
+ CPDF_Dictionary* pAPDict = new CPDF_Dictionary; |
+ pAnnotDict->SetAt("AP", pAPDict); |
+ |
+ CPDF_Stream* pNormalStream = new CPDF_Stream(nullptr, 0, nullptr); |
+ int32_t objnum = pDoc->AddIndirectObject(pNormalStream); |
+ pAnnotDict->GetDictBy("AP")->SetAtReference("N", pDoc, objnum); |
+ |
+ pNormalStream->SetData(reinterpret_cast<uint8_t*>(sAppStream.GetBuffer()), |
+ sAppStream.GetSize(), FALSE, FALSE); |
+ |
+ CPDF_Dictionary* pStreamDict = pNormalStream->GetDict(); |
+ pStreamDict->SetAtInteger("FormType", 1); |
+ pStreamDict->SetAtString("Subtype", "Form"); |
+ pStreamDict->SetAtMatrix("Matrix", CFX_Matrix()); |
+ |
+ CFX_FloatRect rect = pAnnotDict->GetRectBy("Rect"); |
+ pStreamDict->SetAtRect("BBox", rect); |
+ |
+ CPDF_Dictionary* pResourceDict = new CPDF_Dictionary; |
+ pResourceDict->SetAt("ExtGState", pExtGStateDict); |
+ |
+ pStreamDict->SetAt("Resources", pResourceDict); |
+} |
+ |
} // namespace |
bool FPDF_GenerateAP(CPDF_Document* pDoc, CPDF_Dictionary* pAnnotDict) { |
@@ -510,16 +569,11 @@ bool CPVT_GenerateAP::GenerateHighlightAP(CPDF_Document* pDoc, |
return false; |
CFX_ByteTextBuf sAppStream; |
- sAppStream << "/GS gs "; |
+ CFX_ByteString sExtGSDictName = "GS"; |
+ sAppStream << "/" << sExtGSDictName << " gs "; |
- if (pAnnotDict->KeyExist("C")) { |
- CPDF_Array* pColor = pAnnotDict->GetArrayBy("C"); |
- CPVT_Color color = CPVT_Color::ParseColor(*pColor); |
- sAppStream << CPVT_GenerateAP::GenerateColorAP(color, PaintOperation::FILL); |
- } else { |
- // Defaults to 0xFFFF00 color for highlight. |
- sAppStream << "1 1 0 rg \n"; |
- } |
+ sAppStream << GetColorStringWithDefault( |
+ pAnnotDict, CPVT_Color(CPVT_Color::kRGB, 1, 1, 0), PaintOperation::FILL); |
CFX_FloatRect rect = pAnnotDict->GetRectBy("Rect"); |
rect.Normalize(); |
@@ -529,39 +583,39 @@ bool CPVT_GenerateAP::GenerateHighlightAP(CPDF_Document* pDoc, |
<< rect.left << " " << rect.bottom << " l " |
<< "h f\n"; |
- CPDF_Dictionary* pAPDict = new CPDF_Dictionary; |
- pAnnotDict->SetAt("AP", pAPDict); |
+ CPDF_Dictionary* pExtGStateDict = |
+ GenerateExtGStateDict(*pAnnotDict, sExtGSDictName, "Multiply"); |
+ GenerateAndSetAPDict(pDoc, pAnnotDict, sAppStream, pExtGStateDict); |
- CPDF_Stream* pNormalStream = new CPDF_Stream(nullptr, 0, nullptr); |
- int32_t objnum = pDoc->AddIndirectObject(pNormalStream); |
- pAnnotDict->GetDictBy("AP")->SetAtReference("N", pDoc, objnum); |
- |
- pNormalStream->SetData(reinterpret_cast<uint8_t*>(sAppStream.GetBuffer()), |
- sAppStream.GetSize(), FALSE, FALSE); |
+ return true; |
+} |
- CPDF_Dictionary* pStreamDict = pNormalStream->GetDict(); |
- pStreamDict->SetAtInteger("FormType", 1); |
- pStreamDict->SetAtString("Subtype", "Form"); |
- pStreamDict->SetAtMatrix("Matrix", CFX_Matrix()); |
- pStreamDict->SetAtRect("BBox", rect); |
+bool CPVT_GenerateAP::GenerateUnderlineAP(CPDF_Document* pDoc, |
+ CPDF_Dictionary* pAnnotDict) { |
+ // If AP dictionary exists, we use the appearance defined in the |
+ // existing AP dictionary. |
+ if (pAnnotDict->KeyExist("AP")) |
+ return false; |
- CPDF_Dictionary* pGSDict = new CPDF_Dictionary; |
- pGSDict->SetAtString("Type", "ExtGState"); |
+ CFX_ByteTextBuf sAppStream; |
+ CFX_ByteString sExtGSDictName = "GS"; |
+ sAppStream << "/" << sExtGSDictName << " gs "; |
- FX_FLOAT fOpacity = |
- pAnnotDict->KeyExist("CA") ? pAnnotDict->GetNumberBy("CA") : 1; |
- pGSDict->SetAtNumber("ca", fOpacity); |
- pGSDict->SetAtNumber("CA", fOpacity); |
- pGSDict->SetAtBoolean("AIS", false); |
- pGSDict->SetAtString("BM", "Multiply"); |
+ sAppStream << GetColorStringWithDefault(pAnnotDict, |
+ CPVT_Color(CPVT_Color::kRGB, 0, 0, 0), |
+ PaintOperation::STROKE); |
- CPDF_Dictionary* pExtGStateDict = new CPDF_Dictionary; |
- pExtGStateDict->SetAt("GS", pGSDict); |
+ CFX_FloatRect rect = pAnnotDict->GetRectBy("Rect"); |
+ rect.Normalize(); |
- CPDF_Dictionary* pResourceDict = new CPDF_Dictionary; |
- pResourceDict->SetAt("ExtGState", pExtGStateDict); |
+ FX_FLOAT fLineWidth = 1.0; |
+ sAppStream << fLineWidth << " w " << rect.left << " " |
+ << rect.bottom + fLineWidth << " m " << rect.right << " " |
+ << rect.bottom + fLineWidth << " l S\n"; |
- pStreamDict->SetAt("Resources", pResourceDict); |
+ CPDF_Dictionary* pExtGStateDict = |
+ GenerateExtGStateDict(*pAnnotDict, sExtGSDictName, "Normal"); |
+ GenerateAndSetAPDict(pDoc, pAnnotDict, sAppStream, pExtGStateDict); |
return true; |
} |