Create outline for a document

This commit is contained in:
Jakub Melka
2020-11-15 18:15:10 +01:00
parent e06148fa8c
commit cb3a36f891
8 changed files with 334 additions and 1 deletions

View File

@@ -1059,6 +1059,115 @@ QRectF PDFDocumentBuilder::getPolygonsBoundingRect(const Polygons& polygons) con
return rect;
}
PDFObjectReference PDFDocumentBuilder::createOutlineItem(const PDFOutlineItem* root, bool writeOutlineData)
{
PDFObjectFactory objectBuilder;
objectBuilder.beginDictionary();
if (writeOutlineData)
{
// Title
objectBuilder.beginDictionaryItem("Title");
objectBuilder << root->getTitle();
objectBuilder.endDictionaryItem();
// Destination
const PDFActionGoTo* action = dynamic_cast<const PDFActionGoTo*>(root->getAction());
if (action)
{
objectBuilder.beginDictionaryItem("Dest");
objectBuilder << action->getDestination();
objectBuilder.endDictionaryItem();
}
// Color
if (root->getTextColor().isValid() && root->getTextColor() != Qt::black)
{
objectBuilder.beginDictionaryItem("C");
objectBuilder << root->getTextColor();
objectBuilder.endDictionaryItem();
}
// Flags
PDFInteger flags = 0;
if (root->isFontItalics())
{
flags += 1;
}
if (root->isFontBold())
{
flags += 2;
}
if (flags > 0)
{
objectBuilder.beginDictionaryItem("F");
objectBuilder << flags;
objectBuilder.endDictionaryItem();
}
}
// Create descendands
std::vector<PDFObjectReference> children;
children.reserve(root->getChildCount());
for (size_t i = 0; i < root->getChildCount(); ++i)
{
children.push_back(createOutlineItem(root->getChild(i), true));
}
if (!children.empty())
{
// First/Last pointers
objectBuilder.beginDictionaryItem("First");
objectBuilder << children.front();
objectBuilder.endDictionaryItem();
objectBuilder.beginDictionaryItem("Last");
objectBuilder << children.back();
objectBuilder.endDictionaryItem();
}
size_t totalCount = root->getTotalCount();
if (totalCount > 0)
{
objectBuilder.beginDictionaryItem("Count");
objectBuilder << PDFInteger(totalCount);
objectBuilder.endDictionaryItem();
}
objectBuilder.endDictionary();
PDFObjectReference parentReference = addObject(objectBuilder.takeObject());
for (size_t i = 0; i < children.size(); ++i)
{
PDFObjectFactory fixPointersObjectBuilder;
fixPointersObjectBuilder.beginDictionary();
fixPointersObjectBuilder.beginDictionaryItem("Parent");
fixPointersObjectBuilder << parentReference;
fixPointersObjectBuilder.endDictionaryItem();
if (i > 0)
{
fixPointersObjectBuilder.beginDictionaryItem("Prev");
fixPointersObjectBuilder << children[i - 1];
fixPointersObjectBuilder.endDictionaryItem();
}
if (i + 1 < children.size())
{
fixPointersObjectBuilder.beginDictionaryItem("Next");
fixPointersObjectBuilder << children[i + 1];
fixPointersObjectBuilder.endDictionaryItem();
}
fixPointersObjectBuilder.endDictionary();
mergeTo(children[i], fixPointersObjectBuilder.takeObject());
}
return parentReference;
}
void PDFDocumentBuilder::flattenPageTree()
{
PDFObjectReference pageTreeRoot = getPageTreeRoot();
@@ -1166,6 +1275,11 @@ std::vector<PDFObjectReference> PDFDocumentBuilder::getPages() const
return result;
}
void PDFDocumentBuilder::setOutline(const PDFOutlineItem* root)
{
setOutline(createOutlineItem(root, false));
}
std::vector<PDFObject> PDFDocumentBuilder::copyFrom(const std::vector<PDFObject>& objects, const PDFObjectStorage& storage, bool createReferences)
{
// 1) Collect all references, which we must copy. If object is referenced, then
@@ -4180,6 +4294,20 @@ void PDFDocumentBuilder::setLanguage(QLocale locale)
}
void PDFDocumentBuilder::setOutline(PDFObjectReference outline)
{
PDFObjectFactory objectBuilder;
objectBuilder.beginDictionary();
objectBuilder.beginDictionaryItem("Outlines");
objectBuilder << outline;
objectBuilder.endDictionaryItem();
objectBuilder.endDictionary();
PDFObject updatedCatalog = objectBuilder.takeObject();
mergeTo(getCatalogReference(), updatedCatalog);
}
void PDFDocumentBuilder::setPageArtBox(PDFObjectReference page,
QRectF box)
{
@@ -4270,6 +4398,21 @@ void PDFDocumentBuilder::setPageTrimBox(PDFObjectReference page,
}
void PDFDocumentBuilder::setPageUserUnit(PDFObjectReference page,
PDFReal unit)
{
PDFObjectFactory objectBuilder;
objectBuilder.beginDictionary();
objectBuilder.beginDictionaryItem("UserUnit");
objectBuilder << unit;
objectBuilder.endDictionaryItem();
objectBuilder.endDictionary();
PDFObject updatedPageObject = objectBuilder.takeObject();
mergeTo(page, updatedPageObject);
}
void PDFDocumentBuilder::updateTrailerDictionary(PDFInteger objectCount)
{
PDFObjectFactory objectBuilder;

View File

@@ -305,6 +305,11 @@ public:
/// be flattened to use this function. \sa flattenPageTree
std::vector<PDFObjectReference> getPages() const;
/// Sets document outline root item corresponds to invisible root.
/// Top-level items are children of the root.
/// \param root Root item
void setOutline(const PDFOutlineItem* root);
/// Adds a new objet to the object storage
/// \param object Object
PDFObjectReference addObject(PDFObject object);
@@ -1250,6 +1255,11 @@ public:
void setLanguage(QLocale locale);
/// Set document outline.
/// \param outline Document outline root
void setOutline(PDFObjectReference outline);
/// Sets art box to the page. Art box defines page's meaningful content.
/// \param page Page
/// \param box Box
@@ -1296,6 +1306,13 @@ public:
QRectF box);
/// Sets page's user unit. It specifies user space unit, in multiples of 1 / 72 inch.
/// \param page Page
/// \param unit Unit (multiple of 1pt = 1 / 72 inch)
void setPageUserUnit(PDFObjectReference page,
PDFReal unit);
/// This function is used to update trailer dictionary. Must be called each time the final document is
/// being built.
/// \param objectCount Number of objects (including empty ones)
@@ -1312,6 +1329,7 @@ private:
PDFObjectReference getDocumentInfo() const;
void updateDocumentInfo(PDFObject info);
QRectF getPolygonsBoundingRect(const Polygons& Polygons) const;
PDFObjectReference createOutlineItem(const PDFOutlineItem* root, bool writeOutlineData);
PDFObjectStorage m_storage;
PDFVersion m_version;

View File

@@ -23,6 +23,18 @@
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 PDFDocument* document, const PDFObject& root)
{
const PDFObject& rootDereferenced = document->getObject(root);

View File

@@ -31,7 +31,7 @@ namespace pdf
class PDFDocument;
/// Outline item
class PDFOutlineItem
class PDFFORQTLIBSHARED_EXPORT PDFOutlineItem
{
public:
explicit PDFOutlineItem() = default;
@@ -40,6 +40,7 @@ public:
void setTitle(const QString& title) { m_title = title; }
size_t getChildCount() const { return m_children.size(); }
size_t getTotalCount() const;
const PDFOutlineItem* getChild(size_t index) const { return m_children[index].get(); }
void addChild(QSharedPointer<PDFOutlineItem> child) { m_children.emplace_back(qMove(child)); }
QSharedPointer<PDFOutlineItem> getChildPtr(size_t index) const { return m_children[index]; }