2018-12-18 21:08:53 -05:00
|
|
|
#include <qpdf/QPDFOutlineDocumentHelper.hh>
|
2022-02-04 16:31:31 -05:00
|
|
|
|
2018-12-18 21:08:53 -05:00
|
|
|
#include <qpdf/QTC.hh>
|
|
|
|
|
|
|
|
QPDFOutlineDocumentHelper::QPDFOutlineDocumentHelper(QPDF& qpdf) :
|
|
|
|
QPDFDocumentHelper(qpdf),
|
|
|
|
m(new Members())
|
|
|
|
{
|
|
|
|
QPDFObjectHandle root = qpdf.getRoot();
|
2022-04-02 17:14:10 -04:00
|
|
|
if (!root.hasKey("/Outlines")) {
|
2018-12-18 21:08:53 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
QPDFObjectHandle outlines = root.getKey("/Outlines");
|
2022-04-02 17:14:10 -04:00
|
|
|
if (!(outlines.isDictionary() && outlines.hasKey("/First"))) {
|
2018-12-18 21:08:53 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
QPDFObjectHandle cur = outlines.getKey("/First");
|
2021-02-10 16:26:32 -05:00
|
|
|
std::set<QPDFObjGen> seen;
|
2022-04-02 17:14:10 -04:00
|
|
|
while (!cur.isNull()) {
|
2021-02-10 16:26:32 -05:00
|
|
|
auto og = cur.getObjGen();
|
2022-04-02 17:14:10 -04:00
|
|
|
if (seen.count(og)) {
|
2021-02-10 16:26:32 -05:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
seen.insert(og);
|
2018-12-18 21:08:53 -05:00
|
|
|
this->m->outlines.push_back(
|
|
|
|
QPDFOutlineObjectHelper::Accessor::create(cur, *this, 1));
|
|
|
|
cur = cur.getKey("/Next");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
QPDFOutlineDocumentHelper::hasOutlines()
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
return !this->m->outlines.empty();
|
2018-12-18 21:08:53 -05:00
|
|
|
}
|
|
|
|
|
2019-07-03 19:34:02 +02:00
|
|
|
std::vector<QPDFOutlineObjectHelper>
|
2018-12-18 21:08:53 -05:00
|
|
|
QPDFOutlineDocumentHelper::getTopLevelOutlines()
|
|
|
|
{
|
|
|
|
return this->m->outlines;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
QPDFOutlineDocumentHelper::initializeByPage()
|
|
|
|
{
|
|
|
|
std::list<QPDFOutlineObjectHelper> queue;
|
2022-04-02 17:14:10 -04:00
|
|
|
queue.insert(
|
|
|
|
queue.end(), this->m->outlines.begin(), this->m->outlines.end());
|
2018-12-18 21:08:53 -05:00
|
|
|
|
2022-04-02 17:14:10 -04:00
|
|
|
while (!queue.empty()) {
|
2018-12-18 21:08:53 -05:00
|
|
|
QPDFOutlineObjectHelper oh = queue.front();
|
|
|
|
queue.pop_front();
|
|
|
|
this->m->by_page[oh.getDestPage().getObjGen()].push_back(oh);
|
2019-07-03 19:34:02 +02:00
|
|
|
std::vector<QPDFOutlineObjectHelper> kids = oh.getKids();
|
2018-12-18 21:08:53 -05:00
|
|
|
queue.insert(queue.end(), kids.begin(), kids.end());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-03 19:34:02 +02:00
|
|
|
std::vector<QPDFOutlineObjectHelper>
|
2018-12-18 21:08:53 -05:00
|
|
|
QPDFOutlineDocumentHelper::getOutlinesForPage(QPDFObjGen const& og)
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->m->by_page.empty()) {
|
2018-12-18 21:08:53 -05:00
|
|
|
initializeByPage();
|
|
|
|
}
|
2019-07-03 19:34:02 +02:00
|
|
|
std::vector<QPDFOutlineObjectHelper> result;
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->m->by_page.count(og)) {
|
2018-12-18 21:08:53 -05:00
|
|
|
result = this->m->by_page[og];
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPDFObjectHandle
|
|
|
|
QPDFOutlineDocumentHelper::resolveNamedDest(QPDFObjectHandle name)
|
|
|
|
{
|
|
|
|
QPDFObjectHandle result;
|
2022-04-02 17:14:10 -04:00
|
|
|
if (name.isName()) {
|
|
|
|
if (!this->m->dest_dict.isInitialized()) {
|
2018-12-18 21:08:53 -05:00
|
|
|
this->m->dest_dict = this->qpdf.getRoot().getKey("/Dests");
|
|
|
|
}
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->m->dest_dict.isDictionary()) {
|
2018-12-18 21:08:53 -05:00
|
|
|
QTC::TC("qpdf", "QPDFOutlineDocumentHelper name named dest");
|
|
|
|
result = this->m->dest_dict.getKey(name.getName());
|
|
|
|
}
|
2022-04-02 17:14:10 -04:00
|
|
|
} else if (name.isString()) {
|
2022-08-07 10:33:25 +01:00
|
|
|
if (nullptr == this->m->names_dest) {
|
2018-12-18 21:08:53 -05:00
|
|
|
QPDFObjectHandle names = this->qpdf.getRoot().getKey("/Names");
|
2022-04-02 17:14:10 -04:00
|
|
|
if (names.isDictionary()) {
|
2018-12-18 21:08:53 -05:00
|
|
|
QPDFObjectHandle dests = names.getKey("/Dests");
|
2022-04-02 17:14:10 -04:00
|
|
|
if (dests.isDictionary()) {
|
2018-12-18 21:08:53 -05:00
|
|
|
this->m->names_dest =
|
2022-04-09 14:35:56 -04:00
|
|
|
std::make_shared<QPDFNameTreeObjectHelper>(
|
2022-02-06 11:40:24 -05:00
|
|
|
dests, this->qpdf);
|
2018-12-18 21:08:53 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->m->names_dest.get()) {
|
|
|
|
if (this->m->names_dest->findObject(name.getUTF8Value(), result)) {
|
2018-12-18 21:08:53 -05:00
|
|
|
QTC::TC("qpdf", "QPDFOutlineDocumentHelper string named dest");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-04-02 17:14:10 -04:00
|
|
|
if (!result.isInitialized()) {
|
2018-12-18 21:08:53 -05:00
|
|
|
result = QPDFObjectHandle::newNull();
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
QPDFOutlineDocumentHelper::checkSeen(QPDFObjGen const& og)
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->m->seen.count(og) > 0) {
|
2018-12-18 21:08:53 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
this->m->seen.insert(og);
|
|
|
|
return false;
|
|
|
|
}
|