flutter 仿微信通訊錄頁面

具體實(shí)現(xiàn)效果:

呵呵.gif

使用的插件:

//獲取漢字拼音
lpinyin: ^2.0.3

主要注意點(diǎn):
1幼驶、在從接口獲取好友列表后对嚼,需要保存每個(gè)好友名字的拼音首字母

單獨(dú)寫了一個(gè)對(duì)象用來保存每個(gè)好友名字的拼音首字母:

class FriendVo {
  V2TimFriendInfo info;
//名字拼音首字母
  String pin;
  FriendVo({
    this.info,
    this.pin,
  });
}

從接口獲取數(shù)據(jù)源之后的處理

 List<FriendVo> _friendList = [];
//_resultList是接口返回的列表  _friendList為最終的好友列表數(shù)據(jù)源
_resultList.forEach((element) {
        String now = '#';
        String nowPin = PinyinHelper.getFirstWordPinyin(_getShowName(element));
        if (nowPin.length > 0) {
          now = indexList.contains(nowPin.substring(0, 1).toUpperCase())
              ? nowPin.substring(0, 1)
              : '#';
        }
        FriendVo vo = FriendVo(info: element,pin: now.toUpperCase());
        _friendList.add(vo);
      });

//處理好的數(shù)據(jù)源按照首字母進(jìn)行重新排序
_friendList.sort((FriendVo a, FriendVo b) {
        if (a.pin == '#'){
          return 1;
        }
        if (b.pin == '#'){
          return 0;
        }
        return a.pin.compareTo(b.pin);
      });

默認(rèn)創(chuàng)建多個(gè)GlobalKey對(duì)象,用來獲取每個(gè)item距離頂部的距離

_keyList = List.generate(_friendList.length, (index) => GlobalKey());

UI數(shù)據(jù)源處理

List<Widget> itemList = [];
//頂部菜單
if (_menuList.length > 0){
      itemList.addAll(_menuList.map((e){
        int index = _menuList.indexOf(e);
        return MenuItem(
          menuVo: e,
          isHiddenLine: index == _menuList.length - 1,
          onSelect: (int value){
            if (value == 1){
              Navigator.push(context, MaterialPageRoute(builder: (context){
                return GroupListPage();
              }));
            }
          },
        );
      }).toList()
      );
    }
itemList.addAll(_friendList.map((e) {
      int index = _friendList.indexOf(e);
     //此處判斷是都顯示字母分組刑桑,判斷邏輯為棚愤,每個(gè)item只需跟上一個(gè)item的首字母對(duì)比,如果相同就不顯示毛甲,否則顯示
      FriendVo vo = e;
      bool isShow = true;
      if (nextVo != null){
        if (vo.pin == nextVo.pin) {
          isShow = false;
        }
      }
      nextVo = vo;
      return FriendItem(
        key: _keyList[index],
        vo: vo,
        isShowPin: isShow,
        isHiddenLine: index == _friendList.length - 1,
      );
    }).toList());

2、由于需要做到左邊好友列表跟右邊字母列表聯(lián)動(dòng)滾動(dòng)具被,所以需要獲取item的位置玻募。整個(gè)頁面刷新完成即可通過key獲取每個(gè)item距離頂部的距離,然后保存

//保存的當(dāng)前數(shù)據(jù)源包含的字母(不重復(fù))
List<String> _filterList = [];
//保存每個(gè)顯示字母的item距離頂部的距離
List<double> _locList = [];
WidgetsBinding.instance.addPostFrameCallback((timeStamp) {
        _subInitState();
      });
 _subInitState() {
    FriendVo _nextVo;
    for (int i = 0; i < _friendList.length; i++) {
      FriendVo vo = _friendList[i];
      bool isShow = true;
      String now = vo.pin;
      if (_nextVo != null){
        String back = _nextVo.pin;
        if (back == now) {
          isShow = false;
        }
      }
      _nextVo = vo;
      if (isShow) {
        var globalKey = _keyList[i];
        var offsetY = getY(globalKey.currentContext)-_top;
        print('now:${now}');
        print('offsetY:${offsetY}');
        if (!_filterList.contains(vo.pin)){
          _filterList.add(vo.pin);
          _locList.add(offsetY);
        }
      }
    }
    _indexKey.currentState.updateIndex(_filterList);
  }

3一姿、選中字母列表某個(gè)item

onVerticalDragUpdate: (DragUpdateDetails details) {
        print('onVerticalDragUpdate');
        print('globalPosition:${details.globalPosition}');
        _selectIndex = getIdex(context, details.globalPosition);
        double _bubbleY = details.globalPosition.dy;
        bool _bubbleHidden = false;
        if (widget.onSelect != null) {
          widget.onSelect(
              _itemList[_selectIndex < 0
                  ? 0
                  : (_selectIndex > (_itemList.length - 1)
                      ? _itemList.length - 1
                      : _selectIndex)],
              _bubbleY,
              _bubbleHidden);
        }
        setState(() {});
      },
      //按住屏幕移動(dòng)手指實(shí)時(shí)更新觸摸的位置坐標(biāo)
      onVerticalDragDown: (DragDownDetails details) {
        print('onVerticalDragDown');
        _selectIndex = getIdex(context, details.globalPosition);
        double _bubbleY = details.globalPosition.dy;
        bool _bubbleHidden = false;
        if (widget.onSelect != null) {
          widget.onSelect(
              _itemList[_selectIndex < 0
                  ? 0
                  : (_selectIndex > (_itemList.length - 1)
                      ? _itemList.length - 1
                      : _selectIndex)],
              _bubbleY,
              _bubbleHidden);
        }
        print('現(xiàn)在點(diǎn)擊的位置是${details.globalPosition}');
        setState(() {});
      },
      //觸摸開始
      onVerticalDragEnd: (DragEndDetails details) {
        print('onVerticalDragEnd');
        double _bubbleY = 0;
        bool _bubbleHidden = true;
        if (widget.onSelect != null) {
          widget.onSelect(
              _itemList[_selectIndex < 0
                  ? 0
                  : (_selectIndex > (_itemList.length - 1)
                      ? _itemList.length - 1
                      : _selectIndex)],
              _bubbleY,
              _bubbleHidden);
        } //觸摸結(jié)束
        setState(() {});
      },
      onTapDown: (TapDownDetails details) {
        print('onTapDown');
      },
      onTapUp: (TapUpDetails details) {
        print('onTapUp');
        double _bubbleY = 0;
        bool _bubbleHidden = true;
        if (widget.onSelect != null) {
          widget.onSelect(
              _itemList[_selectIndex < 0
                  ? 0
                  : (_selectIndex > (_itemList.length - 1)
                      ? _itemList.length - 1
                      : _selectIndex)],
              _bubbleY,
              _bubbleHidden);
        } //觸摸結(jié)束
        setState(() {});
      },

4七咧、選中字母列表某個(gè)item的回調(diào):

int index = _filterList.indexWhere((element) => element == value);
                  if (index > -1){
                    double max =  _scrollController.position.maxScrollExtent;
                    double offset = _locList[index];
                    if (offset > max){
                      _scrollController.jumpTo(max);
                    }else{
                      _scrollController.jumpTo(offset);
                    }
                  }

5尘喝、整體頁面結(jié)構(gòu)

@override
  Widget build(BuildContext context) {
    double height = MediaQuery.of(context).size.height;
    return Stack(
      children: [
        Positioned(
            left: 0,
            right: 0,
            top: 0,
            bottom: 0,
            child: SingleChildScrollView(
              controller: _scrollController,
              child: Column(
                children: _getItemList(),
              ),
            )
        ),
        Positioned(
            top: height/8,
            height: height/2,
            right: 0,
            child: IndexItem(
              key: _indexKey,
              onSelect: (String value,double offset,bool type) {
              
                if (type == false){
                  int index = _filterList.indexWhere((element) => element == value);
                  if (index > -1){
                    double max =  _scrollController.position.maxScrollExtent;
                    double offset = _locList[index];
                    if (offset > max){
                      _scrollController.jumpTo(max);
                    }else{
                      _scrollController.jumpTo(offset);
                    }
                  }
                }
              //懸浮氣泡
                _bubbleKey.currentState.updateLoc(value,offset,type);
              },
            )),
        Bubble(
          key: _bubbleKey,
        )
      ],
    );
  }

注意:此處一定要用SingleChildScrollView剂娄,因?yàn)長istView會(huì)重新創(chuàng)建、布局惶洲、繪制可見區(qū)域內(nèi)的item,一般會(huì)多繪制可見區(qū)域以外2-4個(gè)item衬横,所以不在屏幕的item通過key獲取的位置是0裹粤。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末终蒂,一起剝皮案震驚了整個(gè)濱河市蜂林,隨后出現(xiàn)的幾起案子遥诉,更是在濱河造成了極大的恐慌,老刑警劉巖噪叙,帶你破解...
    沈念sama閱讀 219,270評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件矮锈,死亡現(xiàn)場離奇詭異,居然都是意外死亡睁蕾,警方通過查閱死者的電腦和手機(jī)苞笨,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,489評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來子眶,“玉大人瀑凝,你說我怎么就攤上這事〕艚埽” “怎么了粤咪?”我有些...
    開封第一講書人閱讀 165,630評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長渴杆。 經(jīng)常有香客問我寥枝,道長,這世上最難降的妖魔是什么磁奖? 我笑而不...
    開封第一講書人閱讀 58,906評(píng)論 1 295
  • 正文 為了忘掉前任囊拜,我火速辦了婚禮,結(jié)果婚禮上比搭,老公的妹妹穿的比我還像新娘冠跷。我一直安慰自己,他們只是感情好身诺,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,928評(píng)論 6 392
  • 文/花漫 我一把揭開白布蔽莱。 她就那樣靜靜地躺著,像睡著了一般戚长。 火紅的嫁衣襯著肌膚如雪盗冷。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,718評(píng)論 1 305
  • 那天同廉,我揣著相機(jī)與錄音仪糖,去河邊找鬼。 笑死迫肖,一個(gè)胖子當(dāng)著我的面吹牛锅劝,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播蟆湖,決...
    沈念sama閱讀 40,442評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼故爵,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼!你這毒婦竟也來了隅津?” 一聲冷哼從身側(cè)響起诬垂,我...
    開封第一講書人閱讀 39,345評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤劲室,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后结窘,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體很洋,經(jīng)...
    沈念sama閱讀 45,802評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,984評(píng)論 3 337
  • 正文 我和宋清朗相戀三年隧枫,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了喉磁。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,117評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡官脓,死狀恐怖协怒,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情卑笨,我是刑警寧澤斤讥,帶...
    沈念sama閱讀 35,810評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站湾趾,受9級(jí)特大地震影響芭商,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜搀缠,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,462評(píng)論 3 331
  • 文/蒙蒙 一铛楣、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧艺普,春花似錦簸州、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,011評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至瑰步,卻和暖如春矢洲,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背缩焦。 一陣腳步聲響...
    開封第一講書人閱讀 33,139評(píng)論 1 272
  • 我被黑心中介騙來泰國打工读虏, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人袁滥。 一個(gè)月前我還...
    沈念sama閱讀 48,377評(píng)論 3 373
  • 正文 我出身青樓盖桥,卻偏偏與公主長得像,于是被迫代替她去往敵國和親题翻。 傳聞我的和親對(duì)象是個(gè)殘疾皇子揩徊,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,060評(píng)論 2 355

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