testsAndMisc-archive/horatio/horatio_app/test/router_test.dart
Krzysztof kuhy Rudnicki 8b5d3e75f2 feat(home): add 'See a demo' button with seeded Hamlet annotation editor
- Add DemoAnnotationEditorScreen: wraps the real AnnotationEditorScreen with
  an in-memory Drift DB seeded with 6 lines of Hamlet's soliloquy, 4 TextMarks,
  4 LineNotes, 4 LineRecordings (3 on line 0 with grades), and 1
  AnnotationSnapshot — all ephemeral, zero writes to disk
- Add /demo route to go_router
- Show 'See a demo' OutlinedButton.icon on the empty library screen only
- Tests: 6 widget tests for DemoAnnotationEditorScreen (including runAsync
  pattern for Drift real-time timer handling), 2 new home screen tests, and
  a router test for the /demo route

All 366 tests pass, 100% branch coverage, flutter analyze --fatal-infos clean
2026-03-29 21:46:28 +02:00

341 lines
10 KiB
Dart

import 'dart:async';
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:horatio_app/bloc/script_import/script_import_cubit.dart';
import 'package:horatio_app/bloc/srs_review/srs_review_cubit.dart';
import 'package:horatio_app/database/daos/annotation_dao.dart';
import 'package:horatio_app/database/daos/recording_dao.dart';
import 'package:horatio_app/router.dart';
import 'package:horatio_app/services/audio_playback_service.dart';
import 'package:horatio_app/services/recording_service.dart';
import 'package:horatio_app/services/script_repository.dart';
import 'package:horatio_core/horatio_core.dart';
import 'package:mocktail/mocktail.dart';
class _MockAnnotationDao extends Mock implements AnnotationDao {}
class _MockRecordingDao extends Mock implements RecordingDao {}
class _MockRecordingService extends Mock implements RecordingService {}
class _MockAudioPlaybackService extends Mock implements AudioPlaybackService {}
Widget _wrapRouter() {
final repository = ScriptRepository();
final mockDao = _MockAnnotationDao();
final mockRecordingDao = _MockRecordingDao();
final mockRecordingService = _MockRecordingService();
final mockPlaybackService = _MockAudioPlaybackService();
when(
() => mockDao.watchMarksForScript(any()),
).thenAnswer((_) => Stream.value([]));
when(
() => mockDao.watchNotesForScript(any()),
).thenAnswer((_) => Stream.value([]));
when(
() => mockDao.watchSnapshotsForScript(any()),
).thenAnswer((_) => Stream.value([]));
when(
() => mockRecordingDao.watchRecordingsForScript(any()),
).thenAnswer((_) => Stream.value([]));
when(mockRecordingService.hasPermission).thenAnswer((_) async => true);
when(
() => mockRecordingService.startRecording(any()),
).thenAnswer((_) async {});
when(mockRecordingService.stopRecording).thenAnswer((_) async => null);
when(() => mockPlaybackService.play(any())).thenAnswer((_) async {});
when(mockPlaybackService.stop).thenAnswer((_) async {});
when(
() => mockPlaybackService.status,
).thenAnswer((_) => const Stream.empty());
when(
() => mockPlaybackService.position,
).thenAnswer((_) => const Stream.empty());
return MultiRepositoryProvider(
providers: [
RepositoryProvider<ScriptRepository>(create: (_) => repository),
RepositoryProvider<AnnotationDao>.value(value: mockDao),
RepositoryProvider<RecordingDao>.value(value: mockRecordingDao),
RepositoryProvider<RecordingService>.value(value: mockRecordingService),
RepositoryProvider<AudioPlaybackService>.value(
value: mockPlaybackService,
),
RepositoryProvider<String>.value(value: '/tmp/test_recordings'),
],
child: MultiBlocProvider(
providers: [
BlocProvider<ScriptImportCubit>(
create: (_) => ScriptImportCubit(repository: repository),
),
BlocProvider<SrsReviewCubit>(create: (_) => SrsReviewCubit()),
],
child: MaterialApp.router(routerConfig: appRouter),
),
);
}
void main() {
group('Router with valid extras', () {
testWidgets('import route shows ImportScreen', (tester) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
appRouter.go(RoutePaths.import_);
await tester.pumpAndSettle();
expect(find.text('Import Script'), findsOneWidget);
});
testWidgets('role-selection route with Script extra', (tester) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
const role = Role(name: 'Hero');
const script = Script(
id: 'router-valid-id',
title: 'Valid',
roles: [role],
scenes: [
Scene(
lines: [
ScriptLine(
text: 'Line.',
role: role,
sceneIndex: 0,
lineIndex: 0,
),
],
),
],
);
unawaited(appRouter.push(RoutePaths.roleSelection, extra: script));
await tester.pumpAndSettle();
expect(find.text('Choose Your Role'), findsOneWidget);
});
testWidgets('schedule route with map extra', (tester) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
const role = Role(name: 'Hero');
const script = Script(
id: 'router-play-id',
title: 'Play',
roles: [role],
scenes: [
Scene(
lines: [
ScriptLine(text: 'Hi.', role: role, sceneIndex: 0, lineIndex: 0),
],
),
],
);
unawaited(
appRouter.push(
RoutePaths.schedule,
extra: {'script': script, 'role': role},
),
);
await tester.pumpAndSettle();
expect(find.text('Memorization Schedule'), findsOneWidget);
});
testWidgets('rehearsal route with map extra', (tester) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
const role = Role(name: 'Hero');
const script = Script(
id: 'router-rehearse-id',
title: 'Rehearse',
roles: [role],
scenes: [
Scene(
lines: [
ScriptLine(text: 'A.', role: role, sceneIndex: 0, lineIndex: 0),
],
),
],
);
unawaited(
appRouter.push(
RoutePaths.rehearsal,
extra: {'script': script, 'role': role},
),
);
await tester.pumpAndSettle();
expect(find.text('Rehearsing: Hero'), findsOneWidget);
});
testWidgets('srs-review route with cards extra', (tester) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
final cards = [SrsCard(id: 'c1', cueText: 'Cue', answerText: 'Ans')];
unawaited(appRouter.push(RoutePaths.srsReview, extra: cards));
await tester.pumpAndSettle();
// SrsReviewScreen is visible.
expect(find.text('No review session active.'), findsOneWidget);
});
testWidgets('error route shows 404', (tester) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
appRouter.go('/nonexistent-route');
await tester.pumpAndSettle();
expect(find.text('Not Found'), findsOneWidget);
});
testWidgets('schedule route with wrong extra type falls back', (
tester,
) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
// Push schedule with a non-Map extra → the builder returns SizedBox.
unawaited(appRouter.push(RoutePaths.schedule, extra: 'wrong'));
await tester.pumpAndSettle();
// Should not crash — shows SizedBox.shrink or redirects.
expect(tester.takeException(), isNull);
});
testWidgets('annotations route with Script extra shows editor', (
tester,
) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
// Reset to home to clear any stale navigation stack.
appRouter.go(RoutePaths.home);
await tester.pumpAndSettle();
const role = Role(name: 'Hero');
const script = Script(
id: 'router-annotate-id',
title: 'Annotate Play',
roles: [role],
scenes: [
Scene(
lines: [
ScriptLine(
text: 'Line.',
role: role,
sceneIndex: 0,
lineIndex: 0,
),
],
),
],
);
unawaited(appRouter.push(RoutePaths.annotations, extra: script));
await tester.pumpAndSettle();
expect(find.text('Annotate: Annotate Play'), findsOneWidget);
});
testWidgets('annotations route with null extra redirects home', (
tester,
) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
appRouter.go(RoutePaths.annotations);
await tester.pumpAndSettle();
// Redirected to home.
expect(find.text('Horatio'), findsOneWidget);
});
testWidgets('annotation-history route with Script extra shows history', (
tester,
) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
const role = Role(name: 'Hero');
const script = Script(
id: 'router-history-id',
title: 'History Play',
roles: [role],
scenes: [
Scene(
lines: [
ScriptLine(
text: 'Line.',
role: role,
sceneIndex: 0,
lineIndex: 0,
),
],
),
],
);
unawaited(appRouter.push(RoutePaths.annotationHistory, extra: script));
await tester.pumpAndSettle();
expect(find.text('History: History Play'), findsOneWidget);
});
testWidgets('annotation-history route with null extra redirects home', (
tester,
) async {
await tester.pumpWidget(_wrapRouter());
await tester.pumpAndSettle();
appRouter.go(RoutePaths.annotationHistory);
await tester.pumpAndSettle();
// Redirected to home.
expect(find.text('Horatio'), findsOneWidget);
});
testWidgets('demo route shows DemoAnnotationEditorScreen', (tester) async {
// DemoAnnotationEditorScreen creates a real in-memory Drift DB.
// All Drift async timers (seeding, stream delivery, disposal cleanup)
// must fire in real time via runAsync to avoid pending fake-async timers.
await tester.runAsync(() async {
await tester.pumpWidget(_wrapRouter());
await tester.pump();
appRouter.go(RoutePaths.demo);
// Process the navigation frame.
await tester.pump();
await tester.pump();
// Wait for seeding to complete in real time.
await Future<void>.delayed(const Duration(seconds: 2));
await tester.pump();
// Allow Drift initial stream deliveries.
await Future<void>.delayed(const Duration(milliseconds: 500));
await tester.pump();
expect(find.textContaining('Hamlet', findRichText: true), findsWidgets);
// Replace entire widget tree to force DemoAnnotationEditorScreen
// disposal inside runAsync so Drift's markAsClosed timers fire in
// real time rather than as pending fake-async timers.
await tester.pumpWidget(const SizedBox.shrink());
await Future<void>.delayed(const Duration(milliseconds: 500));
});
});
});
}