| Index: third_party/WebKit/Source/core/frame/ImageBitmap.cpp
|
| diff --git a/third_party/WebKit/Source/core/frame/ImageBitmap.cpp b/third_party/WebKit/Source/core/frame/ImageBitmap.cpp
|
| index 239edcf652a50e00609b5d112ae7a56b9554e258..c7348a846eba46ffd10fd41e291155fc90e3dc5f 100644
|
| --- a/third_party/WebKit/Source/core/frame/ImageBitmap.cpp
|
| +++ b/third_party/WebKit/Source/core/frame/ImageBitmap.cpp
|
| @@ -12,6 +12,8 @@
|
|
|
| namespace blink {
|
|
|
| +static const char* imageOrientationFlipY = "flipY";
|
| +
|
| static inline IntRect normalizeRect(const IntRect& rect)
|
| {
|
| return IntRect(std::min(rect.x(), rect.maxX()),
|
| @@ -20,15 +22,37 @@ static inline IntRect normalizeRect(const IntRect& rect)
|
| std::max(rect.height(), -rect.height()));
|
| }
|
|
|
| -static PassRefPtr<StaticBitmapImage> cropImage(Image* image, const IntRect& cropRect)
|
| +// TODO(xidachen): this function needs to be changed later on when implementing premultiplyAlpha option
|
| +static SkImage* flipSkImageVertically(SkImage* input)
|
| +{
|
| + int width = input->width();
|
| + int height = input->height();
|
| + SkImageInfo info = SkImageInfo::MakeN32Premul(width, height);
|
| + OwnPtr<uint8_t[]> imagePixels = adoptArrayPtr(new uint8_t[width * height * info.bytesPerPixel()]);
|
| + int imageRowBytes = info.bytesPerPixel() * width;
|
| + input->readPixels(info, imagePixels.get(), imageRowBytes, 0, 0);
|
| +
|
| + for (int i = 0; i < height / 2; i++) {
|
| + int topFirstElement = i * imageRowBytes;
|
| + int topLastElement = (i + 1) * imageRowBytes;
|
| + int bottomFirstElement = (height - 1 - i) * imageRowBytes;
|
| + std::swap_ranges(imagePixels.get() + topFirstElement, imagePixels.get() + topLastElement, imagePixels.get() + bottomFirstElement);
|
| + }
|
| + return SkImage::NewRasterCopy(info, imagePixels.get(), imageRowBytes);
|
| +}
|
| +
|
| +static PassRefPtr<StaticBitmapImage> cropImage(Image* image, const IntRect& cropRect, bool flipYEnabled)
|
| {
|
| ASSERT(image);
|
|
|
| IntRect imgRect(IntPoint(), IntSize(image->width(), image->height()));
|
| const IntRect srcRect = intersection(imgRect, cropRect);
|
|
|
| - if (cropRect == srcRect)
|
| + if (cropRect == srcRect) {
|
| + if (flipYEnabled)
|
| + return StaticBitmapImage::create(adoptRef(flipSkImageVertically(image->imageForCurrentFrame()->newSubset(srcRect))));
|
| return StaticBitmapImage::create(adoptRef(image->imageForCurrentFrame()->newSubset(srcRect)));
|
| + }
|
|
|
| RefPtr<SkSurface> surface = adoptRef(SkSurface::NewRasterN32Premul(cropRect.width(), cropRect.height()));
|
|
|
| @@ -42,12 +66,17 @@ static PassRefPtr<StaticBitmapImage> cropImage(Image* image, const IntRect& crop
|
| if (cropRect.y() < 0)
|
| dstTop = -cropRect.y();
|
| surface->getCanvas()->drawImage(image->imageForCurrentFrame().get(), dstLeft, dstTop);
|
| + if (flipYEnabled)
|
| + return StaticBitmapImage::create(adoptRef(flipSkImageVertically(surface->newImageSnapshot())));
|
| return StaticBitmapImage::create(adoptRef(surface->newImageSnapshot()));
|
| }
|
|
|
| ImageBitmap::ImageBitmap(HTMLImageElement* image, const IntRect& cropRect, Document* document, const ImageBitmapOptions& options)
|
| {
|
| - m_image = cropImage(image->cachedImage()->image(), cropRect);
|
| + if (options.imageOrientation() == imageOrientationFlipY)
|
| + m_image = cropImage(image->cachedImage()->image(), cropRect, true);
|
| + else
|
| + m_image = cropImage(image->cachedImage()->image(), cropRect, false);
|
| m_image->setOriginClean(!image->wouldTaintOrigin(document->securityOrigin()));
|
| }
|
|
|
| @@ -65,14 +94,20 @@ ImageBitmap::ImageBitmap(HTMLVideoElement* video, const IntRect& cropRect, Docum
|
|
|
| IntPoint dstPoint = IntPoint(std::max(0, -cropRect.x()), std::max(0, -cropRect.y()));
|
| video->paintCurrentFrame(buffer->canvas(), IntRect(dstPoint, srcRect.size()), nullptr);
|
| - m_image = StaticBitmapImage::create(buffer->newSkImageSnapshot(PreferNoAcceleration));
|
| + if (options.imageOrientation() == imageOrientationFlipY)
|
| + m_image = StaticBitmapImage::create(adoptRef(flipSkImageVertically(buffer->newSkImageSnapshot(PreferNoAcceleration).get())));
|
| + else
|
| + m_image = StaticBitmapImage::create(buffer->newSkImageSnapshot(PreferNoAcceleration));
|
| m_image->setOriginClean(!video->wouldTaintOrigin(document->securityOrigin()));
|
| }
|
|
|
| ImageBitmap::ImageBitmap(HTMLCanvasElement* canvas, const IntRect& cropRect, const ImageBitmapOptions& options)
|
| {
|
| ASSERT(canvas->isPaintable());
|
| - m_image = cropImage(canvas->copiedImage(BackBuffer, PreferAcceleration).get(), cropRect);
|
| + if (options.imageOrientation() == imageOrientationFlipY)
|
| + m_image = cropImage(canvas->copiedImage(BackBuffer, PreferAcceleration).get(), cropRect, true);
|
| + else
|
| + m_image = cropImage(canvas->copiedImage(BackBuffer, PreferAcceleration).get(), cropRect, false);
|
| m_image->setOriginClean(canvas->originClean());
|
| }
|
|
|
| @@ -95,18 +130,27 @@ ImageBitmap::ImageBitmap(ImageData* data, const IntRect& cropRect, const ImageBi
|
| if (cropRect.y() < 0)
|
| dstPoint.setY(-cropRect.y());
|
| buffer->putByteArray(Unmultiplied, data->data()->data(), data->size(), srcRect, dstPoint);
|
| - m_image = StaticBitmapImage::create(buffer->newSkImageSnapshot(PreferNoAcceleration));
|
| + if (options.imageOrientation() == imageOrientationFlipY)
|
| + m_image = StaticBitmapImage::create(adoptRef(flipSkImageVertically(buffer->newSkImageSnapshot(PreferNoAcceleration).get())));
|
| + else
|
| + m_image = StaticBitmapImage::create(buffer->newSkImageSnapshot(PreferNoAcceleration));
|
| }
|
|
|
| ImageBitmap::ImageBitmap(ImageBitmap* bitmap, const IntRect& cropRect, const ImageBitmapOptions& options)
|
| {
|
| - m_image = cropImage(bitmap->bitmapImage(), cropRect);
|
| + if (options.imageOrientation() == imageOrientationFlipY)
|
| + m_image = cropImage(bitmap->bitmapImage(), cropRect, true);
|
| + else
|
| + m_image = cropImage(bitmap->bitmapImage(), cropRect, false);
|
| m_image->setOriginClean(bitmap->originClean());
|
| }
|
|
|
| ImageBitmap::ImageBitmap(PassRefPtr<StaticBitmapImage> image, const IntRect& cropRect, const ImageBitmapOptions& options)
|
| {
|
| - m_image = cropImage(image.get(), cropRect);
|
| + if (options.imageOrientation() == imageOrientationFlipY)
|
| + m_image = cropImage(image.get(), cropRect, true);
|
| + else
|
| + m_image = cropImage(image.get(), cropRect, false);
|
| m_image->setOriginClean(image->originClean());
|
| }
|
|
|
|
|