1*3ac0a46fSAndroid Build Coastguard Worker // Copyright 2016 The PDFium Authors
2*3ac0a46fSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*3ac0a46fSAndroid Build Coastguard Worker // found in the LICENSE file.
4*3ac0a46fSAndroid Build Coastguard Worker
5*3ac0a46fSAndroid Build Coastguard Worker // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6*3ac0a46fSAndroid Build Coastguard Worker
7*3ac0a46fSAndroid Build Coastguard Worker #include "fpdfsdk/cpdfsdk_annotiterator.h"
8*3ac0a46fSAndroid Build Coastguard Worker
9*3ac0a46fSAndroid Build Coastguard Worker #include <algorithm>
10*3ac0a46fSAndroid Build Coastguard Worker
11*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/page/cpdf_page.h"
12*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_dictionary.h"
13*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/stl_util.h"
14*3ac0a46fSAndroid Build Coastguard Worker #include "fpdfsdk/cpdfsdk_annot.h"
15*3ac0a46fSAndroid Build Coastguard Worker #include "fpdfsdk/cpdfsdk_pageview.h"
16*3ac0a46fSAndroid Build Coastguard Worker #include "fpdfsdk/cpdfsdk_widget.h"
17*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/containers/contains.h"
18*3ac0a46fSAndroid Build Coastguard Worker
19*3ac0a46fSAndroid Build Coastguard Worker namespace {
20*3ac0a46fSAndroid Build Coastguard Worker
GetAnnotRect(const CPDFSDK_Annot * pAnnot)21*3ac0a46fSAndroid Build Coastguard Worker CFX_FloatRect GetAnnotRect(const CPDFSDK_Annot* pAnnot) {
22*3ac0a46fSAndroid Build Coastguard Worker return pAnnot->GetPDFAnnot()->GetRect();
23*3ac0a46fSAndroid Build Coastguard Worker }
24*3ac0a46fSAndroid Build Coastguard Worker
CompareByLeftAscending(const CPDFSDK_Annot * p1,const CPDFSDK_Annot * p2)25*3ac0a46fSAndroid Build Coastguard Worker bool CompareByLeftAscending(const CPDFSDK_Annot* p1, const CPDFSDK_Annot* p2) {
26*3ac0a46fSAndroid Build Coastguard Worker return GetAnnotRect(p1).left < GetAnnotRect(p2).left;
27*3ac0a46fSAndroid Build Coastguard Worker }
28*3ac0a46fSAndroid Build Coastguard Worker
CompareByTopDescending(const CPDFSDK_Annot * p1,const CPDFSDK_Annot * p2)29*3ac0a46fSAndroid Build Coastguard Worker bool CompareByTopDescending(const CPDFSDK_Annot* p1, const CPDFSDK_Annot* p2) {
30*3ac0a46fSAndroid Build Coastguard Worker return GetAnnotRect(p1).top > GetAnnotRect(p2).top;
31*3ac0a46fSAndroid Build Coastguard Worker }
32*3ac0a46fSAndroid Build Coastguard Worker
33*3ac0a46fSAndroid Build Coastguard Worker } // namespace
34*3ac0a46fSAndroid Build Coastguard Worker
CPDFSDK_AnnotIterator(CPDFSDK_PageView * pPageView,const std::vector<CPDF_Annot::Subtype> & subtypes_to_iterate)35*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_AnnotIterator::CPDFSDK_AnnotIterator(
36*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_PageView* pPageView,
37*3ac0a46fSAndroid Build Coastguard Worker const std::vector<CPDF_Annot::Subtype>& subtypes_to_iterate)
38*3ac0a46fSAndroid Build Coastguard Worker : m_pPageView(pPageView),
39*3ac0a46fSAndroid Build Coastguard Worker m_subtypes(subtypes_to_iterate),
40*3ac0a46fSAndroid Build Coastguard Worker m_eTabOrder(GetTabOrder(pPageView)) {
41*3ac0a46fSAndroid Build Coastguard Worker GenerateResults();
42*3ac0a46fSAndroid Build Coastguard Worker }
43*3ac0a46fSAndroid Build Coastguard Worker
44*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_AnnotIterator::~CPDFSDK_AnnotIterator() = default;
45*3ac0a46fSAndroid Build Coastguard Worker
GetFirstAnnot()46*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_Annot* CPDFSDK_AnnotIterator::GetFirstAnnot() {
47*3ac0a46fSAndroid Build Coastguard Worker return m_Annots.empty() ? nullptr : m_Annots.front();
48*3ac0a46fSAndroid Build Coastguard Worker }
49*3ac0a46fSAndroid Build Coastguard Worker
GetLastAnnot()50*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_Annot* CPDFSDK_AnnotIterator::GetLastAnnot() {
51*3ac0a46fSAndroid Build Coastguard Worker return m_Annots.empty() ? nullptr : m_Annots.back();
52*3ac0a46fSAndroid Build Coastguard Worker }
53*3ac0a46fSAndroid Build Coastguard Worker
GetNextAnnot(CPDFSDK_Annot * pAnnot)54*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_Annot* CPDFSDK_AnnotIterator::GetNextAnnot(CPDFSDK_Annot* pAnnot) {
55*3ac0a46fSAndroid Build Coastguard Worker auto iter = std::find(m_Annots.begin(), m_Annots.end(), pAnnot);
56*3ac0a46fSAndroid Build Coastguard Worker if (iter == m_Annots.end())
57*3ac0a46fSAndroid Build Coastguard Worker return nullptr;
58*3ac0a46fSAndroid Build Coastguard Worker ++iter;
59*3ac0a46fSAndroid Build Coastguard Worker if (iter == m_Annots.end())
60*3ac0a46fSAndroid Build Coastguard Worker return nullptr;
61*3ac0a46fSAndroid Build Coastguard Worker return *iter;
62*3ac0a46fSAndroid Build Coastguard Worker }
63*3ac0a46fSAndroid Build Coastguard Worker
GetPrevAnnot(CPDFSDK_Annot * pAnnot)64*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_Annot* CPDFSDK_AnnotIterator::GetPrevAnnot(CPDFSDK_Annot* pAnnot) {
65*3ac0a46fSAndroid Build Coastguard Worker auto iter = std::find(m_Annots.begin(), m_Annots.end(), pAnnot);
66*3ac0a46fSAndroid Build Coastguard Worker if (iter == m_Annots.begin() || iter == m_Annots.end())
67*3ac0a46fSAndroid Build Coastguard Worker return nullptr;
68*3ac0a46fSAndroid Build Coastguard Worker return *(--iter);
69*3ac0a46fSAndroid Build Coastguard Worker }
70*3ac0a46fSAndroid Build Coastguard Worker
CollectAnnots(std::vector<UnownedPtr<CPDFSDK_Annot>> * pArray)71*3ac0a46fSAndroid Build Coastguard Worker void CPDFSDK_AnnotIterator::CollectAnnots(
72*3ac0a46fSAndroid Build Coastguard Worker std::vector<UnownedPtr<CPDFSDK_Annot>>* pArray) {
73*3ac0a46fSAndroid Build Coastguard Worker for (auto* pAnnot : m_pPageView->GetAnnotList()) {
74*3ac0a46fSAndroid Build Coastguard Worker if (pdfium::Contains(m_subtypes, pAnnot->GetAnnotSubtype())) {
75*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_Widget* pWidget = ToCPDFSDKWidget(pAnnot);
76*3ac0a46fSAndroid Build Coastguard Worker if (!pWidget || !pWidget->IsSignatureWidget())
77*3ac0a46fSAndroid Build Coastguard Worker pArray->emplace_back(pAnnot);
78*3ac0a46fSAndroid Build Coastguard Worker }
79*3ac0a46fSAndroid Build Coastguard Worker }
80*3ac0a46fSAndroid Build Coastguard Worker }
81*3ac0a46fSAndroid Build Coastguard Worker
AddToAnnotsList(std::vector<UnownedPtr<CPDFSDK_Annot>> * sa,size_t idx)82*3ac0a46fSAndroid Build Coastguard Worker CFX_FloatRect CPDFSDK_AnnotIterator::AddToAnnotsList(
83*3ac0a46fSAndroid Build Coastguard Worker std::vector<UnownedPtr<CPDFSDK_Annot>>* sa,
84*3ac0a46fSAndroid Build Coastguard Worker size_t idx) {
85*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_Annot* pLeftTopAnnot = sa->at(idx);
86*3ac0a46fSAndroid Build Coastguard Worker CFX_FloatRect rcLeftTop = GetAnnotRect(pLeftTopAnnot);
87*3ac0a46fSAndroid Build Coastguard Worker m_Annots.emplace_back(pLeftTopAnnot);
88*3ac0a46fSAndroid Build Coastguard Worker sa->erase(sa->begin() + idx);
89*3ac0a46fSAndroid Build Coastguard Worker return rcLeftTop;
90*3ac0a46fSAndroid Build Coastguard Worker }
91*3ac0a46fSAndroid Build Coastguard Worker
AddSelectedToAnnots(std::vector<UnownedPtr<CPDFSDK_Annot>> * sa,std::vector<size_t> * aSelect)92*3ac0a46fSAndroid Build Coastguard Worker void CPDFSDK_AnnotIterator::AddSelectedToAnnots(
93*3ac0a46fSAndroid Build Coastguard Worker std::vector<UnownedPtr<CPDFSDK_Annot>>* sa,
94*3ac0a46fSAndroid Build Coastguard Worker std::vector<size_t>* aSelect) {
95*3ac0a46fSAndroid Build Coastguard Worker for (size_t i = 0; i < aSelect->size(); ++i)
96*3ac0a46fSAndroid Build Coastguard Worker m_Annots.emplace_back(sa->at(aSelect->at(i)));
97*3ac0a46fSAndroid Build Coastguard Worker
98*3ac0a46fSAndroid Build Coastguard Worker for (size_t i = aSelect->size(); i > 0; --i)
99*3ac0a46fSAndroid Build Coastguard Worker sa->erase(sa->begin() + aSelect->at(i - 1));
100*3ac0a46fSAndroid Build Coastguard Worker }
101*3ac0a46fSAndroid Build Coastguard Worker
102*3ac0a46fSAndroid Build Coastguard Worker // static
GetTabOrder(CPDFSDK_PageView * pPageView)103*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_AnnotIterator::TabOrder CPDFSDK_AnnotIterator::GetTabOrder(
104*3ac0a46fSAndroid Build Coastguard Worker CPDFSDK_PageView* pPageView) {
105*3ac0a46fSAndroid Build Coastguard Worker CPDF_Page* pPDFPage = pPageView->GetPDFPage();
106*3ac0a46fSAndroid Build Coastguard Worker ByteString sTabs = pPDFPage->GetDict()->GetByteStringFor("Tabs");
107*3ac0a46fSAndroid Build Coastguard Worker if (sTabs == "R")
108*3ac0a46fSAndroid Build Coastguard Worker return kRow;
109*3ac0a46fSAndroid Build Coastguard Worker if (sTabs == "C")
110*3ac0a46fSAndroid Build Coastguard Worker return kColumn;
111*3ac0a46fSAndroid Build Coastguard Worker return kStructure;
112*3ac0a46fSAndroid Build Coastguard Worker }
113*3ac0a46fSAndroid Build Coastguard Worker
GenerateResults()114*3ac0a46fSAndroid Build Coastguard Worker void CPDFSDK_AnnotIterator::GenerateResults() {
115*3ac0a46fSAndroid Build Coastguard Worker switch (m_eTabOrder) {
116*3ac0a46fSAndroid Build Coastguard Worker case kStructure:
117*3ac0a46fSAndroid Build Coastguard Worker CollectAnnots(&m_Annots);
118*3ac0a46fSAndroid Build Coastguard Worker break;
119*3ac0a46fSAndroid Build Coastguard Worker
120*3ac0a46fSAndroid Build Coastguard Worker case kRow: {
121*3ac0a46fSAndroid Build Coastguard Worker std::vector<UnownedPtr<CPDFSDK_Annot>> sa;
122*3ac0a46fSAndroid Build Coastguard Worker CollectAnnots(&sa);
123*3ac0a46fSAndroid Build Coastguard Worker std::sort(sa.begin(), sa.end(), CompareByLeftAscending);
124*3ac0a46fSAndroid Build Coastguard Worker
125*3ac0a46fSAndroid Build Coastguard Worker while (!sa.empty()) {
126*3ac0a46fSAndroid Build Coastguard Worker int nLeftTopIndex = -1;
127*3ac0a46fSAndroid Build Coastguard Worker float fTop = 0.0f;
128*3ac0a46fSAndroid Build Coastguard Worker for (int i = fxcrt::CollectionSize<int>(sa) - 1; i >= 0; i--) {
129*3ac0a46fSAndroid Build Coastguard Worker CFX_FloatRect rcAnnot = GetAnnotRect(sa[i]);
130*3ac0a46fSAndroid Build Coastguard Worker if (rcAnnot.top > fTop) {
131*3ac0a46fSAndroid Build Coastguard Worker nLeftTopIndex = i;
132*3ac0a46fSAndroid Build Coastguard Worker fTop = rcAnnot.top;
133*3ac0a46fSAndroid Build Coastguard Worker }
134*3ac0a46fSAndroid Build Coastguard Worker }
135*3ac0a46fSAndroid Build Coastguard Worker if (nLeftTopIndex < 0)
136*3ac0a46fSAndroid Build Coastguard Worker continue;
137*3ac0a46fSAndroid Build Coastguard Worker
138*3ac0a46fSAndroid Build Coastguard Worker CFX_FloatRect rcLeftTop = AddToAnnotsList(&sa, nLeftTopIndex);
139*3ac0a46fSAndroid Build Coastguard Worker
140*3ac0a46fSAndroid Build Coastguard Worker std::vector<size_t> aSelect;
141*3ac0a46fSAndroid Build Coastguard Worker for (size_t i = 0; i < sa.size(); ++i) {
142*3ac0a46fSAndroid Build Coastguard Worker CFX_FloatRect rcAnnot = GetAnnotRect(sa[i]);
143*3ac0a46fSAndroid Build Coastguard Worker float fCenterY = (rcAnnot.top + rcAnnot.bottom) / 2.0f;
144*3ac0a46fSAndroid Build Coastguard Worker if (fCenterY > rcLeftTop.bottom && fCenterY < rcLeftTop.top)
145*3ac0a46fSAndroid Build Coastguard Worker aSelect.push_back(i);
146*3ac0a46fSAndroid Build Coastguard Worker }
147*3ac0a46fSAndroid Build Coastguard Worker AddSelectedToAnnots(&sa, &aSelect);
148*3ac0a46fSAndroid Build Coastguard Worker }
149*3ac0a46fSAndroid Build Coastguard Worker break;
150*3ac0a46fSAndroid Build Coastguard Worker }
151*3ac0a46fSAndroid Build Coastguard Worker
152*3ac0a46fSAndroid Build Coastguard Worker case kColumn: {
153*3ac0a46fSAndroid Build Coastguard Worker std::vector<UnownedPtr<CPDFSDK_Annot>> sa;
154*3ac0a46fSAndroid Build Coastguard Worker CollectAnnots(&sa);
155*3ac0a46fSAndroid Build Coastguard Worker std::sort(sa.begin(), sa.end(), CompareByTopDescending);
156*3ac0a46fSAndroid Build Coastguard Worker
157*3ac0a46fSAndroid Build Coastguard Worker while (!sa.empty()) {
158*3ac0a46fSAndroid Build Coastguard Worker int nLeftTopIndex = -1;
159*3ac0a46fSAndroid Build Coastguard Worker float fLeft = -1.0f;
160*3ac0a46fSAndroid Build Coastguard Worker for (int i = fxcrt::CollectionSize<int>(sa) - 1; i >= 0; --i) {
161*3ac0a46fSAndroid Build Coastguard Worker CFX_FloatRect rcAnnot = GetAnnotRect(sa[i]);
162*3ac0a46fSAndroid Build Coastguard Worker if (fLeft < 0) {
163*3ac0a46fSAndroid Build Coastguard Worker nLeftTopIndex = 0;
164*3ac0a46fSAndroid Build Coastguard Worker fLeft = rcAnnot.left;
165*3ac0a46fSAndroid Build Coastguard Worker } else if (rcAnnot.left < fLeft) {
166*3ac0a46fSAndroid Build Coastguard Worker nLeftTopIndex = i;
167*3ac0a46fSAndroid Build Coastguard Worker fLeft = rcAnnot.left;
168*3ac0a46fSAndroid Build Coastguard Worker }
169*3ac0a46fSAndroid Build Coastguard Worker }
170*3ac0a46fSAndroid Build Coastguard Worker if (nLeftTopIndex < 0)
171*3ac0a46fSAndroid Build Coastguard Worker continue;
172*3ac0a46fSAndroid Build Coastguard Worker
173*3ac0a46fSAndroid Build Coastguard Worker CFX_FloatRect rcLeftTop = AddToAnnotsList(&sa, nLeftTopIndex);
174*3ac0a46fSAndroid Build Coastguard Worker
175*3ac0a46fSAndroid Build Coastguard Worker std::vector<size_t> aSelect;
176*3ac0a46fSAndroid Build Coastguard Worker for (size_t i = 0; i < sa.size(); ++i) {
177*3ac0a46fSAndroid Build Coastguard Worker CFX_FloatRect rcAnnot = GetAnnotRect(sa[i]);
178*3ac0a46fSAndroid Build Coastguard Worker float fCenterX = (rcAnnot.left + rcAnnot.right) / 2.0f;
179*3ac0a46fSAndroid Build Coastguard Worker if (fCenterX > rcLeftTop.left && fCenterX < rcLeftTop.right)
180*3ac0a46fSAndroid Build Coastguard Worker aSelect.push_back(i);
181*3ac0a46fSAndroid Build Coastguard Worker }
182*3ac0a46fSAndroid Build Coastguard Worker AddSelectedToAnnots(&sa, &aSelect);
183*3ac0a46fSAndroid Build Coastguard Worker }
184*3ac0a46fSAndroid Build Coastguard Worker break;
185*3ac0a46fSAndroid Build Coastguard Worker }
186*3ac0a46fSAndroid Build Coastguard Worker }
187*3ac0a46fSAndroid Build Coastguard Worker }
188