blob: 850d7016ef2dab6889d5ebb841d01b785f8f86a5 [file] [log] [blame]
// 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_array.h"
#include <set>
#include <utility>
#include "core/fpdfapi/parser/cpdf_boolean.h"
#include "core/fpdfapi/parser/cpdf_dictionary.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/fxcrt/fx_stream.h"
#include "third_party/base/check.h"
#include "third_party/base/containers/contains.h"
#include "third_party/base/notreached.h"
CPDF_Array::CPDF_Array() = default;
CPDF_Array::CPDF_Array(const WeakPtr<ByteStringPool>& pPool) : m_pPool(pPool) {}
CPDF_Array::~CPDF_Array() {
// Break cycles for cyclic references.
m_ObjNum = kInvalidObjNum;
for (auto& it : m_Objects) {
if (it->GetObjNum() == kInvalidObjNum)
it.Leak();
}
}
CPDF_Object::Type CPDF_Array::GetType() const {
return kArray;
}
CPDF_Array* CPDF_Array::AsMutableArray() {
return this;
}
RetainPtr<CPDF_Object> CPDF_Array::Clone() const {
return CloneObjectNonCyclic(false);
}
RetainPtr<CPDF_Object> CPDF_Array::CloneNonCyclic(
bool bDirect,
std::set<const CPDF_Object*>* pVisited) const {
pVisited->insert(this);
auto pCopy = pdfium::MakeRetain<CPDF_Array>();
for (const auto& pValue : m_Objects) {
if (!pdfium::Contains(*pVisited, pValue.Get())) {
std::set<const CPDF_Object*> visited(*pVisited);
if (auto obj = pValue->CloneNonCyclic(bDirect, &visited))
pCopy->m_Objects.push_back(std::move(obj));
}
}
return pCopy;
}
CFX_FloatRect CPDF_Array::GetRect() const {
CFX_FloatRect rect;
if (m_Objects.size() != 4)
return rect;
rect.left = GetFloatAt(0);
rect.bottom = GetFloatAt(1);
rect.right = GetFloatAt(2);
rect.top = GetFloatAt(3);
return rect;
}
CFX_Matrix CPDF_Array::GetMatrix() const {
if (m_Objects.size() != 6)
return CFX_Matrix();
return CFX_Matrix(GetFloatAt(0), GetFloatAt(1), GetFloatAt(2), GetFloatAt(3),
GetFloatAt(4), GetFloatAt(5));
}
absl::optional<size_t> CPDF_Array::Find(const CPDF_Object* pThat) const {
for (size_t i = 0; i < size(); ++i) {
if (GetDirectObjectAt(i) == pThat)
return i;
}
return absl::nullopt;
}
bool CPDF_Array::Contains(const CPDF_Object* pThat) const {
return Find(pThat).has_value();
}
CPDF_Object* CPDF_Array::GetMutableObjectAtInternal(size_t index) {
return index < m_Objects.size() ? m_Objects[index].Get() : nullptr;
}
const CPDF_Object* CPDF_Array::GetObjectAtInternal(size_t index) const {
return const_cast<CPDF_Array*>(this)->GetMutableObjectAtInternal(index);
}
RetainPtr<CPDF_Object> CPDF_Array::GetMutableObjectAt(size_t index) {
return pdfium::WrapRetain(GetMutableObjectAtInternal(index));
}
RetainPtr<const CPDF_Object> CPDF_Array::GetObjectAt(size_t index) const {
return pdfium::WrapRetain(GetObjectAtInternal(index));
}
RetainPtr<const CPDF_Object> CPDF_Array::GetDirectObjectAt(size_t index) const {
return const_cast<CPDF_Array*>(this)->GetMutableDirectObjectAt(index);
}
RetainPtr<CPDF_Object> CPDF_Array::GetMutableDirectObjectAt(size_t index) {
RetainPtr<CPDF_Object> pObj = GetMutableObjectAt(index);
return pObj ? pObj->GetMutableDirect() : nullptr;
}
ByteString CPDF_Array::GetByteStringAt(size_t index) const {
if (index >= m_Objects.size())
return ByteString();
return m_Objects[index]->GetString();
}
WideString CPDF_Array::GetUnicodeTextAt(size_t index) const {
if (index >= m_Objects.size())
return WideString();
return m_Objects[index]->GetUnicodeText();
}
bool CPDF_Array::GetBooleanAt(size_t index, bool bDefault) const {
if (index >= m_Objects.size())
return bDefault;
const CPDF_Object* p = m_Objects[index].Get();
return ToBoolean(p) ? p->GetInteger() != 0 : bDefault;
}
int CPDF_Array::GetIntegerAt(size_t index) const {
if (index >= m_Objects.size())
return 0;
return m_Objects[index]->GetInteger();
}
float CPDF_Array::GetFloatAt(size_t index) const {
if (index >= m_Objects.size())
return 0;
return m_Objects[index]->GetNumber();
}
RetainPtr<CPDF_Dictionary> CPDF_Array::GetMutableDictAt(size_t index) {
RetainPtr<CPDF_Object> p = GetMutableDirectObjectAt(index);
if (!p)
return nullptr;
CPDF_Dictionary* pDict = p->AsMutableDictionary();
if (pDict)
return pdfium::WrapRetain(pDict);
CPDF_Stream* pStream = p->AsMutableStream();
if (pStream)
return pStream->GetMutableDict();
return nullptr;
}
RetainPtr<const CPDF_Dictionary> CPDF_Array::GetDictAt(size_t index) const {
return const_cast<CPDF_Array*>(this)->GetMutableDictAt(index);
}
RetainPtr<CPDF_Stream> CPDF_Array::GetMutableStreamAt(size_t index) {
return ToStream(GetMutableDirectObjectAt(index));
}
RetainPtr<const CPDF_Stream> CPDF_Array::GetStreamAt(size_t index) const {
return const_cast<CPDF_Array*>(this)->GetMutableStreamAt(index);
}
RetainPtr<CPDF_Array> CPDF_Array::GetMutableArrayAt(size_t index) {
return ToArray(GetMutableDirectObjectAt(index));
}
RetainPtr<const CPDF_Array> CPDF_Array::GetArrayAt(size_t index) const {
return const_cast<CPDF_Array*>(this)->GetMutableArrayAt(index);
}
RetainPtr<const CPDF_Number> CPDF_Array::GetNumberAt(size_t index) const {
return ToNumber(GetObjectAt(index));
}
RetainPtr<const CPDF_String> CPDF_Array::GetStringAt(size_t index) const {
return ToString(GetObjectAt(index));
}
void CPDF_Array::Clear() {
CHECK(!IsLocked());
m_Objects.clear();
}
void CPDF_Array::RemoveAt(size_t index) {
CHECK(!IsLocked());
if (index < m_Objects.size())
m_Objects.erase(m_Objects.begin() + index);
}
void CPDF_Array::ConvertToIndirectObjectAt(size_t index,
CPDF_IndirectObjectHolder* pHolder) {
CHECK(!IsLocked());
if (index >= m_Objects.size())
return;
if (!m_Objects[index] || m_Objects[index]->IsReference())
return;
CPDF_Object* pNew = pHolder->AddIndirectObject(std::move(m_Objects[index]));
m_Objects[index] = pNew->MakeReference(pHolder);
}
void CPDF_Array::SetAt(size_t index, RetainPtr<CPDF_Object> pObj) {
(void)SetAtInternal(index, std::move(pObj));
}
void CPDF_Array::InsertAt(size_t index, RetainPtr<CPDF_Object> pObj) {
(void)InsertAtInternal(index, std::move(pObj));
}
void CPDF_Array::Append(RetainPtr<CPDF_Object> pObj) {
(void)AppendInternal(std::move(pObj));
}
CPDF_Object* CPDF_Array::SetAtInternal(size_t index,
RetainPtr<CPDF_Object> pObj) {
CHECK(!IsLocked());
CHECK(pObj);
CHECK(pObj->IsInline());
if (index >= m_Objects.size())
return nullptr;
CPDF_Object* pRet = pObj.Get();
m_Objects[index] = std::move(pObj);
return pRet;
}
CPDF_Object* CPDF_Array::InsertAtInternal(size_t index,
RetainPtr<CPDF_Object> pObj) {
CHECK(!IsLocked());
CHECK(pObj);
CHECK(pObj->IsInline());
if (index > m_Objects.size())
return nullptr;
CPDF_Object* pRet = pObj.Get();
m_Objects.insert(m_Objects.begin() + index, std::move(pObj));
return pRet;
}
CPDF_Object* CPDF_Array::AppendInternal(RetainPtr<CPDF_Object> pObj) {
CHECK(!IsLocked());
CHECK(pObj);
CHECK(pObj->IsInline());
CPDF_Object* pRet = pObj.Get();
m_Objects.push_back(std::move(pObj));
return pRet;
}
bool CPDF_Array::WriteTo(IFX_ArchiveStream* archive,
const CPDF_Encryptor* encryptor) const {
if (!archive->WriteString("["))
return false;
for (size_t i = 0; i < size(); ++i) {
if (!GetObjectAt(i)->WriteTo(archive, encryptor))
return false;
}
return archive->WriteString("]");
}
CPDF_ArrayLocker::CPDF_ArrayLocker(const CPDF_Array* pArray)
: m_pArray(pArray) {
m_pArray->m_LockCount++;
}
CPDF_ArrayLocker::CPDF_ArrayLocker(RetainPtr<CPDF_Array> pArray)
: m_pArray(std::move(pArray)) {
m_pArray->m_LockCount++;
}
CPDF_ArrayLocker::CPDF_ArrayLocker(RetainPtr<const CPDF_Array> pArray)
: m_pArray(std::move(pArray)) {
m_pArray->m_LockCount++;
}
CPDF_ArrayLocker::~CPDF_ArrayLocker() {
m_pArray->m_LockCount--;
}