blob: 671a4975268d0b108af2035ea59254297a6c3462 [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/page/cpdf_pageobjectholder.h"
#include <algorithm>
#include <utility>
#include "constants/transparency.h"
#include "core/fpdfapi/page/cpdf_allstates.h"
#include "core/fpdfapi/page/cpdf_contentparser.h"
#include "core/fpdfapi/page/cpdf_pageobject.h"
#include "core/fpdfapi/parser/cpdf_dictionary.h"
#include "core/fpdfapi/parser/cpdf_document.h"
#include "core/fxcrt/fx_extension.h"
#include "core/fxcrt/stl_util.h"
#include "third_party/base/check.h"
#include "third_party/base/check_op.h"
bool GraphicsData::operator<(const GraphicsData& other) const {
if (!FXSYS_SafeEQ(fillAlpha, other.fillAlpha))
return FXSYS_SafeLT(fillAlpha, other.fillAlpha);
if (!FXSYS_SafeEQ(strokeAlpha, other.strokeAlpha))
return FXSYS_SafeLT(strokeAlpha, other.strokeAlpha);
return blendType < other.blendType;
}
bool FontData::operator<(const FontData& other) const {
if (baseFont != other.baseFont)
return baseFont < other.baseFont;
return type < other.type;
}
CPDF_PageObjectHolder::CPDF_PageObjectHolder(CPDF_Document* pDoc,
CPDF_Dictionary* pDict,
CPDF_Dictionary* pPageResources,
CPDF_Dictionary* pResources)
: m_pPageResources(pPageResources),
m_pResources(pResources),
m_pDict(pDict),
m_pDocument(pDoc) {
DCHECK(m_pDict);
}
CPDF_PageObjectHolder::~CPDF_PageObjectHolder() = default;
bool CPDF_PageObjectHolder::IsPage() const {
return false;
}
void CPDF_PageObjectHolder::StartParse(
std::unique_ptr<CPDF_ContentParser> pParser) {
DCHECK_EQ(m_ParseState, ParseState::kNotParsed);
m_pParser = std::move(pParser);
m_ParseState = ParseState::kParsing;
}
void CPDF_PageObjectHolder::ContinueParse(PauseIndicatorIface* pPause) {
if (m_ParseState == ParseState::kParsed)
return;
DCHECK_EQ(m_ParseState, ParseState::kParsing);
if (m_pParser->Continue(pPause))
return;
m_ParseState = ParseState::kParsed;
m_pDocument->IncrementParsedPageCount();
if (m_pParser->GetCurStates())
m_LastCTM = m_pParser->GetCurStates()->m_CTM;
m_pParser.reset();
}
void CPDF_PageObjectHolder::AddImageMaskBoundingBox(const CFX_FloatRect& box) {
m_MaskBoundingBoxes.push_back(box);
}
std::set<int32_t> CPDF_PageObjectHolder::TakeDirtyStreams() {
auto dirty_streams = std::move(m_DirtyStreams);
m_DirtyStreams.clear();
return dirty_streams;
}
absl::optional<ByteString> CPDF_PageObjectHolder::GraphicsMapSearch(
const GraphicsData& gd) {
auto it = m_GraphicsMap.find(gd);
if (it == m_GraphicsMap.end())
return absl::nullopt;
return it->second;
}
void CPDF_PageObjectHolder::GraphicsMapInsert(const GraphicsData& gd,
const ByteString& str) {
m_GraphicsMap[gd] = str;
}
absl::optional<ByteString> CPDF_PageObjectHolder::FontsMapSearch(
const FontData& fd) {
auto it = m_FontsMap.find(fd);
if (it == m_FontsMap.end())
return absl::nullopt;
return it->second;
}
void CPDF_PageObjectHolder::FontsMapInsert(const FontData& fd,
const ByteString& str) {
m_FontsMap[fd] = str;
}
void CPDF_PageObjectHolder::LoadTransparencyInfo() {
CPDF_Dictionary* pGroup = m_pDict->GetDictFor("Group");
if (!pGroup)
return;
if (pGroup->GetStringFor(pdfium::transparency::kGroupSubType) !=
pdfium::transparency::kTransparency) {
return;
}
m_Transparency.SetGroup();
if (pGroup->GetIntegerFor(pdfium::transparency::kI))
m_Transparency.SetIsolated();
}
CPDF_PageObject* CPDF_PageObjectHolder::GetPageObjectByIndex(
size_t index) const {
return fxcrt::IndexInBounds(m_PageObjectList, index)
? m_PageObjectList[index].get()
: nullptr;
}
void CPDF_PageObjectHolder::AppendPageObject(
std::unique_ptr<CPDF_PageObject> pPageObj) {
m_PageObjectList.push_back(std::move(pPageObj));
}
bool CPDF_PageObjectHolder::RemovePageObject(CPDF_PageObject* pPageObj) {
fxcrt::FakeUniquePtr<CPDF_PageObject> p(pPageObj);
auto it =
std::find(std::begin(m_PageObjectList), std::end(m_PageObjectList), p);
if (it == std::end(m_PageObjectList))
return false;
it->release();
m_PageObjectList.erase(it);
int32_t content_stream = pPageObj->GetContentStream();
if (content_stream >= 0)
m_DirtyStreams.insert(content_stream);
return true;
}
bool CPDF_PageObjectHolder::ErasePageObjectAtIndex(size_t index) {
if (index >= m_PageObjectList.size())
return false;
m_PageObjectList.erase(m_PageObjectList.begin() + index);
return true;
}