Flutter實(shí)現(xiàn)Android的SpannableString冠句,尋找所有符合條件字符串并上色

今天遇到一個(gè)需求懦底,后臺返回字符串和目標(biāo)字符基茵,將目標(biāo)字符上色,在Android中很好實(shí)現(xiàn)根灯,使用TextSpannableString就可以烙肺,但是Flutter中只能使用TextSpan,嘗試了幾種方法后終于實(shí)現(xiàn)搏明。

import 'package:flutter/material.dart';
import 'package:easy_flutter/utils/extension/StringUtils.dart';

class SpannableTextWidget extends StatefulWidget {
  final String text;
  final List<String> pattern;
  final TextStyle defStyle;
  final TextStyle patternStyle;
  final EdgeInsets margin;

  SpannableTextWidget(
      {this.text,
      this.pattern,
      this.defStyle,
      this.patternStyle,
      this.margin = EdgeInsets.zero});

  @override
  State<StatefulWidget> createState() => _SpannableTextState();
}

class _SpannableTextState extends State<SpannableTextWidget> {
  List<TextSpan> list = [];
  List<int> indexList = [];
  Map<String, List<int>> _memory = {};
  int index = 0;

  @override
  void initState() {
    super.initState();
    String t = widget.text;
    WidgetsBinding.instance.addPostFrameCallback((timeStamp) {
      _init(t);
    });
  }

  @override
  Widget build(BuildContext context) {
    return Container(
      margin: widget.margin,
      child: list.isEmpty
          ? Text('')
          : Text.rich(
              TextSpan(children: list),
            ),
    );
  }

  _init(String t) {
    if (t != null && t.isNotEmpty) {
      if (widget.pattern != null && widget.pattern.isNotEmpty) {
        widget.pattern.forEach((element) {
          if (_memory[element] == null) _memory[element] = [];
          while (_calc(t, element) != -1) {}
        });
      } else {
        list.add(TextSpan(text: t, style: widget.defStyle));
      }
      var charList = t.toCharList();
      for (int i = 0; i < charList.length; i++) {
        if (indexList.contains(i)) {
          list.add(TextSpan(text: charList[i], style: widget.patternStyle));
        } else
          list.add(TextSpan(text: charList[i], style: widget.defStyle));
      }
      setState(() {});
    }
  }

  int _calc(String t, String element) {
    int indexOf = t.indexOf(
        element, _memory[element].isEmpty ? 0 : _memory[element].last + 1);

    if (indexOf != -1) {

      for (int i = indexOf; i < indexOf + element.length; i++) {
        if (!indexList.contains(i)) indexList.add(i);
      }
      _memory[element].add(indexOf);
    }
    return indexOf;
  }
}
List<String> toCharList() {
  return this.runes.map((e) => String.fromCharCode(e)).toList();
}

使用方法

SpannableTextWidget(
                      text: '11121222211133123453454353412312',
                      pattern: ["1", "3", "5"],
                      defStyle: TextStyle(
                          fontSize: 15, color: Colors.black),
                      patternStyle: TextStyle(fontSize:18,color:Colors.red),
                      margin:
                          EdgeInsets.only(top: 22, left: 20, right: 20),
                    )

運(yùn)行效果


微信截圖_20210922102428.png

剛發(fā)群里沒多久,就被大佬教育了娃循,說我寫的太爛炕檩,然后大佬給出了自己的解決方案:

import 'package:flutter/material.dart';
import 'package:easy_flutter/utils/extension/StringUtils.dart';

class SpannableTextWidget extends StatelessWidget {
  final String text;
  final List<String> pattern;
  final TextStyle defStyle;
  final TextStyle patternStyle;
  final EdgeInsets margin;

  SpannableTextWidget(
      {this.text,
      this.pattern,
      this.defStyle,
      this.patternStyle,
      this.margin = EdgeInsets.zero});

  @override
  Widget build(BuildContext context) {
    RegExp regExp = RegExp('$pattern');

    List<TextSpan> list = text
        .replaceAllMapped(regExp, (match) => '&*${match[0]}&*')
        .split('&*')
        .map<TextSpan>((e) => regExp.matchAsPrefix(e) == null
            ? TextSpan(text: e, style: defStyle)
            : TextSpan(text: e, style: patternStyle))
        .toList();

    return Container(
      margin: margin,
      child: list.isEmpty
          ? Text('')
          : Text.rich(
              TextSpan(children: list),
            ),
    );
  }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末斗蒋,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子笛质,更是在濱河造成了極大的恐慌泉沾,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,884評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件妇押,死亡現(xiàn)場離奇詭異跷究,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)敲霍,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,347評論 3 385
  • 文/潘曉璐 我一進(jìn)店門俊马,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人肩杈,你說我怎么就攤上這事柴我。” “怎么了扩然?”我有些...
    開封第一講書人閱讀 157,435評論 0 348
  • 文/不壞的土叔 我叫張陵艘儒,是天一觀的道長。 經(jīng)常有香客問我夫偶,道長界睁,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,509評論 1 284
  • 正文 為了忘掉前任兵拢,我火速辦了婚禮翻斟,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘说铃。我一直安慰自己访惜,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,611評論 6 386
  • 文/花漫 我一把揭開白布截汪。 她就那樣靜靜地躺著疾牲,像睡著了一般。 火紅的嫁衣襯著肌膚如雪衙解。 梳的紋絲不亂的頭發(fā)上阳柔,一...
    開封第一講書人閱讀 49,837評論 1 290
  • 那天,我揣著相機(jī)與錄音蚓峦,去河邊找鬼舌剂。 笑死济锄,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的霍转。 我是一名探鬼主播荐绝,決...
    沈念sama閱讀 38,987評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼避消!你這毒婦竟也來了低滩?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,730評論 0 267
  • 序言:老撾萬榮一對情侶失蹤岩喷,失蹤者是張志新(化名)和其女友劉穎恕沫,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體纱意,經(jīng)...
    沈念sama閱讀 44,194評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡婶溯,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,525評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了偷霉。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片迄委。...
    茶點(diǎn)故事閱讀 38,664評論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖类少,靈堂內(nèi)的尸體忽然破棺而出叙身,到底是詐尸還是另有隱情,我是刑警寧澤瞒滴,帶...
    沈念sama閱讀 34,334評論 4 330
  • 正文 年R本政府宣布曲梗,位于F島的核電站,受9級特大地震影響妓忍,放射性物質(zhì)發(fā)生泄漏虏两。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,944評論 3 313
  • 文/蒙蒙 一世剖、第九天 我趴在偏房一處隱蔽的房頂上張望定罢。 院中可真熱鬧,春花似錦旁瘫、人聲如沸祖凫。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,764評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽惠况。三九已至,卻和暖如春宁仔,著一層夾襖步出監(jiān)牢的瞬間稠屠,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,997評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留权埠,地道東北人榨了。 一個(gè)月前我還...
    沈念sama閱讀 46,389評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像攘蔽,于是被迫代替她去往敵國和親龙屉。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,554評論 2 349

推薦閱讀更多精彩內(nèi)容