| // Copyright 2016 PDFium Authors. All rights reserved. |
| // 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_shadingpattern.h" |
| |
| #include <algorithm> |
| |
| #include "core/fpdfapi/page/cpdf_docpagedata.h" |
| #include "core/fpdfapi/page/cpdf_function.h" |
| #include "core/fpdfapi/parser/cpdf_array.h" |
| #include "core/fpdfapi/parser/cpdf_dictionary.h" |
| #include "core/fpdfapi/parser/cpdf_document.h" |
| #include "core/fpdfapi/parser/cpdf_object.h" |
| #include "core/fpdfapi/parser/cpdf_stream.h" |
| |
| namespace { |
| |
| ShadingType ToShadingType(int type) { |
| return (type > static_cast<int>(kInvalidShading) && |
| type < static_cast<int>(kMaxShading)) |
| ? static_cast<ShadingType>(type) |
| : kInvalidShading; |
| } |
| |
| } // namespace |
| |
| CPDF_ShadingPattern::CPDF_ShadingPattern(CPDF_Document* pDoc, |
| CPDF_Object* pPatternObj, |
| bool bShading, |
| const CFX_Matrix& parentMatrix) |
| : CPDF_Pattern(pDoc, bShading ? nullptr : pPatternObj, parentMatrix), |
| m_ShadingType(kInvalidShading), |
| m_bShadingObj(bShading), |
| m_pShadingObj(pPatternObj), |
| m_pCS(nullptr), |
| m_pCountedCS(nullptr) { |
| assert(document()); |
| if (!bShading) { |
| m_pShadingObj = pattern_obj()->GetDict()->GetDirectObjectFor("Shading"); |
| SetPatternToFormMatrix(); |
| } |
| } |
| |
| CPDF_ShadingPattern::~CPDF_ShadingPattern() { |
| CPDF_ColorSpace* pCountedCS = m_pCountedCS ? m_pCountedCS->get() : nullptr; |
| if (pCountedCS) { |
| auto* pPageData = document()->GetPageData(); |
| if (pPageData) { |
| m_pCS.Release(); // Give up unowned reference first. |
| pPageData->ReleaseColorSpace(pCountedCS->GetArray()); |
| } |
| } |
| } |
| |
| CPDF_TilingPattern* CPDF_ShadingPattern::AsTilingPattern() { |
| return nullptr; |
| } |
| |
| CPDF_ShadingPattern* CPDF_ShadingPattern::AsShadingPattern() { |
| return this; |
| } |
| |
| bool CPDF_ShadingPattern::Load() { |
| if (m_ShadingType != kInvalidShading) |
| return true; |
| |
| CPDF_Dictionary* pShadingDict = |
| m_pShadingObj ? m_pShadingObj->GetDict() : nullptr; |
| if (!pShadingDict) |
| return false; |
| |
| m_pFunctions.clear(); |
| CPDF_Object* pFunc = pShadingDict->GetDirectObjectFor("Function"); |
| if (pFunc) { |
| if (CPDF_Array* pArray = pFunc->AsArray()) { |
| m_pFunctions.resize(std::min<size_t>(pArray->GetCount(), 4)); |
| for (size_t i = 0; i < m_pFunctions.size(); ++i) |
| m_pFunctions[i] = CPDF_Function::Load(pArray->GetDirectObjectAt(i)); |
| } else { |
| m_pFunctions.push_back(CPDF_Function::Load(pFunc)); |
| } |
| } |
| CPDF_Object* pCSObj = pShadingDict->GetDirectObjectFor("ColorSpace"); |
| if (!pCSObj) |
| return false; |
| |
| CPDF_DocPageData* pDocPageData = document()->GetPageData(); |
| m_pCS = pDocPageData->GetColorSpace(pCSObj, nullptr); |
| if (m_pCS) |
| m_pCountedCS = pDocPageData->FindColorSpacePtr(m_pCS->GetArray()); |
| |
| m_ShadingType = ToShadingType(pShadingDict->GetIntegerFor("ShadingType")); |
| |
| // We expect to have a stream if our shading type is a mesh. |
| if (IsMeshShading() && !ToStream(m_pShadingObj.Get())) |
| return false; |
| |
| return true; |
| } |