summaryrefslogtreecommitdiffstats
path: root/chromium/third_party/WebKit/Source/platform/DragImage.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'chromium/third_party/WebKit/Source/platform/DragImage.cpp')
-rw-r--r--chromium/third_party/WebKit/Source/platform/DragImage.cpp272
1 files changed, 272 insertions, 0 deletions
diff --git a/chromium/third_party/WebKit/Source/platform/DragImage.cpp b/chromium/third_party/WebKit/Source/platform/DragImage.cpp
new file mode 100644
index 00000000000..af4e666fac2
--- /dev/null
+++ b/chromium/third_party/WebKit/Source/platform/DragImage.cpp
@@ -0,0 +1,272 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include "config.h"
+#include "platform/DragImage.h"
+
+#include "platform/fonts/Font.h"
+#include "platform/fonts/FontCache.h"
+#include "platform/fonts/FontDescription.h"
+#include "platform/fonts/FontMetrics.h"
+#include "platform/geometry/FloatPoint.h"
+#include "platform/geometry/FloatRect.h"
+#include "platform/geometry/IntPoint.h"
+#include "platform/graphics/BitmapImage.h"
+#include "platform/graphics/Color.h"
+#include "platform/graphics/GraphicsContext.h"
+#include "platform/graphics/Image.h"
+#include "platform/graphics/ImageBuffer.h"
+#include "platform/graphics/skia/NativeImageSkia.h"
+#include "platform/text/BidiTextRun.h"
+#include "platform/text/StringTruncator.h"
+#include "platform/text/TextRun.h"
+#include "platform/transforms/AffineTransform.h"
+#include "platform/weborigin/KURL.h"
+#include "skia/ext/image_operations.h"
+#include "third_party/skia/include/core/SkCanvas.h"
+#include "third_party/skia/include/core/SkMatrix.h"
+#include "wtf/PassOwnPtr.h"
+#include "wtf/RefPtr.h"
+#include "wtf/text/WTFString.h"
+
+#include <algorithm>
+
+namespace WebCore {
+
+const float kDragLabelBorderX = 4;
+// Keep border_y in synch with DragController::LinkDragBorderInset.
+const float kDragLabelBorderY = 2;
+const float kLabelBorderYOffset = 2;
+
+const float kMaxDragLabelWidth = 300;
+const float kMaxDragLabelStringWidth = (kMaxDragLabelWidth - 2 * kDragLabelBorderX);
+
+const float kDragLinkLabelFontSize = 11;
+const float kDragLinkUrlFontSize = 10;
+
+PassOwnPtr<DragImage> DragImage::create(Image* image, RespectImageOrientationEnum shouldRespectImageOrientation, float deviceScaleFactor)
+{
+ if (!image)
+ return nullptr;
+
+ RefPtr<NativeImageSkia> bitmap = image->nativeImageForCurrentFrame();
+ if (!bitmap)
+ return nullptr;
+
+ if (image->isBitmapImage()) {
+ ImageOrientation orientation = DefaultImageOrientation;
+ BitmapImage* bitmapImage = toBitmapImage(image);
+ IntSize sizeRespectingOrientation = bitmapImage->sizeRespectingOrientation();
+
+ if (shouldRespectImageOrientation == RespectImageOrientation)
+ orientation = bitmapImage->currentFrameOrientation();
+
+ if (orientation != DefaultImageOrientation) {
+ FloatRect destRect(FloatPoint(), sizeRespectingOrientation);
+ if (orientation.usesWidthAsHeight())
+ destRect = destRect.transposedRect();
+
+ SkBitmap skBitmap;
+ if (!skBitmap.allocN32Pixels(sizeRespectingOrientation.width(), sizeRespectingOrientation.height()))
+ return nullptr;
+
+ SkCanvas canvas(skBitmap);
+ canvas.concat(affineTransformToSkMatrix(orientation.transformFromDefault(sizeRespectingOrientation)));
+ canvas.drawBitmapRect(bitmap->bitmap(), 0, destRect);
+
+ return adoptPtr(new DragImage(skBitmap, deviceScaleFactor));
+ }
+ }
+
+ SkBitmap skBitmap;
+ if (!bitmap->bitmap().copyTo(&skBitmap, kPMColor_SkColorType))
+ return nullptr;
+ return adoptPtr(new DragImage(skBitmap, deviceScaleFactor));
+}
+
+static Font deriveDragLabelFont(int size, FontWeight fontWeight, const FontDescription& systemFont)
+{
+ FontDescription description = systemFont;
+ description.setWeight(fontWeight);
+ description.setSpecifiedSize(size);
+ description.setComputedSize(size);
+ Font result(description);
+ result.update(nullptr);
+ return result;
+}
+
+PassOwnPtr<DragImage> DragImage::create(const KURL& url, const String& inLabel, const FontDescription& systemFont, float deviceScaleFactor)
+{
+ const Font labelFont = deriveDragLabelFont(kDragLinkLabelFontSize, FontWeightBold, systemFont);
+ const Font urlFont = deriveDragLabelFont(kDragLinkUrlFontSize, FontWeightNormal, systemFont);
+ FontCachePurgePreventer fontCachePurgePreventer;
+
+ bool drawURLString = true;
+ bool clipURLString = false;
+ bool clipLabelString = false;
+
+ String urlString = url.string();
+ String label = inLabel.stripWhiteSpace();
+ if (label.isEmpty()) {
+ drawURLString = false;
+ label = urlString;
+ }
+
+ // First step is drawing the link drag image width.
+ TextRun labelRun(label.impl());
+ TextRun urlRun(urlString.impl());
+ IntSize labelSize(labelFont.width(labelRun), labelFont.fontMetrics().ascent() + labelFont.fontMetrics().descent());
+
+ if (labelSize.width() > kMaxDragLabelStringWidth) {
+ labelSize.setWidth(kMaxDragLabelStringWidth);
+ clipLabelString = true;
+ }
+
+ IntSize urlStringSize;
+ IntSize imageSize(labelSize.width() + kDragLabelBorderX * 2, labelSize.height() + kDragLabelBorderY * 2);
+
+ if (drawURLString) {
+ urlStringSize.setWidth(urlFont.width(urlRun));
+ urlStringSize.setHeight(urlFont.fontMetrics().ascent() + urlFont.fontMetrics().descent());
+ imageSize.setHeight(imageSize.height() + urlStringSize.height());
+ if (urlStringSize.width() > kMaxDragLabelStringWidth) {
+ imageSize.setWidth(kMaxDragLabelWidth);
+ clipURLString = true;
+ } else
+ imageSize.setWidth(std::max(labelSize.width(), urlStringSize.width()) + kDragLabelBorderX * 2);
+ }
+
+ // We now know how big the image needs to be, so we create and
+ // fill the background
+ IntSize scaledImageSize = imageSize;
+ scaledImageSize.scale(deviceScaleFactor);
+ OwnPtr<ImageBuffer> buffer(ImageBuffer::create(scaledImageSize));
+ if (!buffer)
+ return nullptr;
+ buffer->context()->scale(deviceScaleFactor, deviceScaleFactor);
+
+ const float DragLabelRadius = 5;
+ const IntSize radii(DragLabelRadius, DragLabelRadius);
+ IntRect rect(IntPoint(), imageSize);
+ const Color backgroundColor(140, 140, 140);
+ buffer->context()->fillRoundedRect(rect, radii, radii, radii, radii, backgroundColor);
+
+ // Draw the text
+ if (drawURLString) {
+ if (clipURLString)
+ urlString = StringTruncator::centerTruncate(urlString, imageSize.width() - (kDragLabelBorderX * 2.0f), urlFont, StringTruncator::EnableRoundingHacks);
+ IntPoint textPos(kDragLabelBorderX, imageSize.height() - (kLabelBorderYOffset + urlFont.fontMetrics().descent()));
+ TextRun textRun(urlString);
+ buffer->context()->drawText(urlFont, TextRunPaintInfo(textRun), textPos);
+ }
+
+ if (clipLabelString)
+ label = StringTruncator::rightTruncate(label, imageSize.width() - (kDragLabelBorderX * 2.0f), labelFont, StringTruncator::EnableRoundingHacks);
+
+ bool hasStrongDirectionality;
+ TextRun textRun = textRunWithDirectionality(label, hasStrongDirectionality);
+ IntPoint textPos(kDragLabelBorderX, kDragLabelBorderY + labelFont.fontDescription().computedPixelSize());
+ if (hasStrongDirectionality && textRun.direction() == RTL) {
+ float textWidth = urlFont.width(textRun);
+ int availableWidth = imageSize.width() - kDragLabelBorderX * 2;
+ textPos.setX(availableWidth - ceilf(textWidth));
+ }
+ buffer->context()->drawBidiText(urlFont, TextRunPaintInfo(textRun), textPos);
+
+ RefPtr<Image> image = buffer->copyImage();
+ return DragImage::create(image.get(), DoNotRespectImageOrientation, deviceScaleFactor);
+}
+
+DragImage::DragImage(const SkBitmap& bitmap, float resolutionScale)
+ : m_bitmap(bitmap)
+ , m_resolutionScale(resolutionScale)
+{
+}
+
+DragImage::~DragImage()
+{
+}
+
+void DragImage::fitToMaxSize(const IntSize& srcSize, const IntSize& maxSize)
+{
+ float heightResizeRatio = 0.0f;
+ float widthResizeRatio = 0.0f;
+ float resizeRatio = -1.0f;
+ IntSize originalSize = size();
+
+ if (srcSize.width() > maxSize.width()) {
+ widthResizeRatio = maxSize.width() / static_cast<float>(srcSize.width());
+ resizeRatio = widthResizeRatio;
+ }
+
+ if (srcSize.height() > maxSize.height()) {
+ heightResizeRatio = maxSize.height() / static_cast<float>(srcSize.height());
+ if ((resizeRatio < 0.0f) || (resizeRatio > heightResizeRatio))
+ resizeRatio = heightResizeRatio;
+ }
+
+ if (srcSize == originalSize) {
+ if (resizeRatio > 0.0f)
+ scale(resizeRatio, resizeRatio);
+ return;
+ }
+
+ // The image was scaled in the webpage so at minimum we must account for that scaling
+ float scaleX = srcSize.width() / static_cast<float>(originalSize.width());
+ float scaleY = srcSize.height() / static_cast<float>(originalSize.height());
+ if (resizeRatio > 0.0f) {
+ scaleX *= resizeRatio;
+ scaleY *= resizeRatio;
+ }
+
+ scale(scaleX, scaleY);
+}
+
+void DragImage::scale(float scaleX, float scaleY)
+{
+ int imageWidth = scaleX * m_bitmap.width();
+ int imageHeight = scaleY * m_bitmap.height();
+ m_bitmap = skia::ImageOperations::Resize(
+ m_bitmap, skia::ImageOperations::RESIZE_LANCZOS3, imageWidth, imageHeight);
+}
+
+void DragImage::dissolveToFraction(float fraction)
+{
+ m_bitmap.setAlphaType(kPremul_SkAlphaType);
+ SkAutoLockPixels lock(m_bitmap);
+
+ for (int row = 0; row < m_bitmap.height(); ++row) {
+ for (int column = 0; column < m_bitmap.width(); ++column) {
+ uint32_t* pixel = m_bitmap.getAddr32(column, row);
+ *pixel = SkPreMultiplyARGB(
+ SkColorGetA(*pixel) * fraction,
+ SkColorGetR(*pixel),
+ SkColorGetG(*pixel),
+ SkColorGetB(*pixel));
+ }
+ }
+}
+
+} // namespace WebCore