| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright 2008 The Android Open Source Project | 2 * Copyright 2008 The Android Open Source Project |
| 3 * | 3 * |
| 4 * Use of this source code is governed by a BSD-style license that can be | 4 * Use of this source code is governed by a BSD-style license that can be |
| 5 * found in the LICENSE file. | 5 * found in the LICENSE file. |
| 6 */ | 6 */ |
| 7 | 7 |
| 8 #include "SkBitmapDevice.h" | 8 #include "SkBitmapDevice.h" |
| 9 #include "SkCanvas.h" | 9 #include "SkCanvas.h" |
| 10 #include "SkCanvasPriv.h" | 10 #include "SkCanvasPriv.h" |
| (...skipping 1149 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1160 SkCanvas c(dst); | 1160 SkCanvas c(dst); |
| 1161 | 1161 |
| 1162 SkAutoTUnref<SkImageFilter> localF(filter->newWithLocalMatrix(ctm)); | 1162 SkAutoTUnref<SkImageFilter> localF(filter->newWithLocalMatrix(ctm)); |
| 1163 SkPaint p; | 1163 SkPaint p; |
| 1164 p.setImageFilter(localF); | 1164 p.setImageFilter(localF); |
| 1165 const SkScalar x = SkIntToScalar(src->getOrigin().x()); | 1165 const SkScalar x = SkIntToScalar(src->getOrigin().x()); |
| 1166 const SkScalar y = SkIntToScalar(src->getOrigin().y()); | 1166 const SkScalar y = SkIntToScalar(src->getOrigin().y()); |
| 1167 c.drawBitmap(srcBM, x, y, &p); | 1167 c.drawBitmap(srcBM, x, y, &p); |
| 1168 } | 1168 } |
| 1169 | 1169 |
| 1170 static SkImageInfo make_layer_info(const SkImageInfo& prev, int w, int h, bool i
sOpaque, |
| 1171 const SkPaint* paint) { |
| 1172 // need to force L32 for now if we have an image filter. Once filters suppor
t other colortypes |
| 1173 // e.g. sRGB or F16, we can remove this check |
| 1174 const bool hasImageFilter = paint && paint->getImageFilter(); |
| 1175 |
| 1176 SkAlphaType alphaType = isOpaque ? kOpaque_SkAlphaType : kPremul_SkAlphaType
; |
| 1177 if ((prev.bytesPerPixel() < 4) || hasImageFilter) { |
| 1178 // force to L32 |
| 1179 return SkImageInfo::MakeN32(w, h, alphaType); |
| 1180 } else { |
| 1181 // keep the same characteristics as the prev |
| 1182 return SkImageInfo::Make(w, h, prev.colorType(), alphaType, prev.profile
Type()); |
| 1183 } |
| 1184 } |
| 1185 |
| 1170 void SkCanvas::internalSaveLayer(const SaveLayerRec& rec, SaveLayerStrategy stra
tegy) { | 1186 void SkCanvas::internalSaveLayer(const SaveLayerRec& rec, SaveLayerStrategy stra
tegy) { |
| 1171 const SkRect* bounds = rec.fBounds; | 1187 const SkRect* bounds = rec.fBounds; |
| 1172 const SkPaint* paint = rec.fPaint; | 1188 const SkPaint* paint = rec.fPaint; |
| 1173 SaveLayerFlags saveLayerFlags = rec.fSaveLayerFlags; | 1189 SaveLayerFlags saveLayerFlags = rec.fSaveLayerFlags; |
| 1174 | 1190 |
| 1175 #ifndef SK_SUPPORT_LEGACY_CLIPTOLAYERFLAG | 1191 #ifndef SK_SUPPORT_LEGACY_CLIPTOLAYERFLAG |
| 1176 saveLayerFlags &= ~kDontClipToLayer_PrivateSaveLayerFlag; | 1192 saveLayerFlags &= ~kDontClipToLayer_PrivateSaveLayerFlag; |
| 1177 #endif | 1193 #endif |
| 1178 | 1194 |
| 1179 // do this before we create the layer. We don't call the public save() since | 1195 // do this before we create the layer. We don't call the public save() since |
| (...skipping 15 matching lines...) Expand all Loading... |
| 1195 | 1211 |
| 1196 bool isOpaque = SkToBool(saveLayerFlags & kIsOpaque_SaveLayerFlag); | 1212 bool isOpaque = SkToBool(saveLayerFlags & kIsOpaque_SaveLayerFlag); |
| 1197 SkPixelGeometry geo = fProps.pixelGeometry(); | 1213 SkPixelGeometry geo = fProps.pixelGeometry(); |
| 1198 if (paint) { | 1214 if (paint) { |
| 1199 // TODO: perhaps add a query to filters so we might preserve opaqueness.
.. | 1215 // TODO: perhaps add a query to filters so we might preserve opaqueness.
.. |
| 1200 if (paint->getImageFilter() || paint->getColorFilter()) { | 1216 if (paint->getImageFilter() || paint->getColorFilter()) { |
| 1201 isOpaque = false; | 1217 isOpaque = false; |
| 1202 geo = kUnknown_SkPixelGeometry; | 1218 geo = kUnknown_SkPixelGeometry; |
| 1203 } | 1219 } |
| 1204 } | 1220 } |
| 1205 SkImageInfo info = SkImageInfo::MakeN32(ir.width(), ir.height(), | |
| 1206 isOpaque ? kOpaque_SkAlphaType : kPremul_SkAlphaType); | |
| 1207 | 1221 |
| 1208 SkBaseDevice* device = this->getTopDevice(); | 1222 SkBaseDevice* device = this->getTopDevice(); |
| 1209 if (nullptr == device) { | 1223 if (nullptr == device) { |
| 1210 SkDebugf("Unable to find device for layer."); | 1224 SkDebugf("Unable to find device for layer."); |
| 1211 return; | 1225 return; |
| 1212 } | 1226 } |
| 1213 | 1227 |
| 1228 SkImageInfo info = make_layer_info(device->imageInfo(), ir.width(), ir.heigh
t(), isOpaque, |
| 1229 paint); |
| 1230 |
| 1214 bool forceSpriteOnRestore = false; | 1231 bool forceSpriteOnRestore = false; |
| 1215 { | 1232 { |
| 1216 const bool preserveLCDText = kOpaque_SkAlphaType == info.alphaType() || | 1233 const bool preserveLCDText = kOpaque_SkAlphaType == info.alphaType() || |
| 1217 (saveLayerFlags & kPreserveLCDText_SaveLaye
rFlag); | 1234 (saveLayerFlags & kPreserveLCDText_SaveLaye
rFlag); |
| 1218 const SkBaseDevice::TileUsage usage = SkBaseDevice::kNever_TileUsage; | 1235 const SkBaseDevice::TileUsage usage = SkBaseDevice::kNever_TileUsage; |
| 1219 const SkBaseDevice::CreateInfo createInfo = SkBaseDevice::CreateInfo(inf
o, usage, geo, | 1236 const SkBaseDevice::CreateInfo createInfo = SkBaseDevice::CreateInfo(inf
o, usage, geo, |
| 1220 pres
erveLCDText, false); | 1237 pres
erveLCDText, false); |
| 1221 SkBaseDevice* newDev = device->onCreateDevice(createInfo, paint); | 1238 SkBaseDevice* newDev = device->onCreateDevice(createInfo, paint); |
| 1222 if (nullptr == newDev) { | 1239 if (nullptr == newDev) { |
| 1223 // If onCreateDevice didn't succeed, try raster (e.g. PDF couldn't h
andle the paint) | 1240 // If onCreateDevice didn't succeed, try raster (e.g. PDF couldn't h
andle the paint) |
| (...skipping 969 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2193 if (this->quickReject(tmp)) { | 2210 if (this->quickReject(tmp)) { |
| 2194 return; | 2211 return; |
| 2195 } | 2212 } |
| 2196 } | 2213 } |
| 2197 | 2214 |
| 2198 SkLazyPaint lazy; | 2215 SkLazyPaint lazy; |
| 2199 if (nullptr == paint) { | 2216 if (nullptr == paint) { |
| 2200 paint = lazy.init(); | 2217 paint = lazy.init(); |
| 2201 } | 2218 } |
| 2202 | 2219 |
| 2203 const bool drawAsSprite = this->canDrawBitmapAsSprite(x, y, image->width(),
image->height(), | 2220 bool drawAsSprite = this->canDrawBitmapAsSprite(x, y, image->width(), image-
>height(), |
| 2204 *paint); | 2221 *paint); |
| 2222 if (drawAsSprite && paint->getImageFilter()) { |
| 2223 SkBitmap bitmap; |
| 2224 if (!as_IB(image)->asBitmapForImageFilters(&bitmap)) { |
| 2225 drawAsSprite = false; |
| 2226 } else{ |
| 2227 // Until imagefilters are updated, they cannot handle any src type b
ut N32... |
| 2228 if (bitmap.info().colorType() != kN32_SkColorType || bitmap.info().i
sSRGB()) { |
| 2229 drawAsSprite = false; |
| 2230 } |
| 2231 } |
| 2232 } |
| 2233 |
| 2205 LOOPER_BEGIN_DRAWBITMAP(*paint, drawAsSprite, &bounds) | 2234 LOOPER_BEGIN_DRAWBITMAP(*paint, drawAsSprite, &bounds) |
| 2206 | 2235 |
| 2207 while (iter.next()) { | 2236 while (iter.next()) { |
| 2208 const SkPaint& pnt = looper.paint(); | 2237 const SkPaint& pnt = looper.paint(); |
| 2209 if (drawAsSprite && pnt.getImageFilter()) { | 2238 if (drawAsSprite && pnt.getImageFilter()) { |
| 2210 SkBitmap bitmap; | 2239 SkBitmap bitmap; |
| 2211 if (as_IB(image)->asBitmapForImageFilters(&bitmap)) { | 2240 if (as_IB(image)->asBitmapForImageFilters(&bitmap)) { |
| 2212 SkPoint pt; | 2241 SkPoint pt; |
| 2213 iter.fMatrix->mapXY(x, y, &pt); | 2242 iter.fMatrix->mapXY(x, y, &pt); |
| 2214 iter.fDevice->drawBitmapAsSprite(iter, bitmap, | 2243 iter.fDevice->drawBitmapAsSprite(iter, bitmap, |
| (...skipping 55 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2270 if (paint->canComputeFastBounds()) { | 2299 if (paint->canComputeFastBounds()) { |
| 2271 bitmap.getBounds(&storage); | 2300 bitmap.getBounds(&storage); |
| 2272 matrix.mapRect(&storage); | 2301 matrix.mapRect(&storage); |
| 2273 SkRect tmp = storage; | 2302 SkRect tmp = storage; |
| 2274 if (this->quickReject(paint->computeFastBounds(tmp, &tmp))) { | 2303 if (this->quickReject(paint->computeFastBounds(tmp, &tmp))) { |
| 2275 return; | 2304 return; |
| 2276 } | 2305 } |
| 2277 bounds = &storage; | 2306 bounds = &storage; |
| 2278 } | 2307 } |
| 2279 | 2308 |
| 2280 const bool drawAsSprite = bounds && this->canDrawBitmapAsSprite(x, y, bitmap
.width(), | 2309 bool drawAsSprite = bounds && this->canDrawBitmapAsSprite(x, y, bitmap.width
(), bitmap.height(), |
| 2281 bitmap.heigh
t(), *paint); | 2310 *paint); |
| 2311 if (drawAsSprite && paint->getImageFilter()) { |
| 2312 // Until imagefilters are updated, they cannot handle any src type but N
32... |
| 2313 if (bitmap.info().colorType() != kN32_SkColorType || bitmap.info().isSRG
B()) { |
| 2314 drawAsSprite = false; |
| 2315 } |
| 2316 } |
| 2317 |
| 2282 LOOPER_BEGIN_DRAWBITMAP(*paint, drawAsSprite, bounds) | 2318 LOOPER_BEGIN_DRAWBITMAP(*paint, drawAsSprite, bounds) |
| 2283 | 2319 |
| 2284 while (iter.next()) { | 2320 while (iter.next()) { |
| 2285 const SkPaint& pnt = looper.paint(); | 2321 const SkPaint& pnt = looper.paint(); |
| 2286 if (drawAsSprite && pnt.getImageFilter()) { | 2322 if (drawAsSprite && pnt.getImageFilter()) { |
| 2287 SkPoint pt; | 2323 SkPoint pt; |
| 2288 iter.fMatrix->mapXY(x, y, &pt); | 2324 iter.fMatrix->mapXY(x, y, &pt); |
| 2289 iter.fDevice->drawBitmapAsSprite(iter, bitmap, | 2325 iter.fDevice->drawBitmapAsSprite(iter, bitmap, |
| 2290 SkScalarRoundToInt(pt.fX), | 2326 SkScalarRoundToInt(pt.fX), |
| 2291 SkScalarRoundToInt(pt.fY), pnt); | 2327 SkScalarRoundToInt(pt.fY), pnt); |
| (...skipping 672 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2964 } | 3000 } |
| 2965 | 3001 |
| 2966 if (matrix) { | 3002 if (matrix) { |
| 2967 canvas->concat(*matrix); | 3003 canvas->concat(*matrix); |
| 2968 } | 3004 } |
| 2969 } | 3005 } |
| 2970 | 3006 |
| 2971 SkAutoCanvasMatrixPaint::~SkAutoCanvasMatrixPaint() { | 3007 SkAutoCanvasMatrixPaint::~SkAutoCanvasMatrixPaint() { |
| 2972 fCanvas->restoreToCount(fSaveCount); | 3008 fCanvas->restoreToCount(fSaveCount); |
| 2973 } | 3009 } |
| OLD | NEW |