Files
GitJournal/lib/core/note.dart
Vishesh Handa 895d2f4f1b Give each Note a title
We're moving away from being a Journal Editor to being a Note Editor. I
don't want to discard the Journaling parts, as Journaling is really a
subset of Note Taking - so in the future it would be nice if this could
be somehow toggled.

Maybe each Folder can have different display settings. Probably in the
"pro" version.
2019-12-27 11:05:32 +01:00

217 lines
4.6 KiB
Dart

import 'dart:io';
import 'package:flutter/material.dart';
import 'package:path/path.dart' as p;
import 'package:gitjournal/utils/datetime.dart';
import 'note_fileName.dart';
import 'notes_folder.dart';
import 'serializers.dart';
enum NoteLoadState {
None,
Loading,
Loaded,
NotExists,
}
class Note with ChangeNotifier implements Comparable<Note> {
NotesFolder parent;
String _filePath;
String _title = "";
DateTime _created;
DateTime _modified;
NoteData _data = NoteData();
DateTime _fileLastModified;
var _loadState = NoteLoadState.None;
var _serializer = MarkdownYAMLSerializer();
Note(this.parent, this._filePath) {
_created = DateTime(0, 0, 0, 0, 0, 0, 0, 0);
}
Note.newNote(this.parent) {
_created = DateTime.now();
_data.props['created'] = toIso8601WithTimezone(created);
_filePath = p.join(parent.folderPath, getFileName(this));
}
String get filePath {
return _filePath;
}
DateTime get created {
return _created;
}
set created(DateTime dt) {
_created = dt;
if (hasValidDate()) {
_data.props['created'] = toIso8601WithTimezone(created);
} else {
_data.props.remove('created');
}
notifyListeners();
}
DateTime get modified {
return _modified;
}
set modified(DateTime dt) {
_modified = dt;
if (hasValidDate()) {
_data.props['modified'] = toIso8601WithTimezone(_modified);
} else {
_data.props.remove('modified');
}
notifyListeners();
}
void updateModified() {
modified = DateTime.now();
}
String get body {
return data.body;
}
set body(String newBody) {
data.body = newBody;
notifyListeners();
}
String get title {
return _title;
}
set title(String title) {
_title = title;
if (_title.isEmpty) {
_data.props.remove('title');
} else {
_data.props['title'] = title;
}
notifyListeners();
}
NoteData get data {
return _data;
}
set data(NoteData data) {
_data = data;
if (data.props.containsKey("created")) {
_created = parseDateTime(data.props['created'].toString());
}
if (data.props.containsKey("modified")) {
_modified = parseDateTime(data.props['modified'].toString());
}
if (data.props.containsKey("title")) {
_title = data.props['title'].toString();
}
_created ??= DateTime(0, 0, 0, 0, 0, 0, 0, 0);
notifyListeners();
}
bool hasValidDate() {
// Arbitrary number, when we set the year = 0, it becomes 1, somehow
return created.year > 10;
}
bool isEmpty() {
return body.isEmpty;
}
Future<NoteLoadState> load() async {
if (_loadState == NoteLoadState.Loading) {
return _loadState;
}
final file = File(filePath);
if (_loadState == NoteLoadState.Loaded) {
var fileLastModified = file.lastModifiedSync();
if (fileLastModified == _fileLastModified) {
return _loadState;
}
}
if (!file.existsSync()) {
_loadState = NoteLoadState.NotExists;
notifyListeners();
return _loadState;
}
final string = await file.readAsString();
data = _serializer.decode(string);
_fileLastModified = file.lastModifiedSync();
_loadState = NoteLoadState.Loaded;
notifyListeners();
return _loadState;
}
// FIXME: What about error handling?
Future<void> save() async {
assert(filePath != null);
assert(data != null);
assert(data.body != null);
assert(data.props != null);
var file = File(filePath);
var contents = _serializer.encode(data);
await file.writeAsString(contents);
}
// FIXME: What about error handling?
Future<void> remove() async {
var file = File(filePath);
await file.delete();
}
// FIXME: Can't this part be auto-generated?
@override
int get hashCode => filePath.hashCode ^ created.hashCode ^ data.hashCode;
@override
bool operator ==(Object other) =>
identical(this, other) ||
other is Note &&
runtimeType == other.runtimeType &&
filePath == other.filePath &&
data == other.data;
@override
String toString() {
return 'Note{filePath: $filePath, created: $created, modified: $modified, data: $data}';
}
@override
int compareTo(Note other) {
if (other == null) {
return -1;
}
var dt = modified ?? created;
var otherDt = other.modified ?? other.created;
if (dt == null && otherDt == null) {
return filePath.compareTo(other.filePath);
}
if (dt.year == 0 || otherDt.year == 0) {
return filePath.compareTo(other.filePath);
}
return dt.compareTo(otherDt);
}
}