Flutter实现渐变弧形进度条的示例详解

 更新时间:2023年12月21日 16:29:57   作者:Sinyu1012  
在Flutter开发中,构建一个具有视觉吸引力的、反映进度的圆形弧形进度条是一个常见需求,本文将详细介绍如何使用Flutter和Dart语言实现这一功能,需要的可以参考下
(福利推荐:【腾讯云】服务器最新限时优惠活动,云服务器1核2G仅99元/年、2核4G仅768元/3年,立即抢购>>>:9i0i.cn/qcloud

(福利推荐:你还在原价购买阿里云服务器?现在阿里云0.8折限时抢购活动来啦!4核8G企业云服务器仅2998元/3年,立即抢购>>>:9i0i.cn/aliyun

在Flutter开发中,构建一个具有视觉吸引力的、反映进度的圆形弧形进度条是一个常见需求。本文将详细介绍如何使用Flutter和Dart语言实现这一功能。最终效果如图:

首先,我们需要导入必要的包和库,比如dart:mathpackage:flutter/material.dart。这些库为绘制和样式提供基础支持。

接下来,创建一个ArcProgressPainter类,它继承自CustomPainter。这个类的核心是paint方法,用于绘制进度条。我们使用Canvas对象和Size对象来确定绘制区域,并利用数学运算确定圆心、半径等参数。

此外,文章还将展示如何使用线性渐变(LinearGradient)来美化进度条,以及如何计算角度和绘制圆弧。这包括如何根据进度动态变化圆弧的颜色和位置。

最后,我们将创建一个ArcProgressBar组件,它包装了CustomPaint,并使用上面定义的ArcProgressPainter来实现视觉效果。

整个过程不仅涉及基础的Flutter绘图技术,还包含一些高级的定制化元素,如颜色计算和动态布局调整。通过本文,读者可以学习如何灵活运用Flutter框架的绘图能力,为自己的应用程序添加独特且富有表现力的UI组件。

完整代码如下:

import 'dart:math' as math;
import 'dart:math';

import 'package:flutter/material.dart';
import 'package:otterlife/component/theme/extension.dart';

class ArcProgressPainter extends CustomPainter {
  final double progress;
  final Color backgroundColor;
  final double strokeWidth;
  final TextStyle textStyle;

  ArcProgressPainter({
    required this.progress,
    required this.backgroundColor,
    required this.strokeWidth,
    required this.textStyle,
  });

  @override
  void paint(Canvas canvas, Size size) {
    final gradientColors = [const Color(0xFFFFC75A), const Color(0xFF6DAFF9), const Color(0xFF31A7AE)];
    final gradient = LinearGradient(
      begin: Alignment.centerLeft,
      end: Alignment.centerRight,
      colors: gradientColors,
    );

    Offset center = Offset(size.width / 2, size.height / 2);
    double radius = math.min(size.width / 2, size.height / 2);

    Rect rect = Rect.fromCircle(center: center, radius: radius).inflate(-strokeWidth / 2);

    double degreesToRadians(num deg) => deg * (math.pi / 180.0);
    double startAngle = degreesToRadians(90 + 40);
    double sweepAngle = degreesToRadians(360 - 80);

    for (double i = 0; i < sweepAngle; i += 0.01) {
      double angle = startAngle + i;
      double colorPosition = i / sweepAngle;
      Color color = _calculateGradientColor(gradientColors, colorPosition);

      Paint segmentPaint = Paint()
        ..color = color
        ..strokeWidth = strokeWidth
        ..strokeCap = StrokeCap.round
        ..style = PaintingStyle.stroke;

      canvas.drawArc(
        rect,
        angle,
        0.01, // 绘制小段的角度
        false,
        segmentPaint,
      );
    }

    double sliderAngle = startAngle + progress * sweepAngle;
    Offset sliderPosition = Offset(
      center.dx + (radius - strokeWidth / 2) * cos(sliderAngle),
      center.dy + (radius - strokeWidth / 2) * sin(sliderAngle),
    );

    double sliderRadius = 28 / 2;
    Paint sliderPaint = Paint()..color = _calculateSliderColor(progress); // Assuming you have this method

    canvas.drawCircle(sliderPosition, sliderRadius, sliderPaint);

    Paint whiteCenterPaint = Paint()..color = Colors.white;
    canvas.drawCircle(sliderPosition, 16 / 2, whiteCenterPaint);
  }

  Color _calculateGradientColor(List<Color> colors, double position) {
    int index = (position * (colors.length - 1)).floor();
    double localPosition = (position * (colors.length - 1)) - index;
    return Color.lerp(colors[index], colors[index + 1], localPosition) ?? colors.last;
  }

  Color _calculateSliderColor(double progress) {
    final colors = [const Color(0xFFFFC75A), const Color(0xFF6DAFF9), const Color(0xFF31A7AE)];

    progress = progress.clamp(0.0, 1.0);

    double colorPosition = progress * (colors.length - 1);
    int index = colorPosition.floor();
    int nextIndex = (index + 1).clamp(0, colors.length - 1);

    double t = colorPosition - index;

    return Color.lerp(colors[index], colors[nextIndex], t) ?? colors.first;
  }

  double convertRadiusToSigma(double radius) {
    return radius * 0.57735 + 0.5;
  }

  @override
  bool shouldRepaint(covariant CustomPainter oldDelegate) => true;
}

class ArcProgressBar extends StatelessWidget {
  final double progress;
  final double strokeWidth;

  const ArcProgressBar({
    super.key,
    required this.progress,
    this.strokeWidth = 16,
  });

  @override
  Widget build(BuildContext context) {
    return CustomPaint(
      painter: ArcProgressPainter(
        progress: progress,
        backgroundColor: Colors.red,
        strokeWidth: strokeWidth,
        textStyle: Colors.red,
      ),
    );
  }
}

到此这篇关于Flutter实现渐变弧形进度条的示例详解的文章就介绍到这了,更多相关Flutter渐变弧形进度条内容请搜索程序员之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持程序员之家!

相关文章

  • 安卓图片反复压缩后为什么普遍会变绿而不是其它颜色?

    安卓图片反复压缩后为什么普遍会变绿而不是其它颜色?

    今天小编就为大家分享一篇关于安卓图片反复压缩后为什么普遍会变绿而不是其它颜色?,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • Android?Studio实现简单页面跳转的详细教程

    Android?Studio实现简单页面跳转的详细教程

    这篇文章主要给大家介绍了关于Android?Studio实现简单页面跳转的详细教程,文中通过图文介绍的非常详细,对大家学习或者使用Android?Studio具有一定的参考学习价值,需要的朋友可以参考下
    2023-01-01
  • Android6.0指纹识别开发实例详解

    Android6.0指纹识别开发实例详解

    这篇文章主要介绍了Android6.0指纹识别开发实例详解的相关资料,需要的朋友可以参考下
    2017-04-04
  • Android开发之简单文件管理器实现方法

    Android开发之简单文件管理器实现方法

    这篇文章主要介绍了Android开发之简单文件管理器实现方法,简单实现了Android的文件目录查看,文件重命名,打开,删除等功能,需要的朋友可以参考下
    2016-01-01
  • Android 配置gradle实现VersionCode自增实例

    Android 配置gradle实现VersionCode自增实例

    今天小编就为大家分享一篇Android 配置gradle实现VersionCode自增实例,具有很好的 参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-03-03
  • Android开发笔记SQLite优化记住密码功能

    Android开发笔记SQLite优化记住密码功能

    这篇文章主要为大家详细介绍了Android开发笔记SQLite优化记住密码功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-07-07
  • 全面解析Android中对EditText输入实现监听的方法

    全面解析Android中对EditText输入实现监听的方法

    这篇文章主要介绍了Android中对EditText输入实现监听的方法,包括一个仿iOS的带清除功能的ClearEditText输入框控件的详细使用介绍,需要的朋友可以参考下
    2016-04-04
  • Kotlin?ContentProvider使用方法详解

    Kotlin?ContentProvider使用方法详解

    ContentProvider内容提供者,主要用于再不同的应用程序之前实现数据共享的功能,它提供了一套完整的机制,允许一个程序访问另外一个程序的数据,同时还能保证数据的安全性
    2022-12-12
  • android 布局之ConstraintLayout的使用

    android 布局之ConstraintLayout的使用

    这篇文章主要介绍了android 布局之ConstraintLayout的使用,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-04-04
  • 两种Android打电话实现方法

    两种Android打电话实现方法

    这篇文章主要为大家详细介绍了两种Android打电话实现方法,具有一定的实用性,感兴趣的小伙伴们可以参考一下
    2016-08-08

最新评论

?


http://www.vxiaotou.com