| // 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 'dart:async'; |
| import 'dart:convert' show base64Decode; |
| import 'dart:developer' as developer; |
| import 'dart:ui'; |
| |
| import 'package:litetest/litetest.dart'; |
| import 'package:vm_service/vm_service.dart' as vms; |
| import 'package:vm_service/vm_service_io.dart'; |
| import 'package:vm_service_protos/vm_service_protos.dart'; |
| |
| import '../impeller_enabled.dart'; |
| |
| Future<void> _testChromeFormatTrace(vms.VmService vmService) async { |
| final vms.Timeline timeline = await vmService.getVMTimeline(); |
| |
| int saveLayerRecordCount = 0; |
| int saveLayerCount = 0; |
| int flowEventCount = 0; |
| for (final vms.TimelineEvent event in timeline.traceEvents!) { |
| final Map<String, dynamic> json = event.json!; |
| if (json['ph'] == 'B') { |
| if (json['name'] == 'ui.Canvas::saveLayer (Recorded)') { |
| saveLayerRecordCount += 1; |
| } |
| if (json['name'] == 'Canvas::saveLayer') { |
| saveLayerCount += 1; |
| } |
| } else if (json['ph'] == 's' || json['ph'] == 't' || json['ph'] == 'f') { |
| flowEventCount += 1; |
| } |
| } |
| expect(saveLayerRecordCount, 3); |
| expect(saveLayerCount, impellerEnabled ? 2 : 3); |
| expect(flowEventCount, 5); |
| } |
| |
| Future<void> _testPerfettoFormatTrace(vms.VmService vmService) async { |
| final vms.PerfettoTimeline response = await vmService.getPerfettoVMTimeline(); |
| final List<TracePacket> packets = |
| Trace.fromBuffer(base64Decode(response.trace!)).packet; |
| final Iterable<TrackEvent> events = packets |
| .where((TracePacket packet) => packet.hasTrackEvent()) |
| .map((TracePacket packet) => packet.trackEvent); |
| |
| int saveLayerRecordCount = 0; |
| int saveLayerCount = 0; |
| int flowIdCount = 0; |
| for (final TrackEvent event in events) { |
| if (event.type == TrackEvent_Type.TYPE_SLICE_BEGIN) { |
| if (event.name == 'ui.Canvas::saveLayer (Recorded)') { |
| saveLayerRecordCount += 1; |
| } |
| if (event.name == 'Canvas::saveLayer') { |
| saveLayerCount += 1; |
| } |
| flowIdCount += event.flowIds.length; |
| } |
| } |
| expect(saveLayerRecordCount, 3); |
| expect(saveLayerCount, impellerEnabled ? 2 : 3); |
| expect(flowIdCount, 5); |
| } |
| |
| void main() { |
| test('Canvas.saveLayer emits tracing', () async { |
| final developer.ServiceProtocolInfo info = await developer.Service.getInfo(); |
| |
| if (info.serverUri == null) { |
| fail('This test must not be run with --disable-vm-service.'); |
| } |
| |
| final vms.VmService vmService = await vmServiceConnectUri( |
| 'ws://localhost:${info.serverUri!.port}${info.serverUri!.path}ws', |
| ); |
| |
| final Completer<void> completer = Completer<void>(); |
| PlatformDispatcher.instance.onBeginFrame = (Duration timeStamp) async { |
| final PictureRecorder recorder = PictureRecorder(); |
| final Canvas canvas = Canvas(recorder); |
| canvas.drawColor(const Color(0xff0000ff), BlendMode.srcOut); |
| // Will saveLayer implicitly for Skia, but not Impeller. |
| canvas.drawPaint(Paint()..imageFilter = ImageFilter.blur(sigmaX: 2, sigmaY: 3)); |
| canvas.saveLayer(null, Paint()); |
| canvas.drawRect(const Rect.fromLTRB(10, 10, 20, 20), Paint()); |
| canvas.saveLayer(const Rect.fromLTWH(0, 0, 100, 100), Paint()); |
| canvas.drawRect(const Rect.fromLTRB(10, 10, 20, 20), Paint()); |
| canvas.drawRect(const Rect.fromLTRB(15, 15, 25, 25), Paint()); |
| canvas.restore(); |
| canvas.restore(); |
| final Picture picture = recorder.endRecording(); |
| |
| final SceneBuilder builder = SceneBuilder(); |
| builder.addPicture(Offset.zero, picture); |
| final Scene scene = builder.build(); |
| |
| await scene.toImage(100, 100); |
| scene.dispose(); |
| completer.complete(); |
| }; |
| PlatformDispatcher.instance.scheduleFrame(); |
| await completer.future; |
| |
| await _testChromeFormatTrace(vmService); |
| await _testPerfettoFormatTrace(vmService); |
| await vmService.dispose(); |
| }); |
| } |