Home | History | Annotate | Download | only in core
      1 /*
      2  * Copyright 2010 The Android Open Source Project
      3  *
      4  * Use of this source code is governed by a BSD-style license that can be
      5  * found in the LICENSE file.
      6  */
      7 
      8 #ifndef SkDevice_DEFINED
      9 #define SkDevice_DEFINED
     10 
     11 #include "SkRefCnt.h"
     12 #include "SkCanvas.h"
     13 #include "SkColor.h"
     14 #include "SkImageFilter.h"
     15 #include "SkSurfaceProps.h"
     16 
     17 class SkBitmap;
     18 class SkClipStack;
     19 class SkDraw;
     20 class SkDrawFilter;
     21 struct SkIRect;
     22 class SkMatrix;
     23 class SkMetaData;
     24 class SkRegion;
     25 class GrRenderTarget;
     26 
     27 class SK_API SkBaseDevice : public SkRefCnt {
     28 public:
     29     /**
     30      *  Construct a new device.
     31     */
     32     explicit SkBaseDevice(const SkSurfaceProps&);
     33     virtual ~SkBaseDevice();
     34 
     35     SkMetaData& getMetaData();
     36 
     37     /**
     38      *  Return ImageInfo for this device. If the canvas is not backed by pixels
     39      *  (cpu or gpu), then the info's ColorType will be kUnknown_SkColorType.
     40      */
     41     virtual SkImageInfo imageInfo() const;
     42 
     43     /**
     44      *  Return the bounds of the device in the coordinate space of the root
     45      *  canvas. The root device will have its top-left at 0,0, but other devices
     46      *  such as those associated with saveLayer may have a non-zero origin.
     47      */
     48     void getGlobalBounds(SkIRect* bounds) const {
     49         SkASSERT(bounds);
     50         const SkIPoint& origin = this->getOrigin();
     51         bounds->setXYWH(origin.x(), origin.y(), this->width(), this->height());
     52     }
     53 
     54     SkIRect getGlobalBounds() const {
     55         SkIRect bounds;
     56         this->getGlobalBounds(&bounds);
     57         return bounds;
     58     }
     59 
     60     int width() const {
     61         return this->imageInfo().width();
     62     }
     63 
     64     int height() const {
     65         return this->imageInfo().height();
     66     }
     67 
     68     bool isOpaque() const {
     69         return this->imageInfo().isOpaque();
     70     }
     71 
     72     /** Return the bitmap associated with this device. Call this each time you need
     73         to access the bitmap, as it notifies the subclass to perform any flushing
     74         etc. before you examine the pixels.
     75         @param changePixels set to true if the caller plans to change the pixels
     76         @return the device's bitmap
     77     */
     78     const SkBitmap& accessBitmap(bool changePixels);
     79 
     80     bool writePixels(const SkImageInfo&, const void*, size_t rowBytes, int x, int y);
     81 
     82     /**
     83      *  Try to get write-access to the pixels behind the device. If successful, this returns true
     84      *  and fills-out the pixmap parameter. On success it also bumps the genID of the underlying
     85      *  bitmap.
     86      *
     87      *  On failure, returns false and ignores the pixmap parameter.
     88      */
     89     bool accessPixels(SkPixmap* pmap);
     90 
     91     /**
     92      *  Try to get read-only-access to the pixels behind the device. If successful, this returns
     93      *  true and fills-out the pixmap parameter.
     94      *
     95      *  On failure, returns false and ignores the pixmap parameter.
     96      */
     97     bool peekPixels(SkPixmap*);
     98 
     99     /**
    100      * Return the device's associated gpu render target, or NULL.
    101      */
    102     virtual GrRenderTarget* accessRenderTarget() { return NULL; }
    103 
    104 
    105     /**
    106      *  Return the device's origin: its offset in device coordinates from
    107      *  the default origin in its canvas' matrix/clip
    108      */
    109     const SkIPoint& getOrigin() const { return fOrigin; }
    110 
    111     /**
    112      * onAttachToCanvas is invoked whenever a device is installed in a canvas
    113      * (i.e., setDevice, saveLayer (for the new device created by the save),
    114      * and SkCanvas' SkBaseDevice & SkBitmap -taking ctors). It allows the
    115      * devices to prepare for drawing (e.g., locking their pixels, etc.)
    116      */
    117     virtual void onAttachToCanvas(SkCanvas*) {
    118         SkASSERT(!fAttachedToCanvas);
    119 #ifdef SK_DEBUG
    120         fAttachedToCanvas = true;
    121 #endif
    122     };
    123 
    124     /**
    125      * onDetachFromCanvas notifies a device that it will no longer be drawn to.
    126      * It gives the device a chance to clean up (e.g., unlock its pixels). It
    127      * is invoked from setDevice (for the displaced device), restore and
    128      * possibly from SkCanvas' dtor.
    129      */
    130     virtual void onDetachFromCanvas() {
    131         SkASSERT(fAttachedToCanvas);
    132 #ifdef SK_DEBUG
    133         fAttachedToCanvas = false;
    134 #endif
    135     };
    136 
    137 protected:
    138     enum TileUsage {
    139         kPossible_TileUsage,    //!< the created device may be drawn tiled
    140         kNever_TileUsage,       //!< the created device will never be drawn tiled
    141     };
    142 
    143     struct TextFlags {
    144         uint32_t    fFlags;     // SkPaint::getFlags()
    145     };
    146 
    147     /**
    148      * Returns the text-related flags, possibly modified based on the state of the
    149      * device (e.g. support for LCD).
    150      */
    151     uint32_t filterTextFlags(const SkPaint&) const;
    152 
    153     virtual bool onShouldDisableLCD(const SkPaint&) const { return false; }
    154 
    155     /**
    156      *
    157      *  DEPRECATED: This will be removed in a future change. Device subclasses
    158      *  should use the matrix and clip from the SkDraw passed to draw functions.
    159      *
    160      *  Called with the correct matrix and clip before this device is drawn
    161      *  to using those settings. If your subclass overrides this, be sure to
    162      *  call through to the base class as well.
    163      *
    164      *  The clipstack is another view of the clip. It records the actual
    165      *  geometry that went into building the region. It is present for devices
    166      *  that want to parse it, but is not required: the region is a complete
    167      *  picture of the current clip. (i.e. if you regionize all of the geometry
    168      *  in the clipstack, you will arrive at an equivalent region to the one
    169      *  passed in).
    170      */
    171      virtual void setMatrixClip(const SkMatrix&, const SkRegion&,
    172                                 const SkClipStack&) {};
    173 
    174     /** These are called inside the per-device-layer loop for each draw call.
    175      When these are called, we have already applied any saveLayer operations,
    176      and are handling any looping from the paint, and any effects from the
    177      DrawFilter.
    178      */
    179     virtual void drawPaint(const SkDraw&, const SkPaint& paint) = 0;
    180     virtual void drawPoints(const SkDraw&, SkCanvas::PointMode mode, size_t count,
    181                             const SkPoint[], const SkPaint& paint) = 0;
    182     virtual void drawRect(const SkDraw&, const SkRect& r,
    183                           const SkPaint& paint) = 0;
    184     virtual void drawOval(const SkDraw&, const SkRect& oval,
    185                           const SkPaint& paint) = 0;
    186     virtual void drawRRect(const SkDraw&, const SkRRect& rr,
    187                            const SkPaint& paint) = 0;
    188 
    189     // Default impl calls drawPath()
    190     virtual void drawDRRect(const SkDraw&, const SkRRect& outer,
    191                             const SkRRect& inner, const SkPaint&);
    192 
    193     /**
    194      *  If pathIsMutable, then the implementation is allowed to cast path to a
    195      *  non-const pointer and modify it in place (as an optimization). Canvas
    196      *  may do this to implement helpers such as drawOval, by placing a temp
    197      *  path on the stack to hold the representation of the oval.
    198      *
    199      *  If prePathMatrix is not null, it should logically be applied before any
    200      *  stroking or other effects. If there are no effects on the paint that
    201      *  affect the geometry/rasterization, then the pre matrix can just be
    202      *  pre-concated with the current matrix.
    203      */
    204     virtual void drawPath(const SkDraw&, const SkPath& path,
    205                           const SkPaint& paint,
    206                           const SkMatrix* prePathMatrix = NULL,
    207                           bool pathIsMutable = false) = 0;
    208     virtual void drawBitmap(const SkDraw&, const SkBitmap& bitmap,
    209                             const SkMatrix& matrix, const SkPaint& paint) = 0;
    210     virtual void drawSprite(const SkDraw&, const SkBitmap& bitmap,
    211                             int x, int y, const SkPaint& paint) = 0;
    212 
    213     /**
    214      *  The default impl. will create a bitmap-shader from the bitmap,
    215      *  and call drawRect with it.
    216      */
    217     virtual void drawBitmapRect(const SkDraw&, const SkBitmap&,
    218                                 const SkRect* srcOrNull, const SkRect& dst,
    219                                 const SkPaint& paint,
    220                                 SkCanvas::SrcRectConstraint) = 0;
    221     virtual void drawBitmapNine(const SkDraw&, const SkBitmap&, const SkIRect& center,
    222                                const SkRect& dst, const SkPaint&);
    223 
    224     virtual void drawImage(const SkDraw&, const SkImage*, SkScalar x, SkScalar y, const SkPaint&);
    225     virtual void drawImageRect(const SkDraw&, const SkImage*, const SkRect* src, const SkRect& dst,
    226                                const SkPaint&, SkCanvas::SrcRectConstraint);
    227     virtual void drawImageNine(const SkDraw&, const SkImage*, const SkIRect& center,
    228                                const SkRect& dst, const SkPaint&);
    229 
    230     /**
    231      *  Does not handle text decoration.
    232      *  Decorations (underline and stike-thru) will be handled by SkCanvas.
    233      */
    234     virtual void drawText(const SkDraw&, const void* text, size_t len,
    235                           SkScalar x, SkScalar y, const SkPaint& paint) = 0;
    236     virtual void drawPosText(const SkDraw&, const void* text, size_t len,
    237                              const SkScalar pos[], int scalarsPerPos,
    238                              const SkPoint& offset, const SkPaint& paint) = 0;
    239     virtual void drawVertices(const SkDraw&, SkCanvas::VertexMode, int vertexCount,
    240                               const SkPoint verts[], const SkPoint texs[],
    241                               const SkColor colors[], SkXfermode* xmode,
    242                               const uint16_t indices[], int indexCount,
    243                               const SkPaint& paint) = 0;
    244     // default implementation unrolls the blob runs.
    245     virtual void drawTextBlob(const SkDraw&, const SkTextBlob*, SkScalar x, SkScalar y,
    246                               const SkPaint& paint, SkDrawFilter* drawFilter);
    247     // default implementation calls drawVertices
    248     virtual void drawPatch(const SkDraw&, const SkPoint cubics[12], const SkColor colors[4],
    249                            const SkPoint texCoords[4], SkXfermode* xmode, const SkPaint& paint);
    250 
    251     // default implementation calls drawPath
    252     virtual void drawAtlas(const SkDraw&, const SkImage* atlas, const SkRSXform[], const SkRect[],
    253                            const SkColor[], int count, SkXfermode::Mode, const SkPaint&);
    254 
    255     /** The SkDevice passed will be an SkDevice which was returned by a call to
    256         onCreateDevice on this device with kNeverTile_TileExpectation.
    257      */
    258     virtual void drawDevice(const SkDraw&, SkBaseDevice*, int x, int y,
    259                             const SkPaint&) = 0;
    260 
    261     virtual void drawTextOnPath(const SkDraw&, const void* text, size_t len, const SkPath&,
    262                                 const SkMatrix*, const SkPaint&);
    263 
    264     bool readPixels(const SkImageInfo&, void* dst, size_t rowBytes, int x, int y);
    265 
    266     ///////////////////////////////////////////////////////////////////////////
    267 
    268     /** Update as needed the pixel value in the bitmap, so that the caller can
    269         access the pixels directly.
    270         @return The device contents as a bitmap
    271     */
    272     virtual const SkBitmap& onAccessBitmap() = 0;
    273 
    274     /**
    275      *  Override and return true for filters that the device can handle
    276      *  intrinsically. Doing so means that SkCanvas will pass-through this
    277      *  filter to drawSprite and drawDevice (and potentially filterImage).
    278      *  Returning false means the SkCanvas will have apply the filter itself,
    279      *  and just pass the resulting image to the device.
    280      */
    281     virtual bool canHandleImageFilter(const SkImageFilter*) { return false; }
    282 
    283     /**
    284      *  Related (but not required) to canHandleImageFilter, this method returns
    285      *  true if the device could apply the filter to the src bitmap and return
    286      *  the result (and updates offset as needed).
    287      *  If the device does not recognize or support this filter,
    288      *  it just returns false and leaves result and offset unchanged.
    289      */
    290     virtual bool filterImage(const SkImageFilter*, const SkBitmap&,
    291                              const SkImageFilter::Context&,
    292                              SkBitmap* /*result*/, SkIPoint* /*offset*/) {
    293         return false;
    294     }
    295 
    296 protected:
    297     virtual SkSurface* newSurface(const SkImageInfo&, const SkSurfaceProps&) { return NULL; }
    298     virtual bool onPeekPixels(SkPixmap*) { return false; }
    299 
    300     /**
    301      *  The caller is responsible for "pre-clipping" the dst. The impl can assume that the dst
    302      *  image at the specified x,y offset will fit within the device's bounds.
    303      *
    304      *  This is explicitly asserted in readPixels(), the public way to call this.
    305      */
    306     virtual bool onReadPixels(const SkImageInfo&, void*, size_t, int x, int y);
    307 
    308     /**
    309      *  The caller is responsible for "pre-clipping" the src. The impl can assume that the src
    310      *  image at the specified x,y offset will fit within the device's bounds.
    311      *
    312      *  This is explicitly asserted in writePixelsDirect(), the public way to call this.
    313      */
    314     virtual bool onWritePixels(const SkImageInfo&, const void*, size_t, int x, int y);
    315 
    316     virtual bool onAccessPixels(SkPixmap*) { return false; }
    317 
    318     const SkSurfaceProps& surfaceProps() const {
    319         return fSurfaceProps;
    320     }
    321 
    322     /**
    323      *  PRIVATE / EXPERIMENTAL -- do not call
    324      *  This entry point gives the backend an opportunity to take over the rendering
    325      *  of 'picture'. If optimization data is available (due to an earlier
    326      *  'optimize' call) this entry point should make use of it and return true
    327      *  if all rendering has been done. If false is returned, SkCanvas will
    328      *  perform its own rendering pass. It is acceptable for the backend
    329      *  to perform some device-specific warm up tasks and then let SkCanvas
    330      *  perform the main rendering loop (by return false from here).
    331      */
    332     virtual bool EXPERIMENTAL_drawPicture(SkCanvas*, const SkPicture*, const SkMatrix*,
    333                                           const SkPaint*);
    334 
    335     struct CreateInfo {
    336         static SkPixelGeometry AdjustGeometry(const SkImageInfo&, TileUsage, SkPixelGeometry,
    337                                               bool preserveLCDText);
    338 
    339         // The constructor may change the pixel geometry based on other parameters.
    340         CreateInfo(const SkImageInfo& info,
    341                    TileUsage tileUsage,
    342                    SkPixelGeometry geo)
    343             : fInfo(info)
    344             , fTileUsage(tileUsage)
    345             , fPixelGeometry(AdjustGeometry(info, tileUsage, geo, false))
    346             , fForImageFilter(false) {}
    347 
    348         CreateInfo(const SkImageInfo& info,
    349                    TileUsage tileUsage,
    350                    SkPixelGeometry geo,
    351                    bool preserveLCDText,
    352                    bool forImageFilter)
    353             : fInfo(info)
    354             , fTileUsage(tileUsage)
    355             , fPixelGeometry(AdjustGeometry(info, tileUsage, geo, preserveLCDText))
    356             , fForImageFilter(forImageFilter) {}
    357 
    358         const SkImageInfo       fInfo;
    359         const TileUsage         fTileUsage;
    360         const SkPixelGeometry   fPixelGeometry;
    361         const bool              fForImageFilter;
    362     };
    363 
    364     /**
    365      *  Create a new device based on CreateInfo. If the paint is not null, then it represents a
    366      *  preview of how the new device will be composed with its creator device (this).
    367      *
    368      *  The subclass may be handed this device in drawDevice(), so it must always return
    369      *  a device that it knows how to draw, and that it knows how to identify if it is not of the
    370      *  same subclass (since drawDevice is passed a SkBaseDevice*). If the subclass cannot fulfill
    371      *  that contract (e.g. PDF cannot support some settings on the paint) it should return NULL,
    372      *  and the caller may then decide to explicitly create a bitmapdevice, knowing that later
    373      *  it could not call drawDevice with it (but it could call drawSprite or drawBitmap).
    374      */
    375     virtual SkBaseDevice* onCreateDevice(const CreateInfo&, const SkPaint*) {
    376         return NULL;
    377     }
    378 
    379 private:
    380     friend class SkCanvas;
    381     friend struct DeviceCM; //for setMatrixClip
    382     friend class SkDraw;
    383     friend class SkDrawIter;
    384     friend class SkDeviceFilteredPaint;
    385     friend class SkImageFilter::DeviceProxy;
    386     friend class SkNoPixelsBitmapDevice;
    387     friend class SkSurface_Raster;
    388 
    389     /**
    390      *  Calls through to drawSprite, processing imagefilter as needed.
    391      */
    392     void drawBitmapAsSprite(const SkDraw&, const SkBitmap&, int x, int y, const SkPaint&);
    393 
    394     // used to change the backend's pixels (and possibly config/rowbytes)
    395     // but cannot change the width/height, so there should be no change to
    396     // any clip information.
    397     // TODO: move to SkBitmapDevice
    398     virtual void replaceBitmapBackendForRasterSurface(const SkBitmap&) {}
    399 
    400     virtual bool forceConservativeRasterClip() const { return false; }
    401 
    402     // just called by SkCanvas when built as a layer
    403     void setOrigin(int x, int y) { fOrigin.set(x, y); }
    404 
    405     /** Causes any deferred drawing to the device to be completed.
    406      */
    407     virtual void flush() {}
    408 
    409     virtual SkImageFilter::Cache* getImageFilterCache() { return NULL; }
    410 
    411     SkIPoint    fOrigin;
    412     SkMetaData* fMetaData;
    413     SkSurfaceProps fSurfaceProps;
    414 
    415 #ifdef SK_DEBUG
    416     bool        fAttachedToCanvas;
    417 #endif
    418 
    419     typedef SkRefCnt INHERITED;
    420 };
    421 
    422 #endif
    423