| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 1 | /* |
| vandebo@chromium.org | 2a22e10 | 2011-01-25 21:01:34 +0000 | [diff] [blame] | 2 | * Copyright (C) 2011 Google Inc. |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | |
| 17 | #ifndef SkPDFDevice_DEFINED |
| 18 | #define SkPDFDevice_DEFINED |
| 19 | |
| 20 | #include "SkRefCnt.h" |
| 21 | #include "SkDevice.h" |
| vandebo@chromium.org | cae5fba | 2011-03-28 19:03:50 +0000 | [diff] [blame] | 22 | #include "SkStream.h" |
| vandebo@chromium.org | a518086 | 2010-10-26 19:48:49 +0000 | [diff] [blame] | 23 | #include "SkPaint.h" |
| 24 | #include "SkPath.h" |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 25 | |
| 26 | class SkPDFArray; |
| 27 | class SkPDFDevice; |
| 28 | class SkPDFDict; |
| vandebo@chromium.org | 28be72b | 2010-11-11 21:37:00 +0000 | [diff] [blame] | 29 | class SkPDFFont; |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 30 | class SkPDFGraphicState; |
| 31 | class SkPDFObject; |
| vandebo@chromium.org | da912d6 | 2011-03-08 18:31:02 +0000 | [diff] [blame] | 32 | class SkPDFShader; |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 33 | class SkPDFStream; |
| 34 | |
| 35 | class SkPDFDeviceFactory : public SkDeviceFactory { |
| vandebo@chromium.org | 75f97e4 | 2011-04-11 23:24:18 +0000 | [diff] [blame] | 36 | virtual SkDevice* newDevice(SkCanvas*, SkBitmap::Config, int width, |
| 37 | int height, bool isOpaque, bool isForLayer); |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 38 | }; |
| 39 | |
| 40 | /** \class SkPDFDevice |
| 41 | |
| 42 | The drawing context for the PDF backend. |
| 43 | */ |
| 44 | class SkPDFDevice : public SkDevice { |
| 45 | public: |
| 46 | /** Create a PDF drawing context with the given width and height. |
| 47 | * 72 points/in means letter paper is 612x792. |
| 48 | * @param width Page width in points. |
| 49 | * @param height Page height in points. |
| vandebo@chromium.org | 75f97e4 | 2011-04-11 23:24:18 +0000 | [diff] [blame] | 50 | * @param initialTransform The initial transform to apply to the page. |
| 51 | * This may be useful to, for example, move the origin in and |
| 52 | * over a bit to account for a margin, scale the canvas, |
| 53 | * or apply a rotation. Note1: the SkPDFDevice also applies |
| 54 | * a scale+translate transform to move the origin from the |
| 55 | * bottom left (PDF default) to the top left. Note2: drawDevice |
| 56 | * (used by layer restore) draws the device after this initial |
| 57 | * transform is applied, so the PDF device factory does an |
| 58 | * inverse scale+translate to accommodate the one that SkPDFDevice |
| 59 | * always does. |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 60 | */ |
| vandebo@chromium.org | 75f97e4 | 2011-04-11 23:24:18 +0000 | [diff] [blame] | 61 | SkPDFDevice(int width, int height, const SkMatrix& initialTransform); |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 62 | virtual ~SkPDFDevice(); |
| 63 | |
| 64 | virtual SkDeviceFactory* getDeviceFactory() { |
| 65 | return SkNEW(SkPDFDeviceFactory); |
| 66 | } |
| 67 | |
| vandebo@chromium.org | 35fc62b | 2010-10-26 19:47:30 +0000 | [diff] [blame] | 68 | virtual uint32_t getDeviceCapabilities() { return kVector_Capability; } |
| 69 | |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 70 | virtual int width() const { return fWidth; }; |
| 71 | |
| 72 | virtual int height() const { return fHeight; }; |
| 73 | |
| vandebo@chromium.org | 77bcaa3 | 2011-04-15 20:57:37 +0000 | [diff] [blame] | 74 | virtual void clear(SkColor color); |
| 75 | |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 76 | /** Called with the correct matrix and clip before this device is drawn |
| 77 | to using those settings. If your subclass overrides this, be sure to |
| 78 | call through to the base class as well. |
| 79 | */ |
| reed@google.com | 46799cd | 2011-02-22 20:56:26 +0000 | [diff] [blame] | 80 | virtual void setMatrixClip(const SkMatrix&, const SkRegion&, |
| 81 | const SkClipStack&); |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 82 | |
| reed@android.com | f2b98d6 | 2010-12-20 18:26:13 +0000 | [diff] [blame] | 83 | virtual bool readPixels(const SkIRect& srcRect, SkBitmap* bitmap) { |
| 84 | return false; |
| 85 | } |
| 86 | |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 87 | /** These are called inside the per-device-layer loop for each draw call. |
| 88 | When these are called, we have already applied any saveLayer operations, |
| 89 | and are handling any looping from the paint, and any effects from the |
| 90 | DrawFilter. |
| 91 | */ |
| 92 | virtual void drawPaint(const SkDraw&, const SkPaint& paint); |
| 93 | virtual void drawPoints(const SkDraw&, SkCanvas::PointMode mode, |
| 94 | size_t count, const SkPoint[], |
| 95 | const SkPaint& paint); |
| 96 | virtual void drawRect(const SkDraw&, const SkRect& r, const SkPaint& paint); |
| 97 | virtual void drawPath(const SkDraw&, const SkPath& path, |
| vandebo@chromium.org | 02cc5aa | 2011-01-25 22:06:29 +0000 | [diff] [blame] | 98 | const SkPaint& paint, const SkMatrix* prePathMatrix, |
| 99 | bool pathIsMutable); |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 100 | virtual void drawBitmap(const SkDraw&, const SkBitmap& bitmap, |
| reed@android.com | f2b98d6 | 2010-12-20 18:26:13 +0000 | [diff] [blame] | 101 | const SkIRect* srcRectOrNull, |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 102 | const SkMatrix& matrix, const SkPaint& paint); |
| 103 | virtual void drawSprite(const SkDraw&, const SkBitmap& bitmap, int x, int y, |
| 104 | const SkPaint& paint); |
| 105 | virtual void drawText(const SkDraw&, const void* text, size_t len, |
| 106 | SkScalar x, SkScalar y, const SkPaint& paint); |
| 107 | virtual void drawPosText(const SkDraw&, const void* text, size_t len, |
| 108 | const SkScalar pos[], SkScalar constY, |
| 109 | int scalarsPerPos, const SkPaint& paint); |
| 110 | virtual void drawTextOnPath(const SkDraw&, const void* text, size_t len, |
| 111 | const SkPath& path, const SkMatrix* matrix, |
| 112 | const SkPaint& paint); |
| 113 | virtual void drawVertices(const SkDraw&, SkCanvas::VertexMode, |
| 114 | int vertexCount, const SkPoint verts[], |
| 115 | const SkPoint texs[], const SkColor colors[], |
| 116 | SkXfermode* xmode, const uint16_t indices[], |
| 117 | int indexCount, const SkPaint& paint); |
| 118 | virtual void drawDevice(const SkDraw&, SkDevice*, int x, int y, |
| 119 | const SkPaint&); |
| 120 | |
| 121 | // PDF specific methods. |
| 122 | |
| 123 | /** Returns a reference to the resource dictionary for this device. |
| 124 | */ |
| 125 | const SkRefPtr<SkPDFDict>& getResourceDict(); |
| 126 | |
| vandebo@chromium.org | a518086 | 2010-10-26 19:48:49 +0000 | [diff] [blame] | 127 | /** Get the list of resources (PDF objects) used on this page. |
| 128 | * @param resourceList A list to append the resources to. |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 129 | */ |
| vandebo@chromium.org | a518086 | 2010-10-26 19:48:49 +0000 | [diff] [blame] | 130 | void getResources(SkTDArray<SkPDFObject*>* resourceList) const; |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 131 | |
| 132 | /** Returns the media box for this device. |
| 133 | */ |
| vandebo@chromium.org | a518086 | 2010-10-26 19:48:49 +0000 | [diff] [blame] | 134 | SkRefPtr<SkPDFArray> getMediaBox() const; |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 135 | |
| vandebo@chromium.org | c2a9b7f | 2011-02-24 23:22:30 +0000 | [diff] [blame] | 136 | /** Returns a SkStream with the page contents. The caller is responsible |
| 137 | for a reference to the returned value. |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 138 | */ |
| vandebo@chromium.org | c2a9b7f | 2011-02-24 23:22:30 +0000 | [diff] [blame] | 139 | SkStream* content() const; |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 140 | |
| 141 | private: |
| 142 | int fWidth; |
| 143 | int fHeight; |
| vandebo@chromium.org | 75f97e4 | 2011-04-11 23:24:18 +0000 | [diff] [blame] | 144 | SkMatrix fInitialTransform; |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 145 | SkRefPtr<SkPDFDict> fResourceDict; |
| 146 | |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 147 | SkTDArray<SkPDFGraphicState*> fGraphicStateResources; |
| 148 | SkTDArray<SkPDFObject*> fXObjectResources; |
| vandebo@chromium.org | 28be72b | 2010-11-11 21:37:00 +0000 | [diff] [blame] | 149 | SkTDArray<SkPDFFont*> fFontResources; |
| vandebo@chromium.org | da912d6 | 2011-03-08 18:31:02 +0000 | [diff] [blame] | 150 | SkTDArray<SkPDFShader*> fShaderResources; |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 151 | |
| vandebo@chromium.org | 7e2ff7c | 2010-11-03 23:55:28 +0000 | [diff] [blame] | 152 | // In PDF, transforms and clips can only be undone by popping the graphic |
| 153 | // state to before the transform or clip was applied. Because it can be |
| 154 | // a lot of work to reapply a clip and because this class has to apply |
| 155 | // different transforms to accomplish various operations, the clip is |
| 156 | // always applied before a transform and always at a different graphic |
| 157 | // state-stack level than a transform. This strategy results in the |
| 158 | // following possible states for the graphic state stack: |
| 159 | // empty: (identity transform and clip to page) |
| 160 | // one entry: a transform |
| 161 | // one entry: a clip |
| 162 | // two entries: a clip and then a transform |
| vandebo@chromium.org | 2a22e10 | 2011-01-25 21:01:34 +0000 | [diff] [blame] | 163 | // Pointers are owned by the respective Resources list. |
| vandebo@chromium.org | 7e2ff7c | 2010-11-03 23:55:28 +0000 | [diff] [blame] | 164 | struct GraphicStackEntry { |
| 165 | SkColor fColor; |
| vandebo@chromium.org | 28be72b | 2010-11-11 21:37:00 +0000 | [diff] [blame] | 166 | SkScalar fTextSize; |
| vandebo@chromium.org | 7e2ff7c | 2010-11-03 23:55:28 +0000 | [diff] [blame] | 167 | SkScalar fTextScaleX; |
| vandebo@chromium.org | 28be72b | 2010-11-11 21:37:00 +0000 | [diff] [blame] | 168 | SkPaint::Style fTextFill; |
| vandebo@chromium.org | 2a22e10 | 2011-01-25 21:01:34 +0000 | [diff] [blame] | 169 | SkPDFFont* fFont; |
| vandebo@chromium.org | da912d6 | 2011-03-08 18:31:02 +0000 | [diff] [blame] | 170 | SkPDFShader* fShader; |
| vandebo@chromium.org | 2a22e10 | 2011-01-25 21:01:34 +0000 | [diff] [blame] | 171 | SkPDFGraphicState* fGraphicState; |
| vandebo@chromium.org | 7e2ff7c | 2010-11-03 23:55:28 +0000 | [diff] [blame] | 172 | SkRegion fClip; |
| 173 | SkMatrix fTransform; |
| 174 | }; |
| 175 | struct GraphicStackEntry fGraphicStack[3]; |
| 176 | int fGraphicStackIndex; |
| 177 | |
| vandebo@chromium.org | cae5fba | 2011-03-28 19:03:50 +0000 | [diff] [blame] | 178 | SkDynamicMemoryWStream fContent; |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 179 | |
| vandebo@chromium.org | 77bcaa3 | 2011-04-15 20:57:37 +0000 | [diff] [blame] | 180 | void init(); |
| 181 | void cleanUp(); |
| vandebo@chromium.org | 28be72b | 2010-11-11 21:37:00 +0000 | [diff] [blame] | 182 | void updateGSFromPaint(const SkPaint& newPaint, bool forText); |
| vandebo@chromium.org | 2a22e10 | 2011-01-25 21:01:34 +0000 | [diff] [blame] | 183 | void updateFont(const SkPaint& paint, uint16_t glyphID); |
| ctguil@chromium.org | 9db86bb | 2011-03-04 21:43:27 +0000 | [diff] [blame] | 184 | int getFontResourceIndex(SkTypeface* typeface, uint16_t glyphID); |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 185 | |
| vandebo@chromium.org | 7e2ff7c | 2010-11-03 23:55:28 +0000 | [diff] [blame] | 186 | void pushGS(); |
| 187 | void popGS(); |
| vandebo@chromium.org | 28be72b | 2010-11-11 21:37:00 +0000 | [diff] [blame] | 188 | void setTextTransform(SkScalar x, SkScalar y, SkScalar textSkewX); |
| vandebo@chromium.org | 77bcaa3 | 2011-04-15 20:57:37 +0000 | [diff] [blame] | 189 | void internalDrawPaint(const SkPaint& paint); |
| 190 | void internalDrawRect(const SkRect& r, const SkPaint& paint); |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 191 | void internalDrawBitmap(const SkMatrix& matrix, const SkBitmap& bitmap, |
| vandebo@chromium.org | befebb8 | 2011-01-29 01:38:50 +0000 | [diff] [blame] | 192 | const SkIRect* srcRect, const SkPaint& paint); |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 193 | |
| vandebo@chromium.org | 7e2ff7c | 2010-11-03 23:55:28 +0000 | [diff] [blame] | 194 | SkMatrix setTransform(const SkMatrix& matrix); |
| vandebo@chromium.org | 9b49dc0 | 2010-10-20 22:23:29 +0000 | [diff] [blame] | 195 | }; |
| 196 | |
| 197 | #endif |