數碼管就是咱們不少液晶屏或者小家電上顯示數字的小顯示屏, 一個數字「8」對應一位數碼管,每一個數碼管8個LED:數字「8」的7個筆畫,以及小數點 git
一下是網上找到的數碼管尺寸圖,能夠看到數碼管是呈10度傾斜的 由於我在大學,本身搞了些單片機,因此對這東西很是熟悉由於我本身app的實際須要,並不須要小數點,因此只須要顯示數字8,若是是顯示0~99的數字,那就用Row集合兩個「8」就能夠了。因此問題的關鍵就是顯示數字「8」編程
顯然用0~9 十張圖片是最簡單也是最low的,固然不想使用,因而決定試一試Flutter CustomPainter,配合貝塞爾曲線來繪製。canvas
看上面的尺寸圖,你們能夠看到,「8」的每個筆畫是有編號的,最頂部是a,而後順時針遞增,最中間的筆畫是g,後面描述的時候會用到數組
筆畫a最簡單,經過6個點,便可畫出其輪廓的貝塞爾曲線,而後填充顏色便可,以後的6個筆畫,由於位置未知,因此計算三角函數來得出位置很麻煩,因此這裏使用了3維變換,(x,y)軸平移,z軸旋轉,便可挪到對應位置。好比筆畫b,是經過筆畫a右移筆畫長度(外加二者間隙),而後旋轉(90+10)度完成的,筆畫C是筆畫B移動筆畫長度完成的,以此類推。畫完了數字「8」,再根據輸入的數字是0~9,決定每一個筆畫的顏色。性能優化
因此技術難點解析成了一下幾項:bash
Flutter中,CustomPainter是個抽象類,須要咱們本身繼承子類,而後重寫幾個方法:app
class NumberPart extends CustomPainter {
@override
bool shouldRepaint(CustomPainter oldDelegate) {
return true;
}
@override
void paint(Canvas canvas, Size size) {
}
}
複製代碼
shouldRepaint告訴flutter是否須要重繪,除非爲了性能優化,不然直接返回true就能夠了,這裏不擴展講 paint是繪圖的核心方法,參數canvas是畫布,size是畫布大小less
canvas能夠畫圓,畫線,畫圖片和文字等等,定製內容的話,能夠畫path(貝塞爾曲線)ide
貝塞爾曲線在flutter中的實現也很簡單,是一個Path類,而後經過在Path上添加點/線/弧等,繪製路徑,這裏咱們使用addPolygon來添加多邊形 下面這個方法,就是建立筆畫a,一個相似六邊形的形狀,裏面的width是線寬,lerp這個單詞其實我也說不清意思,就是六邊形左上方的點距離最左邊點的x軸位移,length就是筆畫長度。函數
Path genPath(double length, double width) {
final path = Path();
double lerp = width / 1.7;
path.addPolygon([
Offset(0, 0),
Offset(lerp, -width / 2),
Offset(length - lerp, -width / 2) et,
Offset(length, 0) + offset,
Offset(length - lerp, width / 2)+ offset,
Offset(lerp, width / 2)
], true);
return path;
}
複製代碼
在繪製筆畫到畫布的時候,須要指定paint,也就是染色方式,好比是否填充啊,各類顏色啊啥的,畢竟貝塞爾曲線只是線的走向,既沒寬度也沒顏色的
final highlightPaint = Paint()
..style = PaintingStyle.fill
..color = highlightColor;
final delightPaint = Paint()
..style = PaintingStyle.fill
..color = delightColor;
複製代碼
畫筆是Paint類,而後經過設置style和color,設置成填充某種顏色,highlightPaint是筆畫高亮的時候的顏色,好比亮紅色,delightPaint是暗的時候的顏色,好比暗紅色,不少數碼管,不亮的時候也能看到顏色,爲了逼真咱們也這麼幹
canvas.drawPath(pathA, getPaint());
複製代碼
經過canvas.drawPath,而後傳入路徑和畫筆,便可畫出筆畫a
貝塞爾曲線的移動和旋轉稱爲transform,平移叫translate,旋轉叫rotate,咱們是在水平面旋轉,因此是rotateZ,沿Z軸旋轉。
Path pathB = genPath(Offset.zero, length, width);
transform.translate(length);
transform.translate(gap, gap);
transform.rotateZ((10 + 90) / 180 * 3.14159);
pathB = pathB.transform(transform.storage);
canvas.drawPath(pathB, getPaint());
複製代碼
第一行建立筆畫b的貝塞爾曲線路徑,而後平移筆畫長度,由於筆畫a/b之間有一個間隙,因此x,y軸移動gap,再旋轉90+10度,由於rotateZ的參數是弧度制,因此轉換一下,最後將transform的數值經過transform.storage變成矩陣,傳遞給pathB.transform,就旋轉完了。 旋轉筆畫c的時候,仍是在畫布原點建立,而後在移動b的transform基礎上,再向x軸移動length + gap就能夠了:
Path pathC = genPath(Offset.zero, length, width);
transform.translate(length + gap);
pathC = pathC.transform(transform.storage);
canvas.drawPath(pathC, getPaint(2));
複製代碼
你可能會奇怪,明明筆畫c是筆畫b向左下移動,爲何是translate的x軸?由於transform裏,自己有個旋轉。
數碼管經過7個筆畫(a-g)的明暗,來顯示0~9,因此咱們來經過全局數組來展現編碼:
final matrix = [
[
//0
true,
true,
true,
true,
true,
true,
false,
],
[
//1
false,
true,
true,
false,
false,
false,
false,
],
[
//2
true,
true,
false,
true,
true,
false,
true,
],
[
//3
true,
true,
true,
true,
false,
false,
true,
],
[
//4
false,
true,
true,
false,
false,
true,
true,
],
[
//5
true,
false,
true,
true,
false,
true,
true,
],
[
//6
true,
false,
true,
true,
true,
true,
true,
],
[
//7
true,
true,
true,
false,
false,
false,
false,
],
[
//8
true,
true,
true,
true,
true,
true,
true,
],
[
//9
true,
true,
true,
true,
false,
true,
true,
]
];
複製代碼
第一維是哪一個數字,第二維是哪一個筆畫的明暗 因此經過matrix[num][index]便可反應這個筆畫的明暗 好比數字0的筆畫b的狀態,就是matrix[0][1]==true,也就是筆畫b在顯示數字0時,要亮。
import 'package:flutter/material.dart';
final matrix = [
[
//0
true,
true,
true,
true,
true,
true,
false,
],
[
//1
false,
true,
true,
false,
false,
false,
false,
],
[
//2
true,
true,
false,
true,
true,
false,
true,
],
[
//3
true,
true,
true,
true,
false,
false,
true,
],
[
//4
false,
true,
true,
false,
false,
true,
true,
],
[
//5
true,
false,
true,
true,
false,
true,
true,
],
[
//6
true,
false,
true,
true,
true,
true,
true,
],
[
//7
true,
true,
true,
false,
false,
false,
false,
],
[
//8
true,
true,
true,
true,
true,
true,
true,
],
[
//9
true,
true,
true,
true,
false,
true,
true,
]
];
class DigitalNumber extends StatelessWidget {
final double height;
final double width;
final double lineWidth;
final int num;
final bool dotLight;
final Color highlightColor;
final Color delightColor;
DigitalNumber(
{@required this.height,
@required this.width,
this.lineWidth = 8,
num,
this.dotLight = true,
this.highlightColor = Colors.red,
this.delightColor = const Color(0x33FF0000)})
: this.num = num > 0 ? (num > 9 ? 9 : num) : 0;
@override
Widget build(BuildContext context) {
return CustomPaint(
painter: NumberPart(
lineWidth: lineWidth,
num: num,
dotLight: dotLight,
highlightColor: highlightColor,
delightColor: delightColor),
size: Size(width, height),
);
}
}
class NumberPart extends CustomPainter {
final int num;
final bool dotLight;
final Color highlightColor;
final Color delightColor;
final double lineWidth;
NumberPart(
{@required this.lineWidth,
@required this.num,
@required this.dotLight,
@required this.highlightColor,
@required this.delightColor});
@override
bool shouldRepaint(CustomPainter oldDelegate) {
return true;
}
Paint getPaint(int index) {
final highlightPaint = Paint()
..style = PaintingStyle.fill
..color = highlightColor;
final delightPaint = Paint()
..style = PaintingStyle.fill
..color = delightColor;
return matrix[num][index] ? highlightPaint : delightPaint;
}
Path genPath(Offset offset, double length, double width) {
final path = Path();
double lerp = width / 1.7;
path.addPolygon([
Offset(0, 0) + offset,
Offset(lerp, -width / 2) + offset,
Offset(length - lerp, -width / 2) + offset,
Offset(length, 0) + offset,
Offset(length - lerp, width / 2) + offset,
Offset(lerp, width / 2) + offset
], true);
return path;
}
@override
void paint(Canvas canvas, Size size) {
double width = lineWidth;
double length = (size.width) / 1.5 - width;
double leftOffset = size.width / 3;
double gap = width / 8;
Path pathA = genPath(Offset.zero, length, width);
Matrix4 transform = Matrix4.identity();
transform.translate(leftOffset, width / 2 + 2);
pathA = pathA.transform(transform.storage);
canvas.drawPath(pathA, getPaint(0));
Path pathB = genPath(Offset.zero, length, width);
transform.translate(length);
transform.translate(gap, gap);
transform.rotateZ((10 + 90) / 180 * 3.14159);
pathB = pathB.transform(transform.storage);
canvas.drawPath(pathB, getPaint(1));
Path pathC = genPath(Offset.zero, length, width);
transform.translate(length + gap);
pathC = pathC.transform(transform.storage);
canvas.drawPath(pathC, getPaint(2));
Path pathD = genPath(Offset.zero, length, width);
transform.translate(length + gap, gap);
transform.rotateZ((90 - 10) / 180 * 3.14159);
pathD = pathD.transform(transform.storage);
canvas.drawPath(pathD, getPaint(3));
Path pathE = genPath(Offset.zero, length, width);
transform.translate(length + gap, gap);
transform.rotateZ((90 + 10) / 180 * 3.14159);
pathE = pathE.transform(transform.storage);
canvas.drawPath(pathE, getPaint(4));
Path pathF = genPath(Offset.zero, length, width);
Matrix4 transformF = transform.clone();
transformF.translate(length + gap);
pathF = pathF.transform(transformF.storage);
canvas.drawPath(pathF, getPaint(5));
Path pathG = genPath(Offset.zero, length, width);
transform.translate(length + gap / 2, gap);
transform.rotateZ((90 - 10) / 180 * 3.14159);
pathG = pathG.transform(transform.storage);
canvas.drawPath(pathG, getPaint(6));
}
}
複製代碼
DigitalNumber類就是單個數碼管的widget,須要指定大小(數碼管適應指定的大小),能夠配置筆畫的寬度,指定顯示哪一個數字,以及明暗兩種顏色。dotLight暫時沒有實現
使用的代碼也很簡單:
Row(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
DigitalNumber(
height: 60,
width: 45,
num: 1
lineWidth: 6,
),
DigitalNumber(
height: 60,
width: 45,
num:3,
lineWidth: 6,
),
],
)
複製代碼
這樣就能夠顯示數字13了。
作這個項目的時候,就想到了當時作單片機 單片機的多位數碼管顯示,是經過n+8個引腳控制的,n個引腳對應幾位數碼管,8個引腳對應這一排數碼管的筆畫,這樣組成一個矩陣,而後經過定時器掃描的方式,輪詢逐位顯示每一位數碼管,好比時刻1,使能第0位數碼管,而後經過編碼控制8個引腳,讓數碼管顯示數字1,而後到時刻2,關閉第0位數碼管,使能第1位數碼管,經過編碼顯示數字3,往復掃描,雖然某一時刻只能顯示一位數字,可是由於掃描很快,因此肉眼看到的就是完整的數字13了。這個就是最初的屏幕掃描頻率
在單片機的顯示中,經過某個輸入獲取到數字,到讓數字顯示到數碼管,是兩個邏輯,兩個邏輯都有本身的操做週期,因此兩個不能耦合,因而獲取數字的邏輯,獲取到新的數字之後,會將這個數字(或者對應的數碼管編碼)存放在數組中,而後到了刷新數碼管的週期,數碼管程序經過讀取這個數組的數字,顯示在數碼管上,那麼這個數組,就是顯存啦。哈哈
最後貼上我作的完整app,這是一個遙控車控制app,有前進和轉向兩個搖桿,控制的數據經過udp發送給遙控車,遙控車上有esp8226 wifi芯片,配置成AP模式,也就是wifi基站,app的udp數據發送給esp8226後,下位機轉換成PWM數據,控制舵機和L298N電機驅動芯片,後者控制減速電機讓小車運動。app、下位機電路、esp8226編程,遙控車整車都是我本身作的,很是有樂趣,下次有機會給你們說說我作的遙控車,你們2019年快樂~~~~