- 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>
107 lines
2.9 KiB
Dart
107 lines
2.9 KiB
Dart
import 'package:drift/drift.dart';
|
|
|
|
import '../../core/models/draft.dart';
|
|
import '../../core/repositories/draft_repository.dart';
|
|
import '../db/database.dart';
|
|
|
|
class DraftRepositoryImpl implements DraftRepository {
|
|
DraftRepositoryImpl(this._db);
|
|
|
|
final AppDatabase _db;
|
|
|
|
@override
|
|
Future<SavedDraft> saveDraft({
|
|
int? id,
|
|
String? accountId,
|
|
String? replyToEmailId,
|
|
required String toText,
|
|
required String ccText,
|
|
required String subjectText,
|
|
required String bodyText,
|
|
}) async {
|
|
final now = DateTime.now();
|
|
if (id != null) {
|
|
await (_db.update(_db.drafts)..where((t) => t.id.equals(id))).write(
|
|
DraftsCompanion(
|
|
accountId: Value(accountId),
|
|
replyToEmailId: Value(replyToEmailId),
|
|
toText: Value(toText),
|
|
ccText: Value(ccText),
|
|
subjectText: Value(subjectText),
|
|
bodyText: Value(bodyText),
|
|
updatedAt: Value(now),
|
|
),
|
|
);
|
|
return SavedDraft(
|
|
id: id,
|
|
accountId: accountId,
|
|
replyToEmailId: replyToEmailId,
|
|
toText: toText,
|
|
ccText: ccText,
|
|
subjectText: subjectText,
|
|
bodyText: bodyText,
|
|
updatedAt: now,
|
|
);
|
|
}
|
|
|
|
final newId = await _db.into(_db.drafts).insert(
|
|
DraftsCompanion.insert(
|
|
accountId: Value(accountId),
|
|
replyToEmailId: Value(replyToEmailId),
|
|
toText: Value(toText),
|
|
ccText: Value(ccText),
|
|
subjectText: Value(subjectText),
|
|
bodyText: Value(bodyText),
|
|
updatedAt: now,
|
|
),
|
|
);
|
|
return SavedDraft(
|
|
id: newId,
|
|
accountId: accountId,
|
|
replyToEmailId: replyToEmailId,
|
|
toText: toText,
|
|
ccText: ccText,
|
|
subjectText: subjectText,
|
|
bodyText: bodyText,
|
|
updatedAt: now,
|
|
);
|
|
}
|
|
|
|
@override
|
|
Future<SavedDraft?> findDraft({String? replyToEmailId}) async {
|
|
final query = _db.select(_db.drafts);
|
|
if (replyToEmailId == null) {
|
|
query.where((t) => t.replyToEmailId.isNull());
|
|
} else {
|
|
query.where((t) => t.replyToEmailId.equals(replyToEmailId));
|
|
}
|
|
query.orderBy([(t) => OrderingTerm.desc(t.id)]);
|
|
query.limit(1);
|
|
final row = await query.getSingleOrNull();
|
|
return row == null ? null : _toModel(row);
|
|
}
|
|
|
|
@override
|
|
Future<SavedDraft?> getDraft(int id) async {
|
|
final row = await (_db.select(_db.drafts)..where((t) => t.id.equals(id)))
|
|
.getSingleOrNull();
|
|
return row == null ? null : _toModel(row);
|
|
}
|
|
|
|
@override
|
|
Future<void> deleteDraft(int id) async {
|
|
await (_db.delete(_db.drafts)..where((t) => t.id.equals(id))).go();
|
|
}
|
|
|
|
SavedDraft _toModel(Draft row) => SavedDraft(
|
|
id: row.id,
|
|
accountId: row.accountId,
|
|
replyToEmailId: row.replyToEmailId,
|
|
toText: row.toText,
|
|
ccText: row.ccText,
|
|
subjectText: row.subjectText,
|
|
bodyText: row.bodyText,
|
|
updatedAt: row.updatedAt,
|
|
);
|
|
}
|