blob: d60526c11a07e494b0f7adb6480934a1201cb39e [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
reed374772b2016-10-05 17:33:02 -070015static const char* as_blend_mode(SkBlendMode mode) {
vandebo@chromium.org48543272011-02-08 19:28:07 +000016 switch (mode) {
reed374772b2016-10-05 17:33:02 -070017 case SkBlendMode::kSrcOver:
halcanaryfb62b3d2015-01-21 09:59:14 -080018 return "Normal";
reed374772b2016-10-05 17:33:02 -070019 case SkBlendMode::kMultiply:
halcanaryfb62b3d2015-01-21 09:59:14 -080020 return "Multiply";
reed374772b2016-10-05 17:33:02 -070021 case SkBlendMode::kScreen:
halcanaryfb62b3d2015-01-21 09:59:14 -080022 return "Screen";
reed374772b2016-10-05 17:33:02 -070023 case SkBlendMode::kOverlay:
halcanaryfb62b3d2015-01-21 09:59:14 -080024 return "Overlay";
reed374772b2016-10-05 17:33:02 -070025 case SkBlendMode::kDarken:
halcanaryfb62b3d2015-01-21 09:59:14 -080026 return "Darken";
reed374772b2016-10-05 17:33:02 -070027 case SkBlendMode::kLighten:
halcanaryfb62b3d2015-01-21 09:59:14 -080028 return "Lighten";
reed374772b2016-10-05 17:33:02 -070029 case SkBlendMode::kColorDodge:
halcanaryfb62b3d2015-01-21 09:59:14 -080030 return "ColorDodge";
reed374772b2016-10-05 17:33:02 -070031 case SkBlendMode::kColorBurn:
halcanaryfb62b3d2015-01-21 09:59:14 -080032 return "ColorBurn";
reed374772b2016-10-05 17:33:02 -070033 case SkBlendMode::kHardLight:
halcanaryfb62b3d2015-01-21 09:59:14 -080034 return "HardLight";
reed374772b2016-10-05 17:33:02 -070035 case SkBlendMode::kSoftLight:
halcanaryfb62b3d2015-01-21 09:59:14 -080036 return "SoftLight";
reed374772b2016-10-05 17:33:02 -070037 case SkBlendMode::kDifference:
halcanaryfb62b3d2015-01-21 09:59:14 -080038 return "Difference";
reed374772b2016-10-05 17:33:02 -070039 case SkBlendMode::kExclusion:
halcanaryfb62b3d2015-01-21 09:59:14 -080040 return "Exclusion";
reed374772b2016-10-05 17:33:02 -070041 case SkBlendMode::kHue:
halcanaryfb62b3d2015-01-21 09:59:14 -080042 return "Hue";
reed374772b2016-10-05 17:33:02 -070043 case SkBlendMode::kSaturation:
halcanaryfb62b3d2015-01-21 09:59:14 -080044 return "Saturation";
reed374772b2016-10-05 17:33:02 -070045 case SkBlendMode::kColor:
halcanaryfb62b3d2015-01-21 09:59:14 -080046 return "Color";
reed374772b2016-10-05 17:33:02 -070047 case SkBlendMode::kLuminosity:
halcanaryfb62b3d2015-01-21 09:59:14 -080048 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.
reed374772b2016-10-05 17:33:02 -070051 case SkBlendMode::kClear:
52 case SkBlendMode::kSrc:
53 case SkBlendMode::kDst:
54 case SkBlendMode::kDstOver:
55 case SkBlendMode::kSrcIn:
56 case SkBlendMode::kDstIn:
57 case SkBlendMode::kSrcOut:
58 case SkBlendMode::kDstOut:
59 case SkBlendMode::kSrcATop:
60 case SkBlendMode::kDstATop:
61 case SkBlendMode::kModulate:
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.
reed374772b2016-10-05 17:33:02 -070065 case SkBlendMode::kXor:
66 case SkBlendMode::kPlus:
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.
reed374772b2016-10-05 17:33:02 -070074static SkBlendMode mode_for_pdf(SkBlendMode mode) {
halcanarybe27a112015-04-01 13:31:19 -070075 switch (mode) {
reed374772b2016-10-05 17:33:02 -070076 case SkBlendMode::kSrcOver:
77 case SkBlendMode::kMultiply:
78 case SkBlendMode::kScreen:
79 case SkBlendMode::kOverlay:
80 case SkBlendMode::kDarken:
81 case SkBlendMode::kLighten:
82 case SkBlendMode::kColorDodge:
83 case SkBlendMode::kColorBurn:
84 case SkBlendMode::kHardLight:
85 case SkBlendMode::kSoftLight:
86 case SkBlendMode::kDifference:
87 case SkBlendMode::kExclusion:
88 case SkBlendMode::kHue:
89 case SkBlendMode::kSaturation:
90 case SkBlendMode::kColor:
91 case SkBlendMode::kLuminosity:
halcanarybe27a112015-04-01 13:31:19 -070092 // Mode is suppported and handled by pdf graphics state.
93 return mode;
94 default:
reed374772b2016-10-05 17:33:02 -070095 return SkBlendMode::kSrcOver; // Default mode.
halcanaryfb62b3d2015-01-21 09:59:14 -080096 }
halcanaryfb62b3d2015-01-21 09:59:14 -080097}
98
halcanarybe27a112015-04-01 13:31:19 -070099SkPDFGraphicState::SkPDFGraphicState(const SkPaint& p)
100 : fStrokeWidth(p.getStrokeWidth())
101 , fStrokeMiter(p.getStrokeMiter())
102 , fAlpha(p.getAlpha())
103 , fStrokeCap(SkToU8(p.getStrokeCap()))
104 , fStrokeJoin(SkToU8(p.getStrokeJoin()))
reed374772b2016-10-05 17:33:02 -0700105 , fMode(SkToU8((unsigned)mode_for_pdf(p.getBlendMode()))) {}
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000106
vandebo@chromium.org48543272011-02-08 19:28:07 +0000107// static
halcanaryfb62b3d2015-01-21 09:59:14 -0800108SkPDFGraphicState* SkPDFGraphicState::GetGraphicStateForPaint(
halcanary792c80f2015-02-20 07:21:05 -0800109 SkPDFCanon* canon, const SkPaint& paint) {
110 SkASSERT(canon);
halcanarybe27a112015-04-01 13:31:19 -0700111 SkPDFGraphicState key(paint);
112 if (const SkPDFGraphicState* canonGS = canon->findGraphicState(key)) {
113 // The returned SkPDFGraphicState must be made non-const,
114 // since the emitObject() interface is non-const. But We
115 // promise that there is no way to mutate this object from
116 // here on out.
117 return SkRef(const_cast<SkPDFGraphicState*>(canonGS));
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000118 }
halcanarybe27a112015-04-01 13:31:19 -0700119 SkPDFGraphicState* pdfGraphicState = new SkPDFGraphicState(paint);
halcanary792c80f2015-02-20 07:21:05 -0800120 canon->addGraphicState(pdfGraphicState);
halcanaryfb62b3d2015-01-21 09:59:14 -0800121 return pdfGraphicState;
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000122}
123
halcanary1437c1e2016-03-13 18:30:24 -0700124sk_sp<SkPDFStream> SkPDFGraphicState::MakeInvertFunction() {
halcanarye6ea2442015-01-21 13:13:22 -0800125 // Acrobat crashes if we use a type 0 function, kpdf crashes if we use
126 // a type 2 function, so we use a type 4 function.
halcanaryece83922016-03-08 08:32:12 -0800127 auto domainAndRange = sk_make_sp<SkPDFArray>();
halcanarye6ea2442015-01-21 13:13:22 -0800128 domainAndRange->reserve(2);
129 domainAndRange->appendInt(0);
130 domainAndRange->appendInt(1);
131
132 static const char psInvert[] = "{1 exch sub}";
133 // Do not copy the trailing '\0' into the SkData.
halcanaryfe8f0e02016-07-27 14:14:04 -0700134 auto invertFunction = sk_make_sp<SkPDFStream>(
135 SkData::MakeWithoutCopy(psInvert, strlen(psInvert)));
halcanaryfa251062016-07-29 10:13:18 -0700136 invertFunction->dict()->insertInt("FunctionType", 4);
137 invertFunction->dict()->insertObject("Domain", domainAndRange);
138 invertFunction->dict()->insertObject("Range", std::move(domainAndRange));
halcanary1437c1e2016-03-13 18:30:24 -0700139 return invertFunction;
halcanarye6ea2442015-01-21 13:13:22 -0800140}
141
halcanary1437c1e2016-03-13 18:30:24 -0700142sk_sp<SkPDFDict> SkPDFGraphicState::GetSMaskGraphicState(
halcanarydabd4f02016-08-03 11:16:56 -0700143 sk_sp<SkPDFObject> sMask,
halcanary1437c1e2016-03-13 18:30:24 -0700144 bool invert,
145 SkPDFSMaskMode sMaskMode,
146 SkPDFCanon* canon) {
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000147 // The practical chances of using the same mask more than once are unlikely
148 // enough that it's not worth canonicalizing.
halcanaryece83922016-03-08 08:32:12 -0800149 auto sMaskDict = sk_make_sp<SkPDFDict>("Mask");
commit-bot@chromium.org93a2e212013-07-23 23:16:03 +0000150 if (sMaskMode == kAlpha_SMaskMode) {
151 sMaskDict->insertName("S", "Alpha");
152 } else if (sMaskMode == kLuminosity_SMaskMode) {
153 sMaskDict->insertName("S", "Luminosity");
154 }
halcanarydabd4f02016-08-03 11:16:56 -0700155 sMaskDict->insertObjRef("G", std::move(sMask));
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000156 if (invert) {
halcanary1437c1e2016-03-13 18:30:24 -0700157 // Instead of calling SkPDFGraphicState::MakeInvertFunction,
158 // let the canon deduplicate this object.
159 sMaskDict->insertObjRef("TR", canon->makeInvertFunction());
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000160 }
161
halcanaryece83922016-03-08 08:32:12 -0800162 auto result = sk_make_sp<SkPDFDict>("ExtGState");
halcanary8103a342016-03-08 15:10:16 -0800163 result->insertObject("SMask", std::move(sMaskDict));
halcanary1437c1e2016-03-13 18:30:24 -0700164 return result;
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000165}
166
halcanary1437c1e2016-03-13 18:30:24 -0700167sk_sp<SkPDFDict> SkPDFGraphicState::MakeNoSmaskGraphicState() {
168 auto noSMaskGS = sk_make_sp<SkPDFDict>("ExtGState");
halcanarye6ea2442015-01-21 13:13:22 -0800169 noSMaskGS->insertName("SMask", "None");
170 return noSMaskGS;
171}
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000172
halcanarya060eba2015-08-19 12:26:46 -0700173void SkPDFGraphicState::emitObject(
174 SkWStream* stream,
halcanary530032a2016-08-18 14:22:52 -0700175 const SkPDFObjNumMap& objNumMap) const {
halcanaryece83922016-03-08 08:32:12 -0800176 auto dict = sk_make_sp<SkPDFDict>("ExtGState");
halcanarybe27a112015-04-01 13:31:19 -0700177 dict->insertName("Type", "ExtGState");
vandebo@chromium.org6112c212011-05-13 03:50:38 +0000178
reed80ea19c2015-05-12 10:37:34 -0700179 SkScalar alpha = SkIntToScalar(fAlpha) / 0xFF;
halcanarya25b3372015-04-27 14:00:09 -0700180 dict->insertScalar("CA", alpha);
181 dict->insertScalar("ca", alpha);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000182
halcanarybe27a112015-04-01 13:31:19 -0700183 SkPaint::Cap strokeCap = (SkPaint::Cap)fStrokeCap;
184 SkPaint::Join strokeJoin = (SkPaint::Join)fStrokeJoin;
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000185
bungeman99fe8222015-08-20 07:57:51 -0700186 static_assert(SkPaint::kButt_Cap == 0, "paint_cap_mismatch");
187 static_assert(SkPaint::kRound_Cap == 1, "paint_cap_mismatch");
188 static_assert(SkPaint::kSquare_Cap == 2, "paint_cap_mismatch");
189 static_assert(SkPaint::kCapCount == 3, "paint_cap_mismatch");
halcanarybe27a112015-04-01 13:31:19 -0700190 SkASSERT(strokeCap >= 0 && strokeCap <= 2);
191 dict->insertInt("LC", strokeCap);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000192
bungeman99fe8222015-08-20 07:57:51 -0700193 static_assert(SkPaint::kMiter_Join == 0, "paint_join_mismatch");
194 static_assert(SkPaint::kRound_Join == 1, "paint_join_mismatch");
195 static_assert(SkPaint::kBevel_Join == 2, "paint_join_mismatch");
196 static_assert(SkPaint::kJoinCount == 3, "paint_join_mismatch");
halcanarybe27a112015-04-01 13:31:19 -0700197 SkASSERT(strokeJoin >= 0 && strokeJoin <= 2);
198 dict->insertInt("LJ", strokeJoin);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000199
halcanarybe27a112015-04-01 13:31:19 -0700200 dict->insertScalar("LW", fStrokeWidth);
201 dict->insertScalar("ML", fStrokeMiter);
halcanarya25b3372015-04-27 14:00:09 -0700202 dict->insertBool("SA", true); // SA = Auto stroke adjustment.
reed374772b2016-10-05 17:33:02 -0700203 dict->insertName("BM", as_blend_mode((SkBlendMode)fMode));
halcanary530032a2016-08-18 14:22:52 -0700204 dict->emitObject(stream, objNumMap);
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +0000205}