/**************************************************************************** ** ** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies). ** All rights reserved. ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** This file is part of the QtGui module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** GNU Lesser General Public License Usage ** This file may be used under the terms of the GNU Lesser General Public ** License version 2.1 as published by the Free Software Foundation and ** appearing in the file LICENSE.LGPL included in the packaging of this ** file. Please review the following information to ensure the GNU Lesser ** General Public License version 2.1 requirements will be met: ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU General ** Public License version 3.0 as published by the Free Software Foundation ** and appearing in the file LICENSE.GPL included in the packaging of this ** file. Please review the following information to ensure the GNU General ** Public License version 3.0 requirements will be met: ** http://www.gnu.org/copyleft/gpl.html. ** ** Other Usage ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** ** ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #ifndef QPDF_P_H #define QPDF_P_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists purely as an // implementation detail. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include "QtGui/qmatrix.h" #include "QtCore/qstring.h" #include "QtCore/qvector.h" #include "private/qstroker_p.h" #include "private/qpaintengine_p.h" #include "private/qfontengine_p.h" #include "private/qfontsubset_p.h" // #define USE_NATIVE_GRADIENTS QT_BEGIN_NAMESPACE const char *qt_real_to_string(qreal val, char *buf); const char *qt_int_to_string(int val, char *buf); namespace QPdf { class ByteStream { public: // fileBacking means that ByteStream will buffer the contents on disk // if the size exceeds a certain threshold. In this case, if a byte // array was passed in, its contents may no longer correspond to the // ByteStream contents. explicit ByteStream(bool fileBacking = false); explicit ByteStream(QByteArray *ba, bool fileBacking = false); ~ByteStream(); ByteStream &operator <<(char chr); ByteStream &operator <<(const char *str); ByteStream &operator <<(const QByteArray &str); ByteStream &operator <<(const ByteStream &src); ByteStream &operator <<(qreal val); ByteStream &operator <<(int val); ByteStream &operator <<(const QPointF &p); // Note that the stream may be invalidated by calls that insert data. QIODevice *stream(); void clear(); static inline int maxMemorySize() { return 100000000; } static inline int chunkSize() { return 10000000; } protected: void constructor_helper(QIODevice *dev); void constructor_helper(QByteArray *ba); private: void prepareBuffer(); private: QIODevice *dev; QByteArray ba; bool fileBackingEnabled; bool fileBackingActive; bool handleDirty; }; enum PathFlags { ClipPath, FillPath, StrokePath, FillAndStrokePath }; QByteArray generatePath(const QPainterPath &path, const QTransform &matrix, PathFlags flags); QByteArray generateMatrix(const QTransform &matrix); QByteArray generateDashes(const QPen &pen); QByteArray patternForBrush(const QBrush &b); #ifdef USE_NATIVE_GRADIENTS QByteArray generateLinearGradientShader(const QLinearGradient *lg, const QPointF *page_rect, bool alpha = false); #endif struct Stroker { Stroker(); void setPen(const QPen &pen); void strokePath(const QPainterPath &path); ByteStream *stream; bool first; QTransform matrix; bool cosmeticPen; private: QStroker basicStroker; QDashStroker dashStroker; QStrokerOps *stroker; }; QByteArray ascii85Encode(const QByteArray &input); const char *toHex(ushort u, char *buffer); const char *toHex(uchar u, char *buffer); } class QPdfPage : public QPdf::ByteStream { public: QPdfPage(); QVector images; QVector graphicStates; QVector patterns; QVector fonts; QVector annotations; void streamImage(int w, int h, int object); QSize pageSize; private: }; class QPdfWriter; class QPdfEnginePrivate; class Q_GUI_EXPORT QPdfEngine : public QPaintEngine { Q_DECLARE_PRIVATE(QPdfEngine) friend class QPdfWriter; public: QPdfEngine(); QPdfEngine(QPdfEnginePrivate &d); ~QPdfEngine() {} void setOutputFilename(const QString &filename); inline void setResolution(int resolution); // reimplementations QPaintEngine bool begin(QPaintDevice *pdev); bool end(); void drawPoints(const QPointF *points, int pointCount); void drawLines(const QLineF *lines, int lineCount); void drawRects(const QRectF *rects, int rectCount); void drawPolygon(const QPointF *points, int pointCount, PolygonDrawMode mode); void drawPath (const QPainterPath & path); void drawTextItem(const QPointF &p, const QTextItem &textItem); void drawPixmap (const QRectF & rectangle, const QPixmap & pixmap, const QRectF & sr); void drawImage(const QRectF &r, const QImage &pm, const QRectF &sr, Qt::ImageConversionFlags flags = Qt::AutoColor); void drawTiledPixmap (const QRectF & rectangle, const QPixmap & pixmap, const QPointF & point); void updateState(const QPaintEngineState &state); int metric(QPaintDevice::PaintDeviceMetric metricType) const; Type type() const; // end reimplementations QPaintEngine // Printer stuff... bool newPage(); void setPen(); void setBrush(); void setupGraphicsState(QPaintEngine::DirtyFlags flags); private: void updateClipPath(const QPainterPath & path, Qt::ClipOperation op); }; class Q_GUI_EXPORT QPdfEnginePrivate : public QPaintEnginePrivate { Q_DECLARE_PUBLIC(QPdfEngine) public: QPdfEnginePrivate(); ~QPdfEnginePrivate(); inline uint requestObject() { return currentObject++; } QRect paperRect() const; QRect pageRect() const; int width() const { QRect r = paperRect(); return qRound(r.width()*72./resolution); } int height() const { QRect r = paperRect(); return qRound(r.height()*72./resolution); } void writeHeader(); void writeTail(); int addImage(const QImage &image, bool *bitmap, qint64 serial_no); int addConstantAlphaObject(int brushAlpha, int penAlpha = 255); int addBrushPattern(const QTransform &matrix, bool *specifyColor, int *gStateObject); void drawTextItem(const QPointF &p, const QTextItemInt &ti); QTransform pageMatrix() const; void newPage(); bool postscript; int currentObject; QPdfPage* currentPage; QPdf::Stroker stroker; QPointF brushOrigin; QBrush brush; QPen pen; QList clips; bool clipEnabled; bool allClipped; bool hasPen; bool hasBrush; bool simplePen; qreal opacity; QHash fonts; QPaintDevice *pdev; // the device the output is in the end streamed to. QIODevice *outDevice; bool ownsDevice; // printer options QString outputFileName; QString title; QString creator; bool fullPage; bool embedFonts; int resolution; bool landscape; bool grayscale; // in postscript points QSizeF paperSize; qreal leftMargin, topMargin, rightMargin, bottomMargin; #if !defined(QT_NO_CUPS) && !defined(QT_NO_LIBRARY) QString cupsTempFile; #endif private: #ifdef USE_NATIVE_GRADIENTS int gradientBrush(const QBrush &b, const QMatrix &matrix, int *gStateObject); #endif void writeInfo(); void writePageRoot(); void writeFonts(); void embedFont(QFontSubset *font); QVector xrefPositions; QDataStream* stream; int streampos; int writeImage(const QByteArray &data, int width, int height, int depth, int maskObject, int softMaskObject, bool dct = false); void writePage(); int addXrefEntry(int object, bool printostr = true); void printString(const QString &string); void xprintf(const char* fmt, ...); inline void write(const QByteArray &data) { stream->writeRawData(data.constData(), data.size()); streampos += data.size(); } int writeCompressed(const char *src, int len); inline int writeCompressed(const QByteArray &data) { return writeCompressed(data.constData(), data.length()); } int writeCompressed(QIODevice *dev); // various PDF objects int pageRoot, catalog, info, graphicsState, patternColorSpace; QVector pages; QHash imageCache; QHash, uint > alphaCache; }; void QPdfEngine::setResolution(int resolution) { Q_D(QPdfEngine); d->resolution = resolution; } QT_END_NAMESPACE #endif // QPDF_P_H