| // Copyright 2016 The PDFium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| |
| #include "core/fpdfapi/page/cpdf_allstates.h" |
| |
| #include <algorithm> |
| #include <utility> |
| #include <vector> |
| |
| #include "core/fpdfapi/font/cpdf_font.h" |
| #include "core/fpdfapi/page/cpdf_pageobjectholder.h" |
| #include "core/fpdfapi/page/cpdf_streamcontentparser.h" |
| #include "core/fpdfapi/parser/cpdf_array.h" |
| #include "core/fpdfapi/parser/cpdf_dictionary.h" |
| #include "core/fpdfapi/parser/fpdf_parser_utility.h" |
| #include "core/fxcrt/bytestring.h" |
| #include "core/fxge/cfx_graphstatedata.h" |
| |
| CPDF_AllStates::CPDF_AllStates() = default; |
| |
| CPDF_AllStates::CPDF_AllStates(const CPDF_AllStates& that) = default; |
| |
| CPDF_AllStates& CPDF_AllStates::operator=(const CPDF_AllStates& that) = default; |
| |
| CPDF_AllStates::~CPDF_AllStates() = default; |
| |
| void CPDF_AllStates::SetDefaultStates() { |
| m_GraphicStates.SetDefaultStates(); |
| } |
| |
| void CPDF_AllStates::SetLineDash(const CPDF_Array* pArray, |
| float phase, |
| float scale) { |
| std::vector<float> dashes = ReadArrayElementsToVector(pArray, pArray->size()); |
| mutable_graph_state().SetLineDash(std::move(dashes), phase, scale); |
| } |
| |
| void CPDF_AllStates::ProcessExtGS(const CPDF_Dictionary* pGS, |
| CPDF_StreamContentParser* pParser) { |
| CPDF_DictionaryLocker locker(pGS); |
| for (const auto& it : locker) { |
| RetainPtr<CPDF_Object> pObject = it.second->GetMutableDirect(); |
| if (!pObject) |
| continue; |
| |
| uint32_t key = it.first.GetID(); |
| switch (key) { |
| case FXBSTR_ID('L', 'W', 0, 0): |
| mutable_graph_state().SetLineWidth(pObject->GetNumber()); |
| break; |
| case FXBSTR_ID('L', 'C', 0, 0): |
| mutable_graph_state().SetLineCap( |
| static_cast<CFX_GraphStateData::LineCap>(pObject->GetInteger())); |
| break; |
| case FXBSTR_ID('L', 'J', 0, 0): |
| mutable_graph_state().SetLineJoin( |
| static_cast<CFX_GraphStateData::LineJoin>(pObject->GetInteger())); |
| break; |
| case FXBSTR_ID('M', 'L', 0, 0): |
| mutable_graph_state().SetMiterLimit(pObject->GetNumber()); |
| break; |
| case FXBSTR_ID('D', 0, 0, 0): { |
| const CPDF_Array* pDash = pObject->AsArray(); |
| if (!pDash) |
| break; |
| |
| RetainPtr<const CPDF_Array> pArray = pDash->GetArrayAt(0); |
| if (!pArray) |
| break; |
| |
| SetLineDash(pArray.Get(), pDash->GetFloatAt(1), 1.0f); |
| break; |
| } |
| case FXBSTR_ID('R', 'I', 0, 0): |
| mutable_general_state().SetRenderIntent(pObject->GetString()); |
| break; |
| case FXBSTR_ID('F', 'o', 'n', 't'): { |
| const CPDF_Array* pFont = pObject->AsArray(); |
| if (!pFont) |
| break; |
| |
| mutable_text_state().SetFontSize(pFont->GetFloatAt(1)); |
| mutable_text_state().SetFont( |
| pParser->FindFont(pFont->GetByteStringAt(0))); |
| break; |
| } |
| case FXBSTR_ID('T', 'R', 0, 0): |
| if (pGS->KeyExist("TR2")) { |
| continue; |
| } |
| [[fallthrough]]; |
| case FXBSTR_ID('T', 'R', '2', 0): |
| mutable_general_state().SetTR(!pObject->IsName() ? std::move(pObject) |
| : nullptr); |
| break; |
| case FXBSTR_ID('B', 'M', 0, 0): { |
| const CPDF_Array* pArray = pObject->AsArray(); |
| mutable_general_state().SetBlendMode(pArray ? pArray->GetByteStringAt(0) |
| : pObject->GetString()); |
| if (general_state().GetBlendType() > BlendMode::kMultiply) { |
| pParser->GetPageObjectHolder()->SetBackgroundAlphaNeeded(true); |
| } |
| break; |
| } |
| case FXBSTR_ID('S', 'M', 'a', 's'): { |
| RetainPtr<CPDF_Dictionary> pMaskDict = ToDictionary(pObject); |
| mutable_general_state().SetSoftMask(pMaskDict); |
| if (pMaskDict) |
| mutable_general_state().SetSMaskMatrix( |
| pParser->GetCurStates()->m_CTM); |
| break; |
| } |
| case FXBSTR_ID('C', 'A', 0, 0): |
| mutable_general_state().SetStrokeAlpha( |
| std::clamp(pObject->GetNumber(), 0.0f, 1.0f)); |
| break; |
| case FXBSTR_ID('c', 'a', 0, 0): |
| mutable_general_state().SetFillAlpha( |
| std::clamp(pObject->GetNumber(), 0.0f, 1.0f)); |
| break; |
| case FXBSTR_ID('O', 'P', 0, 0): |
| mutable_general_state().SetStrokeOP(!!pObject->GetInteger()); |
| if (!pGS->KeyExist("op")) |
| mutable_general_state().SetFillOP(!!pObject->GetInteger()); |
| break; |
| case FXBSTR_ID('o', 'p', 0, 0): |
| mutable_general_state().SetFillOP(!!pObject->GetInteger()); |
| break; |
| case FXBSTR_ID('O', 'P', 'M', 0): |
| mutable_general_state().SetOPMode(pObject->GetInteger()); |
| break; |
| case FXBSTR_ID('B', 'G', 0, 0): |
| if (pGS->KeyExist("BG2")) { |
| continue; |
| } |
| [[fallthrough]]; |
| case FXBSTR_ID('B', 'G', '2', 0): |
| mutable_general_state().SetBG(std::move(pObject)); |
| break; |
| case FXBSTR_ID('U', 'C', 'R', 0): |
| if (pGS->KeyExist("UCR2")) { |
| continue; |
| } |
| [[fallthrough]]; |
| case FXBSTR_ID('U', 'C', 'R', '2'): |
| mutable_general_state().SetUCR(std::move(pObject)); |
| break; |
| case FXBSTR_ID('H', 'T', 0, 0): |
| mutable_general_state().SetHT(std::move(pObject)); |
| break; |
| case FXBSTR_ID('F', 'L', 0, 0): |
| mutable_general_state().SetFlatness(pObject->GetNumber()); |
| break; |
| case FXBSTR_ID('S', 'M', 0, 0): |
| mutable_general_state().SetSmoothness(pObject->GetNumber()); |
| break; |
| case FXBSTR_ID('S', 'A', 0, 0): |
| mutable_general_state().SetStrokeAdjust(!!pObject->GetInteger()); |
| break; |
| case FXBSTR_ID('A', 'I', 'S', 0): |
| mutable_general_state().SetAlphaSource(!!pObject->GetInteger()); |
| break; |
| case FXBSTR_ID('T', 'K', 0, 0): |
| mutable_general_state().SetTextKnockout(!!pObject->GetInteger()); |
| break; |
| } |
| } |
| } |
| |
| void CPDF_AllStates::ResetTextPosition() { |
| m_TextLinePos = CFX_PointF(); |
| m_TextPos = CFX_PointF(); |
| } |
| |
| CFX_PointF CPDF_AllStates::GetTransformedTextPosition() const { |
| return m_CTM.Transform(m_TextMatrix.Transform( |
| CFX_PointF(m_TextPos.x, m_TextPos.y + m_TextRise))); |
| } |
| |
| void CPDF_AllStates::MoveTextPoint(const CFX_PointF& point) { |
| m_TextLinePos += point; |
| m_TextPos = m_TextLinePos; |
| } |
| |
| void CPDF_AllStates::MoveTextToNextLine() { |
| m_TextLinePos.y -= m_TextLeading; |
| m_TextPos = m_TextLinePos; |
| } |
| |
| void CPDF_AllStates::IncrementTextPositionX(float value) { |
| m_TextPos.x += value; |
| } |
| |
| void CPDF_AllStates::IncrementTextPositionY(float value) { |
| m_TextPos.y += value; |
| } |