mirror of
https://github.com/flutter/flutter.git
synced 2026-02-04 19:00:09 +08:00
This auto-formats all *.dart files in the repository outside of the `engine` subdirectory and enforces that these files stay formatted with a presubmit check. **Reviewers:** Please carefully review all the commits except for the one titled "formatted". The "formatted" commit was auto-generated by running `dev/tools/format.sh -a -f`. The other commits were hand-crafted to prepare the repo for the formatting change. I recommend reviewing the commits one-by-one via the "Commits" tab and avoiding Github's "Files changed" tab as it will likely slow down your browser because of the size of this PR. --------- Co-authored-by: Kate Lovett <katelovett@google.com> Co-authored-by: LongCatIsLooong <31859944+LongCatIsLooong@users.noreply.github.com>
403 lines
14 KiB
Dart
403 lines
14 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:async';
|
|
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() {
|
|
final TestRenderingFlutterBinding binding = TestRenderingFlutterBinding.ensureInitialized();
|
|
|
|
group('PlatformViewRenderBox', () {
|
|
late FakePlatformViewController fakePlatformViewController;
|
|
late PlatformViewRenderBox platformViewRenderBox;
|
|
setUp(() {
|
|
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 =
|
|
TestRenderingFlutterBinding.instance.ensureSemantics();
|
|
TestRenderingFlutterBinding.instance.pipelineOwner.semanticsOwner!.addListener(() {
|
|
++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);
|
|
|
|
RendererBinding.instance.platformDispatcher.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);
|
|
|
|
RendererBinding.instance.platformDispatcher.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);
|
|
});
|
|
});
|
|
|
|
// Regression test for https://github.com/flutter/flutter/issues/69431
|
|
test('multi-finger touch test', () {
|
|
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.
|
|
RendererBinding.instance.platformDispatcher.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.
|
|
RendererBinding.instance.platformDispatcher.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.
|
|
RendererBinding.instance.platformDispatcher.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.
|
|
});
|
|
|
|
test('created callback is reset when controller is changed', () {
|
|
final FakeAndroidPlatformViewsController viewsController = FakeAndroidPlatformViewsController();
|
|
viewsController.registerViewType('webview');
|
|
final AndroidViewController firstController = PlatformViewsService.initAndroidView(
|
|
id: 0,
|
|
viewType: 'webview',
|
|
layoutDirection: TextDirection.rtl,
|
|
);
|
|
final RenderAndroidView renderBox = RenderAndroidView(
|
|
viewController: firstController,
|
|
hitTestBehavior: PlatformViewHitTestBehavior.opaque,
|
|
gestureRecognizers: <Factory<OneSequenceGestureRecognizer>>{},
|
|
);
|
|
layout(renderBox);
|
|
pumpFrame(phase: EnginePhase.flushSemantics);
|
|
|
|
expect(firstController.createdCallbacks, isNotEmpty);
|
|
expect(firstController.createdCallbacks.length, 1);
|
|
|
|
final AndroidViewController secondController = PlatformViewsService.initAndroidView(
|
|
id: 0,
|
|
viewType: 'webview',
|
|
layoutDirection: TextDirection.rtl,
|
|
);
|
|
// Reset controller.
|
|
renderBox.controller = secondController;
|
|
|
|
expect(firstController.createdCallbacks, isEmpty);
|
|
expect(secondController.createdCallbacks, isNotEmpty);
|
|
expect(secondController.createdCallbacks.length, 1);
|
|
});
|
|
|
|
test('render object changed its visual appearance after texture is created', () {
|
|
FakeAsync().run((FakeAsync async) {
|
|
final AndroidViewController viewController = PlatformViewsService.initAndroidView(
|
|
id: 0,
|
|
viewType: 'webview',
|
|
layoutDirection: TextDirection.rtl,
|
|
);
|
|
final RenderAndroidView renderBox = RenderAndroidView(
|
|
viewController: viewController,
|
|
hitTestBehavior: PlatformViewHitTestBehavior.opaque,
|
|
gestureRecognizers: <Factory<OneSequenceGestureRecognizer>>{},
|
|
);
|
|
|
|
final Completer<void> viewCreation = Completer<void>();
|
|
const MethodChannel channel = MethodChannel('flutter/platform_views');
|
|
binding.defaultBinaryMessenger.setMockMethodCallHandler(channel, (
|
|
MethodCall methodCall,
|
|
) async {
|
|
assert(methodCall.method == 'create', 'Unexpected method call');
|
|
await viewCreation.future;
|
|
return /*textureId=*/ 0;
|
|
});
|
|
|
|
layout(renderBox);
|
|
pumpFrame(phase: EnginePhase.paint);
|
|
|
|
expect(renderBox.debugLayer, isNotNull);
|
|
expect(renderBox.debugLayer!.hasChildren, isFalse);
|
|
expect(viewController.isCreated, isFalse);
|
|
expect(renderBox.debugNeedsPaint, isFalse);
|
|
|
|
viewCreation.complete();
|
|
async.flushMicrotasks();
|
|
|
|
expect(viewController.isCreated, isTrue);
|
|
expect(renderBox.debugNeedsPaint, isTrue);
|
|
expect(renderBox.debugLayer!.hasChildren, isFalse);
|
|
|
|
pumpFrame(phase: EnginePhase.paint);
|
|
expect(renderBox.debugLayer!.hasChildren, isTrue);
|
|
expect(renderBox.debugLayer!.firstChild, isA<TextureLayer>());
|
|
});
|
|
});
|
|
|
|
test('markNeedsPaint does not get called on a disposed RO', () async {
|
|
FakeAsync().run((FakeAsync async) {
|
|
final AndroidViewController viewController = PlatformViewsService.initAndroidView(
|
|
id: 0,
|
|
viewType: 'webview',
|
|
layoutDirection: TextDirection.rtl,
|
|
);
|
|
final RenderAndroidView renderBox = RenderAndroidView(
|
|
viewController: viewController,
|
|
hitTestBehavior: PlatformViewHitTestBehavior.opaque,
|
|
gestureRecognizers: <Factory<OneSequenceGestureRecognizer>>{},
|
|
);
|
|
|
|
final Completer<void> viewCreation = Completer<void>();
|
|
const MethodChannel channel = MethodChannel('flutter/platform_views');
|
|
binding.defaultBinaryMessenger.setMockMethodCallHandler(channel, (
|
|
MethodCall methodCall,
|
|
) async {
|
|
assert(methodCall.method == 'create', 'Unexpected method call');
|
|
await viewCreation.future;
|
|
return /*textureId=*/ 0;
|
|
});
|
|
|
|
layout(renderBox);
|
|
pumpFrame(phase: EnginePhase.paint);
|
|
|
|
expect(renderBox.debugLayer, isNotNull);
|
|
expect(renderBox.debugLayer!.hasChildren, isFalse);
|
|
expect(viewController.isCreated, isFalse);
|
|
expect(renderBox.debugNeedsPaint, isFalse);
|
|
|
|
renderBox.dispose();
|
|
viewCreation.complete();
|
|
async.flushMicrotasks();
|
|
|
|
expect(viewController.isCreated, isTrue);
|
|
expect(renderBox.debugNeedsPaint, isFalse);
|
|
expect(renderBox.debugLayer, isNull);
|
|
|
|
pumpFrame(phase: EnginePhase.paint);
|
|
expect(renderBox.debugLayer, isNull);
|
|
});
|
|
});
|
|
|
|
test('markNeedsPaint does not get called when setting the same viewController', () {
|
|
FakeAsync().run((FakeAsync async) {
|
|
final Completer<void> viewCreation = Completer<void>();
|
|
const MethodChannel channel = MethodChannel('flutter/platform_views');
|
|
binding.defaultBinaryMessenger.setMockMethodCallHandler(channel, (
|
|
MethodCall methodCall,
|
|
) async {
|
|
assert(methodCall.method == 'create', 'Unexpected method call');
|
|
await viewCreation.future;
|
|
return /*textureId=*/ 0;
|
|
});
|
|
|
|
bool futureCallbackRan = false;
|
|
|
|
PlatformViewsService.initUiKitView(
|
|
id: 0,
|
|
viewType: 'webview',
|
|
layoutDirection: TextDirection.ltr,
|
|
).then((UiKitViewController viewController) {
|
|
final RenderUiKitView renderBox = RenderUiKitView(
|
|
viewController: viewController,
|
|
hitTestBehavior: PlatformViewHitTestBehavior.opaque,
|
|
gestureRecognizers: <Factory<OneSequenceGestureRecognizer>>{},
|
|
);
|
|
|
|
layout(renderBox);
|
|
pumpFrame(phase: EnginePhase.paint);
|
|
expect(renderBox.debugNeedsPaint, isFalse);
|
|
|
|
renderBox.viewController = viewController;
|
|
|
|
expect(renderBox.debugNeedsPaint, isFalse);
|
|
|
|
futureCallbackRan = true;
|
|
});
|
|
|
|
viewCreation.complete();
|
|
async.flushMicrotasks();
|
|
expect(futureCallbackRan, true);
|
|
});
|
|
});
|
|
}
|
|
|
|
ui.PointerData _pointerData(
|
|
ui.PointerChange change,
|
|
Offset logicalPosition, {
|
|
int device = 0,
|
|
PointerDeviceKind kind = PointerDeviceKind.mouse,
|
|
int pointer = 0,
|
|
}) {
|
|
final double devicePixelRatio =
|
|
RendererBinding.instance.platformDispatcher.implicitView!.devicePixelRatio;
|
|
return ui.PointerData(
|
|
pointerIdentifier: pointer,
|
|
embedderId: pointer,
|
|
change: change,
|
|
physicalX: logicalPosition.dx * devicePixelRatio,
|
|
physicalY: logicalPosition.dy * devicePixelRatio,
|
|
kind: kind,
|
|
device: device,
|
|
);
|
|
}
|