2023-10-07 15:40:45 +02:00
|
|
|
// Copyright (C) 2019-2023 Jakub Melka
|
2021-09-27 11:14:20 +02:00
|
|
|
//
|
|
|
|
// This file is part of PDF4QT.
|
|
|
|
//
|
|
|
|
// PDF4QT is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// with the written consent of the copyright owner, any later version.
|
|
|
|
//
|
|
|
|
// PDF4QT is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with PDF4QT. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
#include "pdfoutline.h"
|
|
|
|
#include "pdfdocument.h"
|
|
|
|
#include "pdfexception.h"
|
|
|
|
#include "pdfencoding.h"
|
2022-01-30 18:23:25 +01:00
|
|
|
#include "pdfdbgheap.h"
|
2021-09-27 11:14:20 +02:00
|
|
|
|
|
|
|
namespace pdf
|
|
|
|
{
|
|
|
|
|
|
|
|
size_t PDFOutlineItem::getTotalCount() const
|
|
|
|
{
|
|
|
|
size_t count = m_children.size();
|
|
|
|
|
|
|
|
for (size_t i = 0; i < m_children.size(); ++i)
|
|
|
|
{
|
|
|
|
count += getChild(i)->getTotalCount();
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
QSharedPointer<PDFOutlineItem> PDFOutlineItem::parse(const PDFObjectStorage* storage, const PDFObject& root)
|
|
|
|
{
|
|
|
|
const PDFObject& rootDereferenced = storage->getObject(root);
|
|
|
|
if (rootDereferenced.isDictionary())
|
|
|
|
{
|
|
|
|
const PDFDictionary* dictionary = rootDereferenced.getDictionary();
|
|
|
|
const PDFObject& first = dictionary->get("First");
|
|
|
|
|
|
|
|
if (first.isReference())
|
|
|
|
{
|
|
|
|
QSharedPointer<PDFOutlineItem> result(new PDFOutlineItem());
|
|
|
|
std::set<PDFObjectReference> visitedOutlineItems;
|
|
|
|
parseImpl(storage, result.get(), first.getReference(), visitedOutlineItems);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QSharedPointer<PDFOutlineItem>();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PDFOutlineItem::parseImpl(const PDFObjectStorage* storage,
|
|
|
|
PDFOutlineItem* parent,
|
|
|
|
PDFObjectReference currentItem,
|
|
|
|
std::set<PDFObjectReference>& visitedOutlineItems)
|
|
|
|
{
|
|
|
|
auto checkCyclicDependence = [&visitedOutlineItems](PDFObjectReference reference)
|
|
|
|
{
|
|
|
|
if (visitedOutlineItems.count(reference))
|
|
|
|
{
|
2023-08-06 17:10:03 +02:00
|
|
|
return false;
|
2021-09-27 11:14:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
visitedOutlineItems.insert(reference);
|
2023-08-06 17:10:03 +02:00
|
|
|
return true;
|
2021-09-27 11:14:20 +02:00
|
|
|
};
|
2023-08-06 17:10:03 +02:00
|
|
|
|
|
|
|
if (!checkCyclicDependence(currentItem))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2021-09-27 11:14:20 +02:00
|
|
|
|
|
|
|
PDFObject dereferencedItem = storage->getObjectByReference(currentItem);
|
|
|
|
while (dereferencedItem.isDictionary())
|
|
|
|
{
|
|
|
|
const PDFDictionary* dictionary = dereferencedItem.getDictionary();
|
|
|
|
|
|
|
|
QSharedPointer<PDFOutlineItem> currentOutlineItem(new PDFOutlineItem());
|
|
|
|
const PDFObject& titleObject = storage->getObject(dictionary->get("Title"));
|
|
|
|
if (titleObject.isString())
|
|
|
|
{
|
|
|
|
currentOutlineItem->setTitle(PDFEncoding::convertTextString(titleObject.getString()));
|
|
|
|
}
|
|
|
|
currentOutlineItem->setAction(PDFAction::parse(storage, dictionary->get("A")));
|
|
|
|
if (!currentOutlineItem->getAction() && dictionary->hasKey("Dest"))
|
|
|
|
{
|
|
|
|
currentOutlineItem->setAction(PDFActionPtr(new PDFActionGoTo(PDFDestination::parse(storage, dictionary->get("Dest")), PDFDestination())));
|
|
|
|
}
|
|
|
|
|
|
|
|
PDFDocumentDataLoaderDecorator loader(storage);
|
|
|
|
std::vector<PDFReal> colors = loader.readNumberArrayFromDictionary(dictionary, "C", { 0.0, 0.0, 0.0 });
|
|
|
|
colors.resize(3, 0.0);
|
|
|
|
currentOutlineItem->setTextColor(QColor::fromRgbF(colors[0], colors[1], colors[2]));
|
|
|
|
PDFInteger flag = loader.readIntegerFromDictionary(dictionary, "F", 0);
|
|
|
|
currentOutlineItem->setFontItalics(flag & 0x1);
|
|
|
|
currentOutlineItem->setFontBold(flag & 0x2);
|
|
|
|
PDFObject structureElementObject = dictionary->get("SE");
|
|
|
|
if (structureElementObject.isReference())
|
|
|
|
{
|
|
|
|
currentOutlineItem->setStructureElement(structureElementObject.getReference());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse children of this item
|
|
|
|
const PDFObject& firstItem = dictionary->get("First");
|
|
|
|
if (firstItem.isReference())
|
|
|
|
{
|
2023-08-06 17:10:03 +02:00
|
|
|
std::set<PDFObjectReference> currentVisitedOutlineItems = visitedOutlineItems;
|
|
|
|
parseImpl(storage, currentOutlineItem.get(), firstItem.getReference(), currentVisitedOutlineItems);
|
2021-09-27 11:14:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add new child to the parent
|
|
|
|
parent->addChild(qMove(currentOutlineItem));
|
|
|
|
|
|
|
|
// Parse next item
|
|
|
|
const PDFObject& nextItem = dictionary->get("Next");
|
|
|
|
if (nextItem.isReference())
|
|
|
|
{
|
2023-08-06 17:10:03 +02:00
|
|
|
if (!checkCyclicDependence(nextItem.getReference()))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2021-09-27 11:14:20 +02:00
|
|
|
dereferencedItem = storage->getObject(nextItem);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We are finished
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PDFObjectReference PDFOutlineItem::getStructureElement() const
|
|
|
|
{
|
|
|
|
return m_structureElement;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PDFOutlineItem::setStructureElement(PDFObjectReference structureElement)
|
|
|
|
{
|
|
|
|
m_structureElement = structureElement;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PDFOutlineItem::apply(const std::function<void (PDFOutlineItem*)>& functor)
|
|
|
|
{
|
|
|
|
functor(this);
|
|
|
|
|
|
|
|
for (const auto& item : m_children)
|
|
|
|
{
|
|
|
|
item->apply(functor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-01 17:35:42 +02:00
|
|
|
QSharedPointer<PDFOutlineItem> PDFOutlineItem::clone() const
|
|
|
|
{
|
|
|
|
QSharedPointer<PDFOutlineItem> result(new PDFOutlineItem());
|
|
|
|
|
|
|
|
result->setTitle(getTitle());
|
|
|
|
result->setTextColor(getTextColor());
|
|
|
|
result->setStructureElement(getStructureElement());
|
|
|
|
result->setFontItalics(isFontItalics());
|
|
|
|
result->setFontBold(isFontBold());
|
|
|
|
|
|
|
|
if (auto action = getAction())
|
|
|
|
{
|
|
|
|
result->setAction(action->clone());
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < getChildCount(); ++i)
|
|
|
|
{
|
|
|
|
result->addChild(getChild(i)->clone());
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-11-08 19:24:04 +01:00
|
|
|
void PDFOutlineItem::insertChild(size_t index, QSharedPointer<PDFOutlineItem> item)
|
|
|
|
{
|
|
|
|
m_children.insert(std::next(m_children.begin(), index), item);
|
|
|
|
}
|
|
|
|
|
|
|
|
void PDFOutlineItem::removeChild(size_t index)
|
|
|
|
{
|
|
|
|
m_children.erase(std::next(m_children.begin(), index));
|
|
|
|
}
|
|
|
|
|
2021-09-27 11:14:20 +02:00
|
|
|
bool PDFOutlineItem::isFontBold() const
|
|
|
|
{
|
|
|
|
return m_fontBold;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PDFOutlineItem::setFontBold(bool fontBold)
|
|
|
|
{
|
|
|
|
m_fontBold = fontBold;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PDFOutlineItem::isFontItalics() const
|
|
|
|
{
|
|
|
|
return m_fontItalics;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PDFOutlineItem::setFontItalics(bool fontItalics)
|
|
|
|
{
|
|
|
|
m_fontItalics = fontItalics;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor PDFOutlineItem::getTextColor() const
|
|
|
|
{
|
|
|
|
return m_textColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PDFOutlineItem::setTextColor(const QColor& textColor)
|
|
|
|
{
|
|
|
|
m_textColor = textColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
const PDFAction* PDFOutlineItem::getAction() const
|
|
|
|
{
|
|
|
|
return m_action.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
PDFAction* PDFOutlineItem::getAction()
|
|
|
|
{
|
|
|
|
return m_action.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PDFOutlineItem::setAction(const PDFActionPtr& action)
|
|
|
|
{
|
|
|
|
m_action = action;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace pdf
|