xref: /aosp_15_r20/external/skia/src/pdf/SkPDFUtils.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2011 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkBitmap.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkBlendMode.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkImage.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPath.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPathTypes.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRect.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkSize.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkStream.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkString.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "include/docs/SkPDFDocument.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkFixed.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkFloatingPoint.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkPoint_impl.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTo.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkGeometry.h"
23*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkPathPriv.h"
24*c8dee2aaSAndroid Build Coastguard Worker #include "src/image/SkImage_Base.h"
25*c8dee2aaSAndroid Build Coastguard Worker #include "src/pdf/SkPDFResourceDict.h"
26*c8dee2aaSAndroid Build Coastguard Worker #include "src/pdf/SkPDFTypes.h"
27*c8dee2aaSAndroid Build Coastguard Worker #include "src/pdf/SkPDFUtils.h"
28*c8dee2aaSAndroid Build Coastguard Worker 
29*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm>
30*c8dee2aaSAndroid Build Coastguard Worker #include <ctime>
31*c8dee2aaSAndroid Build Coastguard Worker #include <utility>
32*c8dee2aaSAndroid Build Coastguard Worker 
33*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_BUILD_FOR_WIN)
34*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkLeanWindows.h"
35*c8dee2aaSAndroid Build Coastguard Worker #endif
36*c8dee2aaSAndroid Build Coastguard Worker 
BlendModeName(SkBlendMode mode)37*c8dee2aaSAndroid Build Coastguard Worker const char* SkPDFUtils::BlendModeName(SkBlendMode mode) {
38*c8dee2aaSAndroid Build Coastguard Worker     // PDF32000.book section 11.3.5 "Blend Mode"
39*c8dee2aaSAndroid Build Coastguard Worker     switch (mode) {
40*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kSrcOver:     return "Normal";
41*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kXor:         return "Normal";  // (unsupported mode)
42*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kPlus:        return "Normal";  // (unsupported mode)
43*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kScreen:      return "Screen";
44*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kOverlay:     return "Overlay";
45*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kDarken:      return "Darken";
46*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kLighten:     return "Lighten";
47*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kColorDodge:  return "ColorDodge";
48*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kColorBurn:   return "ColorBurn";
49*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kHardLight:   return "HardLight";
50*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kSoftLight:   return "SoftLight";
51*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kDifference:  return "Difference";
52*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kExclusion:   return "Exclusion";
53*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kMultiply:    return "Multiply";
54*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kHue:         return "Hue";
55*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kSaturation:  return "Saturation";
56*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kColor:       return "Color";
57*c8dee2aaSAndroid Build Coastguard Worker         case SkBlendMode::kLuminosity:  return "Luminosity";
58*c8dee2aaSAndroid Build Coastguard Worker         // Other blendmodes are handled in SkPDFDevice::setUpContentEntry.
59*c8dee2aaSAndroid Build Coastguard Worker         default:                        return nullptr;
60*c8dee2aaSAndroid Build Coastguard Worker     }
61*c8dee2aaSAndroid Build Coastguard Worker }
62*c8dee2aaSAndroid Build Coastguard Worker 
RectToArray(const SkRect & r)63*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<SkPDFArray> SkPDFUtils::RectToArray(const SkRect& r) {
64*c8dee2aaSAndroid Build Coastguard Worker     return SkPDFMakeArray(r.left(), r.top(), r.right(), r.bottom());
65*c8dee2aaSAndroid Build Coastguard Worker }
66*c8dee2aaSAndroid Build Coastguard Worker 
MatrixToArray(const SkMatrix & matrix)67*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<SkPDFArray> SkPDFUtils::MatrixToArray(const SkMatrix& matrix) {
68*c8dee2aaSAndroid Build Coastguard Worker     SkScalar a[6];
69*c8dee2aaSAndroid Build Coastguard Worker     if (!matrix.asAffine(a)) {
70*c8dee2aaSAndroid Build Coastguard Worker         SkMatrix::SetAffineIdentity(a);
71*c8dee2aaSAndroid Build Coastguard Worker     }
72*c8dee2aaSAndroid Build Coastguard Worker     return SkPDFMakeArray(a[0], a[1], a[2], a[3], a[4], a[5]);
73*c8dee2aaSAndroid Build Coastguard Worker }
74*c8dee2aaSAndroid Build Coastguard Worker 
MoveTo(SkScalar x,SkScalar y,SkWStream * content)75*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::MoveTo(SkScalar x, SkScalar y, SkWStream* content) {
76*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(x, content);
77*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" ");
78*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(y, content);
79*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" m\n");
80*c8dee2aaSAndroid Build Coastguard Worker }
81*c8dee2aaSAndroid Build Coastguard Worker 
AppendLine(SkScalar x,SkScalar y,SkWStream * content)82*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::AppendLine(SkScalar x, SkScalar y, SkWStream* content) {
83*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(x, content);
84*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" ");
85*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(y, content);
86*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" l\n");
87*c8dee2aaSAndroid Build Coastguard Worker }
88*c8dee2aaSAndroid Build Coastguard Worker 
append_cubic(SkScalar ctl1X,SkScalar ctl1Y,SkScalar ctl2X,SkScalar ctl2Y,SkScalar dstX,SkScalar dstY,SkWStream * content)89*c8dee2aaSAndroid Build Coastguard Worker static void append_cubic(SkScalar ctl1X, SkScalar ctl1Y,
90*c8dee2aaSAndroid Build Coastguard Worker                          SkScalar ctl2X, SkScalar ctl2Y,
91*c8dee2aaSAndroid Build Coastguard Worker                          SkScalar dstX, SkScalar dstY, SkWStream* content) {
92*c8dee2aaSAndroid Build Coastguard Worker     SkString cmd("y\n");
93*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(ctl1X, content);
94*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" ");
95*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(ctl1Y, content);
96*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" ");
97*c8dee2aaSAndroid Build Coastguard Worker     if (ctl2X != dstX || ctl2Y != dstY) {
98*c8dee2aaSAndroid Build Coastguard Worker         cmd.set("c\n");
99*c8dee2aaSAndroid Build Coastguard Worker         SkPDFUtils::AppendScalar(ctl2X, content);
100*c8dee2aaSAndroid Build Coastguard Worker         content->writeText(" ");
101*c8dee2aaSAndroid Build Coastguard Worker         SkPDFUtils::AppendScalar(ctl2Y, content);
102*c8dee2aaSAndroid Build Coastguard Worker         content->writeText(" ");
103*c8dee2aaSAndroid Build Coastguard Worker     }
104*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(dstX, content);
105*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" ");
106*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(dstY, content);
107*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" ");
108*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(cmd.c_str());
109*c8dee2aaSAndroid Build Coastguard Worker }
110*c8dee2aaSAndroid Build Coastguard Worker 
append_quad(const SkPoint quad[],SkWStream * content)111*c8dee2aaSAndroid Build Coastguard Worker static void append_quad(const SkPoint quad[], SkWStream* content) {
112*c8dee2aaSAndroid Build Coastguard Worker     SkPoint cubic[4];
113*c8dee2aaSAndroid Build Coastguard Worker     SkConvertQuadToCubic(quad, cubic);
114*c8dee2aaSAndroid Build Coastguard Worker     append_cubic(cubic[1].fX, cubic[1].fY, cubic[2].fX, cubic[2].fY,
115*c8dee2aaSAndroid Build Coastguard Worker                  cubic[3].fX, cubic[3].fY, content);
116*c8dee2aaSAndroid Build Coastguard Worker }
117*c8dee2aaSAndroid Build Coastguard Worker 
AppendRectangle(const SkRect & rect,SkWStream * content)118*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::AppendRectangle(const SkRect& rect, SkWStream* content) {
119*c8dee2aaSAndroid Build Coastguard Worker     // Skia has 0,0 at top left, pdf at bottom left.  Do the right thing.
120*c8dee2aaSAndroid Build Coastguard Worker     SkScalar bottom = std::min(rect.fBottom, rect.fTop);
121*c8dee2aaSAndroid Build Coastguard Worker 
122*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(rect.fLeft, content);
123*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" ");
124*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(bottom, content);
125*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" ");
126*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(rect.width(), content);
127*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" ");
128*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::AppendScalar(rect.height(), content);
129*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" re\n");
130*c8dee2aaSAndroid Build Coastguard Worker }
131*c8dee2aaSAndroid Build Coastguard Worker 
EmitPath(const SkPath & path,SkPaint::Style paintStyle,bool doConsumeDegerates,SkWStream * content,SkScalar tolerance)132*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::EmitPath(const SkPath& path, SkPaint::Style paintStyle,
133*c8dee2aaSAndroid Build Coastguard Worker                           bool doConsumeDegerates, SkWStream* content,
134*c8dee2aaSAndroid Build Coastguard Worker                           SkScalar tolerance) {
135*c8dee2aaSAndroid Build Coastguard Worker     if (path.isEmpty() && SkPaint::kFill_Style == paintStyle) {
136*c8dee2aaSAndroid Build Coastguard Worker         SkPDFUtils::AppendRectangle({0, 0, 0, 0}, content);
137*c8dee2aaSAndroid Build Coastguard Worker         return;
138*c8dee2aaSAndroid Build Coastguard Worker     }
139*c8dee2aaSAndroid Build Coastguard Worker     // Filling a path with no area results in a drawing in PDF renderers but
140*c8dee2aaSAndroid Build Coastguard Worker     // Chrome expects to be able to draw some such entities with no visible
141*c8dee2aaSAndroid Build Coastguard Worker     // result, so we detect those cases and discard the drawing for them.
142*c8dee2aaSAndroid Build Coastguard Worker     // Specifically: moveTo(X), lineTo(Y) and moveTo(X), lineTo(X), lineTo(Y).
143*c8dee2aaSAndroid Build Coastguard Worker 
144*c8dee2aaSAndroid Build Coastguard Worker     SkRect rect;
145*c8dee2aaSAndroid Build Coastguard Worker     bool isClosed; // Both closure and direction need to be checked.
146*c8dee2aaSAndroid Build Coastguard Worker     SkPathDirection direction;
147*c8dee2aaSAndroid Build Coastguard Worker     if (path.isRect(&rect, &isClosed, &direction) &&
148*c8dee2aaSAndroid Build Coastguard Worker         isClosed &&
149*c8dee2aaSAndroid Build Coastguard Worker         (SkPathDirection::kCW == direction ||
150*c8dee2aaSAndroid Build Coastguard Worker          SkPathFillType::kEvenOdd == path.getFillType()))
151*c8dee2aaSAndroid Build Coastguard Worker     {
152*c8dee2aaSAndroid Build Coastguard Worker         SkPDFUtils::AppendRectangle(rect, content);
153*c8dee2aaSAndroid Build Coastguard Worker         return;
154*c8dee2aaSAndroid Build Coastguard Worker     }
155*c8dee2aaSAndroid Build Coastguard Worker 
156*c8dee2aaSAndroid Build Coastguard Worker     enum SkipFillState {
157*c8dee2aaSAndroid Build Coastguard Worker         kEmpty_SkipFillState,
158*c8dee2aaSAndroid Build Coastguard Worker         kSingleLine_SkipFillState,
159*c8dee2aaSAndroid Build Coastguard Worker         kNonSingleLine_SkipFillState,
160*c8dee2aaSAndroid Build Coastguard Worker     };
161*c8dee2aaSAndroid Build Coastguard Worker     SkipFillState fillState = kEmpty_SkipFillState;
162*c8dee2aaSAndroid Build Coastguard Worker     //if (paintStyle != SkPaint::kFill_Style) {
163*c8dee2aaSAndroid Build Coastguard Worker     //    fillState = kNonSingleLine_SkipFillState;
164*c8dee2aaSAndroid Build Coastguard Worker     //}
165*c8dee2aaSAndroid Build Coastguard Worker     SkPoint lastMovePt = SkPoint::Make(0,0);
166*c8dee2aaSAndroid Build Coastguard Worker     SkDynamicMemoryWStream currentSegment;
167*c8dee2aaSAndroid Build Coastguard Worker     SkPoint args[4];
168*c8dee2aaSAndroid Build Coastguard Worker     SkPath::Iter iter(path, false);
169*c8dee2aaSAndroid Build Coastguard Worker     for (SkPath::Verb verb = iter.next(args);
170*c8dee2aaSAndroid Build Coastguard Worker          verb != SkPath::kDone_Verb;
171*c8dee2aaSAndroid Build Coastguard Worker          verb = iter.next(args)) {
172*c8dee2aaSAndroid Build Coastguard Worker         // args gets all the points, even the implicit first point.
173*c8dee2aaSAndroid Build Coastguard Worker         switch (verb) {
174*c8dee2aaSAndroid Build Coastguard Worker             case SkPath::kMove_Verb:
175*c8dee2aaSAndroid Build Coastguard Worker                 MoveTo(args[0].fX, args[0].fY, &currentSegment);
176*c8dee2aaSAndroid Build Coastguard Worker                 lastMovePt = args[0];
177*c8dee2aaSAndroid Build Coastguard Worker                 fillState = kEmpty_SkipFillState;
178*c8dee2aaSAndroid Build Coastguard Worker                 break;
179*c8dee2aaSAndroid Build Coastguard Worker             case SkPath::kLine_Verb:
180*c8dee2aaSAndroid Build Coastguard Worker                 if (!doConsumeDegerates || !SkPathPriv::AllPointsEq(args, 2)) {
181*c8dee2aaSAndroid Build Coastguard Worker                     AppendLine(args[1].fX, args[1].fY, &currentSegment);
182*c8dee2aaSAndroid Build Coastguard Worker                     if ((fillState == kEmpty_SkipFillState) && (args[0] != lastMovePt)) {
183*c8dee2aaSAndroid Build Coastguard Worker                         fillState = kSingleLine_SkipFillState;
184*c8dee2aaSAndroid Build Coastguard Worker                         break;
185*c8dee2aaSAndroid Build Coastguard Worker                     }
186*c8dee2aaSAndroid Build Coastguard Worker                     fillState = kNonSingleLine_SkipFillState;
187*c8dee2aaSAndroid Build Coastguard Worker                 }
188*c8dee2aaSAndroid Build Coastguard Worker                 break;
189*c8dee2aaSAndroid Build Coastguard Worker             case SkPath::kQuad_Verb:
190*c8dee2aaSAndroid Build Coastguard Worker                 if (!doConsumeDegerates || !SkPathPriv::AllPointsEq(args, 3)) {
191*c8dee2aaSAndroid Build Coastguard Worker                     append_quad(args, &currentSegment);
192*c8dee2aaSAndroid Build Coastguard Worker                     fillState = kNonSingleLine_SkipFillState;
193*c8dee2aaSAndroid Build Coastguard Worker                 }
194*c8dee2aaSAndroid Build Coastguard Worker                 break;
195*c8dee2aaSAndroid Build Coastguard Worker             case SkPath::kConic_Verb:
196*c8dee2aaSAndroid Build Coastguard Worker                 if (!doConsumeDegerates || !SkPathPriv::AllPointsEq(args, 3)) {
197*c8dee2aaSAndroid Build Coastguard Worker                     SkAutoConicToQuads converter;
198*c8dee2aaSAndroid Build Coastguard Worker                     const SkPoint* quads = converter.computeQuads(args, iter.conicWeight(), tolerance);
199*c8dee2aaSAndroid Build Coastguard Worker                     for (int i = 0; i < converter.countQuads(); ++i) {
200*c8dee2aaSAndroid Build Coastguard Worker                         append_quad(&quads[i * 2], &currentSegment);
201*c8dee2aaSAndroid Build Coastguard Worker                     }
202*c8dee2aaSAndroid Build Coastguard Worker                     fillState = kNonSingleLine_SkipFillState;
203*c8dee2aaSAndroid Build Coastguard Worker                 }
204*c8dee2aaSAndroid Build Coastguard Worker                 break;
205*c8dee2aaSAndroid Build Coastguard Worker             case SkPath::kCubic_Verb:
206*c8dee2aaSAndroid Build Coastguard Worker                 if (!doConsumeDegerates || !SkPathPriv::AllPointsEq(args, 4)) {
207*c8dee2aaSAndroid Build Coastguard Worker                     append_cubic(args[1].fX, args[1].fY, args[2].fX, args[2].fY,
208*c8dee2aaSAndroid Build Coastguard Worker                                  args[3].fX, args[3].fY, &currentSegment);
209*c8dee2aaSAndroid Build Coastguard Worker                     fillState = kNonSingleLine_SkipFillState;
210*c8dee2aaSAndroid Build Coastguard Worker                 }
211*c8dee2aaSAndroid Build Coastguard Worker                 break;
212*c8dee2aaSAndroid Build Coastguard Worker             case SkPath::kClose_Verb:
213*c8dee2aaSAndroid Build Coastguard Worker                 ClosePath(&currentSegment);
214*c8dee2aaSAndroid Build Coastguard Worker                 currentSegment.writeToStream(content);
215*c8dee2aaSAndroid Build Coastguard Worker                 currentSegment.reset();
216*c8dee2aaSAndroid Build Coastguard Worker                 break;
217*c8dee2aaSAndroid Build Coastguard Worker             default:
218*c8dee2aaSAndroid Build Coastguard Worker                 SkASSERT(false);
219*c8dee2aaSAndroid Build Coastguard Worker                 break;
220*c8dee2aaSAndroid Build Coastguard Worker         }
221*c8dee2aaSAndroid Build Coastguard Worker     }
222*c8dee2aaSAndroid Build Coastguard Worker     if (currentSegment.bytesWritten() > 0) {
223*c8dee2aaSAndroid Build Coastguard Worker         currentSegment.writeToStream(content);
224*c8dee2aaSAndroid Build Coastguard Worker     }
225*c8dee2aaSAndroid Build Coastguard Worker }
226*c8dee2aaSAndroid Build Coastguard Worker 
ClosePath(SkWStream * content)227*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::ClosePath(SkWStream* content) {
228*c8dee2aaSAndroid Build Coastguard Worker     content->writeText("h\n");
229*c8dee2aaSAndroid Build Coastguard Worker }
230*c8dee2aaSAndroid Build Coastguard Worker 
PaintPath(SkPaint::Style style,SkPathFillType fill,SkWStream * content)231*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::PaintPath(SkPaint::Style style, SkPathFillType fill, SkWStream* content) {
232*c8dee2aaSAndroid Build Coastguard Worker     if (style == SkPaint::kFill_Style) {
233*c8dee2aaSAndroid Build Coastguard Worker         content->writeText("f");
234*c8dee2aaSAndroid Build Coastguard Worker     } else if (style == SkPaint::kStrokeAndFill_Style) {
235*c8dee2aaSAndroid Build Coastguard Worker         content->writeText("B");
236*c8dee2aaSAndroid Build Coastguard Worker     } else if (style == SkPaint::kStroke_Style) {
237*c8dee2aaSAndroid Build Coastguard Worker         content->writeText("S");
238*c8dee2aaSAndroid Build Coastguard Worker     }
239*c8dee2aaSAndroid Build Coastguard Worker 
240*c8dee2aaSAndroid Build Coastguard Worker     if (style != SkPaint::kStroke_Style) {
241*c8dee2aaSAndroid Build Coastguard Worker         NOT_IMPLEMENTED(fill == SkPathFillType::kInverseEvenOdd, false);
242*c8dee2aaSAndroid Build Coastguard Worker         NOT_IMPLEMENTED(fill == SkPathFillType::kInverseWinding, false);
243*c8dee2aaSAndroid Build Coastguard Worker         if (fill == SkPathFillType::kEvenOdd) {
244*c8dee2aaSAndroid Build Coastguard Worker             content->writeText("*");
245*c8dee2aaSAndroid Build Coastguard Worker         }
246*c8dee2aaSAndroid Build Coastguard Worker     }
247*c8dee2aaSAndroid Build Coastguard Worker     content->writeText("\n");
248*c8dee2aaSAndroid Build Coastguard Worker }
249*c8dee2aaSAndroid Build Coastguard Worker 
StrokePath(SkWStream * content)250*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::StrokePath(SkWStream* content) {
251*c8dee2aaSAndroid Build Coastguard Worker     SkPDFUtils::PaintPath(SkPaint::kStroke_Style, SkPathFillType::kWinding, content);
252*c8dee2aaSAndroid Build Coastguard Worker }
253*c8dee2aaSAndroid Build Coastguard Worker 
ApplyGraphicState(int objectIndex,SkWStream * content)254*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::ApplyGraphicState(int objectIndex, SkWStream* content) {
255*c8dee2aaSAndroid Build Coastguard Worker     SkPDFWriteResourceName(content, SkPDFResourceType::kExtGState, objectIndex);
256*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" gs\n");
257*c8dee2aaSAndroid Build Coastguard Worker }
258*c8dee2aaSAndroid Build Coastguard Worker 
ApplyPattern(int objectIndex,SkWStream * content)259*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::ApplyPattern(int objectIndex, SkWStream* content) {
260*c8dee2aaSAndroid Build Coastguard Worker     // Select Pattern color space (CS, cs) and set pattern object as current
261*c8dee2aaSAndroid Build Coastguard Worker     // color (SCN, scn)
262*c8dee2aaSAndroid Build Coastguard Worker     content->writeText("/Pattern CS/Pattern cs");
263*c8dee2aaSAndroid Build Coastguard Worker     SkPDFWriteResourceName(content, SkPDFResourceType::kPattern, objectIndex);
264*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" SCN");
265*c8dee2aaSAndroid Build Coastguard Worker     SkPDFWriteResourceName(content, SkPDFResourceType::kPattern, objectIndex);
266*c8dee2aaSAndroid Build Coastguard Worker     content->writeText(" scn\n");
267*c8dee2aaSAndroid Build Coastguard Worker }
268*c8dee2aaSAndroid Build Coastguard Worker 
269*c8dee2aaSAndroid Build Coastguard Worker // return "x/pow(10, places)", given 0<x<pow(10, places)
270*c8dee2aaSAndroid Build Coastguard Worker // result points to places+2 chars.
print_permil_as_decimal(int x,char * result,unsigned places)271*c8dee2aaSAndroid Build Coastguard Worker static size_t print_permil_as_decimal(int x, char* result, unsigned places) {
272*c8dee2aaSAndroid Build Coastguard Worker     result[0] = '.';
273*c8dee2aaSAndroid Build Coastguard Worker     for (int i = places; i > 0; --i) {
274*c8dee2aaSAndroid Build Coastguard Worker         result[i] = '0' + x % 10;
275*c8dee2aaSAndroid Build Coastguard Worker         x /= 10;
276*c8dee2aaSAndroid Build Coastguard Worker     }
277*c8dee2aaSAndroid Build Coastguard Worker     int j;
278*c8dee2aaSAndroid Build Coastguard Worker     for (j = places; j > 1; --j) {
279*c8dee2aaSAndroid Build Coastguard Worker         if (result[j] != '0') {
280*c8dee2aaSAndroid Build Coastguard Worker             break;
281*c8dee2aaSAndroid Build Coastguard Worker         }
282*c8dee2aaSAndroid Build Coastguard Worker     }
283*c8dee2aaSAndroid Build Coastguard Worker     result[j + 1] = '\0';
284*c8dee2aaSAndroid Build Coastguard Worker     return j + 1;
285*c8dee2aaSAndroid Build Coastguard Worker }
286*c8dee2aaSAndroid Build Coastguard Worker 
287*c8dee2aaSAndroid Build Coastguard Worker 
int_pow(int base,unsigned exp,int acc=1)288*c8dee2aaSAndroid Build Coastguard Worker static constexpr int int_pow(int base, unsigned exp, int acc = 1) {
289*c8dee2aaSAndroid Build Coastguard Worker   return exp < 1 ? acc
290*c8dee2aaSAndroid Build Coastguard Worker                  : int_pow(base * base,
291*c8dee2aaSAndroid Build Coastguard Worker                            exp / 2,
292*c8dee2aaSAndroid Build Coastguard Worker                            (exp % 2) ? acc * base : acc);
293*c8dee2aaSAndroid Build Coastguard Worker }
294*c8dee2aaSAndroid Build Coastguard Worker 
295*c8dee2aaSAndroid Build Coastguard Worker 
ColorToDecimalF(float value,char result[kFloatColorDecimalCount+2])296*c8dee2aaSAndroid Build Coastguard Worker size_t SkPDFUtils::ColorToDecimalF(float value, char result[kFloatColorDecimalCount + 2]) {
297*c8dee2aaSAndroid Build Coastguard Worker     static constexpr int kFactor = int_pow(10, kFloatColorDecimalCount);
298*c8dee2aaSAndroid Build Coastguard Worker     int x = sk_float_round2int(value * kFactor);
299*c8dee2aaSAndroid Build Coastguard Worker     if (x >= kFactor || x <= 0) {  // clamp to 0-1
300*c8dee2aaSAndroid Build Coastguard Worker         result[0] = x > 0 ? '1' : '0';
301*c8dee2aaSAndroid Build Coastguard Worker         result[1] = '\0';
302*c8dee2aaSAndroid Build Coastguard Worker         return 1;
303*c8dee2aaSAndroid Build Coastguard Worker     }
304*c8dee2aaSAndroid Build Coastguard Worker     return print_permil_as_decimal(x, result, kFloatColorDecimalCount);
305*c8dee2aaSAndroid Build Coastguard Worker }
306*c8dee2aaSAndroid Build Coastguard Worker 
ColorToDecimal(uint8_t value,char result[5])307*c8dee2aaSAndroid Build Coastguard Worker size_t SkPDFUtils::ColorToDecimal(uint8_t value, char result[5]) {
308*c8dee2aaSAndroid Build Coastguard Worker     if (value == 255 || value == 0) {
309*c8dee2aaSAndroid Build Coastguard Worker         result[0] = value ? '1' : '0';
310*c8dee2aaSAndroid Build Coastguard Worker         result[1] = '\0';
311*c8dee2aaSAndroid Build Coastguard Worker         return 1;
312*c8dee2aaSAndroid Build Coastguard Worker     }
313*c8dee2aaSAndroid Build Coastguard Worker     // int x = 0.5 + (1000.0 / 255.0) * value;
314*c8dee2aaSAndroid Build Coastguard Worker     int x = SkFixedRoundToInt((SK_Fixed1 * 1000 / 255) * value);
315*c8dee2aaSAndroid Build Coastguard Worker     return print_permil_as_decimal(x, result, 3);
316*c8dee2aaSAndroid Build Coastguard Worker }
317*c8dee2aaSAndroid Build Coastguard Worker 
InverseTransformBBox(const SkMatrix & matrix,SkRect * bbox)318*c8dee2aaSAndroid Build Coastguard Worker bool SkPDFUtils::InverseTransformBBox(const SkMatrix& matrix, SkRect* bbox) {
319*c8dee2aaSAndroid Build Coastguard Worker     SkMatrix inverse;
320*c8dee2aaSAndroid Build Coastguard Worker     if (!matrix.invert(&inverse)) {
321*c8dee2aaSAndroid Build Coastguard Worker         return false;
322*c8dee2aaSAndroid Build Coastguard Worker     }
323*c8dee2aaSAndroid Build Coastguard Worker     inverse.mapRect(bbox);
324*c8dee2aaSAndroid Build Coastguard Worker     return true;
325*c8dee2aaSAndroid Build Coastguard Worker }
326*c8dee2aaSAndroid Build Coastguard Worker 
PopulateTilingPatternDict(SkPDFDict * pattern,SkRect & bbox,std::unique_ptr<SkPDFDict> resources,const SkMatrix & matrix)327*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::PopulateTilingPatternDict(SkPDFDict* pattern,
328*c8dee2aaSAndroid Build Coastguard Worker                                            SkRect& bbox,
329*c8dee2aaSAndroid Build Coastguard Worker                                            std::unique_ptr<SkPDFDict> resources,
330*c8dee2aaSAndroid Build Coastguard Worker                                            const SkMatrix& matrix) {
331*c8dee2aaSAndroid Build Coastguard Worker     const int kTiling_PatternType = 1;
332*c8dee2aaSAndroid Build Coastguard Worker     const int kColoredTilingPattern_PaintType = 1;
333*c8dee2aaSAndroid Build Coastguard Worker     const int kConstantSpacing_TilingType = 1;
334*c8dee2aaSAndroid Build Coastguard Worker 
335*c8dee2aaSAndroid Build Coastguard Worker     pattern->insertName("Type", "Pattern");
336*c8dee2aaSAndroid Build Coastguard Worker     pattern->insertInt("PatternType", kTiling_PatternType);
337*c8dee2aaSAndroid Build Coastguard Worker     pattern->insertInt("PaintType", kColoredTilingPattern_PaintType);
338*c8dee2aaSAndroid Build Coastguard Worker     pattern->insertInt("TilingType", kConstantSpacing_TilingType);
339*c8dee2aaSAndroid Build Coastguard Worker     pattern->insertObject("BBox", SkPDFUtils::RectToArray(bbox));
340*c8dee2aaSAndroid Build Coastguard Worker     pattern->insertScalar("XStep", bbox.width());
341*c8dee2aaSAndroid Build Coastguard Worker     pattern->insertScalar("YStep", bbox.height());
342*c8dee2aaSAndroid Build Coastguard Worker     pattern->insertObject("Resources", std::move(resources));
343*c8dee2aaSAndroid Build Coastguard Worker     if (!matrix.isIdentity()) {
344*c8dee2aaSAndroid Build Coastguard Worker         pattern->insertObject("Matrix", SkPDFUtils::MatrixToArray(matrix));
345*c8dee2aaSAndroid Build Coastguard Worker     }
346*c8dee2aaSAndroid Build Coastguard Worker }
347*c8dee2aaSAndroid Build Coastguard Worker 
ToBitmap(const SkImage * img,SkBitmap * dst)348*c8dee2aaSAndroid Build Coastguard Worker bool SkPDFUtils::ToBitmap(const SkImage* img, SkBitmap* dst) {
349*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(img);
350*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(dst);
351*c8dee2aaSAndroid Build Coastguard Worker     SkBitmap bitmap;
352*c8dee2aaSAndroid Build Coastguard Worker     // TODO: support GPU images
353*c8dee2aaSAndroid Build Coastguard Worker     if(as_IB(img)->getROPixels(nullptr, &bitmap)) {
354*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(bitmap.dimensions() == img->dimensions());
355*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!bitmap.drawsNothing());
356*c8dee2aaSAndroid Build Coastguard Worker         *dst = std::move(bitmap);
357*c8dee2aaSAndroid Build Coastguard Worker         return true;
358*c8dee2aaSAndroid Build Coastguard Worker     }
359*c8dee2aaSAndroid Build Coastguard Worker     return false;
360*c8dee2aaSAndroid Build Coastguard Worker }
361*c8dee2aaSAndroid Build Coastguard Worker 
362*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_PDF_BASE85_BINARY
Base85Encode(std::unique_ptr<SkStreamAsset> stream,SkDynamicMemoryWStream * dst)363*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::Base85Encode(std::unique_ptr<SkStreamAsset> stream, SkDynamicMemoryWStream* dst) {
364*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(dst);
365*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(stream);
366*c8dee2aaSAndroid Build Coastguard Worker     dst->writeText("\n");
367*c8dee2aaSAndroid Build Coastguard Worker     int column = 0;
368*c8dee2aaSAndroid Build Coastguard Worker     while (true) {
369*c8dee2aaSAndroid Build Coastguard Worker         uint8_t src[4] = {0, 0, 0, 0};
370*c8dee2aaSAndroid Build Coastguard Worker         size_t count = stream->read(src, 4);
371*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(count < 5);
372*c8dee2aaSAndroid Build Coastguard Worker         if (0 == count) {
373*c8dee2aaSAndroid Build Coastguard Worker             dst->writeText("~>\n");
374*c8dee2aaSAndroid Build Coastguard Worker             return;
375*c8dee2aaSAndroid Build Coastguard Worker         }
376*c8dee2aaSAndroid Build Coastguard Worker         uint32_t v = ((uint32_t)src[0] << 24) | ((uint32_t)src[1] << 16) |
377*c8dee2aaSAndroid Build Coastguard Worker                      ((uint32_t)src[2] <<  8) | src[3];
378*c8dee2aaSAndroid Build Coastguard Worker         if (v == 0 && count == 4) {
379*c8dee2aaSAndroid Build Coastguard Worker             dst->writeText("z");
380*c8dee2aaSAndroid Build Coastguard Worker             column += 1;
381*c8dee2aaSAndroid Build Coastguard Worker         } else {
382*c8dee2aaSAndroid Build Coastguard Worker             char buffer[5];
383*c8dee2aaSAndroid Build Coastguard Worker             for (int n = 4; n > 0; --n) {
384*c8dee2aaSAndroid Build Coastguard Worker                 buffer[n] = (v % 85) + '!';
385*c8dee2aaSAndroid Build Coastguard Worker                 v /= 85;
386*c8dee2aaSAndroid Build Coastguard Worker             }
387*c8dee2aaSAndroid Build Coastguard Worker             buffer[0] = v + '!';
388*c8dee2aaSAndroid Build Coastguard Worker             dst->write(buffer, count + 1);
389*c8dee2aaSAndroid Build Coastguard Worker             column += count + 1;
390*c8dee2aaSAndroid Build Coastguard Worker         }
391*c8dee2aaSAndroid Build Coastguard Worker         if (column > 74) {
392*c8dee2aaSAndroid Build Coastguard Worker             dst->writeText("\n");
393*c8dee2aaSAndroid Build Coastguard Worker             column = 0;
394*c8dee2aaSAndroid Build Coastguard Worker         }
395*c8dee2aaSAndroid Build Coastguard Worker     }
396*c8dee2aaSAndroid Build Coastguard Worker }
397*c8dee2aaSAndroid Build Coastguard Worker #endif //  SK_PDF_BASE85_BINARY
398*c8dee2aaSAndroid Build Coastguard Worker 
AppendTransform(const SkMatrix & matrix,SkWStream * content)399*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::AppendTransform(const SkMatrix& matrix, SkWStream* content) {
400*c8dee2aaSAndroid Build Coastguard Worker     SkScalar values[6];
401*c8dee2aaSAndroid Build Coastguard Worker     if (!matrix.asAffine(values)) {
402*c8dee2aaSAndroid Build Coastguard Worker         SkMatrix::SetAffineIdentity(values);
403*c8dee2aaSAndroid Build Coastguard Worker     }
404*c8dee2aaSAndroid Build Coastguard Worker     for (SkScalar v : values) {
405*c8dee2aaSAndroid Build Coastguard Worker         SkPDFUtils::AppendScalar(v, content);
406*c8dee2aaSAndroid Build Coastguard Worker         content->writeText(" ");
407*c8dee2aaSAndroid Build Coastguard Worker     }
408*c8dee2aaSAndroid Build Coastguard Worker     content->writeText("cm\n");
409*c8dee2aaSAndroid Build Coastguard Worker }
410*c8dee2aaSAndroid Build Coastguard Worker 
411*c8dee2aaSAndroid Build Coastguard Worker 
412*c8dee2aaSAndroid Build Coastguard Worker #if defined(SK_BUILD_FOR_WIN)
413*c8dee2aaSAndroid Build Coastguard Worker 
GetDateTime(SkPDF::DateTime * dt)414*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::GetDateTime(SkPDF::DateTime* dt) {
415*c8dee2aaSAndroid Build Coastguard Worker     if (dt) {
416*c8dee2aaSAndroid Build Coastguard Worker         SYSTEMTIME st;
417*c8dee2aaSAndroid Build Coastguard Worker         GetSystemTime(&st);
418*c8dee2aaSAndroid Build Coastguard Worker         dt->fTimeZoneMinutes = 0;
419*c8dee2aaSAndroid Build Coastguard Worker         dt->fYear       = st.wYear;
420*c8dee2aaSAndroid Build Coastguard Worker         dt->fMonth      = SkToU8(st.wMonth);
421*c8dee2aaSAndroid Build Coastguard Worker         dt->fDayOfWeek  = SkToU8(st.wDayOfWeek);
422*c8dee2aaSAndroid Build Coastguard Worker         dt->fDay        = SkToU8(st.wDay);
423*c8dee2aaSAndroid Build Coastguard Worker         dt->fHour       = SkToU8(st.wHour);
424*c8dee2aaSAndroid Build Coastguard Worker         dt->fMinute     = SkToU8(st.wMinute);
425*c8dee2aaSAndroid Build Coastguard Worker         dt->fSecond     = SkToU8(st.wSecond);
426*c8dee2aaSAndroid Build Coastguard Worker     }
427*c8dee2aaSAndroid Build Coastguard Worker }
428*c8dee2aaSAndroid Build Coastguard Worker 
429*c8dee2aaSAndroid Build Coastguard Worker #else // SK_BUILD_FOR_WIN
430*c8dee2aaSAndroid Build Coastguard Worker 
GetDateTime(SkPDF::DateTime * dt)431*c8dee2aaSAndroid Build Coastguard Worker void SkPDFUtils::GetDateTime(SkPDF::DateTime* dt) {
432*c8dee2aaSAndroid Build Coastguard Worker     if (dt) {
433*c8dee2aaSAndroid Build Coastguard Worker         time_t m_time;
434*c8dee2aaSAndroid Build Coastguard Worker         time(&m_time);
435*c8dee2aaSAndroid Build Coastguard Worker         struct tm tstruct;
436*c8dee2aaSAndroid Build Coastguard Worker         gmtime_r(&m_time, &tstruct);
437*c8dee2aaSAndroid Build Coastguard Worker         dt->fTimeZoneMinutes = 0;
438*c8dee2aaSAndroid Build Coastguard Worker         dt->fYear       = tstruct.tm_year + 1900;
439*c8dee2aaSAndroid Build Coastguard Worker         dt->fMonth      = SkToU8(tstruct.tm_mon + 1);
440*c8dee2aaSAndroid Build Coastguard Worker         dt->fDayOfWeek  = SkToU8(tstruct.tm_wday);
441*c8dee2aaSAndroid Build Coastguard Worker         dt->fDay        = SkToU8(tstruct.tm_mday);
442*c8dee2aaSAndroid Build Coastguard Worker         dt->fHour       = SkToU8(tstruct.tm_hour);
443*c8dee2aaSAndroid Build Coastguard Worker         dt->fMinute     = SkToU8(tstruct.tm_min);
444*c8dee2aaSAndroid Build Coastguard Worker         dt->fSecond     = SkToU8(tstruct.tm_sec);
445*c8dee2aaSAndroid Build Coastguard Worker     }
446*c8dee2aaSAndroid Build Coastguard Worker }
447*c8dee2aaSAndroid Build Coastguard Worker #endif // SK_BUILD_FOR_WIN
448