UBExportFullPDF.cpp 9.49 KB
Newer Older
Claudio Valerio's avatar
Claudio Valerio committed
1
/*
Craig Watson's avatar
Craig Watson committed
2 3
 * Copyright (C) 2015-2016 Département de l'Instruction Publique (DIP-SEM)
 *
Claudio Valerio's avatar
Claudio Valerio committed
4
 * Copyright (C) 2013 Open Education Foundation
Claudio Valerio's avatar
Claudio Valerio committed
5
 *
Claudio Valerio's avatar
Claudio Valerio committed
6 7
 * Copyright (C) 2010-2013 Groupement d'Intérêt Public pour
 * l'Education Numérique en Afrique (GIP ENA)
8
 *
Claudio Valerio's avatar
Claudio Valerio committed
9 10 11
 * This file is part of OpenBoard.
 *
 * OpenBoard is free software: you can redistribute it and/or modify
Claudio Valerio's avatar
Claudio Valerio committed
12 13
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, version 3 of the License,
14 15 16 17
 * with a specific linking exception for the OpenSSL project's
 * "OpenSSL" library (or with modified versions of it that use the
 * same license as the "OpenSSL" library).
 *
Claudio Valerio's avatar
Claudio Valerio committed
18
 * OpenBoard is distributed in the hope that it will be useful,
Claudio Valerio's avatar
Claudio Valerio committed
19
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Claudio Valerio's avatar
Claudio Valerio committed
20
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Claudio Valerio's avatar
Claudio Valerio committed
21
 * GNU General Public License for more details.
Claudio Valerio's avatar
Claudio Valerio committed
22
 *
Claudio Valerio's avatar
Claudio Valerio committed
23
 * You should have received a copy of the GNU General Public License
Claudio Valerio's avatar
Claudio Valerio committed
24
 * along with OpenBoard. If not, see <http://www.gnu.org/licenses/>.
Claudio Valerio's avatar
Claudio Valerio committed
25 26
 */

27

Claudio Valerio's avatar
Claudio Valerio committed
28

Claudio Valerio's avatar
Claudio Valerio committed
29

Claudio Valerio's avatar
Claudio Valerio committed
30 31 32 33
#include "UBExportFullPDF.h"

#include <QtCore>
#include <QtSvg>
34
#include <QPrinter>
Claudio Valerio's avatar
Claudio Valerio committed
35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54

#include "core/UBApplication.h"
#include "core/UBSettings.h"
#include "core/UBSetting.h"
#include "core/UBPersistenceManager.h"

#include "domain/UBGraphicsScene.h"
#include "domain/UBGraphicsSvgItem.h"
#include "domain/UBGraphicsPDFItem.h"

#include "document/UBDocumentProxy.h"

#include "pdf/GraphicsPDFItem.h"

#include "UBExportPDF.h"

#include <Merger.h>
#include <Exception.h>
#include <Transformation.h>

55 56 57
#include "core/memcheck.h"


Claudio Valerio's avatar
Claudio Valerio committed
58 59 60 61 62 63
using namespace merge_lib;


UBExportFullPDF::UBExportFullPDF(QObject *parent)
    : UBExportAdaptor(parent)
{
64
    //need to calculate screen resolution
Claudio Valerio's avatar
Claudio Valerio committed
65 66
    QDesktopWidget* desktop = UBApplication::desktop();
    int dpiCommon = (desktop->physicalDpiX() + desktop->physicalDpiY()) / 2;
67
    mScaleFactor = 72.0f / dpiCommon; // 1pt = 1/72 inch
68 69

    mSimpleExporter = new UBExportPDF();
Claudio Valerio's avatar
Claudio Valerio committed
70 71 72 73 74 75 76 77 78
}


UBExportFullPDF::~UBExportFullPDF()
{
    // NOOP
}


79
void UBExportFullPDF::saveOverlayPdf(UBDocumentProxy* pDocumentProxy, const QString& filename)
Claudio Valerio's avatar
Claudio Valerio committed
80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
{
    if (!pDocumentProxy || filename.length() == 0 || pDocumentProxy->pageCount() == 0)
        return;

    //PDF
    qDebug() << "exporting document to PDF Merger" << filename;
    QPrinter pdfPrinter;

    pdfPrinter.setOutputFormat(QPrinter::PdfFormat);
    pdfPrinter.setResolution(UBSettings::settings()->pdfResolution->get().toInt());
    pdfPrinter.setOutputFileName(filename);
    pdfPrinter.setFullPage(true);

    QPainter* pdfPainter = 0;

    for(int pageIndex = 0 ; pageIndex < pDocumentProxy->pageCount(); pageIndex++)
    {
        UBGraphicsScene* scene = UBPersistenceManager::persistenceManager()->loadDocumentScene(pDocumentProxy, pageIndex);
        // set background to white, no grid for PDF output
        bool isDark = scene->isDarkBackground();
Craig Watson's avatar
Craig Watson committed
100 101
        UBPageBackground pageBackground = scene->pageBackground();
        scene->setBackground(false, UBPageBackground::plain);
Claudio Valerio's avatar
Claudio Valerio committed
102 103 104 105 106

        // set high res rendering
        scene->setRenderingQuality(UBItem::RenderingQualityHigh);
        scene->setRenderingContext(UBGraphicsScene::PdfExport);

Claudio Valerio's avatar
Claudio Valerio committed
107
        QSize pageSize = scene->nominalSize();
Claudio Valerio's avatar
Claudio Valerio committed
108

Claudio Valerio's avatar
Claudio Valerio committed
109
        UBGraphicsPDFItem *pdfItem = qgraphicsitem_cast<UBGraphicsPDFItem*>(scene->backgroundObject());
Claudio Valerio's avatar
Claudio Valerio committed
110

111 112
        if (pdfItem) mHasPDFBackgrounds = true;
        
Claudio Valerio's avatar
Claudio Valerio committed
113
        pdfPrinter.setPaperSize(QSizeF(pageSize.width()*mScaleFactor, pageSize.height()*mScaleFactor), QPrinter::Point);
Claudio Valerio's avatar
Claudio Valerio committed
114

115
        if (!pdfPainter) pdfPainter = new QPainter(&pdfPrinter);
Claudio Valerio's avatar
Claudio Valerio committed
116

Claudio Valerio's avatar
Claudio Valerio committed
117
        if (pageIndex != 0) pdfPrinter.newPage();
Claudio Valerio's avatar
Claudio Valerio committed
118

119
        //render to PDF
120
        scene->setDrawingMode(true);
Anatoly Mihalchenko's avatar
Anatoly Mihalchenko committed
121
        scene->render(pdfPainter, QRectF(), scene->normalizedSceneRect());
Claudio Valerio's avatar
Claudio Valerio committed
122 123 124 125 126 127

        //restore screen rendering quality
        scene->setRenderingContext(UBGraphicsScene::Screen);
        scene->setRenderingQuality(UBItem::RenderingQualityNormal);

        //restore background state
128
        scene->setDrawingMode(false);
Craig Watson's avatar
Craig Watson committed
129
        scene->setBackground(isDark, pageBackground);
Claudio Valerio's avatar
Claudio Valerio committed
130 131
    }

132
    if (pdfPainter) delete pdfPainter;
Claudio Valerio's avatar
Claudio Valerio committed
133 134 135 136 137
}


void UBExportFullPDF::persist(UBDocumentProxy* pDocumentProxy)
{
138
    persistLocally(pDocumentProxy, tr("Export as PDF File"));
Claudio Valerio's avatar
Claudio Valerio committed
139 140 141
}


142
bool UBExportFullPDF::persistsDocument(UBDocumentProxy* pDocumentProxy, const QString& filename)
Claudio Valerio's avatar
Claudio Valerio committed
143 144
{
    QFile file(filename);
145
    if (file.exists()) file.remove();
Claudio Valerio's avatar
Claudio Valerio committed
146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172

    QString overlayName = filename;
    overlayName.replace(".pdf", "_overlay.pdf");

    QFile previousOverlay(overlayName);
    if (previousOverlay.exists())
        previousOverlay.remove();

    mHasPDFBackgrounds = false;

    saveOverlayPdf(pDocumentProxy, overlayName);

    if (!mHasPDFBackgrounds)
    {
        QFile f(overlayName);
        f.rename(filename);
    }
    else
    {
        Merger merger;
        try
        {
            merger.addOverlayDocument(QFile::encodeName(overlayName).constData());

            MergeDescription mergeInfo;

            int existingPageCount = pDocumentProxy->pageCount();
shibakaneki's avatar
shibakaneki committed
173

Claudio Valerio's avatar
Claudio Valerio committed
174 175 176 177 178
            for(int pageIndex = 0 ; pageIndex < existingPageCount; pageIndex++)
            {
                UBGraphicsScene* scene = UBPersistenceManager::persistenceManager()->loadDocumentScene(pDocumentProxy, pageIndex);
                UBGraphicsPDFItem *pdfItem = qgraphicsitem_cast<UBGraphicsPDFItem*>(scene->backgroundObject());

Claudio Valerio's avatar
Claudio Valerio committed
179
                QSize pageSize = scene->nominalSize();
180
                
Claudio Valerio's avatar
Claudio Valerio committed
181
                if (pdfItem)
Claudio Valerio's avatar
Claudio Valerio committed
182 183 184
                {
                    QString pdfName = UBPersistenceManager::objectDirectory + "/" + pdfItem->fileUuid().toString() + ".pdf";
                    QString backgroundPath = pDocumentProxy->persistencePath() + "/" + pdfName;
185
                    QRectF annotationsRect = scene->annotationsBoundingRect();
shibakaneki's avatar
shibakaneki committed
186 187 188 189 190 191 192 193 194

                    // Original datas
                    double xAnnotation = qRound(annotationsRect.x());
                    double yAnnotation = qRound(annotationsRect.y());
                    double xPdf = qRound(pdfItem->sceneBoundingRect().x());
                    double yPdf = qRound(pdfItem->sceneBoundingRect().y());
                    double hPdf = qRound(pdfItem->sceneBoundingRect().height());

                    // Exportation-transformed datas
195 196
                    double hScaleFactor = pageSize.width()/annotationsRect.width();
                    double vScaleFactor = pageSize.height()/annotationsRect.height();
shibakaneki's avatar
shibakaneki committed
197 198 199 200 201 202 203 204 205 206 207 208 209
                    double scaleFactor = qMin(hScaleFactor, vScaleFactor);

                    double xAnnotationsOffset = 0;
                    double yAnnotationsOffset = 0;
                    double hPdfTransformed = qRound(hPdf * scaleFactor);

                    // Here, we force the PDF page to be on the topleft corner of the page
                    double xPdfOffset = 0;
                    double yPdfOffset = (hPdf - hPdfTransformed) * mScaleFactor;

                    // Now we align the items
                    xPdfOffset += (xPdf - xAnnotation) * scaleFactor * mScaleFactor;
                    yPdfOffset -= (yPdf - yAnnotation) * scaleFactor * mScaleFactor;
Claudio Valerio's avatar
Claudio Valerio committed
210

211 212
                    // If the PDF was scaled when added to the scene (e.g if it was loaded from a document with a different DPI
                    // than the current one), it should also be scaled here.
213
                    qreal pdfScale = pdfItem->scale();
214 215

                    TransformationDescription pdfTransform(xPdfOffset, yPdfOffset, scaleFactor * pdfScale, 0);
shibakaneki's avatar
shibakaneki committed
216
                    TransformationDescription annotationTransform(xAnnotationsOffset, yAnnotationsOffset, 1, 0);
Claudio Valerio's avatar
Claudio Valerio committed
217

218 219
                    MergePageDescription pageDescription(pageSize.width() * mScaleFactor,
                                                         pageSize.height() * mScaleFactor,
Claudio Valerio's avatar
Claudio Valerio committed
220 221
                                                         pdfItem->pageNumber(),
                                                         QFile::encodeName(backgroundPath).constData(),
shibakaneki's avatar
shibakaneki committed
222
                                                         pdfTransform,
Claudio Valerio's avatar
Claudio Valerio committed
223
                                                         pageIndex + 1,
shibakaneki's avatar
shibakaneki committed
224
                                                         annotationTransform,
Claudio Valerio's avatar
Claudio Valerio committed
225 226 227 228 229 230 231 232
                                                         false, false);

                    mergeInfo.push_back(pageDescription);

                    merger.addBaseDocument(QFile::encodeName(backgroundPath).constData());
                }
                else
                {
233 234
                    MergePageDescription pageDescription(pageSize.width() * mScaleFactor,
                             pageSize.height() * mScaleFactor,
Claudio Valerio's avatar
Claudio Valerio committed
235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255
                             0,
                             "",
                             TransformationDescription(),
                             pageIndex + 1,
                             TransformationDescription(),
                             false, true);

                    mergeInfo.push_back(pageDescription);
                }
            }

            merger.merge(QFile::encodeName(overlayName).constData(), mergeInfo);

            merger.saveMergedDocumentsAs(QFile::encodeName(filename).constData());

        }
        catch(Exception e)
        {
            qDebug() << "PdfMerger failed to merge documents to " << filename << " - Exception : " << e.what();

            // default to raster export
256
            mSimpleExporter->persistsDocument(pDocumentProxy, filename);
Claudio Valerio's avatar
Claudio Valerio committed
257 258 259 260 261 262 263
        }

        if (!UBApplication::app()->isVerbose())
        {
            QFile::remove(overlayName);
        }
    }
264 265

    return true;
Claudio Valerio's avatar
Claudio Valerio committed
266 267 268 269 270 271 272 273 274 275 276 277
}


QString UBExportFullPDF::exportExtention()
{
    return QString(".pdf");
}

QString UBExportFullPDF::exportName()
{
    return tr("Export to PDF");
}