1 /*
2  * Copyright 2011 Google Inc.
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 
8 #ifndef SkPDFDevice_DEFINED
9 #define SkPDFDevice_DEFINED
10 
11 #include "SkBitmap.h"
12 #include "SkCanvas.h"
13 #include "SkClipStack.h"
14 #include "SkClipStackDevice.h"
15 #include "SkData.h"
16 #include "SkKeyedImage.h"
17 #include "SkPDFTypes.h"
18 #include "SkPaint.h"
19 #include "SkRect.h"
20 #include "SkRefCnt.h"
21 #include "SkStream.h"
22 #include "SkTHash.h"
23 #include "SkTextBlobPriv.h"
24 
25 #include <vector>
26 
27 class SkGlyphRunList;
28 class SkKeyedImage;
29 class SkPDFArray;
30 class SkPDFDevice;
31 class SkPDFDict;
32 class SkPDFDocument;
33 class SkPDFFont;
34 class SkPDFObject;
35 class SkPath;
36 class SkRRect;
37 struct SkPDFIndirectReference;
38 
39 /**
40  *  \class SkPDFDevice
41  *
42  *  An SkPDFDevice is the drawing context for a page or layer of PDF
43  *  content.
44  */
45 class SkPDFDevice final : public SkClipStackDevice {
46 public:
47     /**
48      *  @param pageSize Page size in point units.
49      *         1 point == 127/360 mm == 1/72 inch
50      *  @param document  A non-null pointer back to the
51      *         PDFDocument object.  The document is responsible for
52      *         de-duplicating across pages (via the SkPDFDocument) and
53      *         for early serializing of large immutable objects, such
54      *         as images (via SkPDFDocument::serialize()).
55      *  @param initialTransform Transform to be applied to the entire page.
56      */
57     SkPDFDevice(SkISize pageSize, SkPDFDocument* document,
58                 const SkMatrix& initialTransform = SkMatrix::I());
59 
makeCongruentDevice()60     sk_sp<SkPDFDevice> makeCongruentDevice() {
61         return sk_make_sp<SkPDFDevice>(this->size(), fDocument);
62     }
63 
64     ~SkPDFDevice() override;
65 
66     /**
67      *  These are called inside the per-device-layer loop for each draw call.
68      *  When these are called, we have already applied any saveLayer
69      *  operations, and are handling any looping from the paint.
70      */
71     void drawPaint(const SkPaint& paint) override;
72     void drawPoints(SkCanvas::PointMode mode,
73                     size_t count, const SkPoint[],
74                     const SkPaint& paint) override;
75     void drawRect(const SkRect& r, const SkPaint& paint) override;
76     void drawOval(const SkRect& oval, const SkPaint& paint) override;
77     void drawRRect(const SkRRect& rr, const SkPaint& paint) override;
78     void drawPath(const SkPath& origpath, const SkPaint& paint, bool pathIsMutable) override;
79     void drawBitmapRect(const SkBitmap& bitmap, const SkRect* src,
80                         const SkRect& dst, const SkPaint&, SkCanvas::SrcRectConstraint) override;
81     void drawBitmap(const SkBitmap& bitmap, SkScalar x, SkScalar y, const SkPaint&) override;
82     void drawSprite(const SkBitmap& bitmap, int x, int y,
83                     const SkPaint& paint) override;
84     void drawImage(const SkImage*,
85                    SkScalar x,
86                    SkScalar y,
87                    const SkPaint&) override;
88     void drawImageRect(const SkImage*,
89                        const SkRect* src,
90                        const SkRect& dst,
91                        const SkPaint&,
92                        SkCanvas::SrcRectConstraint) override;
93     void drawGlyphRunList(const SkGlyphRunList& glyphRunList) override;
94     void drawVertices(const SkVertices*, const SkVertices::Bone bones[], int boneCount, SkBlendMode,
95                       const SkPaint&) override;
96     void drawDevice(SkBaseDevice*, int x, int y,
97                     const SkPaint&) override;
98 
99     // PDF specific methods.
100 
101     /** Create the resource dictionary for this device. Destructive. */
102     std::unique_ptr<SkPDFDict> makeResourceDict();
103 
104     /** return annotations (link to urls and destinations) or nulltpr */
105     std::unique_ptr<SkPDFArray> getAnnotations();
106 
107     /** Add our named destinations to the supplied dictionary.
108      *  @param dict  Dictionary to add destinations to.
109      *  @param page  The PDF object representing the page for this device.
110      */
111     void appendDestinations(SkPDFDict* dict, SkPDFIndirectReference page) const;
112 
113     /** Returns a SkStream with the page contents.
114      */
115     std::unique_ptr<SkStreamAsset> content();
116 
size()117     SkISize size() const { return this->imageInfo().dimensions(); }
bounds()118     SkIRect bounds() const { return this->imageInfo().bounds(); }
119 
120     // It is important to not confuse GraphicStateEntry with SkPDFGraphicState, the
121     // later being our representation of an object in the PDF file.
122     struct GraphicStateEntry {
123         SkMatrix fMatrix = SkMatrix::I();
124         uint32_t fClipStackGenID = SkClipStack::kWideOpenGenID;
125         SkColor4f fColor = {0, 0, 0, 1};
126         SkScalar fTextScaleX = 1;  // Zero means we don't care what the value is.
127         SkPaint::Style fTextFill = SkPaint::kFill_Style;  // Only if TextScaleX is non-zero.
128         int fShaderIndex = -1;
129         int fGraphicStateIndex = -1;
130     };
131 
132     void DrawGlyphRunAsPath(SkPDFDevice* dev, const SkGlyphRun& glyphRun, SkPoint offset);
133 
134 protected:
135     sk_sp<SkSurface> makeSurface(const SkImageInfo&, const SkSurfaceProps&) override;
136 
137     void drawAnnotation(const SkRect&, const char key[], SkData* value) override;
138 
139     void drawSpecial(SkSpecialImage*, int x, int y, const SkPaint&,
140                      SkImage*, const SkMatrix&) override;
141     sk_sp<SkSpecialImage> makeSpecial(const SkBitmap&) override;
142     sk_sp<SkSpecialImage> makeSpecial(const SkImage*) override;
143     sk_sp<SkSpecialImage> snapSpecial() override;
144     SkImageFilterCache* getImageFilterCache() override;
145 
146 private:
147     struct RectWithData {
148         SkRect rect;
149         sk_sp<SkData> data;
150     };
151 
152     struct NamedDestination {
153         sk_sp<SkData> nameData;
154         SkPoint point;
155     };
156 
157     // TODO(vandebo): push most of SkPDFDevice's state into a core object in
158     // order to get the right access levels without using friend.
159     friend class ScopedContentEntry;
160 
161     SkMatrix fInitialTransform;
162 
163     std::vector<RectWithData> fLinkToURLs;
164     std::vector<RectWithData> fLinkToDestinations;
165     std::vector<NamedDestination> fNamedDestinations;
166 
167     SkTHashSet<SkPDFIndirectReference> fGraphicStateResources;
168     SkTHashSet<SkPDFIndirectReference> fXObjectResources;
169     SkTHashSet<SkPDFIndirectReference> fShaderResources;
170     SkTHashSet<SkPDFIndirectReference> fFontResources;
171     int fNodeId;
172 
173     SkDynamicMemoryWStream fContent;
174     SkDynamicMemoryWStream fContentBuffer;
175     bool fNeedsExtraSave = false;
176     struct GraphicStackState {
177         GraphicStackState(SkDynamicMemoryWStream* s = nullptr);
178         void updateClip(const SkClipStack* clipStack, const SkIRect& bounds);
179         void updateMatrix(const SkMatrix& matrix);
180         void updateDrawingState(const SkPDFDevice::GraphicStateEntry& state);
181         void push();
182         void pop();
183         void drainStack();
currentEntryGraphicStackState184         SkPDFDevice::GraphicStateEntry* currentEntry() { return &fEntries[fStackDepth]; }
185         // Must use stack for matrix, and for clip, plus one for no matrix or clip.
186         static constexpr int kMaxStackDepth = 2;
187         SkPDFDevice::GraphicStateEntry fEntries[kMaxStackDepth + 1];
188         int fStackDepth = 0;
189         SkDynamicMemoryWStream* fContentStream;
190     };
191     GraphicStackState fActiveStackState;
192     SkPDFDocument* fDocument;
193 
194     ////////////////////////////////////////////////////////////////////////////
195 
196     SkBaseDevice* onCreateDevice(const CreateInfo&, const SkPaint*) override;
197 
198     // Set alpha to true if making a transparency group form x-objects.
199     SkPDFIndirectReference makeFormXObjectFromDevice(bool alpha = false);
200 
201     void drawFormXObjectWithMask(SkPDFIndirectReference xObject,
202                                  SkPDFIndirectReference sMask,
203                                  SkBlendMode,
204                                  bool invertClip);
205 
206     // If the paint or clip is such that we shouldn't draw anything, this
207     // returns nullptr and does not create a content entry.
208     // setUpContentEntry and finishContentEntry can be used directly, but
209     // the preferred method is to use the ScopedContentEntry helper class.
210     SkDynamicMemoryWStream* setUpContentEntry(const SkClipStack* clipStack,
211                                               const SkMatrix& matrix,
212                                               const SkPaint& paint,
213                                               SkScalar,
214                                               SkPDFIndirectReference* dst);
215     void finishContentEntry(const SkClipStack*, SkBlendMode, SkPDFIndirectReference, SkPath*);
216     bool isContentEmpty();
217 
218     void internalDrawGlyphRun(const SkGlyphRun& glyphRun, SkPoint offset, const SkPaint& runPaint);
219     void drawGlyphRunAsPath(const SkGlyphRun& glyphRun, SkPoint offset, const SkPaint& runPaint);
220 
221     void internalDrawImageRect(SkKeyedImage,
222                                const SkRect* src,
223                                const SkRect& dst,
224                                const SkPaint&,
225                                const SkMatrix& canvasTransformationMatrix);
226 
227     void internalDrawPath(const SkClipStack&,
228                           const SkMatrix&,
229                           const SkPath&,
230                           const SkPaint&,
231                           bool pathIsMutable);
232 
233     void internalDrawPathWithFilter(const SkClipStack& clipStack,
234                                     const SkMatrix& ctm,
235                                     const SkPath& origPath,
236                                     const SkPaint& paint);
237 
238     bool handleInversePath(const SkPath& origPath, const SkPaint& paint, bool pathIsMutable);
239 
240     void addSMaskGraphicState(sk_sp<SkPDFDevice> maskDevice, SkDynamicMemoryWStream*);
241     void clearMaskOnGraphicState(SkDynamicMemoryWStream*);
242     void setGraphicState(SkPDFIndirectReference gs, SkDynamicMemoryWStream*);
243     void drawFormXObject(SkPDFIndirectReference xObject, SkDynamicMemoryWStream*);
244 
hasEmptyClip()245     bool hasEmptyClip() const { return this->cs().isEmpty(this->bounds()); }
246 
247     void reset();
248 
249     typedef SkClipStackDevice INHERITED;
250 };
251 
252 #endif
253