0%

【文章內容使用 Gemini 1.5 Pro 自動翻譯產生】

介紹 Dart 3 alpha

12 月,我們發表了第一篇關於 Dart 3 的部落格文章,讓我們得以一窺未來。今天,在肯亞奈洛比舉行的 Flutter Forward 開發者活動現場,我們分享了 Dart 下一個主要版本的更廣泛願景,以及我們的第一個預覽版本 Dart 3 alpha

我們的 Dart 使命是創造最好的程式語言,以便在任何應用程式平台上構建高品質的應用程式。Dart 易於上手、易於學習,避免不必要的複雜性,並且內部一致。Dart 生產力高;應用程式開發人員面臨著持續的壓力,需要在更短的時間內交付具有不斷變化的需求的應用程式,同時還要保證品質和完成度。因此,我們在構建 Dart 時,非常注重實現快速開發、快速迭代和高品質。Dart 還具有 可移植性,涵蓋最廣泛的平台,使您能夠將應用程式部署到現在和未來的設備上。

多年來,我們一直致力於建立這些品質。藉由 Dart 3,我們打算邁出更大的一步,以顯著的方式改進和擴展易用性、生產力和可移植性。

Dart 3 是一個全新的 主要 版本。部分原因是為了表示新功能的重大進展,部分原因是就語義版本控制而言,它是一個 重大變更版本:我們正在將類型系統更改為 支援 完善的空值安全(在 Dart 2.12 及更高版本中,這是 可選的),並在 Dart 的核心函式庫中進行了相應的重大變更。讓我們深入了解細節!

Dart 3 生產力提升

更具表現力的 Dart 語言:記錄、模式和存取控制

過去幾年,我們以飛快的速度發展 Dart 語言,自 2.0 版以來 新增了 23 項新功能。在 Dart 3 中,我們預計會新增兩項主要功能,即記錄和模式,目標是讓使用結構化資料更有效率。

記錄允許您有效且簡潔地從任何現有資料建立匿名複合值,而無需宣告類別來儲存這些值的思維負擔。使用記錄,您可以輕鬆構建組合現有資料的新資料結構。例如,要返回一對值:

1
2
3
4
(int, int) geoLocation(String address) {
// ...
return (lat, long);
}

記錄允許您組合資料,而模式可以將複合資料解構為其組成部分。例如,要將上面 geoLocation 的返回值(由一對整數組成的記錄)解構為兩個單獨的整數變數 latlong,您可以使用如下模式宣告:

1
var (lat, long) = geoLocation(address);

模式是完全類型安全的,並在開發過程中進行檢查。

您還可以根據值的類型進行模式匹配,例如從類別的階層結構中匹配。switch 可以使用與類型匹配的模式,以及每種類型的單獨欄位,如下所示 calculateArea 的主體:

1
2
3
4
5
6
7
8
9
10
double calculateArea(Shape shape) {
return switch (shape) {
Square(side: var s) => s * s,
Circle(radius: var r) => pi * r * r,
Rectangle(
width: var w,
height: var h
) => w * h,
};
}

總體而言,我們正在添加大量模式,這些模式組合在一起使 Dart 在處理結構化資料時更具表現力。

結合模式,我們還透過幾個新的修飾符為類別新增了 能力控制

  • interface class:不能被繼承。
  • base class:禁用隱式介面,因此不能被實作。
  • final class:不能繼承、實作或混合使用該類別(在目前的函式庫之外)。
  • sealed class:與 abstract + final 相同,並且該類型被視為密封類型族的根,用於窮盡性檢查。例如,以上面的 Shape 類別階層為例。在 Shape 類型的 switch 語句中(如 calculateArea 函數),如果 switch 語句未處理密封類型的所有可能子類型,分析器將觸發錯誤。
  • mixin class:可以用作 mixin 的類別。

每個新功能都會增加語言的複雜性。為了確保 Dart 保持易於上手,類別預設是完全允許的,就像今天一樣,只有一個小例外,即打算用作 mixin 的類別現在必須使用 mixin 關鍵字。

透過直接平台函式庫互操作實現高效的平台整合

我們正在努力擴展 Dart 語言的互操作性,以便 直接從 Dart 調用平台函式庫。在 Apple 平台上,我們正在基於我們在 FFI for C interoperability 上的工作,它支援從 Dart 調用任何遵循 C 調用約定的 C 模組。到目前為止,它支援 C、Go 和 Rust 等語言。現在,我們正在為 iOS/macOS 上的 Swift 和 Objective-C 新增支援。在 Android 上,您可以利用 FFI 和 Android 的 Java Native Interface (JNI) 調用以 Kotlin 編寫的 Jetpack 函式庫和以 Java 編寫的 Android 函式庫。

在新的工具的支持下,Dart 可以根據原始 C/ObjC/Swift/Java/Kotlin 程式碼的標題/介面檔案自動建立具有 Dart 介面的繫結,以及這些繫結背後的跨語言互操作程式碼。有關使用此方法調用 Apple 的 Core Motion API 和 Android 的 HealthConnect API 的示範,請查看 範例應用程式

這些新的互操作機制目前尚處於實驗階段,但我們希望它們至少能在 Dart 3.0 穩定版中達到 beta 品質。我們歡迎您在 Java/KotlinC/ObjectiveC/Swift 的問題追蹤器中提供回饋。最後,我們也在努力改進 Dart 與 JavaScript 的互操作性;我們將在以後的部落格文章中提供更多相關資訊。

可移植性方面的進展

Dart 已經透過我們的原生和網路編譯器支援廣泛的目標平台。在 Web 上,我們目前編譯為 JavaScript。在原生設備上,我們目前支援 Intel 32 位和 64 位,以及 ARM 32 位和 64 位架構。我們正在努力為許多新興的設備和標準新增支援,以確保 Dart 為未來的設備做好準備。

對於 Web,我們正在努力支援 WebAssembly (Wasm),它作為一個平台中立的二進制指令格式,在 所有現代瀏覽器 中都日趨成熟。我們希望 Wasm 能夠使 Dart Web 應用程式的啟動速度與原生設備上的應用程式一樣快。

另一個越來越受關注的平台架構是 RISC-V,這是一種為廣泛使用而設計的開放標準指令集架構 (ISA)。Android 團隊最近就 他們對 RISC-V 的支援工作 進行了演講,我們認為它也將廣泛應用於嵌入式設備。Dart 3 alpha 中提供了對 RISC-V 的預覽支援。

最後,我們正在努力擴展我們目前對 ARM 設備的支援,使其也涵蓋 Windows 上的 ARM64 支援。

Dart 3 的重大變更

介紹了新的 Dart 3 功能後,讓我們來看看我們為了更一致且更易於上手的 Dart 而進行的重大變更。

Dart 3 類型系統:100% 完善的空值安全

我們在 2021 年的 Dart 2.12 中引入了完善的空值安全。在完善的類型系統中,您可以信任類型,這意味著當類型聲明變數不為空時,它就 永遠 不為空。這提高了生產力,因為可以在開發過程中而不是在生產環境中發現問題,並且使 Dart 編譯器能夠產生 更小、更優化的程式碼。正如我們在 12 月 所討論的,Dart 的獨特之處在於將 100% 完善的空值安全引入到現有語言中。

自 Dart 2.12 起,空值安全一直是一個可以 開啟關閉 的設定選項。在 Dart 3 中,所有程式碼都以 開啟 完善的空值安全運行。不再可以運行沒有空值安全的應用程式或在部分空值安全的混合模式下運行應用程式。

我們意識到停止支援不使用空值安全的運行可能會給現有的應用程式和套件帶來問題。但是,我們覺得現在是推進這一變更的合適時機。擁有一個單一的類型系統,其中空值安全始終開啟,可以減少開銷和複雜性。每當開發人員讀取一段 Dart 3 程式碼時,他們都可以放心,所有變數預設都不為空。在實作方面,Dart 的編譯器和運行時可以專注於支援單一的 Dart 程式碼運行方式,從而降低新增新功能的成本和複雜性。

Dart 生態系統已經展現出將現有程式碼遷移到空值安全的堅定決心和意願。截至今天,

  • pub.dev 上排名前 250 位的套件中有 100% 支援空值安全
  • pub.dev 上排名前 1000 位的套件中有 98% 支援空值安全
  • 只有 14% 的 flutter run 階段仍然在沒有空值安全的情況下運行

感謝生態系統為遷移所做的努力!

清理 Dart 3 核心函式庫和語言

隨著 Dart 語言和類型系統的發展,我們核心函式庫中的一些 API 變得冗餘或不夠理想。為了確保 Dart 保持易於上手,我們進行了一些重大變更以清理不必要的 API。有關詳細資訊,請參閱 GitHub 問題,#34233#49529。我們還移除了一個歷史語法,用於預設參數值(#2357)。

隨著 Dart 3 alpha 的完成,我們的注意力現在轉向了 Dart 3 beta。在那裡,我們希望透過新增一些新的 API 來進一步改進 Dart 核心函式庫。您可以在 回饋問題 中提供意見。

準備您的程式碼以迎接 Dart 3 穩定版

一般來說,我們相信大多數已經遷移到使用空值安全(使用 Dart 2.12 或更高版本)的程式碼都可以在 Dart 3 上運行。但是,一小部分套件和應用程式可能會受到上面討論的重大變更的影響。因此,為了讓您有時間準備今年晚些時候發佈的 Dart 3 穩定版,我們沒有計劃在 Dart 3 alpha 之後對 Dart 語言和核心函式庫進行任何進一步的重大變更。

為了利用已經遷移到使用空值安全的程式碼,我們在 Dart 3 中實作了向後相容性,這將允許遷移的套件在 Dart 3 中使用 pub get 解析,即使它們的 SDK 版本限制為 >=2.12 <3.0.0,只要它們不依賴於其他已停止使用的核心函式庫 API 或語言功能。有關更多詳細資訊,包括如何使用本地安裝的 Dart 3 alpha SDK 進行測試,請參閱我們的 Dart 3 空值安全文件

對於發佈在 pub.dev 儲存庫上的套件,我們使用 Dart 3 alpha 進行了分析,並使用 “Dart 3 ready” 標籤標記了通過的套件。這表示該套件很可能在今年晚些時候 Dart 3 穩定版發佈時可以運行。

Dart 3 工具清理

在 Dart 3 發佈之前,我們進行了 許多變更,將我們所有的終端開發者工具都移動到一個統一的 dart 開發者工具中,這使得 Dart 對新開發者來說更容易上手。我們計劃在接下來的幾個月中進行 額外的工具變更 以進一步清理。希望這些變更不會造成太大的破壞,但在 Dart 3 beta 中,工具可能會出現一些小的損壞。

後續步驟

Dart 3 alpha 今天已在 Dart 開發頻道和 Flutter master 頻道中提供。我們邀請您試用它,無論是嘗試記錄和模式等新語言功能,還是嘗試我們新的直接存取平台互操作,或者測試您的應用程式和套件的 Dart 3 相容性

我們將在接下來的幾個月中繼續改進 Dart 3,並希望在今年晚些時候發佈完整的 Dart 3 穩定版。請注意,大多數功能仍在開發中,因此在穩定版發佈之前可能會有所變更。在此之前,祝您使用 Dart 程式碼愉快!


介紹 Dart 3 alpha 最初發佈在 Medium 的 Dart 上,人們在那裡透過醒目顯示和回應這個故事來繼續討論。

【文章內容使用 Gemini 1.5 Pro 自動產生】

宣布 Flutter 新聞工具套件

This image shows Dash is reading newspaper

Google 的 Flutter 團隊與 Google 新聞倡議 (GNI) 密切合作,共同打造了 Flutter 新聞工具套件 - 一個免費的預建 Flutter 應用程式範本,用於加速新聞應用開發。一些早期採用者已經在短短 6 週的時間內建立了 iOS 和 Android 應用程式,聲稱與傳統的 iOS 和 Android 雙端開發相比,節省了高達 80% 的時間。

三個月前,我們在早期試用階段宣布了該工具套件,並收到了來自世界各地組織的壓倒性申請,因此我們很高興宣布 Flutter 新聞工具套件的第一個版本現在已正式推出!

為什麼我們要建立 Flutter 新聞工具套件?

如今,大多數人使用智慧型手機獲取新聞。除了行動網站以外,行動應用程式可以成為新聞出版商與現有讀者互動、接觸新受眾以及創造新收入來源的絕佳方式。但設計、實作和部署行動應用程式需要時間和精力。

對於財力和人力資源有限的新聞出版商來說,從頭開始建立應用程式是一項重大工程。在 Google,我們希望幫助開發人員取得成功並提高生產力,因此我們正在投資一個工具套件,該工具套件提供了新聞出版商所需的所有常見功能,讓他們可以專注於區分他們的品牌和內容,而無需撰寫樣板程式碼。

Flutter 新聞工具套件中包含哪些功能?

我們與全球業界專家和新聞出版商合作,以識別新聞應用程式範本的基本功能和使用者工作流程。有了這些預建的開箱即用功能,新聞出版商只需要連接他們的資料源並自訂 UI 以反映他們的品牌。

使用來自 新聞消費者洞察 的最佳實務,Flutter 新聞工具套件包含以下功能:

  • 使用者入門:透過推送通知權限重新吸引您的受眾;取得廣告追蹤權限
  • 帳戶建立/登入:透過您的平台和使用者社群媒體網路進行驗證(例如 Google 登入);建立無密碼帳戶
  • 內容供稿和內容頁面:顯示新聞文章、影片、圖片、廣告和電子報報名提示
  • 變現:輕鬆加入訂閱和不同的廣告格式
  • 搜尋:允許使用者透過關鍵字搜尋文章或點擊熱門搜尋項目
  • 設定:允許使用者針對通知、登入和廣告追蹤調整偏好設定

您可以直接使用這些預整合功能,也可以輕鬆地修改並將它們替換為您偏好的其他功能。

This image includes sample features provided in the Flutter News Toolkit
Flutter 新聞工具套件中提供的範例功能

早期採用者

Flutter 新聞工具套件已經開始幫助全球各地的新聞出版商加速行動策略並實現業務目標。例如,

  • The Standard(肯亞最大的新聞出版商之一)在建立 The Standard 新聞 時,聲稱節省了高達 80% 的時間。
  • 商業賦能公司 AnyMind Group 在 2 週內為 Khaosod Online(泰國最大的新聞網站之一)建立了應用程式。
  • Boldsports,一家尼日利亞的體育新聞新創公司,在短短幾天內就建立了一個早期版本。
  • Hespress(摩洛哥最大的新聞網站之一)在 6 週內完成了其 英文應用程式 開發,並報告說新應用程式在推出應用程式後的兩個月內, organically 獲得了 93% 的有機使用者,廣告收入增長了 50%。

讓我們用他們自己的話聽聽他們的感受。

開始使用

如果您一直在等待存取 Flutter 新聞工具套件,現在可以開始了!您現在可以直接存取 Github 上託管的程式碼

如果您有興趣採用新聞工具套件,但需要幫助自訂以滿足您的需求,我們也為您提供了支援!只需發送電子郵件至 [email protected],我們可以推薦值得信賴的 Flutter 代理商來幫助您使用 Flutter 新聞工具套件建立應用程式。

雖然 Flutter 新聞工具套件最初是為新聞組織設計的,但任何想要建立流程式內容的人都可以使用範例程式碼。無論您是想建立公司電子報應用程式(以 DiUS 建立的 Medius 為例),還是社群媒體應用程式,您都可能會發現使用者登入、社群分享和內容供稿等預建功能對您的應用程式有所幫助。

我們計畫繼續改進 Flutter 新聞工具套件,同時從出版商和開發人員那裡獲取更多資訊。別忘了查看我們的 影片教學,以瞭解 Flutter 新聞工具套件 的更多詳細資訊!

常見問題解答

什麼是 Flutter?

Flutter 是 Google 的開源 UI 工具套件,用於從單一程式碼庫建立適用於行動裝置、網頁和桌面的精美應用程式。Flutter 大幅減少了建立和發佈應用程式所需的時間。如今,Flutter 是 最受歡迎的跨平台 開發框架,在全球擁有超過 300 萬名開發人員。目前有超過 700,000 個 Flutter 應用程式,包括阿里巴巴、BMW、Google Pay、PUBG、Shein 和微信。

Flutter 容易學習嗎?

是的,對於那些了解 Java 等其他程式設計語言的開發人員來說,學習 Flutter 速度很快。此外,我們還為您提供了豐富的 學習資源 來幫助您入門。

我可以在應用程式中使用其他非 Google 服務,例如廣告、分析和訂閱嗎?

當然可以!Flutter 新聞工具套件只是一個範例。您不需要使用任何您不想要的服務。事實上,很容易移除或將現有整合替換為其他服務。例如,如果您想使用非 Google 廣告服務,您可以在 pub.dev 上找到其他 套件

我需要做多少工作才能發佈應用程式?

雖然新聞範本可以降低建立新聞行動應用程式的入門門檻,但此解決方案仍然需要 Flutter 開發工作。範本不支援非技術使用者透過無程式碼或所見即所得功能或工具(例如儀表板)來修改應用程式品牌和樣式。

我們已繪製出一個高層級的部署流程圖,假設有 2 名 Flutter 工程師參與專案。例如,Hespress 等早期採用者在 6 週內完成了應用程式開發。但我們也有像 AnyMind 這樣的客戶,他們只用 1 名工程師在 2 週內就推出了應用程式。

This image maps out the deployment process at a high-level
高層級的部署流程

如果我沒有內部的開發團隊,如何使用範本?

我們可以推薦值得信賴的 Flutter 代理商,他們可以幫助您將範本完善。您可以發送電子郵件至 [email protected] 獲取更多資訊。聲明:Google 與這些代理商無關,我們也不從他們那裡分享任何收入。


宣布 Flutter 新聞工具套件 最初發佈在 Flutter 上的 Medium,人們在那裡透過突出顯示和回應這個故事來繼續討論。

【文章內容使用 Gemini 1.5 Pro 自動翻譯產生】

使用 Isolate.run() 更好地管理 Isolate

Dart 2.19 引入了一個新函數,可以讓您僅用一行代碼即可輕鬆實現併發。

展示新 Isolate.run() 函數速度的基準測試

所有 Dart 程式碼都在 isolate 中運行。是否要實作多個 isolate 以在 Dart 程式中啟用 併發 取決於您。如何實作多個 isolate 則取決於 Dart 團隊,在 Dart 2.19 中,我們對此流程進行了巨大的升級,我們很興奮能與大家分享。來認識一下 Isolate.run() 吧!

run() 將設定和管理 isolate 的所有複雜性完全抽象成單個函數調用。使用一些基本元素來使用 isolate 已經有一段時間了。但是,即使 isolate 有了所有最近的效能改進,這個過程充其量也只是繁瑣的,最糟糕的情況下則容易出錯。

為了讓您了解 run() 的改進程度,本文將逐步分解先前使用低階基本元素建構功能的方法。然後,我們將其與使用 Isolate.run() 進行對比,並向您展示它在內部的運作方式。即使您以前從未使用過 isolate,我們也相信 run() 會讓您興奮地想要嘗試它們!

Isolate

Isolate 是一個相當簡單的概念。Isolate 本質上是 Dart 中的單個執行緒。它們讓您可以並行執行部分程式碼。您可以啟動新的並行執行(數量不限),並直接從 main(主執行緒或 main isolate)告訴它們要做什麼。Isolate 不共享記憶體;相反,它們透過來回傳遞訊息來進行通訊。因此,您不必擔心典型的多執行緒問題,例如競爭條件或互斥鎖和鎖。

聽起來很棒!但是如何使用它們呢?在 Isolate.run() 之前,事情變得棘手起來。

Isolate API 由低階基本元素組成,這些基本元素提供了廣泛的功能。當您需要自訂 isolate 的功能時,這種粒度非常棒。但是,當 必須 使用 isolate 時,這種粒度就不那麼好了。特別是因為幾乎所有 isolate 的使用案例都需要相同的設定和管理基本配置。這基本上意味著要將每個實作細節都公開給 Dart 的使用者自行處理。

讓我們來看看典型的 isolate 設定,以更好地理解 Isolate.run() 所解決的繁瑣過程。

使用 Isolate(之前)

您可以將 Isolate.spawn() 視為 isolate 的舊起點。Flutter 的 compute 函數就是建立在 spawn() 之上的。它接受一個方法作為其入口點參數,以及該方法的任何參數,以及 isolate 本身的其他配置。過去,此入口點只能是頂級或靜態方法。

1
Isolate.spawn(_readAndParseJson, filename);

Isolate 建立完成!開玩笑的。還差得遠呢。

調用 spawn 不會返回任何實際可用的東西。它只返回一個 isolate 物件,該物件只是確認 isolate 已啟動。

除了建立時傳遞的初始參數之外,main isolate 和 spawned isolate(由 spawn 建立的 isolate)無法直接通訊。實際上,即使您不需要從 spawned isolate 返回任何計算結果,您仍然需要某種驗證來確認計算成功,因此您總是需要返回一條訊息。

為了啟用通訊,您必須設定埠。在調用 spawn 之前,您需要建立一個 ReceivePort 物件。ReceivePort 物件的 sendPort 成員作為 spawn 的另一個參數傳遞給 spawned isolate。

這意味著您傳遞給 spawn 的函數必須 專門 配置為使用該 sendPort 執行某些操作。換句話說,您不能僅將現有函數與 isolate 一起使用。因此,您不要將僅讀取和解析 JSON 檔案的常規函數傳遞給 spawn,而是建立如下內容:

您的專用、isolate 友好型 JSON 解碼函數可以簡單地「返回結果」,然後就完成了,對嗎?不完全是。結果需要透過 responsePort 傳送。這就是 isolate 與埠通訊的方式。您可以使用另一個基本元素 Isolate.exit() 來有效地返回結果並同時關閉 spawned isolate:

1
Isolate.exit(responsePort, result);

exit() 函數將儲存 spawned isolate 中訊息的記憶體 傳輸 到 main isolate(而不是複製它),並安全地關閉 isolate。

讓我們把這些都串在一起。由於此範例中的 result 是解析的 JSON,您可能想要稍微解構它才能實際使用它。為了程式碼的簡潔性,我們應該將建立 receivePort 和 isolate 並等待它們的回應的三行程式碼放在它們自己的函數中,而不是直接放在 main() 中。

正在完成的工作相對簡單。正是實作細節的暴露讓它 感覺 起來很複雜,例如用於訊息傳遞的埠,以及需要一個專用函數來處理埠,而該函數與 isolate 無關。

錯誤處理

到目前為止的範例 仍然 不是一個真正的「完整」、可投入生產的實作。如果您不做任何錯誤處理,那將對您自己不利,但它通常在已經相當多的流程中被遺忘為一個額外的步驟。例如,如果沒有任何錯誤處理,未捕獲的異步錯誤導致 isolate 崩潰,您將不知道導致錯誤的原因,甚至不知道 發生了什麼事

涵蓋 isolate 的所有錯誤處理可能性將會很廣泛,但通常需要對程式碼進行一些額外的補充。

您至少可以將 errorsAreFatalonExitonError 參數加入到 spawn 調用中:

這可以確保即使 spawned isolate 在沒有傳送結果的情況下終止,或者發生任何未捕獲的錯誤,resultPort 都會收到一條訊息。將錯誤設定為致命意味著未捕獲的錯誤會退出 isolate 作為安全預防措施,以確保它一定會終止。

onExit 參數使 isolate 在退出時向埠傳送 nullonError 參數使未捕獲的錯誤向埠傳送兩個字串的列表(錯誤和堆疊追蹤的 toString)。

重複使用結果埠可以避免建立更多埠,因此您只需在一个地方查找訊息。但這也意味著您需要區分 onExitonError 訊息與結果值。在這裡,我們假設 JSON 必須是一個 Map,因此它不能是列表或 null。否則,您還必須將結果包裝在可以識別的內容中。您必須在埠訊息之上建立一個(微不足道的)訊息協定。

除此之外,您還可以檢查回應中的特定錯誤。其中一種情況是檢查 resultPort 是否為 null,這意味著 isolate 在沒有傳送結果的情況下終止:

另一種情況是檢查結果是否為列表,這意味著發生了未捕獲的錯誤:

然後,最後,處理實際的結果:

無論如何,您都希望將 spawn 放在 try 塊中,以檢查將入口點傳送到新的 isolate 是否失敗。如果失敗了,結果埠將不會收到任何訊息,並且需要關閉:

提供最基本的錯誤處理可以確保結果埠始終關閉,並且 *spawnAndReceieve 始終完成,無論 spawned isolate 如何退出。您還可以做得 更好,例如,透過捕獲錯誤和堆疊追蹤並將它們作為實際物件傳送回去,而不僅僅是像 onError 處理程式那樣的字串。

錯誤處理顯然會引入很多變化,以及決定如何處理它以及要考慮哪些因素的心理開銷。可以理解的是,它通常被忽略在基本 isolate 設定之外。

使用 Isolate(之後)

Isolate.run() 使用您以前必須自己使用的基本元素,在單個函數調用中設定 isolate 實作的所有部分:

沒有埠、沒有單獨的生成、退出或錯誤處理,也沒有特殊的返回結構。也許最好的部分是,您傳遞給 run 的入口點可以是任何現有函數:

此範例顯示了一個 異步 函數,但 run 可以同樣輕鬆地執行 同步 函數。run 函數本身始終異步返回,這才是最重要的。

入口點也可以是 函數表達式,直接在您調用 run 的位置內聯編寫。Isolate 和在其上編寫的任何更高級別的 API 不再局限於僅運行靜態或頂級函數。

不再需要額外的訊息參數,您可以避免在列表等資料結構中打包和解包參數。

您根本不需要考慮太多錯誤處理。run 函數結合了本地和遠端錯誤捕獲、處理和跨 isolate 通訊,並將結果公開為單個普通的(異步)錯誤,您可以在標準的 try/catch 中捕獲它。您可以忘記 isolate,並將其視為普通的函數。

Isolate.run() 可以讓程式碼更加簡潔和符合人體工程學。Flutter 的 compute 函數甚至也轉而使用 run 而不是 spawn

Isolate.run() 內部

看看 run 的實作。它深入研究了所有與 isolate 相關的低階 API(這在以前是您的工作),以構建一個「完美」、全面的 isolate 設定。它接受要執行的 computation 方法,並設定所有埠及其返回值,以考慮 isolate 之間的有效訊息傳遞。

對每個潛在情況都進行了 徹底的 錯誤處理。run 函數會檢查 isolate 是否在完成計算之前就已終止。如果計算拋出錯誤,則 isolate 將終止,並將相同的錯誤拋出到 main isolate。

如果發生未捕獲的異步錯誤,則 isolate 將終止,並將錯誤異步報告給 main isolate。如果 main isolate 首先終止,則 spawned isolate 將終止,並將情況視為未捕獲的異步錯誤。

最後,run 總是使用 exit 來安全地關閉。這意味著資料可以有效地在 isolate 之間傳輸,而無需實際複製它。

總結

run 函數非常適合啟動一個計算並等待結果。如果您想為 run 未涵蓋的內容(例如可以多次傳送和接收訊息的 長時間運行的 isolate)建構自己的 isolate 設定,則基本元素仍然可用。但是,在大多數情況下,應該使用單個 run 語句替換 spawn 及其所有支援配置,而不是任何其他配置。

如果您在 run 之前從未嘗試過 isolate 管理,那麼很難相信所有這些功能以前都必須由使用者來實作!Isolate.run()(在 Dart 2.19 和 Flutter 3.7 中可用)使程式碼更加符合人體工程學,並使 isolate 更易於使用。您將如何利用 run 節省的時間呢?


使用 Isolate.run() 更好地管理 Isolate 最初發佈在 Medium 的 Dart 上,人們在那裡透過突出顯示和回應這個故事來繼續討論。

【文章內容使用 Gemini 1.5 Pro 自動產生】

Flutter 3.7 新功能:Material 3 更新、iOS 改進等等!

我們很高興在 2023 年開始時發佈 Flutter 3.7!在 Flutter 3.7 中,我們作為社群,繼續改進框架,加入了一些很棒的新功能,例如:建立自訂選單列、階層式選單、更好地支援國際化的工具、新的除錯工具等等。

我們還繼續完善一些功能,例如全局選取、透過 Impeller 進行更快渲染、DevTools,以及始終如一的效能!

讓我們一起踏上快速旅程,探索 Flutter 3.7 中的新功能吧!

增強的 Material 3 支援

Material 3 支援在 3.7 中得到了極大的增強,以下 Widget 已遷移:

若要使用這些新功能,只需在應用程式的 ThemeData Widget 中開啟 useMaterial3 旗標。若要充分利用 M3 支援,您需要完整的 M3 色彩方案。您可以自己提供,使用新的 主題建構工具,或者 Flutter 可以使用 ThemeData 建構函數的 colorSchemeSeed 參數從單個種子顏色為您生成一個:

1
2
3
4
5
6
7
MaterialApp(
theme: ThemeData(
useMaterial3: true,
colorSchemeSeed: Colors.green,
),
// …
);

有關 Flutter 的 Material 3 支援的最新資訊,請查看 GitHub 上的 總體議題

若要親自試用這些元件,請查看 互動式演示,展示了所有新的 M3 功能:

選單列和階層式選單

Flutter 現在可以建立選單列和階層式上下文選單。

對於 macOS,可以使用 PlatformMenuBar Widget 建立選單列,它定義由 macOS 而不是 Flutter 渲染的平台原生選單列。

而且,對於所有平台,您都可以定義一個 Material Design 選單,它提供階層式選單列 (MenuBar),或者由另一個使用者介面元素觸發的獨立階層式選單 (MenuAnchor)。這些選單完全可自訂,選單項目可以是自訂 Widget,或者您可以使用新的選單項目 Widget (MenuItemButtonSubmenuButton)。

Impeller 預覽

團隊很高興地宣布,新的 Impeller 渲染引擎 已在 stable channel 上的 iOS 上 準備好進行預覽。我們相信 Impeller 的效能對於大多數應用程式來說將會達到或超過 Skia 渲染器,至於保真度,Impeller 實作了所有功能,只有一些很少使用的角落案例除外。我們預計將在即將到來的穩定版本中將 Impeller 設為 iOS 上的預設渲染器,因此請在 GitHub 上繼續提交 Impeller 回饋

儘管我們越來越有信心 iOS 上的 Impeller 將滿足幾乎所有現有 Flutter 應用程式的渲染需求,但 API 涵蓋範圍仍然存在一些差距。在 Flutter wiki 上列出了少數幾個剩餘的差距。使用者也可能會注意到 Skia 和 Impeller 之間在渲染方面的細微視覺差異。這些細微的差異可能是錯誤,因此請勿猶豫 提交議題

社群的貢獻極大地加速了我們在 Impeller 上的進展。特別是 GitHub 使用者 ColdPaleLightguoguo338JsouLiangmagicianA 為此版本貢獻了 291 個(>12%)與 Impeller 相關的修補程式中的 37 個。謝謝!

我們繼續在 Impeller 的 Vulkan 後端上取得進展(在較舊的設備上回退到 OpenGL),但 Android 上的 Impeller 尚未準備好進行預覽。Android 支援正在積極開發中,我們希望在將來的版本中分享更多相關資訊,包括更多關於桌面和 Web 支援的資訊。

在 GitHub 上的 Impeller 專案看板 上關注我們的進展。

iOS 版本驗證

當您發佈 iOS 應用程式時,更新設定的清單 可以確保您的應用程式已準備好提交到 App Store。

flutter build ipa 命令現在會驗證其中一些設定,並在發佈之前通知您是否需要對應用程式進行更改。

DevTools 更新

在此版本中,有幾個新的工具功能和整體改進可以嘗試。DevTools 記憶體除錯工具已徹底改版。有三個新的功能標籤,剖析追蹤差異,它們支援所有以前支援的記憶體除錯功能,並為您的除錯工作添加了更多功能。新功能包括能夠按類別和記憶體類型分析應用程式目前的記憶體配置、調查在運行時哪些程式碼路徑正在為一組類別配置記憶體,以及比較記憶體快照以了解兩個時間點之間的記憶體管理。

所有這些新的記憶體功能都已記錄在 docs.flutter.dev 上,因此請查看文件以獲取更多資訊。

效能頁面也有一些值得注意的新功能。效能頁面頂部的一個新的 畫面分析 標籤提供了所選 Flutter 畫面的洞察力。洞察力可能包括如何更詳細地追蹤 Flutter 畫面中昂貴部分的建議,或者有關在 Flutter 畫面中檢測到的昂貴操作的警告。

這只是幾個亮點,但此版本包含若干錯誤修復和改進,超出此處提到的功能,包括針對 Inspector、網路分析工具和 CPU 分析工具的一些重要錯誤修復。若要更深入地了解更新列表,請查看 DevTools 變更的發行備註,這些變更已加入 Flutter 3.7。

自訂上下文選單

您現在可以在 Flutter 應用程式中的任何位置建立自訂上下文選單。您也可以使用它們自訂內建的上下文選單。

例如,您可以將「發送電子郵件」按鈕加入到使用者選取電子郵件地址時出現的預設文字選取工具列中 (程式碼)。查看 contextMenuBuilder 參數,該參數已加入預設顯示上下文選單的現有 Widget 中,例如 TextField。您可以從 contextMenuBuilder 返回任何您想要的 Widget,包括修改預設的平台自適應上下文選單。

此新功能也適用於文字選取之外。例如,您可以建立一個 Image Widget,當右鍵點擊或長按時顯示「儲存」按鈕 (程式碼)。使用 ContextMenuController 在應用程式中的任何位置顯示目前平台的預設上下文選單或自訂上下文選單。

Flutter 的樣本儲存庫 中查看完整的範例套件。

CupertinoListSection 和 CupertinoListTile Widget

感謝 Github 使用者 Campovski 的努力,Cupertino 有兩個新的 Widget:CupertinoListSectionCupertinoListTile,用於以 iOS 風格顯示可捲軸的 Widget 列表。它們是 Material 中 ListViewListTile 的 Cupertino 版本。

捲軸改進

這個版本包含了一些 捲軸更新:針對觸控板互動的潤色和完善、新的 Widget(例如 ScrollbarsDraggableScrollableSheet),以及改進了在捲軸上下文中選取文字的處理方式。

值得注意的是,MacOS 應用程式現在將會在新增 新的捲軸物理 以匹配桌面平台後,體驗到更高的保真度。

新的 AnimatedGridSliverAnimatedGrid Widget 會為加入(或移除)列表的項目製作動畫。

最後,我們 修復了若干捲軸 Widget 的建構函數中的回歸,例如 ListView。在 Flutter 框架的 NNBD 遷移期間,itemBuilder(允許使用者按需提供 Widget)被遷移到 IndexedWidgetBuilder。這意味著 itemBuilder 不再可以返回 null,而 null(在過去)可以用於表示已到達列表的末尾。此功能已透過 NullableIndexedWidgetBuilder 恢復。感謝 @rrousselGit 注意到這一點——遷移幾年後——並發送了修復程式!

國際化工具和文件

國際化支援已完全改版!我們已完全重寫 gen-l10n 工具,以支援:

  • 描述性語法錯誤。
  • 包含巢狀/多個複數、選擇和佔位符的複雜訊息。

有關更多資訊,請查看更新後的 國際化 Flutter 應用程式 頁面。

全局選取改進

SelectionArea 現在支援鍵盤選取。您可以使用鍵盤快捷鍵(例如 shift+right)擴展現有選取範圍。

背景隔離區

現在,平台通道 可以從任何 隔離區 呼叫。以前,使用者只能從 Flutter 提供的主要隔離區呼叫平台通道。這使得在 PluginAdd-to-app 中使用隔離區和主機平台程式碼變得更好。有關更多資訊,請查看 flutter.dev 上的 撰寫自訂平台特定程式碼 以及深入文章 簡介背景隔離區通道,這是一篇免費的 Medium 文章。

文字放大鏡

在 Android 和 iOS 上文字選取期間出現的放大鏡現在可以在 Flutter 中使用。這對於所有具有文字選取的應用程式來說都是開箱即用的,但如果想停用或自訂它,請查看 magnifierConfiguration 屬性。

Plugin 的 Swift 遷移

隨著 Apple 將重點放在他們自己的 API 上的 Swift,我們希望開發一些參考來幫助 Flutter Plugin 開發人員使用 Swift 遷移或建立新的 Plugin。quick_actions Plugin 已從 Objective-C 遷移到 Swift,可以用作最佳實務的演示。如果您有興趣幫助我們遷移 1P Plugin,請查看 wiki 的 Swift 遷移部分

面向 iOS 開發人員的資源

我們為 iOS 開發人員發佈了一些新的資源,包括:

Bitcode 棄用

從 Xcode 14 開始,watchOS 和 tvOS 應用程式不再需要 Bitcode,App Store 不再接受來自 Xcode 14 的 Bitcode 提交。 因此,Bitcode 支援已從 Flutter 中移除。

預設情況下,Flutter 應用程式沒有啟用 Bitcode,我們預計這不會影響太多開發人員。但是,如果您在 Xcode 專案中手動啟用了 Bitcode,請在升級到 Xcode 14 後立即停用它。您可以透過打開 ios/Runner.xcworkspace 並將 啟用 Bitcode 設為 來執行此操作。 Add-to-app 開發人員應在主機 Xcode 專案中停用它。

若要進一步了解 Bitcode 分發,請查看 Apple 的文件

iOS PlatformView BackdropFilter

我們已添加了原生 iOS 視圖在渲染到模糊的 Flutter Widget 之下時被模糊的功能,並且 UiKitView Widget 現在可以包裝在 BackdropFilter 中。

有關更多資訊,請查看 iOS PlatformView BackdropFilter 設計文件。

記憶體管理

此版本對記憶體管理進行了一些改進,這些改進的共同效果是減少由垃圾收集暫停造成的卡頓,降低因配置速度和背景 GC 線程造成的 CPU 使用率,以及減小記憶體佔用空間。

例如,我們擴展了現有 的手動釋放支援某些 dart:ui Dart 物件的原生資源的做法。以前,原生資源將由 Flutter 引擎保留,直到 Dart VM 垃圾收集 Dart 物件。透過對使用者應用程式和我們自己的基準測試進行分析,我們確定這種策略通常不足以避免不合時宜的 GC 並過度使用記憶體。因此,在此版本中,Flutter 引擎添加了用於明確釋放 VerticesParagraphImageShader 物件持有的原生資源的 API。

在我們對遷移到此 API 的 Flutter 框架進行基準測試時,這些改進將 90% 的畫面建立時間縮短了 30% 以上,最終使用者將體驗到更平滑的動畫,卡頓更少。

此外,Flutter 引擎 不再註冊 GPU 圖片的大小到 Dart VM。如上所述,這些圖片在不再需要時已由框架手動釋放,因此通知 Dart 的 GC 策略支援 Dart 堆物件的 GPU 記憶體大小會不必要地增加 Dart 堆記憶體壓力,從而觸發無法收集任何額外記憶體的不合時宜的 GC。基於類似的思路,Flutter 引擎現在的策略是僅向 Dart VM 報告支援 dart:ui Dart 物件的原生物件的淺層大小

在我們的基準測試中,此變更在建立畫面時會消除同步 GC 工作,當時 Widget 會建立 GPU 常駐圖片。

在此版本中,Flutter 引擎也更好地動態更新 Dart VM,提供有關 Flutter 應用程式狀態的資訊。特別是,Flutter 現在使用 Dart VM 的 RAIL 風格 API 在路由轉場動畫期間進入 低延遲模式。在此低延遲模式下,Dart VM 的記憶體配置器優先選擇堆增長而不是垃圾收集,以避免使用 GC 暫停中斷轉場動畫。雖然此變更沒有帶來任何顯著的效能改善,但我們計劃在將來的版本中擴展此模型的使用範圍,以進一步消除不合時宜的 GC 暫停。此外,我們已 修復了邏輯中的錯誤,這些邏輯決定何時通知 Dart VM Flutter 引擎處於空閒狀態。修復這些錯誤還可以防止 GC 相關的卡頓。最後,對於 Add-to-app Flutter 應用程式,Flutter 引擎現在會 通知 Dart VM Flutter 視圖不再顯示。這現在導致 Dart VM 為與視圖關聯的隔離區觸發最終的重大 GC。此變更在沒有 Flutter 視圖可見時會減少 Flutter 的記憶體佔用空間。

停止支援 macOS 10.11 到 10.13

先前宣布,Flutter 不再支援 macOS 版本 10.11 和 10.12。自那次宣布以來,進一步的分析 顯示,移除對 10.13 的支援也會產生有限的額外影響,並且將有助於團隊大幅簡化程式碼庫。這意味著使用此版本及更高版本建立的針對穩定 Flutter SDK 的應用程式將不再在這些版本上運行,並且 Flutter 支援的最低 macOS 版本將升級到 10.14 Mojave。

因此,由於 Flutter 支援的所有 iOS 和 macOS 版本都包含 Metal 支援,因此 OpenGL 後端已從 iOS 和 macOS 嵌入器中移除。移除這些後端使 Flutter 引擎的壓縮大小減少了約 100KB。

toImageSync

此版本 添加了方法 Picture.toImageSyncScene.toImageSyncdart:ui 中,類似於異步方法 Picture.toImageScene.toImagePicture.toImageSync 同步返回一個 Image 的句柄,其中 Image 的光柵化會在背景中異步執行。然後,當 GPU 上下文可用時,圖片會保留為 GPU 常駐,這意味著與透過 toImage 生成的圖片相比,它的繪製速度更快。(透過 toImage 生成的圖片也可以保留為 GPU 常駐,但這種優化尚未在這種情況下實作。)

新的 toImageSync API 支援以下用例:

  • 迅速捕捉一個光柵化代價高昂的圖片,以便在多個畫面中重複使用。
  • 將多通道濾鏡應用於圖片。
  • 應用自訂著色器。

例如,Flutter 框架現在 使用此 API 來改進 Android 上的頁面轉場效能,這幾乎將畫面光柵化時間減半,減少了卡頓,並允許動畫在支援這些更新率的設備上達到 90/120fps。

自訂著色器支援改進

此版本包含 Flutter 對自訂片段著色器支援的許多改進。Flutter SDK 現在包含一個著色器編譯器,它會將 pubspec.yaml 檔案中列出的 GLSL 著色器編譯為目標平台的正確後端特定格式。此外,自訂著色器現在可以熱重新載入,以實現便捷的開發週期。自訂著色器現在也受 iOS 上的 Skia 和 Impeller 後端支援。

我們對社群已經分享的演示非常印象深刻,並且期待看到自訂著色器在 Flutter 中的進一步創新使用:

在 docs.flutter.dev 上查看 撰寫和使用自訂片段著色器 的詳細文件,並查看 pub.dev 上的實用程式套件 flutter_shaders

字體資產熱重新載入

以前,將新字體添加到 pubspec.yaml 檔案中需要重新建立應用程式才能看到它們,這與可以熱重新載入的其他資產類型不同。現在,對字體清單的更改(包括新增新字體)可以在應用程式中熱重新載入。

減少 iOS 設備上的動畫卡頓

感謝 luckysmg 的開源貢獻,兩個改進減少了 iOS 上的動畫卡頓。特別是,在手勢期間在主線程上添加一個虛擬 CADisplayLink 現在會強制以最大更新率刷新。此外,鍵盤動畫 現在會將 CADisplayLink 的更新率設定為與 Flutter 引擎的動畫器使用的相同更新率。由於這些更改,使用者應該會注意到 120Hz iOS 設備上更一致的平滑動畫。

總結

毫不誇張地說,沒有才華橫溢且充滿熱情的貢獻者社群,Flutter 就無法成為今天的傑出體驗。我們將繼續一起踏上這段旅程,Google 的 Flutter 團隊希望大家知道,沒有你們,我們是無法做到的。謝謝!

這股勢頭不會減弱,敬請關注未來的更新!


Flutter 3.7 的新功能 最初發佈在 Flutter 上的 Medium,人們在那裡透過突出顯示和回應這個故事來繼續討論。

【文章內容使用 Gemini 1.5 Pro 自動產生】

展示 Flutter 和機器學習的虛擬照片亭體驗

隆重推出 Flutter Forward Holobooth,這是一個展示 FlutterFirebase 和機器學習(使用 MediaPipeTensorFlow.js)功能的全新體驗,以虛擬照片亭呈現。首先選擇您的頭像(Dash 或 Sparky),然後將自己傳送到熱帶海灘、火山山脈、外太空、海底或其他地方!由於我們無法將所有人運送到奈洛比現場參加 Flutter Forward,因此我們想要提供同樣令人興奮的虛擬體驗!使用 Holobooth,您可以捕捉一段短片來紀念您的虛擬之旅。然後,透過分享到 Twitter 或 Facebook 來與您的朋友炫耀。

Flutter Forward Holobooth 網頁應用程式的登陸畫面。左邊,Dash 在一個以紫色和藍色色調以及 Flutter 標誌裝飾的照片亭裡拍照。右邊是一個開始按鈕。
holobooth.flutter.dev 上嘗試 Flutter Forward Holobooth

Holobooth 基於 Google I/O 2021 的第一版 Photo Booth 應用程式。Holobooth 沒有拍攝你和 Dash 或 Sparky 的照片,而是使用機器學習,透過你的臉部表情控制 Dash 或 Sparky 的動畫。

我們將分解說明我們的團隊如何與 Google 合作,透過運用 Google 工具的功能,打造更具沉浸感和未來感的照片亭體驗。我們使用 FlutterFirebase 建立 Holobooth 應用程式。JavaScript 中的 Web ML 讓我們能夠透過虛擬、互動式、3D 頭像將體驗提升到新的水平。讓我們深入了解我們是如何建立它的!

使用 TensorFlow.js 偵測臉部

Holobooth 最令人興奮的功能之一是能夠將您臉部的即時影片,映射到 Dash(或 Sparky)的 3D 模型上,同時您穿梭在他們的虛擬世界中。如果您臉部表達驚訝,Dash 的臉部也會表達驚訝,依此類推。為了實現這一點,我們使用了 Flutter Web 的 camera Plugin 和 TensorFlow.js 來偵測相機畫面中使用者的臉部。更具體地說,我們使用了 MediaPipe FaceMesh 模型,它可以實時估計 468 個 3D 臉部特徵點,以偵測 Web 和行動瀏覽器中相機畫面中使用者臉部的特徵。

一個穿著灰色襯衫和眼鏡坐在椅子上的男人。在他臉上是一堆紅點,映射到他的五官上。在他的眼睛周圍和嘴巴周圍有高濃度的紅點。
使用 MediaPipe FaceMesh 模型偵測到的特徵

根據每個臉部特徵的位置,我們可以判斷使用者是否在畫面中,他們的眼睛或嘴巴是否睜開,等等。當使用者在相機視圖中移動時,MediaPipe FaceMesh 模型(透過 TensorFlow.js Face Landmarks Detection 套件 提供)確保我們可以追蹤使用者特徵的確切座標,以便我們可以在 Dash 或 Sparky 上鏡像它們。如需更多詳細資訊,您可以深入了解 face_geometry.dart 檔案。雖然目前還沒有針對 TensorFlow.js 的官方 Dart 套件,但 Dart 的 JS 套件允許我們將 JavaScript 函式庫導入 Flutter Web 應用程式(如需更多詳細資訊,請參閱 tensorflow_models 套件資料夾)。

  FaceGeometry({
required tf.Face face,
required tf.Size size,
}) : this._(
rotation: FaceRotation(keypoints: face.keypoints),
leftEye: LeftEyeGeometry(
keypoints: face.keypoints,
boundingBox: face.boundingBox,
),
rightEye: RightEyeGeometry(
keypoints: face.keypoints,
boundingBox: face.boundingBox,
),
mouth: MouthGeometry(
keypoints: face.keypoints,
boundingBox: face.boundingBox,
),
distance: FaceDistance(
boundingBox: face.boundingBox,
imageSize: size,
),
);

const FaceGeometry._({
required this.rotation,
required this.mouth,
required this.leftEye,
required this.rightEye,
required this.distance,
});

使用 Rive 和 TensorFlow.js 動畫背景和頭像

我們求助於 Rive 來讓 Holobooth 動畫栩栩如生。Rive 是一個使用 Flutter 建立的 Web 應用程式,它提供了用於建立高性能、輕量級、互動式動畫的工具,這些動畫易於整合到 Flutter 應用程式中。我們與 Rive 的才華洋溢的設計師和 HOPR 設計工作室 合作,創造出可以在我們的 Flutter 應用程式中無縫運作的動畫 Rive 圖形。動畫背景和頭像都是 Rive 動畫。

左邊是一個向左、向右、向上、向下移動、眨眼,然後張嘴的臉。Dash 模仿了左邊臉移動的相同動作。
移動您的臉,看看 Rive 模型如何模仿您的行為

頭像使用 Rive 狀態機,讓我們可以控制頭像的行為和外觀。在 Rive 狀態機中,設計師指定所有輸入。輸入是由您的應用程式控制的值。您可以將它們視為設計和工程團隊之間的契約。您的產品程式碼可以隨時變更輸入的值,狀態機會對這些變更做出反應。

對於 Holobooth,我們使用輸入來控制嘴巴張開或閉合的程度。使用來自 FaceMesh 模型的特徵偵測,我們可以將它們映射到頭像模型上的對應座標。使用 StateMachineController,我們轉換來自模型的輸入,以確定頭像在螢幕上的顯示方式。

class CharacterStateMachineController extends StateMachineController {
CharacterStateMachineController(Artboard artboard)
: super(
artboard.animations.whereType<StateMachine>().firstWhere(
(stateMachine) => stateMachine.name == 'State Machine 1',
),

例如,頭像模型有一個屬性用於測量 嘴巴的張開程度(從 0 到 1 測量,其中 0 表示完全閉合,1 表示完全張開)。如果使用者在相機視圖中閉合嘴巴,應用程式會提供對應的值並將其饋送到頭像模型中,因此您會看到您的頭像嘴巴在螢幕上也閉合了。

使用 Firebase 捕捉動態照片

Holobooth 的主要功能是您可以分享的 GIF 或影片,以慶祝 Flutter Forward。我們求助於 Firebase 的雲端函數 來幫助我們生成和上傳您的動態照片到 Firebase 的雲端儲存空間。當您按下相機按鈕時,應用程式會開始捕捉畫面 5 秒鐘。使用 ffmpeg,我們使用雲端函數將畫面轉換為單個 GIF 和影片,然後上傳到 Firebase 的雲端儲存空間。您可以選擇下載 GIF 或影片以供日後觀賞,或手動上傳到社交媒體。

使用者選擇 Dash 作為頭像,然後選擇外太空的動畫背景,其中有行星和星星。一個火箭斜向上向左移動,在 Dash 身後。使用者選擇一頂藍色的巫師帽,上面有星星,搭配一件相同的襯衫和一個 Flutter 馬克杯,然後按下相機按鈕錄製動態照片。最終的照片在一個單獨的螢幕上顯示,其中有按鈕可以分享照片、下載照片或重新拍攝照片。
捕捉動態照片

若要將您的 GIF 直接分享到 Twitter 或 Facebook,您可以點擊分享按鈕。然後,您將被帶到所選平台,其中包含一個預先填寫的帖子,其中包含您的影片第一畫面的照片。若要觀看完整的影片,請點擊您的 holocard 連結 - 一個網頁,它會完整顯示您的影片,並有一個按鈕引導訪客自己嘗試 Holobooth!

Holocard 頁面,左邊是使用者動態照片的第一個畫面。Dash 穿著太空人服,站在未來城市前。右邊是 Flutter Forward 活動標誌,文字為“Check out my Flutter holocard”,以及一個顯示“Try now”的按鈕,使用者可以在其中使用 Holobooth 拍攝自己的照片。
Holocard 範例

挑戰和解決方案

Holobooth 包含許多元素,這些元素擴展了 Flutter 的可能性 - 例如使用機器學習模型和 Rive 圖形,同時確保為使用者提供高性能、流暢的體驗。

與 TensorFlow.js 合作對 Very Good Ventures 來說是第一次。目前沒有官方的 Flutter 函式庫,因此我們在這個專案上的早期工作很大一部分是專注於對可用模型進行實驗,以確定哪個模型適合我們的需求。當我們確定了特徵點偵測模型後,我們就必須理解模型輸出的資料,並將它們映射到 Rive 動畫。以下是一個早期探索臉部偵測的範例:

一個穿著淡藍色襯衫和紅色帽衫的男人正在螢幕上移動他的臉。他的臉周圍有一個藍色方框,紅點映射到他的臉上,他的眼睛和嘴巴周圍有高濃度的點。當他移動他的臉時,紅點隨著他的臉部特徵一起移動。
臉部偵測的早期探索

官方 Flutter 的 camera Plugin 為我們提供了開箱即用的許多功能,但目前它不支持在 Web 上串流圖片。對於 Holobooth,我們 分支 了 camera Plugin 以添加此功能。我們希望這種功能可以在 未來由官方 Plugin 支援

另一個挑戰是效能優化。錄製螢幕可能會是一項昂貴的操作,因為應用程式正在捕捉大量數據。我們還必須考慮使用者會從不同的瀏覽器和設備訪問這個應用程式。我們希望確保該應用程式具有高性能,並且無論使用者使用什麼設備,都能提供流暢的體驗。當在桌面端訪問 Holobooth 時,影片背景會動畫化並反映橫向模式。為了優化行動瀏覽器,背景是靜態的,並被裁剪以適合縱向模式。由於行動螢幕比桌面螢幕更小,因此我們還優化了圖片素材的解析度,以減少初始頁面載入時間和設備使用的數據量。

如需更詳細地了解我們是如何解決這些挑戰以及更多問題,您可以查看 開源程式碼。我們希望這能為想要嘗試 TensorFlow.js、Rive 和影片的開發人員提供靈感,甚至對於那些只是想優化他們 Web 應用程式的開發人員也能提供靈感。

展望未來

在建立這個演示過程中,我們想要探索 Flutter Web 應用程式以簡單、高性能且有趣的方式與 TensorFlow.js 模型整合的潛力。雖然我們建立的許多內容仍然是實驗性的,但我們對機器學習在 Flutter 應用程式中的未來感到興奮,它可以為使用者在任何設備上創造出令人愉悅的體驗!加入社群對話,告訴我們您的想法,以及您如何在您的下一個 Flutter 專案中使用機器學習。

在 Holobooth 中拍一段影片,並在社交媒體上與我們分享!


它是如何製作的:Holobooth 最初發佈在 Flutter 上的 Medium,人們在那裡透過突出顯示和回應這個故事來繼續討論。

【文章內容使用 Gemini 1.5 Pro 自動產生】

現在,從 Flutter 3.7 開始,我很高興地宣布,開發者可以從 *任何* 隔離區使用 Plugin 和平台通道。這一直是我們 [排名最高的議題之一](https://github.com/flutter/flutter/issues/13937),並且從 2018 年就開始存在。由於實作起來並不簡單,而且存在一種雖然繁瑣但可行的解決方案:始終從根隔離區(Flutter 提供的隔離區)使用 Plugin,因此它被降級了。但是,隨著 Flutter 的成熟,它越來越注重效能,遵循軟體界的古老格言「讓它運作,讓它正確,讓它快速」。選擇實作此功能是改善效能和讓 Flutter 更易於使用的幸福交集。因此,它成為了很容易做出的投資決定。

如果您想了解此功能的用法,請查看 GitHub 上的 [範例程式碼](http://tbd)。

用例

為什麼有人想要從背景隔離區使用 Plugin 呢?很明顯 Plugin 有其存在的必要性,因為並非世界上所有的程式碼都是用 Dart 編寫的。社群花費了數年的努力來讓這些程式碼透過 Plugin 變得可用,例如:path_provider 的尋找臨時目錄功能,或 flutter_local_notifications 的發佈通知功能。

接下來的邏輯問題是:「為什麼有人要在背景隔離區上執行程式碼?」答案是,有時您別無選擇,某些函式庫可能會在背景隔離區上呼叫回呼函式,例如 android_alarm_manager_plus。或者,應用程式可能會進行大量的計算,而開發者不希望這些計算干擾 UI。

在我幫助 Google 的其他團隊採用 Flutter 的時間裡,隨著產品的成熟,他們最終會遇到根隔離區成為瓶頸的情況,這幾乎是不可避免的。因此,我們需要確保框架中的所有內容都經過優化,並為使用者提供在必要時輕鬆卸載工作的工具。

以下是一個為背景隔離區通道設計的虛構具體用例:

想像一個使用 AI 從文字提示生成高解析度圖像的應用程式。使用者的先前創作儲存在 Firebase 雲端儲存空間中,並且有一個功能可以從使用者的手機匯出和分享創作。Flutter 應用程式啟動一個背景隔離區,它從 Firebase 雲端儲存空間下載圖像的 8k 版本,將圖像縮放到所需的匯出大小,將圖像儲存到相機膠捲,最後在匯出完成時發佈本地通知。

在此範例中,至少有 3 個 Plugin 是從背景隔離區使用的,一個用於從 Firebase 雲端儲存空間讀取,一個用於儲存到手機的相機膠捲,一個用於發佈本地通知。如果沒有背景隔離區通道,應用程式必須將 8k 圖像從根隔離區複製到背景隔離區,才能對其進行縮放。今天,使用 Dart 無法將其變成恆定時間運作。

快速範例

以下是一個使用新 API 從背景隔離區呼叫 shared_preferences Plugin 的快速範例:

import ‘package:flutter/services.dart’;
import ‘package:shared_preferences/shared_preferences.dart’;
void main() {
// 識別要傳遞給背景隔離區的根隔離區。
// (在 Flutter 3.7 中引入的 API)
RootIsolateToken rootIsolateToken = RootIsolateToken.instance!;
Isolate.spawn(_isolateMain, rootIsolateToken);
}
void _isolateMain(RootIsolateToken rootIsolateToken) async {
// 將背景隔離區註冊到根隔離區。
BackgroundIsolateBinaryMessenger
.ensureInitialized(rootIsolateToken);
// 您現在可以使用 shared_preferences Plugin。
SharedPreferences sharedPreferences =
await SharedPreferences.getInstance();
print(sharedPreferences.getBool(‘isDebug’));
}

技術細節

以下是平台通道工作原理的高階概觀:

當平台通道的結果被呼叫時,會有一個硬編碼跳轉到 platform 線程。為了讓背景隔離區通道能夠運作,必須儲存傳送訊息的隔離區,以便引擎可以在該隔離區的事件迴圈中排程結果。這透過使用 [Dart 的 ports](https://github.com/dart-lang/sdk/blob/eb9554d70e386bb3177f63509ba8f7e4bbf500a0/runtime/include/dart_native_api.h#L125) 來實作。Dart ports 儲存擁有它們的隔離區,並且是從 C API 排程這些隔離區的唯一方法。

另一個需要實作的內容是將背景隔離區與其根隔離區關聯起來的方法。這讓我很驚訝,但是為了在 Flutter 引擎被破壞時關閉平台通道,必須知道哪些背景隔離區與該引擎關聯。否則,背景隔離區可能會嘗試與正在被破壞的 Flutter 引擎通訊。這在最終的 API 中可以被視為一種後果,其中需要使用 RootIsolateToken 初始化 BackgroundIsolateBinaryMessenger

若要進一步了解實作,請查看 [隔離區平台通道](https://docs.google.com/document/d/1yAFw-6kBefuurXWTur9jdEUAckWiWJVukP1Iay8ehyU/edit#heading=h.722pnbmlqbkx) 設計文件。此文件還包含反向通訊的提議,但尚未實作或被接受。

感謝 Flutter 社群的支持,希望大家都能為這個新功能找到絕妙的用途。


介紹背景隔離區通道 最初發佈在 Medium 上的 Flutter,人們在那裡透過突出顯示和回應這個故事來繼續討論。

【文章內容使用 Gemini 1.5 Pro 自動產生】

探索可變字型和著色器

簡介

我是 Brian,Flutter 新手,六個月前加入了 Google 的 Flutter 團隊擔任 UX 工程師。作為 UX 工程師,我將我的圖形設計背景與編碼技能結合起來,幫助團隊構建 Flutter 本身,並展示 Flutter 的能力。這篇部落格文章分享了我作為混合設計師 - 開發人員的第一次 Flutter 實驗之一:一個簡單的以排版為主題的益智遊戲。它結合了我設計中的創意背景和對幾個您可能還沒有嘗試過的 Flutter API 的技術探索:FontVariationAnimationController。我創造這個應用程式玩得很開心,並透過它親身感受了 Flutter 對那些會編碼的設計師或擁有設計眼光的開發人員來說有多棒。

使用可變字型和著色器自訂 Flutter 文字 Widget 的調色盤。

排版

圖形設計師喜歡深入研究「排版」,這只是一個描述文字樣式的花哨詞彙。這包括選擇字型、大小、佈局、間距等等。透過所有這些可以設定的選項,字母本身也可以成為一種藝術形式!作為一名受過設計訓練的設計師,我知道我為這個探索性專案所創造的任何事物,都會以某種方式涉及使用 Flutter 深入研究排版。在過去幾年中,可變字型 已經成為設計師前所未有地控制文字樣式的一種有趣方法。這是我一直想進一步探索的領域,而 Flutter 對可變字型的支援,似乎是這個完美的機會。

您可能已經熟悉一般字型,它決定了文字區塊中每個字元的樣式。常規字型通常會提供我們可以選擇的幾個設定,例如粗細:粗體、常規或輕細。可變字型將這種自訂提升到一個新的層次。例如,在粗細方面,我們可能會選擇 100(極輕)到 1000(極粗)之間的任何數值設定。而且我們並不局限於粗細;字型創作者可以讓使用者設定各種項目,例如字母的寬度或下降字元(例如「p」和「y」等小寫字母下降的距離)的深度等等。所有這些不同的設定都稱為「軸」,就像圖表的軸一樣。

Roboto Flex 可變字型中的字母「g」,沿著寬度軸(從左到右)和粗細軸(從上到下)變化。這些只是它許多生效的軸中的兩個!

Flutter 提供了一種簡單的方法,可以在 TextStyle Widget 中透過使用 fontVariations 欄位來調整這些設定。例如,以下程式碼片段將在 Roboto Flex 可變字型中建立一個文字樣式,大小為 18,具有針對粗細(wght)和寬度(wdth)的可變軸設定:

1
2
3
4
5
6
7
8
TextStyle(
fontFamily: 'RobotoFlex',
fontSize: 18,
fontVariations:[
FontVariation('wght', 374),
FontVariation('wdth', 118)
],
)

若要進一步了解可變字型,請查看 Google Fonts 網站上的 可變字型資源

設定變化非常容易,但是我想要從一個變化設定動畫到另一個設定,以及在設定組合之間動畫。幸運的是,Flutter 和 Dart 讓將兩個現有的底層功能(FontVariationAnimationController)粘合在一起以製作我自己的自訂動態排版效果變得非常容易。

對於我的展示,我製作了一個 WonkyChar Widget 和一個 WonkyAnimPalette 協助類別,它提供了一種簡單的方法來選擇在螢幕上顯示的任何字母,控制文字大小,並輸入與可變字型軸相關的各種設定。WonkyChar Widget 還包含一個標準的 Flutter AnimationController 物件,我用它來為設定製作動畫。在此範例中,字母「M」將以 200 的大小顯示,並在 4 秒的動畫中將其粗細(粗體)從極細動畫到極粗:

1
2
3
4
5
6
7
8
9
10
11
WonkyChar(
text: 'M',
size: 200,
animationDurationMillis: 4000,
animationSettings: [
WonkyAnimPalette.weight(
from: 100,
to: 900,
),
],
)

自訂著色器

可變字型的效果,特別是加上動畫,非常棒,但我想要進一步提升我的視覺創造力,看看我可以用字母形狀做些什麼。順便說一句,我聽說 Flutter 剛剛加入了對自訂片段著色器的支援。著色器是在電腦的圖形處理單元 (GPU) 上執行的程式,讓開發人員可以在維持高畫面更新率的同時製作各種視覺效果。Flutter 支援使用 GLSL 撰寫著色器,GLSL 是最出名且文件最齊全的著色器語言之一,有許多書籍、網站、YouTube 影片和大型開放式線上課程 (MOOC) 中提供了許多教學和範例。我當然不是著色器或圖形程式的專家,但我以前曾玩過,所以這是另一個在建立自己的技能的同時探索 Flutter 功能的好機會。

注意:著色器是一個相當進階的編碼主題,有關將它們插入 Flutter 應用程式的詳細資訊超出了這篇部落格文章的範圍。如果您想自己嘗試撰寫一些著色器,請查看 Flutter 文件中的 [撰寫和使用片段著色器](https://docs.flutter.dev/development/ui/advanced/shaders)。這是一種真正引人入勝的編碼方法!

最初,只是為了展示一種可能性,我撰寫了一個簡單的 GLSL 著色器,以作為我其中一個 WonkyChar Widget(如上所述)的濾鏡,以製作以下效果:

應用著色器之前和之後的 `WonkyChar` Widget。

專案描述

這些原始元件 - 排版、可變字型支援和著色器 - 是一個有趣的開始,但它們需要實際整合到一個連貫的專案中。作為一個自我賦予的任務,以進一步了解 Flutter 和 Flutter 社群中的編碼,我製作了一個簡單的基於磁磚的益智遊戲 Type Jam,靈感來自 2022 Flutter 益智遊戲黑客大賽。在益智遊戲黑客大賽提示的背景下,針對這些想法進行集思廣益,我提出了以下概念:您,作為玩家,必須幫助一家陷入困境的字型公司透過解決一系列基於磁磚的益智遊戲來修復一個故障的字型檔案。

應用程式的每個畫面都有一個益智遊戲,由一個故障字母的混亂碎片組成,周圍環繞著完整的字母,根據可變字型軸(例如粗細或寬度)製作動畫。磁磚碎片上的故障效果是在每個頁面上透過不同的著色器來建立的,導致磁磚碎片看起來波浪狀、斷裂等等。而且,為了與通常的磁磚益智遊戲機制有所區別,我決定讓玩家透過旋轉磁磚來解決益智遊戲,而不是滑動它們,這樣一來,在完成後就可以看到整個字母,沒有空槽。回想起來,這可能讓益智遊戲變得過於容易,但我可以為 2.0 版想出更多挑戰!

在益智遊戲應用程式中應用可變字型加上著色器效果(在磁磚碎片上)。這個畫面的字母在「光學大小」軸上製作動畫。

如果您想看看它是如何整合在一起的,請查看 flutter/samples 儲存庫 中「experimental」資料夾中的 Type Jam 應用程式的 varfont_shader_puzzle 範例。正如我之前提到的,我仍然是 Flutter 的新手,而且我只對著色器有基本的了解,所以我希望在未來能夠改進它!目前,請將它視為一個快速展示,而不是模板。

結語

作為一名以設計為中心的 UX 工程師,將我的設計和技術興趣結合起來以製作有趣的排版,並讓其他人也能玩弄字母,這讓我感到非常滿足。此外,我親眼見證了 Flutter 作為一種創造性工具的潛力,它可以超越應用程式設計的典型模式。我製作這個專案玩得很開心,希望它能激勵其他設計師 - 編碼人員嘗試他們奇怪而有趣的想法,並將他們的設計作為真正的應用程式呈現出來!


使用 Flutter 製作有趣的排版 最初發佈在 Flutter 上的 Medium,人們在那裡透過突出顯示和回應這個故事來繼續討論。

【文章內容使用 Gemini 1.5 Pro 自動翻譯產生】

pub.dev 的螢幕截圖和自動發佈功能

從今天開始,pub.dev 上有一些新功能:您現在可以為套件宣告螢幕截圖,讓搜尋方式更具視覺化。而且現在發佈流程可以完全自動化,由 GitHub Actions 支援。

將螢幕截圖加入 pub.dev

pub.dev 的核心目標之一是讓開發者更容易找到適合特定用途的套件。當您搜尋 Widget 或其他視覺元件時,圖片扮演著重要的角色。我們現在在套件搜尋結果中顯示螢幕截圖,讓您更容易找到套件。

套件搜尋和套件頁面中的螢幕截圖

螢幕截圖的縮圖現在會顯示在搜尋結果中,您也可以篩選搜尋結果,僅列出包含螢幕截圖的套件。

螢幕截圖縮圖也會顯示在套件頁面上,點擊縮圖會觸發包含所有螢幕截圖的圖片輪播。

pubspec.yaml 中宣告螢幕截圖

將螢幕截圖加入到套件非常簡單。在 pubspec.yamlscreenshots 欄位中宣告檔案,並附上描述和路徑。以下是 animations 套件的範例。

1
2
3
4
5
6
7
8
name: animations
...
screenshots:
- description: '容器轉換模式的範例。'
path: example/screenshots/container_transform_lineup.webp
- description: '淡入淡出模式的範例。'
path: example/screenshots/fade_lineup.webp
...

更多詳細資訊,請參閱 dart.dev/tools/pub/pubspec#screenshots

pub.dev 的自動發佈

數千名 Dart 社群成員在 pub.dev 上發佈套件,造福所有 Dart 和 Flutter 開發者。傳統上,這些套件發佈者透過運行本地終端機命令 dart pub publish 並使用他們的 Google 帳戶進行驗證來發佈到 pub.dev。在幕後,refresh token 會儲存在他們開發機器上的設定檔中,讓他們無需再次驗證即可發佈新版本。這個流程簡單易懂,但需要許多手動步驟。我們很高興推出新的自動發佈流程,可以直接從 GitHub Action 發佈新的 pub.dev 版本。這有幾個好處:

  • 每個版本都會在 GitHub 中自動標記一致的版本標籤
  • 設定審查和批准新版本發佈的流程變得更容易。
  • 發佈過程由 GitHub 簽名的 token 進行驗證,確保沒有憑證需要儲存在磁碟上。

確保自動發佈的安全性

目前,一些使用者已採用將包含 refresh tokenpub-credentials.json 設定檔儲存在 GitHub Actions 上的秘密環境變數中的方法。透過這種方式,他們可以編寫一個 GitHub Action 工作流程,恢復設定檔並使用它來發佈套件的新版本。雖然這種流程在技術上可行,但工具設計時並未考慮到這一點,而且該流程有多個缺點:

  • pub-credentials.json 檔案可以用於發佈作者有權發佈的任何套件,而不僅僅是從設定的流程發佈的一個套件。
  • 如果洩露,pub-credentials.json 可以用於發佈新版本和新套件,偽裝成建立 pub-credentials.json 檔案的使用者。
  • pub-credentials.json撤銷 refresh token 的使用者介面 並不容易找到。

事實上,不止一次發生過套件發佈時意外包含了作者的 pub-credentials.json 檔案的情況,因為作者在 CI 作業中提取了檔案。發佈套件以及更新套件所需的憑證是一個非常嚴重的安全漏洞,我們最終在 dart pub publish實作了洩露偵測

我們新的自動發佈支援減輕了這些問題,可以使用以下其中一種方法驗證到 pub.dev:

  • 由 Github Actions 簽名的臨時 OIDC token
  • 可以從大多數 GCP 資源(如 Google Cloud Build 或其他雲端環境,例如 Azure 或 AWS)使用的 GCP 服務帳戶。

這些流程啟用了依賴長期秘密 token 的自動發佈,如果意外洩露,這些 token 很容易被濫用。如果您將發佈流程部署到自訂機器上,您當然可以依賴匯出的服務帳戶金鑰,但必須非常小心地妥善保護這些金鑰。

啟用安全發佈

只需幾個步驟,您就可以從 GitHub 啟用自動發佈 - 有關所有詳細資訊,請參閱 dart.dev/go/automated-publishing


pub.dev 的螢幕截圖和自動發佈 最初發佈在 Medium 的 Dart 上,人們在那裡透過醒目顯示和回應這個故事來繼續對話。

http://creativecommons.org/licenses/by/4.0/

【文章內容使用 Gemini 1.5 Pro 自動產生】

福爾摩斯與華生;花生醬與果醬;Flutter 與 Material!這些搭配都是天作之合!

Material 是一個設計系統,可以幫助您在行動、網頁和桌面跨平台構建出色的、無障礙的應用程式。

在 Flutter 中,Material 函式庫 為開發人員提供了 UI 所需的所有構建模組。自 2021 年 Google I/O 推出 Material 3 以來,Flutter 團隊一直在更新 Flutter 的 Material 函式庫以支援這些新的變更。

我於 2022 年 8 月加入 Flutter 的 Material 團隊。從那以後,我一直在幫助更新 Widget 以符合 Material 3 規格,同時支援開源 Flutter 社群中的 Material 3 使用者。

在本文中,我將向您展示 Material 3 的新功能,如何將它們整合到您的 Flutter 應用程式中,並向您更新接下來的規劃。

Flutter Material 3 範例應用程式

Material 3

若要查看所有最新的 Material 3 更新,請查看我們的 範例應用程式。它讓您可以與所有 Widget 進行實時互動。

此應用程式展示了更新後的組件,讓您可以在 Material 2 和 Material 3、亮模式和暗模式之間切換,並嘗試不同的顏色方案。

入門

若要將 Material 3 加入到您的應用程式中,請在主題的建構函式中將 useMaterial3 旗標設定為 true:

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      theme: ThemeData.light(useMaterial3: true),
      body: MyHomePage(),
    );
  }
}

有關遷移現有 Flutter 應用程式的資訊,請查看 [將 Flutter 應用程式遷移到 Material 3](https://blog.codemagic.io/migrating-a-flutter-app-to-material-3/),這是 Flutter 貢獻者 Taha Tesser 在 CodeMagic 上發表的文章。

在撰寫本文時,您可以 *選擇* 使用 Material 3。未來,它將成為 Flutter 應用程式的預設(以及強制性)Material 版本。

新的 Widget

若要充分利用 Widget 變更,請設定 useMaterial3 旗標。但是,有些 Widget 在 Material 3 中發生了很大的變更,因此已被新的 Widget 取代。以下是其中一些新 Widget:

NavigationBar & NavigationDrawer

BottomNavigationBar Widget 已被 NavigationBar 取代。它稍微高一點,而且沒有下拉陰影(表示高度)。

分段按鈕

分段按鈕讓您的使用者可以在 *單個 Widget* 中從多個選項中選擇可切換的選項。預設情況下,您只能選擇一個項目,除非您指定 multiSelectionEnabled 參數。

分段按鈕 - 單選和多選

填充按鈕

我們已增加了按鈕的選項,其中包括新的 FilledButton。此按鈕會建立一個有色且沒有高度的按鈕。FilledButton.tonalmethod 方法會將目前的背景設定為輔助顏色。

凸起、填充和輪廓按鈕

徽章

新的 Badge Widget 可用於為父 Widget 提供額外的資訊,通常表示計數或狀態變更。它們有大型和小型可供選擇:

附加到父圖示的徽章 Widget

產生顏色方案

Material 3 的一項全新的功能讓您可以從單個種子顏色設定整個應用程式的顏色主題。在主題的建構函式中設定 color scheme seed 參數,然後從 *單一* 輸入,Flutter 會為應用程式中 **所有** Widget 產生一個和諧的顏色方案。此方案在亮模式和暗模式下都有效!

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      theme: ThemeData.light(
        useMaterial3: true,
        colorSchemeSeed: Color.fromRGBO(188, 0, 74, 1.0);
        ),
      body: MyHomePage(),
    );
  }
}
從種子顏色產生的顏色方案和色調值

如果您想進一步自訂顏色方案,請訪問 Material 生成器應用程式。您可以定義輔助顏色和第三顏色以建立完全獨特的調色盤。最棒的是,它會匯出 Dart 檔案,您可以立即在 Flutter 應用程式中使用。

接下來的規劃

Flutter 的 Material 3 遷移仍在繼續。您可以在 將 Material 3 帶入 Flutter GitHub 議題中追蹤進度。您也可以貢獻!我們歡迎錯誤報告和修復,讓 Material 3 變得對像您一樣的 Flutter 開發人員來說更加出色。

我們最近的更新帶來了 SnackBarTabBarSlider Widget 來符合 Material 3 規格。到目前為止,我們已經遷移了 27/30 個組件,並加入了一系列新功能以支援動態顏色、文字主題、Android 12 功能等等。

如果您想更多地參與 Flutter 社群,請前往 flutter.dev/community,並在您最喜歡的社交平台上找到我們。在那裡見!


Flutter 的 Material 3 最初發佈在 Flutter 上的 Medium,人們在那裡透過突出顯示和回應這個故事來繼續討論。

http://creativecommons.org/licenses/by/4.0/

【文章內容使用 Gemini 1.5 Pro 自動產生】

加入我們參與 #17DaysOfFlutter

#17DaysOfFlutter 的橫幅

新年快樂!

為什麼不藉此機會提升您的 Flutter 技能來迎接新的一年呢?從 1 月 9 日開始,直至我們在 1 月 25 日舉辦的 Flutter Forward 活動 之前,我們將舉辦為期 17 天的 Flutter 活動,一場 Flutter 內容馬拉松,我們的 Flutter 專家將在星期一至星期五公開發表全新的內容。此外,您還有機會應用所學,並與 Flutter 社群分享您的作品。無論您是 Flutter 的新手,還是經驗豐富的 Flutter 專家,我們都有新的 YouTube 影片、Codelab、現場活動和其他精心策劃的內容,可以幫助您在學習過程中取得進展。

#17DaysOfFlutter 時間表

通往 Flutter Forward 的內容安排如下:

第一週

第一週將以趣味和遊戲為主!觀看開發者關係工程師 KhanhEricLearning to Fly 第二季 中打造他們首個 Web 遊戲(使用 Flame)。跟著他們學習遊戲開發,或按照自己的步調,使用我們的全新遊戲 Codelab 從頭開始打造一個遊戲。在本週稍晚,在 Widget of the Week 的新一集中,了解 Flame 遊戲引擎。我們提供多種方法來幫助您在 Flutter 中構建遊戲。

第二週

在第二週,我們將發佈另一個 Codelab 和一些關於 Material 3 和 UX 設計的部落格文章,這些文章將重新定義您對 Flutter 應用程式體驗的想法。然後觀看 1 月 17 日由 CraigFlutter YouTube 頻道 上主持的 The Boring Show - Live。為了結束 #17DaysOfFlutter 的第二週,我們將在一個新的 Decoding Flutter 影片中,揭開 dart fix 的神秘面紗。

社群遊戲提交

透過分享您自己的遊戲來參與 Flutter 社群!從 1 月 14 日開始,您將有機會參與遊戲編碼挑戰。為 Learning To Fly 的 Doodle Dash 和遊戲 Codelab 增添新功能或敵人,提升它們的功能。您可以在 17DaysOfFlutter 網站上找到更多關於如何與社群分享您的遊戲的詳細資訊。

請務必在 Twitter 上關注 FlutterDev,以獲取有關 #17DaysOfFlutter 的更新。

(為什麼是 17DaysOfFlutter?Flutter 團隊的人決定在星期三開始 Flutter Forward,而我們希望至少能有兩個完整的星期來準備內容!)


加入我們參與 #17DaysOfFlutter 最初發佈在 Flutter 上的 Medium,人們在那裡透過突出顯示和回應這個故事來繼續討論。