const { reg } = require('../../registry.js'); const Folder = require('../../models/Folder').default; const BaseModel = require('../../BaseModel').default; const Note = require('../../models/Note').default; const Resource = require('../../models/Resource').default; const ResourceFetcher = require('../../services/ResourceFetcher').default; const DecryptionWorker = require('../../services/DecryptionWorker').default; const Setting = require('../../models/Setting').default; const Mutex = require('async-mutex').Mutex; const shared = {}; // If saveNoteButton_press is called multiple times in short intervals, it might result in // the same new note being created twice, so we need to a mutex to access this function. const saveNoteMutex_ = new Mutex(); shared.noteExists = async function(noteId) { const existingNote = await Note.load(noteId); return !!existingNote; }; // Note has been deleted while user was modifying it. In that case, we // just save a new note so that user can keep editing. shared.handleNoteDeletedWhileEditing_ = async (note) => { if (await shared.noteExists(note.id)) return null; reg.logger().info('Note has been deleted while it was being edited - recreating it.'); let newNote = Object.assign({}, note); delete newNote.id; newNote = await Note.save(newNote); return Note.load(newNote.id); }; shared.saveNoteButton_press = async function(comp, folderId = null, options = null) { options = Object.assign({}, { autoTitle: true, }, options); const releaseMutex = await saveNoteMutex_.acquire(); let note = Object.assign({}, comp.state.note); const recreatedNote = await shared.handleNoteDeletedWhileEditing_(note); if (recreatedNote) note = recreatedNote; if (folderId) { note.parent_id = folderId; } else if (!note.parent_id) { const activeFolderId = Setting.value('activeFolderId'); let folder = await Folder.load(activeFolderId); if (!folder) folder = await Folder.defaultFolder(); if (!folder) return releaseMutex(); note.parent_id = folder.id; } const isProvisionalNote = comp.props.provisionalNoteIds.includes(note.id); const saveOptions = { userSideValidation: true, fields: BaseModel.diffObjectsFields(comp.state.lastSavedNote, note), }; const hasAutoTitle = comp.state.newAndNoTitleChangeNoteId || (isProvisionalNote && !note.title); if (hasAutoTitle && options.autoTitle) { note.title = Note.defaultTitle(note.body); if (saveOptions.fields && saveOptions.fields.indexOf('title') < 0) saveOptions.fields.push('title'); } const savedNote = 'fields' in saveOptions && !saveOptions.fields.length ? Object.assign({}, note) : await Note.save(note, saveOptions); const stateNote = comp.state.note; // Note was reloaded while being saved. if (!recreatedNote && (!stateNote || stateNote.id !== savedNote.id)) return releaseMutex(); // Re-assign any property that might have changed during saving (updated_time, etc.) note = Object.assign(note, savedNote); if (stateNote.id === note.id) { // But we preserve the current title and body because // the user might have changed them between the time // saveNoteButton_press was called and the note was // saved (it's done asynchronously). // // If the title was auto-assigned above, we don't restore // it from the state because it will be empty there. if (!hasAutoTitle) note.title = stateNote.title; note.body = stateNote.body; } const newState = { lastSavedNote: Object.assign({}, note), note: note, }; if (isProvisionalNote && hasAutoTitle) newState.newAndNoTitleChangeNoteId = note.id; if (!options.autoTitle) newState.newAndNoTitleChangeNoteId = null; comp.setState(newState); if (isProvisionalNote) { Note.updateGeolocation(note.id).then(geoNote => { const stateNote = comp.state.note; if (!stateNote || !geoNote) return; if (stateNote.id !== geoNote.id) return; // Another note has been loaded while geoloc was being retrieved // Geo-location for this note has been saved to the database however the properties // are is not in the state so set them now. const geoInfo = { longitude: geoNote.longitude, latitude: geoNote.latitude, altitude: geoNote.altitude, }; const modNote = Object.assign({}, stateNote, geoInfo); const modLastSavedNote = Object.assign({}, comp.state.lastSavedNote, geoInfo); comp.setState({ note: modNote, lastSavedNote: modLastSavedNote }); }); } releaseMutex(); }; shared.saveOneProperty = async function(comp, name, value) { let note = Object.assign({}, comp.state.note); const recreatedNote = await shared.handleNoteDeletedWhileEditing_(note); if (recreatedNote) note = recreatedNote; let toSave = { id: note.id }; toSave[name] = value; toSave = await Note.save(toSave); note[name] = toSave[name]; comp.setState({ lastSavedNote: Object.assign({}, note), note: note, }); }; shared.noteComponent_change = function(comp, propName, propValue) { const newState = {}; const note = Object.assign({}, comp.state.note); note[propName] = propValue; newState.note = note; comp.setState(newState); }; let resourceCache_ = {}; shared.clearResourceCache = function() { resourceCache_ = {}; }; shared.attachedResources = async function(noteBody) { if (!noteBody) return {}; const resourceIds = await Note.linkedItemIdsByType(BaseModel.TYPE_RESOURCE, noteBody); const output = {}; for (let i = 0; i < resourceIds.length; i++) { const id = resourceIds[i]; if (resourceCache_[id]) { output[id] = resourceCache_[id]; } else { const resource = await Resource.load(id); const localState = await Resource.localState(resource); const o = { item: resource, localState: localState, }; // eslint-disable-next-line require-atomic-updates resourceCache_[id] = o; output[id] = o; } } return output; }; shared.isModified = function(comp) { if (!comp.state.note || !comp.state.lastSavedNote) return false; const diff = BaseModel.diffObjects(comp.state.lastSavedNote, comp.state.note); delete diff.type_; return !!Object.getOwnPropertyNames(diff).length; }; shared.initState = async function(comp) { const isProvisionalNote = comp.props.provisionalNoteIds.includes(comp.props.noteId); const note = await Note.load(comp.props.noteId); let mode = 'view'; if (isProvisionalNote && !comp.props.sharedData) { mode = 'edit'; comp.scheduleFocusUpdate(); } const folder = Folder.byId(comp.props.folders, note.parent_id); comp.setState({ lastSavedNote: Object.assign({}, note), note: note, mode: mode, folder: folder, isLoading: false, fromShare: comp.props.sharedData ? true : false, noteResources: await shared.attachedResources(note ? note.body : ''), }); if (comp.props.sharedData) { if (comp.props.sharedData.title) { this.noteComponent_change(comp, 'title', comp.props.sharedData.title); } if (comp.props.sharedData.text) { this.noteComponent_change(comp, 'body', comp.props.sharedData.text); } if (comp.props.sharedData.resources) { for (let i = 0; i < comp.props.sharedData.resources.length; i++) { const resource = comp.props.sharedData.resources[i]; reg.logger().info(`about to attach resource ${JSON.stringify(resource)}`); await comp.attachFile({ uri: resource.uri, type: resource.mimeType, name: resource.name, }, null); } } } // eslint-disable-next-line require-atomic-updates comp.lastLoadedNoteId_ = note.id; }; shared.toggleIsTodo_onPress = function(comp) { const newNote = Note.toggleIsTodo(comp.state.note); const newState = { note: newNote }; comp.setState(newState); }; shared.toggleCheckbox = function(ipcMessage, noteBody) { const newBody = noteBody.split('\n'); const p = ipcMessage.split(':'); const lineIndex = Number(p[p.length - 1]); if (lineIndex >= newBody.length) { reg.logger().warn('Checkbox line out of bounds: ', ipcMessage); return newBody.join('\n'); } let line = newBody[lineIndex]; const noCrossIndex = line.trim().indexOf('- [ ] '); let crossIndex = line.trim().indexOf('- [x] '); if (crossIndex < 0) crossIndex = line.trim().indexOf('- [X] '); if (noCrossIndex < 0 && crossIndex < 0) { reg.logger().warn('Could not find matching checkbox for message: ', ipcMessage); return newBody.join('\n'); } let isCrossLine = false; if (noCrossIndex >= 0 && crossIndex >= 0) { isCrossLine = crossIndex < noCrossIndex; } else { isCrossLine = crossIndex >= 0; } if (!isCrossLine) { line = line.replace(/- \[ \] /, '- [x] '); } else { line = line.replace(/- \[x\] /i, '- [ ] '); } newBody[lineIndex] = line; return newBody.join('\n'); }; shared.installResourceHandling = function(refreshResourceHandler) { ResourceFetcher.instance().on('downloadComplete', refreshResourceHandler); ResourceFetcher.instance().on('downloadStarted', refreshResourceHandler); DecryptionWorker.instance().on('resourceDecrypted', refreshResourceHandler); }; shared.uninstallResourceHandling = function(refreshResourceHandler) { ResourceFetcher.instance().off('downloadComplete', refreshResourceHandler); ResourceFetcher.instance().off('downloadStarted', refreshResourceHandler); DecryptionWorker.instance().off('resourceDecrypted', refreshResourceHandler); }; module.exports = shared;