mirror of
https://github.com/kuhyx/todo-app.git
synced 2026-07-04 15:23:02 +02:00
Notes list & filtering: - Text-search filter plus independent date-range filters for both created and last-updated (AND-combined), a priority filter, and a new status filter. Default view hides Done/Abandoned and renders as "unfiltered" (no badge for the default state); fixed badge clipping. - NoteSort options wired into the list UI; watchCount() for the "N saved". Status & priority: - New Status enum (toDo/inProgress/Done/Abandoned) as a settable + filterable attribute on every note, with capture-screen dropdown. - Removed "None" priority: every note is Low/Medium/High, default Medium. Schema migration v2->v3 rewrites legacy priority 0 -> Medium. Export / import: - NotesMarkdown round-trippable single-file format with HTML-comment markers. - Settings "Export notes" (mobile share sheet / desktop writes ~/todo/BACKLOG.md) and "Import notes" (file picker + safe newer-wins merge by id). Structured template: - Every new note pre-fills the richer what/where/must/nice/out/done/depends/ estimate/refs scaffold. Tests: - New fast (~5s), deterministic suite via FakeNoteRepository (no DB timers) and injected http/file-selector/url-launcher fakes. 86 tests, 96.2% line coverage (note.dart & sync_service.dart at 100%, settings 98.7%). Mobile-only share branch excluded via coverage:ignore (unreachable on the Linux test host). Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
105 lines
2.8 KiB
Dart
105 lines
2.8 KiB
Dart
import 'dart:async';
|
|
|
|
import 'package:sqlite_crdt/sqlite_crdt.dart';
|
|
import 'package:todo/data/note.dart';
|
|
import 'package:todo/data/note_repository.dart';
|
|
|
|
/// In-memory stand-in for [NoteRepository] used by widget tests.
|
|
///
|
|
/// It implements the same public API but backs it with a plain list and a
|
|
/// broadcast [StreamController] — no SQLite, so no pending sqflite timers to
|
|
/// fight the widget tester's fake-async clock. Streams emit synchronously on
|
|
/// every change, making tests fast and deterministic.
|
|
///
|
|
/// It also records the last [NoteSort]/[NoteFilter] passed to [watchNotes],
|
|
/// so list-screen tests can assert the UI built the right query without
|
|
/// re-testing the repository's SQL (covered separately by unit tests).
|
|
class FakeNoteRepository implements NoteRepository {
|
|
FakeNoteRepository([List<Note>? initial]) : _notes = [...?initial];
|
|
|
|
final List<Note> _notes;
|
|
final _controller = StreamController<List<Note>>.broadcast();
|
|
|
|
NoteSort? lastSort;
|
|
NoteFilter? lastFilter;
|
|
|
|
/// Emits the current snapshot to a new subscriber first (so late-binding
|
|
/// [StreamBuilder]s get the seed), then forwards subsequent changes.
|
|
Stream<List<Note>> _snapshots() async* {
|
|
yield List.unmodifiable(_notes);
|
|
yield* _controller.stream;
|
|
}
|
|
|
|
void _emit() {
|
|
if (!_controller.isClosed) _controller.add(List.unmodifiable(_notes));
|
|
}
|
|
|
|
@override
|
|
Future<void> upsert(Note note) async {
|
|
_notes
|
|
..removeWhere((n) => n.id == note.id)
|
|
..add(note);
|
|
_emit();
|
|
}
|
|
|
|
@override
|
|
Future<void> delete(String id) async {
|
|
_notes.removeWhere((n) => n.id == id);
|
|
_emit();
|
|
}
|
|
|
|
@override
|
|
Future<ImportOutcome> importNotes(List<Note> incoming) async {
|
|
var added = 0;
|
|
var updated = 0;
|
|
var skipped = 0;
|
|
for (final note in incoming) {
|
|
final i = _notes.indexWhere((n) => n.id == note.id);
|
|
if (i < 0) {
|
|
_notes.add(note);
|
|
added++;
|
|
} else if (note.updatedAt.isAfter(_notes[i].updatedAt)) {
|
|
_notes[i] = note;
|
|
updated++;
|
|
} else {
|
|
skipped++;
|
|
}
|
|
}
|
|
_emit();
|
|
return ImportOutcome(added: added, updated: updated, skipped: skipped);
|
|
}
|
|
|
|
@override
|
|
Future<List<Note>> listNotes({
|
|
NoteSort sort = NoteSort.modifiedDesc,
|
|
NoteFilter filter = const NoteFilter(),
|
|
}) async => List.unmodifiable(_notes);
|
|
|
|
@override
|
|
Stream<List<Note>> watchNotes({
|
|
NoteSort sort = NoteSort.modifiedDesc,
|
|
NoteFilter filter = const NoteFilter(),
|
|
}) {
|
|
lastSort = sort;
|
|
lastFilter = filter;
|
|
return _snapshots();
|
|
}
|
|
|
|
@override
|
|
Stream<int> watchCount() => _snapshots().map((n) => n.length);
|
|
|
|
@override
|
|
String get nodeId => 'fake-node';
|
|
|
|
@override
|
|
Future<CrdtChangeset> getChangeset() async => {};
|
|
|
|
@override
|
|
Future<void> merge(CrdtChangeset changeset) async {}
|
|
|
|
@override
|
|
Future<void> close() async {
|
|
await _controller.close();
|
|
}
|
|
}
|