reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 1 | #include "SkBlurDrawLooper.h" |
| 2 | #include "SkBlurMaskFilter.h" |
| 3 | #include "SkCanvas.h" |
| 4 | #include "SkPaint.h" |
| 5 | #include "SkMaskFilter.h" |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 6 | #include "SkColorFilter.h" |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 7 | |
| 8 | SkBlurDrawLooper::SkBlurDrawLooper(SkScalar radius, SkScalar dx, SkScalar dy, |
senorblanco@chromium.org | 038aff6 | 2010-12-06 23:45:58 +0000 | [diff] [blame] | 9 | SkColor color, uint32_t flags) |
| 10 | : fDx(dx), fDy(dy), fBlurColor(color), fBlurFlags(flags) |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 11 | { |
senorblanco@chromium.org | 038aff6 | 2010-12-06 23:45:58 +0000 | [diff] [blame] | 12 | SkASSERT(flags <= kAll_BlurFlag); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 13 | if (radius > 0) |
senorblanco@chromium.org | 038aff6 | 2010-12-06 23:45:58 +0000 | [diff] [blame] | 14 | { |
reed@google.com | 82065d6 | 2011-02-07 15:30:46 +0000 | [diff] [blame] | 15 | uint32_t blurFlags = flags & kIgnoreTransform_BlurFlag ? |
| 16 | SkBlurMaskFilter::kIgnoreTransform_BlurFlag : |
senorblanco@chromium.org | 038aff6 | 2010-12-06 23:45:58 +0000 | [diff] [blame] | 17 | SkBlurMaskFilter::kNone_BlurFlag; |
| 18 | |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 19 | blurFlags |= flags & kHighQuality_BlurFlag ? |
| 20 | SkBlurMaskFilter::kHighQuality_BlurFlag : |
| 21 | SkBlurMaskFilter::kNone_BlurFlag; |
| 22 | |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 23 | fBlur = SkBlurMaskFilter::Create(radius, |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 24 | SkBlurMaskFilter::kNormal_BlurStyle, |
senorblanco@chromium.org | 038aff6 | 2010-12-06 23:45:58 +0000 | [diff] [blame] | 25 | blurFlags); |
| 26 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 27 | else |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 28 | { |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 29 | fBlur = NULL; |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 30 | } |
| 31 | |
| 32 | if (flags & kOverrideColor_BlurFlag) |
| 33 | { |
junov@google.com | a639891 | 2011-03-11 16:45:00 +0000 | [diff] [blame] | 34 | // Set alpha to 1 for the override since transparency will already |
| 35 | // be baked into the blurred mask. |
| 36 | SkColor opaqueColor = SkColorSetA(color, 255); |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 37 | //The SrcIn xfer mode will multiply 'color' by the incoming alpha |
junov@google.com | a639891 | 2011-03-11 16:45:00 +0000 | [diff] [blame] | 38 | fColorFilter = SkColorFilter::CreateModeFilter(opaqueColor, SkXfermode::kSrcIn_Mode); |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 39 | } |
| 40 | else |
| 41 | { |
| 42 | fColorFilter = NULL; |
| 43 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 44 | } |
| 45 | |
| 46 | SkBlurDrawLooper::SkBlurDrawLooper(SkFlattenableReadBuffer& buffer) |
| 47 | { |
| 48 | fDx = buffer.readScalar(); |
| 49 | fDy = buffer.readScalar(); |
| 50 | fBlurColor = buffer.readU32(); |
| 51 | fBlur = static_cast<SkMaskFilter*>(buffer.readFlattenable()); |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 52 | fColorFilter = static_cast<SkColorFilter*>(buffer.readFlattenable()); |
senorblanco@chromium.org | 038aff6 | 2010-12-06 23:45:58 +0000 | [diff] [blame] | 53 | fBlurFlags = buffer.readU32() & kAll_BlurFlag; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 54 | } |
| 55 | |
| 56 | SkBlurDrawLooper::~SkBlurDrawLooper() |
| 57 | { |
reed@google.com | 82065d6 | 2011-02-07 15:30:46 +0000 | [diff] [blame] | 58 | SkSafeUnref(fBlur); |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 59 | SkSafeUnref(fColorFilter); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 60 | } |
| 61 | |
| 62 | void SkBlurDrawLooper::flatten(SkFlattenableWriteBuffer& buffer) |
| 63 | { |
| 64 | buffer.writeScalar(fDx); |
| 65 | buffer.writeScalar(fDy); |
| 66 | buffer.write32(fBlurColor); |
| 67 | buffer.writeFlattenable(fBlur); |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 68 | buffer.writeFlattenable(fColorFilter); |
senorblanco@chromium.org | 038aff6 | 2010-12-06 23:45:58 +0000 | [diff] [blame] | 69 | buffer.write32(fBlurFlags); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 70 | } |
| 71 | |
| 72 | void SkBlurDrawLooper::init(SkCanvas* canvas, SkPaint* paint) |
| 73 | { |
| 74 | // we do nothing if a maskfilter is already installed |
| 75 | if (paint->getMaskFilter() != NULL) |
| 76 | fState = kDone; |
| 77 | else |
| 78 | { |
| 79 | fState = kBeforeEdge; |
| 80 | fPaint = paint; |
| 81 | fCanvas = canvas; |
| 82 | fSaveCount = canvas->getSaveCount(); |
| 83 | } |
| 84 | } |
| 85 | |
| 86 | bool SkBlurDrawLooper::next() |
| 87 | { |
| 88 | switch (fState) { |
| 89 | case kBeforeEdge: |
| 90 | fSavedColor = fPaint->getColor(); |
| 91 | fPaint->setColor(fBlurColor); |
| 92 | fPaint->setMaskFilter(fBlur); |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 93 | fPaint->setColorFilter(fColorFilter); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 94 | fCanvas->save(SkCanvas::kMatrix_SaveFlag); |
senorblanco@chromium.org | 038aff6 | 2010-12-06 23:45:58 +0000 | [diff] [blame] | 95 | if (fBlurFlags & kIgnoreTransform_BlurFlag) |
| 96 | { |
| 97 | SkMatrix transform(fCanvas->getTotalMatrix()); |
| 98 | transform.postTranslate(fDx, fDy); |
| 99 | fCanvas->setMatrix(transform); |
| 100 | } |
| 101 | else |
| 102 | { |
| 103 | fCanvas->translate(fDx, fDy); |
| 104 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 105 | fState = kAfterEdge; |
| 106 | return true; |
| 107 | case kAfterEdge: |
| 108 | fPaint->setColor(fSavedColor); |
| 109 | fPaint->setMaskFilter(NULL); |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 110 | fPaint->setColorFilter(NULL); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 111 | fCanvas->restore(); // to remove the translate we did earlier |
| 112 | fState = kDone; |
| 113 | return true; |
| 114 | default: |
| 115 | SkASSERT(kDone == fState); |
| 116 | return false; |
| 117 | } |
| 118 | } |
| 119 | |
| 120 | void SkBlurDrawLooper::restore() |
| 121 | { |
| 122 | if (kAfterEdge == fState) |
| 123 | { |
| 124 | fPaint->setColor(fSavedColor); |
| 125 | fPaint->setMaskFilter(NULL); |
senorblanco@chromium.org | 4868e6b | 2011-02-18 19:03:01 +0000 | [diff] [blame] | 126 | fPaint->setColorFilter(NULL); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 127 | fCanvas->restore(); // to remove the translate we did earlier |
| 128 | fState = kDone; |
| 129 | } |
| 130 | } |
| 131 | |
| 132 | /////////////////////////////////////////////////////////////////////////////// |
| 133 | |
| 134 | static SkFlattenable::Registrar gReg("SkBlurDrawLooper", |
| 135 | SkBlurDrawLooper::CreateProc); |
| 136 | |