[搬運]flutter如何在Widget上疊加其他overlay widget

来源:https://www.cnblogs.com/ligun123/archive/2020/06/12/13099573.html
-Advertisement-
Play Games

原文在這裡 作者簡介:Jose,剛大學畢業,現帶領團隊負責維護Flutter的Material庫, Material是一個幫助團隊建設高質量用戶體驗的設計體系。 假設你的ui里有一個widget,並且您希望在該widget的頂部覆蓋一個浮動widget。 可能該widget被旋轉或應用了其他轉換,如 ...


原文在這裡

作者簡介:Jose,剛大學畢業,現帶領團隊負責維護Flutter的Material庫,
Material是一個幫助團隊建設高質量用戶體驗的設計體系。

假設你的ui里有一個widget,並且您希望在該widget的頂部覆蓋一個浮動widget。
可能該widget被旋轉或應用了其他轉換,如何將widget的位置和轉換信息傳遞到浮動widget呢?

你可以使用CompositedTransformTarget, CompositedTransformWidget、LayerLink、Overlay和OverlayEntry來完成上述操作。

在Flutter中,overlay允許您將視覺元素插入到overlay的堆棧中,從而將它們顯示在其他widget的頂部。使用OverlayEntry將widget插入到overlay中,同時可以使用Positioned和AnimatedPositioned來定位你的widget。當你需要把一個widget浮動在另一個widget上面時,就可以使用這種方法,並且這些widget都可以復用。

在這一篇文章中,開發者想要對現有的TextField增加自動建議功能,我們當然可以使用Stack來實現這個功能,但這種方法並不友好,具有很強的侵入性,還需要將整個屏幕設計為Stack。如本文所述,建議使用Overlay來實現這種效果,這種場景非常常見。

直接使用使用Overlay表現的很直觀,但在Flutter中實現起來卻有限困難:使用一個builder回調函數來創建OverlayEntry實例,並把它插入到Overlay的堆棧中,同時你還需要持有該OverlayEntry實例的引用,可以方便的對該實例進行更新、刪除操作。OverlayEntry的position、transformation依賴的widget必須也要存在於overlay中,否則就會發生衝突。因為overlay的MediaQuery的context不同於常規的contenxt。調用Overlay之前,在widget中使用padding或margin的時候就會發生類似問題。幸運的是,flutter已經為你處理了這些問題。(這一段還需要潤色,要銜接上下文中心思想)

如果你的overlay entry所依賴的“target”不在overlay堆棧中,那麼我們需要使用CompositedTransformTarget、CompositedTransformFollower和LayerLink將它們粘合在一起:

  • 用CompositedTransformFollower包裝需要浮動的widget

  • CompositedTransformFollower必須是CompositedTransformTarget的子孫接點

  • 用CompositedTransformTarget包裝你要跟隨依賴的“target”

  • 使用LayerLink將CompositedTransformTarget和CompositedTransformFollower粘在一起

下圖Gif中,藍色的Container不在overlay中,而綠色的在overlay中。藍色的Container作為CompositedTransformTarget的child節點,綠色的作為CompostedTransformFollower的child節點,他們通過LayerLink實例連接到一起。註意綠色overlay widget是如何知道藍色widget的bounds數據,因為藍色widget並不在overlay中:

建議你使用DartPad親自嘗試下。
下麵是實例代碼(作者Hans Muller):

import 'package:flutter/material.dart';

void main() {
  runApp(MaterialApp(home: Slide()));
}

class Indicator extends StatelessWidget {
  Indicator({ Key key, this.link, this.offset }) : super(key: key);

  final LayerLink link;
  final Offset offset;

  @override
  Widget build(BuildContext context) {
    return CompositedTransformFollower(
      offset: offset,
      link: link,
      child: Container(
        color: Colors.green,
      ),
    );
  }
}

class Slide extends StatefulWidget {
  Slide({ Key key }) : super(key: key);

  @override
  _SlideState createState() => _SlideState();
}

class _SlideState extends State<Slide> {
  final double indicatorWidth = 24.0;
  final double indicatorHeight = 300.0;
  final double slideHeight = 200.0;
  final double slideWidth = 400.0;

  final LayerLink layerLink = LayerLink();
  OverlayEntry overlayEntry;
  Offset indicatorOffset;

  Offset getIndicatorOffset(Offset dragOffset) {
    final double x = (dragOffset.dx - (indicatorWidth / 2.0)).clamp(0.0, slideWidth - indicatorWidth);
    final double y = (slideHeight - indicatorHeight) / 2.0;
    return Offset(x, y);
  }

  void showIndicator(DragStartDetails details) {
    indicatorOffset = getIndicatorOffset(details.localPosition);
    overlayEntry = OverlayEntry(
      builder: (BuildContext context) {
        return Positioned(
          top: 0.0,
          left: 0.0,
          child: SizedBox(
            width: indicatorWidth,
            height: indicatorHeight,
            child: Indicator(
                offset: indicatorOffset,
                link: layerLink
            ),
          ),
        );
      },
    );
    Overlay.of(context).insert(overlayEntry);
  }

  void updateIndicator(DragUpdateDetails details) {
    indicatorOffset = getIndicatorOffset(details.localPosition);
    overlayEntry.markNeedsBuild();
  }

  void hideIndicator(DragEndDetails details) {
    overlayEntry.remove();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text('Overlay Indicator')),
      body: Center(
        child: CompositedTransformTarget(
          link: layerLink,
          child: Container(
            width: slideWidth,
            height: slideHeight,
            color: Colors.blue.withOpacity(0.2),
            child: GestureDetector(
              onPanStart: showIndicator,
              onPanUpdate: updateIndicator,
              onPanEnd: hideIndicator,
            ),
          ),
        ),
      ),
    );
  }
}

上面例子展示瞭如何將浮動widget和它依賴的“target”粘合到一起使用。接下來我們要對CompositedTransformTarget做一些Transformation來展示這些widget的真正強大之處。你會註意到綠色的overlay widget也會自動被施加這些Transformation,這一切都得益於LayerLink。

下麵的gif圖中,藍色的container依然是不在overlay,綠色在。這一次我們對CompositedTransformTarget(即藍色container)做了一個旋轉的Transformation,如你所見,儘管CompositeTransformFollower位於overlay中,它依然感知得到“target”的位置和被施加的transformations。

建議你使用DartPad親自嘗試下。
下麵是實例代碼(作者Hans Muller):

import 'dart:math' as math;

import 'package:flutter/material.dart';

void main() {
  runApp(MaterialApp(home: Slide()));
}

class Indicator extends StatelessWidget {
  Indicator({ Key key, this.link, this.offset }) : super(key: key);

  final LayerLink link;
  final Offset offset;

  @override
  Widget build(BuildContext context) {
    return CompositedTransformFollower(
      offset: offset,
      link: link,
      child: Container(
        color: Colors.green,
      ),
    );
  }
}

class Slide extends StatefulWidget {
  Slide({ Key key }) : super(key: key);

  @override
  _SlideState createState() => _SlideState();
}

class _SlideState extends State<Slide> {
  final double indicatorWidth = 24.0;
  final double indicatorHeight = 300.0;
  final double slideHeight = 200.0;
  final double slideWidth = 400.0;

  final LayerLink layerLink = LayerLink();
  OverlayEntry overlayEntry;
  Offset indicatorOffset;

  Offset getIndicatorOffset(Offset dragOffset) {
    final double x = (dragOffset.dx - (indicatorWidth / 2.0)).clamp(0.0, slideWidth - indicatorWidth);
    final double y = (slideHeight - indicatorHeight) / 2.0;
    return Offset(x, y);
  }

  void showIndicator(DragStartDetails details) {
    indicatorOffset = getIndicatorOffset(details.localPosition);
    overlayEntry = OverlayEntry(
      builder: (BuildContext context) {
        return Positioned(
          top: 0.0,
          left: 0.0,
          child: SizedBox(
            width: indicatorWidth,
            height: indicatorHeight,
            child: Indicator(
                offset: indicatorOffset,
                link: layerLink
            ),
          ),
        );
      },
    );
    Overlay.of(context).insert(overlayEntry);
  }

  void updateIndicator(DragUpdateDetails details) {
    indicatorOffset = getIndicatorOffset(details.localPosition);
    overlayEntry.markNeedsBuild();
  }

  void hideIndicator(DragEndDetails details) {
    overlayEntry.remove();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text('Overlay Indicator')),
      body: Transform.rotate(
        angle: -math.pi / 12.0,
        child: Center(
          child: CompositedTransformTarget(
            link: layerLink,
            child: Container(
              width: slideWidth,
              height: slideHeight,
              color: Colors.blue.withOpacity(0.2),
              child: GestureDetector(
                onPanStart: showIndicator,
                onPanUpdate: updateIndicator,
                onPanEnd: hideIndicator,
              ),
            ),
          ),
        ),
      ),
    );
  }
}

總結

當你要實現一個widget浮動在另一個widget之上時,使用overlay,然後連接這些widget,非常簡單易用。本文中,你學會瞭如何使用LayerLink來粘合這些widget。現在該你自己動手了。

想瞭解更多關於Jose,可以訪問他的GitHub、LinkedIn
YouTubeInstagram


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • SQL--SQL詳解(DDL,DML,DQL,DCL) 博客說明 文章所涉及的資料來自互聯網整理和個人總結,意在於個人學習和經驗彙總,如有什麼地方侵權,請聯繫本人刪除,謝謝! 什麼是SQL? Structured Query Language:結構化查詢語言 SQL通用語法 SQL 語句可以單行或多 ...
  • 一、MongoDB用戶認證機制簡介 為了認證客戶端,你必須要添加一個對應的用戶到MongoDB。基本的步驟分為以下幾步: 用戶管理介面:db.createUser()方法可以創建一個用戶,添加完成後可以分配角色給用戶,第一個用戶必須是管理員,用來管理其他用戶。你也可以更新存在的用戶,必須修改密碼和權 ...
  • 第一步:開啟郵箱功能 -開啟發郵件功能 exec sp_configure 'show advanced options',1 –顯示為1時,代表功能開啟 reconfigure with override go exec sp_configure 'database mail xps',1 –顯示 ...
  • 所有知識體系文章,GitHub已收錄,歡迎老闆們前來Star! GitHub地址: https://github.com/Ziphtracks/JavaLearningmanual MySQL觸發器 一、什麼是觸發器 觸發器(trigger)是MySQL提供給程式員和數據分析員來保證數據完整性的一種 ...
  • redis交叉編譯 平臺: Windows: x86 x86_64 Linux: arm aarch64 armv8l 倉庫地址: https://github.com/huskar-t/redis 成品地址 github 藍奏 編譯過程 有需要過程的底下留言 ...
  • 數據存儲和有效期 在 redis 工作流程中,過期的數據並不需要馬上就要執行刪除操作。因為這些刪不刪除只是一種狀態表示,可以非同步的去處理,在不忙的時候去把這些不緊急的刪除操作做了,從而保證 redis 的高效 數據的存儲 在redis中數據的存儲不僅僅需要保存數據本身還要保存數據的生命周期,也就是過 ...
  • 1.啟動oracle埠監聽 lsnrctl start 2.啟動資料庫 sqlplus / as sysdba 回車 開啟資料庫:startup 關閉監聽 lsnrctl stop 關閉資料庫服務 shutdown ...
  • 一、findViewById函數和Toast類 (1)Button button1 = (Button) findViewById(R.id.button_1); 通過findViewById方法,去佈局中找出R.id.button_1名稱的Button按鈕,這個方法會返回一個View對象,然後進行 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...