blob: 93509fedd8e00b6476a1ed707fe147b2f1eff181 [file] [log] [blame]
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +00001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * Copyright 2011 Google Inc.
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +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.
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +00006 */
7
epoger@google.comec3ed6a2011-07-28 14:26:00 +00008
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +00009#ifndef SkPDFUtils_DEFINED
10#define SkPDFUtils_DEFINED
11
vandebo@chromium.org683001c2012-05-09 17:17:51 +000012#include "SkPaint.h"
vandebo@chromium.orgf71b2102011-04-04 19:46:31 +000013#include "SkPath.h"
Hal Canaryd12a6762017-05-26 17:01:16 -040014#include "SkShader.h"
halcanarya76a10b72016-07-07 12:31:55 -070015#include "SkStream.h"
halcanaryf59d18a2016-09-16 14:44:57 -070016#include "SkUtils.h"
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +000017
18class SkMatrix;
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +000019class SkPDFArray;
vandebo@chromium.org8989d312011-05-18 22:10:09 +000020struct SkRect;
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +000021
Hal Canary94fd66c2017-07-05 11:25:42 -040022template <typename T>
23bool SkPackedArrayEqual(T* u, T* v, size_t n) {
24 SkASSERT(u);
25 SkASSERT(v);
26 return 0 == memcmp(u, v, n * sizeof(T));
27}
28
vandebo@chromium.orgf71b2102011-04-04 19:46:31 +000029#if 0
30#define PRINT_NOT_IMPL(str) fprintf(stderr, str)
31#else
32#define PRINT_NOT_IMPL(str)
33#endif
34
35#define NOT_IMPLEMENTED(condition, assert) \
36 do { \
mtklein@google.com330313a2013-08-22 15:37:26 +000037 if ((bool)(condition)) { \
vandebo@chromium.orgf71b2102011-04-04 19:46:31 +000038 PRINT_NOT_IMPL("NOT_IMPLEMENTED: " #condition "\n"); \
39 SkDEBUGCODE(SkASSERT(!assert);) \
40 } \
ctguil@chromium.orga5c72342011-08-15 23:55:03 +000041 } while (0)
vandebo@chromium.orgf71b2102011-04-04 19:46:31 +000042
halcanarya76a10b72016-07-07 12:31:55 -070043namespace SkPDFUtils {
vandebo@chromium.orgcae5fba2011-03-28 19:03:50 +000044
Hal Canarya0622582017-06-29 18:51:35 -040045constexpr float kDpiForRasterScaleOne = 72.0f;
46
halcanarya76a10b72016-07-07 12:31:55 -070047sk_sp<SkPDFArray> RectToArray(const SkRect& rect);
48sk_sp<SkPDFArray> MatrixToArray(const SkMatrix& matrix);
49void AppendTransform(const SkMatrix& matrix, SkWStream* content);
halcanary8e9f5e32016-02-24 15:46:46 -080050
halcanarya76a10b72016-07-07 12:31:55 -070051void MoveTo(SkScalar x, SkScalar y, SkWStream* content);
52void AppendLine(SkScalar x, SkScalar y, SkWStream* content);
53void AppendCubic(SkScalar ctl1X, SkScalar ctl1Y,
54 SkScalar ctl2X, SkScalar ctl2Y,
55 SkScalar dstX, SkScalar dstY, SkWStream* content);
56void AppendRectangle(const SkRect& rect, SkWStream* content);
57void EmitPath(const SkPath& path, SkPaint::Style paintStyle,
Hal Canary385468f2017-02-13 11:03:23 -050058 bool doConsumeDegerates, SkWStream* content, SkScalar tolerance = 0.25f);
halcanarya76a10b72016-07-07 12:31:55 -070059inline void EmitPath(const SkPath& path, SkPaint::Style paintStyle,
Hal Canary385468f2017-02-13 11:03:23 -050060 SkWStream* content, SkScalar tolerance = 0.25f) {
61 SkPDFUtils::EmitPath(path, paintStyle, true, content, tolerance);
halcanarya76a10b72016-07-07 12:31:55 -070062}
63void ClosePath(SkWStream* content);
64void PaintPath(SkPaint::Style style, SkPath::FillType fill,
65 SkWStream* content);
66void StrokePath(SkWStream* content);
67void DrawFormXObject(int objectIndex, SkWStream* content);
68void ApplyGraphicState(int objectIndex, SkWStream* content);
69void ApplyPattern(int objectIndex, SkWStream* content);
70
halcanaryeb92cb32016-07-15 13:41:27 -070071// Converts (value / 255.0) with three significant digits of accuracy.
72// Writes value as string into result. Returns strlen() of result.
73size_t ColorToDecimal(uint8_t value, char result[5]);
74inline void AppendColorComponent(uint8_t value, SkWStream* wStream) {
75 char buffer[5];
76 size_t len = SkPDFUtils::ColorToDecimal(value, buffer);
77 wStream->write(buffer, len);
78}
79
halcanarya76a10b72016-07-07 12:31:55 -070080// 3 = '-', '.', and '\0' characters.
81// 9 = number of significant digits
82// abs(FLT_MIN_10_EXP) = number of zeros in FLT_MIN
83const size_t kMaximumFloatDecimalLength = 3 + 9 - FLT_MIN_10_EXP;
84// FloatToDecimal is exposed for unit tests.
85size_t FloatToDecimal(float value,
86 char output[kMaximumFloatDecimalLength]);
87void AppendScalar(SkScalar value, SkWStream* stream);
88void WriteString(SkWStream* wStream, const char* input, size_t len);
89
90inline void WriteUInt16BE(SkDynamicMemoryWStream* wStream, uint16_t value) {
halcanarya76a10b72016-07-07 12:31:55 -070091 char result[4];
Hal Canaryd6e6e662017-06-17 10:38:13 -040092 result[0] = SkHexadecimalDigits::gUpper[ value >> 12 ];
93 result[1] = SkHexadecimalDigits::gUpper[0xF & (value >> 8 )];
94 result[2] = SkHexadecimalDigits::gUpper[0xF & (value >> 4 )];
95 result[3] = SkHexadecimalDigits::gUpper[0xF & (value )];
halcanarya76a10b72016-07-07 12:31:55 -070096 wStream->write(result, 4);
97}
Hal Canaryd6e6e662017-06-17 10:38:13 -040098
halcanaryf0c30f52016-07-15 13:35:45 -070099inline void WriteUInt8(SkDynamicMemoryWStream* wStream, uint8_t value) {
Hal Canaryd6e6e662017-06-17 10:38:13 -0400100 char result[2] = { SkHexadecimalDigits::gUpper[value >> 4],
101 SkHexadecimalDigits::gUpper[value & 0xF] };
halcanaryf0c30f52016-07-15 13:35:45 -0700102 wStream->write(result, 2);
103}
Hal Canaryd6e6e662017-06-17 10:38:13 -0400104
halcanaryf59d18a2016-09-16 14:44:57 -0700105inline void WriteUTF16beHex(SkDynamicMemoryWStream* wStream, SkUnichar utf32) {
106 uint16_t utf16[2] = {0, 0};
107 size_t len = SkUTF16_FromUnichar(utf32, utf16);
108 SkASSERT(len == 1 || len == 2);
109 SkPDFUtils::WriteUInt16BE(wStream, utf16[0]);
110 if (len == 2) {
111 SkPDFUtils::WriteUInt16BE(wStream, utf16[1]);
112 }
113}
Hal Canary5c1b3602017-04-17 16:30:06 -0400114
Hal Canaryd12a6762017-05-26 17:01:16 -0400115inline SkMatrix GetShaderLocalMatrix(const SkShader* shader) {
116 SkMatrix localMatrix;
117 if (sk_sp<SkShader> s = shader->makeAsALocalMatrixShader(&localMatrix)) {
118 return SkMatrix::Concat(s->getLocalMatrix(), localMatrix);
119 }
120 return shader->getLocalMatrix();
121}
Hal Canary94fd66c2017-07-05 11:25:42 -0400122bool InverseTransformBBox(const SkMatrix& matrix, SkRect* bbox);
123void PopulateTilingPatternDict(SkPDFDict* pattern,
124 SkRect& bbox,
125 sk_sp<SkPDFDict> resources,
126 const SkMatrix& matrix);
halcanarya76a10b72016-07-07 12:31:55 -0700127} // namespace SkPDFUtils
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +0000128
129#endif