// Copyright (C) 2020 Jakub Melka // // This file is part of PdfForQt. // // PdfForQt 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 // (at your option) any later version. // // PdfForQt 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 PDFForQt. If not, see . #include "pdfdocumentwriter.h" #include "pdfconstants.h" #include "pdfvisitor.h" #include "pdfparser.h" #include namespace pdf { class PDFWriteObjectVisitor : public PDFAbstractVisitor { public: explicit PDFWriteObjectVisitor(QIODevice* device) : m_device(device) { } virtual void visitNull() override; virtual void visitBool(bool value) override; virtual void visitInt(PDFInteger value) override; virtual void visitReal(PDFReal value) override; virtual void visitString(const PDFString* string) override; virtual void visitName(const PDFString* name) override; virtual void visitArray(const PDFArray* array) override; virtual void visitDictionary(const PDFDictionary* dictionary) override; virtual void visitStream(const PDFStream* stream) override; virtual void visitReference(const PDFObjectReference reference) override; PDFObject getDecryptedObject(); private: void writeName(const QByteArray& string); QIODevice* m_device; }; void PDFWriteObjectVisitor::visitNull() { m_device->write("null "); } void PDFWriteObjectVisitor::visitBool(bool value) { if (value) { m_device->write("true "); } else { m_device->write("false "); } } void PDFWriteObjectVisitor::visitInt(PDFInteger value) { m_device->write(QString::number(value).toLatin1()); m_device->write(" "); } void PDFWriteObjectVisitor::visitReal(PDFReal value) { // Jakub Melka: we use 5 digits, because they are specified // in PDF 1.7 specification, appendix C, Table C.1, where it is defined, // that number of significant digits of precision is 5. m_device->write(QString::number(value, 'f', 5).toLatin1()); m_device->write(" "); } void PDFWriteObjectVisitor::visitString(const PDFString* string) { const QByteArray& data = string->getString(); if (data.indexOf('(') != -1 || data.indexOf(')') != -1 || data.indexOf('\\') != -1) { m_device->write("<"); m_device->write(data.toHex()); m_device->write(">"); } else { m_device->write("("); m_device->write(data); m_device->write(")"); } m_device->write(" "); } void PDFWriteObjectVisitor::writeName(const QByteArray& string) { m_device->write("/"); for (const char character : string) { if (PDFLexicalAnalyzer::isRegular(character)) { m_device->write(&character, 1); } else { m_device->write("#"); m_device->write(QByteArray(&character, 1).toHex()); } } m_device->write(" "); } void PDFWriteObjectVisitor::visitName(const PDFString* name) { writeName(name->getString()); } void PDFWriteObjectVisitor::visitArray(const PDFArray* array) { m_device->write("[ "); acceptArray(array); m_device->write("] "); } void PDFWriteObjectVisitor::visitDictionary(const PDFDictionary* dictionary) { m_device->write("<< "); for (size_t i = 0, count = dictionary->getCount(); i < count; ++i) { writeName(dictionary->getKey(i)); dictionary->getValue(i).accept(this); } m_device->write(">> "); } void PDFWriteObjectVisitor::visitStream(const PDFStream* stream) { visitDictionary(stream->getDictionary()); m_device->write("stream"); m_device->write("\x0D\x0A"); m_device->write(*stream->getContent()); m_device->write("\x0D\x0A"); m_device->write("endstream"); m_device->write("\x0D\x0A"); } void PDFWriteObjectVisitor::visitReference(const PDFObjectReference reference) { visitInt(reference.objectNumber); visitInt(reference.generation); m_device->write("R "); } PDFOperationResult PDFDocumentWriter::write(const QString& fileName, const PDFDocument* document) { QFile file(fileName); if (file.open(QFile::WriteOnly | QFile::Truncate)) { PDFOperationResult result = write(&file, document); file.close(); return result; } else { return tr("File '%1' can't be opened for writing. %2").arg(fileName, file.errorString()); } } PDFOperationResult PDFDocumentWriter::write(QIODevice* device, const PDFDocument* document) { if (!device->isWritable()) { return tr("Device is not writable."); } const PDFObjectStorage& storage = document->getStorage(); const PDFObjectStorage::PDFObjects& objects = storage.getObjects(); const size_t objectCount = objects.size(); if (storage.getSecurityHandler()->getMode() != EncryptionMode::None) { return tr("Writing of encrypted documents is not supported."); } // Write header PDFVersion version = document->getInfo()->version; device->write(QString("%PDF-%1.%2").arg(version.major).arg(version.minor).toLatin1()); writeCRLF(device); device->write("% PDF producer: "); device->write(PDF_LIBRARY_NAME); writeCRLF(device); writeCRLF(device); writeCRLF(device); // Write objects std::vector offsets(objectCount, -1); for (size_t i = 0; i < objectCount; ++i) { const PDFObjectStorage::Entry& entry = objects[i]; if (entry.object.isNull()) { continue; } // Jakub Melka: we must mark actual position of object offsets[i] = device->pos(); PDFWriteObjectVisitor visitor(device); writeObjectHeader(device, PDFObjectReference(i, entry.generation)); entry.object.accept(&visitor); writeObjectFooter(device); } // Write cross-reference table PDFInteger xrefOffset = device->pos(); device->write("xref"); writeCRLF(device); device->write(QString("0 %1").arg(objectCount).toLatin1()); writeCRLF(device); for (size_t i = 0; i < objectCount; ++i) { const PDFObjectStorage::Entry& entry = objects[i]; PDFInteger generation = entry.generation; if (i == 0) { generation = 65535; } PDFInteger offset = offsets[i]; if (offset == -1) { offset = 0; } QString offsetString = QString::number(offset).rightJustified(10, QChar('0'), true); QString generationString = QString::number(generation).rightJustified(5, QChar('0'), true); device->write(offsetString.toLatin1()); device->write(" "); device->write(generationString.toLatin1()); device->write(" "); device->write(entry.object.isNull() ? "f" : "n"); writeCRLF(device); } device->write("trailer"); writeCRLF(device); PDFWriteObjectVisitor trailerVisitor(device); storage.getTrailerDictionary().accept(&trailerVisitor); writeCRLF(device); device->write("startxref"); writeCRLF(device); device->write(QString::number(xrefOffset).toLatin1()); writeCRLF(device); // Write footer device->write("%%EOF"); return true; } void PDFDocumentWriter::writeCRLF(QIODevice* device) { device->write("\x0D\x0A"); } void PDFDocumentWriter::writeObjectHeader(QIODevice* device, PDFObjectReference reference) { QString objectHeader = QString("%1 %2 obj").arg(QString::number(reference.objectNumber)).arg(QString::number(reference.generation)); device->write(objectHeader.toLatin1()); writeCRLF(device); } void PDFDocumentWriter::writeObjectFooter(QIODevice* device) { device->write("endobj"); writeCRLF(device); } } // namespace pdf