172 lines
6.9 KiB
Dart
172 lines
6.9 KiB
Dart
// Copyright 2014 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 'dart:ui' as ui;
|
|
|
|
import 'package:fake_async/fake_async.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/gestures.dart';
|
|
import 'package:flutter/rendering.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
import '../services/fake_platform_views.dart';
|
|
import 'rendering_tester.dart';
|
|
|
|
void main() {
|
|
|
|
group('PlatformViewRenderBox', () {
|
|
late FakePlatformViewController fakePlatformViewController;
|
|
late PlatformViewRenderBox platformViewRenderBox;
|
|
setUp(() {
|
|
renderer; // Initialize bindings
|
|
fakePlatformViewController = FakePlatformViewController(0);
|
|
platformViewRenderBox = PlatformViewRenderBox(
|
|
controller: fakePlatformViewController,
|
|
hitTestBehavior: PlatformViewHitTestBehavior.opaque,
|
|
gestureRecognizers: <Factory<OneSequenceGestureRecognizer>>{
|
|
Factory<VerticalDragGestureRecognizer>(
|
|
() {
|
|
return VerticalDragGestureRecognizer();
|
|
},
|
|
),
|
|
},
|
|
);
|
|
});
|
|
|
|
test('layout should size to max constraint', () {
|
|
layout(platformViewRenderBox);
|
|
platformViewRenderBox.layout(const BoxConstraints(minWidth: 50, minHeight: 50, maxWidth: 100, maxHeight: 100));
|
|
expect(platformViewRenderBox.size, const Size(100, 100));
|
|
});
|
|
|
|
test('send semantics update if id is changed', () {
|
|
final RenderConstrainedBox tree = RenderConstrainedBox(
|
|
additionalConstraints: const BoxConstraints.tightFor(height: 20.0, width: 20.0),
|
|
child: platformViewRenderBox,
|
|
);
|
|
int semanticsUpdateCount = 0;
|
|
final SemanticsHandle semanticsHandle = renderer.pipelineOwner.ensureSemantics(
|
|
listener: () {
|
|
++semanticsUpdateCount;
|
|
},
|
|
);
|
|
layout(tree, phase: EnginePhase.flushSemantics);
|
|
// Initial semantics update
|
|
expect(semanticsUpdateCount, 1);
|
|
|
|
semanticsUpdateCount = 0;
|
|
|
|
// Request semantics update even though nothing changed.
|
|
platformViewRenderBox.markNeedsSemanticsUpdate();
|
|
pumpFrame(phase: EnginePhase.flushSemantics);
|
|
expect(semanticsUpdateCount, 0);
|
|
|
|
semanticsUpdateCount = 0;
|
|
|
|
final FakePlatformViewController updatedFakePlatformViewController = FakePlatformViewController(10);
|
|
platformViewRenderBox.controller = updatedFakePlatformViewController;
|
|
pumpFrame(phase: EnginePhase.flushSemantics);
|
|
// Update id should update the semantics.
|
|
expect(semanticsUpdateCount, 1);
|
|
|
|
semanticsHandle.dispose();
|
|
});
|
|
|
|
test('mouse hover events are dispatched via PlatformViewController.dispatchPointerEvent', () {
|
|
layout(platformViewRenderBox);
|
|
pumpFrame(phase: EnginePhase.flushSemantics);
|
|
|
|
ui.window.onPointerDataPacket!(ui.PointerDataPacket(data: <ui.PointerData>[
|
|
_pointerData(ui.PointerChange.add, Offset.zero),
|
|
_pointerData(ui.PointerChange.hover, const Offset(10, 10)),
|
|
_pointerData(ui.PointerChange.remove, const Offset(10, 10)),
|
|
]));
|
|
|
|
expect(fakePlatformViewController.dispatchedPointerEvents, isNotEmpty);
|
|
});
|
|
|
|
test('touch hover events are dispatched via PlatformViewController.dispatchPointerEvent', () {
|
|
layout(platformViewRenderBox);
|
|
pumpFrame(phase: EnginePhase.flushSemantics);
|
|
|
|
ui.window.onPointerDataPacket!(ui.PointerDataPacket(data: <ui.PointerData>[
|
|
_pointerData(ui.PointerChange.add, Offset.zero),
|
|
_pointerData(ui.PointerChange.hover, const Offset(10, 10)),
|
|
_pointerData(ui.PointerChange.remove, const Offset(10, 10)),
|
|
]));
|
|
|
|
expect(fakePlatformViewController.dispatchedPointerEvents, isNotEmpty);
|
|
});
|
|
|
|
}, skip: isBrowser); // TODO(yjbanov): fails on Web with obscured stack trace: https://github.com/flutter/flutter/issues/42770
|
|
|
|
// Regression test for https://github.com/flutter/flutter/issues/69431
|
|
test('multi-finger touch test', () {
|
|
renderer; // Initialize bindings.
|
|
final FakeAndroidPlatformViewsController viewsController = FakeAndroidPlatformViewsController();
|
|
viewsController.registerViewType('webview');
|
|
final AndroidViewController viewController =
|
|
PlatformViewsService.initAndroidView(id: 0, viewType: 'webview', layoutDirection: TextDirection.rtl);
|
|
final PlatformViewRenderBox platformViewRenderBox = PlatformViewRenderBox(
|
|
controller: viewController,
|
|
hitTestBehavior: PlatformViewHitTestBehavior.opaque,
|
|
gestureRecognizers: <Factory<OneSequenceGestureRecognizer>>{
|
|
Factory<VerticalDragGestureRecognizer>(
|
|
() => VerticalDragGestureRecognizer(),
|
|
),
|
|
},
|
|
);
|
|
layout(platformViewRenderBox);
|
|
pumpFrame(phase: EnginePhase.flushSemantics);
|
|
|
|
viewController.pointTransformer = (Offset offset) => platformViewRenderBox.globalToLocal(offset);
|
|
|
|
FakeAsync().run((FakeAsync async) {
|
|
// Put one pointer down.
|
|
ui.window.onPointerDataPacket!(ui.PointerDataPacket(data: <ui.PointerData>[
|
|
_pointerData(ui.PointerChange.add, Offset.zero, pointer: 1, kind: PointerDeviceKind.touch),
|
|
_pointerData(ui.PointerChange.down, const Offset(10, 10), pointer: 1, kind: PointerDeviceKind.touch),
|
|
_pointerData(ui.PointerChange.remove, const Offset(10, 10), pointer: 1, kind: PointerDeviceKind.touch),
|
|
]));
|
|
async.flushMicrotasks();
|
|
|
|
// Put another pointer down and then cancel it.
|
|
ui.window.onPointerDataPacket!(ui.PointerDataPacket(data: <ui.PointerData>[
|
|
_pointerData(ui.PointerChange.add, Offset.zero, pointer: 2, kind: PointerDeviceKind.touch),
|
|
_pointerData(ui.PointerChange.down, const Offset(20, 10), pointer: 2, kind: PointerDeviceKind.touch),
|
|
_pointerData(ui.PointerChange.cancel, const Offset(20, 10), pointer: 2, kind: PointerDeviceKind.touch),
|
|
]));
|
|
async.flushMicrotasks();
|
|
|
|
// The first pointer can still moving without crashing.
|
|
ui.window.onPointerDataPacket!(ui.PointerDataPacket(data: <ui.PointerData>[
|
|
_pointerData(ui.PointerChange.add, Offset.zero, pointer: 1, kind: PointerDeviceKind.touch),
|
|
_pointerData(ui.PointerChange.move, const Offset(10, 10), pointer: 1, kind: PointerDeviceKind.touch),
|
|
_pointerData(ui.PointerChange.remove, const Offset(10, 10), pointer: 1, kind: PointerDeviceKind.touch),
|
|
]));
|
|
async.flushMicrotasks();
|
|
});
|
|
|
|
// Passes if no crashes.
|
|
});
|
|
}
|
|
|
|
ui.PointerData _pointerData(
|
|
ui.PointerChange change,
|
|
Offset logicalPosition, {
|
|
int device = 0,
|
|
PointerDeviceKind kind = PointerDeviceKind.mouse,
|
|
int pointer = 0,
|
|
}) {
|
|
return ui.PointerData(
|
|
pointerIdentifier: pointer,
|
|
embedderId: pointer,
|
|
change: change,
|
|
physicalX: logicalPosition.dx * ui.window.devicePixelRatio,
|
|
physicalY: logicalPosition.dy * ui.window.devicePixelRatio,
|
|
kind: kind,
|
|
device: device,
|
|
);
|
|
}
|