blob: 27d0a3adbe20b8b45017c3b5409ba05c33dab0e3 [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 */
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +00007#ifndef SkPDFUtils_DEFINED
8#define SkPDFUtils_DEFINED
9
Hal Canary4f29c202017-07-18 10:28:31 -040010#include "SkPDFTypes.h"
vandebo@chromium.org683001c2012-05-09 17:17:51 +000011#include "SkPaint.h"
vandebo@chromium.orgf71b2102011-04-04 19:46:31 +000012#include "SkPath.h"
Hal Canaryd12a6762017-05-26 17:01:16 -040013#include "SkShader.h"
halcanarya76a10b72016-07-07 12:31:55 -070014#include "SkStream.h"
halcanaryf59d18a2016-09-16 14:44:57 -070015#include "SkUtils.h"
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +000016
17class SkMatrix;
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +000018class SkPDFArray;
vandebo@chromium.org8989d312011-05-18 22:10:09 +000019struct SkRect;
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +000020
Hal Canary94fd66c2017-07-05 11:25:42 -040021template <typename T>
22bool SkPackedArrayEqual(T* u, T* v, size_t n) {
23 SkASSERT(u);
24 SkASSERT(v);
25 return 0 == memcmp(u, v, n * sizeof(T));
26}
27
vandebo@chromium.orgf71b2102011-04-04 19:46:31 +000028#if 0
29#define PRINT_NOT_IMPL(str) fprintf(stderr, str)
30#else
31#define PRINT_NOT_IMPL(str)
32#endif
33
34#define NOT_IMPLEMENTED(condition, assert) \
35 do { \
mtklein@google.com330313a2013-08-22 15:37:26 +000036 if ((bool)(condition)) { \
vandebo@chromium.orgf71b2102011-04-04 19:46:31 +000037 PRINT_NOT_IMPL("NOT_IMPLEMENTED: " #condition "\n"); \
38 SkDEBUGCODE(SkASSERT(!assert);) \
39 } \
ctguil@chromium.orga5c72342011-08-15 23:55:03 +000040 } while (0)
vandebo@chromium.orgf71b2102011-04-04 19:46:31 +000041
halcanarya76a10b72016-07-07 12:31:55 -070042namespace SkPDFUtils {
vandebo@chromium.orgcae5fba2011-03-28 19:03:50 +000043
Hal Canarya0622582017-06-29 18:51:35 -040044constexpr float kDpiForRasterScaleOne = 72.0f;
45
halcanarya76a10b72016-07-07 12:31:55 -070046sk_sp<SkPDFArray> RectToArray(const SkRect& rect);
47sk_sp<SkPDFArray> MatrixToArray(const SkMatrix& matrix);
48void AppendTransform(const SkMatrix& matrix, SkWStream* content);
halcanary8e9f5e32016-02-24 15:46:46 -080049
halcanarya76a10b72016-07-07 12:31:55 -070050void MoveTo(SkScalar x, SkScalar y, SkWStream* content);
51void AppendLine(SkScalar x, SkScalar y, SkWStream* content);
52void AppendCubic(SkScalar ctl1X, SkScalar ctl1Y,
53 SkScalar ctl2X, SkScalar ctl2Y,
54 SkScalar dstX, SkScalar dstY, SkWStream* content);
55void AppendRectangle(const SkRect& rect, SkWStream* content);
56void EmitPath(const SkPath& path, SkPaint::Style paintStyle,
Hal Canary385468f2017-02-13 11:03:23 -050057 bool doConsumeDegerates, SkWStream* content, SkScalar tolerance = 0.25f);
halcanarya76a10b72016-07-07 12:31:55 -070058inline void EmitPath(const SkPath& path, SkPaint::Style paintStyle,
Hal Canary385468f2017-02-13 11:03:23 -050059 SkWStream* content, SkScalar tolerance = 0.25f) {
60 SkPDFUtils::EmitPath(path, paintStyle, true, content, tolerance);
halcanarya76a10b72016-07-07 12:31:55 -070061}
62void ClosePath(SkWStream* content);
63void PaintPath(SkPaint::Style style, SkPath::FillType fill,
64 SkWStream* content);
65void StrokePath(SkWStream* content);
66void DrawFormXObject(int objectIndex, SkWStream* content);
67void ApplyGraphicState(int objectIndex, SkWStream* content);
68void ApplyPattern(int objectIndex, SkWStream* content);
69
halcanaryeb92cb32016-07-15 13:41:27 -070070// Converts (value / 255.0) with three significant digits of accuracy.
71// Writes value as string into result. Returns strlen() of result.
72size_t ColorToDecimal(uint8_t value, char result[5]);
73inline void AppendColorComponent(uint8_t value, SkWStream* wStream) {
74 char buffer[5];
75 size_t len = SkPDFUtils::ColorToDecimal(value, buffer);
76 wStream->write(buffer, len);
77}
78
halcanarya76a10b72016-07-07 12:31:55 -070079// 3 = '-', '.', and '\0' characters.
80// 9 = number of significant digits
81// abs(FLT_MIN_10_EXP) = number of zeros in FLT_MIN
82const size_t kMaximumFloatDecimalLength = 3 + 9 - FLT_MIN_10_EXP;
83// FloatToDecimal is exposed for unit tests.
84size_t FloatToDecimal(float value,
85 char output[kMaximumFloatDecimalLength]);
86void AppendScalar(SkScalar value, SkWStream* stream);
87void WriteString(SkWStream* wStream, const char* input, size_t len);
88
89inline void WriteUInt16BE(SkDynamicMemoryWStream* wStream, uint16_t value) {
halcanarya76a10b72016-07-07 12:31:55 -070090 char result[4];
Hal Canaryd6e6e662017-06-17 10:38:13 -040091 result[0] = SkHexadecimalDigits::gUpper[ value >> 12 ];
92 result[1] = SkHexadecimalDigits::gUpper[0xF & (value >> 8 )];
93 result[2] = SkHexadecimalDigits::gUpper[0xF & (value >> 4 )];
94 result[3] = SkHexadecimalDigits::gUpper[0xF & (value )];
halcanarya76a10b72016-07-07 12:31:55 -070095 wStream->write(result, 4);
96}
Hal Canaryd6e6e662017-06-17 10:38:13 -040097
halcanaryf0c30f52016-07-15 13:35:45 -070098inline void WriteUInt8(SkDynamicMemoryWStream* wStream, uint8_t value) {
Hal Canaryd6e6e662017-06-17 10:38:13 -040099 char result[2] = { SkHexadecimalDigits::gUpper[value >> 4],
100 SkHexadecimalDigits::gUpper[value & 0xF] };
halcanaryf0c30f52016-07-15 13:35:45 -0700101 wStream->write(result, 2);
102}
Hal Canaryd6e6e662017-06-17 10:38:13 -0400103
halcanaryf59d18a2016-09-16 14:44:57 -0700104inline void WriteUTF16beHex(SkDynamicMemoryWStream* wStream, SkUnichar utf32) {
105 uint16_t utf16[2] = {0, 0};
106 size_t len = SkUTF16_FromUnichar(utf32, utf16);
107 SkASSERT(len == 1 || len == 2);
108 SkPDFUtils::WriteUInt16BE(wStream, utf16[0]);
109 if (len == 2) {
110 SkPDFUtils::WriteUInt16BE(wStream, utf16[1]);
111 }
112}
Hal Canary5c1b3602017-04-17 16:30:06 -0400113
Hal Canaryd12a6762017-05-26 17:01:16 -0400114inline SkMatrix GetShaderLocalMatrix(const SkShader* shader) {
115 SkMatrix localMatrix;
116 if (sk_sp<SkShader> s = shader->makeAsALocalMatrixShader(&localMatrix)) {
117 return SkMatrix::Concat(s->getLocalMatrix(), localMatrix);
118 }
119 return shader->getLocalMatrix();
120}
Hal Canary94fd66c2017-07-05 11:25:42 -0400121bool InverseTransformBBox(const SkMatrix& matrix, SkRect* bbox);
122void PopulateTilingPatternDict(SkPDFDict* pattern,
123 SkRect& bbox,
124 sk_sp<SkPDFDict> resources,
125 const SkMatrix& matrix);
Hal Canary4f29c202017-07-18 10:28:31 -0400126
127bool ToBitmap(const SkImage* img, SkBitmap* dst);
halcanarya76a10b72016-07-07 12:31:55 -0700128} // namespace SkPDFUtils
ctguil@chromium.orgf966fd32011-03-04 21:47:04 +0000129
130#endif