Chromium Code Reviews| Index: core/fpdfdoc/cpvt_generateap.cpp |
| diff --git a/core/fpdfdoc/cpvt_generateap.cpp b/core/fpdfdoc/cpvt_generateap.cpp |
| index aaf4b798588f511a3cdaa3ca8198d1ddbc229d46..f0b710b8571c4f9b83c4c30c5aaa89e27388c2b4 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 (pAnnotDict->KeyExist("C")) { |
|
Lei Zhang
2016/08/02 21:08:22
Can we just omit this and instead check the return
jaepark
2016/08/02 22:15:20
Done.
|
| + 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(CPDF_Dictionary* pAnnotDict, |
|
Lei Zhang
2016/08/02 21:08:22
Can |pAnnotDict| be passed by const-ref instead?
jaepark
2016/08/02 22:15:20
Done.
|
| + 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; |
| +} |
| + |
| +void GenerateAndSetAPDict(CPDF_Document* pDoc, |
| + CPDF_Dictionary* pAnnotDict, |
| + const CFX_ByteTextBuf& sAppStream, |
| + CPDF_Dictionary* pExtGStateDict) { |
|
Lei Zhang
2016/08/02 21:08:22
Can you add a comment to say "take ownership of |p
jaepark
2016/08/02 22:15:20
Done.
|
| + 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; |
| } |