blob: 3a5e00be6718b118629868bafbac797ce6fe07b5 [file] [log] [blame]
/*
* Copyright (C) 2008, 2013 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 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 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 "DragImage.h"
#include "BitmapInfo.h"
#include "CachedImage.h"
#include "GraphicsContext.h"
#include "GraphicsContextPlatformPrivateCairo.h"
#include "HWndDC.h"
#include "Image.h"
#include <cairo-win32.h>
#include <windows.h>
#include <wtf/RetainPtr.h>
#include <wtf/win/GDIObject.h>
namespace WebCore {
void deallocContext(PlatformContextCairo* target)
{
delete target;
}
GDIObject<HBITMAP> allocImage(HDC dc, IntSize size, PlatformContextCairo** targetRef)
{
BitmapInfo bmpInfo = BitmapInfo::create(size);
LPVOID bits;
auto hbmp = adoptGDIObject(::CreateDIBSection(dc, &bmpInfo, DIB_RGB_COLORS, &bits, 0, 0));
// At this point, we have a Cairo surface that points to a Windows DIB. The DIB interprets
// with the opposite meaning of positive Y axis, so everything we draw into this cairo
// context is going to be upside down.
if (!targetRef)
return hbmp;
cairo_surface_t* bitmapContext = cairo_image_surface_create_for_data((unsigned char*)bits,
CAIRO_FORMAT_ARGB32,
bmpInfo.bmiHeader.biWidth,
bmpInfo.bmiHeader.biHeight,
bmpInfo.bmiHeader.biWidth * 4);
if (!bitmapContext)
return GDIObject<HBITMAP>();
cairo_t* cr = cairo_create(bitmapContext);
cairo_surface_destroy(bitmapContext);
// At this point, we have a Cairo surface that points to a Windows DIB. The DIB interprets
// with the opposite meaning of positive Y axis, so everything we draw into this cairo
// context is going to be upside down.
//
// So, we must invert the CTM for the context so that drawing commands will be flipped
// before they get written to the internal buffer.
cairo_matrix_t matrix;
cairo_matrix_init(&matrix, 1.0, 0.0, 0.0, -1.0, 0.0, size.height());
cairo_set_matrix(cr, &matrix);
*targetRef = new PlatformGraphicsContext(cr);
cairo_destroy(cr);
return hbmp;
}
static cairo_surface_t* createCairoContextFromBitmap(HBITMAP bitmap)
{
BITMAP info;
GetObject(bitmap, sizeof(info), &info);
ASSERT(info.bmBitsPixel == 32);
// At this point, we have a Cairo surface that points to a Windows BITMAP. The BITMAP
// has the opposite meaning of positive Y axis, so everything we draw into this cairo
// context is going to be upside down.
return cairo_image_surface_create_for_data((unsigned char*)info.bmBits,
CAIRO_FORMAT_ARGB32,
info.bmWidth,
info.bmHeight,
info.bmWidthBytes);
}
DragImageRef scaleDragImage(DragImageRef imageRef, FloatSize scale)
{
// FIXME: due to the way drag images are done on windows we need
// to preprocess the alpha channel <rdar://problem/5015946>
if (!imageRef)
return 0;
GDIObject<HBITMAP> hbmp;
auto image = adoptGDIObject(imageRef);
IntSize srcSize = dragImageSize(image.get());
IntSize dstSize(static_cast<int>(srcSize.width() * scale.width()), static_cast<int>(srcSize.height() * scale.height()));
HWndDC dc(0);
auto dstDC = adoptGDIObject(::CreateCompatibleDC(dc));
if (!dstDC)
goto exit;
PlatformContextCairo* targetContext;
hbmp = allocImage(dstDC.get(), dstSize, &targetContext);
if (!hbmp)
goto exit;
cairo_surface_t* srcImage = createCairoContextFromBitmap(image.get());
// Scale the target surface to the new image size, and flip it
// so that when we set the srcImage as the surface it will draw
// right-side-up.
cairo_t* cr = targetContext->cr();
cairo_translate(cr, 0, dstSize.height());
cairo_scale(cr, scale.width(), -scale.height());
cairo_set_source_surface(cr, srcImage, 0.0, 0.0);
// Now we can paint and get the correct result
cairo_paint(cr);
cairo_surface_destroy(srcImage);
deallocContext(targetContext);
exit:
if (!hbmp)
hbmp.swap(image);
return hbmp.leak();
}
DragImageRef createDragImageFromImage(Image* img, ImageOrientation)
{
HWndDC dc(0);
auto workingDC = adoptGDIObject(::CreateCompatibleDC(dc));
if (!workingDC)
return 0;
PlatformContextCairo* drawContext = 0;
auto hbmp = allocImage(workingDC.get(), IntSize(img->size()), &drawContext);
if (!hbmp || !drawContext)
return 0;
cairo_t* cr = drawContext->cr();
cairo_set_source_rgb(cr, 1.0, 0.0, 1.0);
cairo_fill_preserve(cr);
RefPtr<cairo_surface_t> surface = img->nativeImageForCurrentFrame();
if (surface) {
// Draw the image.
cairo_set_source_surface(cr, surface.get(), 0.0, 0.0);
cairo_paint(cr);
}
deallocContext(drawContext);
return hbmp.leak();
}
}