亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

浮動(dòng)AppBar中的textField焦點(diǎn)回滾問題解決

 更新時(shí)間:2022年08月16日 15:51:45   作者:shirnewei  
這篇文章主要為大家介紹了浮動(dòng)AppBar中的textField焦點(diǎn)回滾問題解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

完整問題描述

SliverAppBar的floating=true,pinned=false模式中嵌套的TextField,會(huì)在獲取焦點(diǎn)時(shí)觸發(fā)CustomScrollView滾動(dòng)到頂部。

問題表現(xiàn)

CustomScrollView和SliverAppBar的介紹和演示,參見官方文檔。

在floating=true和pinned=false 這兩個(gè)組合參數(shù)的模式下,SliverAppBar表現(xiàn)為:列表向上滑動(dòng)時(shí)隨列表向上滑動(dòng)直至消失。

列表在任何位置向下滑動(dòng)時(shí),會(huì)立即從上方滑入直至全部展現(xiàn)。

如果該組件內(nèi)嵌套了TextField,在列表上滑一段距離,再下滑至SliverAppBar及其內(nèi)嵌套的TextField出現(xiàn)時(shí)(此時(shí)列表尚未滑動(dòng)到頂端),點(diǎn)擊TextField使其獲取焦點(diǎn)以輸入文字,此時(shí)列表會(huì)立即滾動(dòng)至頂。

如圖:

初步探索

開始調(diào)試問題,嘗試了各種參數(shù)組合,只要pinned為true就沒有這個(gè)問題,因?yàn)镾liverAppBar總會(huì)展現(xiàn)在最頂端。然后想到了在獲取焦點(diǎn)的同時(shí),將CustomScrollView的physics設(shè)置為 NeverScrollableScrollPhysics(意為禁止?jié)L動(dòng)),此時(shí)并不影響CustomScrollView的滾動(dòng)位置,然后在輸入完成或失去焦點(diǎn)時(shí),再取消禁止?jié)L動(dòng)的狀態(tài),即可避免獲取焦點(diǎn)時(shí)列表滾動(dòng)至頂端的問題。解決代碼如下:

class CustomScrollTextFieldPage extends StatefulWidget {
  const CustomScrollTextFieldPage({Key? key}) : super(key: key);
  @override
  State<CustomScrollTextFieldPage> createState() =>
      _CustomScrollTextFieldPageState();
}
class _CustomScrollTextFieldPageState extends State<CustomScrollTextFieldPage> {
  final textController = TextEditingController();
  final editableTextController = TextEditingController();
  bool focused = false;
  final focusNode = FocusNode();
  final buttonFocus = FocusNode();
  final textFocus = FocusNode();
  @override
  void initState() {
    super.initState();
    focusNode.addListener(_onFocus);
  }
  @override
  void dispose() {
    focusNode.removeListener(_onFocus);
    super.dispose();
  }
  _onFocus() {
    setState(() {
      focused = focusNode.hasFocus;
    });
  }
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: GestureDetector(
        behavior: HitTestBehavior.translucent,
        onTapDown: () {
          FocusManager.instance.rootScope.requestFocus(FocusNode());
        },
        child: CustomScrollView(
          physics: focused ? const NeverScrollableScrollPhysics() : null,
          slivers: <Widget>[
            SliverAppBar(
              floating: true,
              pinned: false,
              expandedHeight: 250.0,
              flexibleSpace: FlexibleSpaceBar(
                expandedTitleScale: 1,
                title: Row(
                  crossAxisAlignment: CrossAxisAlignment.end,
                  children: [
                    Expanded(
                      child: TextField(
                        focusNode: focusNode,
                        controller: textController,
                        onEditingComplete: () {
                          FocusManager.instance.rootScope.requestFocus(FocusNode());
                        },
                        style: const TextStyle(color: Colors.white),
                        decoration: const InputDecoration(
                          border: UnderlineInputBorder(
                            borderSide: BorderSide(color: Colors.white),
                          ),
                          focusedBorder: UnderlineInputBorder(
                            borderSide: BorderSide(color: Colors.white),
                          ),
                        ),
                      ),
                    ),
                    Padding(
                      padding: EdgeInsets.symmetric(horizontal: 16),
                      child: IconButton(
                        visualDensity:
                            VisualDensity(horizontal: 0, vertical: -4),
                        padding: EdgeInsets.zero,
                        onPressed: () {
                          print('btn clicked');
                          buttonFocus.requestFocus();
                        },
                        focusNode: buttonFocus,
                        icon: Icon(Icons.heart_broken),
                      ),
                    )
                  ],
                ),
              ),
            ),
            SliverGrid(
              gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent(
                maxCrossAxisExtent: 200.0,
                mainAxisSpacing: 10.0,
                crossAxisSpacing: 10.0,
                childAspectRatio: 4.0,
              ),
              delegate: SliverChildBuilderDelegate(
                (BuildContext context, int index) {
                  return Container(
                    alignment: Alignment.center,
                    color: Colors.teal[100 * (index % 9)],
                    child: Text('Grid Item $index'),
                  );
                },
                childCount: 20,
              ),
            ),
            SliverFixedExtentList(
              itemExtent: 50.0,
              delegate: SliverChildBuilderDelegate(
                (BuildContext context, int index) {
                  return Container(
                    alignment: Alignment.center,
                    color: Colors.lightBlue[100 * (index % 9)],
                    child: Text('List Item $index'),
                  );
                },
              ),
            ),
          ],
        ),
      ),
    );
  }
}

這個(gè)解決方法有點(diǎn)不完美的表現(xiàn),就是輸入完成時(shí)不點(diǎn)擊頁(yè)面,而是直接點(diǎn)擊收起鍵盤,這時(shí)不會(huì)觸發(fā)onTapDown也不會(huì)觸發(fā) onEditingComplete ,就需要在屏幕再點(diǎn)擊或者滑動(dòng)時(shí)才能重置列表的可滾動(dòng)狀態(tài)。

更好的解決辦法

經(jīng)過進(jìn)一步測(cè)試,發(fā)現(xiàn)在輸入框內(nèi)的EditableText中對(duì)focus進(jìn)行了監(jiān)聽,在獲取焦點(diǎn)時(shí)遞歸調(diào)用了RenderObject的showOnScreen方法,會(huì)一直向上追溯Render樹,最終調(diào)用到RenderSliverList中,觸發(fā)了滾動(dòng)事件。

是不是可以在TextField外包裹一個(gè)自定義了RenderBox的組件,把這個(gè)showOnScreen調(diào)用給切斷呢?于是翻了下官方的幾個(gè)組件寫法,照貓畫虎寫了個(gè)自定義的組件

class IgnoreShowOnScreenWidget extends SingleChildRenderObjectWidget {
  const IgnoreShowOnScreenWidget({
    Key? key,
    Widget? child,
    this.ignoreShowOnScreen = true,
  }) : super(key: key, child: child);
  final bool ignoreShowOnScreen;
  @override
  RenderObject createRenderObject(BuildContext context) {
    return IgnoreShowOnScreenRenderObject(
      ignoreShowOnScreen: ignoreShowOnScreen,
    );
  }
}
class IgnoreShowOnScreenRenderObject extends RenderProxyBox {
  IgnoreShowOnScreenRenderObject({
    RenderBox? child,
    this.ignoreShowOnScreen = true,
  });
  final bool ignoreShowOnScreen;
  @override
  void showOnScreen({
    RenderObject? descendant,
    Rect? rect,
    Duration duration = Duration.zero,
    Curve curve = Curves.ease,
  }) {
    if (!ignoreShowOnScreen) {
      return super.showOnScreen(
        descendant: descendant,
        rect: rect,
        duration: duration,
        curve: curve,
      );
    }
  }
}

使用方法

class CustomScrollTextFieldPage extends StatefulWidget {
  const CustomScrollTextFieldPage({Key? key}) : super(key: key);
  @override
  State<CustomScrollTextFieldPage> createState() =>
      _CustomScrollTextFieldPageState();
}
class _CustomScrollTextFieldPageState extends State<CustomScrollTextFieldPage> {
  final textController = TextEditingController();
  final focusNode = FocusNode();
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: GestureDetector(
        behavior: HitTestBehavior.translucent,
        onTap: () {
          FocusManager.instance.rootScope.requestFocus(FocusNode());
        },
        child: CustomScrollView(
          slivers: <Widget>[
            SliverAppBar(
              floating: true,
              pinned: false,
              expandedHeight: 250.0,
              flexibleSpace: FlexibleSpaceBar(
                expandedTitleScale: 1,
                title: Row(
                  crossAxisAlignment: CrossAxisAlignment.end,
                  children: [
                    Expanded(
                      child: IgnoreShowOnScreenWidget(
                        child: TextField(
                          focusNode: focusNode ,
                          controller: textController ,
                          style: const TextStyle(color: Colors.white),
                          decoration: const InputDecoration(
                            border: UnderlineInputBorder(
                              borderSide: BorderSide(color: Colors.white),
                            ),
                            focusedBorder: UnderlineInputBorder(
                              borderSide: BorderSide(color: Colors.white),
                            ),
                          ),
                        ),
                      ),
                    ),
                    Padding(
                      padding: EdgeInsets.symmetric(horizontal: 16),
                      child: IconButton(
                        visualDensity:
                            VisualDensity(horizontal: 0, vertical: -4),
                        padding: EdgeInsets.zero,
                        onPressed: () {
                          print('btn clicked');
                        },
                        icon: Icon(Icons.heart_broken),
                      ),
                    )
                  ],
                ),
              ),
            ),
            SliverGrid(
              gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent(
                maxCrossAxisExtent: 200.0,
                mainAxisSpacing: 10.0,
                crossAxisSpacing: 10.0,
                childAspectRatio: 4.0,
              ),
              delegate: SliverChildBuilderDelegate(
                (BuildContext context, int index) {
                  return Container(
                    alignment: Alignment.center,
                    color: Colors.teal[100 * (index % 9)],
                    child: Text('Grid Item $index'),
                  );
                },
                childCount: 20,
              ),
            ),
            SliverFixedExtentList(
              itemExtent: 50.0,
              delegate: SliverChildBuilderDelegate(
                (BuildContext context, int index) {
                  return Container(
                    alignment: Alignment.center,
                    color: Colors.lightBlue[100 * (index % 9)],
                    child: Text('List Item $index'),
                  );
                },
              ),
            ),
          ],
        ),
      ),
    );
  }
}

初步嘗試,確實(shí)可以更方便地解決問題。

效果如圖:

目前還未發(fā)現(xiàn)有什么副作用,如果哪位大神有更好的解決辦法,

以上就是浮動(dòng)AppBar中的textField焦點(diǎn)回滾問題解決的詳細(xì)內(nèi)容,更多關(guān)于AppBar浮動(dòng)textField焦點(diǎn)回滾的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Android xml文件的序列化實(shí)現(xiàn)代碼

    Android xml文件的序列化實(shí)現(xiàn)代碼

    Android提供了XmlSerializer來實(shí)現(xiàn)XML文件的序列化。相比傳統(tǒng)方式,更高效安全,需要的朋友可以參考下
    2014-02-02
  • Android?實(shí)現(xiàn)卡片堆疊錢包管理動(dòng)畫效果

    Android?實(shí)現(xiàn)卡片堆疊錢包管理動(dòng)畫效果

    這篇文章主要介紹了Android?實(shí)現(xiàn)卡片堆疊錢包管理動(dòng)畫效果,實(shí)現(xiàn)思路是在動(dòng)畫回調(diào)中requestLayout?實(shí)現(xiàn)動(dòng)畫效果,用Bounds?對(duì)象記錄每一個(gè)CardView?對(duì)象的初始位置,當(dāng)前位置,運(yùn)動(dòng)目標(biāo)位置,需要的朋友可以參考下
    2022-07-07
  • Android 中圖片和按鈕按下狀態(tài)變化實(shí)例代碼解析

    Android 中圖片和按鈕按下狀態(tài)變化實(shí)例代碼解析

    這篇文章通過實(shí)例代碼給大家總結(jié)了android 中圖片和按鈕按下狀態(tài)變化問題,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),感興趣的朋友跟隨腳本之家小編一起學(xué)習(xí)吧
    2018-06-06
  • Android 常見的圖片加載框架詳細(xì)介紹

    Android 常見的圖片加載框架詳細(xì)介紹

    這篇文章主要介紹了Android 常見的圖片加載框架詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下
    2016-12-12
  • 深入學(xué)習(xí)Android?ANR?的原理分析及解決辦法

    深入學(xué)習(xí)Android?ANR?的原理分析及解決辦法

    Android系統(tǒng)中,AMS和WMS會(huì)檢測(cè)App的響應(yīng)時(shí)間,如果App在特定時(shí)間無(wú)法相應(yīng)屏幕觸摸或鍵盤輸入時(shí)間,或者特定事件沒有處理完畢,就會(huì)出現(xiàn)ANR。本文將帶領(lǐng)大學(xué)深入學(xué)習(xí)一下ANR的原理及解決辦法,感興趣的同學(xué)可以學(xué)習(xí)一下
    2021-11-11
  • flutter 微信聊天輸入框功能實(shí)現(xiàn)

    flutter 微信聊天輸入框功能實(shí)現(xiàn)

    這篇文章主要介紹了flutter 微信聊天輸入框功能實(shí)現(xiàn),本文通過實(shí)例圖文相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-03-03
  • android?studio?項(xiàng)目?:UI設(shè)計(jì)高精度實(shí)現(xiàn)簡(jiǎn)單計(jì)算器

    android?studio?項(xiàng)目?:UI設(shè)計(jì)高精度實(shí)現(xiàn)簡(jiǎn)單計(jì)算器

    這篇文章主要介紹了android?studio?項(xiàng)目?:UI設(shè)計(jì)高精度實(shí)現(xiàn)簡(jiǎn)單計(jì)算器,自主完成一個(gè)簡(jiǎn)單APP的設(shè)計(jì)工作,綜合應(yīng)用已經(jīng)學(xué)到的Android?UI設(shè)計(jì)技巧,下面來看看實(shí)驗(yàn)實(shí)現(xiàn)過程
    2021-12-12
  • Android Studio中使用lambda表達(dá)式的方法

    Android Studio中使用lambda表達(dá)式的方法

    這篇文章主要介紹了Android Studio中使用lambda表達(dá)式的方法,需要的朋友可以參考下
    2017-06-06
  • Android實(shí)例代碼理解設(shè)計(jì)模式SOLID六大原則

    Android實(shí)例代碼理解設(shè)計(jì)模式SOLID六大原則

    程序設(shè)計(jì)領(lǐng)域, SOLID (單一功能、開閉原則、里氏替換、接口隔離以及依賴反轉(zhuǎn))是由羅伯特·C·馬丁在21世紀(jì)早期 引入的記憶術(shù)首字母縮略字,指代了面向?qū)ο缶幊毯兔嫦驅(qū)ο笤O(shè)計(jì)的基本原則
    2021-10-10
  • Android?RecyclerChart其它圖表繪制示例詳解

    Android?RecyclerChart其它圖表繪制示例詳解

    這篇文章主要為大家介紹了Android?RecyclerChart其它圖表繪制示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-12-12

最新評(píng)論