2017-08-30 10:25:58 +08:00
|
|
|
const tags = {
|
2017-06-12 04:04:07 +08:00
|
|
|
1: "<b>",
|
|
|
|
2: "</b>",
|
|
|
|
3: "<i>",
|
|
|
|
4: "</i>",
|
|
|
|
5: "<u>",
|
|
|
|
6: "</u>",
|
|
|
|
9: "<s>",
|
|
|
|
10: "</s>"
|
|
|
|
};
|
|
|
|
|
2017-06-12 09:04:04 +08:00
|
|
|
let noteChangeDisabled = false;
|
|
|
|
|
2017-08-17 08:41:41 +08:00
|
|
|
let isNoteChanged = false;
|
|
|
|
|
2017-06-12 04:04:07 +08:00
|
|
|
function noteChanged() {
|
2017-06-12 09:04:04 +08:00
|
|
|
if (noteChangeDisabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-08-17 08:41:41 +08:00
|
|
|
isNoteChanged = true;
|
|
|
|
}
|
|
|
|
|
2017-09-05 09:28:07 +08:00
|
|
|
function saveNoteIfChanged(callback) {
|
|
|
|
if (!isNoteChanged) {
|
|
|
|
if (callback) {
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-09-10 00:06:15 +08:00
|
|
|
const note = globalCurrentNote;
|
2017-09-05 09:28:07 +08:00
|
|
|
|
|
|
|
updateNoteFromInputs(note);
|
|
|
|
|
2017-09-06 11:48:41 +08:00
|
|
|
encryptNoteIfNecessary(note);
|
|
|
|
|
|
|
|
saveNoteToServer(note, callback);
|
2017-09-05 09:28:07 +08:00
|
|
|
}
|
|
|
|
|
2017-08-17 08:41:41 +08:00
|
|
|
setInterval(saveNoteIfChanged, 5000);
|
|
|
|
|
2017-10-10 06:53:11 +08:00
|
|
|
$(document).ready(() => {
|
2017-10-11 08:19:16 +08:00
|
|
|
$("#note-title").on('input', () => {
|
2017-06-12 04:04:07 +08:00
|
|
|
noteChanged();
|
|
|
|
});
|
|
|
|
|
2017-10-11 08:19:16 +08:00
|
|
|
$('#note-detail').summernote({
|
2017-06-12 04:04:07 +08:00
|
|
|
airMode: true,
|
2017-06-12 09:04:04 +08:00
|
|
|
height: 300,
|
2017-06-12 04:04:07 +08:00
|
|
|
callbacks: {
|
|
|
|
onChange: noteChanged
|
|
|
|
}
|
|
|
|
});
|
2017-08-26 07:30:26 +08:00
|
|
|
|
|
|
|
// so that tab jumps from note title (which has tabindex 1)
|
|
|
|
$(".note-editable").attr("tabindex", 2);
|
2017-06-12 04:04:07 +08:00
|
|
|
});
|
2017-09-10 00:06:15 +08:00
|
|
|
|
2017-10-11 07:47:46 +08:00
|
|
|
function parseHtml(contents, note) {
|
2017-09-27 12:04:50 +08:00
|
|
|
note.links = [];
|
|
|
|
note.images = [];
|
|
|
|
|
|
|
|
note.detail.note_text = contents;
|
|
|
|
|
|
|
|
if (!note.detail.encryption) {
|
|
|
|
const linkRegexp = /<a[^>]+?href="[^"]*kapp#([A-Za-z0-9]{22})"[^>]*?>[^<]+?<\/a>/g;
|
|
|
|
let match;
|
|
|
|
|
|
|
|
while (match = linkRegexp.exec(contents)) {
|
|
|
|
console.log("adding link for " + match[1]);
|
|
|
|
|
|
|
|
note.links.push({
|
|
|
|
note_id: note.detail.note_id,
|
|
|
|
target_note_id: match[1]
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-10 00:06:15 +08:00
|
|
|
function updateNoteFromInputs(note) {
|
2017-10-11 08:19:16 +08:00
|
|
|
let contents = $('#note-detail').summernote('code');
|
2017-09-10 00:06:15 +08:00
|
|
|
|
2017-10-11 07:47:46 +08:00
|
|
|
parseHtml(contents, note);
|
2017-09-10 00:06:15 +08:00
|
|
|
|
2017-10-11 08:19:16 +08:00
|
|
|
let title = $('#note-title').val();
|
2017-09-10 00:06:15 +08:00
|
|
|
|
|
|
|
getNodeByKey(note.detail.note_id).setTitle(title);
|
|
|
|
|
|
|
|
note.detail.note_title = title;
|
|
|
|
}
|
|
|
|
|
|
|
|
function saveNoteToServer(note, callback) {
|
|
|
|
$.ajax({
|
2017-09-30 22:05:12 +08:00
|
|
|
url: baseApiUrl + 'notes/' + note.detail.note_id,
|
2017-09-10 00:06:15 +08:00
|
|
|
type: 'PUT',
|
|
|
|
data: JSON.stringify(note),
|
|
|
|
contentType: "application/json",
|
2017-10-10 06:53:11 +08:00
|
|
|
success: () => {
|
2017-09-10 00:06:15 +08:00
|
|
|
isNoteChanged = false;
|
|
|
|
|
|
|
|
message("Saved!");
|
|
|
|
|
|
|
|
if (callback) {
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
},
|
2017-10-10 06:53:11 +08:00
|
|
|
error: () => {
|
2017-09-10 00:06:15 +08:00
|
|
|
error("Error saving the note!");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
let globalCurrentNote;
|
2017-06-12 04:04:07 +08:00
|
|
|
|
2017-06-12 09:04:04 +08:00
|
|
|
function createNewTopLevelNote() {
|
2017-09-07 09:34:54 +08:00
|
|
|
let rootNode = globalTree.fancytree("getRootNode");
|
2017-06-12 09:04:04 +08:00
|
|
|
|
|
|
|
createNote(rootNode, "root", "into");
|
|
|
|
}
|
|
|
|
|
2017-06-14 10:36:56 +08:00
|
|
|
let newNoteCreated = false;
|
|
|
|
|
2017-09-09 10:43:02 +08:00
|
|
|
function createNote(node, parentKey, target, encryption) {
|
|
|
|
// if encryption isn't available (user didn't enter password yet), then note is created as unencrypted
|
|
|
|
// but this is quite weird since user doesn't see where the note is being created so it shouldn't occur often
|
|
|
|
if (!encryption || !isEncryptionAvailable()) {
|
|
|
|
encryption = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const newNoteName = "new note";
|
|
|
|
const newNoteNameEncryptedIfNecessary = encryption > 0 ? encryptString(newNoteName) : newNoteName;
|
2017-06-12 04:04:07 +08:00
|
|
|
|
|
|
|
$.ajax({
|
2017-09-30 22:05:12 +08:00
|
|
|
url: baseApiUrl + 'notes/' + parentKey + '/children' ,
|
2017-06-12 04:04:07 +08:00
|
|
|
type: 'POST',
|
|
|
|
data: JSON.stringify({
|
2017-09-09 10:43:02 +08:00
|
|
|
note_title: newNoteNameEncryptedIfNecessary,
|
2017-06-12 04:04:07 +08:00
|
|
|
target: target,
|
2017-09-09 10:43:02 +08:00
|
|
|
target_note_id: node.key,
|
|
|
|
encryption: encryption
|
2017-06-12 04:04:07 +08:00
|
|
|
}),
|
|
|
|
contentType: "application/json",
|
2017-10-10 06:53:11 +08:00
|
|
|
success: result => {
|
2017-09-09 11:14:42 +08:00
|
|
|
const newNode = {
|
2017-09-09 10:43:02 +08:00
|
|
|
title: newNoteName,
|
|
|
|
key: result.note_id,
|
|
|
|
note_id: result.note_id,
|
2017-09-09 11:14:42 +08:00
|
|
|
encryption: encryption,
|
|
|
|
extraClasses: encryption ? "encrypted" : ""
|
2017-06-12 04:04:07 +08:00
|
|
|
};
|
|
|
|
|
2017-09-04 06:50:56 +08:00
|
|
|
globalAllNoteIds.push(result.note_id);
|
2017-08-27 22:30:32 +08:00
|
|
|
|
2017-06-14 10:36:56 +08:00
|
|
|
newNoteCreated = true;
|
|
|
|
|
2017-08-22 09:31:23 +08:00
|
|
|
if (target === 'after') {
|
2017-06-12 04:04:07 +08:00
|
|
|
node.appendSibling(newNode).setActive(true);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
node.addChildren(newNode).setActive(true);
|
|
|
|
|
|
|
|
node.folder = true;
|
|
|
|
node.renderTitle();
|
|
|
|
}
|
|
|
|
|
|
|
|
message("Created!");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-09-18 00:46:14 +08:00
|
|
|
function setTreeBasedOnEncryption(note) {
|
|
|
|
const node = getNodeByKey(note.detail.note_id);
|
|
|
|
node.toggleClass("encrypted", note.detail.encryption > 0);
|
|
|
|
}
|
|
|
|
|
2017-09-05 09:37:55 +08:00
|
|
|
function setNoteBackgroundIfEncrypted(note) {
|
|
|
|
if (note.detail.encryption > 0) {
|
|
|
|
$(".note-editable").addClass("encrypted");
|
2017-10-11 08:19:16 +08:00
|
|
|
$("#encrypt-button").hide();
|
|
|
|
$("#decrypt-button").show();
|
2017-09-05 09:37:55 +08:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
$(".note-editable").removeClass("encrypted");
|
2017-10-11 08:19:16 +08:00
|
|
|
$("#encrypt-button").show();
|
|
|
|
$("#decrypt-button").hide();
|
2017-09-05 09:37:55 +08:00
|
|
|
}
|
2017-09-09 11:41:29 +08:00
|
|
|
|
2017-09-18 00:46:14 +08:00
|
|
|
setTreeBasedOnEncryption(note);
|
2017-09-05 09:37:55 +08:00
|
|
|
}
|
|
|
|
|
2017-10-12 08:37:27 +08:00
|
|
|
function loadNoteToEditor(noteId) {
|
2017-10-10 06:53:11 +08:00
|
|
|
$.get(baseApiUrl + 'notes/' + noteId).then(note => {
|
2017-09-10 00:06:15 +08:00
|
|
|
globalCurrentNote = note;
|
2017-06-12 04:04:07 +08:00
|
|
|
|
2017-06-14 10:36:56 +08:00
|
|
|
if (newNoteCreated) {
|
|
|
|
newNoteCreated = false;
|
|
|
|
|
2017-10-11 08:19:16 +08:00
|
|
|
$("#note-title").focus().select();
|
2017-06-14 10:36:56 +08:00
|
|
|
}
|
|
|
|
|
2017-09-06 11:51:50 +08:00
|
|
|
handleEncryption(note.detail.encryption > 0, false, () => {
|
2017-10-11 08:19:16 +08:00
|
|
|
$("#note-detail-wrapper").show();
|
2017-09-07 10:03:53 +08:00
|
|
|
|
|
|
|
// this may fal if the dialog has not been previously opened
|
2017-09-06 11:48:41 +08:00
|
|
|
try {
|
2017-10-11 08:19:16 +08:00
|
|
|
$("#encryption-password-dialog").dialog('close');
|
2017-09-06 11:48:41 +08:00
|
|
|
}
|
|
|
|
catch(e) {}
|
|
|
|
|
2017-10-11 08:19:16 +08:00
|
|
|
$("#encryption-password").val('');
|
2017-09-06 11:48:41 +08:00
|
|
|
|
2017-09-07 11:13:39 +08:00
|
|
|
decryptNoteIfNecessary(note);
|
|
|
|
|
2017-10-11 08:19:16 +08:00
|
|
|
$("#note-title").val(note.detail.note_title);
|
2017-06-12 09:04:04 +08:00
|
|
|
|
2017-09-05 09:28:07 +08:00
|
|
|
noteChangeDisabled = true;
|
2017-06-12 09:04:04 +08:00
|
|
|
|
2017-09-05 09:28:07 +08:00
|
|
|
// Clear contents and remove all stored history. This is to prevent undo from going across notes
|
2017-10-11 08:19:16 +08:00
|
|
|
$('#note-detail').summernote('reset');
|
2017-08-23 09:32:03 +08:00
|
|
|
|
2017-10-11 08:19:16 +08:00
|
|
|
$('#note-detail').summernote('code', note.detail.note_text);
|
2017-08-16 10:32:30 +08:00
|
|
|
|
2017-09-05 09:28:07 +08:00
|
|
|
document.location.hash = noteId;
|
2017-08-23 10:40:54 +08:00
|
|
|
|
2017-09-05 09:28:07 +08:00
|
|
|
addRecentNote(noteId, note.detail.note_id);
|
|
|
|
|
|
|
|
noteChangeDisabled = false;
|
2017-09-05 09:37:55 +08:00
|
|
|
|
|
|
|
setNoteBackgroundIfEncrypted(note);
|
2017-09-05 09:28:07 +08:00
|
|
|
});
|
2017-06-12 04:04:07 +08:00
|
|
|
});
|
2017-10-12 08:37:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
function loadNote(noteId, callback) {
|
|
|
|
$.get(baseApiUrl + 'notes/' + noteId).then(note => {
|
|
|
|
if (note.detail.encryption > 0 && !isEncryptionAvailable()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
decryptNoteIfNecessary(note);
|
|
|
|
|
|
|
|
callback(note);
|
|
|
|
});
|
2017-09-10 00:06:15 +08:00
|
|
|
}
|