mirror of
https://github.com/kuhyx/todo-app.git
synced 2026-07-04 11:43:10 +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>
104 lines
3.5 KiB
Dart
104 lines
3.5 KiB
Dart
import 'package:uuid/uuid.dart';
|
|
|
|
import '../data/note.dart';
|
|
|
|
/// Serialises notes to (and parses them back from) a single Markdown file.
|
|
///
|
|
/// The whole document is valid Markdown: each note is preceded by an HTML
|
|
/// comment carrying its metadata (id, priority, status, timestamps), which
|
|
/// renders invisibly, followed by the note body verbatim. Keeping the `id`
|
|
/// lets [NoteRepository.importNotes] re-import a file as a *merge* (by id)
|
|
/// rather than creating duplicates — the basis for "never lose ideas"
|
|
/// recovery and round-tripping a backup.
|
|
class NotesMarkdown {
|
|
const NotesMarkdown._();
|
|
|
|
static const _uuid = Uuid();
|
|
|
|
/// First line of an exported file; identifies the format/version.
|
|
static const header = '<!-- todo-backlog v1 -->';
|
|
|
|
/// Matches a per-note metadata marker at the start of a line. The body is
|
|
/// everything between one marker and the next (or end of file).
|
|
static final _markerPattern = RegExp(
|
|
r'^<!--\s*@note\s+(.*?)\s*-->[ \t]*$',
|
|
multiLine: true,
|
|
);
|
|
|
|
/// Matches `key="value"` attribute pairs inside a marker.
|
|
static final _attrPattern = RegExp(r'(\w+)="([^"]*)"');
|
|
|
|
/// Renders [notes] to a single Markdown document.
|
|
static String export(List<Note> notes) {
|
|
final buffer = StringBuffer()
|
|
..writeln(header)
|
|
..writeln();
|
|
for (final note in notes) {
|
|
buffer
|
|
..writeln(
|
|
'<!-- @note id="${note.id}" priority="${note.priority.name}" '
|
|
'status="${note.status.name}" '
|
|
'created="${note.createdAt.toIso8601String()}" '
|
|
'updated="${note.updatedAt.toIso8601String()}" -->',
|
|
)
|
|
..writeln(note.text)
|
|
..writeln();
|
|
}
|
|
return buffer.toString();
|
|
}
|
|
|
|
/// Parses a previously exported (or hand-written) document back into notes.
|
|
///
|
|
/// Tolerant by design: a missing/blank `id` gets a fresh UUID (treated as
|
|
/// a new note), and unknown/missing priority/status/timestamps fall back
|
|
/// to sensible defaults so a partially hand-edited file never throws.
|
|
static List<Note> parse(String content) {
|
|
final markers = _markerPattern.allMatches(content).toList();
|
|
final notes = <Note>[];
|
|
for (var i = 0; i < markers.length; i++) {
|
|
final marker = markers[i];
|
|
final attrs = _parseAttrs(marker.group(1) ?? '');
|
|
final bodyStart = marker.end;
|
|
final bodyEnd = i + 1 < markers.length
|
|
? markers[i + 1].start
|
|
: content.length;
|
|
final body = content.substring(bodyStart, bodyEnd).trim();
|
|
|
|
final id = attrs['id'];
|
|
notes.add(
|
|
Note(
|
|
id: (id != null && id.isNotEmpty) ? id : _uuid.v4(),
|
|
text: body,
|
|
priority: _enumByName(
|
|
Priority.values,
|
|
attrs['priority'],
|
|
Priority.defaultValue,
|
|
),
|
|
status: _enumByName(Status.values, attrs['status'], Status.todo),
|
|
createdAt:
|
|
DateTime.tryParse(attrs['created'] ?? '') ?? DateTime.now(),
|
|
updatedAt:
|
|
DateTime.tryParse(attrs['updated'] ?? '') ?? DateTime.now(),
|
|
),
|
|
);
|
|
}
|
|
return notes;
|
|
}
|
|
|
|
/// Extracts `key="value"` pairs from a marker's attribute string.
|
|
static Map<String, String> _parseAttrs(String raw) {
|
|
return {
|
|
for (final m in _attrPattern.allMatches(raw)) m.group(1)!: m.group(2)!,
|
|
};
|
|
}
|
|
|
|
/// Resolves an enum value by its [Enum.name], falling back to [fallback].
|
|
static T _enumByName<T extends Enum>(
|
|
List<T> values,
|
|
String? name,
|
|
T fallback,
|
|
) {
|
|
return values.firstWhere((v) => v.name == name, orElse: () => fallback);
|
|
}
|
|
}
|