/// Text的使用Demo
class TextWidgetDemo extends StatelessWidget {
final textContent = "《定風波》 蘇軾 /n莫聽穿林打葉聲,何妨吟嘯且徐行。/n竹杖芒鞋輕勝馬,誰怕?一蓑煙雨任平生。/n";
// final textContent = "《定風波》 蘇軾 莫聽穿林打葉聲,何妨吟嘯且徐行。竹杖芒鞋輕勝馬,誰怕?一蓑煙雨任平生。";

@override
Widget build(BuildContext context) {
return Text(
textContent,
// 默認情況下,Text是包裹文字的,文字內容太少時可能看不出效果
textAlign: TextAlign.center,
// 文字最多顯示2行
maxLines: 3,
// 文字超過2行時,顯示...
overflow: TextOverflow.ellipsis,
// 縮放因子,默認是1
textScaleFactor: 1,
style: TextStyle(
fontSize: 30,
color: Colors.red,
fontWeight: FontWeight.bold, // 字體加粗
),
);
}
}

注意:Text 并不是最終渲染的 Widget,最終渲染的是 RichText。Text 的父類是 StatelessWidget,最終渲染的 Widget 是 build()方法創建出來的 RenderObjectWidget,即 RichText

2、富文本(Text.rich())

Text.rich() 有一個必須參數 InlineSpan textSpan,InlineSpan 是抽象類且無工廠構造函數,無法直接創建,故需要使用其子類:

  • TextSpan : 用于構建純文本的 Span
  • WidgetSpan : 用于構建內嵌 Widget 的 Span (比如:Icon)

/// Text.rich()的使用Demo
class TextRichDemo extends StatelessWidget {br/>@override
Widget build(BuildContext context) {
return Text.rich(
TextSpan(
// text: "Hello lqr",
// style: TextStyle(color: Colors.red,fontSize: 20),
children: [
TextSpan(text: "Hello World", style: TextStyle(color: Colors.red)),
TextSpan(
text: "Hello Flutter", style: TextStyle(color: Colors.green)),
WidgetSpan(child: Icon(Icons.favorite, color: Colors.red)),
TextSpan(text: "Hello Dart", style: TextStyle(color: Colors.blue)),
],
),
style: TextStyle(fontSize: 26),
);
}
}

二、按鈕 Widget

1、常見 Button

在使用這些常見 Widget 時,經常會看到構造方法中有兩類 "必傳"參數

class HomePage extends StatelessWidget {br/>@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text("基礎Widget")),
body: HomeContent(),
// 4. FloatingActionButton
floatingActionButton: FloatingActionButton(
child: Icon(Icons.add),
onPressed: () => print("FloatingActionButton Click"),
),
floatingActionButtonLocation: FloatingActionButtonLocation.centerFloat,
);
}
}

...

class ButtonDemo extends StatelessWidget {br/>@override
Widget build(BuildContext context) {
return Column(
children: [
// 1. 突出的Button(從 v1.25.0 過時,推薦使用 ElevatedButton)
RaisedButton(
child: Text("RaisedButton"),
textColor: Colors.red, // 文字顏色(也可以在Text的style中設置)
color: Colors.blue, // 按鈕背景色
onPressed: () => print("RaisedButton Click"),
),

// 2. 扁平的Button(從 v1.25.0 過時,推薦使用 TextButton)
FlatButton(
child: Text("FlatButton"),
color: Colors.orange,
onPressed: () => print("FlatButton Click"),
),

// 3. 邊框Button(從 v1.25.0 過時,推薦使用 OutlinedButton)
OutlineButton(
child: Text("OutlineButton"),
onPressed: () => print("OutlineButton Click")),

// 4. FloatingActionButton,一般用在Scaffold中
// FloatingActionButton(onPressed: onPressed)

// 5. 自定義Button:圖標-文字-背景-圓角
FlatButton(
color: Colors.amberAccent,
shape: RoundedRectangleBorder( // 圓角
borderRadius: BorderRadius.circular(8),
),
child: Row(
mainAxisSize: MainAxisSize.min, // 默認是max,占滿父Widget;min是包裹內容
children: [
Icon(Icons.favorite, color: Colors.red), // 圖標
Text("喜歡作者"), // 文字
],
),
onPressed: () => print("自定義Button"), // onPressed必傳,否則樣式可能會出問題
)
],
);
}
}

2、定制 Button

class ButtonExtensionDemo extends StatelessWidget {br/>@override
Widget build(BuildContext context) {
return Column(
children: [
/// 1. 默認情況下Button上下有一定有間隔
/// MaterialTapTargetSize.padded:當按鈕寬(或高)不足48px時,就把寬(或高)擴展到48px。
/// MaterialTapTargetSize.shrinkWrap:緊縮包裹,可以去除上下的間隔。
FlatButton(
color: Colors.red,
child: Text("Flat Button1"),
textColor: Colors.white,
onPressed: () {},
materialTapTargetSize: MaterialTapTargetSize.shrinkWrap,
),
FlatButton(
color: Colors.red,
child: Text("Flat Button2"),
textColor: Colors.white,
onPressed: () {},
materialTapTargetSize: MaterialTapTargetSize.shrinkWrap,
),

/// 2. 修改按鈕的最小寬度:ButtonTheme
FlatButton(
minWidth: 30,
height: 30,
color: Colors.red,
child: Text(""),
onPressed: () {},
),
ButtonTheme(
minWidth: 30,
height: 30,
child: FlatButton(
color: Colors.red,
child: Text(""),
onPressed: () {},
),
),

/// 3. 修改按鈕的內間距
FlatButton(
padding: EdgeInsets.all(0),
// 只能去除左右內間距,上下內間距可以指定一個固定height解決
color: Colors.red,
child: Text("Float Button3"),
textColor: Colors.white,
onPressed: () {},
),
],
);
}
}

三、圖片 Widget

Image 控件需要一個必傳參數 ImageProvider image,常見子類如下:

1、NetworkImage

class ImageDemo01 extends StatelessWidget {

@override
Widget build(BuildContext context) {
return GridView.count(
crossAxisSpacing: 8,
mainAxisSpacing: 8,
crossAxisCount: 3,
children: [
wrapItem(
Image(image: NetworkImage(imageUrl)),
"NetworkImage",
),
wrapItem(
Image.network(imageUrl),
"Image.network()",
),
Container(),
wrapItem(
Image.network(imageUrl, fit: BoxFit.fill), // 拉伸
"BoxFit.fill",
),
wrapItem(
Image.network(imageUrl, fit: BoxFit.contain), // 內容縮放至最長的一邊貼邊
"BoxFit.contain",
),
wrapItem(
Image.network(imageUrl, fit: BoxFit.cover), // 內容縮放至最短的一邊貼邊
"BoxFit.cover",
),
wrapItem(
Image.network(imageUrl, fit: BoxFit.fitWidth), // 寬度一定,高度自適應
"BoxFit.fitWidth",
),
wrapItem(
Image.network(imageUrl, fit: BoxFit.fitHeight), //高度一定,寬度自適應
"BoxFit.fitHeight",
),
Container(),
wrapItem(
Image.network(imageUrl, alignment: Alignment.bottomCenter),
"Alignment.topLeft",
),
wrapItem(
Image.network(imageUrl, alignment: Alignment.center),
"Alignment.center",
),
wrapItem(
// 左上角是(-1, -1),右下角是(1, 1)
Image.network(imageUrl, alignment: Alignment(0, -1)),
"Alignment(0, -1)",
),
wrapItem(
// color不是背景色,而是用于圖像混入的顏色,配合 colorBlendMode 使用
Image.network(imageUrl,
color: Colors.green, colorBlendMode: BlendMode.colorDodge),
"BlendMode.colorDodge",
),
wrapItem(
Image.network(imageUrl, repeat: ImageRepeat.repeatY),
"ImageRepeat.repeatY",
),
],
);
}

Widget wrapItem(Widget widget, String tip) {
Text genTip(String tip) {
return Text(
tip,
style: TextStyle(
fontSize: 14,
color: Colors.white,
backgroundColor: Colors.black,
),
);
}

return Stack(
fit: StackFit.expand,
children: [
Container(color: Colors.red[100], child: widget),
Positioned(left: 4, bottom: 4, child: genTip(tip)),
],
);
}
}

2、AssetImage

使用 AssetImage 加載包內圖片步驟如下:

  1. 在 Flutter 項目中創建一個文件夾目錄(比如 assets/image),存儲圖片
  2. pubspec.yaml 進行配置

assets:

- assets/images/FSA_QR.png # 配置單張圖片

  1. 使用圖片

class ImageDemo02 extends StatelessWidget {

@override
Widget build(BuildContext context) {
// 簡單寫法
// Image.asset(assets/images/FSA_QR.png);

return Image(
image: AssetImage(assets/images/FSA_QR.png),
);
}
}

3、占位圖(placeHolder)

在網絡圖片未加載出來之前顯示的圖片稱為占位圖,可以使用 FadeInImage 實現占位圖功能:

class ImageExtensionDemo extends StatelessWidget {
final imageUrl =
"
https://up.enterdesk.com/edpic_source/ab/a0/40/aba040ce2daa32fa9cb0cc624b385c0a.jpg";

@override
Widget build(BuildContext context) {
return FadeInImage(
fadeInDuration: Duration(milliseconds: 1),
fadeOutDuration: Duration(milliseconds: 1),
placeholder: AssetImage("assets/images/FSA_QR.png"),
image: NetworkImage(imageUrl),
);
}
}

Flutter 會自動進行圖片緩存(默認最多緩存 1000 張,緩存空間最多 100m)
[api.flutter.dev/flutter/wid

《Android學習筆記總結+最新移動架構視頻+大廠安卓面試真題+項目實戰源碼講義》

【docs.qq.com/doc/DSkNLaERkbnFoS0ZF】 完整內容開源分享

…]( )

[api.flutter.dev/flutter/pai…]( )

四、字體圖標 Widget

Icon 字體圖標和圖片圖標對比:

  • 字體圖標是矢量圖(放大的時候不會失真)
  • 字體圖標可以設置顏色
  • 圖標很多時,字體圖標占據空間更小

Icon 控件接收一個必傳參數 IconData icon,Icons 中配備了大量常用 icon (如 Icons.pets),可以使用 Icons.xxxIconData(編碼,字體) 這 2 種方式來得到 IconData 對象。另外,IconData 的本質就是字體,因此也可以使用 Text 來顯示字體圖標:

class IconDemo extends StatelessWidget {
@override

最后

對于很多初中級Android工程師而言,想要提升技能,往往是自己摸索成長。而不成體系的學習效果低效漫長且無助。時間久了,付出巨大的時間成本和努力,沒有看到應有的效果,會氣餒是再正常不過的。

所以學習一定要找到最適合自己的方式,有一個思路方法,不然不止浪費時間,更可能把未來發展都一起耽誤了。

如果你是卡在缺少學習資源的瓶頸上,那么剛剛好我能幫到你。

本文已被[CODING開源項目:《Android學習筆記總結+移動架構視頻+大廠面試真題+項目實戰源碼》]( )收錄