Flutter 中的 Widget Test 與 Routing 驗證

更新於 發佈於 閱讀時間約 18 分鐘


raw-image

在開發 Flutter 時,我們可以寫 Widget Test 確保功能在我們重構之後,還是保持正常運作。我們會針對許多不同的情境進行測試,其中一種情境是當使用者進行某些操作,或者當某些情況發生,把使用者導到其他頁面,今天就來分享如何使用 Widget Test 驗證 Routing。

舉個例子

假設我們有常見的清單頁面,其中列滿了各種狗狗品種,當我們點擊了某一個品種之後,App 會把使用者導向另一個頁面,並向隨機呈現一張該品種的圖片。在這個例子中,我們使用 DogAPI,有興趣的觀眾也可以參考看看。

raw-image

這個需求並不複雜,經過一番操作之後,我們在相對應的 ListTitle 上加上 GestureDetector 並使用 Navigator 把使用者導到下一個頁面,也告訴下一個頁面要顯示哪種品種的狗狗,最後成功在畫面上隨機顯示一張該品種的狗狗圖片。

class BreedListPage extends StatelessWidget {

...

@override
Widget build(BuildContext context) {
return ListView.builder(
itemCount: breedList.length,
itemBuilder: (context, index) {
return GestureDetector(
onTap: () => Navigator.of(context).pushNamed(
"/dog_image",
arguments: breedList[index],
),
child: ListTile(
title: Text(breedList[index]),
),
);
},
);
}
}

但是當我們完成功能之後,可能會思考,我們該如何進行測試呢?我們該如何功能正確,並且在未來也都持續保持正確呢?

Arrange 與 Act

對測試熟悉的觀眾,可能很快就能完成 Arrange 與 Act 的部分,關於測試 3A 原則可以參考這邊。在下面測試中,我們使用 mocktail 套件準備 DogAPI 的回傳結果,顯示 BreedListPage,接著我們點擊其中一個品種,一切都輕鬆寫意。

testWidgets('should open dog image page when click breed tile', (tester) async {
// Arrange
MockClient mockClient = MockClient();
when(() => mockClient.get(Uri.parse("https://dog.ceo/api/breeds/list/all"))).thenAnswer((_) async {
return Response('{"message": {"affenpinscher": [], "african": [], "airedale": []}, "status": "success"}', 200);
});

await tester.pumpWidget(
MaterialApp(
home: BreedListPage(client: mockClient)
));
await tester.pump();

// Act
await tester.tap(find.text("affenpinscher"));

// How to assert routing?

});

class MockClient extends Mock implements Client {}

接下來的問題便是,我們如何驗證 Routing 是否成功呢?

Mock NavigatorObserver

如果有使用過 firebase_analytics 的觀眾,可能會知道可以使用套件中的 FirebaseAnalyticsObserver 協助我們追中使用 Routing 狀況,當 App 進行 Routing 時,會呼叫 NavigatorObserver.didPush 方法並透過參數告知當前 Route 與上一個 Route,此時 firebase_analytics 套件就有機會追蹤使用者的 Routing 行為。

同樣地,我們也可以 mock 一個測試用的 MockNavigatorObserver,並驗證 didPush 方法是否有被呼叫,來達到驗證 Routing 的效果,那就讓我們使用 MockRoutingObserver 來驗證一下上述例子吧。

在下面例子中,我們宣告了一個 MockNavigatorObserver,並把它傳給 MaterialApp,由此我們就能在測試中監聽 App Routing 的狀況。在 Assert 的地方中,我們使用驗證了 mockNavigatorObserver.didPush 是否有被呼叫,除此之外,我們還使用 captureAny() 來捕捉參數,驗證參數中的 Route 名稱是否符合預期。

testWidgets('should open dog image page when click breed tile', (tester) async {
// Arrange
...

MockNavigatorObserver mockNavigatorObserver = MockNavigatorObserver();
await tester.pumpWidget(
MaterialApp(
home: BreedListPage(client: mockClient),
navigatorObservers: [mockNavigatorObserver],
));
await tester.pump();

// Act
await tester.tap(find.text("affenpinscher"));

// Assert
var result = verify(() => mockNavigatorObserver.didPush(captureAny(), any()));
expect(result.captured[1].settings.name, "/dog_image");
});

class MockNavigatorObserver extends Mock implements NavigatorObserver {}

值得注意的是,在例子中我們使用了 captured[1] 來驗證,是因為在測試中,當我們在準備 BreedListPage 時,實際上也進行了一次 Routing,但這次 Routing 我們並不關心,我們關心的是第二次 Routing 結果,所以在上面例子中,我們驗證 captured[1] 的結果。

找不到 Routing 錯誤

當我們完成上面測試並運行後,會發現測試還是錯誤的,並在錯誤訊息中發現以下錯誤訊息。

Could not find a generator for route RouteSettings("/dog_image", affenpinscher) in the _WidgetsAppState.
Make sure your root app widget has provided a way to generate this route.

原因是我們在測試中沒有定義 /dog_image 這個 Route,所以當運行測試,程式走到 Navigator.of(context).pushNamed 時,就發生了錯誤。為了解決這個問題,我們只要在測試中給假的 Route 即可。

testWidgets('should open dog image page when click breed tile', (tester) async {
// Arrange
...

await tester.pumpWidget(
MaterialApp(
...
routes: {"/dog_image": (_) => const SizedBox()},
));
await tester.pump();

// Act
...

// Assert
...
});

當我們加上假的 Route 之後,再次運行測試,就能通過測試得到綠燈了。

除了驗證測試路徑之外

在上面測試中,我們雖然成功驗證了 Routing 是否符合預期,但是其實還有一件事我們沒有驗證到,那就是我們少驗證了參數,我們除了把使用者導到下一個頁面之外,也會告訴下一個頁面要顯示哪一個品種的狗。在我們完成上面的測試之後,我們想再額外驗證參數其實就相對容易,我們只要加上另外一個 expect 即可。

testWidgets('should open dog image page when click breed tile', (tester) async {
// Arrange
...

// Act
...

// Assert
var result = verify(() => mockNavigatorObserver.didPush(captureAny(), any()));
expect(result.captured[1].settings.name, DogImagePage.routeName);
expect(result.captured[1].settings.arguments, "affenpinscher");
});

自定義 RouteMatcher

當我們完成測試之後,除了重構一下程式碼之外,我們也必須重構一下測試,讓我們的測試保持簡單易懂,在驗證 Routing 的部分,我們可以自定義一個 RouteMatcher 來增加測試可讀性,讓我們不必每次都在測試中把 captured 挖出來一個一個檢查,那就讓我們來重構一下 Assert 的部分吧。

testWidgets('should open dog image page when click breed tile', (tester) async {
// Arrange
...

// Act
...

// Assert
verify(() => mockNavigatorObserver.didPush(
captureAny(
that: RouteMatcher(
routeName: "/dog_image",
arguments: "affenpinscher",
),
),
any(),
));
});

class RouteMatcher extends Matcher {
final String routeName;
final dynamic arguments;

RouteMatcher({required this.routeName, this.arguments});

@override
Description describe(Description description) {
return description.add('routeName: $routeName, arguments: $arguments');
}

@override
bool matches(item, Map matchState) {
return item.settings.name == routeName &&
item.settings.arguments == arguments;
}
}

我們新增了一個 RouteMatcher 來協助比較 Route 是否符合預期,此後當我們閱讀 Routing 測試時,就能更直觀的在 Assert 中看到我們預期什麼路徑與參數,增加測試的可讀性,當然我們還可以利用抽取方法進一步的調整,讓測試真正變成容易閱讀的需求文件,像是下面程式碼那樣,這邊就不做過多贅述,我們以後會做一集專門講解(誤)。

raw-image

一定得使用 NavigatorObserver 嗎?

除了使用 NavigatorObserver 來測試 Routing,其實也可以直接針對整個 App 測試,我們也就不用做假的 RouteMockNavigatorObserver 了,聽起來好像十分省事,對吧。讓我們簡單地改寫一下測試:

  1. 準備兩個頁面所必須使用的資料:狗狗品種清單與隨機一張狗狗圖片
  2. 顯示 MainApp,而不是 BreedListPage
  3. 按下其中一個品種
  4. 驗證 Image 所顯示的圖片是否符合預期。
testWidgets('should open dog image page when click breed tile', (tester) async {
//Arrange
when(() => mockClient.get(Uri.parse("https://dog.ceo/api/breeds/list/all"))).thenAnswer((_) async {
return Response( '{"message": {"affenpinscher": [], "african": [], "airedale": []}, "status": "success"}', 200);
});

when(() => mockClient.get(Uri.parse("https://dog.ceo/api/breed/african/images/random"))).thenAnswer((_) async {
return Response( '{"message": "https://images.dog.ceo/breeds/bulldog-boston/n02096585_355.jpg", "status": "success"}', 200);
});

await tester.pumpWidget(MainApp(client: mockClient));
await tester.pump();

// Act
await tester.tap(find.text("african"));
await tester.pumpAndSettle();

// Assert
expect(findNetworkImage(tester).url, "https://images.dog.ceo/breeds/bulldog-boston/n02096585_355.jpg");
});

NetworkImage findNetworkImage(WidgetTester tester) => tester.widget<Image>(find.byType(Image)).image as NetworkImage;

在上面例子中,測試看起來也更簡單俐落,也更貼近使用者的真實狀況,沒有 MockNavigatorObserver 好像看起來更好了。事實上,在這個例子中,也確實如此,使用 MockNavigatorObserver 反而增加了不必要的麻煩。

但是在實務上,有時並非如此,當我們的 App 功能越來越多,越來越複雜時,若測試的進入點是整個 App,但是我們卻想測試某個頁面的行為,可能就得做很多準備工作,最後才能進到我們真正想測試的地方,雖然測試很貼近使用者的真實狀況,但同時也變得很難寫,變得脆弱。

結論

無論選擇使用 MockNavigatorObserver 協助測試,或是直接測試整個 App,我們應該依照當下情況調整,但是不管如何,我們都有義務為功能撰寫測試 ,這是開發人員必要工作之一,測試可以維護產品品質,也增加我們重構時的信心,更可以用來描述產品行為,讓後人可以透過測試案例來了解產品行為,是一石三鳥的好投資。

P.S. 以上程式碼都只有片段,如果有興趣看更完整的 Demo 的觀眾,可以到這邊

分享各種軟體開發技巧與心得
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
這篇文章介紹如何在多臺 MacBook 上同步開發工具與設定,以提高開發效率。文章重點在於如何同步 IntelliJ、IdeaVim 和 Alfred 配置,並解決因設定不同而影響開發效率的問題。透過簡單的步驟,開發者可以在不同設備上無縫運作,持續專注於開發工作,而不必因為熱鍵或工具失效而浪費時間。
本文探討在客戶端程式開發中,如何有效處理根據後端不同資料型態變化的畫面顯示。透過列舉 Shortgun Surgery 問題及其對代碼維護的影響,分析各種設計模式,包括轉接器模式和策略模式,來改善資料的處理方式。最終提出根據具體情況選擇合適解法的重要性,以確保開發效率與代碼可維護性。
本文介紹如何利用 Flutter 框架和 Flame 遊戲引擎製作一個簡單的點擊小遊戲。從導入 Flame 套件,到使用 GameWidget,接著為遊戲中的騎士角色添加動作,最後實現點擊計數功能,這篇文章一步步帶你體驗遊戲開發的樂趣,讓你掌握基本的遊戲設計和邏輯編程技巧。
本文分享瞭如何使用 Notion Web Clipper 來儲存文章,並結合 Habit Stacking 技術克服維持新習慣的困難。同時探討如何使用 Flutter Web 和 Notion API 開發自用的 Chrome 擴展功能,提升個人資訊管理和閱讀效率。
Flutter Widget 能加速開發,但誤用 MediaQuery 可能導致不預期的重建。範例中,頁面因鍵盤觸發高度變動而刷新,隨機數重新生成。使用固定比例設計避免重建,顯示深入理解框架對穩定性的重要性。
Flutter 習慣在最頂層的 MaterialApp 或 CupertinoApp 中統一定義整個 app 的路由管理。當我們把所有頁面的路由管理都放在最頂層時,就會讓它變得很長,不容易維護。或許應該適時思考,是否某些頁面的路由不應該被管理在最頂層。
這篇文章介紹如何在多臺 MacBook 上同步開發工具與設定,以提高開發效率。文章重點在於如何同步 IntelliJ、IdeaVim 和 Alfred 配置,並解決因設定不同而影響開發效率的問題。透過簡單的步驟,開發者可以在不同設備上無縫運作,持續專注於開發工作,而不必因為熱鍵或工具失效而浪費時間。
本文探討在客戶端程式開發中,如何有效處理根據後端不同資料型態變化的畫面顯示。透過列舉 Shortgun Surgery 問題及其對代碼維護的影響,分析各種設計模式,包括轉接器模式和策略模式,來改善資料的處理方式。最終提出根據具體情況選擇合適解法的重要性,以確保開發效率與代碼可維護性。
本文介紹如何利用 Flutter 框架和 Flame 遊戲引擎製作一個簡單的點擊小遊戲。從導入 Flame 套件,到使用 GameWidget,接著為遊戲中的騎士角色添加動作,最後實現點擊計數功能,這篇文章一步步帶你體驗遊戲開發的樂趣,讓你掌握基本的遊戲設計和邏輯編程技巧。
本文分享瞭如何使用 Notion Web Clipper 來儲存文章,並結合 Habit Stacking 技術克服維持新習慣的困難。同時探討如何使用 Flutter Web 和 Notion API 開發自用的 Chrome 擴展功能,提升個人資訊管理和閱讀效率。
Flutter Widget 能加速開發,但誤用 MediaQuery 可能導致不預期的重建。範例中,頁面因鍵盤觸發高度變動而刷新,隨機數重新生成。使用固定比例設計避免重建,顯示深入理解框架對穩定性的重要性。
Flutter 習慣在最頂層的 MaterialApp 或 CupertinoApp 中統一定義整個 app 的路由管理。當我們把所有頁面的路由管理都放在最頂層時,就會讓它變得很長,不容易維護。或許應該適時思考,是否某些頁面的路由不應該被管理在最頂層。
你可能也想看
Google News 追蹤
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
打開 jupyter notebook 寫一段 python 程式,可以完成五花八門的工作,這是玩程式最簡便的方式,其中可以獲得很多快樂,在現今這種資訊發達的時代,幾乎沒有門檻,只要願意,人人可享用。 下一步,希望程式可以隨時待命聽我吩咐,不想每次都要開電腦,啟動開發環境,只為完成一個重複性高
Thumbnail
這篇內容,將簡單介紹Asset Browser、Workspace、Inspector、Code Browser,作為入門的介面導覽。
Thumbnail
當我們架好站、WebService測試完,接著就是測試區域網路連線啦~
Thumbnail
你好,在下最近在學習開發web,學了html css js,也得出一些心得,由於網路上已有許多教學,所以我會著重在如何開發出to do List,以及解釋我寫的程式碼。相關的教學我會直接貼網址。如果我有什麼地方出錯,或者是可以寫得更好,歡迎在下方留言,討論。 首先先介紹我的開發環境: 我用了vs
Thumbnail
# 簡介 身為一位專注於 Vue.js 的前端開發者,這是我第一次嘗試構建 Flutter 網頁應用。讓我們開始吧! ## 第一次嘗試 ### 第一步:創建一個 Flutter 應用 首先,通過運行以下命令來創建一個新的 Flutter 項目: ```sh flutter
※ TypeScript範例說明: interface ITest { test1: string test2: number print: (arg: string[]) => boolean } class Test implements ITest { public te
Thumbnail
有沒有想過,即使沒有任何編程背景,你的創意也能在六個月內轉化成真實的App?我可以以自身經歷跟你說有了 No-Code Tool (無代碼工具) 和 AI 的幫助,這一切都是可能的!你一行 code 都不需要打,甚至也無須學習任何編程語言!沒有什麼比實踐一個自小認為不可能的任務還振奮人心的事了!
Thumbnail
這是為了搭建自己想要的工作流而開始的研究工作。
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
打開 jupyter notebook 寫一段 python 程式,可以完成五花八門的工作,這是玩程式最簡便的方式,其中可以獲得很多快樂,在現今這種資訊發達的時代,幾乎沒有門檻,只要願意,人人可享用。 下一步,希望程式可以隨時待命聽我吩咐,不想每次都要開電腦,啟動開發環境,只為完成一個重複性高
Thumbnail
這篇內容,將簡單介紹Asset Browser、Workspace、Inspector、Code Browser,作為入門的介面導覽。
Thumbnail
當我們架好站、WebService測試完,接著就是測試區域網路連線啦~
Thumbnail
你好,在下最近在學習開發web,學了html css js,也得出一些心得,由於網路上已有許多教學,所以我會著重在如何開發出to do List,以及解釋我寫的程式碼。相關的教學我會直接貼網址。如果我有什麼地方出錯,或者是可以寫得更好,歡迎在下方留言,討論。 首先先介紹我的開發環境: 我用了vs
Thumbnail
# 簡介 身為一位專注於 Vue.js 的前端開發者,這是我第一次嘗試構建 Flutter 網頁應用。讓我們開始吧! ## 第一次嘗試 ### 第一步:創建一個 Flutter 應用 首先,通過運行以下命令來創建一個新的 Flutter 項目: ```sh flutter
※ TypeScript範例說明: interface ITest { test1: string test2: number print: (arg: string[]) => boolean } class Test implements ITest { public te
Thumbnail
有沒有想過,即使沒有任何編程背景,你的創意也能在六個月內轉化成真實的App?我可以以自身經歷跟你說有了 No-Code Tool (無代碼工具) 和 AI 的幫助,這一切都是可能的!你一行 code 都不需要打,甚至也無須學習任何編程語言!沒有什麼比實踐一個自小認為不可能的任務還振奮人心的事了!
Thumbnail
這是為了搭建自己想要的工作流而開始的研究工作。