flutter 雷达图

通过CustomPainter自定义雷达图

效果如下

 主要代码

import 'package:flutter/material.dart';
import 'dart:math';
import 'dash_painter.dart';
import 'model/charts_model.dart';class RadarChart extends StatelessWidget {final List<ChartModel> list;final double maxValue;final Color radarColor;final Color dataColor;const RadarChart({super.key,required this.list,this.maxValue = 100,this.radarColor = Colors.grey,this.dataColor = Colors.green,});@overrideWidget build(BuildContext context) {return CustomPaint(painter: RadarChartPainter(list: list,maxValue: maxValue,radarColor: radarColor,dataColor: dataColor,numLayers: 4,),);}
}class RadarChartPainter extends CustomPainter {final int numLayers;final List<ChartModel> list;final double maxValue;final Color radarColor;final Color dataColor;List<Offset> startList = []; //存放第一层的点List<Offset> endList = []; //存放最外层的点List<Offset> textOffsetList = [];final int _offsetDy = 20;final int _offsetDx = 5;RadarChartPainter({required this.numLayers,required this.list,required this.maxValue,required this.radarColor,required this.dataColor,});@overridevoid paint(Canvas canvas, Size size) {final center = Offset(size.width / 2, size.height / 2);final radius = size.width / 2;final dataPoints = <Offset>[];final radarPaint = Paint()..color = radarColor..style = PaintingStyle.stroke;final bgPaint = Paint()..color = const Color(0xFFCAD0E8).withOpacity(.4)..style = PaintingStyle.fill;//画背景颜色for (var layer = 1; layer <= numLayers; layer++) {if (layer == 3) {final layerRadius = radius * (layer / numLayers);final radarPath = Path();for (var i = 0; i < 6; i++) {//每一层的6个点final angle = (2 * pi / 6) * i - (pi / 2);final x = center.dx + layerRadius * cos(angle);final y = center.dy + layerRadius * sin(angle);final point = Offset(x, y);if (i == 0) {radarPath.moveTo(point.dx, point.dy);} else {radarPath.lineTo(point.dx, point.dy);}}radarPath.close();canvas.drawPath(radarPath, bgPaint);}}//连接每一层的6个点和文字for (var layer = 1; layer <= numLayers; layer++) {final layerRadius = radius * (layer / numLayers);final radarPath = Path();for (var i = 0; i < 6; i++) {final angle = (2 * pi / 6) * i - (pi / 2);final x = center.dx + layerRadius * cos(angle);final y = center.dy + layerRadius * sin(angle);final point = Offset(x, y);if (i == 0) {radarPath.moveTo(point.dx, point.dy);} else {radarPath.lineTo(point.dx, point.dy);}//保存第一层的点if (layer == 1) {startList.add(point);}//保存最外层的点,用于后面画第一层的最外层点的连线if (layer == 4) {endList.add(point);//画label文字const textStyle = TextStyle(fontSize: 12, fontWeight: FontWeight.bold, color: Color(0XFF999999));final textSpan = TextSpan(text: list[i].label, style: textStyle);final textPainter = TextPainter(text: textSpan,textDirection: TextDirection.ltr,);textPainter.layout();double w = textPainter.width;Offset off = const Offset(0, 0);if (i == 0) {off = Offset(point.dx - w * 0.5, point.dy - _offsetDy);} else if (i == 1 || i == 2) {off = Offset(point.dx + _offsetDx, point.dy);} else if (i == 4 || i == 5) {off = Offset(point.dx - _offsetDx - w, point.dy);} else if (i == 3) {off = Offset(point.dx - w * 0.5, point.dy);}textPainter.paint(canvas, off);}}radarPath.close();const DashPainter(span: 3, step: 3).paint(canvas, radarPath, radarPaint);}//画第一层的点到最外层的点的连线for (var i = 0; i < 6; i++) {final path = Path();path.moveTo(startList[i].dx, startList[i].dy);path.lineTo(endList[i].dx, endList[i].dy);const DashPainter(span: 4, step: 9).paint(canvas, path, radarPaint);}//画数据区域final dataPaint = Paint()..color = dataColor..style = PaintingStyle.stroke..strokeWidth = 2.0;final dataFillPaint = Paint()..color = dataColor.withOpacity(.3)..style = PaintingStyle.fill..strokeWidth = 2.0;final dataPath = Path();for (var i = 0; i < 6; i++) {final angle = (2 * pi / 6) * i - (pi / 2);final value = list[i].y;final normalizedValue = value / maxValue;final dataRadius = radius * 0.75 * normalizedValue + radius * 0.25;final x = center.dx + dataRadius * cos(angle);final y = center.dy + dataRadius * sin(angle);final point = Offset(x, y);if (i == 0) {dataPath.moveTo(point.dx, point.dy);} else {dataPath.lineTo(point.dx, point.dy);}dataPoints.add(point);}dataPath.close();canvas.drawPath(dataPath, dataFillPaint);canvas.drawPath(dataPath, dataPaint);final dataPointPaint = Paint()..color = dataColor;for (var point in dataPoints) {canvas.drawCircle(point, 4.0, dataPointPaint);}}@overridebool shouldRepaint(covariant CustomPainter oldDelegate) {return true;}
}

项目地址 : flutter_radar: flutter 雷达图

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/51388.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Flutter系列文章-Flutter 插件开发

在本篇文章中&#xff0c;我们将学习如何开发 Flutter 插件&#xff0c;实现 Flutter 与原生平台的交互。我们将详细介绍插件的开发过程&#xff0c;包括如何创建插件项目、实现方法通信、处理异步任务等。最后&#xff0c;我们还将演示如何将插件打包并发布到 Flutter 社区。 …

Attention is all you need 官方 tensorflow 1.x 实现

https://github.com/tensorflow/tensor2tensor/blob/master/tensor2tensor/models/transformer.py 1&#xff0c;搭建cuda10.0环境&#xff0c; 2&#xff0c;安装tensorflow 1.14.0 3&#xff0c;安装python3的 tensor2tensor 包 4&#xff0c;示例代码&#xff1a; impor…

5.物联网LWIP之UDP编程,stm32作为服务器实现大小写转化

UDP编程模型 1.UDP C/S模型&#xff08;代码流程只需要根据以下模型去输入即可&#xff09; 2.UDP API socket int socket(int domain, int type, int protocol); domain: AF_INET 这是大多数用来产生socket的协议&#xff0c;使用TCP或UDP来传输&#xff0c;用IPv4的地址…

从0开始配置eslint

没有在.eslintrc文件中配置parserOptions指定语言版本和模块类型 {"parserOptions": {"ecmaVersion": 7, //指定es版本为es2016"sourceType": "module", //使用import导入模块} }eslint还不能识别jsx语法 {"parserOptions"…

网站是如何识别网络爬虫的?

在爬取数据时&#xff0c;你常常会遇到各种网站的反爬机制。网站是如何检测和拦截网络爬虫的呢&#xff1f;本文将为你揭秘网站使用的几种常见的反爬手段&#xff0c;并为你提供一些解决方案&#xff0c;助你越过反爬壁垒&#xff0c;提升你的实际操作效率。 一、Cookie检测 …

centos7物理机安装并配置外网访问

安装准备工作 安装之前需要准备一下&#xff0c;需要一个U盘&#xff0c;其次需要准备以下内容 1.需要centos7的ISO系统镜像 2.使用UltraISO软件写入ISO镜像 3.一台windows系统 将系统写入到U盘&#xff0c;写入步骤 打开UltraISO点击文件 → 打开&#xff0c;选择Linux镜…

Go 语言入门指南:基础语法和常用特性解析

变量 goCopy code package mainimport ("fmt""math" )func main() {// 声明并初始化变量var a "initial"// 声明并初始化多个变量var b, c int 1, 2// 声明并初始化变量&#xff0c;并推断其类型var d true// 声明变量&#xff0c;但没有初始…

告别数字化系统“物理叠加”,华为云推动智慧门店价值跃迁

文|智能相对论 作者|叶远风 有大屏幕滚动播放广告&#xff1b; 有人脸识别系统让消费者自助结账&#xff1b; 有订单管理系统综合分析一段时间内总体经营情况&#xff1b; 有全门店监控直连总部机房&#xff1b; …… 以搭载数字化系统的硬件设备为表面特征的智慧门店&a…

更改Docker的存储位置

要更改Docker的存储位置&#xff0c;可以按照以下步骤进行操作 文章目录 停止Docker服务编辑Docker配置文件 /etc/docker/daemon.json保存并关闭配置文件,并服务程序配置文件生效移动现有的Docker数据目录到新的存储路径更新文件权限以确保Docker可以访问新的数据目录启动Dock…

Windows快速关闭占用某端口的进程

在Windows操作系统中&#xff0c;你可以使用以下方法来快速关闭占用特定端口的进程&#xff0c;比如端口8848&#xff1a; 使用命令行工具 打开命令提示符&#xff08;Command Prompt&#xff09;。 输入以下命令&#xff0c;将端口号替换为你需要关闭的端口&#xff08;例如…

Java项目时间字段问题-MySQL

Java项目时间字段问题-MySQL ​ 在 Java 项目中&#xff0c;与 MySQL 数据库中的时间字段对应的 Java 类型通常使用 java.time 包中的类&#xff0c;这是从 Java 8 开始引入的日期和时间 API。以下是常见的时间字段和它们在 Java 实体类中的对应类型示例&#xff1a; 1、DATE…

前端面试:【性能优化】前端缓存、CDN、懒加载和预加载

亲爱的前端开发者&#xff0c;Web性能对用户体验至关重要。如果你想让你的网站更快、更具吸引力&#xff0c;就需要关注前端性能优化。在这篇文章中&#xff0c;我们将深入探讨四个关键的性能优化策略&#xff1a;前端缓存、CDN&#xff08;内容分发网络&#xff09;、懒加载和…

按软件开发阶段的角度划分:单元测试、集成测试、系统测试、验收测试

1.单元测试&#xff08;Unit Testing&#xff09; 单元测试&#xff0c;又称模块测试。对软件的组成单位进行测试&#xff0c;其目的是检验软件基本组成单位的正确性。测试的对象是软件里测试的最小单位&#xff1a;模块。 测试阶段&#xff1a;编码后或者编码前&#xff08;…

Unity OnDrawGizmos的简单应用 绘制圆形

编辑器和配置表各有各的好。 卡牌游戏即使再复杂&#xff0c;哪怕是梦幻西游&#xff0c;大话西游那种&#xff0c;甚至wow那种&#xff0c;用配表都完全没问题。但是崩坏3&#xff0c;或者鬼泣&#xff0c;格斗游戏&#xff0c;可视化编辑器是唯一的选择。 开发初期刚开始配技…

人工智能技术

人工智能技术是什么&#xff1f; 人工智能技术&#xff08;Artificial Intelligence Technology&#xff0c;AI技术&#xff09;是一种模仿人类智能和思维方式的计算机技术&#xff0c;旨在使计算机能够执行需要人类智能才能完成的任务。这些任务包括理解自然语言、解决问题、…

高手进阶之路---pyqt自定义信号

高手进阶之路—pyqt自定义信号 1.思考问题为什么要自定义信号&#xff0c;qt5本身已有信号槽函数 # pushButton 被clicked的时候connect 函数print self.pushButton.clicked.connect(self.print)def print(self):print("我被点击了")或者使用 # 需要引入 pyqtSlo…

ubuntu20.04 编译安装运行emqx

文章目录 安装依赖编译运行登录dashboard压力测试 安装依赖 Erlang/OTP OTP 24 或 25 版本 apt-get install libncurses5-dev sudo apt-get install erlang如果安装的erlang版本小于24的话&#xff0c;可以使用如下方法自行编译erlang 1.源码获取 wget https://github.com/erla…

微服务中间件--多级缓存

多级缓存 多级缓存a.JVM进程缓存1) Caffeine2) 案例 b.Lua语法1) 变量和循环2) 条件控制、函数 c.多级缓存1) 安装OpenResty2) 请求参数处理3) 查询Tomcat4) Redis缓存预热5) 查询Redis缓存6) Nginx本地缓存 d.缓存同步1) 数据同步策略2) 安装Canal2.a) 开启MySQL主从2.b) 安装…

mysql(八)事务隔离级别及加锁流程详解

目录 MySQL 锁简介什么是锁锁的作用锁的种类共享排他锁共享锁排它锁 粒度锁全局锁表级锁页级锁行级锁种类 意向锁间隙临键记录锁记录锁间隙锁 加锁的流程锁的内存结构加锁的基本流程根据主键加锁根据二级索引加锁根据非索引字段查询加锁加锁规律 锁信息查看查看锁的sql语句 数据…

龙迅LT7911UX TYPE-C/DP转MIPI/LVDS,内有HDCP

1. 描述 LT7911UX是一种高性能的Type-C/DP1.4a到MIPI或LVDS芯片。HDCP RX作为HDCP中继器的上游端&#xff0c;可以与其他芯片的HDCP TX协同工作&#xff0c;实现中继器的功能。 对于DP1.4a输入&#xff0c;LT7911UX可以配置为1/2/4车道。自适应均衡使其适用于长电缆应用&#…