blob: 7fed9408c7b03e639329ef99f65e98484ca1c8b0 [file] [log] [blame]
/*
* Copyright (C) 2004-2017 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.
*/
#import "config.h"
#import "Cursor.h"
#if PLATFORM(MAC)
#import <objc/runtime.h>
#import <pal/spi/mac/HIServicesSPI.h>
#import <wtf/BlockObjCExceptions.h>
#import <wtf/StdLibExtras.h>
@interface WebCoreCursorBundle : NSObject { }
@end
@implementation WebCoreCursorBundle
@end
namespace WebCore {
static NSCursor *busyButClickableNSCursor;
static NSCursor *makeAliasNSCursor;
static NSCursor *moveNSCursor;
static NSCursor *resizeEastNSCursor;
static NSCursor *resizeEastWestNSCursor;
static NSCursor *resizeNorthNSCursor;
static NSCursor *resizeNorthSouthNSCursor;
static NSCursor *resizeNortheastNSCursor;
static NSCursor *resizeNortheastSouthwestNSCursor;
static NSCursor *resizeNorthwestNSCursor;
static NSCursor *resizeNorthwestSoutheastNSCursor;
static NSCursor *resizeSouthNSCursor;
static NSCursor *resizeSoutheastNSCursor;
static NSCursor *resizeSouthwestNSCursor;
static NSCursor *resizeWestNSCursor;
static NSCursor *cellNSCursor;
static NSCursor *helpNSCursor;
static NSCursor *zoomInNSCursor;
static NSCursor *zoomOutNSCursor;
static NSInteger WKCoreCursor_coreCursorType(id self, SEL)
{
if (self == busyButClickableNSCursor)
return kCoreCursorBusyButClickable;
if (self == makeAliasNSCursor)
return kCoreCursorMakeAlias;
if (self == moveNSCursor)
return kCoreCursorWindowMove;
if (self == resizeEastNSCursor)
return kCoreCursorWindowResizeEast;
if (self == resizeEastWestNSCursor)
return kCoreCursorWindowResizeEastWest;
if (self == resizeNorthNSCursor)
return kCoreCursorWindowResizeNorth;
if (self == resizeNorthSouthNSCursor)
return kCoreCursorWindowResizeNorthSouth;
if (self == resizeNortheastNSCursor)
return kCoreCursorWindowResizeNorthEast;
if (self == resizeNortheastSouthwestNSCursor)
return kCoreCursorWindowResizeNorthEastSouthWest;
if (self == resizeNorthwestNSCursor)
return kCoreCursorWindowResizeNorthWest;
if (self == resizeNorthwestSoutheastNSCursor)
return kCoreCursorWindowResizeNorthWestSouthEast;
if (self == resizeSouthNSCursor)
return kCoreCursorWindowResizeSouth;
if (self == resizeSoutheastNSCursor)
return kCoreCursorWindowResizeSouthEast;
if (self == resizeSouthwestNSCursor)
return kCoreCursorWindowResizeSouthWest;
if (self == resizeWestNSCursor)
return kCoreCursorWindowResizeWest;
if (self == cellNSCursor)
return kCoreCursorCell;
if (self == helpNSCursor)
return kCoreCursorHelp;
if (self == zoomInNSCursor)
return kCoreCursorZoomIn;
if (self == zoomOutNSCursor)
return kCoreCursorZoomOut;
return NSNotFound;
}
static Class createCoreCursorClass()
{
Class coreCursorClass = objc_allocateClassPair([NSCursor class], "WKCoreCursor", 0);
SEL coreCursorType = NSSelectorFromString(@"_coreCursorType");
class_addMethod(coreCursorClass, coreCursorType, (IMP)WKCoreCursor_coreCursorType, method_getTypeEncoding(class_getInstanceMethod([NSCursor class], coreCursorType)));
objc_registerClassPair(coreCursorClass);
return coreCursorClass;
}
static Class coreCursorClass()
{
Class coreCursorClass = objc_lookUpClass("WKCoreCursor");
if (!coreCursorClass)
coreCursorClass = createCoreCursorClass();
return coreCursorClass;
}
static NSCursor *cursor(const char *name)
{
__strong NSCursor **slot = nullptr;
if (!strcmp(name, "BusyButClickable"))
slot = &busyButClickableNSCursor;
else if (!strcmp(name, "MakeAlias"))
slot = &makeAliasNSCursor;
else if (!strcmp(name, "Move"))
slot = &moveNSCursor;
else if (!strcmp(name, "ResizeEast"))
slot = &resizeEastNSCursor;
else if (!strcmp(name, "ResizeEastWest"))
slot = &resizeEastWestNSCursor;
else if (!strcmp(name, "ResizeNorth"))
slot = &resizeNorthNSCursor;
else if (!strcmp(name, "ResizeNorthSouth"))
slot = &resizeNorthSouthNSCursor;
else if (!strcmp(name, "ResizeNortheast"))
slot = &resizeNortheastNSCursor;
else if (!strcmp(name, "ResizeNortheastSouthwest"))
slot = &resizeNortheastSouthwestNSCursor;
else if (!strcmp(name, "ResizeNorthwest"))
slot = &resizeNorthwestNSCursor;
else if (!strcmp(name, "ResizeNorthwestSoutheast"))
slot = &resizeNorthwestSoutheastNSCursor;
else if (!strcmp(name, "ResizeSouth"))
slot = &resizeSouthNSCursor;
else if (!strcmp(name, "ResizeSoutheast"))
slot = &resizeSoutheastNSCursor;
else if (!strcmp(name, "ResizeSouthwest"))
slot = &resizeSouthwestNSCursor;
else if (!strcmp(name, "ResizeWest"))
slot = &resizeWestNSCursor;
else if (!strcmp(name, "Cell"))
slot = &cellNSCursor;
else if (!strcmp(name, "Help"))
slot = &helpNSCursor;
else if (!strcmp(name, "ZoomIn"))
slot = &zoomInNSCursor;
else if (!strcmp(name, "ZoomOut"))
slot = &zoomOutNSCursor;
if (!slot)
return nil;
if (!*slot)
*slot = [[coreCursorClass() alloc] init];
return *slot;
}
// Simple NSCursor calls shouldn't need protection,
// but creating a cursor with a bad image might throw.
#if ENABLE(MOUSE_CURSOR_SCALE)
static RetainPtr<NSCursor> createCustomCursor(Image* image, const IntPoint& hotSpot, float scale)
#else
static RetainPtr<NSCursor> createCustomCursor(Image* image, const IntPoint& hotSpot)
#endif
{
// FIXME: The cursor won't animate. Not sure if that's a big deal.
auto nsImage = image->snapshotNSImage();
if (!nsImage)
return nullptr;
BEGIN_BLOCK_OBJC_EXCEPTIONS;
#if ENABLE(MOUSE_CURSOR_SCALE)
NSSize size = NSMakeSize(image->width() / scale, image->height() / scale);
NSSize expandedSize = NSMakeSize(ceil(size.width), ceil(size.height));
// Pad the image with transparent pixels so it has an integer boundary.
if (size.width != expandedSize.width || size.height != expandedSize.height) {
RetainPtr<NSImage> expandedImage = adoptNS([[NSImage alloc] initWithSize:expandedSize]);
NSRect toRect = NSMakeRect(0, expandedSize.height - size.height, size.width, size.height);
NSRect fromRect = NSMakeRect(0, 0, image->width(), image->height());
[expandedImage lockFocus];
[nsImage drawInRect:toRect fromRect:fromRect operation:NSCompositingOperationSourceOver fraction:1];
[expandedImage unlockFocus];
return adoptNS([[NSCursor alloc] initWithImage:expandedImage.get() hotSpot:hotSpot]);
}
// Scale the image and its representation to match retina resolution.
[nsImage setSize:expandedSize];
[[[nsImage representations] objectAtIndex:0] setSize:expandedSize];
#endif
return adoptNS([[NSCursor alloc] initWithImage:nsImage.get() hotSpot:hotSpot]);
END_BLOCK_OBJC_EXCEPTIONS;
return nullptr;
}
void Cursor::ensurePlatformCursor() const
{
if (m_platformCursor)
return;
switch (m_type) {
case Cursor::Pointer:
m_platformCursor = [NSCursor arrowCursor];
break;
case Cursor::Cross:
m_platformCursor = [NSCursor crosshairCursor];
break;
case Cursor::Hand:
m_platformCursor = [NSCursor pointingHandCursor];
break;
case Cursor::IBeam:
m_platformCursor = [NSCursor IBeamCursor];
break;
case Cursor::Wait:
m_platformCursor = cursor("BusyButClickable");
break;
case Cursor::Help:
m_platformCursor = cursor("Help");
break;
case Cursor::Move:
case Cursor::MiddlePanning:
m_platformCursor = cursor("Move");
break;
case Cursor::EastResize:
case Cursor::EastPanning:
m_platformCursor = cursor("ResizeEast");
break;
case Cursor::NorthResize:
case Cursor::NorthPanning:
m_platformCursor = cursor("ResizeNorth");
break;
case Cursor::NorthEastResize:
case Cursor::NorthEastPanning:
m_platformCursor = cursor("ResizeNortheast");
break;
case Cursor::NorthWestResize:
case Cursor::NorthWestPanning:
m_platformCursor = cursor("ResizeNorthwest");
break;
case Cursor::SouthResize:
case Cursor::SouthPanning:
m_platformCursor = cursor("ResizeSouth");
break;
case Cursor::SouthEastResize:
case Cursor::SouthEastPanning:
m_platformCursor = cursor("ResizeSoutheast");
break;
case Cursor::SouthWestResize:
case Cursor::SouthWestPanning:
m_platformCursor = cursor("ResizeSouthwest");
break;
case Cursor::WestResize:
case Cursor::WestPanning:
m_platformCursor = cursor("ResizeWest");
break;
case Cursor::NorthSouthResize:
m_platformCursor = cursor("ResizeNorthSouth");
break;
case Cursor::EastWestResize:
m_platformCursor = cursor("ResizeEastWest");
break;
case Cursor::NorthEastSouthWestResize:
m_platformCursor = cursor("ResizeNortheastSouthwest");
break;
case Cursor::NorthWestSouthEastResize:
m_platformCursor = cursor("ResizeNorthwestSoutheast");
break;
case Cursor::ColumnResize:
m_platformCursor = [NSCursor resizeLeftRightCursor];
break;
case Cursor::RowResize:
m_platformCursor = [NSCursor resizeUpDownCursor];
break;
case Cursor::VerticalText:
m_platformCursor = [NSCursor IBeamCursorForVerticalLayout];
break;
case Cursor::Cell:
m_platformCursor = cursor("Cell");
break;
case Cursor::ContextMenu:
m_platformCursor = [NSCursor contextualMenuCursor];
break;
case Cursor::Alias:
m_platformCursor = cursor("MakeAlias");
break;
case Cursor::Progress:
m_platformCursor = cursor("BusyButClickable");
break;
case Cursor::NoDrop:
m_platformCursor = [NSCursor operationNotAllowedCursor];
break;
case Cursor::Copy:
m_platformCursor = [NSCursor dragCopyCursor];
break;
case Cursor::None:
m_platformCursor = adoptNS([[NSCursor alloc] initWithImage:adoptNS([[NSImage alloc] initWithSize:NSMakeSize(1, 1)]).get() hotSpot:NSZeroPoint]);
break;
case Cursor::NotAllowed:
m_platformCursor = [NSCursor operationNotAllowedCursor];
break;
case Cursor::ZoomIn:
m_platformCursor = cursor("ZoomIn");
break;
case Cursor::ZoomOut:
m_platformCursor = cursor("ZoomOut");
break;
case Cursor::Grab:
m_platformCursor = [NSCursor openHandCursor];
break;
case Cursor::Grabbing:
m_platformCursor = [NSCursor closedHandCursor];
break;
case Cursor::Custom:
#if ENABLE(MOUSE_CURSOR_SCALE)
m_platformCursor = createCustomCursor(m_image.get(), m_hotSpot, m_imageScaleFactor);
#else
m_platformCursor = createCustomCursor(m_image.get(), m_hotSpot);
#endif
break;
}
}
NSCursor *Cursor::platformCursor() const
{
ensurePlatformCursor();
return m_platformCursor.get();
}
} // namespace WebCore
#endif // PLATFORM(MAC)