Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 /* | 1 /* |
| 2 * Copyright 2013 Google Inc. | 2 * Copyright 2013 Google Inc. |
| 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 "SkAlphaThresholdFilter.h" | 8 #include "SkAlphaThresholdFilter.h" |
| 9 | 9 |
| 10 #include "SkBitmap.h" | 10 #include "SkBitmap.h" |
| 11 #include "SkDevice.h" | |
| 12 #include "SkReadBuffer.h" | 11 #include "SkReadBuffer.h" |
| 12 #include "SkSpecialImage.h" | |
| 13 #include "SkWriteBuffer.h" | 13 #include "SkWriteBuffer.h" |
| 14 #include "SkRegion.h" | 14 #include "SkRegion.h" |
| 15 #if SK_SUPPORT_GPU | 15 #if SK_SUPPORT_GPU |
| 16 #include "GrDrawContext.h" | 16 #include "GrDrawContext.h" |
| 17 #endif | 17 #endif |
| 18 | 18 |
| 19 class SK_API SkAlphaThresholdFilterImpl : public SkImageFilter { | 19 class SK_API SkAlphaThresholdFilterImpl : public SkImageFilter { |
| 20 public: | 20 public: |
| 21 SkAlphaThresholdFilterImpl(const SkRegion& region, SkScalar innerThreshold, | 21 SkAlphaThresholdFilterImpl(const SkRegion& region, SkScalar innerThreshold, |
| 22 SkScalar outerThreshold, sk_sp<SkImageFilter> inp ut); | 22 SkScalar outerThreshold, sk_sp<SkImageFilter> inp ut); |
| 23 | 23 |
| 24 SK_TO_STRING_OVERRIDE() | 24 SK_TO_STRING_OVERRIDE() |
| 25 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkAlphaThresholdFilterIm pl) | 25 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkAlphaThresholdFilterIm pl) |
| 26 friend void SkAlphaThresholdFilter::InitializeFlattenables(); | 26 friend void SkAlphaThresholdFilter::InitializeFlattenables(); |
| 27 | 27 |
| 28 protected: | 28 protected: |
| 29 void flatten(SkWriteBuffer&) const override; | 29 void flatten(SkWriteBuffer&) const override; |
| 30 | 30 |
| 31 bool onFilterImageDeprecated(Proxy*, const SkBitmap& src, const Context&, | 31 sk_sp<SkSpecialImage> onFilterImage(SkSpecialImage* source, const Context&, |
| 32 SkBitmap* result, SkIPoint* offset) const overr ide; | 32 SkIPoint* offset) const override; |
| 33 | |
| 33 #if SK_SUPPORT_GPU | 34 #if SK_SUPPORT_GPU |
| 34 bool asFragmentProcessor(GrFragmentProcessor**, GrTexture*, const SkMatrix&, | 35 GrTexture* createMaskTexture(GrContext*, const SkMatrix&, const SkIRect& bou nds) const; |
|
Stephen White
2016/04/11 20:48:20
Should this return an sk_sp<GrTexture> instead?
robertphillips
2016/04/12 15:36:39
Done.
| |
| 35 const SkIRect& bounds) const override; | |
| 36 #endif | 36 #endif |
| 37 | 37 |
| 38 private: | 38 private: |
| 39 SkRegion fRegion; | 39 SkRegion fRegion; |
| 40 SkScalar fInnerThreshold; | 40 SkScalar fInnerThreshold; |
| 41 SkScalar fOuterThreshold; | 41 SkScalar fOuterThreshold; |
| 42 typedef SkImageFilter INHERITED; | 42 typedef SkImageFilter INHERITED; |
| 43 }; | 43 }; |
| 44 | 44 |
| 45 SK_DEFINE_FLATTENABLE_REGISTRAR_GROUP_START(SkAlphaThresholdFilter) | 45 SK_DEFINE_FLATTENABLE_REGISTRAR_GROUP_START(SkAlphaThresholdFilter) |
| (...skipping 232 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 278 SkScalar innerThreshold, | 278 SkScalar innerThreshold, |
| 279 SkScalar outerThreshold, | 279 SkScalar outerThreshold, |
| 280 sk_sp<SkImageFilter> inpu t) | 280 sk_sp<SkImageFilter> inpu t) |
| 281 : INHERITED(&input, 1, nullptr) | 281 : INHERITED(&input, 1, nullptr) |
| 282 , fRegion(region) | 282 , fRegion(region) |
| 283 , fInnerThreshold(innerThreshold) | 283 , fInnerThreshold(innerThreshold) |
| 284 , fOuterThreshold(outerThreshold) { | 284 , fOuterThreshold(outerThreshold) { |
| 285 } | 285 } |
| 286 | 286 |
| 287 #if SK_SUPPORT_GPU | 287 #if SK_SUPPORT_GPU |
| 288 bool SkAlphaThresholdFilterImpl::asFragmentProcessor(GrFragmentProcessor** fp, | 288 GrTexture* SkAlphaThresholdFilterImpl::createMaskTexture(GrContext* context, |
| 289 GrTexture* texture, | 289 const SkMatrix& inMatri x, |
| 290 const SkMatrix& inMatrix, | 290 const SkIRect& bounds) const { |
| 291 const SkIRect& bounds) cons t { | 291 GrSurfaceDesc maskDesc; |
| 292 if (fp) { | 292 if (context->caps()->isConfigRenderable(kAlpha_8_GrPixelConfig, false)) { |
| 293 GrContext* context = texture->getContext(); | 293 maskDesc.fConfig = kAlpha_8_GrPixelConfig; |
| 294 GrSurfaceDesc maskDesc; | 294 } else { |
| 295 if (context->caps()->isConfigRenderable(kAlpha_8_GrPixelConfig, false)) { | 295 maskDesc.fConfig = kRGBA_8888_GrPixelConfig; |
| 296 maskDesc.fConfig = kAlpha_8_GrPixelConfig; | 296 } |
| 297 } else { | 297 maskDesc.fFlags = kRenderTarget_GrSurfaceFlag; |
| 298 maskDesc.fConfig = kRGBA_8888_GrPixelConfig; | 298 // Add one pixel of border to ensure that clamp mode will be all zeros |
| 299 } | 299 // the outside. |
| 300 maskDesc.fFlags = kRenderTarget_GrSurfaceFlag; | 300 maskDesc.fWidth = bounds.width(); |
| 301 // Add one pixel of border to ensure that clamp mode will be all zeros | 301 maskDesc.fHeight = bounds.height(); |
| 302 // the outside. | 302 SkAutoTUnref<GrTexture> maskTexture(context->textureProvider()->createApprox Texture(maskDesc)); |
| 303 maskDesc.fWidth = bounds.width(); | 303 if (!maskTexture) { |
| 304 maskDesc.fHeight = bounds.height(); | 304 return nullptr; |
| 305 SkAutoTUnref<GrTexture> maskTexture( | 305 } |
| 306 context->textureProvider()->createApproxTexture(maskDesc)); | |
| 307 if (!maskTexture) { | |
| 308 return false; | |
| 309 } | |
| 310 | 306 |
| 311 SkAutoTUnref<GrDrawContext> drawContext( | 307 SkAutoTUnref<GrDrawContext> drawContext(context->drawContext(maskTexture->as RenderTarget())); |
| 312 context->drawContext(maskTexture->as RenderTarget())); | 308 if (!drawContext) { |
| 313 if (drawContext) { | 309 return nullptr; |
| 314 GrPaint grPaint; | 310 } |
| 315 grPaint.setPorterDuffXPFactory(SkXfermode::kSrc_Mode); | |
| 316 SkRegion::Iterator iter(fRegion); | |
| 317 drawContext->clear(nullptr, 0x0, true); | |
| 318 | 311 |
| 319 GrClip clip(SkRect::Make(SkIRect::MakeWH(bounds.width(), bounds.heig ht()))); | 312 GrPaint grPaint; |
| 320 while (!iter.done()) { | 313 grPaint.setPorterDuffXPFactory(SkXfermode::kSrc_Mode); |
| 321 SkRect rect = SkRect::Make(iter.rect()); | 314 SkRegion::Iterator iter(fRegion); |
| 322 drawContext->drawRect(clip, grPaint, inMatrix, rect); | 315 drawContext->clear(nullptr, 0x0, true); |
|
Stephen White
2016/04/11 20:48:20
Add a reference to the "clear may not be necessary
robertphillips
2016/04/12 15:36:39
This one is necessary since we don't know where th
Stephen White
2016/04/12 15:41:40
Good point. Perhaps we could do a pass unioning al
| |
| 323 iter.next(); | |
| 324 } | |
| 325 } | |
| 326 | 316 |
| 327 *fp = AlphaThresholdEffect::Create(texture, | 317 GrClip clip(SkRect::Make(SkIRect::MakeWH(bounds.width(), bounds.height()))); |
| 328 maskTexture, | 318 while (!iter.done()) { |
| 329 fInnerThreshold, | 319 SkRect rect = SkRect::Make(iter.rect()); |
| 330 fOuterThreshold, | 320 drawContext->drawRect(clip, grPaint, inMatrix, rect); |
| 331 bounds); | 321 iter.next(); |
| 332 } | 322 } |
| 333 return true; | 323 |
| 324 return maskTexture.release(); | |
| 334 } | 325 } |
| 335 #endif | 326 #endif |
| 336 | 327 |
| 337 void SkAlphaThresholdFilterImpl::flatten(SkWriteBuffer& buffer) const { | 328 void SkAlphaThresholdFilterImpl::flatten(SkWriteBuffer& buffer) const { |
| 338 this->INHERITED::flatten(buffer); | 329 this->INHERITED::flatten(buffer); |
| 339 buffer.writeScalar(fInnerThreshold); | 330 buffer.writeScalar(fInnerThreshold); |
| 340 buffer.writeScalar(fOuterThreshold); | 331 buffer.writeScalar(fOuterThreshold); |
| 341 buffer.writeRegion(fRegion); | 332 buffer.writeRegion(fRegion); |
| 342 } | 333 } |
| 343 | 334 |
| 344 bool SkAlphaThresholdFilterImpl::onFilterImageDeprecated(Proxy* proxy, const SkB itmap& src, | 335 sk_sp<SkSpecialImage> SkAlphaThresholdFilterImpl::onFilterImage(SkSpecialImage* source, |
| 345 const Context& ctx, SkB itmap* dst, | 336 const Context& c tx, |
| 346 SkIPoint* offset) const { | 337 SkIPoint* offset ) const { |
| 338 SkIPoint inputOffset = SkIPoint::Make(0, 0); | |
| 339 sk_sp<SkSpecialImage> input(this->filterInput(0, source, ctx, &inputOffset)) ; | |
| 340 if (!input) { | |
| 341 return nullptr; | |
| 342 } | |
| 347 | 343 |
| 348 if (src.colorType() != kN32_SkColorType) { | 344 const SkIRect inputBounds = SkIRect::MakeXYWH(inputOffset.x(), inputOffset.y (), |
| 349 return false; | 345 input->width(), input->height( )); |
| 346 | |
| 347 SkIRect bounds; | |
| 348 if (!this->applyCropRect(ctx, inputBounds, &bounds)) { | |
|
Stephen White
2016/04/11 20:48:20
Given that there's no create param to add a crop r
robertphillips
2016/04/12 15:36:39
applyCropRect also does interesting things with th
Stephen White
2016/04/12 15:41:39
Thanks!
| |
| 349 return nullptr; | |
| 350 } | 350 } |
| 351 | 351 |
| 352 #if SK_SUPPORT_GPU | |
| 353 if (source->isTextureBacked()) { | |
| 354 GrContext* context = source->getContext(); | |
| 355 | |
| 356 sk_sp<GrTexture> inputTexture(input->asTextureRef(context)); | |
| 357 if (!inputTexture) { | |
|
Stephen White
2016/04/11 20:48:20
Please assert this is non-null instead.
robertphillips
2016/04/12 15:36:39
Done.
| |
| 358 return nullptr; | |
| 359 } | |
| 360 | |
| 361 offset->fX = bounds.left(); | |
| 362 offset->fY = bounds.top(); | |
| 363 | |
| 364 bounds.offset(-inputOffset); | |
| 365 | |
| 366 SkMatrix matrix(ctx.ctm()); | |
| 367 matrix.postTranslate(SkIntToScalar(-bounds.left()), SkIntToScalar(-bound s.top())); | |
| 368 | |
| 369 sk_sp<GrTexture> maskTexture(this->createMaskTexture(context, matrix, bo unds)); | |
| 370 if (!maskTexture) { | |
| 371 return nullptr; | |
| 372 } | |
| 373 | |
| 374 // SRGBTODO: handle sRGB here | |
|
Stephen White
2016/04/11 20:48:20
Is there a bug for that?
robertphillips
2016/04/12 15:36:39
This is Brian O.'s TODO. I believe he is just trac
Stephen White
2016/04/12 15:41:39
Acknowledged.
| |
| 375 sk_sp<GrFragmentProcessor> fp(AlphaThresholdEffect::Create(inputTexture. get(), | |
| 376 maskTexture.g et(), | |
| 377 fInnerThresho ld, | |
| 378 fOuterThresho ld, | |
| 379 bounds)); | |
| 380 if (!fp) { | |
| 381 return nullptr; | |
| 382 } | |
| 383 | |
| 384 GrPaint paint; | |
| 385 paint.addColorFragmentProcessor(fp.get()); | |
| 386 paint.setPorterDuffXPFactory(SkXfermode::kSrc_Mode); | |
| 387 | |
| 388 GrSurfaceDesc desc; | |
| 389 desc.fFlags = kRenderTarget_GrSurfaceFlag; | |
| 390 desc.fWidth = bounds.width(); | |
| 391 desc.fHeight = bounds.height(); | |
| 392 desc.fConfig = kRGBA_8888_GrPixelConfig; | |
| 393 | |
| 394 SkAutoTUnref<GrTexture> dst(context->textureProvider()->createApproxText ure(desc)); | |
| 395 if (!dst) { | |
| 396 return nullptr; | |
| 397 } | |
| 398 | |
| 399 SkAutoTUnref<GrDrawContext> drawContext(context->drawContext(dst->asRend erTarget())); | |
| 400 if (!drawContext) { | |
| 401 return nullptr; | |
| 402 } | |
| 403 | |
| 404 SkRect srcRect = SkRect::Make(bounds); | |
| 405 SkRect dstRect = SkRect::MakeWH(srcRect.width(), srcRect.height()); | |
| 406 GrClip clip(dstRect); | |
| 407 drawContext->fillRectToRect(clip, paint, SkMatrix::I(), dstRect, srcRect ); | |
| 408 | |
| 409 return SkSpecialImage::MakeFromGpu(source->internal_getProxy(), | |
| 410 SkIRect::MakeWH(bounds.width(), bound s.height()), | |
| 411 kNeedNewImageUniqueID_SpecialImage, | |
| 412 dst); | |
|
Stephen White
2016/04/11 20:48:20
Kind of a shame the GPU code got so much longer in
robertphillips
2016/04/12 15:36:39
I've added a SkImageFilter::DrawWithFP helper func
Stephen White
2016/04/12 15:41:39
Acknowledged.
| |
| 413 } | |
| 414 #endif | |
| 415 | |
| 416 SkBitmap inputBM; | |
| 417 | |
| 418 if (!input->getROPixels(&inputBM)) { | |
| 419 return nullptr; | |
| 420 } | |
| 421 | |
| 422 if (inputBM.colorType() != kN32_SkColorType) { | |
| 423 return nullptr; | |
| 424 } | |
| 425 | |
| 426 SkAutoLockPixels inputLock(inputBM); | |
| 427 | |
| 428 if (!inputBM.getPixels() || inputBM.width() <= 0 || inputBM.height() <= 0) { | |
| 429 return nullptr; | |
| 430 } | |
| 431 | |
| 432 | |
| 352 SkMatrix localInverse; | 433 SkMatrix localInverse; |
| 353 if (!ctx.ctm().invert(&localInverse)) { | 434 if (!ctx.ctm().invert(&localInverse)) { |
| 354 return false; | 435 return nullptr; |
| 355 } | 436 } |
| 356 | 437 |
| 357 SkAutoLockPixels alp(src); | 438 SkImageInfo info = SkImageInfo::MakeN32(bounds.width(), bounds.height(), |
| 358 SkASSERT(src.getPixels()); | 439 inputBM.alphaType()); |
| 359 if (!src.getPixels() || src.width() <= 0 || src.height() <= 0) { | 440 |
| 360 return false; | 441 SkBitmap dst; |
| 442 if (!dst.tryAllocPixels(info)) { | |
| 443 return nullptr; | |
| 361 } | 444 } |
| 362 | 445 |
| 363 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(src.width(), src.heigh t())); | 446 SkAutoLockPixels dstLock(dst); |
| 364 if (!device) { | |
| 365 return false; | |
| 366 } | |
| 367 *dst = device->accessBitmap(false); | |
| 368 SkAutoLockPixels alp_dst(*dst); | |
| 369 | 447 |
| 370 U8CPU innerThreshold = (U8CPU)(fInnerThreshold * 0xFF); | 448 U8CPU innerThreshold = (U8CPU)(fInnerThreshold * 0xFF); |
| 371 U8CPU outerThreshold = (U8CPU)(fOuterThreshold * 0xFF); | 449 U8CPU outerThreshold = (U8CPU)(fOuterThreshold * 0xFF); |
| 372 SkColor* sptr = src.getAddr32(0, 0); | 450 SkColor* dptr = dst.getAddr32(0, 0); |
| 373 SkColor* dptr = dst->getAddr32(0, 0); | 451 int dstWidth = dst.width(), dstHeight = dst.height(); |
| 374 int width = src.width(), height = src.height(); | 452 for (int y = 0; y < dstHeight; ++y) { |
| 375 for (int y = 0; y < height; ++y) { | 453 const SkColor* sptr = inputBM.getAddr32(bounds.fLeft, bounds.fTop+y); |
| 376 for (int x = 0; x < width; ++x) { | 454 |
| 377 const SkColor& source = sptr[y * width + x]; | 455 for (int x = 0; x < dstWidth; ++x) { |
| 378 SkColor output_color(source); | 456 const SkColor& source = sptr[x]; |
| 457 SkColor outputColor(source); | |
| 379 SkPoint position; | 458 SkPoint position; |
| 380 localInverse.mapXY((SkScalar)x, (SkScalar)y, &position); | 459 localInverse.mapXY((SkScalar)x + bounds.fLeft, (SkScalar)y + bounds. fTop, &position); |
| 381 if (fRegion.contains((int32_t)position.x(), (int32_t)position.y())) { | 460 if (fRegion.contains((int32_t)position.x(), (int32_t)position.y())) { |
| 382 if (SkColorGetA(source) < innerThreshold) { | 461 if (SkColorGetA(source) < innerThreshold) { |
| 383 U8CPU alpha = SkColorGetA(source); | 462 U8CPU alpha = SkColorGetA(source); |
| 384 if (alpha == 0) | 463 if (alpha == 0) { |
| 385 alpha = 1; | 464 alpha = 1; |
| 465 } | |
| 386 float scale = (float)innerThreshold / alpha; | 466 float scale = (float)innerThreshold / alpha; |
| 387 output_color = SkColorSetARGB(innerThreshold, | 467 outputColor = SkColorSetARGB(innerThreshold, |
| 388 (U8CPU)(SkColorGetR(source) * scale), | 468 (U8CPU)(SkColorGetR(source) * scale), |
| 389 (U8CPU)(SkColorGetG(source) * scale), | 469 (U8CPU)(SkColorGetG(source) * scale), |
| 390 (U8CPU)(SkColorGetB(source) * scale)); | 470 (U8CPU)(SkColorGetB(source) * scale)); |
| 391 } | 471 } |
| 392 } else { | 472 } else { |
| 393 if (SkColorGetA(source) > outerThreshold) { | 473 if (SkColorGetA(source) > outerThreshold) { |
| 394 float scale = (float)outerThreshold / SkColorGetA(source); | 474 float scale = (float)outerThreshold / SkColorGetA(source); |
| 395 output_color = SkColorSetARGB(outerThreshold, | 475 outputColor = SkColorSetARGB(outerThreshold, |
| 396 (U8CPU)(SkColorGetR(source) * scale), | 476 (U8CPU)(SkColorGetR(source) * scale), |
| 397 (U8CPU)(SkColorGetG(source) * scale), | 477 (U8CPU)(SkColorGetG(source) * scale), |
| 398 (U8CPU)(SkColorGetB(source) * scale)); | 478 (U8CPU)(SkColorGetB(source) * scale)); |
| 399 } | 479 } |
| 400 } | 480 } |
| 401 dptr[y * dst->width() + x] = output_color; | 481 dptr[y * dstWidth + x] = outputColor; |
| 402 } | 482 } |
| 403 } | 483 } |
| 404 | 484 |
| 405 return true; | 485 offset->fX = bounds.left(); |
| 486 offset->fY = bounds.top(); | |
| 487 return SkSpecialImage::MakeFromRaster(source->internal_getProxy(), | |
| 488 SkIRect::MakeWH(bounds.width(), bounds .height()), | |
| 489 dst); | |
| 406 } | 490 } |
| 407 | 491 |
| 408 #ifndef SK_IGNORE_TO_STRING | 492 #ifndef SK_IGNORE_TO_STRING |
| 409 void SkAlphaThresholdFilterImpl::toString(SkString* str) const { | 493 void SkAlphaThresholdFilterImpl::toString(SkString* str) const { |
| 410 str->appendf("SkAlphaThresholdImageFilter: ("); | 494 str->appendf("SkAlphaThresholdImageFilter: ("); |
| 411 str->appendf("inner: %f outer: %f", fInnerThreshold, fOuterThreshold); | 495 str->appendf("inner: %f outer: %f", fInnerThreshold, fOuterThreshold); |
| 412 str->append(")"); | 496 str->append(")"); |
| 413 } | 497 } |
| 414 #endif | 498 #endif |
| OLD | NEW |