Files
notesnook/packages/core/collections/notebooks.js
2021-02-25 21:01:19 +05:00

137 lines
4.1 KiB
JavaScript

import Collection from "./collection";
import Notebook from "../models/notebook";
import sort from "fast-sort";
import getId from "../utils/id";
import { CHECK_IDS, sendCheckUserStatusEvent } from "../common";
import { qclone } from "qclone";
export default class Notebooks extends Collection {
async merge(remoteNotebook) {
const id = remoteNotebook.id || getId();
let localNotebook = this._collection.getItem(id);
if (localNotebook && localNotebook.topics?.length) {
const lastSyncedTimestamp = await this._db.lastSynced();
// 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() });
}
// 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.
else if (newTopic && oldTopic.dateEdited > newTopic.dateEdited) {
remoteNotebook.topics[newTopicIndex] = {
...oldTopic,
dateEdited: Date.now(),
};
}
}
remoteNotebook.dateEdited = Date.now(); // we update the dateEdited so it can be synced back
}
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 sendCheckUserStatusEvent(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,
dateCreated: notebook.dateCreated,
dateEdited: notebook.dateEdited,
pinned: !!notebook.pinned,
topics: notebook.topics || [],
};
if (
!oldNotebook &&
notebook.topics.findIndex((topic) => topic.title === "General") <= -1
) {
notebook.topics.splice(0, 0, "General");
}
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 sort(this._collection.getItems()).desc((t) => t.pinned);
}
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);
}
}
}