OLD | NEW |
(Empty) | |
| 1 // Copyright 2016 PDFium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. |
| 4 |
| 5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| 6 |
| 7 #include "core/include/fpdfapi/cpdf_dictionary.h" |
| 8 |
| 9 #include "core/include/fpdfapi/cpdf_array.h" |
| 10 #include "core/include/fpdfapi/cpdf_boolean.h" |
| 11 #include "core/include/fpdfapi/cpdf_name.h" |
| 12 #include "core/include/fpdfapi/cpdf_number.h" |
| 13 #include "core/include/fpdfapi/cpdf_reference.h" |
| 14 #include "core/include/fpdfapi/cpdf_stream.h" |
| 15 #include "core/include/fpdfapi/cpdf_string.h" |
| 16 #include "third_party/base/stl_util.h" |
| 17 |
| 18 CPDF_Dictionary::CPDF_Dictionary() {} |
| 19 |
| 20 CPDF_Dictionary::~CPDF_Dictionary() { |
| 21 for (const auto& it : m_Map) |
| 22 it.second->Release(); |
| 23 } |
| 24 |
| 25 CPDF_Object::Type CPDF_Dictionary::GetType() const { |
| 26 return DICTIONARY; |
| 27 } |
| 28 |
| 29 CPDF_Dictionary* CPDF_Dictionary::GetDict() const { |
| 30 // The method should be made non-const if we want to not be const. |
| 31 // See bug #234. |
| 32 return const_cast<CPDF_Dictionary*>(this); |
| 33 } |
| 34 |
| 35 bool CPDF_Dictionary::IsDictionary() const { |
| 36 return true; |
| 37 } |
| 38 |
| 39 CPDF_Dictionary* CPDF_Dictionary::AsDictionary() { |
| 40 return this; |
| 41 } |
| 42 |
| 43 const CPDF_Dictionary* CPDF_Dictionary::AsDictionary() const { |
| 44 return this; |
| 45 } |
| 46 |
| 47 CPDF_Object* CPDF_Dictionary::Clone(FX_BOOL bDirect) const { |
| 48 CPDF_Dictionary* pCopy = new CPDF_Dictionary(); |
| 49 for (const auto& it : *this) |
| 50 pCopy->m_Map.insert(std::make_pair(it.first, it.second->Clone(bDirect))); |
| 51 return pCopy; |
| 52 } |
| 53 |
| 54 CPDF_Object* CPDF_Dictionary::GetElement(const CFX_ByteStringC& key) const { |
| 55 auto it = m_Map.find(key); |
| 56 if (it == m_Map.end()) |
| 57 return nullptr; |
| 58 return it->second; |
| 59 } |
| 60 CPDF_Object* CPDF_Dictionary::GetElementValue( |
| 61 const CFX_ByteStringC& key) const { |
| 62 CPDF_Object* p = GetElement(key); |
| 63 return p ? p->GetDirect() : nullptr; |
| 64 } |
| 65 |
| 66 CFX_ByteString CPDF_Dictionary::GetStringBy(const CFX_ByteStringC& key) const { |
| 67 CPDF_Object* p = GetElement(key); |
| 68 return p ? p->GetString() : CFX_ByteString(); |
| 69 } |
| 70 |
| 71 CFX_ByteStringC CPDF_Dictionary::GetConstStringBy( |
| 72 const CFX_ByteStringC& key) const { |
| 73 CPDF_Object* p = GetElement(key); |
| 74 return p ? p->GetConstString() : CFX_ByteStringC(); |
| 75 } |
| 76 |
| 77 CFX_WideString CPDF_Dictionary::GetUnicodeTextBy( |
| 78 const CFX_ByteStringC& key) const { |
| 79 CPDF_Object* p = GetElement(key); |
| 80 if (CPDF_Reference* pRef = ToReference(p)) |
| 81 p = pRef->GetDirect(); |
| 82 return p ? p->GetUnicodeText() : CFX_WideString(); |
| 83 } |
| 84 |
| 85 CFX_ByteString CPDF_Dictionary::GetStringBy(const CFX_ByteStringC& key, |
| 86 const CFX_ByteStringC& def) const { |
| 87 CPDF_Object* p = GetElement(key); |
| 88 return p ? p->GetString() : CFX_ByteString(def); |
| 89 } |
| 90 |
| 91 CFX_ByteStringC CPDF_Dictionary::GetConstStringBy( |
| 92 const CFX_ByteStringC& key, |
| 93 const CFX_ByteStringC& def) const { |
| 94 CPDF_Object* p = GetElement(key); |
| 95 return p ? p->GetConstString() : CFX_ByteStringC(def); |
| 96 } |
| 97 |
| 98 int CPDF_Dictionary::GetIntegerBy(const CFX_ByteStringC& key) const { |
| 99 CPDF_Object* p = GetElement(key); |
| 100 return p ? p->GetInteger() : 0; |
| 101 } |
| 102 |
| 103 int CPDF_Dictionary::GetIntegerBy(const CFX_ByteStringC& key, int def) const { |
| 104 CPDF_Object* p = GetElement(key); |
| 105 return p ? p->GetInteger() : def; |
| 106 } |
| 107 |
| 108 FX_FLOAT CPDF_Dictionary::GetNumberBy(const CFX_ByteStringC& key) const { |
| 109 CPDF_Object* p = GetElement(key); |
| 110 return p ? p->GetNumber() : 0; |
| 111 } |
| 112 |
| 113 FX_BOOL CPDF_Dictionary::GetBooleanBy(const CFX_ByteStringC& key, |
| 114 FX_BOOL bDefault) const { |
| 115 CPDF_Object* p = GetElement(key); |
| 116 return ToBoolean(p) ? p->GetInteger() : bDefault; |
| 117 } |
| 118 |
| 119 CPDF_Dictionary* CPDF_Dictionary::GetDictBy(const CFX_ByteStringC& key) const { |
| 120 CPDF_Object* p = GetElementValue(key); |
| 121 if (!p) |
| 122 return nullptr; |
| 123 if (CPDF_Dictionary* pDict = p->AsDictionary()) |
| 124 return pDict; |
| 125 if (CPDF_Stream* pStream = p->AsStream()) |
| 126 return pStream->GetDict(); |
| 127 return nullptr; |
| 128 } |
| 129 |
| 130 CPDF_Array* CPDF_Dictionary::GetArrayBy(const CFX_ByteStringC& key) const { |
| 131 return ToArray(GetElementValue(key)); |
| 132 } |
| 133 |
| 134 CPDF_Stream* CPDF_Dictionary::GetStreamBy(const CFX_ByteStringC& key) const { |
| 135 return ToStream(GetElementValue(key)); |
| 136 } |
| 137 |
| 138 CFX_FloatRect CPDF_Dictionary::GetRectBy(const CFX_ByteStringC& key) const { |
| 139 CFX_FloatRect rect; |
| 140 CPDF_Array* pArray = GetArrayBy(key); |
| 141 if (pArray) |
| 142 rect = pArray->GetRect(); |
| 143 return rect; |
| 144 } |
| 145 |
| 146 CFX_Matrix CPDF_Dictionary::GetMatrixBy(const CFX_ByteStringC& key) const { |
| 147 CFX_Matrix matrix; |
| 148 CPDF_Array* pArray = GetArrayBy(key); |
| 149 if (pArray) |
| 150 matrix = pArray->GetMatrix(); |
| 151 return matrix; |
| 152 } |
| 153 |
| 154 FX_BOOL CPDF_Dictionary::KeyExist(const CFX_ByteStringC& key) const { |
| 155 return pdfium::ContainsKey(m_Map, key); |
| 156 } |
| 157 |
| 158 bool CPDF_Dictionary::IsSignatureDict() const { |
| 159 CPDF_Object* pType = GetElementValue("Type"); |
| 160 if (!pType) |
| 161 pType = GetElementValue("FT"); |
| 162 return pType && pType->GetString() == "Sig"; |
| 163 } |
| 164 |
| 165 void CPDF_Dictionary::SetAt(const CFX_ByteStringC& key, CPDF_Object* pObj) { |
| 166 ASSERT(IsDictionary()); |
| 167 // Avoid 2 constructions of CFX_ByteString. |
| 168 CFX_ByteString key_bytestring = key; |
| 169 auto it = m_Map.find(key_bytestring); |
| 170 if (it == m_Map.end()) { |
| 171 if (pObj) |
| 172 m_Map.insert(std::make_pair(key_bytestring, pObj)); |
| 173 return; |
| 174 } |
| 175 |
| 176 if (it->second == pObj) |
| 177 return; |
| 178 it->second->Release(); |
| 179 |
| 180 if (pObj) |
| 181 it->second = pObj; |
| 182 else |
| 183 m_Map.erase(it); |
| 184 } |
| 185 |
| 186 void CPDF_Dictionary::RemoveAt(const CFX_ByteStringC& key) { |
| 187 auto it = m_Map.find(key); |
| 188 if (it == m_Map.end()) |
| 189 return; |
| 190 |
| 191 it->second->Release(); |
| 192 m_Map.erase(it); |
| 193 } |
| 194 |
| 195 void CPDF_Dictionary::ReplaceKey(const CFX_ByteStringC& oldkey, |
| 196 const CFX_ByteStringC& newkey) { |
| 197 auto old_it = m_Map.find(oldkey); |
| 198 if (old_it == m_Map.end()) |
| 199 return; |
| 200 |
| 201 // Avoid 2 constructions of CFX_ByteString. |
| 202 CFX_ByteString newkey_bytestring = newkey; |
| 203 auto new_it = m_Map.find(newkey_bytestring); |
| 204 if (new_it == old_it) |
| 205 return; |
| 206 |
| 207 if (new_it != m_Map.end()) { |
| 208 new_it->second->Release(); |
| 209 new_it->second = old_it->second; |
| 210 } else { |
| 211 m_Map.insert(std::make_pair(newkey_bytestring, old_it->second)); |
| 212 } |
| 213 m_Map.erase(old_it); |
| 214 } |
| 215 |
| 216 void CPDF_Dictionary::SetAtInteger(const CFX_ByteStringC& key, int i) { |
| 217 SetAt(key, new CPDF_Number(i)); |
| 218 } |
| 219 |
| 220 void CPDF_Dictionary::SetAtName(const CFX_ByteStringC& key, |
| 221 const CFX_ByteString& name) { |
| 222 SetAt(key, new CPDF_Name(name)); |
| 223 } |
| 224 |
| 225 void CPDF_Dictionary::SetAtString(const CFX_ByteStringC& key, |
| 226 const CFX_ByteString& str) { |
| 227 SetAt(key, new CPDF_String(str, FALSE)); |
| 228 } |
| 229 |
| 230 void CPDF_Dictionary::SetAtReference(const CFX_ByteStringC& key, |
| 231 CPDF_IndirectObjectHolder* pDoc, |
| 232 FX_DWORD objnum) { |
| 233 SetAt(key, new CPDF_Reference(pDoc, objnum)); |
| 234 } |
| 235 |
| 236 void CPDF_Dictionary::AddReference(const CFX_ByteStringC& key, |
| 237 CPDF_IndirectObjectHolder* pDoc, |
| 238 FX_DWORD objnum) { |
| 239 SetAt(key, new CPDF_Reference(pDoc, objnum)); |
| 240 } |
| 241 |
| 242 void CPDF_Dictionary::SetAtNumber(const CFX_ByteStringC& key, FX_FLOAT f) { |
| 243 CPDF_Number* pNumber = new CPDF_Number(f); |
| 244 SetAt(key, pNumber); |
| 245 } |
| 246 |
| 247 void CPDF_Dictionary::SetAtBoolean(const CFX_ByteStringC& key, FX_BOOL bValue) { |
| 248 SetAt(key, new CPDF_Boolean(bValue)); |
| 249 } |
| 250 |
| 251 void CPDF_Dictionary::SetAtRect(const CFX_ByteStringC& key, |
| 252 const CFX_FloatRect& rect) { |
| 253 CPDF_Array* pArray = new CPDF_Array; |
| 254 pArray->AddNumber(rect.left); |
| 255 pArray->AddNumber(rect.bottom); |
| 256 pArray->AddNumber(rect.right); |
| 257 pArray->AddNumber(rect.top); |
| 258 SetAt(key, pArray); |
| 259 } |
| 260 |
| 261 void CPDF_Dictionary::SetAtMatrix(const CFX_ByteStringC& key, |
| 262 const CFX_Matrix& matrix) { |
| 263 CPDF_Array* pArray = new CPDF_Array; |
| 264 pArray->AddNumber(matrix.a); |
| 265 pArray->AddNumber(matrix.b); |
| 266 pArray->AddNumber(matrix.c); |
| 267 pArray->AddNumber(matrix.d); |
| 268 pArray->AddNumber(matrix.e); |
| 269 pArray->AddNumber(matrix.f); |
| 270 SetAt(key, pArray); |
| 271 } |
OLD | NEW |