2021-04-30 20:12:10 +02:00
// Copyright (C) 2019-2021 Jakub Melka
2019-02-14 19:45:07 +01:00
//
2020-12-20 19:03:58 +01:00
// This file is part of Pdf4Qt.
2019-02-14 19:45:07 +01:00
//
2020-12-20 19:03:58 +01:00
// Pdf4Qt is free software: you can redistribute it and/or modify
2019-02-14 19:45:07 +01:00
// 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
2021-04-30 20:12:10 +02:00
// with the written consent of the copyright owner, any later version.
2019-02-14 19:45:07 +01:00
//
2020-12-20 19:03:58 +01:00
// Pdf4Qt is distributed in the hope that it will be useful,
2019-02-14 19:45:07 +01:00
// 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
2020-12-20 19:03:58 +01:00
// along with Pdf4Qt. If not, see <https://www.gnu.org/licenses/>.
2019-02-14 19:45:07 +01:00
2019-02-24 17:48:37 +01:00
# ifndef PDFPAGECONTENTPROCESSOR_H
# define PDFPAGECONTENTPROCESSOR_H
2019-02-14 19:45:07 +01:00
# include "pdfrenderer.h"
# include "pdfcolorspaces.h"
2019-04-29 17:03:19 +02:00
# include "pdfparser.h"
2019-03-25 18:44:45 +01:00
# include "pdffont.h"
2019-03-30 18:45:30 +01:00
# include "pdfutils.h"
2019-09-28 18:26:31 +02:00
# include "pdfmeshqualitysettings.h"
2019-09-29 15:44:35 +02:00
# include "pdfblendfunction.h"
2019-12-28 19:21:29 +01:00
# include "pdftextlayout.h"
2019-02-14 19:45:07 +01:00
# include <QMatrix>
# include <QPainterPath>
# include <QSharedPointer>
# include <stack>
# include <tuple>
# include <type_traits>
namespace pdf
{
2019-12-25 17:56:17 +01:00
class PDFCMS ;
2019-08-31 15:55:59 +02:00
class PDFMesh ;
2021-02-21 16:42:24 +01:00
class PDFImage ;
2019-09-25 19:17:52 +02:00
class PDFTilingPattern ;
2021-03-01 15:24:04 +01:00
class PDFShadingPattern ;
2019-07-04 17:52:38 +02:00
class PDFOptionalContentActivity ;
2019-02-14 19:45:07 +01:00
2019-07-04 17:52:38 +02:00
static constexpr const char * PDF_RESOURCE_EXTGSTATE = " ExtGState " ;
2019-06-15 14:29:49 +02:00
2020-03-29 18:53:04 +02:00
class PDFLineDashPattern
{
public :
explicit inline PDFLineDashPattern ( ) = default ;
2020-08-09 17:10:42 +02:00
explicit PDFLineDashPattern ( const std : : vector < PDFReal > & dashArray , PDFReal dashOffset ) ;
2020-03-29 18:53:04 +02:00
inline const std : : vector < PDFReal > & getDashArray ( ) const { return m_dashArray ; }
inline void setDashArray ( const std : : vector < PDFReal > & dashArray ) { m_dashArray = dashArray ; }
inline PDFReal getDashOffset ( ) const { return m_dashOffset ; }
inline void setDashOffset ( PDFReal dashOffset ) { m_dashOffset = dashOffset ; }
inline bool operator = = ( const PDFLineDashPattern & other ) const { return m_dashArray = = other . m_dashArray & & m_dashOffset = = other . m_dashOffset ; }
inline bool operator ! = ( const PDFLineDashPattern & other ) const { return ! ( * this = = other ) ; }
/// Is line solid? Function returns true, if yes.
bool isSolid ( ) const { return m_dashArray . empty ( ) ; }
2020-08-09 17:10:42 +02:00
/// Fix line dash pattern according to the specification
void fix ( ) ;
2020-03-29 18:53:04 +02:00
private :
std : : vector < PDFReal > m_dashArray ;
PDFReal m_dashOffset = 0.0 ;
} ;
2019-02-14 19:45:07 +01:00
/// Process the contents of the page.
2020-12-20 19:03:58 +01:00
class Pdf4QtLIBSHARED_EXPORT PDFPageContentProcessor : public PDFRenderErrorReporter
2019-02-14 19:45:07 +01:00
{
public :
2019-07-04 17:52:38 +02:00
explicit PDFPageContentProcessor ( const PDFPage * page ,
const PDFDocument * document ,
const PDFFontCache * fontCache ,
2019-12-25 17:56:17 +01:00
const PDFCMS * CMS ,
2019-08-25 18:16:37 +02:00
const PDFOptionalContentActivity * optionalContentActivity ,
2019-09-28 18:26:31 +02:00
QMatrix pagePointToDevicePointMatrix ,
const PDFMeshQualitySettings & meshQualitySettings ) ;
2019-02-24 17:48:37 +01:00
virtual ~ PDFPageContentProcessor ( ) ;
2019-02-14 19:45:07 +01:00
enum class Operator
{
// General graphic state w, J, j, M, d, ri, i, gs
SetLineWidth , ///< w, sets the line width
SetLineCap , ///< J, sets the line cap
SetLineJoin , ///< j, sets the line join
SetMitterLimit , ///< M, sets the mitter limit
SetLineDashPattern , ///< d, sets the line dash pattern
SetRenderingIntent , ///< ri, sets the rendering intent
SetFlatness , ///< i, sets the flattness (number in range from 0 to 100)
SetGraphicState , ///< gs, sets the whole graphic state (stored in resource dictionary)
// Special graphic state: q, Q, cm
SaveGraphicState , ///< q, saves the graphic state
RestoreGraphicState , ///< Q, restores the graphic state
AdjustCurrentTransformationMatrix , ///< cm, modify the current transformation matrix by matrix multiplication
// Path construction: m, l, c, v, y, h, re
MoveCurrentPoint , ///< m, begin a new subpath by moving to the desired point
LineTo , ///< l, appends a straight line segment to the subpath
Bezier123To , ///< c, appends a Bézier curve with control points 1, 2, 3
Bezier23To , ///< v, appends a Bézier curve with control points 2, 3
Bezier13To , ///< y, appends a Bézier curve with control points 1, 3
EndSubpath , ///< h, ends current subpath by adding straight line segment from the last point to the beginning
Rectangle , ///< re, adds rectangle
// Path painting: S, s, f, F, f*, B, B*, b, b*, n
2019-02-16 18:26:16 +01:00
PathStroke , ///< S, Stroke
PathCloseStroke , ///< s, Close, Stroke (equivalent of operators h S)
PathFillWinding , ///< f, Fill, Winding
PathFillWinding2 , ///< F, same as previous, see PDF Reference 1.7, Table 4.10
PathFillEvenOdd , ///< f*, Fill, Even-Odd
PathFillStrokeWinding , ///< B, Fill, Stroke, Winding
PathFillStrokeEvenOdd , ///< B*, Fill, Stroke, Even-Odd
PathCloseFillStrokeWinding , ///< b, Close, Fill, Stroke, Winding (equivalent of operators h B)
PathCloseFillStrokeEvenOdd , ///< b*, Close, Fill, Stroke, Even-Odd (equivalent of operators h B*)
PathClear , ///< n, clear path (close current) path, "no-operation", used with clipping
2019-02-14 19:45:07 +01:00
// Clipping paths: W, W*
ClipWinding , ///< W, modify current clipping path by intersecting it with current path using "Non zero winding number rule"
ClipEvenOdd , ///< W*, modify current clipping path by intersecting it with current path using "Even-odd rule"
// Text object: BT, ET
TextBegin , ///< BT, begin text object, initialize text matrices, cannot be nested
TextEnd , ///< ET, end text object, cannot be nested
// Text state: Tc, Tw, Tz, TL, Tf, Tr, Ts
TextSetCharacterSpacing , ///< Tc, set text character spacing
TextSetWordSpacing , ///< Tw, set text word spacing
TextSetHorizontalScale , ///< Tz, set text horizontal scaling (in percents, 100% = normal scaling)
TextSetLeading , ///< TL, set text leading
TextSetFontAndFontSize , ///< Tf, set text font (name from dictionary) and its size
TextSetRenderMode , ///< Tr, set text render mode
TextSetRise , ///< Ts, set text rise
// Text positioning: Td, TD, Tm, T*
TextMoveByOffset , ///< Td, move by offset
2019-03-25 18:44:45 +01:00
TextSetLeadingAndMoveByOffset , ///< TD, sets text leading and moves by offset, x y TD is equivalent to sequence -y TL x y Td
2019-02-14 19:45:07 +01:00
TextSetMatrix , ///< Tm, set text matrix
TextMoveByLeading , ///< T*, moves text by leading, equivalent to 0 leading Td
// Text showing: Tj, TJ, ', "
TextShowTextString , ///< Tj, show text string
TextShowTextIndividualSpacing , ///< TJ, show text, allow individual text spacing
TextNextLineShowText , ///< ', move to the next line and show text ("string '" is equivalent to "T* string Tj")
TextSetSpacingAndShowText , ///< ", move to the next line, set spacing and show text (equivalent to sequence "w1 Tw w2 Tc string '")
// Type 3 font: d0, d1
Type3FontSetOffset , ///< d0, set width information, see PDF 1.7 Reference, Table 5.10
Type3FontSetOffsetAndBB , ///< d1, set offset and glyph bounding box
// Color: CS, cs, SC, SCN, sc, scn, G, g, RG, rg, K, k
ColorSetStrokingColorSpace , ///< CS, set current color space for stroking operations
ColorSetFillingColorSpace , ///< cs, set current color space for filling operations
ColorSetStrokingColor , ///< SC, set current stroking color
2019-02-17 18:01:22 +01:00
ColorSetStrokingColorN , ///< SCN, same as SC, but also supports Pattern, Separation, DeviceN and ICCBased color spaces
2019-02-14 19:45:07 +01:00
ColorSetFillingColor , ///< sc, set current filling color
2019-02-17 18:01:22 +01:00
ColorSetFillingColorN , ///< scn, same as sc, but also supports Pattern, Separation, DeviceN and ICCBased color spaces
2019-02-14 19:45:07 +01:00
ColorSetDeviceGrayStroking , ///< G, set DeviceGray color space for stroking color and set color
ColorSetDeviceGrayFilling , ///< g, set DeviceGray color space for filling color and set color
ColorSetDeviceRGBStroking , ///< RG, set DeviceRGB color space for stroking color and set color
ColorSetDeviceRGBFilling , ///< rg, set DeviceRGB color space for filling color and set color
ColorSetDeviceCMYKStroking , ///< K, set DeviceCMYK color space for stroking color and set color
ColorSetDeviceCMYKFilling , ///< k, set DeviceCMYK color space for filling color and set color
// Shading pattern: sh
ShadingPaintShape , ///< sh, paint shape
// Inline images: BI, ID, EI
InlineImageBegin , ///< BI, begin inline image
InlineImageData , ///< ID, inline image data
InlineImageEnd , ///< EI, end of inline image
// XObject: Do
PaintXObject , ///< Do, paint the X Object (image, form, ...)
// Marked content: MP, DP, BMC, BDC, EMC
MarkedContentPoint , ///< MP, marked content point
MarkedContentPointWithProperties , ///< DP, marked content point with properties
MarkedContentBegin , ///< BMC, begin of sequence of marked content
MarkedContentBeginWithProperties , ///< BDC, begin of sequence of marked content with properties
MarkedContentEnd , ///< EMC, end of marked content sequence
// Compatibility: BX, EX
CompatibilityBegin , ///< BX, Compatibility mode begin (unrecognized operators are ignored)
CompatibilityEnd , ///< EX, Compatibility mode end
Invalid ///< Invalid operator, use for error reporting
} ;
2020-08-02 15:29:10 +02:00
enum ProcedureSet
{
EmptyProcSet = 0x0000 ,
NoProcSet = 0x0001 ,
PDF = 0x0002 ,
Text = 0x0004 ,
ImageB = 0x0008 ,
ImageC = 0x0010 ,
ImageI = 0x0020
} ;
Q_DECLARE_FLAGS ( ProcedureSets , ProcedureSet )
2019-02-14 19:45:07 +01:00
/// Process the contents of the page
QList < PDFRenderError > processContents ( ) ;
2019-04-30 14:39:48 +02:00
virtual void reportRenderError ( RenderErrorType type , QString message ) override ;
2019-10-05 13:08:32 +02:00
/// Reports render error, but only once - if same error was already reported,
/// then no new error is reported.
2019-12-25 17:56:17 +01:00
virtual void reportRenderErrorOnce ( RenderErrorType type , QString message ) override ;
2019-10-05 13:08:32 +02:00
2020-03-07 17:38:50 +01:00
/// Processes form (XObject of type form)
/// \param Matrix Transformation matrix from form coordinate system to page coordinate system
/// \param boundingBox Bounding box, to which is drawed content clipped
/// \param resources Resources, assigned to the form
/// \param transparencyGroup Transparency group object
/// \param content Content stream of the form
2020-10-17 16:56:39 +02:00
/// \param formStructuralParent Structural parent key for form
void processForm ( const QMatrix & matrix ,
const QRectF & boundingBox ,
const PDFObject & resources ,
const PDFObject & transparencyGroup ,
const QByteArray & content ,
PDFInteger formStructuralParent ) ;
2020-03-07 17:38:50 +01:00
/// Initialize stream processor for processing content streams. For example,
/// graphic state is initialized to default, and default color spaces are initialized.
void initializeProcessor ( ) ;
/// Computes visibility of OCG/OCMD - returns false, if it is not suppressed,
/// or true, if it is suppressed.
virtual bool isContentSuppressedByOC ( PDFObjectReference ocgOrOcmd ) ;
2019-02-16 18:26:16 +01:00
protected :
2019-02-21 19:35:07 +01:00
2019-09-29 18:09:09 +02:00
struct PDFTransparencyGroup
{
PDFColorSpacePointer colorSpacePointer ;
bool isolated = false ;
bool knockout = false ;
} ;
2021-03-24 19:55:31 +01:00
/// Parses transparency group
PDFTransparencyGroup parseTransparencyGroup ( const PDFObject & object ) ;
/// Soft mask definition
class PDFSoftMaskDefinition
{
public :
enum class Type
{
Invalid ,
Alpha ,
Luminosity
} ;
Type getType ( ) const { return m_type ; }
const PDFStream * getFormStream ( ) const { return m_formStream ; }
const PDFDictionary * getFormDictionary ( ) const { return m_formStream ? m_formStream - > getDictionary ( ) : nullptr ; }
const PDFTransparencyGroup & getTransparencyGroup ( ) const { return m_transparencyGroup ; }
const PDFColor & getBackdropColor ( ) const { return m_backdropColor ; }
const PDFFunction * getTransferFunction ( ) const { return m_transferFunction . get ( ) ; }
static PDFSoftMaskDefinition parse ( const PDFDictionary * softMask , PDFPageContentProcessor * processor ) ;
private :
Type m_type = Type : : Invalid ;
const PDFStream * m_formStream = nullptr ;
PDFTransparencyGroup m_transparencyGroup ;
PDFColor m_backdropColor ;
PDFFunctionPtr m_transferFunction ;
friend class PDFPageContentProcessor ;
} ;
2019-10-04 17:21:26 +02:00
struct PDFOverprintMode
{
bool overprintStroking = false ;
bool overprintFilling = false ;
int overprintMode = 0 ;
inline bool operator = = ( const PDFOverprintMode & other ) const
{
return std : : tie ( overprintStroking , overprintFilling , overprintMode ) = = std : : tie ( other . overprintStroking , other . overprintFilling , other . overprintMode ) ;
}
inline bool operator ! = ( const PDFOverprintMode & other ) const
{
return ! ( * this = = other ) ;
}
} ;
2019-02-17 18:01:22 +01:00
/// Represents graphic state of the PDF (holding current graphic state parameters).
/// Please see PDF Reference 1.7, Chapter 4.3 "Graphic State"
class PDFPageContentProcessorState
{
public :
explicit PDFPageContentProcessorState ( ) ;
~ PDFPageContentProcessorState ( ) ;
PDFPageContentProcessorState ( const PDFPageContentProcessorState & ) = default ;
PDFPageContentProcessorState ( PDFPageContentProcessorState & & ) = default ;
PDFPageContentProcessorState & operator = ( PDFPageContentProcessorState & & ) = delete ;
PDFPageContentProcessorState & operator = ( const PDFPageContentProcessorState & other ) ;
2020-08-09 17:10:42 +02:00
enum StateFlag : uint64_t
2019-02-17 18:01:22 +01:00
{
2020-08-09 17:10:42 +02:00
StateUnchanged = 0x0000000000000000 ,
StateCurrentTransformationMatrix = 0x0000000000000001 ,
StateStrokeColorSpace = 0x0000000000000002 ,
StateFillColorSpace = 0x0000000000000004 ,
StateStrokeColor = 0x0000000000000008 ,
StateFillColor = 0x0000000000000010 ,
StateLineWidth = 0x0000000000000020 ,
StateLineCapStyle = 0x0000000000000040 ,
StateLineJoinStyle = 0x0000000000000080 ,
StateMitterLimit = 0x0000000000000100 ,
StateLineDashPattern = 0x0000000000000200 ,
StateRenderingIntentName = 0x0000000000000400 ,
StateFlatness = 0x0000000000000800 ,
StateSmoothness = 0x0000000000001000 ,
StateTextMatrix = 0x0000000000002000 ,
StateTextLineMatrix = 0x0000000000004000 ,
StateTextCharacterSpacing = 0x0000000000008000 ,
StateTextWordSpacing = 0x0000000000010000 ,
StateTextHorizontalScaling = 0x0000000000020000 ,
StateTextLeading = 0x0000000000040000 ,
StateTextFont = 0x0000000000080000 ,
StateTextFontSize = 0x0000000000100000 ,
StateTextRenderingMode = 0x0000000000200000 ,
StateTextRise = 0x0000000000400000 ,
StateTextKnockout = 0x0000000000800000 ,
StateAlphaStroking = 0x0000000001000000 ,
StateAlphaFilling = 0x0000000002000000 ,
StateBlendMode = 0x0000000004000000 ,
StateRenderingIntent = 0x0000000008000000 ,
StateOverprint = 0x0000000010000000 ,
StateAlphaIsShape = 0x0000000020000000 ,
StateStrokeAdjustment = 0x0000000040000000 ,
StateSoftMask = 0x0000000080000000 ,
StateBlackPointCompensation = 0x0000000100000000 ,
StateBlackGenerationFunction = 0x0000000200000000 ,
StateUndercolorRemovalFunction = 0x0000000400000000 ,
StateTransferFunction = 0x0000000800000000 ,
StateHalftone = 0x0000001000000000 ,
StateHalftoneOrigin = 0x0000002000000000 ,
StateAll = 0xFFFFFFFFFFFFFFFF
2019-02-17 18:01:22 +01:00
} ;
2020-08-09 17:10:42 +02:00
using StateFlags = PDFFlags < StateFlag > ;
2019-02-17 18:01:22 +01:00
const QMatrix & getCurrentTransformationMatrix ( ) const { return m_currentTransformationMatrix ; }
void setCurrentTransformationMatrix ( const QMatrix & currentTransformationMatrix ) ;
const PDFAbstractColorSpace * getStrokeColorSpace ( ) const { return m_strokeColorSpace . data ( ) ; }
void setStrokeColorSpace ( const QSharedPointer < PDFAbstractColorSpace > & strokeColorSpace ) ;
const PDFAbstractColorSpace * getFillColorSpace ( ) const { return m_fillColorSpace . data ( ) ; }
void setFillColorSpace ( const QSharedPointer < PDFAbstractColorSpace > & fillColorSpace ) ;
const QColor & getStrokeColor ( ) const { return m_strokeColor ; }
2021-02-02 19:23:51 +01:00
const PDFColor & getStrokeColorOriginal ( ) const { return m_strokeColorOriginal ; }
void setStrokeColor ( const QColor & strokeColor , const PDFColor & originalColor ) ;
2019-02-17 18:01:22 +01:00
const QColor & getFillColor ( ) const { return m_fillColor ; }
2021-02-02 19:23:51 +01:00
const PDFColor & getFillColorOriginal ( ) const { return m_fillColorOriginal ; }
void setFillColor ( const QColor & fillColor , const PDFColor & originalColor ) ;
2019-02-17 18:01:22 +01:00
PDFReal getLineWidth ( ) const { return m_lineWidth ; }
void setLineWidth ( PDFReal lineWidth ) ;
Qt : : PenCapStyle getLineCapStyle ( ) const { return m_lineCapStyle ; }
void setLineCapStyle ( Qt : : PenCapStyle lineCapStyle ) ;
Qt : : PenJoinStyle getLineJoinStyle ( ) const { return m_lineJoinStyle ; }
void setLineJoinStyle ( Qt : : PenJoinStyle lineJoinStyle ) ;
PDFReal getMitterLimit ( ) const { return m_mitterLimit ; }
2019-03-25 18:44:45 +01:00
void setMitterLimit ( PDFReal mitterLimit ) ;
2019-02-17 18:01:22 +01:00
2019-02-21 19:35:07 +01:00
const PDFLineDashPattern & getLineDashPattern ( ) const { return m_lineDashPattern ; }
void setLineDashPattern ( PDFLineDashPattern pattern ) ;
2019-10-04 17:21:26 +02:00
const QByteArray & getRenderingIntentName ( ) const { return m_renderingIntentName ; }
void setRenderingIntentName ( const QByteArray & renderingIntentName ) ;
2019-02-17 18:01:22 +01:00
PDFReal getFlatness ( ) const { return m_flatness ; }
void setFlatness ( PDFReal flatness ) ;
PDFReal getSmoothness ( ) const { return m_smoothness ; }
void setSmoothness ( PDFReal smoothness ) ;
StateFlags getStateFlags ( ) const { return m_stateFlags ; }
void setStateFlags ( StateFlags stateFlags ) { m_stateFlags = stateFlags ; }
2019-03-25 18:44:45 +01:00
PDFReal getTextCharacterSpacing ( ) const { return m_textCharacterSpacing ; }
void setTextCharacterSpacing ( PDFReal textCharacterSpacing ) ;
PDFReal getTextWordSpacing ( ) const { return m_textWordSpacing ; }
void setTextWordSpacing ( PDFReal textWordSpacing ) ;
PDFReal getTextHorizontalScaling ( ) const { return m_textHorizontalScaling ; }
void setTextHorizontalScaling ( PDFReal textHorizontalScaling ) ;
PDFReal getTextLeading ( ) const { return m_textLeading ; }
void setTextLeading ( PDFReal textLeading ) ;
const PDFFontPointer & getTextFont ( ) const { return m_textFont ; }
void setTextFont ( const PDFFontPointer & textFont ) ;
PDFReal getTextFontSize ( ) const { return m_textFontSize ; }
void setTextFontSize ( PDFReal textFontSize ) ;
TextRenderingMode getTextRenderingMode ( ) const { return m_textRenderingMode ; }
void setTextRenderingMode ( TextRenderingMode textRenderingMode ) ;
PDFReal getTextRise ( ) const { return m_textRise ; }
void setTextRise ( PDFReal textRise ) ;
bool getTextKnockout ( ) const { return m_textKnockout ; }
void setTextKnockout ( bool textKnockout ) ;
const QMatrix & getTextMatrix ( ) const { return m_textMatrix ; }
void setTextMatrix ( const QMatrix & textMatrix ) ;
const QMatrix & getTextLineMatrix ( ) const { return m_textLineMatrix ; }
void setTextLineMatrix ( const QMatrix & textLineMatrix ) ;
2019-09-29 15:44:35 +02:00
PDFReal getAlphaStroking ( ) const { return m_alphaStroking ; }
void setAlphaStroking ( PDFReal alpha ) ;
PDFReal getAlphaFilling ( ) const { return m_alphaFilling ; }
void setAlphaFilling ( PDFReal alpha ) ;
BlendMode getBlendMode ( ) const { return m_blendMode ; }
void setBlendMode ( BlendMode mode ) ;
2019-10-04 17:21:26 +02:00
RenderingIntent getRenderingIntent ( ) const { return m_renderingIntent ; }
void setRenderingIntent ( RenderingIntent renderingIntent ) ;
2019-09-29 15:44:35 +02:00
/// Returns stroke color with alpha channel
QColor getStrokeColorWithAlpha ( ) const ;
/// Returns fill color with alpha channel
QColor getFillColorWithAlpha ( ) const ;
2019-10-04 17:21:26 +02:00
PDFOverprintMode getOverprintMode ( ) const { return m_overprintMode ; }
void setOverprintMode ( PDFOverprintMode overprintMode ) ;
bool getAlphaIsShape ( ) const { return m_alphaIsShape ; }
void setAlphaIsShape ( bool alphaIsShape ) ;
2020-08-09 14:08:31 +02:00
bool getStrokeAdjustment ( ) const ;
void setStrokeAdjustment ( bool strokeAdjustment ) ;
const PDFDictionary * getSoftMask ( ) const ;
void setSoftMask ( const PDFDictionary * softMask ) ;
2020-08-09 17:10:42 +02:00
BlackPointCompensationMode getBlackPointCompensationMode ( ) const ;
void setBlackPointCompensationMode ( BlackPointCompensationMode blackPointCompensationMode ) ;
PDFObject getBlackGenerationFunction ( ) const ;
void setBlackGenerationFunction ( const PDFObject & blackGenerationFunction ) ;
PDFObject getUndercolorRemovalFunction ( ) const ;
void setUndercolorRemovalFunction ( const PDFObject & undercolorRemovalFunction ) ;
PDFObject getTransferFunction ( ) const ;
void setTransferFunction ( const PDFObject & transferFunction ) ;
PDFObject getHalftone ( ) const ;
void setHalftone ( const PDFObject & halftone ) ;
QPointF getHalftoneOrigin ( ) const ;
void setHalftoneOrigin ( const QPointF & halftoneOrigin ) ;
2019-02-17 18:01:22 +01:00
private :
QMatrix m_currentTransformationMatrix ;
PDFColorSpacePointer m_strokeColorSpace ;
PDFColorSpacePointer m_fillColorSpace ;
QColor m_strokeColor ;
2021-02-02 19:23:51 +01:00
PDFColor m_strokeColorOriginal ;
2019-02-17 18:01:22 +01:00
QColor m_fillColor ;
2021-02-02 19:23:51 +01:00
PDFColor m_fillColorOriginal ;
2019-02-17 18:01:22 +01:00
PDFReal m_lineWidth ;
Qt : : PenCapStyle m_lineCapStyle ;
Qt : : PenJoinStyle m_lineJoinStyle ;
PDFReal m_mitterLimit ;
2019-02-21 19:35:07 +01:00
PDFLineDashPattern m_lineDashPattern ;
2019-10-04 17:21:26 +02:00
QByteArray m_renderingIntentName ;
2019-02-17 18:01:22 +01:00
PDFReal m_flatness ;
PDFReal m_smoothness ;
2019-03-25 18:44:45 +01:00
PDFReal m_textCharacterSpacing ; // T_c
PDFReal m_textWordSpacing ; // T_w
PDFReal m_textHorizontalScaling ; // T_h, percentage
PDFReal m_textLeading ; // T_l
PDFFontPointer m_textFont ; // Text font
PDFReal m_textFontSize ; // T_fs
TextRenderingMode m_textRenderingMode ; // Text rendering mode
PDFReal m_textRise ; // T_rise
bool m_textKnockout ;
QMatrix m_textMatrix ;
QMatrix m_textLineMatrix ;
2019-09-29 15:44:35 +02:00
PDFReal m_alphaStroking ;
PDFReal m_alphaFilling ;
BlendMode m_blendMode ;
2019-10-04 17:21:26 +02:00
RenderingIntent m_renderingIntent ;
PDFOverprintMode m_overprintMode ;
bool m_alphaIsShape ;
2020-08-09 14:08:31 +02:00
bool m_strokeAdjustment ;
const PDFDictionary * m_softMask ;
2020-08-09 17:10:42 +02:00
BlackPointCompensationMode m_blackPointCompensationMode ;
PDFObject m_blackGenerationFunction ;
PDFObject m_undercolorRemovalFunction ;
PDFObject m_transferFunction ;
PDFObject m_halftone ;
QPointF m_halftoneOrigin ;
2019-02-17 18:01:22 +01:00
StateFlags m_stateFlags ;
} ;
2019-02-21 19:35:07 +01:00
enum class ProcessOrder
{
BeforeOperation ,
AfterOperation
} ;
2019-02-16 18:26:16 +01:00
/// This function has to be implemented in the client drawing implementation, it should
/// draw the path according to the parameters.
/// \param path Path, which should be drawn (can be emtpy - in that case nothing happens)
/// \param stroke Stroke the path
/// \param fill Fill the path using given rule
2019-05-04 18:22:40 +02:00
/// \param text Is text being drawn?
2019-02-16 18:26:16 +01:00
/// \param fillRule Fill rule used in the fill mode
2019-05-04 18:22:40 +02:00
virtual void performPathPainting ( const QPainterPath & path , bool stroke , bool fill , bool text , Qt : : FillRule fillRule ) ;
2019-02-16 18:26:16 +01:00
2021-03-01 15:24:04 +01:00
/// This function is used, when we want to implement custom fill using shading. If path is successfully
/// filled by shading, then true should be returned.
/// \param path Path to be filled
/// \param stroke Is path actually stroked?
/// \param fill Is path actually filled?
/// \param shadingPattern Shading pattern
virtual bool performPathPaintingUsingShading ( const QPainterPath & path , bool stroke , bool fill , const PDFShadingPattern * shadingPattern ) ;
/// This function is called after path paintig is finished
virtual void performFinishPathPainting ( ) ;
2019-02-17 18:01:22 +01:00
/// This function has to be implemented in the client drawing implementation, it should
/// clip along the path (intersect with current clipping path).
virtual void performClipping ( const QPainterPath & path , Qt : : FillRule fillRule ) ;
2021-02-21 16:42:24 +01:00
/// Performs image processing on original image. If processor processes
/// original image, it should return true, so no conversion to QImage occurs,
/// which can be performance bottleneck.
/// \param image Image
/// \returns true, if image is successfully processed
virtual bool performOriginalImagePainting ( const PDFImage & image ) ;
2019-05-07 18:21:22 +02:00
/// This function has to be implemented in the client drawing implementation, it should
/// draw the image.
/// \param image Image to be painted
virtual void performImagePainting ( const QImage & image ) ;
2019-08-31 15:55:59 +02:00
/// This function has to be implemented in the client drawing implementation, it should
/// draw the mesh. Mesh is in device space coordinates (so world transformation matrix
/// is identity matrix).
/// \param mesh Mesh to be drawn
virtual void performMeshPainting ( const PDFMesh & mesh ) ;
2019-02-17 18:01:22 +01:00
/// This function has to be implemented in the client drawing implementation, it should
2019-02-21 19:35:07 +01:00
/// update the device according to the graphic state change. The flags are set when
2019-02-17 18:01:22 +01:00
/// the value differs from the previous graphic state.
virtual void performUpdateGraphicsState ( const PDFPageContentProcessorState & state ) ;
2019-02-21 19:35:07 +01:00
/// Implement to perform save of the graphic state. This function is called two times -
/// before the operation and after the operation. Parameter \p order determines when
/// this function is called.
/// \param order If this function is called before the operation, or after the operation.
virtual void performSaveGraphicState ( ProcessOrder order ) ;
/// Implement to perform restore of the graphic state. This function is called two times -
/// before the operation and after the operation. Parameter \p order determines when
/// this function is called.
/// \param order If this function is called before the operation, or after the operation.
virtual void performRestoreGraphicState ( ProcessOrder order ) ;
2019-07-04 17:52:38 +02:00
/// Implement to react on marked content point. Properties object can be null, in case
/// that no properties are provided.
/// \param tag Tag of the marked content point
/// \param properties Properties of the marked content point.
virtual void performMarkedContentPoint ( const QByteArray & tag , const PDFObject & properties ) ;
/// Implement to react on marked content begin.
/// \param tag Tag of the marked content point
/// \param properties Properties of the marked content point.
virtual void performMarkedContentBegin ( const QByteArray & tag , const PDFObject & properties ) ;
/// Implement to react on marked content end
virtual void performMarkedContentEnd ( ) ;
2019-07-14 19:03:15 +02:00
/// Implement to react on set char width request
2019-07-21 17:31:39 +02:00
virtual void performSetCharWidth ( PDFReal wx , PDFReal wy ) ;
2019-07-14 19:03:15 +02:00
/// Implement to react on set cache device request
2019-07-21 17:31:39 +02:00
virtual void performSetCacheDevice ( PDFReal wx , PDFReal wy , PDFReal llx , PDFReal lly , PDFReal urx , PDFReal ury ) ;
2019-07-14 19:03:15 +02:00
2019-09-29 18:09:09 +02:00
/// Implement to begin new transparency group
/// \param Order, in which is function called (before/after setting new transparency group)
/// \param transparencyGroup Transparency group
virtual void performBeginTransparencyGroup ( ProcessOrder order , const PDFTransparencyGroup & transparencyGroup ) ;
/// Implement to end current transparency group
/// \param Order, in which is function called (before/after setting new transparency group)
/// \param transparencyGroup Transparency group
virtual void performEndTransparencyGroup ( ProcessOrder order , const PDFTransparencyGroup & transparencyGroup ) ;
2019-12-28 19:21:29 +01:00
/// Implement to react on character printing
virtual void performOutputCharacter ( const PDFTextCharacterInfo & info ) ;
2021-02-07 18:14:36 +01:00
/// Implement to respond to text begin operator
virtual void performTextBegin ( ProcessOrder order ) ;
/// Implement to respond to text end operator
virtual void performTextEnd ( ProcessOrder order ) ;
2019-12-31 17:39:31 +01:00
enum class ContentKind
{
Shapes , ///< General shapes (they can be also shaded / tiled)
Text , ///< Text outlines (they can be also shaded / tiled)
Images , ///< Images
Shading , ///< Shading
Tiling , ///< Tiling
} ;
/// Override this function to disable particular content type (for example
/// shading, images, ...)
virtual bool isContentKindSuppressed ( ContentKind kind ) const ;
2021-03-24 19:55:31 +01:00
/// Sets current graphic state and updates data
/// \param state New graphic state
void setGraphicsState ( const PDFPageContentProcessorState & state ) ;
2020-10-17 16:56:39 +02:00
/// Returns current structural parent key
PDFInteger getStructuralParentKey ( ) const { return m_structuralParentKey ; }
2019-02-24 17:48:37 +01:00
/// Returns current graphic state
const PDFPageContentProcessorState * getGraphicState ( ) const { return & m_graphicState ; }
/// Adds error to the error list
/// \param error Error message
void addError ( const QString & error ) { m_errorList . append ( PDFRenderError ( RenderErrorType : : Error , error ) ) ; }
2019-07-04 17:52:38 +02:00
/// Returns true, if graphic content is suppressed
bool isContentSuppressed ( ) const ;
2019-08-31 14:37:18 +02:00
/// Returns page point to device point matrix
const QMatrix & getPagePointToDevicePointMatrix ( ) const { return m_pagePointToDevicePointMatrix ; }
/// Returns base matrix for patterns
const QMatrix & getPatternBaseMatrix ( ) const { return m_patternBaseMatrix ; }
/// Returns current world matrix (translating actual point to the device point)
QMatrix getCurrentWorldMatrix ( ) const { return getGraphicState ( ) - > getCurrentTransformationMatrix ( ) * m_pagePointToDevicePointMatrix ; }
/// Returns page bounding rectangle in device space
const QRectF & getPageBoundingRectDeviceSpace ( ) const { return m_pageBoundingRectDeviceSpace ; }
2020-08-02 15:29:10 +02:00
/// Returns current procedure sets. Procedure sets are deprecated in PDF 2.0 and are here
/// only for compatibility purposes. See chapter 14.2 in PDF 2.0 specification.
ProcedureSets getProcedureSets ( ) const { return m_procedureSets ; }
2021-01-14 19:33:23 +01:00
/// Returns page
const PDFPage * getPage ( ) const { return m_page ; }
/// Returns document
const PDFDocument * getDocument ( ) const { return m_document ; }
2021-01-19 20:14:50 +01:00
/// Returns color management system
const PDFCMS * getCMS ( ) const { return m_CMS ; }
2021-03-24 19:55:31 +01:00
/// Returns font cache
const PDFFontCache * getFontCache ( ) const { return m_fontCache ; }
/// Returns optional content activity
const PDFOptionalContentActivity * getOptionalContentActivity ( ) const { return m_optionalContentActivity ; }
2021-01-14 19:33:23 +01:00
class PDFTransparencyGroupGuard
{
public :
explicit PDFTransparencyGroupGuard ( PDFPageContentProcessor * processor , PDFTransparencyGroup & & group ) ;
~ PDFTransparencyGroupGuard ( ) ;
private :
PDFPageContentProcessor * m_processor ;
} ;
2021-03-24 19:55:31 +01:00
/// Process form using form stream
void processForm ( const PDFStream * stream ) ;
2019-02-14 19:45:07 +01:00
private :
2019-06-16 16:32:23 +02:00
/// Initializes the resources dictionaries
void initDictionaries ( const PDFObject & resourcesObject ) ;
2019-02-14 19:45:07 +01:00
/// Process the content stream
void processContentStream ( const PDFStream * stream ) ;
2019-06-16 16:32:23 +02:00
/// Process the content
void processContent ( const QByteArray & content ) ;
2019-02-14 19:45:07 +01:00
/// Processes single command
void processCommand ( const QByteArray & command ) ;
2019-08-31 15:55:59 +02:00
/// Performs path painting
/// \param path Path, which should be drawn (can be emtpy - in that case nothing happens)
/// \param stroke Stroke the path
/// \param fill Fill the path using given rule
/// \param text Is text being drawn?
/// \param fillRule Fill rule used in the fill mode
void processPathPainting ( const QPainterPath & path , bool stroke , bool fill , bool text , Qt : : FillRule fillRule ) ;
2019-09-25 19:17:52 +02:00
/// Performs tiling pattern painting
/// \param tilingPattern Tiling pattern to be painted
/// \param path Clipping path
2019-09-26 19:14:04 +02:00
/// \param uncoloredPatternColorSpace Color space for uncolored color patterns
/// \param uncoloredPatternColor Uncolored color pattern color
void processTillingPatternPainting ( const PDFTilingPattern * tilingPattern ,
const QPainterPath & path ,
PDFColorSpacePointer uncoloredPatternColorSpace ,
PDFColor uncoloredPatternColor ) ;
2019-09-25 19:17:52 +02:00
2019-09-27 18:41:56 +02:00
/// Applies graphic state dictionary
/// \param graphicStateDictionary Dictionary to be applied to the current graphic state
void processApplyGraphicState ( const PDFDictionary * graphicStateDictionary ) ;
2019-07-04 17:52:38 +02:00
enum class MarkedContentKind
{
OptionalContent ,
Other
} ;
struct MarkedContentState
{
inline explicit MarkedContentState ( ) = default ;
inline explicit MarkedContentState ( const QByteArray & tag , MarkedContentKind kind , bool contentSuppressed ) :
tag ( tag ) ,
kind ( kind ) ,
contentSuppressed ( contentSuppressed )
{
}
QByteArray tag ;
MarkedContentKind kind = MarkedContentKind : : Other ;
bool contentSuppressed = false ;
} ;
2019-09-29 18:09:09 +02:00
class PDFPageContentProcessorStateGuard
2019-06-16 16:32:23 +02:00
{
public :
explicit PDFPageContentProcessorStateGuard ( PDFPageContentProcessor * processor ) ;
~ PDFPageContentProcessorStateGuard ( ) ;
private :
PDFPageContentProcessor * m_processor ;
// Stored resources
const PDFDictionary * m_colorSpaceDictionary ;
const PDFDictionary * m_fontDictionary ;
const PDFDictionary * m_xobjectDictionary ;
const PDFDictionary * m_extendedGraphicStateDictionary ;
2019-07-04 17:52:38 +02:00
const PDFDictionary * m_propertiesDictionary ;
2019-08-25 18:16:37 +02:00
const PDFDictionary * m_shadingDictionary ;
2019-09-13 16:28:20 +02:00
const PDFDictionary * m_patternDictionary ;
2020-08-02 15:29:10 +02:00
ProcedureSets m_procedureSets ;
2019-06-16 16:32:23 +02:00
} ;
2019-09-29 18:09:09 +02:00
class PDFPageContentProcessorGraphicStateSaveRestoreGuard
2019-09-25 19:17:52 +02:00
{
public :
inline explicit PDFPageContentProcessorGraphicStateSaveRestoreGuard ( PDFPageContentProcessor * processor ) :
m_processor ( processor )
{
m_processor - > operatorSaveGraphicState ( ) ;
}
inline ~ PDFPageContentProcessorGraphicStateSaveRestoreGuard ( )
{
m_processor - > operatorRestoreGraphicState ( ) ;
}
private :
PDFPageContentProcessor * m_processor ;
} ;
2019-02-17 18:01:22 +01:00
/// Wrapper for PDF Name
2019-03-30 18:45:30 +01:00
struct PDFOperandName
2019-02-17 18:01:22 +01:00
{
QByteArray name ;
} ;
2019-03-30 18:45:30 +01:00
/// Wrapper for PDF String
struct PDFOperandString
{
QByteArray string ;
} ;
2019-02-14 19:45:07 +01:00
template < typename T >
T readOperand ( size_t index ) const ;
template < >
PDFReal readOperand < PDFReal > ( size_t index ) const ;
2019-02-21 19:35:07 +01:00
template < >
PDFInteger readOperand < PDFInteger > ( size_t index ) const ;
2019-02-17 18:01:22 +01:00
template < >
2019-03-30 18:45:30 +01:00
PDFOperandName readOperand < PDFOperandName > ( size_t index ) const ;
template < >
PDFOperandString readOperand < PDFOperandString > ( size_t index ) const ;
2019-02-17 18:01:22 +01:00
2019-02-14 19:45:07 +01:00
template < size_t index , typename T >
inline T readOperand ( ) const { return readOperand < T > ( index ) ; }
template < typename Tuple , class F , std : : size_t . . . I >
inline void invokeOperatorImpl ( F function , std : : index_sequence < I . . . > )
{
( this - > * function ) ( readOperand < I , typename std : : tuple_element < I , Tuple > : : type > ( ) . . . ) ;
}
/// Function invokes operator (function) with given arguments. For that reason, variadic
/// templates are used. Basically, for each argument of the function, we need type of the argument,
/// and its index. To retrieve it, we use std::tuple, variadic template and functionality
/// analogic to std::apply implementation.
template < typename . . . Operands >
inline void invokeOperator ( void ( PDFPageContentProcessor : : * function ) ( Operands . . . ) )
{
invokeOperatorImpl < std : : tuple < Operands . . . > > ( function , std : : make_index_sequence < std : : tuple_size_v < std : : remove_reference_t < std : : tuple < Operands . . . > > > > { } ) ;
}
/// Returns the current poin in the path. If path doesn't exist, then
/// exception is thrown.
QPointF getCurrentPoint ( ) const ;
2019-02-17 18:01:22 +01:00
/// Notifies the updated graphic state. If nothing changed in graphic state, then nothing happens.
void updateGraphicState ( ) ;
template < typename . . . Operands >
inline QColor getColorFromColorSpace ( const PDFAbstractColorSpace * colorSpace , Operands . . . operands )
{
constexpr const size_t operandCount = sizeof . . . ( Operands ) ;
const size_t colorSpaceComponentCount = colorSpace - > getColorComponentCount ( ) ;
if ( operandCount = = colorSpaceComponentCount )
{
2020-09-19 17:07:40 +02:00
return colorSpace - > getColor ( PDFColor ( static_cast < PDFColorComponent > ( operands ) . . . ) , m_CMS , m_graphicState . getRenderingIntent ( ) , this , true ) ;
2019-02-17 18:01:22 +01:00
}
else
{
throw PDFRendererException ( RenderErrorType : : Error , PDFTranslationContext : : tr ( " Invalid color component count. Provided %1, required %2. " ) . arg ( operandCount ) . arg ( colorSpaceComponentCount ) ) ;
}
return QColor ( ) ;
}
2019-02-23 15:44:14 +01:00
/// Converts PDF line cap to Qt's pen cap style. Function always succeeds,
/// if invalid \p lineCap occurs, then some valid pen cap style is returned.
/// \param lineCap PDF Line cap style (see PDF Reference 1.7, values can be 0, 1, and 2)
static Qt : : PenCapStyle convertLineCapToPenCapStyle ( PDFInteger lineCap ) ;
/// Convers Qt's pen cap style to PDF's line cap style (defined in the PDF Reference)
/// \param penCapStyle Qt's pen cap style to be converted
static PDFInteger convertPenCapStyleToLineCap ( Qt : : PenCapStyle penCapStyle ) ;
/// Converts PDF line join to Qt's pen join style. Function always succeeds,
/// if invalid \p lineJoin occurs, then some valid pen join style is returned.
/// \param lineJoin PDF Line join style (see PDF Reference 1.7, values can be 0, 1, and 2)
static Qt : : PenJoinStyle convertLineJoinToPenJoinStyle ( PDFInteger lineJoin ) ;
/// Convers Qt's pen join style to PDF's line join style (defined in the PDF Reference)
/// \param penJoinStyle Qt's pen join style to be converted
static PDFInteger convertPenJoinStyleToLineJoin ( Qt : : PenJoinStyle penJoinStyle ) ;
2019-02-21 19:35:07 +01:00
// General graphic state w, J, j, M, d, ri, i, gs
void operatorSetLineWidth ( PDFReal lineWidth ) ; ///< w, sets the line width
void operatorSetLineCap ( PDFInteger lineCap ) ; ///< J, sets the line cap
void operatorSetLineJoin ( PDFInteger lineJoin ) ; ///< j, sets the line join
void operatorSetMitterLimit ( PDFReal mitterLimit ) ; ///< M, sets the mitter limit
void operatorSetLineDashPattern ( ) ; ///< d, sets the line dash pattern
2019-03-30 18:45:30 +01:00
void operatorSetRenderingIntent ( PDFOperandName intent ) ; ///< ri, sets the rendering intent
2019-02-21 19:35:07 +01:00
void operatorSetFlatness ( PDFReal flatness ) ; ///< i, sets the flattness (number in range from 0 to 100)
2019-03-30 18:45:30 +01:00
void operatorSetGraphicState ( PDFOperandName dictionaryName ) ; ///< gs, sets the whole graphic state (stored in resource dictionary)
2019-02-21 19:35:07 +01:00
// Special graphic state: q, Q, cm
void operatorSaveGraphicState ( ) ; ///< q, saves the graphic state
void operatorRestoreGraphicState ( ) ; ///< Q, restores the graphic state
void operatorAdjustCurrentTransformationMatrix ( PDFReal a , PDFReal b , PDFReal c , PDFReal d , PDFReal e , PDFReal f ) ; ///< cm, modify the current transformation matrix by matrix multiplication
2019-02-14 19:45:07 +01:00
// Path construction operators
void operatorMoveCurrentPoint ( PDFReal x , PDFReal y ) ;
void operatorLineTo ( PDFReal x , PDFReal y ) ;
void operatorBezier123To ( PDFReal x1 , PDFReal y1 , PDFReal x2 , PDFReal y2 , PDFReal x3 , PDFReal y3 ) ;
void operatorBezier23To ( PDFReal x2 , PDFReal y2 , PDFReal x3 , PDFReal y3 ) ;
void operatorBezier13To ( PDFReal x1 , PDFReal y1 , PDFReal x3 , PDFReal y3 ) ;
void operatorEndSubpath ( ) ;
void operatorRectangle ( PDFReal x , PDFReal y , PDFReal width , PDFReal height ) ;
2019-02-16 18:26:16 +01:00
// Path painting operators
void operatorPathStroke ( ) ;
void operatorPathCloseStroke ( ) ;
void operatorPathFillWinding ( ) ;
void operatorPathFillEvenOdd ( ) ;
void operatorPathFillStrokeWinding ( ) ;
void operatorPathFillStrokeEvenOdd ( ) ;
void operatorPathCloseFillStrokeWinding ( ) ;
void operatorPathCloseFillStrokeEvenOdd ( ) ;
void operatorPathClear ( ) ;
2019-02-17 18:01:22 +01:00
// Clipping paths: W, W*
void operatorClipWinding ( ) ; ///< W, modify current clipping path by intersecting it with current path using "Non zero winding number rule"
void operatorClipEvenOdd ( ) ; ///< W*, modify current clipping path by intersecting it with current path using "Even-odd rule"
2019-07-14 19:03:15 +02:00
// Type 3 font: d0, d1
void operatorType3FontSetOffset ( PDFReal wx , PDFReal wy ) ; ///< d0, set width information, see PDF 1.7 Reference, Table 5.10
void operatorType3FontSetOffsetAndBB ( PDFReal wx , PDFReal wy , PDFReal llx , PDFReal lly , PDFReal urx , PDFReal ury ) ; ///< d1, set offset and glyph bounding box
2019-02-17 18:01:22 +01:00
// Color: CS, cs, SC, SCN, sc, scn, G, g, RG, rg, K, k
2019-03-30 18:45:30 +01:00
void operatorColorSetStrokingColorSpace ( PDFOperandName name ) ; ///< CS, set current color space for stroking operations
void operatorColorSetFillingColorSpace ( PDFOperandName name ) ; ///< cs, set current color space for filling operations
2019-02-17 18:01:22 +01:00
void operatorColorSetStrokingColor ( ) ; ///< SC, set current stroking color
void operatorColorSetStrokingColorN ( ) ; ///< SCN, same as SC, but also supports Pattern, Separation, DeviceN and ICCBased color spaces
void operatorColorSetFillingColor ( ) ; ///< sc, set current filling color
void operatorColorSetFillingColorN ( ) ; ///< scn, same as sc, but also supports Pattern, Separation, DeviceN and ICCBased color spaces
void operatorColorSetDeviceGrayStroking ( PDFReal gray ) ; ///< G, set DeviceGray color space for stroking color and set color
void operatorColorSetDeviceGrayFilling ( PDFReal gray ) ; ///< g, set DeviceGray color space for filling color and set color
void operatorColorSetDeviceRGBStroking ( PDFReal r , PDFReal g , PDFReal b ) ; ///< RG, set DeviceRGB color space for stroking color and set color
void operatorColorSetDeviceRGBFilling ( PDFReal r , PDFReal g , PDFReal b ) ; ///< rg, set DeviceRGB color space for filling color and set color
void operatorColorSetDeviceCMYKStroking ( PDFReal c , PDFReal m , PDFReal y , PDFReal k ) ; ///< K, set DeviceCMYK color space for stroking color and set color
void operatorColorSetDeviceCMYKFilling ( PDFReal c , PDFReal m , PDFReal y , PDFReal k ) ; ///< k, set DeviceCMYK color space for filling color and set color
2019-02-14 19:45:07 +01:00
2019-03-25 18:44:45 +01:00
// Text object: BT, ET
void operatorTextBegin ( ) ; ///< BT, begin text object, initialize text matrices, cannot be nested
void operatorTextEnd ( ) ; ///< ET, end text object, cannot be nested
// Text state: Tc, Tw, Tz, TL, Tf, Tr, Ts
void operatorTextSetCharacterSpacing ( PDFReal charSpacing ) ; ///< Tc, set text character spacing
void operatorTextSetWordSpacing ( PDFReal wordSpacing ) ; ///< Tw, set text word spacing
void operatorTextSetHorizontalScale ( PDFReal horizontalScaling ) ; ///< Tz, set text horizontal scaling (in percents, 100% = normal scaling)
void operatorTextSetLeading ( PDFReal leading ) ; ///< TL, set text leading
2019-03-30 18:45:30 +01:00
void operatorTextSetFontAndFontSize ( PDFOperandName fontName , PDFReal fontSize ) ; ///< Tf, set text font (name from dictionary) and its size
2019-03-25 18:44:45 +01:00
void operatorTextSetRenderMode ( PDFInteger mode ) ; ///< Tr, set text render mode
void operatorTextSetRise ( PDFReal rise ) ; ///< Ts, set text rise
// Text positioning: Td, TD, Tm, T*
void operatorTextMoveByOffset ( PDFReal t_x , PDFReal t_y ) ; ///< Td, move by offset
void operatorTextSetLeadingAndMoveByOffset ( PDFReal t_x , PDFReal t_y ) ; ///< TD, sets text leading and moves by offset, x y TD is equivalent to sequence -y TL x y Td
void operatorTextSetMatrix ( PDFReal a , PDFReal b , PDFReal c , PDFReal d , PDFReal e , PDFReal f ) ; ///< Tm, set text matrix
void operatorTextMoveByLeading ( ) ; ///< T*, moves text by leading, equivalent to 0 leading Td
2019-03-30 18:45:30 +01:00
// Text showing: Tj, TJ, ', "
void operatorTextShowTextString ( PDFOperandString text ) ; ///< Tj, show text string
void operatorTextShowTextIndividualSpacing ( ) ; ///< TJ, show text, allow individual text spacing
void operatorTextNextLineShowText ( PDFOperandString text ) ; ///< ', move to the next line and show text ("string '" is equivalent to "T* string Tj")
void operatorTextSetSpacingAndShowText ( PDFReal t_w , PDFReal t_c , PDFOperandString text ) ; ///< ", move to the next line, set spacing and show text (equivalent to sequence "w1 Tw w2 Tc string '")
2019-08-25 18:16:37 +02:00
// Shading pattern: sh
void operatorShadingPaintShape ( PDFOperandName name ) ; ///< sh, paint shape
2019-05-07 18:21:22 +02:00
// XObject: Do
void operatorPaintXObject ( PDFOperandName name ) ; ///< Do, paint the X Object (image, form, ...)
2019-07-04 17:52:38 +02:00
// Marked content: MP, DP, BMC, BDC, EMC
void operatorMarkedContentPoint ( PDFOperandName name ) ; ///< MP, marked content point
void operatorMarkedContentPointWithProperties ( PDFOperandName name , PDFObject properties ) ; ///< DP, marked content point with properties
void operatorMarkedContentBegin ( PDFOperandName name ) ; ///< BMC, begin of sequence of marked content
void operatorMarkedContentBeginWithProperties ( PDFOperandName name , PDFObject properties ) ; ///< BDC, begin of sequence of marked content with properties
void operatorMarkedContentEnd ( ) ; ///< EMC, end of marked content sequence
2019-06-23 18:35:32 +02:00
// Compatibility: BX, EX
void operatorCompatibilityBegin ( ) ; ///< BX, Compatibility mode begin (unrecognized operators are ignored)
void operatorCompatibilityEnd ( ) ; ///< EX, Compatibility mode end
2019-03-30 18:45:30 +01:00
// Draws the text using the text sequence
void drawText ( const TextSequence & textSequence ) ;
/// Returns realized font
2019-04-07 19:39:29 +02:00
const PDFRealizedFontPointer & getRealizedFont ( ) { return m_realizedFont . get ( this , & PDFPageContentProcessor : : getRealizedFontImpl ) ; }
2019-03-30 18:45:30 +01:00
/// Returns realized font (or empty font, if font can't be realized)
2019-10-02 19:37:19 +02:00
PDFRealizedFontPointer getRealizedFontImpl ( ) ;
2019-03-30 18:45:30 +01:00
2019-04-29 14:14:06 +02:00
/// Checks, if stroking color is valid
void checkStrokingColor ( ) ;
/// Checks, if filling color is valid
void checkFillingColor ( ) ;
2019-07-04 17:52:38 +02:00
/// Read object from operand stack
PDFObject readObjectFromOperandStack ( size_t startPosition ) const ;
2019-07-21 17:31:39 +02:00
/// Implementation of painting of XObject image
void paintXObjectImage ( const PDFStream * stream ) ;
2019-09-27 18:41:56 +02:00
/// Report warning about color operators in uncolored tiling pattern
void reportWarningAboutColorOperatorsInUTP ( ) ;
2019-10-04 17:21:26 +02:00
/// Set rendering intent by name
void setRenderingIntentByName ( QByteArray renderingIntentName ) ;
2020-10-17 16:56:39 +02:00
/// Finishes marked content (if end of marked content is missing)
void finishMarkedContent ( ) ;
2019-02-14 19:45:07 +01:00
const PDFPage * m_page ;
const PDFDocument * m_document ;
2019-04-12 19:17:19 +02:00
const PDFFontCache * m_fontCache ;
2019-12-25 17:56:17 +01:00
const PDFCMS * m_CMS ;
2019-07-04 17:52:38 +02:00
const PDFOptionalContentActivity * m_optionalContentActivity ;
2019-02-17 18:01:22 +01:00
const PDFDictionary * m_colorSpaceDictionary ;
2019-03-30 18:45:30 +01:00
const PDFDictionary * m_fontDictionary ;
2019-05-07 18:21:22 +02:00
const PDFDictionary * m_xobjectDictionary ;
2019-06-16 16:32:23 +02:00
const PDFDictionary * m_extendedGraphicStateDictionary ;
2019-07-04 17:52:38 +02:00
const PDFDictionary * m_propertiesDictionary ;
2019-08-25 18:16:37 +02:00
const PDFDictionary * m_shadingDictionary ;
2019-09-13 16:28:20 +02:00
const PDFDictionary * m_patternDictionary ;
2020-08-02 15:29:10 +02:00
ProcedureSets m_procedureSets ;
2019-02-17 18:01:22 +01:00
// Default color spaces
PDFColorSpacePointer m_deviceGrayColorSpace ;
PDFColorSpacePointer m_deviceRGBColorSpace ;
PDFColorSpacePointer m_deviceCMYKColorSpace ;
2019-02-14 19:45:07 +01:00
/// Array with current operand arguments
PDFFlatArray < PDFLexicalAnalyzer : : Token , 33 > m_operands ;
2019-02-17 18:01:22 +01:00
/// Stack with saved graphic states
2019-02-14 19:45:07 +01:00
std : : stack < PDFPageContentProcessorState > m_stack ;
2019-09-29 18:09:09 +02:00
/// Stack with transparency groups
std : : stack < PDFTransparencyGroup > m_transparencyGroupStack ;
2019-07-04 17:52:38 +02:00
/// Stack with marked content
std : : vector < MarkedContentState > m_markedContentStack ;
2019-02-17 18:01:22 +01:00
/// Current graphic state
PDFPageContentProcessorState m_graphicState ;
2019-02-14 19:45:07 +01:00
/// List of errors
QList < PDFRenderError > m_errorList ;
/// Current painter path
QPainterPath m_currentPath ;
2019-03-25 18:44:45 +01:00
/// Nesting level of the begin/end of text object
int m_textBeginEndState ;
2019-03-30 18:45:30 +01:00
2019-06-23 18:35:32 +02:00
/// Compatibility level (if positive, then unrecognized operators are ignored)
int m_compatibilityBeginEndState ;
2019-09-27 18:41:56 +02:00
/// Is drawing uncolored tiling pattern?
int m_drawingUncoloredTilingPatternState ;
2019-03-30 18:45:30 +01:00
/// Actually realized physical font
2019-04-07 19:39:29 +02:00
PDFCachedItem < PDFRealizedFontPointer > m_realizedFont ;
2019-04-29 18:13:16 +02:00
/// Actual clipping path obtained from text. Clipping path
/// is in device space coordinates.
QPainterPath m_textClippingPath ;
2019-08-25 18:16:37 +02:00
/// Base matrix to be used when drawing patterns. Concatenate this matrix
/// with pattern matrix to get transformation from pattern space to device space.
QMatrix m_patternBaseMatrix ;
2019-08-31 14:37:18 +02:00
/// Matrix mapping page points to the device points
QMatrix m_pagePointToDevicePointMatrix ;
2019-08-25 18:16:37 +02:00
/// Bounding rectangle of pages media box in device space coordinates. If drawing rotation
/// is zero, then it corresponds to the scaled media box of the page.
QRectF m_pageBoundingRectDeviceSpace ;
2019-09-28 18:26:31 +02:00
/// Mesh quality settings
PDFMeshQualitySettings m_meshQualitySettings ;
2019-10-05 13:08:32 +02:00
/// Set with rendering errors, which were reported (and should be reported once)
std : : set < QString > m_onceReportedErrors ;
2020-10-17 16:56:39 +02:00
/// Active structural parent key
PDFInteger m_structuralParentKey ;
2019-02-14 19:45:07 +01:00
} ;
} // namespace pdf
2019-02-24 17:48:37 +01:00
# endif // PDFPAGECONTENTPROCESSOR_H