blob: 5827a42739fb9fae35f2a0268875d08b33bf7a0a [file] [log] [blame]
jaepark611adb82016-08-17 11:34:36 -07001// Copyright 2016 PDFium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6
Dan Sinclaircbf76e62018-03-28 21:00:35 +00007#include "fpdfsdk/cpdfsdk_annotiterator.h"
jaepark611adb82016-08-17 11:34:36 -07008
Dan Sinclair85c8e7f2016-11-21 13:50:32 -05009#include <algorithm>
10
dsinclair41872fa2016-10-04 11:29:35 -070011#include "core/fpdfapi/page/cpdf_page.h"
Lei Zhang81535612018-10-09 21:15:17 +000012#include "core/fpdfapi/parser/cpdf_dictionary.h"
dsinclair114e46a2016-09-29 17:18:21 -070013#include "fpdfsdk/cpdfsdk_annot.h"
14#include "fpdfsdk/cpdfsdk_pageview.h"
Lei Zhanga39ead92020-04-22 01:52:36 +000015#include "third_party/base/stl_util.h"
jaepark611adb82016-08-17 11:34:36 -070016
dsinclair8afe15a2016-10-05 12:00:34 -070017namespace {
18
19CFX_FloatRect GetAnnotRect(const CPDFSDK_Annot* pAnnot) {
20 return pAnnot->GetPDFAnnot()->GetRect();
21}
22
23bool CompareByLeftAscending(const CPDFSDK_Annot* p1, const CPDFSDK_Annot* p2) {
jaepark611adb82016-08-17 11:34:36 -070024 return GetAnnotRect(p1).left < GetAnnotRect(p2).left;
25}
26
dsinclair8afe15a2016-10-05 12:00:34 -070027bool CompareByTopDescending(const CPDFSDK_Annot* p1, const CPDFSDK_Annot* p2) {
jaepark611adb82016-08-17 11:34:36 -070028 return GetAnnotRect(p1).top > GetAnnotRect(p2).top;
29}
30
Lei Zhang7fb895f2018-10-09 19:02:24 +000031CPDFSDK_AnnotIterator::TabOrder GetTabOrder(CPDFSDK_PageView* pPageView) {
32 CPDF_Page* pPDFPage = pPageView->GetPDFPage();
33 ByteString sTabs = pPDFPage->GetDict()->GetStringFor("Tabs");
34 if (sTabs == "R")
35 return CPDFSDK_AnnotIterator::ROW;
36 if (sTabs == "C")
37 return CPDFSDK_AnnotIterator::COLUMN;
38 return CPDFSDK_AnnotIterator::STRUCTURE;
39}
40
dsinclair8afe15a2016-10-05 12:00:34 -070041} // namespace
42
Dan Sinclaircbf76e62018-03-28 21:00:35 +000043CPDFSDK_AnnotIterator::CPDFSDK_AnnotIterator(CPDFSDK_PageView* pPageView,
44 CPDF_Annot::Subtype nAnnotSubtype)
Neha Gupta1eb6ddd2020-03-19 08:37:15 +000045 : CPDFSDK_AnnotIterator(
46 pPageView,
47 std::vector<CPDF_Annot::Subtype>(1, nAnnotSubtype)) {}
48
49CPDFSDK_AnnotIterator::CPDFSDK_AnnotIterator(
50 CPDFSDK_PageView* pPageView,
51 const std::vector<CPDF_Annot::Subtype>& subtypes_to_iterate)
Lei Zhang7fb895f2018-10-09 19:02:24 +000052 : m_pPageView(pPageView),
Neha Gupta1eb6ddd2020-03-19 08:37:15 +000053 m_subtypes(subtypes_to_iterate),
Lei Zhang7fb895f2018-10-09 19:02:24 +000054 m_eTabOrder(GetTabOrder(pPageView)) {
jaepark611adb82016-08-17 11:34:36 -070055 GenerateResults();
56}
57
Lei Zhang81535612018-10-09 21:15:17 +000058CPDFSDK_AnnotIterator::~CPDFSDK_AnnotIterator() = default;
jaepark611adb82016-08-17 11:34:36 -070059
Dan Sinclaircbf76e62018-03-28 21:00:35 +000060CPDFSDK_Annot* CPDFSDK_AnnotIterator::GetFirstAnnot() {
jaepark611adb82016-08-17 11:34:36 -070061 return m_Annots.empty() ? nullptr : m_Annots.front();
62}
63
Dan Sinclaircbf76e62018-03-28 21:00:35 +000064CPDFSDK_Annot* CPDFSDK_AnnotIterator::GetLastAnnot() {
jaepark611adb82016-08-17 11:34:36 -070065 return m_Annots.empty() ? nullptr : m_Annots.back();
66}
67
Dan Sinclaircbf76e62018-03-28 21:00:35 +000068CPDFSDK_Annot* CPDFSDK_AnnotIterator::GetNextAnnot(CPDFSDK_Annot* pAnnot) {
jaepark611adb82016-08-17 11:34:36 -070069 auto iter = std::find(m_Annots.begin(), m_Annots.end(), pAnnot);
70 if (iter == m_Annots.end())
71 return nullptr;
72 ++iter;
73 if (iter == m_Annots.end())
Ankit Kumarafc869e2020-04-07 21:21:51 +000074 return nullptr;
jaepark611adb82016-08-17 11:34:36 -070075 return *iter;
76}
77
Dan Sinclaircbf76e62018-03-28 21:00:35 +000078CPDFSDK_Annot* CPDFSDK_AnnotIterator::GetPrevAnnot(CPDFSDK_Annot* pAnnot) {
jaepark611adb82016-08-17 11:34:36 -070079 auto iter = std::find(m_Annots.begin(), m_Annots.end(), pAnnot);
Ankit Kumarafc869e2020-04-07 21:21:51 +000080 if (iter == m_Annots.begin() || iter == m_Annots.end())
jaepark611adb82016-08-17 11:34:36 -070081 return nullptr;
jaepark611adb82016-08-17 11:34:36 -070082 return *(--iter);
83}
84
Dan Sinclaircbf76e62018-03-28 21:00:35 +000085void CPDFSDK_AnnotIterator::CollectAnnots(std::vector<CPDFSDK_Annot*>* pArray) {
Lei Zhang375c2762017-03-10 14:37:14 -080086 for (auto* pAnnot : m_pPageView->GetAnnotList()) {
Lei Zhangf245ae62020-05-15 21:49:46 +000087 if (pdfium::Contains(m_subtypes, pAnnot->GetAnnotSubtype()) &&
dsinclair8afe15a2016-10-05 12:00:34 -070088 !pAnnot->IsSignatureWidget()) {
89 pArray->push_back(pAnnot);
90 }
91 }
92}
93
Dan Sinclaircbf76e62018-03-28 21:00:35 +000094CFX_FloatRect CPDFSDK_AnnotIterator::AddToAnnotsList(
dsinclair8afe15a2016-10-05 12:00:34 -070095 std::vector<CPDFSDK_Annot*>* sa,
96 size_t idx) {
97 CPDFSDK_Annot* pLeftTopAnnot = sa->at(idx);
98 CFX_FloatRect rcLeftTop = GetAnnotRect(pLeftTopAnnot);
99 m_Annots.push_back(pLeftTopAnnot);
100 sa->erase(sa->begin() + idx);
101 return rcLeftTop;
102}
103
Dan Sinclaircbf76e62018-03-28 21:00:35 +0000104void CPDFSDK_AnnotIterator::AddSelectedToAnnots(std::vector<CPDFSDK_Annot*>* sa,
105 std::vector<size_t>* aSelect) {
dsinclair8afe15a2016-10-05 12:00:34 -0700106 for (size_t i = 0; i < aSelect->size(); ++i)
107 m_Annots.push_back(sa->at(aSelect->at(i)));
108
109 for (int i = aSelect->size() - 1; i >= 0; --i)
110 sa->erase(sa->begin() + aSelect->at(i));
111}
112
Dan Sinclaircbf76e62018-03-28 21:00:35 +0000113void CPDFSDK_AnnotIterator::GenerateResults() {
jaepark611adb82016-08-17 11:34:36 -0700114 switch (m_eTabOrder) {
dsinclair8afe15a2016-10-05 12:00:34 -0700115 case STRUCTURE:
116 CollectAnnots(&m_Annots);
jaepark611adb82016-08-17 11:34:36 -0700117 break;
dsinclair8afe15a2016-10-05 12:00:34 -0700118
jaepark611adb82016-08-17 11:34:36 -0700119 case ROW: {
120 std::vector<CPDFSDK_Annot*> sa;
dsinclair8afe15a2016-10-05 12:00:34 -0700121 CollectAnnots(&sa);
jaepark611adb82016-08-17 11:34:36 -0700122 std::sort(sa.begin(), sa.end(), CompareByLeftAscending);
dsinclair8afe15a2016-10-05 12:00:34 -0700123
jaepark611adb82016-08-17 11:34:36 -0700124 while (!sa.empty()) {
125 int nLeftTopIndex = -1;
Dan Sinclair05df0752017-03-14 14:43:42 -0400126 float fTop = 0.0f;
jaepark611adb82016-08-17 11:34:36 -0700127 for (int i = sa.size() - 1; i >= 0; i--) {
128 CFX_FloatRect rcAnnot = GetAnnotRect(sa[i]);
129 if (rcAnnot.top > fTop) {
130 nLeftTopIndex = i;
131 fTop = rcAnnot.top;
132 }
133 }
dsinclair8afe15a2016-10-05 12:00:34 -0700134 if (nLeftTopIndex < 0)
135 continue;
jaepark611adb82016-08-17 11:34:36 -0700136
dsinclair8afe15a2016-10-05 12:00:34 -0700137 CFX_FloatRect rcLeftTop = AddToAnnotsList(&sa, nLeftTopIndex);
jaepark611adb82016-08-17 11:34:36 -0700138
dsinclair8afe15a2016-10-05 12:00:34 -0700139 std::vector<size_t> aSelect;
140 for (size_t i = 0; i < sa.size(); ++i) {
141 CFX_FloatRect rcAnnot = GetAnnotRect(sa[i]);
Dan Sinclair05df0752017-03-14 14:43:42 -0400142 float fCenterY = (rcAnnot.top + rcAnnot.bottom) / 2.0f;
dsinclair8afe15a2016-10-05 12:00:34 -0700143 if (fCenterY > rcLeftTop.bottom && fCenterY < rcLeftTop.top)
144 aSelect.push_back(i);
jaepark611adb82016-08-17 11:34:36 -0700145 }
dsinclair8afe15a2016-10-05 12:00:34 -0700146 AddSelectedToAnnots(&sa, &aSelect);
jaepark611adb82016-08-17 11:34:36 -0700147 }
148 break;
149 }
dsinclair8afe15a2016-10-05 12:00:34 -0700150
jaepark611adb82016-08-17 11:34:36 -0700151 case COLUMN: {
152 std::vector<CPDFSDK_Annot*> sa;
dsinclair8afe15a2016-10-05 12:00:34 -0700153 CollectAnnots(&sa);
jaepark611adb82016-08-17 11:34:36 -0700154 std::sort(sa.begin(), sa.end(), CompareByTopDescending);
dsinclair8afe15a2016-10-05 12:00:34 -0700155
jaepark611adb82016-08-17 11:34:36 -0700156 while (!sa.empty()) {
157 int nLeftTopIndex = -1;
Dan Sinclair05df0752017-03-14 14:43:42 -0400158 float fLeft = -1.0f;
jaepark611adb82016-08-17 11:34:36 -0700159 for (int i = sa.size() - 1; i >= 0; --i) {
160 CFX_FloatRect rcAnnot = GetAnnotRect(sa[i]);
161 if (fLeft < 0) {
162 nLeftTopIndex = 0;
163 fLeft = rcAnnot.left;
164 } else if (rcAnnot.left < fLeft) {
165 nLeftTopIndex = i;
166 fLeft = rcAnnot.left;
167 }
168 }
dsinclair8afe15a2016-10-05 12:00:34 -0700169 if (nLeftTopIndex < 0)
170 continue;
jaepark611adb82016-08-17 11:34:36 -0700171
dsinclair8afe15a2016-10-05 12:00:34 -0700172 CFX_FloatRect rcLeftTop = AddToAnnotsList(&sa, nLeftTopIndex);
jaepark611adb82016-08-17 11:34:36 -0700173
dsinclair8afe15a2016-10-05 12:00:34 -0700174 std::vector<size_t> aSelect;
175 for (size_t i = 0; i < sa.size(); ++i) {
176 CFX_FloatRect rcAnnot = GetAnnotRect(sa[i]);
Dan Sinclair05df0752017-03-14 14:43:42 -0400177 float fCenterX = (rcAnnot.left + rcAnnot.right) / 2.0f;
dsinclair8afe15a2016-10-05 12:00:34 -0700178 if (fCenterX > rcLeftTop.left && fCenterX < rcLeftTop.right)
179 aSelect.push_back(i);
jaepark611adb82016-08-17 11:34:36 -0700180 }
dsinclair8afe15a2016-10-05 12:00:34 -0700181 AddSelectedToAnnots(&sa, &aSelect);
jaepark611adb82016-08-17 11:34:36 -0700182 }
183 break;
184 }
185 }
186}