blob: 5a603ce8da4f1f571650a5b28bd866cadd86f9e4 [file] [log] [blame]
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +00001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * Copyright 2011 Google Inc.
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +00003 *
epoger@google.comec3ed6a2011-07-28 14:26:00 +00004 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +00006 */
7
halcanary67ec1f82014-06-27 11:36:20 -07008#include "SkData.h"
martina.kollarovab8d6af12016-06-29 05:12:31 -07009#include "SkPaint.h"
halcanaryfb62b3d2015-01-21 09:59:14 -080010#include "SkPDFCanon.h"
vandebo@chromium.org6112c212011-05-13 03:50:38 +000011#include "SkPDFFormXObject.h"
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +000012#include "SkPDFGraphicState.h"
vandebo@chromium.orgf71b2102011-04-04 19:46:31 +000013#include "SkPDFUtils.h"
vandebo@chromium.org48543272011-02-08 19:28:07 +000014
halcanaryfb62b3d2015-01-21 09:59:14 -080015static const char* as_blend_mode(SkXfermode::Mode mode) {
vandebo@chromium.org48543272011-02-08 19:28:07 +000016 switch (mode) {
halcanaryfb62b3d2015-01-21 09:59:14 -080017 case SkXfermode::kSrcOver_Mode:
18 return "Normal";
19 case SkXfermode::kMultiply_Mode:
20 return "Multiply";
21 case SkXfermode::kScreen_Mode:
22 return "Screen";
23 case SkXfermode::kOverlay_Mode:
24 return "Overlay";
25 case SkXfermode::kDarken_Mode:
26 return "Darken";
27 case SkXfermode::kLighten_Mode:
28 return "Lighten";
29 case SkXfermode::kColorDodge_Mode:
30 return "ColorDodge";
31 case SkXfermode::kColorBurn_Mode:
32 return "ColorBurn";
33 case SkXfermode::kHardLight_Mode:
34 return "HardLight";
35 case SkXfermode::kSoftLight_Mode:
36 return "SoftLight";
37 case SkXfermode::kDifference_Mode:
38 return "Difference";
39 case SkXfermode::kExclusion_Mode:
40 return "Exclusion";
41 case SkXfermode::kHue_Mode:
42 return "Hue";
43 case SkXfermode::kSaturation_Mode:
44 return "Saturation";
45 case SkXfermode::kColor_Mode:
46 return "Color";
47 case SkXfermode::kLuminosity_Mode:
48 return "Luminosity";
vandebo@chromium.org48543272011-02-08 19:28:07 +000049
vandebo@chromium.org25adce82011-05-09 08:05:01 +000050 // These are handled in SkPDFDevice::setUpContentEntry.
vandebo@chromium.org48543272011-02-08 19:28:07 +000051 case SkXfermode::kClear_Mode:
52 case SkXfermode::kSrc_Mode:
53 case SkXfermode::kDst_Mode:
54 case SkXfermode::kDstOver_Mode:
55 case SkXfermode::kSrcIn_Mode:
56 case SkXfermode::kDstIn_Mode:
57 case SkXfermode::kSrcOut_Mode:
58 case SkXfermode::kDstOut_Mode:
vandebo@chromium.org3b416212013-10-30 20:48:05 +000059 case SkXfermode::kSrcATop_Mode:
60 case SkXfermode::kDstATop_Mode:
61 case SkXfermode::kModulate_Mode:
vandebo@chromium.org6112c212011-05-13 03:50:38 +000062 return "Normal";
63
ctguil@chromium.org769fa6a2011-08-20 00:36:18 +000064 // TODO(vandebo): Figure out if we can support more of these modes.
vandebo@chromium.org48543272011-02-08 19:28:07 +000065 case SkXfermode::kXor_Mode:
66 case SkXfermode::kPlus_Mode:
halcanary96fcdcc2015-08-27 07:41:13 -070067 return nullptr;
vandebo@chromium.org48543272011-02-08 19:28:07 +000068 }
halcanary96fcdcc2015-08-27 07:41:13 -070069 return nullptr;
vandebo@chromium.org48543272011-02-08 19:28:07 +000070}
71
halcanarybe27a112015-04-01 13:31:19 -070072// If a SkXfermode is unsupported in PDF, this function returns
73// SrcOver, otherwise, it returns that Xfermode as a Mode.
74static SkXfermode::Mode mode_for_pdf(const SkXfermode* xfermode) {
75 SkXfermode::Mode mode = SkXfermode::kSrcOver_Mode;
76 if (xfermode) {
77 xfermode->asMode(&mode);
halcanaryfb62b3d2015-01-21 09:59:14 -080078 }
halcanarybe27a112015-04-01 13:31:19 -070079 switch (mode) {
80 case SkXfermode::kSrcOver_Mode:
81 case SkXfermode::kMultiply_Mode:
82 case SkXfermode::kScreen_Mode:
83 case SkXfermode::kOverlay_Mode:
84 case SkXfermode::kDarken_Mode:
85 case SkXfermode::kLighten_Mode:
86 case SkXfermode::kColorDodge_Mode:
87 case SkXfermode::kColorBurn_Mode:
88 case SkXfermode::kHardLight_Mode:
89 case SkXfermode::kSoftLight_Mode:
90 case SkXfermode::kDifference_Mode:
91 case SkXfermode::kExclusion_Mode:
92 case SkXfermode::kHue_Mode:
93 case SkXfermode::kSaturation_Mode:
94 case SkXfermode::kColor_Mode:
95 case SkXfermode::kLuminosity_Mode:
96 // Mode is suppported and handled by pdf graphics state.
97 return mode;
98 default:
99 return SkXfermode::kSrcOver_Mode; // Default mode.
halcanaryfb62b3d2015-01-21 09:59:14 -0800100 }
halcanaryfb62b3d2015-01-21 09:59:14 -0800101}
102
halcanarybe27a112015-04-01 13:31:19 -0700103SkPDFGraphicState::SkPDFGraphicState(const SkPaint& p)
104 : fStrokeWidth(p.getStrokeWidth())
105 , fStrokeMiter(p.getStrokeMiter())
106 , fAlpha(p.getAlpha())
107 , fStrokeCap(SkToU8(p.getStrokeCap()))
108 , fStrokeJoin(SkToU8(p.getStrokeJoin()))
109 , fMode(SkToU8(mode_for_pdf(p.getXfermode()))) {}
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000110
vandebo@chromium.org48543272011-02-08 19:28:07 +0000111// static
halcanaryfb62b3d2015-01-21 09:59:14 -0800112SkPDFGraphicState* SkPDFGraphicState::GetGraphicStateForPaint(
halcanary792c80f2015-02-20 07:21:05 -0800113 SkPDFCanon* canon, const SkPaint& paint) {
114 SkASSERT(canon);
halcanarybe27a112015-04-01 13:31:19 -0700115 SkPDFGraphicState key(paint);
116 if (const SkPDFGraphicState* canonGS = canon->findGraphicState(key)) {
117 // The returned SkPDFGraphicState must be made non-const,
118 // since the emitObject() interface is non-const. But We
119 // promise that there is no way to mutate this object from
120 // here on out.
121 return SkRef(const_cast<SkPDFGraphicState*>(canonGS));
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000122 }
halcanarybe27a112015-04-01 13:31:19 -0700123 SkPDFGraphicState* pdfGraphicState = new SkPDFGraphicState(paint);
halcanary792c80f2015-02-20 07:21:05 -0800124 canon->addGraphicState(pdfGraphicState);
halcanaryfb62b3d2015-01-21 09:59:14 -0800125 return pdfGraphicState;
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000126}
127
halcanary1437c1e2016-03-13 18:30:24 -0700128sk_sp<SkPDFStream> SkPDFGraphicState::MakeInvertFunction() {
halcanarye6ea2442015-01-21 13:13:22 -0800129 // Acrobat crashes if we use a type 0 function, kpdf crashes if we use
130 // a type 2 function, so we use a type 4 function.
halcanaryece83922016-03-08 08:32:12 -0800131 auto domainAndRange = sk_make_sp<SkPDFArray>();
halcanarye6ea2442015-01-21 13:13:22 -0800132 domainAndRange->reserve(2);
133 domainAndRange->appendInt(0);
134 domainAndRange->appendInt(1);
135
136 static const char psInvert[] = "{1 exch sub}";
137 // Do not copy the trailing '\0' into the SkData.
halcanaryfe8f0e02016-07-27 14:14:04 -0700138 auto invertFunction = sk_make_sp<SkPDFStream>(
139 SkData::MakeWithoutCopy(psInvert, strlen(psInvert)));
halcanaryfa251062016-07-29 10:13:18 -0700140 invertFunction->dict()->insertInt("FunctionType", 4);
141 invertFunction->dict()->insertObject("Domain", domainAndRange);
142 invertFunction->dict()->insertObject("Range", std::move(domainAndRange));
halcanary1437c1e2016-03-13 18:30:24 -0700143 return invertFunction;
halcanarye6ea2442015-01-21 13:13:22 -0800144}
145
halcanary1437c1e2016-03-13 18:30:24 -0700146sk_sp<SkPDFDict> SkPDFGraphicState::GetSMaskGraphicState(
halcanarydabd4f02016-08-03 11:16:56 -0700147 sk_sp<SkPDFObject> sMask,
halcanary1437c1e2016-03-13 18:30:24 -0700148 bool invert,
149 SkPDFSMaskMode sMaskMode,
150 SkPDFCanon* canon) {
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000151 // The practical chances of using the same mask more than once are unlikely
152 // enough that it's not worth canonicalizing.
halcanaryece83922016-03-08 08:32:12 -0800153 auto sMaskDict = sk_make_sp<SkPDFDict>("Mask");
commit-bot@chromium.org93a2e212013-07-23 23:16:03 +0000154 if (sMaskMode == kAlpha_SMaskMode) {
155 sMaskDict->insertName("S", "Alpha");
156 } else if (sMaskMode == kLuminosity_SMaskMode) {
157 sMaskDict->insertName("S", "Luminosity");
158 }
halcanarydabd4f02016-08-03 11:16:56 -0700159 sMaskDict->insertObjRef("G", std::move(sMask));
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000160 if (invert) {
halcanary1437c1e2016-03-13 18:30:24 -0700161 // Instead of calling SkPDFGraphicState::MakeInvertFunction,
162 // let the canon deduplicate this object.
163 sMaskDict->insertObjRef("TR", canon->makeInvertFunction());
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000164 }
165
halcanaryece83922016-03-08 08:32:12 -0800166 auto result = sk_make_sp<SkPDFDict>("ExtGState");
halcanary8103a342016-03-08 15:10:16 -0800167 result->insertObject("SMask", std::move(sMaskDict));
halcanary1437c1e2016-03-13 18:30:24 -0700168 return result;
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000169}
170
halcanary1437c1e2016-03-13 18:30:24 -0700171sk_sp<SkPDFDict> SkPDFGraphicState::MakeNoSmaskGraphicState() {
172 auto noSMaskGS = sk_make_sp<SkPDFDict>("ExtGState");
halcanarye6ea2442015-01-21 13:13:22 -0800173 noSMaskGS->insertName("SMask", "None");
174 return noSMaskGS;
175}
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000176
halcanarya060eba2015-08-19 12:26:46 -0700177void SkPDFGraphicState::emitObject(
178 SkWStream* stream,
179 const SkPDFObjNumMap& objNumMap,
180 const SkPDFSubstituteMap& substitutes) const {
halcanaryece83922016-03-08 08:32:12 -0800181 auto dict = sk_make_sp<SkPDFDict>("ExtGState");
halcanarybe27a112015-04-01 13:31:19 -0700182 dict->insertName("Type", "ExtGState");
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000183
reed80ea19c2015-05-12 10:37:34 -0700184 SkScalar alpha = SkIntToScalar(fAlpha) / 0xFF;
halcanarya25b3372015-04-27 14:00:09 -0700185 dict->insertScalar("CA", alpha);
186 dict->insertScalar("ca", alpha);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000187
halcanarybe27a112015-04-01 13:31:19 -0700188 SkPaint::Cap strokeCap = (SkPaint::Cap)fStrokeCap;
189 SkPaint::Join strokeJoin = (SkPaint::Join)fStrokeJoin;
190 SkXfermode::Mode xferMode = (SkXfermode::Mode)fMode;
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000191
bungeman99fe8222015-08-20 07:57:51 -0700192 static_assert(SkPaint::kButt_Cap == 0, "paint_cap_mismatch");
193 static_assert(SkPaint::kRound_Cap == 1, "paint_cap_mismatch");
194 static_assert(SkPaint::kSquare_Cap == 2, "paint_cap_mismatch");
195 static_assert(SkPaint::kCapCount == 3, "paint_cap_mismatch");
halcanarybe27a112015-04-01 13:31:19 -0700196 SkASSERT(strokeCap >= 0 && strokeCap <= 2);
197 dict->insertInt("LC", strokeCap);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000198
bungeman99fe8222015-08-20 07:57:51 -0700199 static_assert(SkPaint::kMiter_Join == 0, "paint_join_mismatch");
200 static_assert(SkPaint::kRound_Join == 1, "paint_join_mismatch");
201 static_assert(SkPaint::kBevel_Join == 2, "paint_join_mismatch");
202 static_assert(SkPaint::kJoinCount == 3, "paint_join_mismatch");
halcanarybe27a112015-04-01 13:31:19 -0700203 SkASSERT(strokeJoin >= 0 && strokeJoin <= 2);
204 dict->insertInt("LJ", strokeJoin);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000205
halcanarybe27a112015-04-01 13:31:19 -0700206 dict->insertScalar("LW", fStrokeWidth);
207 dict->insertScalar("ML", fStrokeMiter);
halcanarya25b3372015-04-27 14:00:09 -0700208 dict->insertBool("SA", true); // SA = Auto stroke adjustment.
halcanarybe27a112015-04-01 13:31:19 -0700209 dict->insertName("BM", as_blend_mode(xferMode));
210 dict->emitObject(stream, objNumMap, substitutes);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000211}