Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 /* | 1 /* |
| 2 * Copyright (C) 2004, 2006, 2007 Apple Inc. All rights reserved. | 2 * Copyright (C) 2004, 2006, 2007 Apple Inc. All rights reserved. |
| 3 * Copyright (C) 2007 Alp Toker <alp@atoker.com> | 3 * Copyright (C) 2007 Alp Toker <alp@atoker.com> |
| 4 * Copyright (C) 2010 Torch Mobile (Beijing) Co. Ltd. All rights reserved. | 4 * Copyright (C) 2010 Torch Mobile (Beijing) Co. Ltd. All rights reserved. |
| 5 * | 5 * |
| 6 * Redistribution and use in source and binary forms, with or without | 6 * Redistribution and use in source and binary forms, with or without |
| 7 * modification, are permitted provided that the following conditions | 7 * modification, are permitted provided that the following conditions |
| 8 * are met: | 8 * are met: |
| 9 * 1. Redistributions of source code must retain the above copyright | 9 * 1. Redistributions of source code must retain the above copyright |
| 10 * notice, this list of conditions and the following disclaimer. | 10 * notice, this list of conditions and the following disclaimer. |
| (...skipping 265 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 276 clearCopiedImage(); | 276 clearCopiedImage(); |
| 277 if (layoutObject()) | 277 if (layoutObject()) |
| 278 layoutObject()->setMayNeedPaintInvalidation(); | 278 layoutObject()->setMayNeedPaintInvalidation(); |
| 279 m_dirtyRect.unite(rect); | 279 m_dirtyRect.unite(rect); |
| 280 if (m_context && m_context->is2d() && hasImageBuffer()) | 280 if (m_context && m_context->is2d() && hasImageBuffer()) |
| 281 buffer()->didDraw(rect); | 281 buffer()->didDraw(rect); |
| 282 } | 282 } |
| 283 | 283 |
| 284 void HTMLCanvasElement::didFinalizeFrame() | 284 void HTMLCanvasElement::didFinalizeFrame() |
| 285 { | 285 { |
| 286 notifyListenersCanvasChanged(); | |
| 287 | |
| 286 if (m_dirtyRect.isEmpty()) | 288 if (m_dirtyRect.isEmpty()) |
| 287 return; | 289 return; |
| 288 | 290 |
| 289 // Propagate the m_dirtyRect accumulated so far to the compositor | 291 // Propagate the m_dirtyRect accumulated so far to the compositor |
| 290 // before restarting with a blank dirty rect. | 292 // before restarting with a blank dirty rect. |
| 291 FloatRect srcRect(0, 0, size().width(), size().height()); | 293 FloatRect srcRect(0, 0, size().width(), size().height()); |
| 292 m_dirtyRect.intersect(srcRect); | 294 m_dirtyRect.intersect(srcRect); |
| 293 LayoutBox* ro = layoutBox(); | 295 LayoutBox* ro = layoutBox(); |
| 294 // Canvas content updates do not need to be propagated as | 296 // Canvas content updates do not need to be propagated as |
| 295 // paint invalidations if the canvas is accelerated, since | 297 // paint invalidations if the canvas is accelerated, since |
| (...skipping 90 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 386 } | 388 } |
| 387 } | 389 } |
| 388 } | 390 } |
| 389 | 391 |
| 390 bool HTMLCanvasElement::paintsIntoCanvasBuffer() const | 392 bool HTMLCanvasElement::paintsIntoCanvasBuffer() const |
| 391 { | 393 { |
| 392 ASSERT(m_context); | 394 ASSERT(m_context); |
| 393 | 395 |
| 394 if (!m_context->isAccelerated()) | 396 if (!m_context->isAccelerated()) |
| 395 return true; | 397 return true; |
| 396 | |
| 397 if (layoutBox() && layoutBox()->hasAcceleratedCompositing()) | 398 if (layoutBox() && layoutBox()->hasAcceleratedCompositing()) |
| 398 return false; | 399 return false; |
| 399 | 400 |
| 400 return true; | 401 return true; |
| 401 } | 402 } |
| 402 | 403 |
| 404 void HTMLCanvasElement::notifyListenersCanvasChanged() | |
| 405 { | |
| 406 if (!originClean()) { | |
| 407 m_listeners.clear(); | |
| 408 return; | |
| 409 } | |
| 410 | |
| 411 for (CanvasDrawListener* listener : m_listeners) { | |
| 412 if (listener->needsNewFrameCapture()) { | |
| 413 SourceImageStatus status; | |
| 414 RefPtr<Image> sourceImage = getSourceImageForCanvas(&status, PreferN oAcceleration); | |
|
esprehn
2015/12/01 20:04:26
getSourceImageForCanvas is expensive I think it se
emircan
2015/12/01 21:07:34
Done. My bad, I was initially designing it with a
| |
| 415 if (status != NormalSourceImageStatus) | |
| 416 return; | |
| 417 listener->setNewFrameCapture(sourceImage->imageForCurrentFrame()); | |
| 418 } | |
| 419 } | |
| 420 } | |
| 421 | |
| 403 void HTMLCanvasElement::paint(GraphicsContext* context, const LayoutRect& r) | 422 void HTMLCanvasElement::paint(GraphicsContext* context, const LayoutRect& r) |
| 404 { | 423 { |
| 405 // FIXME: crbug.com/438240; there is a bug with the new CSS blending and com positing feature. | 424 // FIXME: crbug.com/438240; there is a bug with the new CSS blending and com positing feature. |
| 406 if (!m_context) | 425 if (!m_context) |
| 407 return; | 426 return; |
| 408 if (!paintsIntoCanvasBuffer() && !document().printing()) | 427 if (!paintsIntoCanvasBuffer() && !document().printing()) |
| 409 return; | 428 return; |
| 410 | 429 |
| 411 m_context->paintRenderingResultsToCanvas(FrontBuffer); | 430 m_context->paintRenderingResultsToCanvas(FrontBuffer); |
| 412 if (hasImageBuffer()) { | 431 if (hasImageBuffer()) { |
| (...skipping 148 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 561 RefPtr<DOMUint8ClampedArray> imageDataRef(imageData->data()); | 580 RefPtr<DOMUint8ClampedArray> imageDataRef(imageData->data()); |
| 562 | 581 |
| 563 RefPtr<CanvasAsyncBlobCreator> asyncCreatorRef = CanvasAsyncBlobCreator::cre ate(imageDataRef.release(), encodingMimeType, imageData->size(), callback); | 582 RefPtr<CanvasAsyncBlobCreator> asyncCreatorRef = CanvasAsyncBlobCreator::cre ate(imageDataRef.release(), encodingMimeType, imageData->size(), callback); |
| 564 if (Platform::current()->isThreadedCompositingEnabled() && (encodingMimeType == DefaultMimeType)) { | 583 if (Platform::current()->isThreadedCompositingEnabled() && (encodingMimeType == DefaultMimeType)) { |
| 565 asyncCreatorRef->scheduleAsyncBlobCreation(true); | 584 asyncCreatorRef->scheduleAsyncBlobCreation(true); |
| 566 } else { | 585 } else { |
| 567 asyncCreatorRef->scheduleAsyncBlobCreation(false, quality); | 586 asyncCreatorRef->scheduleAsyncBlobCreation(false, quality); |
| 568 } | 587 } |
| 569 } | 588 } |
| 570 | 589 |
| 590 void HTMLCanvasElement::addListener(CanvasDrawListener* listener) | |
| 591 { | |
| 592 m_listeners.add(listener); | |
| 593 notifyListenersCanvasChanged(); | |
|
esprehn
2015/12/01 20:04:26
hmm, this means we do a sync readback the first ti
emircan
2015/12/01 21:07:34
I see that frame might not be completed. I was try
| |
| 594 } | |
| 595 | |
| 596 void HTMLCanvasElement::removeListener(CanvasDrawListener* listener) | |
| 597 { | |
| 598 m_listeners.remove(listener); | |
| 599 } | |
| 600 | |
| 571 SecurityOrigin* HTMLCanvasElement::securityOrigin() const | 601 SecurityOrigin* HTMLCanvasElement::securityOrigin() const |
| 572 { | 602 { |
| 573 return document().securityOrigin(); | 603 return document().securityOrigin(); |
| 574 } | 604 } |
| 575 | 605 |
| 576 bool HTMLCanvasElement::originClean() const | 606 bool HTMLCanvasElement::originClean() const |
| 577 { | 607 { |
| 578 if (document().settings() && document().settings()->disableReadingFromCanvas ()) | 608 if (document().settings() && document().settings()->disableReadingFromCanvas ()) |
| 579 return false; | 609 return false; |
| 580 return m_originClean; | 610 return m_originClean; |
| (...skipping 155 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 736 } | 766 } |
| 737 | 767 |
| 738 void HTMLCanvasElement::notifySurfaceInvalid() | 768 void HTMLCanvasElement::notifySurfaceInvalid() |
| 739 { | 769 { |
| 740 if (m_context && m_context->is2d()) | 770 if (m_context && m_context->is2d()) |
| 741 m_context->loseContext(CanvasRenderingContext::RealLostContext); | 771 m_context->loseContext(CanvasRenderingContext::RealLostContext); |
| 742 } | 772 } |
| 743 | 773 |
| 744 DEFINE_TRACE(HTMLCanvasElement) | 774 DEFINE_TRACE(HTMLCanvasElement) |
| 745 { | 775 { |
| 776 visitor->trace(m_listeners); | |
| 746 visitor->trace(m_context); | 777 visitor->trace(m_context); |
| 747 DocumentVisibilityObserver::trace(visitor); | 778 DocumentVisibilityObserver::trace(visitor); |
| 748 HTMLElement::trace(visitor); | 779 HTMLElement::trace(visitor); |
| 749 } | 780 } |
| 750 | 781 |
| 751 void HTMLCanvasElement::updateExternallyAllocatedMemory() const | 782 void HTMLCanvasElement::updateExternallyAllocatedMemory() const |
| 752 { | 783 { |
| 753 int bufferCount = 0; | 784 int bufferCount = 0; |
| 754 if (m_imageBuffer) { | 785 if (m_imageBuffer) { |
| 755 bufferCount++; | 786 bufferCount++; |
| (...skipping 196 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 952 } | 983 } |
| 953 return ImageBitmapSource::fulfillImageBitmap(scriptState, isPaintable() ? Im ageBitmap::create(this, IntRect(sx, sy, sw, sh)) : nullptr); | 984 return ImageBitmapSource::fulfillImageBitmap(scriptState, isPaintable() ? Im ageBitmap::create(this, IntRect(sx, sy, sw, sh)) : nullptr); |
| 954 } | 985 } |
| 955 | 986 |
| 956 bool HTMLCanvasElement::isOpaque() const | 987 bool HTMLCanvasElement::isOpaque() const |
| 957 { | 988 { |
| 958 return m_context && !m_context->hasAlpha(); | 989 return m_context && !m_context->hasAlpha(); |
| 959 } | 990 } |
| 960 | 991 |
| 961 } // blink | 992 } // blink |
| OLD | NEW |