dependencies:
flutter_star: $lastVersion
複製代碼
目標: 使用canvas手工打造,一個完美的星星評分組件。github
---->[StarScore 星星顯示組件]----
[1] 好比顯示4.2: 會有5顆星, 前四顆填滿,後一刻填充20%
StarScore 爲 Stateless組件,僅負責顯示的需求
---->[CustomRating 星星評分組件]----
[2] 可指定最大值,也就是顯示多少個星星
[3] 點擊時會改變狀態,進行評分,支持半星評分
[4] 支持評分回調
---->[StarWidget組件]----
[5]. 可定義星星的顯示進度狀況 0% ~ 100 % 無死角
[6]. 可定義星星的角數
[7]. 可定義星星的顏色、大小
複製代碼
分數展現組件編程
名稱 | 類型 | 功能 | 備註 | 默認 |
---|---|---|---|---|
score | double | 分數 | - | 0 |
star | Star | 見 第四點 |
星星屬性配置 | Star() |
tail | Widget | 尾部的組件 | - | null |
StarScore(
score: 4.8,
star: Star(
fillColor: Colors.tealAccent,
emptyColor: Colors.grey.withAlpha(88)),
tail: Column(
children: <Widget>[
Text("綜合評分"),
Text("4.8"),
],
),
),
複製代碼
評分組件canvas
名稱 | 類型 | 功能 | 備註 | 默認 |
---|---|---|---|---|
max | int | 最大星星數 | - | 5 |
score | double | 分數 | - | 0 |
star | Star | 見 第四點 |
星星屬性配置 | Star() |
onRating | Fluction(double) | 點擊回調 | @required | null |
CustomRating(onRating: (s) {
print(s);
}),
複製代碼
CustomRating(
max: 6,
score: 3.0,
star: Star(
num: 12,
fillColor: Colors.orangeAccent,
fat: 0.6,
emptyColor: Colors.grey.withAlpha(88)),
onRating: (s) {
print(s);
}),
複製代碼
星星組件 : 高度可定製的配置類瀏覽器
名稱 | 類型 | 功能 | 備註 | 默認 |
---|---|---|---|---|
progress | double | 填充的進度 | [0,1] | 0.0 |
num | int | 星星的角數 | 大於3 | 5 |
fat | double | 星星的胖瘦 | (0,1] | 0.5 |
emptyColor | Color | 星星的色 | - | Colors.grey |
fillColor | Color | 星星的填充色 | - | Colors.yellow |
size | double | 星星的大小 | - | 20 |
progress
num
fat
fillColor和emptyColor
展現結束,下面進入正文bash
class Star {
final int num;
final double progress;
final Color emptyColor;
final Color fillColor;
final double size;
final double fat;
const Star({this.progress = 0,
this.fat = 0.5,
this.fillColor = Colors.yellow,
this.emptyColor = Colors.grey,
this.num = 5,
this.size = 25});
}
複製代碼
class _StarPainter extends CustomPainter {
Star star;
Paint _paint;
Paint _filePaint;
Path _path;
double _radius;
_StarPainter(this.star) {
_paint = Paint()
..color = (star.emptyColor)
..isAntiAlias = true;
_filePaint = Paint()
..color = (star.fillColor);
_path = Path();
_radius = star.size / 2.0;
}
@override
void paint(Canvas canvas, Size size) {
//TODO 繪製
}
@override
bool shouldRepaint(CustomPainter oldDelegate) {
return true;
}
}
複製代碼
若是說StarWidget是評分組件的基礎,那麼繪製的路徑就是星星的靈魂
下面的nStarPath是繪製n角星路徑的核心方法微信
Path nStarPath(int num, double R, double r, {dx = 0, dy = 0, rotate = 0}) {
_path.reset(); //重置路徑
double perRad = 2 * pi / num; //每份的角度
double radA = perRad / 2 / 2 + rotate; //a角
double radB = 2 * pi / (num - 1) / 2 - radA / 2 + radA + rotate; //起始b角
_path.moveTo(cos(radA) * R + dx, -sin(radA) * R + dy); //移動到起點
for (int i = 0; i < num; i++) {
//循環生成點,路徑連至
_path.lineTo(
cos(radA + perRad * i) * R + dx, -sin(radA + perRad * i) * R + dy);
_path.lineTo(
cos(radB + perRad * i) * r + dx, -sin(radB + perRad * i) * r + dy);
}
_path.close();
return _path;
}
複製代碼
在初始化的時候爲路徑賦值less
class _StarPainter extends CustomPainter {
...
_StarPainter(this.star) {
...
_path = Path();
_radius = star.size / 2.0;
nStarPath(star.num, _radius, _radius * star.fat);
}
複製代碼
繪製也很是簡單,其中有個進度問題,能夠先畫背景的星星, 再畫一個填充的星星,再用canvas.clipRect進行裁剪便可。ide
@override
void paint(Canvas canvas, Size size) {
canvas.translate(_radius, _radius);
canvas.drawPath(_path, _paint);
canvas.clipRect(Rect.fromLTRB(
-_radius, -_radius, _radius * 2 * star.progress - _radius, _radius));
canvas.drawPath(_path, _filePaint);
}
複製代碼
將畫板放入CustomPaint中便可ui
class StarWidget extends StatelessWidget {
final Star star;
StarWidget({this.star = const Star()});
@override
Widget build(BuildContext context) {
return Container(
width: star.size,
height: star.size,
child: CustomPaint(
painter: _StarPainter(star),
),
);
}
}
複製代碼
這就是星星組件的全部代碼,也不超過百行。
該組件的目的是顯示評分,能夠精確的顯示百分進度
其實也沒啥,僅是用幾個StarWidget拼起來而已,代碼也就下面一丟丟
class StarScore extends StatelessWidget {
final Star star;
final double score;
final Widget tail;
StarScore({this.star = const Star(), this.score, this.tail});
@override
Widget build(BuildContext context) {
var li = <StarWidget>[];
int count = score.floor();
for (int i = 0; i < count; i++) {
li.add(StarWidget(star: star.copyWith(progress: 1.0)));
}
if (score - count > 0) {
li.add(StarWidget(star: star.copyWith(progress: score - count)));
}
return Wrap(
crossAxisAlignment: WrapCrossAlignment.center,
children: [
...li,
SizedBox(
width: 10,
),
if (tail!=null) tail
],
);
}
}
複製代碼
因爲點擊須要本身響應進行狀態改變,因此使用StatefulWidget 最核心的是GestureDetector進行觸點的獲取並計算出當前值。其中對.5進行處理,以及越界的處理。
class CustomRating extends StatefulWidget {
final int max;
final Star star;
final double score;
final Function(double) onRating;
CustomRating(
{this.max = 5,
this.score = 0,
this.star = const Star(),
@required this.onRating})
: assert(score <= max);
@override
_CustomRatingState createState() => _CustomRatingState();
}
class _CustomRatingState extends State<CustomRating> {
double _score;
@override
void initState() {
_score = widget.score;
super.initState();
}
@override
Widget build(BuildContext context) {
var li = <StarWidget>[];
int count = _score.floor(); //滿星
for (int i = 0; i < count; i++) {
li.add(StarWidget(star: widget.star.copyWith(progress: 1.0)));
}
if (_score != widget.max.toDouble())
li.add(StarWidget(
star: widget.star.copyWith(progress: _score - count))); //不滿星
var empty = widget.max - count - 1; // 空星
for (int i = 0; i < empty; i++) {
li.add(StarWidget(star: widget.star.copyWith(progress: 0)));
}
return GestureDetector(
onTapDown: (d) {
setState(() {
_score = d.localPosition.dx / widget.star.size;
if (_score - _score.floor() > 0.5) {
_score = _score.floor() + 1.0;
} else {
_score = _score.floor() + 0.5;
}
if (_score >= widget.max.toDouble()) {
_score = widget.max.toDouble();
}
widget.onRating(_score);
});
},
child: Wrap(
children: li,
),
);
}
}
複製代碼
須要在
pubspec.yaml
進行一些配置
name 是名稱
description 是描述 60 ~ 180 之間,過短或太長會扣分
version 版本
author 做者信息,會報warning 但我就想寫
homepage 主頁
複製代碼
---->[pubspec.yaml]----
name: flutter_star
description: You can create a star easily and decide how many angle or color of the star, even the fat and progress of the star.
version: 0.1.2
author: 張風捷特烈<1981462002@qq.com>
homepage: https://juejin.im/user/5b42c0656fb9a04fe727eb37/collections
複製代碼
否則會扣分
它會在這裏,給使用者看
flutter packages pub publish --server=https://pub.dartlang.org
複製代碼
而後須要權限驗證,記得
所有複製在瀏覽器打開,不用在控制檯點連接
,因爲控制檯的換行而致使url不全。
而後就看你的網給不給力了。
flutter_star
歡迎使用
另外本人有一個Flutter微信交流羣,歡迎小夥伴加入,共同探討Flutter的問題,期待與你的交流與切磋。
@張風捷特烈 2019.02.23 未允禁轉
個人公衆號:編程之王
聯繫我--郵箱:1981462002@qq.com --微信:zdl1994328
~ END ~