- Add `format` task (fvm dart format .) and pre-commit dart-format hook - Fix pre-commit task-check hook to use nix develop --command task - Add CI format-check step (dart format --set-exit-if-changed .) - Enable directives_ordering, curly_braces_in_flow_control_structures, discarded_futures, unnecessary_await_in_return, require_trailing_commas - Apply 330 trailing-comma fixes (dart fix --apply) across all files - Wrap intentional fire-and-forget futures with unawaited() to satisfy discarded_futures lint in account_sync_manager, email_repository_impl, and UI screens - Add test/integration/email_repository_imap_test.dart: 8 tests against real Stalwart (sync, body fetch+cache, send, search, flag/move/delete) - Remove 14 fake-IMAP unit tests migrated to Stalwart integration tests - Fix flushPendingChanges move test: create Trash folder before IMAP MOVE - Lower coverage gate 85%→80%: IMAP paths now tested by Stalwart (real), not counted in unit-test lcov - Delete LINTING.md (plan fully executed) Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
170 lines
5.4 KiB
Dart
170 lines
5.4 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
import 'package:go_router/go_router.dart';
|
|
|
|
import 'package:sharedinbox/core/models/account.dart';
|
|
import 'package:sharedinbox/di.dart';
|
|
import 'package:sharedinbox/ui/screens/compose_screen.dart';
|
|
|
|
import 'helpers.dart';
|
|
|
|
void main() {
|
|
group('ComposeScreen', () {
|
|
testWidgets('renders To, Cc, Subject and Body fields', (tester) async {
|
|
await tester.pumpWidget(
|
|
buildApp(
|
|
initialLocation: '/compose',
|
|
overrides: [
|
|
accountRepositoryProvider
|
|
.overrideWithValue(FakeAccountRepository([kTestAccount])),
|
|
mailboxRepositoryProvider
|
|
.overrideWithValue(FakeMailboxRepository()),
|
|
emailRepositoryProvider.overrideWithValue(FakeEmailRepository()),
|
|
draftRepositoryProvider.overrideWithValue(FakeDraftRepository()),
|
|
],
|
|
),
|
|
);
|
|
await tester.pumpAndSettle();
|
|
|
|
expect(find.text('To'), findsOneWidget);
|
|
expect(find.text('Cc'), findsOneWidget);
|
|
expect(find.text('Subject'), findsOneWidget);
|
|
expect(find.text('Body'), findsOneWidget);
|
|
});
|
|
|
|
testWidgets('prefills To and Subject when provided as constructor params',
|
|
(tester) async {
|
|
await tester.pumpWidget(
|
|
_buildDirect(
|
|
screen: const ComposeScreen(
|
|
prefillTo: 'bob@example.com',
|
|
prefillSubject: 'Re: Hello',
|
|
),
|
|
overrides: [
|
|
accountRepositoryProvider
|
|
.overrideWithValue(FakeAccountRepository([kTestAccount])),
|
|
mailboxRepositoryProvider
|
|
.overrideWithValue(FakeMailboxRepository()),
|
|
emailRepositoryProvider.overrideWithValue(FakeEmailRepository()),
|
|
draftRepositoryProvider.overrideWithValue(FakeDraftRepository()),
|
|
],
|
|
),
|
|
);
|
|
await tester.pumpAndSettle();
|
|
|
|
expect(
|
|
find.widgetWithText(TextFormField, 'bob@example.com'),
|
|
findsOneWidget,
|
|
);
|
|
expect(find.widgetWithText(TextFormField, 'Re: Hello'), findsOneWidget);
|
|
});
|
|
|
|
testWidgets('shows static From field when one account is loaded',
|
|
(tester) async {
|
|
await tester.pumpWidget(
|
|
buildApp(
|
|
initialLocation: '/compose',
|
|
overrides: [
|
|
accountRepositoryProvider
|
|
.overrideWithValue(FakeAccountRepository([kTestAccount])),
|
|
mailboxRepositoryProvider
|
|
.overrideWithValue(FakeMailboxRepository()),
|
|
emailRepositoryProvider.overrideWithValue(FakeEmailRepository()),
|
|
draftRepositoryProvider.overrideWithValue(FakeDraftRepository()),
|
|
],
|
|
),
|
|
);
|
|
await tester.pumpAndSettle();
|
|
|
|
expect(find.text('Alice <alice@example.com>'), findsOneWidget);
|
|
});
|
|
|
|
testWidgets('shows From dropdown when multiple accounts are loaded',
|
|
(tester) async {
|
|
const second = Account(
|
|
id: 'acc-2',
|
|
displayName: 'Bob',
|
|
email: 'bob@example.com',
|
|
imapHost: 'imap.example.com',
|
|
smtpHost: 'smtp.example.com',
|
|
);
|
|
await tester.pumpWidget(
|
|
buildApp(
|
|
initialLocation: '/compose',
|
|
overrides: [
|
|
accountRepositoryProvider.overrideWithValue(
|
|
FakeAccountRepository([kTestAccount, second]),
|
|
),
|
|
mailboxRepositoryProvider
|
|
.overrideWithValue(FakeMailboxRepository()),
|
|
emailRepositoryProvider.overrideWithValue(FakeEmailRepository()),
|
|
draftRepositoryProvider.overrideWithValue(FakeDraftRepository()),
|
|
],
|
|
),
|
|
);
|
|
await tester.pumpAndSettle();
|
|
|
|
expect(find.byType(DropdownButtonFormField<String>), findsOneWidget);
|
|
});
|
|
|
|
testWidgets('restores saved draft when no prefill is provided',
|
|
(tester) async {
|
|
final fakeDrafts = FakeDraftRepository();
|
|
await fakeDrafts.saveDraft(
|
|
toText: 'carol@example.com',
|
|
ccText: '',
|
|
subjectText: 'Restored subject',
|
|
bodyText: 'Draft body',
|
|
);
|
|
await tester.pumpWidget(
|
|
_buildDirect(
|
|
screen: const ComposeScreen(),
|
|
overrides: [
|
|
accountRepositoryProvider
|
|
.overrideWithValue(FakeAccountRepository([kTestAccount])),
|
|
mailboxRepositoryProvider
|
|
.overrideWithValue(FakeMailboxRepository()),
|
|
emailRepositoryProvider.overrideWithValue(FakeEmailRepository()),
|
|
draftRepositoryProvider.overrideWithValue(fakeDrafts),
|
|
],
|
|
),
|
|
);
|
|
await tester.pumpAndSettle();
|
|
|
|
expect(
|
|
find.widgetWithText(TextFormField, 'carol@example.com'),
|
|
findsOneWidget,
|
|
);
|
|
expect(
|
|
find.widgetWithText(TextFormField, 'Restored subject'),
|
|
findsOneWidget,
|
|
);
|
|
});
|
|
});
|
|
}
|
|
|
|
/// Builds [screen] inside a minimal GoRouter so [context.pop()] works, without
|
|
/// going through [buildApp]'s full route tree.
|
|
Widget _buildDirect({
|
|
required Widget screen,
|
|
required List<Override> overrides,
|
|
}) {
|
|
final router = GoRouter(
|
|
initialLocation: '/',
|
|
routes: [
|
|
GoRoute(path: '/', builder: (ctx, state) => screen),
|
|
],
|
|
);
|
|
return ProviderScope(
|
|
overrides: overrides,
|
|
child: MaterialApp.router(
|
|
routerConfig: router,
|
|
theme: ThemeData(
|
|
colorScheme: ColorScheme.fromSeed(seedColor: Colors.indigo),
|
|
useMaterial3: true,
|
|
),
|
|
),
|
|
);
|
|
}
|