2017-11-03 00:09:34 +00:00
|
|
|
const fs = require('fs-extra');
|
|
|
|
const { BaseCommand } = require('./base-command.js');
|
2020-11-07 15:59:37 +00:00
|
|
|
const { splitCommandString } = require('@joplin/lib/string-utils.js');
|
|
|
|
const uuid = require('@joplin/lib/uuid').default;
|
2017-11-03 00:09:34 +00:00
|
|
|
const { app } = require('./app.js');
|
2020-11-07 15:59:37 +00:00
|
|
|
const { _ } = require('@joplin/lib/locale');
|
|
|
|
const Note = require('@joplin/lib/models/Note.js');
|
|
|
|
const Setting = require('@joplin/lib/models/Setting').default;
|
|
|
|
const BaseModel = require('@joplin/lib/BaseModel').default;
|
2017-07-10 20:03:46 +00:00
|
|
|
|
|
|
|
class Command extends BaseCommand {
|
|
|
|
usage() {
|
2017-08-04 17:51:01 +00:00
|
|
|
return 'edit <note>';
|
2017-07-10 20:03:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
description() {
|
2017-07-18 18:21:03 +00:00
|
|
|
return _('Edit note.');
|
2017-07-10 20:03:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async action(args) {
|
2017-10-14 18:03:23 +00:00
|
|
|
let tempFilePath = null;
|
2017-07-10 20:03:46 +00:00
|
|
|
|
2017-07-13 21:26:45 +00:00
|
|
|
const onFinishedEditing = async () => {
|
2017-10-14 18:03:23 +00:00
|
|
|
if (tempFilePath) fs.removeSync(tempFilePath);
|
2019-07-30 07:35:42 +00:00
|
|
|
};
|
2017-07-10 20:03:46 +00:00
|
|
|
|
|
|
|
const textEditorPath = () => {
|
|
|
|
if (Setting.value('editor')) return Setting.value('editor');
|
|
|
|
if (process.env.EDITOR) return process.env.EDITOR;
|
|
|
|
throw new Error(_('No text editor is defined. Please set it using `config editor <editor-path>`'));
|
2019-07-30 07:35:42 +00:00
|
|
|
};
|
2017-07-10 20:03:46 +00:00
|
|
|
|
2019-07-30 07:35:42 +00:00
|
|
|
try {
|
2017-10-14 18:03:23 +00:00
|
|
|
// -------------------------------------------------------------------------
|
|
|
|
// Load note or create it if it doesn't exist
|
|
|
|
// -------------------------------------------------------------------------
|
|
|
|
|
2020-03-13 23:46:14 +00:00
|
|
|
const title = args['note'];
|
2017-07-10 20:03:46 +00:00
|
|
|
|
|
|
|
if (!app().currentFolder()) throw new Error(_('No active notebook.'));
|
2017-07-10 20:47:01 +00:00
|
|
|
let note = await app().loadItem(BaseModel.TYPE_NOTE, title);
|
2017-07-10 20:03:46 +00:00
|
|
|
|
2017-12-26 10:38:53 +00:00
|
|
|
this.encryptionCheck(note);
|
|
|
|
|
2017-07-13 21:26:45 +00:00
|
|
|
if (!note) {
|
2017-10-08 17:50:43 +00:00
|
|
|
const ok = await this.prompt(_('Note does not exist: "%s". Create it?', title));
|
2017-07-27 17:25:42 +00:00
|
|
|
if (!ok) return;
|
2017-10-14 18:03:23 +00:00
|
|
|
note = await Note.save({ title: title, parent_id: app().currentFolder().id });
|
|
|
|
note = await Note.load(note.id);
|
2017-07-13 21:26:45 +00:00
|
|
|
}
|
2017-07-10 20:03:46 +00:00
|
|
|
|
2017-10-14 18:03:23 +00:00
|
|
|
// -------------------------------------------------------------------------
|
|
|
|
// Create the file to be edited and prepare the editor program arguments
|
|
|
|
// -------------------------------------------------------------------------
|
|
|
|
|
2017-07-10 20:03:46 +00:00
|
|
|
let editorPath = textEditorPath();
|
2019-11-01 22:42:05 +00:00
|
|
|
let editorArgs = splitCommandString(editorPath);
|
2017-07-10 20:03:46 +00:00
|
|
|
|
|
|
|
editorPath = editorArgs[0];
|
|
|
|
editorArgs = editorArgs.splice(1);
|
|
|
|
|
2017-10-14 18:03:23 +00:00
|
|
|
const originalContent = await Note.serializeForEdit(note);
|
2017-07-10 20:03:46 +00:00
|
|
|
|
2019-09-19 21:51:18 +00:00
|
|
|
tempFilePath = `${Setting.value('tempDir')}/${uuid.create()}.md`;
|
2017-07-10 20:03:46 +00:00
|
|
|
editorArgs.push(tempFilePath);
|
|
|
|
|
2017-10-14 18:03:23 +00:00
|
|
|
await fs.writeFile(tempFilePath, originalContent);
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------------
|
|
|
|
// Start editing the file
|
|
|
|
// -------------------------------------------------------------------------
|
|
|
|
|
|
|
|
this.logger().info('Disabling fullscreen...');
|
2017-07-10 20:03:46 +00:00
|
|
|
|
2019-11-01 22:42:05 +00:00
|
|
|
app().gui().showModalOverlay(_('Starting to edit note. Close the editor to get back to the prompt.'));
|
|
|
|
await app().gui().forceRender();
|
|
|
|
const termState = app().gui().termSaveState();
|
2019-07-30 07:35:42 +00:00
|
|
|
|
|
|
|
const spawnSync = require('child_process').spawnSync;
|
2018-01-17 18:10:07 +00:00
|
|
|
const result = spawnSync(editorPath, editorArgs, { stdio: 'inherit' });
|
|
|
|
|
|
|
|
if (result.error) this.stdout(_('Error opening note in editor: %s', result.error.message));
|
2017-07-10 20:03:46 +00:00
|
|
|
|
2019-11-01 22:42:05 +00:00
|
|
|
app().gui().termRestoreState(termState);
|
|
|
|
app().gui().hideModalOverlay();
|
|
|
|
app().gui().forceRender();
|
2017-07-10 20:03:46 +00:00
|
|
|
|
2017-10-14 18:03:23 +00:00
|
|
|
// -------------------------------------------------------------------------
|
|
|
|
// Save the note and clean up
|
|
|
|
// -------------------------------------------------------------------------
|
2017-07-10 20:03:46 +00:00
|
|
|
|
2017-10-14 18:03:23 +00:00
|
|
|
const updatedContent = await fs.readFile(tempFilePath, 'utf8');
|
|
|
|
if (updatedContent !== originalContent) {
|
2020-03-13 23:46:14 +00:00
|
|
|
const updatedNote = await Note.unserializeForEdit(updatedContent);
|
2017-10-14 18:03:23 +00:00
|
|
|
updatedNote.id = note.id;
|
|
|
|
await Note.save(updatedNote);
|
2017-10-17 21:56:22 +00:00
|
|
|
this.stdout(_('Note has been saved.'));
|
2017-10-14 18:03:23 +00:00
|
|
|
}
|
|
|
|
|
2017-10-14 21:44:50 +00:00
|
|
|
this.dispatch({
|
2017-11-08 21:22:24 +00:00
|
|
|
type: 'NOTE_SELECT',
|
|
|
|
id: note.id,
|
2017-10-14 21:44:50 +00:00
|
|
|
});
|
|
|
|
|
2017-10-14 18:03:23 +00:00
|
|
|
await onFinishedEditing();
|
2019-07-30 07:35:42 +00:00
|
|
|
} catch (error) {
|
2017-07-13 21:26:45 +00:00
|
|
|
await onFinishedEditing();
|
2017-07-10 20:03:46 +00:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-30 07:35:42 +00:00
|
|
|
module.exports = Command;
|