import Collection from "./collection"; import Notebook from "../models/notebook"; import getId from "../utils/id"; import { CHECK_IDS, checkIsUserPremium } from "../common"; import { qclone } from "qclone"; import setManipulator from "../utils/set"; export default class Notebooks extends Collection { async merge(remoteNotebook) { if (remoteNotebook.deleted) return await this._collection.addItem(remoteNotebook); const id = remoteNotebook.id || getId(); let localNotebook = this._collection.getItem(id); if (localNotebook && localNotebook.topics?.length) { const lastSyncedTimestamp = await this._db.lastSynced(); let isChanged = false; // merge new and old topics // We need to handle 3 cases: for (let oldTopic of localNotebook.topics) { const newTopicIndex = remoteNotebook.topics.findIndex( (t) => t.id === oldTopic.id ); const newTopic = remoteNotebook.topics[newTopicIndex]; // CASE 1: if topic exists in old notebook but not in new notebook, it's deleted. // However, if the dateEdited of topic in the old notebook is > lastSyncedTimestamp // it was newly added or edited so add it to the new notebook. if (!newTopic && oldTopic.dateEdited > lastSyncedTimestamp) { remoteNotebook.topics.push({ ...oldTopic, dateEdited: Date.now() }); isChanged = true; } // CASE 2: if topic exists in new notebook but not in old notebook, it's new. // This case will be automatically handled as the new notebook is our source of truth. // CASE 3: if topic exists in both notebooks: // if oldTopic.dateEdited > newTopic.dateEdited: we keep oldTopic // and merge the notes of both topics. else if (newTopic && oldTopic.dateEdited > newTopic.dateEdited) { remoteNotebook.topics[newTopicIndex] = { ...oldTopic, notes: setManipulator.union(oldTopic.notes, newTopic.notes), dateEdited: Date.now(), }; isChanged = true; } } remoteNotebook.remote = !isChanged; } return await this._collection.addItem(remoteNotebook); } async add(notebookArg) { if (!notebookArg) throw new Error("Notebook cannot be undefined or null."); if (notebookArg.remote) throw new Error( "Please use db.notebooks.merge to merge remote notebooks" ); //TODO reliably and efficiently check for duplicates. const id = notebookArg.id || getId(); let oldNotebook = this._collection.getItem(id); if ( !oldNotebook && this.all.length >= 3 && !(await checkIsUserPremium(CHECK_IDS.notebookAdd)) ) return; let notebook = { ...oldNotebook, ...notebookArg, }; if (!notebook.title) throw new Error("Notebook must contain a title."); notebook = { id, type: "notebook", title: notebook.title, description: notebook.description, pinned: !!notebook.pinned, topics: notebook.topics || [], dateCreated: notebook.dateCreated, dateModified: notebook.dateModified, dateEdited: Date.now(), }; await this._collection.addItem(notebook); if (!oldNotebook) { await this.notebook(notebook).topics.add(...notebook.topics); } return id; } get raw() { return this._collection.getRaw(); } get all() { return this._collection.getItems(); } get pinned() { return this.all.filter((item) => item.pinned === true); } get deleted() { return this.raw.filter((item) => item.dateDeleted > 0); } /** * * @param {string} id The id of the notebook * @returns {Notebook} The notebook of the given id */ notebook(id) { let notebook = id.type ? id : this._collection.getItem(id); if (!notebook || notebook.deleted) return; return new Notebook(notebook, this._db); } async delete(...ids) { for (let id of ids) { let notebook = this.notebook(id); if (!notebook) continue; const notebookData = qclone(notebook.data); await notebook.topics.delete(...notebook.data.topics); await this._collection.removeItem(id); await this._db.settings.unpin(id); await this._db.trash.add(notebookData); } } async repairReferences() { for (let notebook of this.all) { const _notebook = this.notebook(notebook); for (let topic of notebook.topics) { const _topic = _notebook.topics.topic(topic.id); await _topic.add(...topic.notes); } } } }