// 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/parser/cpdf_dictionary.h" #include <set> #include <utility> #include "core/fpdfapi/parser/cpdf_array.h" #include "core/fpdfapi/parser/cpdf_boolean.h" #include "core/fpdfapi/parser/cpdf_name.h" #include "core/fpdfapi/parser/cpdf_number.h" #include "core/fpdfapi/parser/cpdf_reference.h" #include "core/fpdfapi/parser/cpdf_stream.h" #include "core/fpdfapi/parser/cpdf_string.h" #include "core/fpdfapi/parser/fpdf_parser_decode.h" #include "third_party/base/logging.h" #include "third_party/base/stl_util.h" CPDF_Dictionary::CPDF_Dictionary() : CPDF_Dictionary(CFX_WeakPtr<CFX_ByteStringPool>()) {} CPDF_Dictionary::CPDF_Dictionary(const CFX_WeakPtr<CFX_ByteStringPool>& pPool) : m_pPool(pPool) {} CPDF_Dictionary::~CPDF_Dictionary() { // Mark the object as deleted so that it will not be deleted again, // and break cyclic references. m_ObjNum = kInvalidObjNum; for (auto& it : m_Map) { if (it.second && it.second->GetObjNum() == kInvalidObjNum) it.second.release(); } } CPDF_Object::Type CPDF_Dictionary::GetType() const { return DICTIONARY; } CPDF_Dictionary* CPDF_Dictionary::GetDict() const { // The method should be made non-const if we want to not be const. // See bug #234. return const_cast<CPDF_Dictionary*>(this); } bool CPDF_Dictionary::IsDictionary() const { return true; } CPDF_Dictionary* CPDF_Dictionary::AsDictionary() { return this; } const CPDF_Dictionary* CPDF_Dictionary::AsDictionary() const { return this; } std::unique_ptr<CPDF_Object> CPDF_Dictionary::Clone() const { return CloneObjectNonCyclic(false); } std::unique_ptr<CPDF_Object> CPDF_Dictionary::CloneNonCyclic( bool bDirect, std::set<const CPDF_Object*>* pVisited) const { pVisited->insert(this); auto pCopy = pdfium::MakeUnique<CPDF_Dictionary>(m_pPool); for (const auto& it : *this) { if (!pdfium::ContainsKey(*pVisited, it.second.get())) { std::set<const CPDF_Object*> visited(*pVisited); if (auto obj = it.second->CloneNonCyclic(bDirect, &visited)) pCopy->m_Map.insert(std::make_pair(it.first, std::move(obj))); } } return std::move(pCopy); } CPDF_Object* CPDF_Dictionary::GetObjectFor(const CFX_ByteString& key) const { auto it = m_Map.find(key); return it != m_Map.end() ? it->second.get() : nullptr; } CPDF_Object* CPDF_Dictionary::GetDirectObjectFor( const CFX_ByteString& key) const { CPDF_Object* p = GetObjectFor(key); return p ? p->GetDirect() : nullptr; } CFX_ByteString CPDF_Dictionary::GetStringFor(const CFX_ByteString& key) const { CPDF_Object* p = GetObjectFor(key); return p ? p->GetString() : CFX_ByteString(); } CFX_WideString CPDF_Dictionary::GetUnicodeTextFor( const CFX_ByteString& key) const { CPDF_Object* p = GetObjectFor(key); if (CPDF_Reference* pRef = ToReference(p)) p = pRef->GetDirect(); return p ? p->GetUnicodeText() : CFX_WideString(); } CFX_ByteString CPDF_Dictionary::GetStringFor(const CFX_ByteString& key, const CFX_ByteString& def) const { CPDF_Object* p = GetObjectFor(key); return p ? p->GetString() : CFX_ByteString(def); } int CPDF_Dictionary::GetIntegerFor(const CFX_ByteString& key) const { CPDF_Object* p = GetObjectFor(key); return p ? p->GetInteger() : 0; } int CPDF_Dictionary::GetIntegerFor(const CFX_ByteString& key, int def) const { CPDF_Object* p = GetObjectFor(key); return p ? p->GetInteger() : def; } float CPDF_Dictionary::GetNumberFor(const CFX_ByteString& key) const { CPDF_Object* p = GetObjectFor(key); return p ? p->GetNumber() : 0; } bool CPDF_Dictionary::GetBooleanFor(const CFX_ByteString& key, bool bDefault) const { CPDF_Object* p = GetObjectFor(key); return ToBoolean(p) ? p->GetInteger() != 0 : bDefault; } CPDF_Dictionary* CPDF_Dictionary::GetDictFor(const CFX_ByteString& key) const { CPDF_Object* p = GetDirectObjectFor(key); if (!p) return nullptr; if (CPDF_Dictionary* pDict = p->AsDictionary()) return pDict; if (CPDF_Stream* pStream = p->AsStream()) return pStream->GetDict(); return nullptr; } CPDF_Array* CPDF_Dictionary::GetArrayFor(const CFX_ByteString& key) const { return ToArray(GetDirectObjectFor(key)); } CPDF_Stream* CPDF_Dictionary::GetStreamFor(const CFX_ByteString& key) const { return ToStream(GetDirectObjectFor(key)); } CFX_FloatRect CPDF_Dictionary::GetRectFor(const CFX_ByteString& key) const { CFX_FloatRect rect; CPDF_Array* pArray = GetArrayFor(key); if (pArray) rect = pArray->GetRect(); return rect; } CFX_Matrix CPDF_Dictionary::GetMatrixFor(const CFX_ByteString& key) const { CFX_Matrix matrix; CPDF_Array* pArray = GetArrayFor(key); if (pArray) matrix = pArray->GetMatrix(); return matrix; } bool CPDF_Dictionary::KeyExist(const CFX_ByteString& key) const { return pdfium::ContainsKey(m_Map, key); } bool CPDF_Dictionary::IsSignatureDict() const { CPDF_Object* pType = GetDirectObjectFor("Type"); if (!pType) pType = GetDirectObjectFor("FT"); return pType && pType->GetString() == "Sig"; } CPDF_Object* CPDF_Dictionary::SetFor(const CFX_ByteString& key, std::unique_ptr<CPDF_Object> pObj) { if (!pObj) { m_Map.erase(key); return nullptr; } ASSERT(pObj->IsInline()); CPDF_Object* pRet = pObj.get(); m_Map[MaybeIntern(key)] = std::move(pObj); return pRet; } void CPDF_Dictionary::ConvertToIndirectObjectFor( const CFX_ByteString& key, CPDF_IndirectObjectHolder* pHolder) { auto it = m_Map.find(key); if (it == m_Map.end() || it->second->IsReference()) return; CPDF_Object* pObj = pHolder->AddIndirectObject(std::move(it->second)); it->second = pdfium::MakeUnique<CPDF_Reference>(pHolder, pObj->GetObjNum()); } void CPDF_Dictionary::RemoveFor(const CFX_ByteString& key) { m_Map.erase(key); } void CPDF_Dictionary::ReplaceKey(const CFX_ByteString& oldkey, const CFX_ByteString& newkey) { auto old_it = m_Map.find(oldkey); if (old_it == m_Map.end()) return; auto new_it = m_Map.find(newkey); if (new_it == old_it) return; m_Map[MaybeIntern(newkey)] = std::move(old_it->second); m_Map.erase(old_it); } void CPDF_Dictionary::SetRectFor(const CFX_ByteString& key, const CFX_FloatRect& rect) { CPDF_Array* pArray = SetNewFor<CPDF_Array>(key); pArray->AddNew<CPDF_Number>(rect.left); pArray->AddNew<CPDF_Number>(rect.bottom); pArray->AddNew<CPDF_Number>(rect.right); pArray->AddNew<CPDF_Number>(rect.top); } void CPDF_Dictionary::SetMatrixFor(const CFX_ByteString& key, const CFX_Matrix& matrix) { CPDF_Array* pArray = SetNewFor<CPDF_Array>(key); pArray->AddNew<CPDF_Number>(matrix.a); pArray->AddNew<CPDF_Number>(matrix.b); pArray->AddNew<CPDF_Number>(matrix.c); pArray->AddNew<CPDF_Number>(matrix.d); pArray->AddNew<CPDF_Number>(matrix.e); pArray->AddNew<CPDF_Number>(matrix.f); } CFX_ByteString CPDF_Dictionary::MaybeIntern(const CFX_ByteString& str) { return m_pPool ? m_pPool->Intern(str) : str; } bool CPDF_Dictionary::WriteTo(CFX_FileBufferArchive* archive, FX_FILESIZE* offset) const { if (archive->AppendString("<<") < 0) return false; *offset += 2; for (const auto& it : *this) { const CFX_ByteString& key = it.first; CPDF_Object* pValue = it.second.get(); if (archive->AppendString("/") < 0) return false; int32_t len = archive->AppendString(PDF_NameEncode(key).AsStringC()); if (len < 0) return false; *offset += len + 1; if (!pValue->IsInline()) { if (archive->AppendString(" ") < 0) return false; len = archive->AppendDWord(pValue->GetObjNum()); if (len < 0) return false; if (archive->AppendString(" 0 R") < 0) return false; *offset += len + 5; } else { if (!pValue->WriteTo(archive, offset)) return false; } } if (archive->AppendString(">>") < 0) return false; *offset += 2; return true; }