Files
GitJournal/lib/screens/journal_editor.dart
2020-01-18 11:47:20 +01:00

265 lines
7.6 KiB
Dart

import 'package:flutter/material.dart';
import 'package:gitjournal/core/note.dart';
import 'package:gitjournal/core/notes_folder.dart';
import 'package:gitjournal/state_container.dart';
import 'package:gitjournal/widgets/journal_editor_header.dart';
import 'package:gitjournal/core/note_data.dart';
import 'package:gitjournal/core/note_data_serializers.dart';
enum NoteEditorDropDownChoices { Discard, SwitchEditor }
class JournalEditor extends StatefulWidget {
final Note note;
final NotesFolder notesFolder;
JournalEditor.fromNote(this.note) : notesFolder = null;
JournalEditor.newNote(this.notesFolder) : note = null;
@override
JournalEditorState createState() {
if (note == null) {
return JournalEditorState.newNote(notesFolder);
} else {
return JournalEditorState.fromNote(note);
}
}
}
class JournalEditorState extends State<JournalEditor> {
Note note;
final bool newNote;
TextEditingController _textController = TextEditingController();
TextEditingController _titleTextController = TextEditingController();
bool rawEditor = false;
final serializer = MarkdownYAMLSerializer();
final bool journalMode = false;
final bool showTitle = true;
JournalEditorState.newNote(NotesFolder folder) : newNote = true {
note = Note.newNote(folder);
}
JournalEditorState.fromNote(this.note) : newNote = false {
_textController = TextEditingController(text: note.body);
_titleTextController = TextEditingController(text: note.title);
}
@override
void dispose() {
_textController.dispose();
_titleTextController.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
Widget editor = Column(
children: <Widget>[
if (journalMode) JournalEditorHeader(note),
if (showTitle && !rawEditor) NoteTitleEditor(_titleTextController),
NoteMarkdownEditor(_textController, false),
],
);
if (rawEditor) {
editor = NoteMarkdownEditor(_textController, true);
}
var title = newNote ? "New Note" : "Edit Note";
var newJournalScreen = Scaffold(
appBar: AppBar(
title: Text(title),
leading: IconButton(
key: const ValueKey("NewEntry"),
icon: Icon(Icons.check),
onPressed: () {
_saveNote(context);
Navigator.pop(context);
},
),
actions: <Widget>[
PopupMenuButton<NoteEditorDropDownChoices>(
onSelected: (NoteEditorDropDownChoices choice) {
switch (choice) {
case NoteEditorDropDownChoices.Discard:
if (_noteModified()) {
showDialog(context: context, builder: _buildAlertDialog);
} else {
Navigator.pop(context);
}
break;
case NoteEditorDropDownChoices.SwitchEditor:
note.title = _titleTextController.text.trim();
setState(() {
if (rawEditor) {
rawEditor = false;
note.data = serializer.decode(_textController.text);
_textController.text = note.body;
_titleTextController.text = note.title;
} else {
rawEditor = true;
var noteData =
NoteData(_textController.text, note.data.props);
_textController.text = serializer.encode(noteData);
}
});
break;
}
},
itemBuilder: (BuildContext context) =>
<PopupMenuEntry<NoteEditorDropDownChoices>>[
const PopupMenuItem<NoteEditorDropDownChoices>(
value: NoteEditorDropDownChoices.Discard,
child: Text('Discard'),
),
PopupMenuItem<NoteEditorDropDownChoices>(
value: NoteEditorDropDownChoices.SwitchEditor,
child: rawEditor
? const Text('Rich Editor')
: const Text('Raw Editor'),
),
],
),
],
),
body: Padding(
padding: const EdgeInsets.all(16.0),
child: SingleChildScrollView(
child: editor,
),
),
floatingActionButton: FloatingActionButton(
child: Icon(Icons.check),
onPressed: () {
_saveNote(context);
Navigator.pop(context);
},
),
);
return WillPopScope(
onWillPop: () async {
_saveNote(context);
return true;
},
child: newJournalScreen,
);
}
Widget _buildAlertDialog(BuildContext context) {
var title = newNote
? "Do you want to discard this?"
: "Do you want to ignore the changes?";
var editText = newNote ? "Keep Writing" : "Keep Editing";
var discardText = newNote ? "Discard" : "Discard Changes";
return AlertDialog(
title: Text(title),
actions: <Widget>[
FlatButton(
onPressed: () => Navigator.of(context).pop(false),
child: Text(editText),
),
FlatButton(
onPressed: () {
Navigator.pop(context); // Alert box
Navigator.pop(context); // Note Editor
},
child: Text(discardText),
),
],
);
}
bool _noteModified() {
var noteContent = _textController.text.trim();
var titleContent = _titleTextController.text.trim();
if (noteContent.isEmpty && titleContent.isEmpty) {
return false;
}
if (note != null) {
if (rawEditor) {
return serializer.encode(note.data) != noteContent;
} else {
return noteContent != note.body || titleContent != note.title;
}
}
return false;
}
void _saveNote(BuildContext context) {
if (!_noteModified()) return;
final stateContainer = StateContainer.of(context);
if (rawEditor == false) {
note.body = _textController.text.trim();
note.title = _titleTextController.text.trim();
} else {
note.data = serializer.decode(_textController.text);
}
if (!note.isEmpty()) {
newNote ? stateContainer.addNote(note) : stateContainer.updateNote(note);
}
}
}
class NoteMarkdownEditor extends StatelessWidget {
final TextEditingController textController;
final bool useMonospace;
NoteMarkdownEditor(this.textController, this.useMonospace);
@override
Widget build(BuildContext context) {
var style = Theme.of(context).textTheme.subhead;
if (useMonospace) {
style = style.copyWith(fontFamily: "Roboto Mono");
}
return TextField(
autofocus: true,
autocorrect: false,
keyboardType: TextInputType.multiline,
maxLines: null,
style: style,
decoration: InputDecoration(
hintText: 'Write here',
border: InputBorder.none,
isDense: true,
),
controller: textController,
textCapitalization: TextCapitalization.sentences,
scrollPadding: const EdgeInsets.all(0.0),
);
}
}
class NoteTitleEditor extends StatelessWidget {
final TextEditingController textController;
NoteTitleEditor(this.textController);
@override
Widget build(BuildContext context) {
var style = Theme.of(context).textTheme.title;
return TextField(
autofocus: false,
keyboardType: TextInputType.text,
maxLines: 1,
style: style,
decoration: InputDecoration(
hintText: 'Title',
border: InputBorder.none,
isDense: true,
),
controller: textController,
textCapitalization: TextCapitalization.sentences,
);
}
}