// Copyright 2015 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. library sky_tools.build; import 'dart:async'; import 'dart:io'; import 'package:archive/archive.dart'; import 'package:args/args.dart'; import 'package:yaml/yaml.dart'; import 'artifacts.dart'; import 'common.dart'; const String _kSnapshotKey = 'snapshot_blob.bin'; const List _kDensities = const ['drawable-xxhdpi']; const List _kThemes = const ['white', 'black']; const List _kSizes = const [24]; class _Asset { final String base; final String key; _Asset({ this.base, this.key }); } Iterable<_Asset> _parseAssets(Map manifestDescriptor, String manifestPath) sync* { if (manifestDescriptor == null || !manifestDescriptor.containsKey('assets')) return; String basePath = new File(manifestPath).parent.path; for (String asset in manifestDescriptor['assets']) yield new _Asset(base: basePath, key: asset); } class _MaterialAsset { final String name; final String density; final String theme; final int size; _MaterialAsset(Map descriptor) : name = descriptor['name'], density = descriptor['density'], theme = descriptor['theme'], size = descriptor['size']; String get key { List parts = name.split('/'); String category = parts[0]; String subtype = parts[1]; return '$category/$density/ic_${subtype}_${theme}_${size}dp.png'; } } List _generateValues(Map assetDescriptor, String key, List defaults) { if (assetDescriptor.containsKey(key)) return [assetDescriptor[key]]; return defaults; } Iterable<_MaterialAsset> _generateMaterialAssets(Map assetDescriptor) sync* { Map currentAssetDescriptor = new Map.from(assetDescriptor); for (String density in _generateValues(assetDescriptor, 'density', _kDensities)) { currentAssetDescriptor['density'] = density; for (String theme in _generateValues(assetDescriptor, 'theme', _kThemes)) { currentAssetDescriptor['theme'] = theme; for (int size in _generateValues(assetDescriptor, 'size', _kSizes)) { currentAssetDescriptor['size'] = size; yield new _MaterialAsset(currentAssetDescriptor); } } } } Iterable<_MaterialAsset> _parseMaterialAssets(Map manifestDescriptor) sync* { if (manifestDescriptor == null || !manifestDescriptor.containsKey('material-design-icons')) return; for (Map assetDescriptor in manifestDescriptor['material-design-icons']) { for (_MaterialAsset asset in _generateMaterialAssets(assetDescriptor)) { yield asset; } } } Future _loadManifest(String manifestPath) async { if (manifestPath == null) return null; String manifestDescriptor = await new File(manifestPath).readAsString(); return loadYaml(manifestDescriptor); } Future _createFile(String key, String assetBase) async { File file = new File('${assetBase}/${key}'); if (!await file.exists()) return null; List content = await file.readAsBytes(); return new ArchiveFile.noCompress(key, content.length, content); } Future _compileSnapshot({ String mainPath, String packageRoot, String snapshotPath }) async { File compiler = await artifactStore.getPath(Artifact.FlutterCompiler); await Process.run(compiler.path, [ mainPath, '--package-root=$packageRoot', '--snapshot=$snapshotPath' ]); } Future _createSnapshotFile(String snapshotPath) async { File file = new File(snapshotPath); List content = await file.readAsBytes(); return new ArchiveFile(_kSnapshotKey, content.length, content); } class BuildCommandHandler extends CommandHandler { BuildCommandHandler() : super('build', 'Create a Flutter package.'); ArgParser get parser { ArgParser parser = new ArgParser(); parser.addFlag('help', abbr: 'h', negatable: false); parser.addOption('asset-base', defaultsTo: 'packages/material_design_icons/icons'); parser.addOption('main', defaultsTo: 'lib/main.dart'); parser.addOption('manifest', defaultsTo: 'flutter.yaml'); parser.addOption('output-file', abbr: 'o', defaultsTo: 'app.flx'); parser.addOption('package-root', defaultsTo: 'packages'); parser.addOption('snapshot', defaultsTo: 'snapshot_blob.bin'); return parser; } @override Future processArgResults(ArgResults results) async { if (results['help']) { print(parser.usage); return 0; } String manifestPath = results['manifest']; Map manifestDescriptor = await _loadManifest(manifestPath); Iterable<_Asset> assets = _parseAssets(manifestDescriptor, manifestPath); Iterable<_MaterialAsset> materialAssets = _parseMaterialAssets(manifestDescriptor); Archive archive = new Archive(); String snapshotPath = results['snapshot']; await _compileSnapshot( mainPath: results['main'], packageRoot: results['package-root'], snapshotPath: snapshotPath); archive.addFile(await _createSnapshotFile(snapshotPath)); for (_Asset asset in assets) archive.addFile(await _createFile(asset.key, asset.base)); for (_MaterialAsset asset in materialAssets) { ArchiveFile file = await _createFile(asset.key, results['asset-base']); if (file != null) archive.addFile(file); } File outputFile = new File(results['output-file']); await outputFile.writeAsBytes(new ZipEncoder().encode(archive)); return 0; } }