xref: /aosp_15_r20/external/pdfium/fpdfsdk/cpdfsdk_annotiterator.cpp (revision 3ac0a46f773bac49fa9476ec2b1cf3f8da5ec3a4)
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