Flutter【03】图片输出package依赖关系

2024-09-02 15:12

本文主要是介绍Flutter【03】图片输出package依赖关系,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

环境准备

安装 graphviz

arch -arm64 brew install graphviz

项目根目录pubspec.yaml文件内添加

dev_dependencies:
yaml: ^3.1.1
gviz: ^0.4.0

执行脚本

项目根目录下添加dart文件,运行main函数

import 'dart:io';
import 'dart:convert';
import 'package:yaml/yaml.dart' as yaml;
import 'package:gviz/gviz.dart';void main() async {final projectPath = await _getProjectPath();final file = File('$projectPath/pubspec.yaml');final fileContent = file.readAsStringSync();final yamlMap = yaml.loadYaml(fileContent) as yaml.YamlMap;final appName = yamlMap['name'].toString();print('开始 ...');final dependencyContent = await _getComponentDependencyTree(projectPath: projectPath,);print('... 开始遍历组件依赖节点');print(dependencyContent);final dependencyNodes = _traversalComponentDependencyTree(dependencyContent);print('... 完成遍历组件依赖节点');final graph = Gviz(name: appName,graphProperties: {'pad': '0.5','nodesep': '1','ranksep': '2',},edgeProperties: {'fontcolor': 'gray',},);print('... 开始转换 dot 节点');_generateDotByNodes(dependencyNodes,graph: graph,edgeCache: <String>[],);print('... 完成转换 dot 节点');final dotDirectoryPath = '$projectPath/dotGenerateDir';final dotDirectory = Directory(dotDirectoryPath);if (!dotDirectory.existsSync()) {await dotDirectory.create();print('... 创建 dotGenerate 文件夹');}final dotFileName = '$appName.dot';final dotPngName = '$appName.png';final dotFile = File('$dotDirectoryPath/$dotFileName');final dotPngFile = File('$dotDirectoryPath/$dotPngName');if (dotFile.existsSync()) {await dotFile.delete();print('... 删除原有 dot 生成文件');}if (dotPngFile.existsSync()) {await dotPngFile.delete();print('... 删除原有 dot 依赖关系图');}await dotFile.create();final dotResult = await dotFile.writeAsString(graph.toString());print('dot 文件生成成功: ${dotResult.path}');print('... 开始生成 dot png');await _runCommand(executable: 'dot',projectPath: projectPath,commandArgs: ['$dotDirectoryPath/$dotFileName','-T','png','-o','$dotDirectoryPath/$dotPngName'],);print('png 文件生成成功:$dotDirectoryPath/$dotPngName');await Process.run('open',[dotDirectoryPath],);
}const List<String> ignoreDependency = <String>['flutter','flutter_test','flutter_lints','cupertino_icons','gviz','yaml','injectable_generator','build_runner',
];Future<String> _getComponentDependencyTree({required String projectPath,
}) {return _runCommand(projectPath: projectPath,commandArgs: ['pub', 'deps', '--json'],).then((value) {if (value.contains('dependencies:') &&value.contains('dev dependencies:')) {final start = value.indexOf('dependencies:');final end = value.indexOf('dev dependencies:');return value.substring(start, end);} else {return value;}},);
}List<DependencyNode> _traversalComponentDependencyTree(String dependencyContent) {final dependencyJson = jsonDecode(dependencyContent) as Map<String, dynamic>;final packages = dependencyJson['packages'] as List<dynamic>;final nodeMap = <String, DependencyNode>{};for (var package in packages) {final node = DependencyNode.fromMap(package);nodeMap[node.name] = node;}final rootNode = nodeMap.values.firstWhere((element) => element.isRootNode);void mapDependencies(DependencyNode node, Set<String> visitedNodes) {if (visitedNodes.contains(node.name)) {return;}visitedNodes.add(node.name);for (final itemName in node.dependencies) {if (!ignoreDependency.contains(itemName)) {final itemNode = nodeMap[itemName];if (itemNode != null) {mapDependencies(itemNode, visitedNodes);node.children.add(itemNode);itemNode.isLevel1Node = false;}}}}final visitedNodes = <String>{};mapDependencies(rootNode, visitedNodes);// 使用新的 rebuildDependencyTree 函数来创建一个没有重复依赖的新树DependencyNode newRootNode = rebuildDependencyTree(rootNode, Set<String>());return [newRootNode];
}
DependencyNode rebuildDependencyTree(DependencyNode originalNode, Set<String> seenDependencies) {// 创建一个新的节点,复制原始节点的属性DependencyNode newNode = DependencyNode(name: originalNode.name,version: originalNode.version,kind: originalNode.kind,source: originalNode.source,dependencies: originalNode.dependencies,);newNode.isLevel1Node = originalNode.isLevel1Node;// 如果这个节点已经被处理过,直接返回新节点(没有子节点)if (seenDependencies.contains(newNode.name)) {return newNode;}// 将这个节点添加到已处理集合中seenDependencies.add(newNode.name);// 处理子节点for (var childNode in originalNode.children) {if (!ignoreDependency.contains(childNode.name)) {var newChildNode = rebuildDependencyTree(childNode, seenDependencies);newNode.children.add(newChildNode);}}return newNode;
}
Future<String> _getProjectPath() async {final originProjectPath = await Process.run('pwd',[],);final projectPath = (originProjectPath.stdout as String).replaceAll('\n','',);return projectPath;
}void _generateDotByNodes(List<DependencyNode> nodes, {required Gviz graph,required List<String> edgeCache,}) {if (nodes.isEmpty) {return;}for (int index = 0; index < nodes.length; index++) {final itemNode = nodes[index];final from = '${itemNode.name}\n${itemNode.version}';if (!graph.nodeExists(from)) {graph.addNode(from,properties: {'color': 'black','shape': 'rectangle','margin': '1,0.8','penwidth': '7','style': 'filled','fillcolor': 'gray','fontsize': itemNode.isLevel1Node ? '60' : '55',},);}final toArr = itemNode.children.map((e) => '${e.name}\n${e.version}').toList();for (var element in toArr) {final edgeKey = '$from-$element';if (!edgeCache.contains(edgeKey)) {graph.addEdge(from,element,properties: {'penwidth': '2','style': 'dashed','arrowed': 'vee',},);edgeCache.add(edgeKey);}}_generateDotByNodes(itemNode.children,graph: graph,edgeCache: edgeCache,);}
}Future<String> _runCommand({String executable = 'flutter',required String projectPath,required List<String> commandArgs,
}) {return Process.run(executable,commandArgs,runInShell: true,workingDirectory: projectPath,).then((result) => result.stdout as String);
}class DependencyNode {final String name;final String version;final String kind;final String source;final List<String> dependencies;final children = <DependencyNode>[];bool isLevel1Node = true;factory DependencyNode.fromMap(Map<String, dynamic> map) {return DependencyNode(name: map['name'] as String,version: map['version'] as String,kind: map['kind'] as String,source: map['source'] as String,dependencies: (map['dependencies'] as List<dynamic>).map((e) => e as String).toList(),);}bool get isRootNode => kind == 'root';bool operator ==(Object other) =>identical(this, other) ||other is DependencyNode &&runtimeType == other.runtimeType &&name == other.name;int get hashCode => name.hashCode;DependencyNode({required this.name,required this.version,required this.kind,required this.source,required this.dependencies,});
}

输出示例


这篇关于Flutter【03】图片输出package依赖关系的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1130305

相关文章

Java使用Spire.Doc for Java实现Word自动化插入图片

《Java使用Spire.DocforJava实现Word自动化插入图片》在日常工作中,Word文档是不可或缺的工具,而图片作为信息传达的重要载体,其在文档中的插入与布局显得尤为关键,下面我们就来... 目录1. Spire.Doc for Java库介绍与安装2. 使用特定的环绕方式插入图片3. 在指定位

Java中自旋锁与CAS机制的深层关系与区别

《Java中自旋锁与CAS机制的深层关系与区别》CAS算法即比较并替换,是一种实现并发编程时常用到的算法,Java并发包中的很多类都使用了CAS算法,:本文主要介绍Java中自旋锁与CAS机制深层... 目录1. 引言2. 比较并交换 (Compare-and-Swap, CAS) 核心原理2.1 CAS

Python多任务爬虫实现爬取图片和GDP数据

《Python多任务爬虫实现爬取图片和GDP数据》本文主要介绍了基于FastAPI开发Web站点的方法,包括搭建Web服务器、处理图片资源、实现多任务爬虫和数据可视化,同时,还简要介绍了Python爬... 目录一. 基于FastAPI之Web站点开发1. 基于FastAPI搭建Web服务器2. Web服务

利用Python将PDF文件转换为PNG图片的代码示例

《利用Python将PDF文件转换为PNG图片的代码示例》在日常工作和开发中,我们经常需要处理各种文档格式,PDF作为一种通用且跨平台的文档格式,被广泛应用于合同、报告、电子书等场景,然而,有时我们需... 目录引言为什么选择 python 进行 PDF 转 PNG?Spire.PDF for Python

Python版本与package版本兼容性检查方法总结

《Python版本与package版本兼容性检查方法总结》:本文主要介绍Python版本与package版本兼容性检查方法的相关资料,文中提供四种检查方法,分别是pip查询、conda管理、PyP... 目录引言为什么会出现兼容性问题方法一:用 pip 官方命令查询可用版本方法二:conda 管理包环境方法

uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)

《uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)》在uni-app开发中,文件上传和图片处理是很常见的需求,但也经常会遇到各种问题,下面:本文主要介绍uni-app小程序项目中实... 目录方式一:使用<canvas>实现图片压缩(推荐,兼容性好)示例代码(小程序平台):方式二:使用uni

python依赖管理工具UV的安装和使用教程

《python依赖管理工具UV的安装和使用教程》UV是一个用Rust编写的Python包安装和依赖管理工具,比传统工具(如pip)有着更快、更高效的体验,:本文主要介绍python依赖管理工具UV... 目录前言一、命令安装uv二、手动编译安装2.1在archlinux安装uv的依赖工具2.2从github

Java 中的 equals 和 hashCode 方法关系与正确重写实践案例

《Java中的equals和hashCode方法关系与正确重写实践案例》在Java中,equals和hashCode方法是Object类的核心方法,广泛用于对象比较和哈希集合(如HashMa... 目录一、背景与需求分析1.1 equals 和 hashCode 的背景1.2 需求分析1.3 技术挑战1.4

Android实现图片浏览功能的示例详解(附带源码)

《Android实现图片浏览功能的示例详解(附带源码)》在许多应用中,都需要展示图片并支持用户进行浏览,本文主要为大家介绍了如何通过Android实现图片浏览功能,感兴趣的小伙伴可以跟随小编一起学习一... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

javacv依赖太大导致jar包也大的解决办法

《javacv依赖太大导致jar包也大的解决办法》随着项目的复杂度和依赖关系的增加,打包后的JAR包可能会变得很大,:本文主要介绍javacv依赖太大导致jar包也大的解决办法,文中通过代码介绍的... 目录前言1.检查依赖2.更改依赖3.检查副依赖总结 前言最近在写项目时,用到了Javacv里的获取视频