blob: 0ad1853f3e9b1e6ed77ffc4a9b9274475a51860c [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"
Tom Hudson2880df22015-10-29 09:55:42 -04009#include "SkOncePtr.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#include "SkTypes.h"
15
halcanaryfb62b3d2015-01-21 09:59:14 -080016static const char* as_blend_mode(SkXfermode::Mode mode) {
vandebo@chromium.org48543272011-02-08 19:28:07 +000017 switch (mode) {
halcanaryfb62b3d2015-01-21 09:59:14 -080018 case SkXfermode::kSrcOver_Mode:
19 return "Normal";
20 case SkXfermode::kMultiply_Mode:
21 return "Multiply";
22 case SkXfermode::kScreen_Mode:
23 return "Screen";
24 case SkXfermode::kOverlay_Mode:
25 return "Overlay";
26 case SkXfermode::kDarken_Mode:
27 return "Darken";
28 case SkXfermode::kLighten_Mode:
29 return "Lighten";
30 case SkXfermode::kColorDodge_Mode:
31 return "ColorDodge";
32 case SkXfermode::kColorBurn_Mode:
33 return "ColorBurn";
34 case SkXfermode::kHardLight_Mode:
35 return "HardLight";
36 case SkXfermode::kSoftLight_Mode:
37 return "SoftLight";
38 case SkXfermode::kDifference_Mode:
39 return "Difference";
40 case SkXfermode::kExclusion_Mode:
41 return "Exclusion";
42 case SkXfermode::kHue_Mode:
43 return "Hue";
44 case SkXfermode::kSaturation_Mode:
45 return "Saturation";
46 case SkXfermode::kColor_Mode:
47 return "Color";
48 case SkXfermode::kLuminosity_Mode:
49 return "Luminosity";
vandebo@chromium.org48543272011-02-08 19:28:07 +000050
vandebo@chromium.org25adce82011-05-09 08:05:01 +000051 // These are handled in SkPDFDevice::setUpContentEntry.
vandebo@chromium.org48543272011-02-08 19:28:07 +000052 case SkXfermode::kClear_Mode:
53 case SkXfermode::kSrc_Mode:
54 case SkXfermode::kDst_Mode:
55 case SkXfermode::kDstOver_Mode:
56 case SkXfermode::kSrcIn_Mode:
57 case SkXfermode::kDstIn_Mode:
58 case SkXfermode::kSrcOut_Mode:
59 case SkXfermode::kDstOut_Mode:
vandebo@chromium.org3b416212013-10-30 20:48:05 +000060 case SkXfermode::kSrcATop_Mode:
61 case SkXfermode::kDstATop_Mode:
62 case SkXfermode::kModulate_Mode:
vandebo@chromium.org6112c212011-05-13 03:50:38 +000063 return "Normal";
64
ctguil@chromium.org769fa6a2011-08-20 00:36:18 +000065 // TODO(vandebo): Figure out if we can support more of these modes.
vandebo@chromium.org48543272011-02-08 19:28:07 +000066 case SkXfermode::kXor_Mode:
67 case SkXfermode::kPlus_Mode:
Tom Hudson2880df22015-10-29 09:55:42 -040068 return nullptr;
vandebo@chromium.org48543272011-02-08 19:28:07 +000069 }
Tom Hudson2880df22015-10-29 09:55:42 -040070 return nullptr;
vandebo@chromium.org48543272011-02-08 19:28:07 +000071}
72
halcanarybe27a112015-04-01 13:31:19 -070073// If a SkXfermode is unsupported in PDF, this function returns
74// SrcOver, otherwise, it returns that Xfermode as a Mode.
75static SkXfermode::Mode mode_for_pdf(const SkXfermode* xfermode) {
76 SkXfermode::Mode mode = SkXfermode::kSrcOver_Mode;
77 if (xfermode) {
78 xfermode->asMode(&mode);
halcanaryfb62b3d2015-01-21 09:59:14 -080079 }
halcanarybe27a112015-04-01 13:31:19 -070080 switch (mode) {
81 case SkXfermode::kSrcOver_Mode:
82 case SkXfermode::kMultiply_Mode:
83 case SkXfermode::kScreen_Mode:
84 case SkXfermode::kOverlay_Mode:
85 case SkXfermode::kDarken_Mode:
86 case SkXfermode::kLighten_Mode:
87 case SkXfermode::kColorDodge_Mode:
88 case SkXfermode::kColorBurn_Mode:
89 case SkXfermode::kHardLight_Mode:
90 case SkXfermode::kSoftLight_Mode:
91 case SkXfermode::kDifference_Mode:
92 case SkXfermode::kExclusion_Mode:
93 case SkXfermode::kHue_Mode:
94 case SkXfermode::kSaturation_Mode:
95 case SkXfermode::kColor_Mode:
96 case SkXfermode::kLuminosity_Mode:
97 // Mode is suppported and handled by pdf graphics state.
98 return mode;
99 default:
100 return SkXfermode::kSrcOver_Mode; // Default mode.
halcanaryfb62b3d2015-01-21 09:59:14 -0800101 }
halcanaryfb62b3d2015-01-21 09:59:14 -0800102}
103
halcanarybe27a112015-04-01 13:31:19 -0700104SkPDFGraphicState::SkPDFGraphicState(const SkPaint& p)
105 : fStrokeWidth(p.getStrokeWidth())
106 , fStrokeMiter(p.getStrokeMiter())
107 , fAlpha(p.getAlpha())
108 , fStrokeCap(SkToU8(p.getStrokeCap()))
109 , fStrokeJoin(SkToU8(p.getStrokeJoin()))
110 , fMode(SkToU8(mode_for_pdf(p.getXfermode()))) {}
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000111
vandebo@chromium.org48543272011-02-08 19:28:07 +0000112// static
halcanaryfb62b3d2015-01-21 09:59:14 -0800113SkPDFGraphicState* SkPDFGraphicState::GetGraphicStateForPaint(
halcanary792c80f2015-02-20 07:21:05 -0800114 SkPDFCanon* canon, const SkPaint& paint) {
115 SkASSERT(canon);
halcanarybe27a112015-04-01 13:31:19 -0700116 SkPDFGraphicState key(paint);
117 if (const SkPDFGraphicState* canonGS = canon->findGraphicState(key)) {
118 // The returned SkPDFGraphicState must be made non-const,
119 // since the emitObject() interface is non-const. But We
120 // promise that there is no way to mutate this object from
121 // here on out.
122 return SkRef(const_cast<SkPDFGraphicState*>(canonGS));
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000123 }
halcanarybe27a112015-04-01 13:31:19 -0700124 SkPDFGraphicState* pdfGraphicState = new SkPDFGraphicState(paint);
halcanary792c80f2015-02-20 07:21:05 -0800125 canon->addGraphicState(pdfGraphicState);
halcanaryfb62b3d2015-01-21 09:59:14 -0800126 return pdfGraphicState;
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000127}
128
Tom Hudson2880df22015-10-29 09:55:42 -0400129static SkPDFObject* create_invert_function() {
halcanarye6ea2442015-01-21 13:13:22 -0800130 // Acrobat crashes if we use a type 0 function, kpdf crashes if we use
131 // a type 2 function, so we use a type 4 function.
132 SkAutoTUnref<SkPDFArray> domainAndRange(new SkPDFArray);
133 domainAndRange->reserve(2);
134 domainAndRange->appendInt(0);
135 domainAndRange->appendInt(1);
136
137 static const char psInvert[] = "{1 exch sub}";
138 // Do not copy the trailing '\0' into the SkData.
139 SkAutoTUnref<SkData> psInvertStream(
140 SkData::NewWithoutCopy(psInvert, strlen(psInvert)));
141
Tom Hudson2880df22015-10-29 09:55:42 -0400142 SkPDFStream* invertFunction = new SkPDFStream(psInvertStream.get());
halcanarye6ea2442015-01-21 13:13:22 -0800143 invertFunction->insertInt("FunctionType", 4);
halcanarya25b3372015-04-27 14:00:09 -0700144 invertFunction->insertObject("Domain", SkRef(domainAndRange.get()));
145 invertFunction->insertObject("Range", domainAndRange.detach());
halcanarye6ea2442015-01-21 13:13:22 -0800146 return invertFunction;
147}
148
Tom Hudson2880df22015-10-29 09:55:42 -0400149SK_DECLARE_STATIC_ONCE_PTR(SkPDFObject, invertFunction);
halcanarye6ea2442015-01-21 13:13:22 -0800150
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000151// static
halcanarybe27a112015-04-01 13:31:19 -0700152SkPDFDict* SkPDFGraphicState::GetSMaskGraphicState(SkPDFFormXObject* sMask,
153 bool invert,
154 SkPDFSMaskMode sMaskMode) {
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000155 // The practical chances of using the same mask more than once are unlikely
156 // enough that it's not worth canonicalizing.
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000157 SkAutoTUnref<SkPDFDict> sMaskDict(new SkPDFDict("Mask"));
commit-bot@chromium.org93a2e212013-07-23 23:16:03 +0000158 if (sMaskMode == kAlpha_SMaskMode) {
159 sMaskDict->insertName("S", "Alpha");
160 } else if (sMaskMode == kLuminosity_SMaskMode) {
161 sMaskDict->insertName("S", "Luminosity");
162 }
halcanarya25b3372015-04-27 14:00:09 -0700163 sMaskDict->insertObjRef("G", SkRef(sMask));
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000164 if (invert) {
Tom Hudson2880df22015-10-29 09:55:42 -0400165 sMaskDict->insertObjRef("TR", SkRef(invertFunction.get(create_invert_function)));
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000166 }
167
halcanarybe27a112015-04-01 13:31:19 -0700168 SkPDFDict* result = new SkPDFDict("ExtGState");
halcanarya25b3372015-04-27 14:00:09 -0700169 result->insertObject("SMask", sMaskDict.detach());
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000170 return result;
171}
172
Tom Hudson2880df22015-10-29 09:55:42 -0400173static SkPDFDict* create_no_smask_graphic_state() {
halcanarybe27a112015-04-01 13:31:19 -0700174 SkPDFDict* noSMaskGS = new SkPDFDict("ExtGState");
halcanarye6ea2442015-01-21 13:13:22 -0800175 noSMaskGS->insertName("SMask", "None");
176 return noSMaskGS;
177}
Tom Hudson2880df22015-10-29 09:55:42 -0400178SK_DECLARE_STATIC_ONCE_PTR(SkPDFDict, noSMaskGraphicState);
halcanarye6ea2442015-01-21 13:13:22 -0800179
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000180// static
halcanarybe27a112015-04-01 13:31:19 -0700181SkPDFDict* SkPDFGraphicState::GetNoSMaskGraphicState() {
Tom Hudson2880df22015-10-29 09:55:42 -0400182 return SkRef(noSMaskGraphicState.get(create_no_smask_graphic_state));
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000183}
184
Tom Hudson2880df22015-10-29 09:55:42 -0400185void SkPDFGraphicState::emitObject(
186 SkWStream* stream,
187 const SkPDFObjNumMap& objNumMap,
188 const SkPDFSubstituteMap& substitutes) const {
189 SkAutoTUnref<SkPDFDict> dict(new SkPDFDict("ExtGState"));
halcanarybe27a112015-04-01 13:31:19 -0700190 dict->insertName("Type", "ExtGState");
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000191
reed80ea19c2015-05-12 10:37:34 -0700192 SkScalar alpha = SkIntToScalar(fAlpha) / 0xFF;
halcanarya25b3372015-04-27 14:00:09 -0700193 dict->insertScalar("CA", alpha);
194 dict->insertScalar("ca", alpha);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000195
halcanarybe27a112015-04-01 13:31:19 -0700196 SkPaint::Cap strokeCap = (SkPaint::Cap)fStrokeCap;
197 SkPaint::Join strokeJoin = (SkPaint::Join)fStrokeJoin;
198 SkXfermode::Mode xferMode = (SkXfermode::Mode)fMode;
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000199
Tom Hudson2880df22015-10-29 09:55:42 -0400200 static_assert(SkPaint::kButt_Cap == 0, "paint_cap_mismatch");
201 static_assert(SkPaint::kRound_Cap == 1, "paint_cap_mismatch");
202 static_assert(SkPaint::kSquare_Cap == 2, "paint_cap_mismatch");
203 static_assert(SkPaint::kCapCount == 3, "paint_cap_mismatch");
halcanarybe27a112015-04-01 13:31:19 -0700204 SkASSERT(strokeCap >= 0 && strokeCap <= 2);
205 dict->insertInt("LC", strokeCap);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000206
Tom Hudson2880df22015-10-29 09:55:42 -0400207 static_assert(SkPaint::kMiter_Join == 0, "paint_join_mismatch");
208 static_assert(SkPaint::kRound_Join == 1, "paint_join_mismatch");
209 static_assert(SkPaint::kBevel_Join == 2, "paint_join_mismatch");
210 static_assert(SkPaint::kJoinCount == 3, "paint_join_mismatch");
halcanarybe27a112015-04-01 13:31:19 -0700211 SkASSERT(strokeJoin >= 0 && strokeJoin <= 2);
212 dict->insertInt("LJ", strokeJoin);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000213
halcanarybe27a112015-04-01 13:31:19 -0700214 dict->insertScalar("LW", fStrokeWidth);
215 dict->insertScalar("ML", fStrokeMiter);
halcanarya25b3372015-04-27 14:00:09 -0700216 dict->insertBool("SA", true); // SA = Auto stroke adjustment.
halcanarybe27a112015-04-01 13:31:19 -0700217 dict->insertName("BM", as_blend_mode(xferMode));
218 dict->emitObject(stream, objNumMap, substitutes);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000219}