// Copyright 2016 The Chromium 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:flutter/rendering.dart'; import 'package:flutter/scheduler.dart'; import 'package:flutter/widgets.dart'; import 'package:flutter_driver/src/common/find.dart'; import 'package:flutter_driver/src/common/request_data.dart'; import 'package:flutter_driver/src/extension/extension.dart'; import 'package:flutter_test/flutter_test.dart'; void main() { group('waitUntilNoTransientCallbacks', () { FlutterDriverExtension extension; Map result; int messageId = 0; final List log = []; setUp(() { result = null; extension = new FlutterDriverExtension((String message) async { log.add(message); return (messageId += 1).toString(); }, false); }); testWidgets('returns immediately when transient callback queue is empty', (WidgetTester tester) async { extension.call(new WaitUntilNoTransientCallbacks().serialize()) .then(expectAsync1((Map r) { result = r; })); await tester.idle(); expect( result, { 'isError': false, 'response': null, }, ); }); testWidgets('waits until no transient callbacks', (WidgetTester tester) async { SchedulerBinding.instance.scheduleFrameCallback((_) { // Intentionally blank. We only care about existence of a callback. }); extension.call(new WaitUntilNoTransientCallbacks().serialize()) .then(expectAsync1((Map r) { result = r; })); // Nothing should happen until the next frame. await tester.idle(); expect(result, isNull); // NOW we should receive the result. await tester.pump(); expect( result, { 'isError': false, 'response': null, }, ); }); testWidgets('handler', (WidgetTester tester) async { expect(log, isEmpty); final dynamic result = RequestDataResult.fromJson((await extension.call(new RequestData('hello').serialize()))['response']); expect(log, ['hello']); expect(result.message, '1'); }); }); group('getSemanticsId', () { FlutterDriverExtension extension; setUp(() { extension = new FlutterDriverExtension((String arg) async {}, true); }); testWidgets('works when semantics are enabled', (WidgetTester tester) async { final SemanticsHandle semantics = RendererBinding.instance.pipelineOwner.ensureSemantics(); await tester.pumpWidget( const Text('hello', textDirection: TextDirection.ltr)); final Map arguments = new GetSemanticsId(new ByText('hello')).serialize(); final GetSemanticsIdResult result = GetSemanticsIdResult.fromJson((await extension.call(arguments))['response']); expect(result.id, 1); semantics.dispose(); }); testWidgets('throws state error if no data is found', (WidgetTester tester) async { await tester.pumpWidget( const Text('hello', textDirection: TextDirection.ltr)); final Map arguments = new GetSemanticsId(new ByText('hello')).serialize(); final Map response = await extension.call(arguments); expect(response['isError'], true); expect(response['response'], contains('Bad state: No semantics data found')); }); testWidgets('throws state error multiple matches are found', (WidgetTester tester) async { final SemanticsHandle semantics = RendererBinding.instance.pipelineOwner.ensureSemantics(); await tester.pumpWidget( new Directionality( textDirection: TextDirection.ltr, child: new ListView(children: const [ SizedBox(width: 100.0, height: 100.0, child: Text('hello')), SizedBox(width: 100.0, height: 100.0, child: Text('hello')), ]), ), ); final Map arguments = new GetSemanticsId(new ByText('hello')).serialize(); final Map response = await extension.call(arguments); expect(response['isError'], true); expect(response['response'], contains('Bad state: Too many elements')); semantics.dispose(); }); }); }