// 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 'dart:async'; import 'dart:convert'; import '../base/common.dart'; import '../base/file_system.dart'; import '../base/utils.dart'; import '../device.dart'; import '../globals.dart'; import '../runner/flutter_command.dart'; import '../vmservice.dart'; const String _kOut = 'out'; const String _kType = 'type'; const String _kObservatoryPort = 'observatory-port'; const String _kDeviceType = 'device'; const String _kSkiaType = 'skia'; const String _kRasterizerType = 'rasterizer'; class ScreenshotCommand extends FlutterCommand { ScreenshotCommand() { argParser.addOption( _kOut, abbr: 'o', valueHelp: 'path/to/file', help: 'Location to write the screenshot.', ); argParser.addOption( _kObservatoryPort, valueHelp: 'port', help: 'The observatory port to connect to.\n' 'This is required when --$_kType is "$_kSkiaType" or "$_kRasterizerType".\n' 'To find the observatory port number, use "flutter run --verbose"\n' 'and look for "Forwarded host port ... for Observatory" in the output.', ); argParser.addOption( _kType, valueHelp: 'type', help: 'The type of screenshot to retrieve.', allowed: const <String>[_kDeviceType, _kSkiaType, _kRasterizerType], allowedHelp: const <String, String>{ _kDeviceType: 'Delegate to the device\'s native screenshot capabilities. This\n' 'screenshots the entire screen currently being displayed (including content\n' 'not rendered by Flutter, like the device status bar).', _kSkiaType: 'Render the Flutter app as a Skia picture. Requires --$_kObservatoryPort', _kRasterizerType: 'Render the Flutter app using the rasterizer. Requires --$_kObservatoryPort', }, defaultsTo: _kDeviceType, ); } @override String get name => 'screenshot'; @override String get description => 'Take a screenshot from a connected device.'; @override final List<String> aliases = <String>['pic']; Device device; @override Future<Null> verifyThenRunCommand() async { device = await findTargetDevice(); if (device == null) throwToolExit('Must have a connected device'); if (argResults[_kType] == _kDeviceType && !device.supportsScreenshot) throwToolExit('Screenshot not supported for ${device.name}.'); if (argResults[_kType] != _kDeviceType && argResults[_kObservatoryPort] == null) throwToolExit('Observatory port must be specified for screenshot type ${argResults[_kType]}'); return super.verifyThenRunCommand(); } @override Future<Null> runCommand() async { File outputFile; if (argResults.wasParsed(_kOut)) outputFile = fs.file(argResults[_kOut]); switch (argResults[_kType]) { case _kDeviceType: return runScreenshot(outputFile); case _kSkiaType: return runSkia(outputFile); case _kRasterizerType: return runRasterizer(outputFile); } } Future<Null> runScreenshot(File outputFile) async { outputFile ??= getUniqueFile(fs.currentDirectory, 'flutter', 'png'); try { await device.takeScreenshot(outputFile); } catch (error) { throwToolExit('Error taking screenshot: $error'); } await showOutputFileInfo(outputFile); } Future<Null> runSkia(File outputFile) async { final Map<String, dynamic> skp = await _invokeVmServiceRpc('_flutter.screenshotSkp'); outputFile ??= getUniqueFile(fs.currentDirectory, 'flutter', 'skp'); final IOSink sink = outputFile.openWrite(); sink.add(base64.decode(skp['skp'])); await sink.close(); await showOutputFileInfo(outputFile); await _ensureOutputIsNotJsonRpcError(outputFile); } Future<void> runRasterizer(File outputFile) async { final Map<String, dynamic> response = await _invokeVmServiceRpc('_flutter.screenshot'); outputFile ??= getUniqueFile(fs.currentDirectory, 'flutter', 'png'); final IOSink sink = outputFile.openWrite(); sink.add(base64.decode(response['screenshot'])); await sink.close(); await showOutputFileInfo(outputFile); await _ensureOutputIsNotJsonRpcError(outputFile); } Future<Map<String, dynamic>> _invokeVmServiceRpc(String method) async { final Uri observatoryUri = new Uri(scheme: 'http', host: '127.0.0.1', port: int.parse(argResults[_kObservatoryPort])); final VMService vmService = await VMService.connect(observatoryUri); return await vmService.vm.invokeRpcRaw(method); } Future<void> _ensureOutputIsNotJsonRpcError(File outputFile) async { if (await outputFile.length() < 1000) { final String content = await outputFile.readAsString( encoding: const AsciiCodec(allowInvalid: true), ); if (content.startsWith('{"jsonrpc":"2.0", "error"')) throwToolExit('\nIt appears the output file contains an error message, not valid skia output.'); } } Future<Null> showOutputFileInfo(File outputFile) async { final int sizeKB = (await outputFile.length()) ~/ 1024; printStatus('Screenshot written to ${fs.path.relative(outputFile.path)} (${sizeKB}kB).'); } }