diff --git a/AUTHORS b/AUTHORS index 1211ffab5ac3a..e0e7902bbba3f 100644 --- a/AUTHORS +++ b/AUTHORS @@ -23,4 +23,5 @@ Koutaro Mori TheOneWithTheBraid Twin Sun, LLC Qixing Cao -LinXunFeng \ No newline at end of file +LinXunFeng +Tim Maffett diff --git a/lib/web_ui/README.md b/lib/web_ui/README.md index 4eac836446d40..24bc7e42b61e2 100644 --- a/lib/web_ui/README.md +++ b/lib/web_ui/README.md @@ -27,7 +27,7 @@ The `build` subcommand builds web engine gn/ninja targets. Targets can be individually specified in the command line invocation, or if none are specified, all web engine targets are built. Common targets are as follows: * `sdk` - The flutter_web_sdk itself. - * `canvaskit` - Flutter's version of canvakit. + * `canvaskit` - Flutter's version of canvaskit. * `canvaskit_chromium` - A version of canvaskit optimized for use with chromium-based browsers. * `skwasm` - Builds experimental skia wasm module renderer. diff --git a/lib/web_ui/lib/src/engine/mouse_cursor.dart b/lib/web_ui/lib/src/engine/mouse_cursor.dart index 0ccfbed03b129..cedabffbf6cf6 100644 --- a/lib/web_ui/lib/src/engine/mouse_cursor.dart +++ b/lib/web_ui/lib/src/engine/mouse_cursor.dart @@ -62,7 +62,19 @@ class MouseCursor { 'zoomOut': 'zoom-out', }; static String _mapKindToCssValue(String? kind) { - return _kindToCssValueMap[kind] ?? 'default'; + // Allow 'image-set(...)'/'-webkit-image-set(...)' css commands thru for setting DevicePixelRatio + // aware images for cursors (for newer browsers) and allow 'url(...)' strings thru as fallback + // for older browsers (bare url() commands are always 1.0x dpr). The hotspot coordinates are always + // supplied in 1.0x dpr coordinates. All of these css methods can use data-uri versions of url's + // which allow for inline definition of image data to define the cursor. + if (kind != null && + (kind.startsWith('url') || + kind.startsWith('image-set') || + kind.startsWith('-webkit-image-set'))) { + return kind; + } else { + return _kindToCssValueMap[kind] ?? 'default'; + } } void activateSystemCursor(String? kind) { diff --git a/lib/web_ui/test/engine/mouse_cursor_test.dart b/lib/web_ui/test/engine/mouse_cursor_test.dart new file mode 100644 index 0000000000000..75114e778659d --- /dev/null +++ b/lib/web_ui/test/engine/mouse_cursor_test.dart @@ -0,0 +1,124 @@ +// Copyright 2013 The Flutter Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +import 'package:test/bootstrap/browser.dart'; +import 'package:test/test.dart'; +import 'package:ui/src/engine.dart'; + +const String flutterLogoPngAsDataUri = ''; + +// Following [_kindToCssValueMap] array definition is from flutter engine /lib/web_ui/lib/src/engine/mouse_cursor.dart +// +// Map from Flutter's kind values to CSS's cursor values. +// +// This map must be kept in sync with Flutter framework's +// rendering/mouse_cursor.dart. +const Map _kindToCssValueMap = { + 'alias': 'alias', + 'allScroll': 'all-scroll', + 'basic': 'default', + 'cell': 'cell', + 'click': 'pointer', + 'contextMenu': 'context-menu', + 'copy': 'copy', + 'forbidden': 'not-allowed', + 'grab': 'grab', + 'grabbing': 'grabbing', + 'help': 'help', + 'move': 'move', + 'none': 'none', + 'noDrop': 'no-drop', + 'precise': 'crosshair', + 'progress': 'progress', + 'text': 'text', + 'resizeColumn': 'col-resize', + 'resizeDown': 's-resize', + 'resizeDownLeft': 'sw-resize', + 'resizeDownRight': 'se-resize', + 'resizeLeft': 'w-resize', + 'resizeLeftRight': 'ew-resize', + 'resizeRight': 'e-resize', + 'resizeRow': 'row-resize', + 'resizeUp': 'n-resize', + 'resizeUpDown': 'ns-resize', + 'resizeUpLeft': 'nw-resize', + 'resizeUpRight': 'ne-resize', + 'resizeUpLeftDownRight': 'nwse-resize', + 'resizeUpRightDownLeft': 'nesw-resize', + 'verticalText': 'vertical-text', + 'wait': 'wait', + 'zoomIn': 'zoom-in', + 'zoomOut': 'zoom-out', +}; + +void main() { + internalBootstrapBrowserTest(() => testMain); +} + +void testMain() { + ensureFlutterViewEmbedderInitialized(); + final DomElement flutterViewElement = flutterViewEmbedder.flutterViewElement; + + setUp(() { + }); + + MouseCursor.initialize(); + final MouseCursor? mouseCursorTry = MouseCursor.instance; + + assert(mouseCursorTry!=null,'Error MouseCursor.instance is null'); + + final MouseCursor mouseCursorInstance = mouseCursorTry!; + + void setCursorKind(String kindToTest) { + mouseCursorInstance.activateSystemCursor( kindToTest ); + } + + String getSetCursorCSSStyle() { + final DomCSSStyleDeclaration style = domWindow.getComputedStyle(flutterViewElement); + return style.getPropertyValue('cursor'); + } + + // Make sure that all 'built in' MouseCursor kinds ([_kindToCssValueMap]) make it through to there css style mapping + test('MouseCursor.activateSystemCursor test built in system cursor to browser css map', () { + _kindToCssValueMap.forEach( (String key, String expectedResult) { + setCursorKind(key); + expect( getSetCursorCSSStyle(), equals(expectedResult) ); + } + ); + }); + + // We need to test that cursor key/kind values that start with + // 'url(..)', 'image-set(...)'/'-webkit-image-set(...)' css commands make it through. + test('MouseCursor.activateSystemCursor allows url/-webkit-image-set kind value prefix to pass', () { + const String urlCss = 'url("cursor.png")'; + const String webkitImageSetCss = '-webkit-image-set($urlCss 1x) 0 0, $urlCss, help'; + + setCursorKind('grabbing'); // reset to different cursor before testing 'url...' + + setCursorKind('$urlCss 4 12, default'); + expect( getSetCursorCSSStyle(), startsWith('url') ); + + setCursorKind('progress'); // reset to different cursor before testing '-webkit-image-set...' + + // we do test for '-webkit-image-set' or 'image-set' prefix, browsers may change '-webkit-image-set' to 'image-set'. + setCursorKind(webkitImageSetCss); + expect( getSetCursorCSSStyle(), anyOf(startsWith('-webkit-image-set'),startsWith('image-set')) ); + }); + + test('MouseCursor.activateSystemCursor allows url/-webkit-image-set (with datauri url) kind value prefix to pass', () { + const String urlCss = 'url("$flutterLogoPngAsDataUri")'; + const String webkitImageSetCss = '-webkit-image-set($urlCss 1x) 0 0, $urlCss, help'; + + setCursorKind('move'); // reset to different cursor before testing 'url...' + + setCursorKind('$urlCss 4 12, default'); + expect( getSetCursorCSSStyle(), startsWith('url') ); + + setCursorKind('progress'); // reset to different cursor before testing '-webkit-image-set...' + + // we do test for '-webkit-image-set' or 'image-set' prefix, browsers may change '-webkit-image-set' to 'image-set'. + setCursorKind(webkitImageSetCss); + expect( getSetCursorCSSStyle(), anyOf(startsWith('-webkit-image-set'),startsWith('image-set')) ); + }); +}