mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-20 18:48:28 +02:00
1537 lines
48 KiB
TypeScript
1537 lines
48 KiB
TypeScript
import produce, { Draft, original } from 'immer';
|
|
import pluginServiceReducer, { stateRootKey as pluginServiceStateRootKey, defaultState as pluginServiceDefaultState, State as PluginServiceState } from './services/plugins/reducer';
|
|
import shareServiceReducer, { stateRootKey as shareServiceStateRootKey, defaultState as shareServiceDefaultState, State as ShareServiceState } from './services/share/reducer';
|
|
import Note from './models/Note';
|
|
import Folder from './models/Folder';
|
|
import BaseModel from './BaseModel';
|
|
import { Store } from 'redux';
|
|
import { ProfileConfig } from './services/profileConfig/types';
|
|
import * as ArrayUtils from './ArrayUtils';
|
|
import { FolderEntity, NoteEntity, NoteTagEntity } from './services/database/types';
|
|
import { getListRendererIds } from './services/noteList/renderers';
|
|
import { ProcessResultsRow } from './services/search/SearchEngine';
|
|
import { getDisplayParentId } from './services/trash';
|
|
import Logger from '@joplin/utils/Logger';
|
|
import { SettingsRecord } from './models/settings/types';
|
|
const fastDeepEqual = require('fast-deep-equal');
|
|
const { ALL_NOTES_FILTER_ID } = require('./reserved-ids');
|
|
const { createSelectorCreator, defaultMemoize } = require('reselect');
|
|
const { createCachedSelector } = require('re-reselect');
|
|
|
|
const logger = Logger.create('lib/reducer');
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const additionalReducers: any[] = [];
|
|
|
|
additionalReducers.push({
|
|
stateRootKey: pluginServiceStateRootKey,
|
|
defaultState: pluginServiceDefaultState,
|
|
reducer: pluginServiceReducer,
|
|
});
|
|
|
|
additionalReducers.push({
|
|
stateRootKey: shareServiceStateRootKey,
|
|
defaultState: shareServiceDefaultState,
|
|
reducer: shareServiceReducer,
|
|
});
|
|
|
|
interface StateLastSelectedNotesIds {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
Folder: any;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
Tag: any;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
Search: any;
|
|
}
|
|
|
|
interface StateClipperServer {
|
|
startState: string;
|
|
port: number;
|
|
}
|
|
|
|
export interface StateDecryptionWorker {
|
|
state: string;
|
|
itemIndex: number;
|
|
itemCount: number;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
decryptedItemCounts: any;
|
|
decryptedItemCount: number;
|
|
skippedItemCount: number;
|
|
}
|
|
|
|
export interface StateResourceFetcher {
|
|
toFetchCount: number;
|
|
fetchingCount: number;
|
|
}
|
|
|
|
export interface StateLastDeletion {
|
|
noteIds: string[];
|
|
folderIds: string[];
|
|
timestamp: number;
|
|
}
|
|
|
|
export interface WindowState {
|
|
windowId: string;
|
|
notes: NoteEntity[];
|
|
noteSelectionEnabled?: boolean;
|
|
notesSource: string;
|
|
notesParentType: string;
|
|
selectedNoteTags: NoteTagEntity[];
|
|
searchQuery: string;
|
|
|
|
selectedNoteIds: string[];
|
|
selectedNoteHash: string;
|
|
selectedFolderId: string;
|
|
selectedTagId: string;
|
|
selectedSearchId: string;
|
|
selectedItemType: string;
|
|
selectedSmartFilterId: string;
|
|
|
|
backwardHistoryNotes: NoteEntity[];
|
|
forwardHistoryNotes: NoteEntity[];
|
|
lastSelectedNotesIds: StateLastSelectedNotesIds;
|
|
}
|
|
|
|
export const defaultWindowId = 'default';
|
|
export const defaultWindowState: WindowState = {
|
|
windowId: defaultWindowId,
|
|
searchQuery: '',
|
|
notes: [],
|
|
notesSource: '',
|
|
notesParentType: null,
|
|
selectedNoteIds: [],
|
|
selectedNoteHash: '',
|
|
selectedFolderId: null,
|
|
selectedTagId: null,
|
|
selectedSearchId: null,
|
|
selectedSmartFilterId: null,
|
|
selectedItemType: 'note',
|
|
selectedNoteTags: [],
|
|
backwardHistoryNotes: [],
|
|
forwardHistoryNotes: [],
|
|
lastSelectedNotesIds: {
|
|
Folder: {},
|
|
Tag: {},
|
|
Search: {},
|
|
},
|
|
};
|
|
|
|
export interface EditorNoteStatuses {
|
|
[id: string]: string;
|
|
}
|
|
|
|
export interface State extends WindowState {
|
|
// Contains state specific to windows that currently don't have focus.
|
|
// See spec/background_windows.md for details.
|
|
backgroundWindows: Record<string, WindowState>;
|
|
|
|
folders: FolderEntity[];
|
|
tags: NoteTagEntity[];
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
masterKeys: any[];
|
|
notLoadedMasterKeys: string[];
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
searches: any[];
|
|
highlightedWords: string[];
|
|
showSideMenu: boolean;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
screens: any;
|
|
historyCanGoBack: boolean;
|
|
syncStarted: boolean;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
syncReport: any;
|
|
searchResults: ProcessResultsRow[];
|
|
settings: Partial<SettingsRecord>;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
sharedData: any;
|
|
appState: string;
|
|
biometricsDone: boolean;
|
|
hasDisabledSyncItems: boolean;
|
|
hasDisabledEncryptionItems: boolean;
|
|
customViewerCss: string;
|
|
customChromeCssPaths: string[];
|
|
collapsedFolderIds: string[];
|
|
clipperServer: StateClipperServer;
|
|
decryptionWorker: StateDecryptionWorker;
|
|
resourceFetcher: StateResourceFetcher;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
pluginsLegacy: any;
|
|
provisionalNoteIds: string[];
|
|
editorNoteStatuses: EditorNoteStatuses;
|
|
isInsertingNotes: boolean;
|
|
hasEncryptedItems: boolean;
|
|
needApiAuth: boolean;
|
|
profileConfig: ProfileConfig;
|
|
noteListRendererIds: string[];
|
|
noteListLastSortTime: number;
|
|
lastDeletion: StateLastDeletion;
|
|
lastDeletionNotificationTime: number;
|
|
mustUpgradeAppMessage: string;
|
|
mustAuthenticate: boolean;
|
|
|
|
allowSelectionInOtherFolders: boolean;
|
|
|
|
// Extra reducer keys go here:
|
|
pluginService: PluginServiceState;
|
|
shareService: ShareServiceState;
|
|
}
|
|
|
|
export const defaultState: State = {
|
|
...defaultWindowState,
|
|
backgroundWindows: {},
|
|
folders: [],
|
|
tags: [],
|
|
masterKeys: [],
|
|
notLoadedMasterKeys: [],
|
|
searches: [],
|
|
highlightedWords: [],
|
|
showSideMenu: false,
|
|
screens: {},
|
|
historyCanGoBack: false,
|
|
syncStarted: false,
|
|
syncReport: {},
|
|
searchQuery: '',
|
|
searchResults: [],
|
|
settings: {},
|
|
sharedData: null,
|
|
appState: 'starting',
|
|
biometricsDone: false,
|
|
hasDisabledSyncItems: false,
|
|
hasDisabledEncryptionItems: false,
|
|
customViewerCss: '',
|
|
customChromeCssPaths: [],
|
|
collapsedFolderIds: [],
|
|
clipperServer: {
|
|
startState: 'idle',
|
|
port: null,
|
|
},
|
|
decryptionWorker: {
|
|
state: 'idle',
|
|
itemIndex: 0,
|
|
itemCount: 0,
|
|
decryptedItemCounts: {},
|
|
decryptedItemCount: 0,
|
|
skippedItemCount: 0,
|
|
},
|
|
resourceFetcher: {
|
|
toFetchCount: 0,
|
|
fetchingCount: 0,
|
|
},
|
|
// pluginsLegacy is the original plugin system, which eventually was used only for GotoAnything.
|
|
// GotoAnything should be refactored to part of core and when it's done the pluginsLegacy key can
|
|
// be removed. It was originally named "plugins", then renamed "pluginsLegacy" so as not to conflict
|
|
// with the new "plugins" key used for the new plugin system.
|
|
pluginsLegacy: {},
|
|
provisionalNoteIds: [],
|
|
editorNoteStatuses: {},
|
|
isInsertingNotes: false,
|
|
hasEncryptedItems: false,
|
|
needApiAuth: false,
|
|
profileConfig: null,
|
|
noteListRendererIds: getListRendererIds(),
|
|
noteListLastSortTime: 0,
|
|
lastDeletion: {
|
|
noteIds: [],
|
|
folderIds: [],
|
|
timestamp: 0,
|
|
},
|
|
lastDeletionNotificationTime: 0,
|
|
mustUpgradeAppMessage: '',
|
|
mustAuthenticate: false,
|
|
allowSelectionInOtherFolders: false,
|
|
|
|
pluginService: pluginServiceDefaultState,
|
|
shareService: shareServiceDefaultState,
|
|
};
|
|
|
|
for (const additionalReducer of additionalReducers) {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(defaultState as any)[additionalReducer.stateRootKey] = additionalReducer.defaultState;
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
let store_: Store<any> = null;
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
export function setStore(v: Store<any>) {
|
|
store_ = v;
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
export function store(): Store<any> {
|
|
return store_;
|
|
}
|
|
|
|
export const MAX_HISTORY = 200;
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const derivedStateCache_: any = {};
|
|
|
|
// Allows, for a given state, to return the same derived
|
|
// objects, to prevent unnecessary updates on calling components.
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const cacheEnabledOutput = (key: string, output: any) => {
|
|
key = `${key}_${JSON.stringify(output)}`;
|
|
if (derivedStateCache_[key]) return derivedStateCache_[key];
|
|
|
|
derivedStateCache_[key] = output;
|
|
return derivedStateCache_[key];
|
|
};
|
|
|
|
const createShallowArrayEqualSelector = createSelectorCreator(
|
|
defaultMemoize,
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(prev: any[], next: any[]) => {
|
|
if (prev.length !== next.length) return false;
|
|
for (let i = 0; i < prev.length; i++) {
|
|
if (prev[i] !== next[i]) return false;
|
|
}
|
|
return true;
|
|
},
|
|
);
|
|
|
|
const selectArrayShallow = createCachedSelector(
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(state: any) => state.array,
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(array: any[]) => array,
|
|
)({
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
keySelector: (_state: any, cacheKey: any) => {
|
|
return cacheKey;
|
|
},
|
|
selectorCreator: createShallowArrayEqualSelector,
|
|
});
|
|
|
|
class StateUtils {
|
|
|
|
// Given an input array, this selector ensures that the same array is returned
|
|
// if its content hasn't changed.
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
public selectArrayShallow(props: any, cacheKey: any) {
|
|
return selectArrayShallow(props, cacheKey);
|
|
}
|
|
|
|
public oneNoteSelected(state: WindowState): boolean {
|
|
return state.selectedNoteIds.length === 1;
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
public notesOrder(stateSettings: any) {
|
|
if (stateSettings['notes.sortOrder.field'] === 'order') {
|
|
return cacheEnabledOutput('notesOrder', [
|
|
{
|
|
by: 'order',
|
|
dir: 'DESC',
|
|
},
|
|
{
|
|
by: 'user_created_time',
|
|
dir: 'DESC',
|
|
},
|
|
]);
|
|
} else {
|
|
return cacheEnabledOutput('notesOrder', [
|
|
{
|
|
by: stateSettings['notes.sortOrder.field'],
|
|
dir: stateSettings['notes.sortOrder.reverse'] ? 'DESC' : 'ASC',
|
|
},
|
|
]);
|
|
}
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
public foldersOrder(stateSettings: any) {
|
|
return cacheEnabledOutput('foldersOrder', [
|
|
{
|
|
by: stateSettings['folders.sortOrder.field'],
|
|
dir: stateSettings['folders.sortOrder.reverse'] ? 'DESC' : 'ASC',
|
|
},
|
|
]);
|
|
}
|
|
|
|
public hasNotesBeingSaved(state: State): boolean {
|
|
for (const id in state.editorNoteStatuses) {
|
|
if (state.editorNoteStatuses[id] === 'saving') return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public parentItem(state: WindowState) {
|
|
const t = state.notesParentType;
|
|
let id = null;
|
|
if (t === 'Folder') id = state.selectedFolderId;
|
|
if (t === 'Tag') id = state.selectedTagId;
|
|
if (t === 'Search') id = state.selectedSearchId;
|
|
if (!t || !id) return null;
|
|
return { type: t, id: id };
|
|
}
|
|
|
|
public lastSelectedNoteIds(state: WindowState): string[] {
|
|
const parent = this.parentItem(state);
|
|
if (!parent) return [];
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const output = (state.lastSelectedNotesIds as any)[parent.type][parent.id];
|
|
return output ? output : [];
|
|
}
|
|
|
|
public selectedNote(state: WindowState): NoteEntity {
|
|
const noteId = this.selectedNoteId(state);
|
|
return noteId ? BaseModel.byId(state.notes, noteId) : null;
|
|
}
|
|
|
|
public selectedNoteId(state: WindowState): string|null {
|
|
return state.selectedNoteIds.length ? state.selectedNoteIds[0] : null;
|
|
}
|
|
|
|
public activeWindowId(state: State) {
|
|
return state.windowId;
|
|
}
|
|
|
|
private allWindowIds(state: State) {
|
|
return [state.windowId, ...Object.keys(state.backgroundWindows)];
|
|
}
|
|
|
|
public allWindowStates<T extends State>(state: T) {
|
|
return this.allWindowIds(state).map(id => this.windowStateById(state, id));
|
|
}
|
|
|
|
public windowStateById<StateType extends State>(
|
|
state: StateType, id: string,
|
|
) {
|
|
// States for the different Joplin apps can have different types for backgroundWindows -- this
|
|
// makes sure that the correct type is returned.
|
|
type AppWindowState = StateType['backgroundWindows'][keyof StateType['backgroundWindows']];
|
|
const result = id === state.windowId ? state : state.backgroundWindows[id];
|
|
return result as AppWindowState;
|
|
}
|
|
|
|
public mainWindowState(state: State) {
|
|
return this.windowStateById(state, defaultWindowId);
|
|
}
|
|
|
|
public secondaryWindowStates(state: State) {
|
|
const windowIds = [state.windowId, ...Object.keys(state.backgroundWindows)];
|
|
return windowIds
|
|
.filter(id => (id !== defaultWindowId))
|
|
.map(id => this.windowStateById(state, id));
|
|
}
|
|
|
|
public windowIdToSelectedNoteIds(state: State) {
|
|
const result: Record<string, string[]> = {};
|
|
for (const id of this.allWindowIds(state)) {
|
|
result[id] = this.windowStateById(state, id).selectedNoteIds;
|
|
}
|
|
return result;
|
|
}
|
|
}
|
|
|
|
export const stateUtils: StateUtils = new StateUtils();
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
function arrayHasEncryptedItems(array: any[]) {
|
|
for (let i = 0; i < array.length; i++) {
|
|
if (array[i].encryption_applied) return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function stateHasEncryptedItems(state: State) {
|
|
if (arrayHasEncryptedItems(state.notes)) return true;
|
|
if (arrayHasEncryptedItems(state.folders)) return true;
|
|
if (arrayHasEncryptedItems(state.tags)) return true;
|
|
return false;
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
function folderSetCollapsed(draft: Draft<State>, action: any) {
|
|
const collapsedFolderIds = draft.collapsedFolderIds.slice();
|
|
const idx = collapsedFolderIds.indexOf(action.id);
|
|
|
|
if (action.collapsed) {
|
|
if (idx >= 0) return;
|
|
collapsedFolderIds.push(action.id);
|
|
} else {
|
|
if (idx < 0) return;
|
|
collapsedFolderIds.splice(idx, 1);
|
|
}
|
|
|
|
draft.collapsedFolderIds = collapsedFolderIds;
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
function removeAdjacentDuplicates(items: any[]) {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
return items.filter((item: any, idx: number) => (idx >= 1) ? items[idx - 1].id !== item.id : true);
|
|
}
|
|
|
|
// When deleting a note, tag or folder
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
function handleItemDelete(draft: Draft<State>, action: any) {
|
|
type SelectionKey = 'selectedFolderId'|'selectedNoteIds'|'selectedTagId'|'selectedSearchId';
|
|
const map: Record<string, [keyof State, SelectionKey, boolean]> = {
|
|
FOLDER_DELETE: ['folders', 'selectedFolderId', true],
|
|
NOTE_DELETE: ['notes', 'selectedNoteIds', false],
|
|
TAG_DELETE: ['tags', 'selectedTagId', true],
|
|
SEARCH_DELETE: ['searches', 'selectedSearchId', true],
|
|
};
|
|
|
|
const listKey = map[action.type][0];
|
|
const selectedItemKey = map[action.type][1];
|
|
const isSingular = map[action.type][2];
|
|
|
|
for (const windowDraft of stateUtils.allWindowStates(draft)) {
|
|
const selectedItemKeys = isSingular ? [windowDraft[selectedItemKey]] : windowDraft[selectedItemKey];
|
|
const isSelected = selectedItemKeys.includes(action.id);
|
|
|
|
const items = listKey in windowDraft ? windowDraft[listKey as keyof WindowState] : draft[listKey];
|
|
const newItems = [];
|
|
let newSelectedIndexes: number[] = [];
|
|
|
|
for (let i = 0; i < items.length; i++) {
|
|
const item = items[i];
|
|
if (isSelected) {
|
|
// the selected item is deleted so select the following item
|
|
// if multiple items are selected then just use the first one
|
|
if (selectedItemKeys[0] === item.id) {
|
|
newSelectedIndexes.push(newItems.length);
|
|
}
|
|
} else {
|
|
// the selected item/s is not deleted so keep it selected
|
|
if (selectedItemKeys.includes(item.id)) {
|
|
newSelectedIndexes.push(newItems.length);
|
|
}
|
|
}
|
|
if (item.id === action.id) {
|
|
continue;
|
|
}
|
|
newItems.push(item);
|
|
}
|
|
|
|
if (newItems.length === 0) {
|
|
newSelectedIndexes = []; // no remaining items so no selection
|
|
|
|
} else if (newSelectedIndexes.length === 0) {
|
|
newSelectedIndexes.push(0); // no selection exists so select the top
|
|
|
|
} else {
|
|
// when the items at end of list are deleted then select the end
|
|
for (let i = 0; i < newSelectedIndexes.length; i++) {
|
|
if (newSelectedIndexes[i] >= newItems.length) {
|
|
newSelectedIndexes = [newItems.length - 1];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (listKey in windowDraft) {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(windowDraft as any)[listKey] = newItems;
|
|
}
|
|
|
|
const newIds = [];
|
|
for (let i = 0; i < newSelectedIndexes.length; i++) {
|
|
newIds.push(newItems[newSelectedIndexes[i]].id);
|
|
}
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(windowDraft as any)[selectedItemKey] = isSingular ? newIds[0] : newIds;
|
|
|
|
if ((newIds.length === 0) && windowDraft.notesParentType !== 'Folder') {
|
|
windowDraft.notesParentType = 'Folder';
|
|
}
|
|
}
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
function updateOneItem(draft: Draft<State|WindowState>, action: any, keyName = '') {
|
|
let itemsKey = null;
|
|
if (keyName) { itemsKey = keyName; } else {
|
|
if (action.type === 'TAG_UPDATE_ONE') itemsKey = 'tags';
|
|
if (action.type === 'FOLDER_UPDATE_ONE') itemsKey = 'folders';
|
|
if (action.type === 'MASTERKEY_UPDATE_ONE') itemsKey = 'masterKeys';
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const newItems = (draft as any)[itemsKey].slice();
|
|
const item = action.item;
|
|
|
|
let found = false;
|
|
for (let i = 0; i < newItems.length; i++) {
|
|
const n = newItems[i];
|
|
if (n.id === item.id) {
|
|
newItems[i] = { ...newItems[i], ...item };
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found) newItems.push(item);
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(draft as any)[itemsKey] = newItems;
|
|
}
|
|
|
|
function updateSelectedNotesFromExistingNotes(draft: Draft<State>) {
|
|
const newSelectedNoteIds = [];
|
|
for (const selectedNoteId of draft.selectedNoteIds) {
|
|
for (const n of draft.notes) {
|
|
if (n.id === selectedNoteId) {
|
|
newSelectedNoteIds.push(n.id);
|
|
}
|
|
}
|
|
}
|
|
if (JSON.stringify(draft.selectedNoteIds) === JSON.stringify(newSelectedNoteIds)) return;
|
|
draft.selectedNoteIds = newSelectedNoteIds;
|
|
}
|
|
|
|
function defaultNotesParentType(draft: Draft<State>, exclusion: string) {
|
|
let newNotesParentType = null;
|
|
|
|
if (exclusion !== 'SmartFilter' && draft.selectedSmartFilterId) {
|
|
newNotesParentType = 'SmartFilter';
|
|
} else if (exclusion !== 'Folder' && draft.selectedFolderId) {
|
|
newNotesParentType = 'Folder';
|
|
} else if (exclusion !== 'Tag' && draft.selectedTagId) {
|
|
newNotesParentType = 'Tag';
|
|
} else if (exclusion !== 'Search' && draft.selectedSearchId) {
|
|
newNotesParentType = 'Search';
|
|
}
|
|
|
|
return newNotesParentType;
|
|
}
|
|
|
|
export type NotesParentType = 'Folder' | 'Tag' | 'SmartFilter';
|
|
|
|
export interface NotesParent {
|
|
type: NotesParentType;
|
|
selectedItemId: string;
|
|
}
|
|
|
|
export const serializeNotesParent = (n: NotesParent) => {
|
|
return JSON.stringify(n);
|
|
};
|
|
|
|
export const parseNotesParent = (s: string, activeFolderId: string): NotesParent => {
|
|
const defaultValue: NotesParent = {
|
|
type: 'Folder',
|
|
selectedItemId: activeFolderId,
|
|
};
|
|
|
|
if (!s) return defaultValue;
|
|
|
|
try {
|
|
const parsed = JSON.parse(s);
|
|
return parsed;
|
|
} catch (error) {
|
|
return defaultValue;
|
|
}
|
|
};
|
|
|
|
export const getNotesParent = (state: State): NotesParent => {
|
|
let type = state.notesParentType as NotesParentType;
|
|
let selectedItemId = '';
|
|
|
|
if (type === 'Folder') {
|
|
selectedItemId = state.selectedFolderId;
|
|
} else if (type === 'Tag') {
|
|
selectedItemId = state.selectedTagId;
|
|
} else if (type === 'SmartFilter') {
|
|
selectedItemId = state.selectedSmartFilterId;
|
|
} else {
|
|
type = 'Folder';
|
|
selectedItemId = state.selectedFolderId;
|
|
}
|
|
|
|
return { type, selectedItemId };
|
|
};
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
function changeSelectedFolder(draft: Draft<State>, action: any, options: any = null) {
|
|
if (!options) options = {};
|
|
draft.selectedFolderId = 'folderId' in action ? action.folderId : action.id;
|
|
if (!draft.selectedFolderId) {
|
|
draft.notesParentType = defaultNotesParentType(draft, 'Folder');
|
|
} else {
|
|
draft.notesParentType = 'Folder';
|
|
}
|
|
|
|
if (options.clearSelectedNoteIds) draft.selectedNoteIds = [];
|
|
}
|
|
|
|
function recordLastSelectedNoteIds(draft: Draft<State>, noteIds: string[]) {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const newOnes: any = { ...draft.lastSelectedNotesIds };
|
|
const parent = stateUtils.parentItem(draft);
|
|
if (!parent) return;
|
|
|
|
newOnes[parent.type] = { ...newOnes[parent.type] };
|
|
newOnes[parent.type][parent.id] = noteIds.slice();
|
|
|
|
draft.lastSelectedNotesIds = newOnes;
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
function changeSelectedNotes(draft: Draft<State>, action: any, options: any = null) {
|
|
if (!options) options = {};
|
|
|
|
let noteIds = [];
|
|
if (action.id) noteIds = [action.id];
|
|
if (action.ids) noteIds = action.ids;
|
|
if (action.noteId) noteIds = [action.noteId];
|
|
if (action.index) noteIds = [draft.notes[action.index].id];
|
|
|
|
if (action.type === 'NOTE_SELECT') {
|
|
if (JSON.stringify(draft.selectedNoteIds) === JSON.stringify(noteIds)) return;
|
|
draft.selectedNoteIds = noteIds;
|
|
draft.selectedNoteHash = action.hash ? action.hash : '';
|
|
} else if (action.type === 'NOTE_SELECT_ADD') {
|
|
if (!noteIds.length) return;
|
|
draft.selectedNoteIds = ArrayUtils.unique(draft.selectedNoteIds.concat(noteIds));
|
|
} else if (action.type === 'NOTE_SELECT_REMOVE') {
|
|
if (!noteIds.length) return; // Nothing to unselect
|
|
if (draft.selectedNoteIds.length <= 1) return; // Cannot unselect the last note
|
|
|
|
const newSelectedNoteIds = [];
|
|
for (let i = 0; i < draft.selectedNoteIds.length; i++) {
|
|
const id = draft.selectedNoteIds[i];
|
|
if (noteIds.indexOf(id) >= 0) continue;
|
|
newSelectedNoteIds.push(id);
|
|
}
|
|
draft.selectedNoteIds = newSelectedNoteIds;
|
|
} else if (action.type === 'NOTE_SELECT_TOGGLE') {
|
|
if (!noteIds.length) return;
|
|
|
|
if (draft.selectedNoteIds.indexOf(noteIds[0]) >= 0) {
|
|
changeSelectedNotes(draft, { type: 'NOTE_SELECT_REMOVE', id: noteIds[0] });
|
|
} else {
|
|
changeSelectedNotes(draft, { type: 'NOTE_SELECT_ADD', id: noteIds[0] });
|
|
}
|
|
} else {
|
|
throw new Error('Unreachable');
|
|
}
|
|
|
|
recordLastSelectedNoteIds(draft, draft.selectedNoteIds);
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
function removeItemFromArray(array: any[], property: any, value: any) {
|
|
for (let i = 0; i !== array.length; ++i) {
|
|
const currentItem = array[i];
|
|
if (currentItem[property] === value) {
|
|
array = array.slice();
|
|
array.splice(i, 1);
|
|
break;
|
|
}
|
|
}
|
|
return array;
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const getContextFromHistory = (ctx: any) => {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const result: any = {};
|
|
result.notesParentType = ctx.notesParentType;
|
|
if (result.notesParentType === 'Folder') {
|
|
result.selectedFolderId = ctx.selectedFolderId;
|
|
} else if (result.notesParentType === 'Tag') {
|
|
result.selectedTagId = ctx.selectedTagId;
|
|
} else if (result.notesParentType === 'Search') {
|
|
result.selectedSearchId = ctx.selectedSearchId;
|
|
result.searches = ctx.searches;
|
|
} else if (result.notesParentType === 'SmartFilter') {
|
|
result.selectedSmartFilterId = ctx.selectedSmartFilterId;
|
|
}
|
|
return result;
|
|
};
|
|
|
|
function getNoteHistoryInfo(state: State) {
|
|
const selectedNoteIds = state.selectedNoteIds;
|
|
const notes = state.notes;
|
|
if (selectedNoteIds && selectedNoteIds.length > 0) {
|
|
const currNote = notes.find(note => note.id === selectedNoteIds[0]);
|
|
if (currNote) {
|
|
return {
|
|
id: currNote.id,
|
|
parent_id: currNote.parent_id,
|
|
notesParentType: state.notesParentType,
|
|
selectedFolderId: state.selectedFolderId,
|
|
selectedTagId: state.selectedTagId,
|
|
selectedSearchId: state.selectedSearchId,
|
|
searches: state.searches,
|
|
selectedSmartFilterId: state.selectedSmartFilterId,
|
|
};
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
function handleHistory(draft: Draft<State>, action: any) {
|
|
const currentNote = getNoteHistoryInfo(draft);
|
|
switch (action.type) {
|
|
case 'HISTORY_BACKWARD': {
|
|
const note = draft.backwardHistoryNotes[draft.backwardHistoryNotes.length - 1];
|
|
if (currentNote && (draft.forwardHistoryNotes.length === 0 || currentNote.id !== draft.forwardHistoryNotes[draft.forwardHistoryNotes.length - 1].id)) {
|
|
draft.forwardHistoryNotes = draft.forwardHistoryNotes.concat(currentNote).slice(-MAX_HISTORY);
|
|
}
|
|
|
|
changeSelectedFolder(draft, { ...action, type: 'FOLDER_SELECT', folderId: note.parent_id });
|
|
changeSelectedNotes(draft, { ...action, type: 'NOTE_SELECT', noteId: note.id });
|
|
|
|
const ctx = draft.backwardHistoryNotes[draft.backwardHistoryNotes.length - 1];
|
|
Object.assign(draft, getContextFromHistory(ctx));
|
|
|
|
draft.backwardHistoryNotes.pop();
|
|
break;
|
|
}
|
|
case 'HISTORY_FORWARD': {
|
|
const note = draft.forwardHistoryNotes[draft.forwardHistoryNotes.length - 1];
|
|
|
|
if (currentNote && (draft.backwardHistoryNotes.length === 0 || currentNote.id !== draft.backwardHistoryNotes[draft.backwardHistoryNotes.length - 1].id)) {
|
|
draft.backwardHistoryNotes = draft.backwardHistoryNotes.concat(currentNote).slice(-MAX_HISTORY);
|
|
}
|
|
|
|
changeSelectedFolder(draft, { ...action, type: 'FOLDER_SELECT', folderId: note.parent_id });
|
|
changeSelectedNotes(draft, { ...action, type: 'NOTE_SELECT', noteId: note.id });
|
|
|
|
const ctx = draft.forwardHistoryNotes[draft.forwardHistoryNotes.length - 1];
|
|
Object.assign(draft, getContextFromHistory(ctx));
|
|
|
|
|
|
draft.forwardHistoryNotes.pop();
|
|
break;
|
|
}
|
|
case 'NOTE_SELECT':
|
|
if (currentNote && action.id !== currentNote.id) {
|
|
draft.forwardHistoryNotes = [];
|
|
draft.backwardHistoryNotes = draft.backwardHistoryNotes.concat(currentNote).slice(-MAX_HISTORY);
|
|
}
|
|
// History should be free from duplicates.
|
|
if (draft.backwardHistoryNotes && draft.backwardHistoryNotes.length > 0 &&
|
|
action.id === draft.backwardHistoryNotes[draft.backwardHistoryNotes.length - 1].id) {
|
|
draft.backwardHistoryNotes.pop();
|
|
}
|
|
break;
|
|
case 'TAG_SELECT':
|
|
case 'FOLDER_AND_NOTE_SELECT':
|
|
case 'FOLDER_SELECT':
|
|
if (currentNote) {
|
|
if (draft.forwardHistoryNotes.length) draft.forwardHistoryNotes = [];
|
|
draft.backwardHistoryNotes = draft.backwardHistoryNotes.concat(currentNote).slice(-MAX_HISTORY);
|
|
}
|
|
break;
|
|
case 'SEARCH_UPDATE':
|
|
if (currentNote && (draft.backwardHistoryNotes.length === 0 ||
|
|
draft.backwardHistoryNotes[draft.backwardHistoryNotes.length - 1].id !== currentNote.id)) {
|
|
if (draft.forwardHistoryNotes.length) draft.forwardHistoryNotes = [];
|
|
draft.backwardHistoryNotes = draft.backwardHistoryNotes.concat(currentNote).slice(-MAX_HISTORY);
|
|
}
|
|
break;
|
|
|
|
case 'SEARCH_RESULTS_SET':
|
|
draft.searchResults = action.value;
|
|
break;
|
|
}
|
|
|
|
const updateWindowHistory = (windowDraft: Draft<WindowState>) => {
|
|
switch (action.type) {
|
|
case 'NOTE_UPDATE_ONE': {
|
|
const modNote = action.note;
|
|
|
|
windowDraft.backwardHistoryNotes = windowDraft.backwardHistoryNotes.map(note => {
|
|
if (note.id === modNote.id) {
|
|
return { ...note, parent_id: modNote.parent_id, selectedFolderId: modNote.parent_id };
|
|
}
|
|
return note;
|
|
});
|
|
|
|
windowDraft.forwardHistoryNotes = windowDraft.forwardHistoryNotes.map(note => {
|
|
if (note.id === modNote.id) {
|
|
return { ...note, parent_id: modNote.parent_id, selectedFolderId: modNote.parent_id };
|
|
}
|
|
return note;
|
|
});
|
|
|
|
break;
|
|
}
|
|
|
|
case 'FOLDER_DELETE':
|
|
windowDraft.backwardHistoryNotes = windowDraft.backwardHistoryNotes.filter(note => note.parent_id !== action.id);
|
|
windowDraft.forwardHistoryNotes = windowDraft.forwardHistoryNotes.filter(note => note.parent_id !== action.id);
|
|
|
|
windowDraft.backwardHistoryNotes = removeAdjacentDuplicates(windowDraft.backwardHistoryNotes);
|
|
windowDraft.forwardHistoryNotes = removeAdjacentDuplicates(windowDraft.forwardHistoryNotes);
|
|
break;
|
|
case 'NOTE_DELETE': {
|
|
windowDraft.backwardHistoryNotes = windowDraft.backwardHistoryNotes.filter(note => note.id !== action.id);
|
|
windowDraft.forwardHistoryNotes = windowDraft.forwardHistoryNotes.filter(note => note.id !== action.id);
|
|
|
|
windowDraft.backwardHistoryNotes = removeAdjacentDuplicates(windowDraft.backwardHistoryNotes);
|
|
windowDraft.forwardHistoryNotes = removeAdjacentDuplicates(windowDraft.forwardHistoryNotes);
|
|
|
|
// Fix the case where after deletion the currently selected note is also the latest in history
|
|
const selectedNoteIds = windowDraft.selectedNoteIds;
|
|
if (selectedNoteIds.length && windowDraft.backwardHistoryNotes.length && windowDraft.backwardHistoryNotes[windowDraft.backwardHistoryNotes.length - 1].id === selectedNoteIds[0]) {
|
|
windowDraft.backwardHistoryNotes = windowDraft.backwardHistoryNotes.slice(0, windowDraft.backwardHistoryNotes.length - 1);
|
|
}
|
|
if (selectedNoteIds.length && windowDraft.forwardHistoryNotes.length && windowDraft.forwardHistoryNotes[windowDraft.forwardHistoryNotes.length - 1].id === selectedNoteIds[0]) {
|
|
windowDraft.forwardHistoryNotes = windowDraft.forwardHistoryNotes.slice(0, windowDraft.forwardHistoryNotes.length - 1);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
|
|
updateWindowHistory(draft);
|
|
for (const id in draft.backgroundWindows) {
|
|
updateWindowHistory(draft.backgroundWindows[id]);
|
|
}
|
|
}
|
|
|
|
type WindowAction = {
|
|
type: 'WINDOW_OPEN';
|
|
windowId: string;
|
|
folderId: string;
|
|
noteId: string;
|
|
defaultAppWindowState: Record<string, unknown>;
|
|
}|{
|
|
type: 'WINDOW_FOCUS'|'WINDOW_CLOSE';
|
|
windowId: string;
|
|
};
|
|
|
|
const handleWindowActions = (draft: Draft<State>, action: WindowAction) => {
|
|
switch (action.type) {
|
|
|
|
case 'WINDOW_OPEN': {
|
|
if (action.windowId in draft.backgroundWindows) {
|
|
throw new Error(`Window with id ${action.windowId} is already open!`);
|
|
}
|
|
|
|
draft.backgroundWindows[action.windowId] = {
|
|
...defaultWindowState,
|
|
...action.defaultAppWindowState,
|
|
|
|
lastSelectedNotesIds: {
|
|
...defaultWindowState.lastSelectedNotesIds,
|
|
Folder: {
|
|
[action.folderId]: [action.noteId],
|
|
},
|
|
},
|
|
notesParentType: 'Folder',
|
|
selectedFolderId: action.folderId,
|
|
windowId: action.windowId,
|
|
selectedNoteIds: [action.noteId],
|
|
};
|
|
break;
|
|
}
|
|
case 'WINDOW_FOCUS': {
|
|
// Only allow bringing a background window to the foreground
|
|
if (draft.windowId !== action.windowId) {
|
|
const windowId = action.windowId;
|
|
const previousWindowId = draft.windowId;
|
|
|
|
const focusingWindowState = draft.backgroundWindows[windowId];
|
|
const previousWindowState = { ...defaultWindowState };
|
|
|
|
for (const key of Object.keys(focusingWindowState)) {
|
|
const stateKey = key as keyof WindowState;
|
|
|
|
type AssignableWindowState = Record<keyof WindowState, unknown>;
|
|
(previousWindowState as AssignableWindowState)[stateKey] = draft[stateKey];
|
|
(draft as AssignableWindowState)[stateKey] = focusingWindowState[stateKey];
|
|
}
|
|
|
|
delete draft.backgroundWindows[windowId];
|
|
draft.backgroundWindows[previousWindowId] = previousWindowState;
|
|
}
|
|
break;
|
|
}
|
|
case 'WINDOW_CLOSE': {
|
|
delete draft.backgroundWindows[action.windowId];
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const reducer = produce((draft: Draft<State> = defaultState, action: any) => {
|
|
|
|
// const reducer = (state:State = defaultState, action:any) => {
|
|
// if (!['SIDE_MENU_OPEN_PERCENT'].includes(action.type)) console.info('Action', action.type, action);
|
|
|
|
// let newState = state;
|
|
|
|
// NOTE_DELETE requires post processing
|
|
if (action.type !== 'NOTE_DELETE') {
|
|
handleHistory(draft, action);
|
|
}
|
|
|
|
handleWindowActions(draft, action);
|
|
|
|
try {
|
|
switch (action.type) {
|
|
|
|
case 'NOTE_SELECT':
|
|
case 'NOTE_SELECT_ADD':
|
|
case 'NOTE_SELECT_REMOVE':
|
|
case 'NOTE_SELECT_TOGGLE':
|
|
changeSelectedNotes(draft, action);
|
|
break;
|
|
case 'NOTE_SELECT_EXTEND':
|
|
{
|
|
if (!draft.selectedNoteIds.length) {
|
|
draft.selectedNoteIds = [action.id];
|
|
} else {
|
|
const selectRangeId1 = draft.selectedNoteIds[draft.selectedNoteIds.length - 1];
|
|
const selectRangeId2 = action.id;
|
|
if (selectRangeId1 === selectRangeId2) {
|
|
// Nothing
|
|
} else {
|
|
const newSelectedNoteIds = draft.selectedNoteIds.slice();
|
|
let selectionStarted = false;
|
|
for (let i = 0; i < draft.notes.length; i++) {
|
|
const id = draft.notes[i].id;
|
|
|
|
if (!selectionStarted && (id === selectRangeId1 || id === selectRangeId2)) {
|
|
selectionStarted = true;
|
|
if (newSelectedNoteIds.indexOf(id) < 0) newSelectedNoteIds.push(id);
|
|
continue;
|
|
} else if (selectionStarted && (id === selectRangeId1 || id === selectRangeId2)) {
|
|
if (newSelectedNoteIds.indexOf(id) < 0) newSelectedNoteIds.push(id);
|
|
break;
|
|
}
|
|
|
|
if (selectionStarted && newSelectedNoteIds.indexOf(id) < 0) {
|
|
newSelectedNoteIds.push(id);
|
|
}
|
|
}
|
|
draft.selectedNoteIds = newSelectedNoteIds;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'NOTE_SELECT_ALL':
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
draft.selectedNoteIds = draft.notes.map((n: any) => n.id);
|
|
break;
|
|
|
|
case 'NOTE_SELECT_ALL_TOGGLE': {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const allSelected = draft.notes.every((n: any) => draft.selectedNoteIds.includes(n.id));
|
|
if (allSelected) {
|
|
draft.selectedNoteIds = [];
|
|
} else {
|
|
draft.selectedNoteIds = draft.notes.map(n => n.id);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 'SMART_FILTER_SELECT':
|
|
draft.notesParentType = 'SmartFilter';
|
|
draft.selectedSmartFilterId = action.id;
|
|
break;
|
|
|
|
case 'FOLDER_SELECT':
|
|
changeSelectedFolder(draft, action, { clearSelectedNoteIds: true });
|
|
break;
|
|
|
|
case 'FOLDER_AND_NOTE_SELECT':
|
|
{
|
|
changeSelectedFolder(draft, action);
|
|
const noteSelectAction = { ...action, type: 'NOTE_SELECT' };
|
|
changeSelectedNotes(draft, noteSelectAction);
|
|
}
|
|
break;
|
|
|
|
case 'SETTING_UPDATE_ALL':
|
|
draft.settings = action.settings;
|
|
break;
|
|
|
|
case 'SETTING_UPDATE_ONE':
|
|
{
|
|
const newSettings = { ...draft.settings };
|
|
newSettings[action.key] = action.value;
|
|
draft.settings = newSettings;
|
|
}
|
|
break;
|
|
|
|
case 'ITEMS_TRASHED':
|
|
|
|
draft.lastDeletion = {
|
|
...action.value,
|
|
timestamp: Date.now(),
|
|
};
|
|
break;
|
|
|
|
case 'DELETION_NOTIFICATION_DONE':
|
|
|
|
draft.lastDeletionNotificationTime = Date.now();
|
|
break;
|
|
|
|
case 'NOTE_PROVISIONAL_FLAG_CLEAR':
|
|
{
|
|
const newIds = ArrayUtils.removeElement(draft.provisionalNoteIds, action.id);
|
|
if (newIds !== draft.provisionalNoteIds) {
|
|
draft.provisionalNoteIds = newIds;
|
|
}
|
|
}
|
|
break;
|
|
|
|
// Replace all the notes with the provided array
|
|
case 'NOTE_UPDATE_ALL':
|
|
draft.notes = action.notes;
|
|
draft.notesSource = action.notesSource;
|
|
draft.noteListLastSortTime = Date.now(); // Notes are already sorted when they are set this way.
|
|
if (!draft.allowSelectionInOtherFolders) {
|
|
updateSelectedNotesFromExistingNotes(draft);
|
|
}
|
|
break;
|
|
|
|
// Insert the note into the note list if it's new, or
|
|
// update it within the note array if it already exists.
|
|
case 'NOTE_UPDATE_ONE':
|
|
{
|
|
const modNote: NoteEntity = action.note;
|
|
const handleWindowState = (windowDraft: Draft<WindowState>) => {
|
|
const isViewingAllNotes = (windowDraft.notesParentType === 'SmartFilter' && windowDraft.selectedSmartFilterId === ALL_NOTES_FILTER_ID);
|
|
const isViewingConflictFolder = windowDraft.notesParentType === 'Folder' && windowDraft.selectedFolderId === Folder.conflictFolderId();
|
|
|
|
const noteIsInFolder = function(note: NoteEntity, folderId: string) {
|
|
if (note.is_conflict && isViewingConflictFolder) return true;
|
|
const noteDisplayParentId = getDisplayParentId(note, draft.folders.find(f => f.id === note.parent_id));
|
|
return folderId === noteDisplayParentId;
|
|
};
|
|
|
|
let movedNotePreviousIndex = 0;
|
|
let noteFolderHasChanged = false;
|
|
const newNotes = windowDraft.notes.slice();
|
|
let found = false;
|
|
for (let i = 0; i < newNotes.length; i++) {
|
|
const n = newNotes[i];
|
|
if (n.id === modNote.id) {
|
|
const previousDisplayParentId = ('parent_id' in n) ? getDisplayParentId(n, draft.folders.find(f => f.id === n.parent_id)) : '';
|
|
if (n.is_conflict && !modNote.is_conflict) {
|
|
// Note was a conflict but was moved outside of
|
|
// the conflict folder
|
|
newNotes.splice(i, 1);
|
|
noteFolderHasChanged = true;
|
|
movedNotePreviousIndex = i;
|
|
} else if (isViewingAllNotes || noteIsInFolder(modNote, previousDisplayParentId)) {
|
|
// Note is still in the same folder
|
|
// Merge the properties that have changed (in modNote) into
|
|
// the object we already have.
|
|
newNotes[i] = { ...newNotes[i] };
|
|
|
|
for (const n in modNote) {
|
|
if (!modNote.hasOwnProperty(n)) continue;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(newNotes[i] as any)[n] = (modNote as any)[n];
|
|
}
|
|
} else {
|
|
// Note has moved to a different folder
|
|
newNotes.splice(i, 1);
|
|
noteFolderHasChanged = true;
|
|
movedNotePreviousIndex = i;
|
|
}
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Note was not found - if the current folder is the same as the note folder,
|
|
// add it to it.
|
|
if (!found) {
|
|
if (isViewingAllNotes || noteIsInFolder(modNote, windowDraft.selectedFolderId)) {
|
|
newNotes.push(modNote);
|
|
}
|
|
}
|
|
|
|
windowDraft.notes = newNotes;
|
|
|
|
// Ensure that the selected note is still in the current folder.
|
|
// For example, if the user drags the current note to a different folder,
|
|
// a new note should be selected.
|
|
// In some cases, however, the selection needs to be preserved (e.g. the mobile app).
|
|
const preserveSelection = action.preserveSelection ?? draft.allowSelectionInOtherFolders;
|
|
if (noteFolderHasChanged && !preserveSelection) {
|
|
let newIndex = movedNotePreviousIndex;
|
|
if (newIndex >= newNotes.length) newIndex = newNotes.length - 1;
|
|
if (!newNotes.length) newIndex = -1;
|
|
windowDraft.selectedNoteIds = newIndex >= 0 ? [newNotes[newIndex].id] : [];
|
|
}
|
|
|
|
if (!action.ignoreProvisionalFlag) {
|
|
let newProvisionalNoteIds = draft.provisionalNoteIds;
|
|
|
|
const idx = newProvisionalNoteIds.indexOf(modNote.id);
|
|
if (action.provisional) {
|
|
if (idx < 0) {
|
|
newProvisionalNoteIds = newProvisionalNoteIds.slice();
|
|
newProvisionalNoteIds.push(modNote.id);
|
|
}
|
|
} else if (idx >= 0) {
|
|
newProvisionalNoteIds = newProvisionalNoteIds.slice();
|
|
newProvisionalNoteIds.splice(idx, 1);
|
|
}
|
|
|
|
draft.provisionalNoteIds = newProvisionalNoteIds;
|
|
}
|
|
};
|
|
|
|
handleWindowState(draft);
|
|
for (const backgroundWindow of Object.values(draft.backgroundWindows)) {
|
|
handleWindowState(backgroundWindow);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'NOTE_DELETE':
|
|
|
|
{
|
|
handleItemDelete(draft, action);
|
|
|
|
const idx = draft.provisionalNoteIds.indexOf(action.id);
|
|
if (idx >= 0) {
|
|
const t = draft.provisionalNoteIds.slice();
|
|
t.splice(idx, 1);
|
|
draft.provisionalNoteIds = t;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'NOTE_SORT':
|
|
|
|
{
|
|
if (draft.notesParentType === 'Search') {
|
|
logger.debug('Not sorting the note list -- sorting should be done by search.');
|
|
} else {
|
|
draft.notes = Note.sortNotes(draft.notes, stateUtils.notesOrder(draft.settings), draft.settings.uncompletedTodosOnTop);
|
|
}
|
|
draft.noteListLastSortTime = Date.now();
|
|
}
|
|
break;
|
|
|
|
case 'NOTE_IS_INSERTING_NOTES':
|
|
|
|
if (draft.isInsertingNotes !== action.value) {
|
|
draft.isInsertingNotes = action.value;
|
|
}
|
|
break;
|
|
|
|
case 'TAG_DELETE':
|
|
handleItemDelete(draft, action);
|
|
draft.selectedNoteTags = removeItemFromArray(draft.selectedNoteTags, 'id', action.id);
|
|
break;
|
|
|
|
case 'FOLDER_UPDATE_ALL':
|
|
draft.folders = action.items;
|
|
break;
|
|
|
|
case 'FOLDER_SET_COLLAPSED':
|
|
folderSetCollapsed(draft, action);
|
|
break;
|
|
|
|
case 'FOLDER_TOGGLE':
|
|
if (draft.collapsedFolderIds.indexOf(action.id) >= 0) {
|
|
folderSetCollapsed(draft, { collapsed: false, ...action });
|
|
} else {
|
|
folderSetCollapsed(draft, { collapsed: true, ...action });
|
|
}
|
|
break;
|
|
|
|
case 'FOLDER_SET_COLLAPSED_ALL':
|
|
draft.collapsedFolderIds = action.ids.slice();
|
|
break;
|
|
|
|
case 'TAG_UPDATE_ALL':
|
|
if (!fastDeepEqual(original(draft.tags), action.items)) {
|
|
draft.tags = action.items;
|
|
}
|
|
break;
|
|
|
|
case 'TAG_SELECT':
|
|
|
|
if (draft.selectedTagId !== action.id || draft.notesParentType !== 'Tag') {
|
|
draft.selectedTagId = action.id;
|
|
if (!action.id) {
|
|
draft.notesParentType = defaultNotesParentType(draft, 'Tag');
|
|
} else {
|
|
draft.notesParentType = 'Tag';
|
|
}
|
|
draft.selectedNoteIds = [];
|
|
}
|
|
break;
|
|
|
|
case 'TAG_UPDATE_ONE':
|
|
{
|
|
updateOneItem(draft, action);
|
|
|
|
for (const windowStateDraft of stateUtils.allWindowStates(draft)) {
|
|
// We only want to update the selected note tags if the tag belongs to the currently open note
|
|
const selectedNoteHasTag = !!windowStateDraft.selectedNoteTags.find(tag => tag.id === action.item.id);
|
|
if (selectedNoteHasTag) {
|
|
updateOneItem(windowStateDraft, action, 'selectedNoteTags');
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'NOTE_TAG_REMOVE':
|
|
{
|
|
updateOneItem(draft, action, 'tags');
|
|
const tagRemoved = action.item;
|
|
for (const windowStateDraft of stateUtils.allWindowStates(draft)) {
|
|
windowStateDraft.selectedNoteTags = removeItemFromArray(windowStateDraft.selectedNoteTags, 'id', tagRemoved.id);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'EDITOR_NOTE_STATUS_SET':
|
|
|
|
{
|
|
draft.editorNoteStatuses[action.id] = action.status;
|
|
}
|
|
break;
|
|
|
|
case 'EDITOR_NOTE_STATUS_REMOVE':
|
|
|
|
{
|
|
delete draft.editorNoteStatuses[action.id];
|
|
}
|
|
break;
|
|
|
|
case 'FOLDER_UPDATE_ONE':
|
|
case 'MASTERKEY_UPDATE_ONE':
|
|
updateOneItem(draft, action);
|
|
break;
|
|
|
|
case 'FOLDER_DELETE':
|
|
handleItemDelete(draft, action);
|
|
break;
|
|
|
|
// case 'MASTERKEY_UPDATE_ALL':
|
|
// draft.masterKeys = action.items;
|
|
// break;
|
|
|
|
case 'MASTERKEY_SET_NOT_LOADED':
|
|
draft.notLoadedMasterKeys = action.ids;
|
|
break;
|
|
|
|
case 'MASTERKEY_ADD_NOT_LOADED':
|
|
{
|
|
if (draft.notLoadedMasterKeys.indexOf(action.id) < 0) {
|
|
const keys = draft.notLoadedMasterKeys.slice();
|
|
keys.push(action.id);
|
|
draft.notLoadedMasterKeys = keys;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'MASTERKEY_REMOVE_NOT_LOADED':
|
|
{
|
|
const ids = action.id ? [action.id] : action.ids;
|
|
for (let i = 0; i < ids.length; i++) {
|
|
const id = ids[i];
|
|
const index = draft.notLoadedMasterKeys.indexOf(id);
|
|
if (index >= 0) {
|
|
const keys = draft.notLoadedMasterKeys.slice();
|
|
keys.splice(index, 1);
|
|
draft.notLoadedMasterKeys = keys;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'SYNC_STARTED':
|
|
draft.syncStarted = true;
|
|
break;
|
|
|
|
case 'SYNC_COMPLETED':
|
|
draft.syncStarted = false;
|
|
break;
|
|
|
|
case 'SYNC_REPORT_UPDATE':
|
|
draft.syncReport = action.report;
|
|
break;
|
|
|
|
case 'SEARCH_QUERY':
|
|
draft.searchQuery = action.query.trim();
|
|
break;
|
|
|
|
case 'SEARCH_ADD':
|
|
{
|
|
const searches = draft.searches.slice();
|
|
searches.push(action.search);
|
|
draft.searches = searches;
|
|
}
|
|
break;
|
|
|
|
case 'SEARCH_UPDATE':
|
|
{
|
|
const searches = draft.searches.slice();
|
|
let found = false;
|
|
for (let i = 0; i < searches.length; i++) {
|
|
if (searches[i].id === action.search.id) {
|
|
searches[i] = { ...action.search };
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found) searches.push(action.search);
|
|
|
|
draft.notesParentType = 'Search';
|
|
draft.searches = searches;
|
|
}
|
|
break;
|
|
|
|
case 'SEARCH_DELETE':
|
|
handleItemDelete(draft, action);
|
|
break;
|
|
|
|
case 'SEARCH_SELECT':
|
|
draft.selectedSearchId = action.id;
|
|
if (!action.id) {
|
|
draft.notesParentType = defaultNotesParentType(draft, 'Search');
|
|
} else {
|
|
draft.notesParentType = 'Search';
|
|
}
|
|
draft.selectedNoteIds = [];
|
|
break;
|
|
case 'SET_HIGHLIGHTED':
|
|
draft.highlightedWords = action.words;
|
|
break;
|
|
|
|
case 'APP_STATE_SET':
|
|
draft.appState = action.state;
|
|
break;
|
|
|
|
case 'BIOMETRICS_DONE_SET':
|
|
draft.biometricsDone = action.value;
|
|
break;
|
|
|
|
case 'SYNC_HAS_DISABLED_SYNC_ITEMS':
|
|
draft.hasDisabledSyncItems = 'value' in action ? action.value : true;
|
|
break;
|
|
|
|
case 'ENCRYPTION_HAS_DISABLED_ITEMS':
|
|
draft.hasDisabledEncryptionItems = action.value;
|
|
break;
|
|
|
|
case 'CLIPPER_SERVER_SET':
|
|
{
|
|
const clipperServer = { ...draft.clipperServer };
|
|
if ('startState' in action) clipperServer.startState = action.startState;
|
|
if ('port' in action) clipperServer.port = action.port;
|
|
draft.clipperServer = clipperServer;
|
|
}
|
|
break;
|
|
|
|
case 'DECRYPTION_WORKER_SET':
|
|
{
|
|
const decryptionWorker = { ...draft.decryptionWorker };
|
|
for (const n in action) {
|
|
if (!action.hasOwnProperty(n) || n === 'type') continue;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(decryptionWorker as any)[n] = action[n];
|
|
}
|
|
draft.decryptionWorker = decryptionWorker;
|
|
}
|
|
break;
|
|
|
|
case 'RESOURCE_FETCHER_SET':
|
|
{
|
|
const rf = { ...action };
|
|
delete rf.type;
|
|
draft.resourceFetcher = rf;
|
|
}
|
|
break;
|
|
|
|
case 'CUSTOM_VIEWER_CSS_APPEND':
|
|
draft.customViewerCss += action.css;
|
|
break;
|
|
|
|
case 'CUSTOM_CHROME_CSS_ADD':
|
|
// To enable/disable custom CSS, some plugins add the same chrome CSS file multiple times.
|
|
// For performance, only apply the last copy of each file.
|
|
draft.customChromeCssPaths = draft.customChromeCssPaths.filter(path => path !== action.filePath);
|
|
draft.customChromeCssPaths.push(action.filePath);
|
|
break;
|
|
|
|
case 'SET_NOTE_TAGS':
|
|
if (!fastDeepEqual(original(draft.selectedNoteTags), action.items)) {
|
|
draft.selectedNoteTags = action.items;
|
|
}
|
|
break;
|
|
|
|
case 'PLUGINLEGACY_DIALOG_SET':
|
|
{
|
|
if (!action.pluginName) throw new Error('action.pluginName not specified');
|
|
const newPluginsLegacy = { ...draft.pluginsLegacy };
|
|
const newPlugin = draft.pluginsLegacy[action.pluginName] ? { ...draft.pluginsLegacy[action.pluginName] } : {};
|
|
if ('open' in action) newPlugin.dialogOpen = action.open;
|
|
if ('userData' in action) newPlugin.userData = action.userData;
|
|
newPluginsLegacy[action.pluginName] = newPlugin;
|
|
draft.pluginsLegacy = newPluginsLegacy;
|
|
}
|
|
break;
|
|
|
|
case 'API_NEED_AUTH_SET':
|
|
draft.needApiAuth = action.value;
|
|
break;
|
|
|
|
case 'PROFILE_CONFIG_SET':
|
|
draft.profileConfig = action.value;
|
|
break;
|
|
|
|
case 'MUST_UPGRADE_APP':
|
|
draft.mustUpgradeAppMessage = action.message;
|
|
break;
|
|
|
|
case 'MUST_AUTHENTICATE':
|
|
draft.mustAuthenticate = action.value;
|
|
break;
|
|
|
|
case 'NOTE_LIST_RENDERER_ADD':
|
|
{
|
|
const noteListRendererIds = draft.noteListRendererIds.slice();
|
|
if (noteListRendererIds.includes(action.value)) throw new Error(`Note list renderer is already registered: ${action.value}`);
|
|
noteListRendererIds.push(action.value);
|
|
draft.noteListRendererIds = noteListRendererIds;
|
|
}
|
|
break;
|
|
|
|
}
|
|
} catch (error) {
|
|
error.message = `In reducer: ${error.message} Action: ${JSON.stringify(action)}`;
|
|
throw error;
|
|
}
|
|
|
|
if (action.type.indexOf('NOTE_UPDATE') === 0 || action.type.indexOf('FOLDER_UPDATE') === 0 || action.type.indexOf('TAG_UPDATE') === 0) {
|
|
draft.hasEncryptedItems = stateHasEncryptedItems(draft);
|
|
}
|
|
|
|
if (action.type === 'NOTE_DELETE') {
|
|
handleHistory(draft, action);
|
|
}
|
|
|
|
if (action.type === 'SETTING_UPDATE_ALL' || (action.type === 'SETTING_UPDATE_ONE' && action.key === 'activeFolderId')) {
|
|
// To allow creating notes when opening the app with all notes and/or tags,
|
|
// a "last selected folder ID" needs to be set.
|
|
draft.selectedFolderId ??= draft.settings.activeFolderId;
|
|
}
|
|
|
|
for (const additionalReducer of additionalReducers) {
|
|
additionalReducer.reducer(draft, action);
|
|
}
|
|
|
|
// if (Setting.value('env') === 'dev') {
|
|
// return Object.freeze(newState);
|
|
// } else {
|
|
// return newState;
|
|
// }
|
|
});
|
|
|
|
export default reducer;
|