티스토리 뷰

Flutter 앱 개발에서 위젯 테스트는 안정적이고 버그 없는 UI를 구축하는 데 필수적인 과정입니다. 이 글에서는 Flutter에서 위젯 테스트를 수행하는 방법과 그 중요성, 그리고 실질적인 구현 예제를 통해 개발자가 알아야 할 핵심 포인트를 자세히 설명합니다.

참고. An introduction to widget testing

위젯 테스트란 무엇인가?

위젯 테스트는 개별 위젯의 동작을 검증하여 UI의 기능성과 안정성을 확인하는 테스트 방법입니다. 단위 테스트보다 상위 수준이며, 통합 테스트보다 하위 수준에 위치합니다. 위젯 테스트를 통해 다음과 같은 이점을 얻을 수 있습니다:

  • UI 동작 검증: 사용자 인터페이스가 예상대로 동작하는지 확인합니다.
  • 빠른 피드백: 테스트 실행 속도가 빨라 코드 변경 시 즉각적인 피드백을 받을 수 있습니다.
  • 버그 조기 발견: 개발 초기 단계에서 UI 관련 버그를 발견하여 수정 비용을 절감합니다.

Flutter에서 위젯 테스트 설정하기

1. 테스트 환경 설정

flutter_test 패키지는 Flutter 프로젝트 생성 시 기본적으로 포함됩니다. 별도의 설치 없이 바로 사용할 수 있습니다.

2. 테스트 파일 생성

프로젝트의 test 디렉토리에 테스트 파일을 생성합니다. 예를 들어, widget_test.dart 파일을 만들 수 있습니다.

my_project/
├── lib/
│   └── main.dart
└── test/
    └── widget_test.dart

기본 위젯 테스트 작성하기

아래는 간단한 카운터 앱의 위젯 테스트 예제입니다.

1. 필요한 패키지 임포트

import 'package:flutter_test/flutter_test.dart';
import 'package:my_app/main.dart';

2. 테스트 함수 작성

void main() {
  testWidgets('Counter increments smoke test', (WidgetTester tester) async {
    // 앱 위젯 빌드 및 렌더링
    await tester.pumpWidget(MyApp());

    // 초기 값이 '0'인지 확인
    expect(find.text('0'), findsOneWidget);
    expect(find.text('1'), findsNothing);

    // '+' 아이콘을 탭
    await tester.tap(find.byIcon(Icons.add));
    await tester.pump();

    // 값이 '1'로 증가했는지 확인
    expect(find.text('0'), findsNothing);
    expect(find.text('1'), findsOneWidget);
  });
}

주요 함수와 개념 이해하기

1. testWidgets

  • 위젯 테스트를 정의하는 함수로, 비동기식으로 동작합니다.
  • 첫 번째 인자로 테스트 설명 문자열, 두 번째 인자로 테스트 콜백 함수를 받습니다.

2. WidgetTester

  • 위젯을 빌드하고 상호 작용을 모방하는 데 사용되는 유틸리티 클래스입니다.
  • 주요 메서드:
    • pumpWidget(): 위젯을 빌드하고 렌더링합니다.
    • tap(): 특정 위젯을 탭하는 동작을 시뮬레이션합니다.
    • pump(): 위젯 트리를 다시 빌드하여 상태 변화를 반영합니다.

3. find

  • 위젯 트리에서 특정 위젯을 찾는 데 사용됩니다.
  • 주요 메서드:
    • find.text(): 텍스트 위젯을 찾습니다.
    • find.byIcon(): 아이콘 위젯을 찾습니다.
    • find.byType(): 특정 타입의 위젯을 찾습니다.

4. expect

  • 실제 값이 예상 값과 일치하는지 검증합니다.

고급 위젯 테스트 기법

1. 상태 변경 테스트

위젯의 상태 변화에 따른 동작을 테스트합니다.

testWidgets('Checkbox toggles test', (WidgetTester tester) async {
  await tester.pumpWidget(MyApp());

  // 체크박스 초기 상태 확인
  expect(find.byType(Checkbox), findsOneWidget);
  expect(tester.widget<Checkbox>(find.byType(Checkbox)).value, false);

  // 체크박스 탭하여 상태 변경
  await tester.tap(find.byType(Checkbox));
  await tester.pump();

  // 상태 변경 확인
  expect(tester.widget<Checkbox>(find.byType(Checkbox)).value, true);
});

2. 비동기 동작 테스트

네트워크 요청이나 타이머와 같은 비동기 동작을 테스트합니다.

testWidgets('Async data load test', (WidgetTester tester) async {
  await tester.pumpWidget(MyApp());

  // 로딩 인디케이터 확인
  expect(find.byType(CircularProgressIndicator), findsOneWidget);

  // 비동기 작업 완료까지 대기
  await tester.pumpAndSettle();

  // 데이터 로드 후 위젯 확인
  expect(find.text('Loaded Data'), findsOneWidget);
});

위젯 테스트 작성 시 고려사항

  1. 테스트의 독립성 유지: 각 테스트는 독립적으로 실행되어야 하며, 다른 테스트에 영향을 주지 않아야 합니다.
  2. UI 변화에 대한 유연성 확보: UI 레이아웃 변경에 너무 민감하지 않도록 테스트를 작성합니다.
  3. 실제 사용자 시나리오 반영: 테스트는 실제 사용자의 행동을 모방해야 합니다.
  4. 적절한 pump 사용: 상태 변화나 애니메이션 후에는 pump() 또는 pumpAndSettle()을 사용하여 위젯 트리를 업데이트합니다.

위젯 테스트의 이점

  • 버그 예방: 코드 변경 시 발생할 수 있는 UI 버그를 미리 발견합니다.
  • 리팩토링 지원: UI를 변경하더라도 기능이 유지되는지 확인할 수 있습니다.
  • 개발 효율성 향상: 수동 테스트에 소요되는 시간을 절약합니다.

결론

Flutter에서 위젯 테스트는 안정적이고 신뢰할 수 있는 앱을 개발하는 데 필수적인 과정입니다. 위에서 소개한 방법과 예제를 따라 위젯 테스트를 작성하면 앱의 품질을 크게 향상시킬 수 있습니다. 지속적인 테스트를 통해 사용자에게 최상의 경험을 제공하는 앱을 만들어 보세요.

 

import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';

void main() {
  // Define a test. The TestWidgets function also provides a WidgetTester
  // to work with. The WidgetTester allows building and interacting
  // with widgets in the test environment.
  testWidgets('MyWidget has a title and message', (tester) async {
    // Create the widget by telling the tester to build it.
    await tester.pumpWidget(const MyWidget(title: 'T', message: 'M'));

    // Create the Finders.
    final titleFinder = find.text('T');
    final messageFinder = find.text('M');

    // Use the `findsOneWidget` matcher provided by flutter_test to
    // verify that the Text widgets appear exactly once in the widget tree.
    expect(titleFinder, findsOneWidget);
    expect(messageFinder, findsOneWidget);
  });
}

class MyWidget extends StatelessWidget {
  const MyWidget({
    super.key,
    required this.title,
    required this.message,
  });

  final String title;
  final String message;

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Flutter Demo',
      home: Scaffold(
        appBar: AppBar(
          title: Text(title),
        ),
        body: Center(
          child: Text(message),
        ),
      ),
    );
  }
}