mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-17 18:44:45 +02:00
310 lines
7.3 KiB
JavaScript
310 lines
7.3 KiB
JavaScript
const { Note } = require('lib/models/note.js');
|
|
|
|
const defaultState = {
|
|
notes: [],
|
|
notesSource: '',
|
|
notesParentType: null,
|
|
folders: [],
|
|
tags: [],
|
|
searches: [],
|
|
selectedNoteId: null,
|
|
selectedFolderId: null,
|
|
selectedTagId: null,
|
|
selectedSearchId: null,
|
|
selectedItemType: 'note',
|
|
showSideMenu: false,
|
|
screens: {},
|
|
historyCanGoBack: false,
|
|
notesOrder: [
|
|
{ by: 'user_updated_time', dir: 'DESC' },
|
|
],
|
|
syncStarted: false,
|
|
syncReport: {},
|
|
searchQuery: '',
|
|
settings: {},
|
|
appState: 'starting',
|
|
windowContentSize: { width: 0, height: 0 },
|
|
};
|
|
|
|
function folderOrNoteDelete(state, action) {
|
|
let newState = Object.assign({}, state);
|
|
|
|
const idKey = action.type === 'FOLDER_DELETE' ? 'folderId' : 'noteId';
|
|
const listKey = action.type === 'FOLDER_DELETE' ? 'folders' : 'notes';
|
|
const selectedItemKey = action.type === 'FOLDER_DELETE' ? 'selectedFolderId' : 'selectedNoteId';
|
|
|
|
let previousIndex = 0;
|
|
let newItems = [];
|
|
const items = state[listKey];
|
|
for (let i = 0; i < items.length; i++) {
|
|
let f = items[i];
|
|
if (f.id == action[idKey]) {
|
|
previousIndex = i;
|
|
continue;
|
|
}
|
|
newItems.push(f);
|
|
}
|
|
|
|
newState = Object.assign({}, state);
|
|
newState[listKey] = newItems;
|
|
|
|
if (previousIndex >= newItems.length) {
|
|
previousIndex = newItems.length - 1;
|
|
}
|
|
|
|
const newIndex = previousIndex >= 0 ? newItems[previousIndex].id : null;
|
|
newState[selectedItemKey] = newIndex;
|
|
|
|
return newState;
|
|
}
|
|
|
|
function updateOneTagOrFolder(state, action) {
|
|
let newItems = action.type === 'TAGS_UPDATE_ONE' ? state.tags.splice(0) : state.folders.splice(0);
|
|
let item = action.type === 'TAGS_UPDATE_ONE' ? action.tag : action.folder;
|
|
|
|
var found = false;
|
|
for (let i = 0; i < newItems.length; i++) {
|
|
let n = newItems[i];
|
|
if (n.id == item.id) {
|
|
newItems[i] = Object.assign(newItems[i], item);
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found) newItems.push(item);
|
|
|
|
let newState = Object.assign({}, state);
|
|
|
|
if (action.type === 'TAGS_UPDATE_ONE') {
|
|
newState.tags = newItems;
|
|
} else {
|
|
newState.folders = newItems;
|
|
}
|
|
|
|
return newState;
|
|
}
|
|
|
|
function defaultNotesParentType(state, exclusion) {
|
|
let newNotesParentType = null;
|
|
|
|
if (exclusion !== 'Folder' && state.selectedFolderId) {
|
|
newNotesParentType = 'Folder';
|
|
} else if (exclusion !== 'Tag' && state.selectedTagId) {
|
|
newNotesParentType = 'Tag';
|
|
} else if (exclusion !== 'Search' && state.selectedSearchId) {
|
|
newNotesParentType = 'Search';
|
|
}
|
|
|
|
return newNotesParentType;
|
|
}
|
|
|
|
const reducer = (state = defaultState, action) => {
|
|
let newState = state;
|
|
|
|
try {
|
|
switch (action.type) {
|
|
|
|
case 'NOTES_SELECT':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.selectedNoteId = action.noteId;
|
|
break;
|
|
|
|
case 'FOLDERS_SELECT':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.selectedFolderId = action.id;
|
|
if (!action.id) {
|
|
newState.notesParentType = defaultNotesParentType(state, 'Folder');
|
|
} else {
|
|
newState.notesParentType = 'Folder';
|
|
}
|
|
break;
|
|
|
|
case 'SETTINGS_UPDATE_ALL':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.settings = action.settings;
|
|
break;
|
|
|
|
case 'SETTINGS_UPDATE_ONE':
|
|
|
|
newState = Object.assign({}, state);
|
|
let newSettings = Object.assign({}, state.settings);
|
|
newSettings[action.key] = action.value;
|
|
newState.settings = newSettings;
|
|
break;
|
|
|
|
// Replace all the notes with the provided array
|
|
case 'NOTES_UPDATE_ALL':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.notes = action.notes;
|
|
newState.notesSource = action.notesSource;
|
|
break;
|
|
|
|
// Insert the note into the note list if it's new, or
|
|
// update it within the note array if it already exists.
|
|
case 'NOTES_UPDATE_ONE':
|
|
|
|
const modNote = action.note;
|
|
|
|
let noteFolderHasChanged = false;
|
|
let newNotes = state.notes.slice();
|
|
var found = false;
|
|
for (let i = 0; i < newNotes.length; i++) {
|
|
let n = newNotes[i];
|
|
if (n.id == modNote.id) {
|
|
|
|
// Note is still in the same folder
|
|
if (!('parent_id' in modNote) || modNote.parent_id == n.parent_id) {
|
|
// Merge the properties that have changed (in modNote) into
|
|
// the object we already have.
|
|
newNotes[i] = Object.assign({}, newNotes[i]);
|
|
|
|
for (let n in modNote) {
|
|
if (!modNote.hasOwnProperty(n)) continue;
|
|
newNotes[i][n] = modNote[n];
|
|
}
|
|
|
|
} else { // Note has moved to a different folder
|
|
newNotes.splice(i, 1);
|
|
noteFolderHasChanged = true;
|
|
}
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found && ('parent_id' in modNote) && modNote.parent_id == state.selectedFolderId) newNotes.push(modNote);
|
|
|
|
newNotes = Note.sortNotes(newNotes, state.notesOrder, newState.settings.uncompletedTodosOnTop);
|
|
newState = Object.assign({}, state);
|
|
newState.notes = newNotes;
|
|
|
|
if (noteFolderHasChanged) {
|
|
newState.selectedNoteId = newNotes.length ? newNotes[0].id : null;
|
|
}
|
|
break;
|
|
|
|
case 'NOTES_DELETE':
|
|
|
|
newState = folderOrNoteDelete(state, action);
|
|
break;
|
|
|
|
case 'FOLDERS_UPDATE_ALL':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.folders = action.folders;
|
|
break;
|
|
|
|
case 'TAGS_UPDATE_ALL':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.tags = action.tags;
|
|
break;
|
|
|
|
case 'TAGS_SELECT':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.selectedTagId = action.id;
|
|
if (!action.id) {
|
|
newState.notesParentType = defaultNotesParentType(state, 'Tag');
|
|
} else {
|
|
newState.notesParentType = 'Tag';
|
|
}
|
|
break;
|
|
|
|
case 'TAGS_UPDATE_ONE':
|
|
|
|
newState = updateOneTagOrFolder(state, action);
|
|
break;
|
|
|
|
case 'FOLDERS_UPDATE_ONE':
|
|
|
|
newState = updateOneTagOrFolder(state, action);
|
|
break;
|
|
|
|
case 'FOLDER_DELETE':
|
|
|
|
newState = folderOrNoteDelete(state, action);
|
|
break;
|
|
|
|
case 'SYNC_STARTED':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.syncStarted = true;
|
|
break;
|
|
|
|
case 'SYNC_COMPLETED':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.syncStarted = false;
|
|
break;
|
|
|
|
case 'SYNC_REPORT_UPDATE':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.syncReport = action.report;
|
|
break;
|
|
|
|
case 'SEARCH_QUERY':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.searchQuery = action.query.trim();
|
|
break;
|
|
|
|
case 'SEARCH_ADD':
|
|
|
|
newState = Object.assign({}, state);
|
|
let searches = newState.searches.slice();
|
|
searches.push(action.search);
|
|
newState.searches = searches;
|
|
break;
|
|
|
|
case 'SEARCH_REMOVE':
|
|
|
|
let foundIndex = -1;
|
|
for (let i = 0; i < state.searches.length; i++) {
|
|
if (state.searches[i].id === action.id) {
|
|
foundIndex = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (foundIndex >= 0) {
|
|
newState = Object.assign({}, state);
|
|
let newSearches = newState.searches.slice();
|
|
newSearches.splice(foundIndex, 1);
|
|
newState.searches = newSearches;
|
|
}
|
|
break;
|
|
|
|
case 'SEARCH_SELECT':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.selectedSearchId = action.id;
|
|
if (!action.id) {
|
|
newState.notesParentType = defaultNotesParentType(state, 'Search');
|
|
} else {
|
|
newState.notesParentType = 'Search';
|
|
}
|
|
break;
|
|
|
|
case 'SET_APP_STATE':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.appState = action.state;
|
|
break;
|
|
|
|
}
|
|
} catch (error) {
|
|
error.message = 'In reducer: ' + error.message + ' Action: ' + JSON.stringify(action);
|
|
throw error;
|
|
}
|
|
|
|
return newState;
|
|
}
|
|
|
|
module.exports = { reducer, defaultState }; |