blob: 172e128cd57e07be361c5832e2bfcb8a2349ad57 [file] [log] [blame]
// 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_contentmarks.h"
#include <algorithm>
#include <utility>
#include "core/fpdfapi/parser/cpdf_dictionary.h"
#include "core/fxcrt/check_op.h"
CPDF_ContentMarks::CPDF_ContentMarks() = default;
CPDF_ContentMarks::~CPDF_ContentMarks() = default;
std::unique_ptr<CPDF_ContentMarks> CPDF_ContentMarks::Clone() {
auto result = std::make_unique<CPDF_ContentMarks>();
if (mark_data_) {
result->mark_data_ = pdfium::MakeRetain<MarkData>(*mark_data_);
}
return result;
}
size_t CPDF_ContentMarks::CountItems() const {
return mark_data_ ? mark_data_->CountItems() : 0;
}
bool CPDF_ContentMarks::ContainsItem(const CPDF_ContentMarkItem* pItem) const {
return mark_data_ && mark_data_->ContainsItem(pItem);
}
CPDF_ContentMarkItem* CPDF_ContentMarks::GetItem(size_t index) {
return mark_data_->GetItem(index);
}
const CPDF_ContentMarkItem* CPDF_ContentMarks::GetItem(size_t index) const {
return mark_data_->GetItem(index);
}
int CPDF_ContentMarks::GetMarkedContentID() const {
return mark_data_ ? mark_data_->GetMarkedContentID() : -1;
}
void CPDF_ContentMarks::AddMark(ByteString name) {
EnsureMarkDataExists();
mark_data_->AddMark(std::move(name));
}
void CPDF_ContentMarks::AddMarkWithDirectDict(
ByteString name,
RetainPtr<CPDF_Dictionary> pDict) {
EnsureMarkDataExists();
mark_data_->AddMarkWithDirectDict(std::move(name), std::move(pDict));
}
void CPDF_ContentMarks::AddMarkWithPropertiesHolder(
const ByteString& name,
RetainPtr<CPDF_Dictionary> pDict,
const ByteString& property_name) {
EnsureMarkDataExists();
mark_data_->AddMarkWithPropertiesHolder(name, std::move(pDict),
property_name);
}
bool CPDF_ContentMarks::RemoveMark(CPDF_ContentMarkItem* pMarkItem) {
return mark_data_ && mark_data_->RemoveMark(pMarkItem);
}
void CPDF_ContentMarks::EnsureMarkDataExists() {
if (!mark_data_) {
mark_data_ = pdfium::MakeRetain<MarkData>();
}
}
size_t CPDF_ContentMarks::FindFirstDifference(
const CPDF_ContentMarks* other) const {
if (mark_data_ == other->mark_data_) {
return CountItems();
}
size_t min_len = std::min(CountItems(), other->CountItems());
for (size_t i = 0; i < min_len; ++i) {
if (GetItem(i) != other->GetItem(i)) {
return i;
}
}
return min_len;
}
CPDF_ContentMarks::MarkData::MarkData() = default;
CPDF_ContentMarks::MarkData::MarkData(const MarkData& src)
: marks_(src.marks_) {}
CPDF_ContentMarks::MarkData::~MarkData() = default;
size_t CPDF_ContentMarks::MarkData::CountItems() const {
return marks_.size();
}
bool CPDF_ContentMarks::MarkData::ContainsItem(
const CPDF_ContentMarkItem* pItem) const {
for (const auto& pMark : marks_) {
if (pMark == pItem) {
return true;
}
}
return false;
}
CPDF_ContentMarkItem* CPDF_ContentMarks::MarkData::GetItem(size_t index) {
CHECK_LT(index, marks_.size());
return marks_[index].Get();
}
const CPDF_ContentMarkItem* CPDF_ContentMarks::MarkData::GetItem(
size_t index) const {
CHECK_LT(index, marks_.size());
return marks_[index].Get();
}
int CPDF_ContentMarks::MarkData::GetMarkedContentID() const {
for (const auto& pMark : marks_) {
RetainPtr<const CPDF_Dictionary> pDict = pMark->GetParam();
if (pDict && pDict->KeyExist("MCID")) {
return pDict->GetIntegerFor("MCID");
}
}
return -1;
}
void CPDF_ContentMarks::MarkData::AddMark(ByteString name) {
auto pItem = pdfium::MakeRetain<CPDF_ContentMarkItem>(std::move(name));
marks_.push_back(pItem);
}
void CPDF_ContentMarks::MarkData::AddMarkWithDirectDict(
ByteString name,
RetainPtr<CPDF_Dictionary> pDict) {
auto pItem = pdfium::MakeRetain<CPDF_ContentMarkItem>(std::move(name));
pItem->SetDirectDict(ToDictionary(pDict->Clone()));
marks_.push_back(pItem);
}
void CPDF_ContentMarks::MarkData::AddMarkWithPropertiesHolder(
const ByteString& name,
RetainPtr<CPDF_Dictionary> pDict,
const ByteString& property_name) {
auto pItem = pdfium::MakeRetain<CPDF_ContentMarkItem>(name);
pItem->SetPropertiesHolder(std::move(pDict), property_name);
marks_.push_back(std::move(pItem));
}
bool CPDF_ContentMarks::MarkData::RemoveMark(CPDF_ContentMarkItem* pMarkItem) {
for (auto it = marks_.begin(); it != marks_.end(); ++it) {
if (*it == pMarkItem) {
marks_.erase(it);
return true;
}
}
return false;
}