Files
habo/test/widgets/habit_list_widget_test.dart
dazhuang aa69f2a91e feat: initial commit - Habo habit tracking app
- Complete MVP with Repository Pattern, SQLite storage
- Provider + ChangeNotifier state management
- Navigation 2.0 with deep link support
- Habit CRUD with twoDayRule, notifications, categories
- Backup/Restore via JSON
- Statistics with streak tracking
- Material You theme support
- Biometric lock support
- Desktop widget support
- 27 languages i18n structure
- Comprehensive test suite (87/89 passing)
2026-04-13 15:02:30 +00:00

155 lines
4.6 KiB
Dart

import 'dart:collection';
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:habo/habits/habit.dart';
import 'package:habo/habits/habits_manager.dart';
import 'package:habo/model/habit_data.dart';
import 'package:habo/repositories/habit_repository.dart';
import 'package:habo/repositories/event_repository.dart';
import 'package:habo/repositories/category_repository.dart';
import 'package:habo/services/backup_service.dart';
import 'package:habo/services/notification_service.dart';
import 'package:habo/services/ui_feedback_service.dart';
import 'package:mocktail/mocktail.dart';
import 'package:provider/provider.dart';
class MockHabitRepository extends Mock implements HabitRepository {}
class MockEventRepository extends Mock implements EventRepository {}
class MockCategoryRepository extends Mock implements CategoryRepository {}
class MockBackupService extends Mock implements BackupService {}
class MockNotificationService extends Mock implements NotificationService {}
class MockUIFeedbackService extends Mock implements UIFeedbackService {}
void main() {
late HabitsManager habitsManager;
late MockHabitRepository mockHabitRepository;
late MockEventRepository mockEventRepository;
late MockCategoryRepository mockCategoryRepository;
late MockBackupService mockBackupService;
late MockNotificationService mockNotificationService;
late MockUIFeedbackService mockUIFeedbackService;
setUp(() {
mockHabitRepository = MockHabitRepository();
mockEventRepository = MockEventRepository();
mockCategoryRepository = MockCategoryRepository();
mockBackupService = MockBackupService();
mockNotificationService = MockNotificationService();
mockUIFeedbackService = MockUIFeedbackService();
habitsManager = HabitsManager(
habitRepository: mockHabitRepository,
eventRepository: mockEventRepository,
categoryRepository: mockCategoryRepository,
backupService: mockBackupService,
notificationService: mockNotificationService,
uiFeedbackService: mockUIFeedbackService,
);
});
group('HabitListWidget Tests', () {
testWidgets('should display empty state when no habits',
(WidgetTester tester) async {
// Build our app and trigger a frame
await tester.pumpWidget(
MaterialApp(
home: ChangeNotifierProvider.value(
value: habitsManager,
child: const Scaffold(
body: HabitListWidget(),
),
),
),
);
// Verify empty state is displayed
expect(find.text('No habits yet'), findsOneWidget);
expect(find.byType(HabitCard), findsNothing);
});
testWidgets('should display habits when available',
(WidgetTester tester) async {
// Add test habits
final testHabit1 = Habit(
habitData: HabitData(
position: 0,
title: 'Test Habit 1',
twoDayRule: false,
cue: 'Cue 1',
routine: 'Routine 1',
reward: 'Reward 1',
showReward: true,
advanced: false,
events: SplayTreeMap<DateTime, List>(),
notification: false,
notTime: const TimeOfDay(hour: 9, minute: 0),
sanction: '',
showSanction: false,
accountant: '',
),
);
testHabit1.setId = 1;
habitsManager.allHabits.add(testHabit1);
await tester.pumpWidget(
MaterialApp(
home: ChangeNotifierProvider.value(
value: habitsManager,
child: const Scaffold(
body: HabitListWidget(),
),
),
),
);
// Verify habits are displayed
expect(find.text('Test Habit 1'), findsOneWidget);
expect(find.byType(HabitCard), findsOneWidget);
});
});
}
// Mock widget classes for testing
class HabitListWidget extends StatelessWidget {
const HabitListWidget({super.key});
@override
Widget build(BuildContext context) {
final habitsManager = context.watch<HabitsManager>();
if (habitsManager.allHabits.isEmpty) {
return const Center(child: Text('No habits yet'));
}
return ListView.builder(
itemCount: habitsManager.allHabits.length,
itemBuilder: (context, index) {
final habit = habitsManager.allHabits[index];
return HabitCard(habit: habit);
},
);
}
}
class HabitCard extends StatelessWidget {
final Habit habit;
const HabitCard({super.key, required this.habit});
@override
Widget build(BuildContext context) {
return Card(
child: ListTile(
title: Text(habit.habitData.title),
subtitle: Text(habit.habitData.cue),
),
);
}
}