diff --git a/app/package.json b/app/package.json index 2e1bb853..9c7d8a31 100644 --- a/app/package.json +++ b/app/package.json @@ -1,6 +1,6 @@ { "name": "micropad", - "version": "3.29.1", + "version": "3.30.0", "private": true, "scripts": { "preinstall": "python3 ../libs/build-libs.py; ./get_precache_files.py > src/extraPrecacheFiles.ts", diff --git a/app/src/app/ReadOnly.ts b/app/src/app/ReadOnly.ts new file mode 100644 index 00000000..0dfe8d46 --- /dev/null +++ b/app/src/app/ReadOnly.ts @@ -0,0 +1,10 @@ +import { isDev } from './util'; + +export function isReadOnlyNotebook(title: string) { + // All notebooks are editable in dev mode + if (isDev()) { + return false; + } + + return title === 'Help'; +} diff --git a/app/src/app/actions.ts b/app/src/app/actions.ts index fb96e219..4b254033 100644 --- a/app/src/app/actions.ts +++ b/app/src/app/actions.ts @@ -62,12 +62,11 @@ export const actions = { imagePasted: actionCreator.async('IMAGE_PASTED'), exportAll: actionCreator.async('EXPORT_ALL_NOTEPADS'), exportToMarkdown: actionCreator.async('EXPORT_ALL_NOTEPADS_TO_MD'), - clearOldData: actionCreator.async('CLEAR_OLD_DATA'), + clearOldData: actionCreator.async<{ silent: boolean }, void, Error>('CLEAR_OLD_DATA'), getHelp: actionCreator.async('GET_HELP'), getDueDates: actionCreator.async('GET_DUE_DATES'), moveObjAcrossNotepads: actionCreator.async('CROSS_NOTEPAD_MOVE'), - started: actionCreator('APP_STARTED'), restoreJsonNotepad: actionCreator('PARSE_JSON_NOTEPAD'), restoreJsonNotepadAndLoadNote: actionCreator('PARSE_JSON_NOTEPAD_AND_LOAD_NOTE'), newNotepad: actionCreator('NEW_NOTEPAD'), @@ -120,3 +119,25 @@ export const actions = { closeNotepad: actionCreator('CLOSE_NOTEPAD'), importMarkdown: actionCreator('IMPORT_FROM_MARKDOWN') }; + +export const READ_ONLY_ACTIONS: ReadonlySet = new Set([ + actions.quickNote.started.type, + actions.quickNote.done.type, + actions.quickNote.failed.type, + + actions.imagePasted.started.type, + actions.imagePasted.done.type, + actions.imagePasted.failed.type, + + actions.updateElement.type, + actions.quickMarkdownInsert.type, + actions.insertElement.type, + actions.toggleInsertMenu.type, + actions.openEditor.type, + actions.renameNotepadObject.type, + actions.newSection.type, + actions.newNote.type, + actions.moveNotepadObject.type, + actions.deleteElement.type, + actions.deleteNotepadObject.type +]); diff --git a/app/src/app/assets/Help.npx b/app/src/app/assets/Help.npx index bb29a4d5..47494ca6 100644 --- a/app/src/app/assets/Help.npx +++ b/app/src/app/assets/Help.npx @@ -1,4 +1,4 @@ -
https://getmicropad.com/# What is µPad? +
https://getmicropad.com/# What is µPad? µPad, also known as *MicroPad*, is a note taking application that is built upon the principles of power and openness. @@ -14,6 +14,16 @@ of power and openness. - This may be bundled with Electron. The licence for Electron can be found [here](https://github.com/electron/electron/blob/master/LICENSE).
.]]>
\ No newline at end of file +*N.B. Binary asset handling changed as of µPad v2. More information coming soon.*.]]> diff --git a/app/src/app/components/explorer/NotepadExplorerComponent.tsx b/app/src/app/components/explorer/NotepadExplorerComponent.tsx index 0c263558..10c46165 100644 --- a/app/src/app/components/explorer/NotepadExplorerComponent.tsx +++ b/app/src/app/components/explorer/NotepadExplorerComponent.tsx @@ -54,7 +54,10 @@ export default class NotepadExplorerComponent extends React.Component - {notepad.title} + + {notepad.title} + {this.props.isReadOnly && (Read-Only)} + diff --git a/app/src/app/components/explorer/NotepadExplorerContainer.ts b/app/src/app/components/explorer/NotepadExplorerContainer.ts index eec82cc6..856b9b0b 100644 --- a/app/src/app/components/explorer/NotepadExplorerContainer.ts +++ b/app/src/app/components/explorer/NotepadExplorerContainer.ts @@ -21,7 +21,8 @@ export function mapStateToProps({ notepads, explorer, app, currentNote }: IStore openSections: explorer.openSections, isFullScreen: app.isFullScreen, openNote: note, - theme: ThemeValues[app.theme] + theme: ThemeValues[app.theme], + isReadOnly: !!notepads?.notepad?.isReadOnly }; } diff --git a/app/src/app/components/explorer/app-settings/AppSettingsContainer.ts b/app/src/app/components/explorer/app-settings/AppSettingsContainer.ts index ef97d2a4..af133964 100644 --- a/app/src/app/components/explorer/app-settings/AppSettingsContainer.ts +++ b/app/src/app/components/explorer/app-settings/AppSettingsContainer.ts @@ -3,7 +3,7 @@ import { actions } from '../../../actions'; import AppSettingsComponent from './AppSettingsComponent'; export const appSettingsContainer = connect(() => ({}), dispatch => ({ - clearOldData: () => dispatch(actions.clearOldData.started()) + clearOldData: () => dispatch(actions.clearOldData.started({ silent: false })) })); export default appSettingsContainer(AppSettingsComponent); diff --git a/app/src/app/components/explorer/explorer-options/ExplorerOptionsComponent.tsx b/app/src/app/components/explorer/explorer-options/ExplorerOptionsComponent.tsx index 4e7f5e64..72165c5d 100644 --- a/app/src/app/components/explorer/explorer-options/ExplorerOptionsComponent.tsx +++ b/app/src/app/components/explorer/explorer-options/ExplorerOptionsComponent.tsx @@ -1,4 +1,5 @@ import * as React from 'react'; +import { FormEvent } from 'react'; import { Button, Col, Icon, Input, Modal, Row } from 'react-materialize'; import { Notepad } from 'upad-parse/dist'; import { NPXObject } from 'upad-parse/dist/NPXObject'; @@ -88,7 +89,8 @@ export default class ExplorerOptionsComponent extends React.Component { ); } - private rename = () => { + private rename = (e: FormEvent) => { + e.preventDefault(); const { objToEdit, type, renameNotepad, renameNotepadObject } = this.props; const value = this.titleInput.state.value; @@ -107,6 +109,8 @@ export default class ExplorerOptionsComponent extends React.Component { default: break; } + + return false; } private delete = async () => { diff --git a/app/src/app/components/header/NotepadDropdownComponent.tsx b/app/src/app/components/header/NotepadDropdownComponent.tsx index 81463179..0d0fba68 100644 --- a/app/src/app/components/header/NotepadDropdownComponent.tsx +++ b/app/src/app/components/header/NotepadDropdownComponent.tsx @@ -97,6 +97,7 @@ export default class NotepadDropdownComponent extends React.Component { const title = await Dialog.prompt('Notebook/Notepad Title:'); + if (!title) return; let notepad = new FlatNotepad(title); let section = FlatNotepad.makeFlatSection('Unorganised Notes'); diff --git a/app/src/app/components/note-viewer/NoteViewerComponent.tsx b/app/src/app/components/note-viewer/NoteViewerComponent.tsx index 4171bf19..644e6946 100644 --- a/app/src/app/components/note-viewer/NoteViewerComponent.tsx +++ b/app/src/app/components/note-viewer/NoteViewerComponent.tsx @@ -29,6 +29,7 @@ export interface INoteViewerComponentProps { downloadAsset?: (filename: string, uuid: string) => void; updateElement?: (id: string, changes: NoteElement, newAsset?: Blob) => void; toggleInsertMenu?: (opts: Partial) => void; + hideInsert?: () => void; insert?: (element: NoteElement) => void; deleteElement?: (id: string) => void; deleteNotepad?: () => void; @@ -141,14 +142,14 @@ export default class NoteViewerComponent extends React.Component edit!('')); setInterval(() => { if (this.scrolling) { this.scrolling = false; - toggleInsertMenu!({ enabled: false }); + hideInsert?.(); } }, 250); } diff --git a/app/src/app/containers/NoteViewerContainer.ts b/app/src/app/containers/NoteViewerContainer.ts index 68b1dcc5..a5dbf9a2 100644 --- a/app/src/app/containers/NoteViewerContainer.ts +++ b/app/src/app/containers/NoteViewerContainer.ts @@ -12,9 +12,11 @@ import { Note } from 'upad-parse/dist'; let noteRef: string = ''; let note: Note | null; let notepadTitle: string = ''; +let isInsertMenuOpen: boolean = false; export function mapStateToProps({ notepads, currentNote, app }: IStoreState) { noteRef = currentNote.ref; + isInsertMenuOpen = currentNote.insertElement.enabled; if (currentNote.ref.length !== 0) { note = notepads.notepad!.item!.notes[currentNote.ref]; @@ -62,7 +64,11 @@ export function mapDispatchToProps(dispatch: Dispatch): Partial dispatch(actions.quickNotepad(undefined)), makeQuickNote: () => dispatch(actions.quickNote.started(undefined)), - deleteNotepad: () => dispatch(actions.deleteNotepad(notepadTitle)) + deleteNotepad: () => dispatch(actions.deleteNotepad(notepadTitle)), + hideInsert: () => { + if (!isInsertMenuOpen) return; + return dispatch(actions.toggleInsertMenu({ enabled: false })); + } }; } diff --git a/app/src/app/epics/HelpEpics.ts b/app/src/app/epics/HelpEpics.ts index e71fd340..72f682d0 100644 --- a/app/src/app/epics/HelpEpics.ts +++ b/app/src/app/epics/HelpEpics.ts @@ -7,7 +7,9 @@ import { filterTruthy } from '../util'; import { actions, MicroPadAction } from '../actions'; import { Dialog } from '../services/dialogs'; -export const getHelp$ = (action$: Observable, store: EpicStore) => +const HELP_READONLY_DATE = new Date('2021-06-18T14:34:30.958+12:00'); + +export const getHelp$ = (action$: Observable, store: EpicStore, { getStorage }: EpicDeps) => action$.pipe( ofType(actions.getHelp.started.type), concatMap(() => @@ -15,7 +17,15 @@ export const getHelp$ = (action$: Observable, store: EpicStore) const notepadList = store.getState().notepads.savedNotepadTitles; if (!notepadList || !notepadList.includes('Help')) return true; - return Dialog.confirm(`You have already imported the Help notebook. It can be accessed from the notebooks dropdown. If you continue you will lose any changes made to the notebook.`); + const helpLastModified: string | null = await getStorage().notepadStorage.getItem('Help') + .then(np => np ? JSON.parse(np).lastModified : null) + .catch(err => { console.error(err); return null; }); + + if (!helpLastModified || new Date(helpLastModified).getTime() < HELP_READONLY_DATE.getTime()) { + return Dialog.confirm(`You have already imported the Help notebook. It can be accessed from the notebooks dropdown. If you continue you will lose any changes made to the notebook.`); + } + + return true; })()) ), filterTruthy(), diff --git a/app/src/app/epics/NoteEpics.ts b/app/src/app/epics/NoteEpics.ts index 14f1711c..e80d1353 100644 --- a/app/src/app/epics/NoteEpics.ts +++ b/app/src/app/epics/NoteEpics.ts @@ -31,8 +31,8 @@ const loadNote$ = (action$: Observable, store: EpicStore) => ]; } - const error = new Error(`MicroPad couldn't load the current note`); - console.warn(error); + const error = new Error(`MicroPad couldn't load the current note (handled in loadNote$)`); + console.error(error); return [actions.loadNote.failed({ params: ref, error })]; }) ); diff --git a/app/src/app/epics/NotepadEpics.ts b/app/src/app/epics/NotepadEpics.ts index 81a0ed76..6058c839 100644 --- a/app/src/app/epics/NotepadEpics.ts +++ b/app/src/app/epics/NotepadEpics.ts @@ -1,4 +1,4 @@ -import { actions, MicroPadAction } from '../actions'; +import { actions, MicroPadAction, READ_ONLY_ACTIONS } from '../actions'; import { catchError, combineLatest, @@ -43,8 +43,9 @@ import { Dispatch } from 'redux'; import { format } from 'date-fns'; import { NotepadShell } from 'upad-parse/dist/interfaces'; import { fromShell } from '../services/CryptoService'; -import { ASSET_STORAGE, NOTEPAD_STORAGE } from '../root'; +import { ASSET_STORAGE, NOTEPAD_STORAGE, store as STORE } from '../root'; import { EpicDeps, EpicStore } from './index'; +import * as Materialize from 'materialize-css/dist/js/materialize'; const parseQueue: string[] = []; @@ -519,6 +520,27 @@ const moveObjAcrossNotepadsFailure$ = (actions$: Observable) => noEmit() ); +const warnOnReadOnlyEdit$ = (actions$: Observable, store: EpicStore, { getToastEventHandler }: EpicDeps) => + actions$.pipe( + filter(() => !!store.getState().notepads.notepad?.isReadOnly), + filter(action => READ_ONLY_ACTIONS.has(action.type)), + tap(() => { + Materialize.Toast.removeAll(); + const guid = getToastEventHandler().register(async () => { + const newTitle = await Dialog.prompt('New Title:'); + if (!newTitle) return; + + STORE.dispatch(actions.renameNotepad.started(newTitle)); + }) + + Materialize.toast(`This notepad is read-only. Changes will not be saved.
` + + `Please create a notebook or open another one using the notebooks dropdown if you want to edit a notebook.
` + + `If you have made changes to this notebook, you can make it editable by renaming it.
` + + `RENAME`, 10_000); + }), + noEmit() + ) + export const notepadEpics$ = combineEpics( parseNpx$, syncOnNotepadParsed$ as any, @@ -543,7 +565,8 @@ export const notepadEpics$ = combineEpics( quickNotepad$, autoFillNewNotepads$, moveObjAcrossNotepads$, - moveObjAcrossNotepadsFailure$ + moveObjAcrossNotepadsFailure$, + warnOnReadOnlyEdit$ ); interface IExportedNotepad { diff --git a/app/src/app/epics/StorageEpics.ts b/app/src/app/epics/StorageEpics.ts index c76115f3..73b27eff 100644 --- a/app/src/app/epics/StorageEpics.ts +++ b/app/src/app/epics/StorageEpics.ts @@ -4,6 +4,7 @@ import { concatMap, debounceTime, distinctUntilChanged, + distinctUntilKeyChanged, filter, map, mergeMap, @@ -27,6 +28,7 @@ import { NotepadShell } from 'upad-parse/dist/interfaces'; import { ASSET_STORAGE, NOTEPAD_STORAGE } from '../root'; import { ICurrentNoteState } from '../reducers/NoteReducer'; import { EpicDeps, EpicStore } from './index'; +import { isReadOnlyNotebook } from '../ReadOnly'; let currentNotepadTitle = ''; @@ -55,13 +57,14 @@ const saveOnChanges$ = (action$: Observable, store: EpicStore) = map((notepadState: INotepadStoreState) => notepadState.item), filterTruthy(), debounceTime(1000), - distinctUntilChanged(), + distinctUntilKeyChanged('lastModified'), filter((notepad: FlatNotepad) => { const condition = notepad.title === currentNotepadTitle; currentNotepadTitle = notepad.title; return condition; }), + filter(notepad => !isReadOnlyNotebook(notepad.title)), map((notepad: FlatNotepad) => notepad.toNotepad()), mergeMap((notepad: Notepad) => { const actionsToReturn: Action[] = []; @@ -157,7 +160,7 @@ const deleteNotepad$ = (action$: Observable) => ofType>(actions.deleteNotepad.type), map((action: Action) => action.payload), tap((notepadTitle: string) => from(NOTEPAD_STORAGE.removeItem(notepadTitle))), - noEmit() + map(() => actions.clearOldData.started({ silent: true })) ); export type LastOpenedNotepad = { notepadTitle: string, noteRef?: string }; @@ -229,17 +232,17 @@ const clearLastOpenedNotepad$ = (action$: Observable) => const clearOldData$ = (action$: Observable, store: EpicStore) => action$.pipe( - ofType(actions.clearOldData.started.type), - concatMap(() => - from(cleanHangingAssets(NOTEPAD_STORAGE, ASSET_STORAGE, store.getState())).pipe( + ofType>(actions.clearOldData.started.type), + concatMap(action => + from(cleanHangingAssets(NOTEPAD_STORAGE, ASSET_STORAGE, store.getState(), action.payload.silent)).pipe( mergeMap((addPasskeyActions: Action[]) => [ - actions.clearOldData.done({ params: undefined, result: undefined }), + actions.clearOldData.done({ params: action.payload, result: undefined }), ...addPasskeyActions ]), catchError(error => { Dialog.alert('There was an error clearing old data'); console.error(error); - return of(actions.clearOldData.failed({ params: undefined, error })); + return of(actions.clearOldData.failed({ params: action.payload, error })); }) ) ) @@ -247,8 +250,8 @@ const clearOldData$ = (action$: Observable, store: EpicStore) => const notifyOnClearOldDataSuccess$ = (action$: Observable>>) => action$.pipe( - ofType(actions.clearOldData.done.type), - tap(() => Dialog.alert('The spring cleaning has been done!')), + ofType>>(actions.clearOldData.done.type), + tap(action => !action.payload.params.silent && Dialog.alert('The spring cleaning has been done!')), noEmit() ); @@ -271,13 +274,18 @@ export const storageEpics$ = combineEpics( /** * Clean up all the assets that aren't in any notepads yet */ -async function cleanHangingAssets(notepadStorage: LocalForage, assetStorage: LocalForage, state: IStoreState): Promise[]> { +async function cleanHangingAssets(notepadStorage: LocalForage, assetStorage: LocalForage, state: IStoreState, silent): Promise[]> { const cryptoPasskeys: Action[] = []; - const notepads: Promise[] = []; + const notepads: Promise[] = []; await notepadStorage.iterate((json: string) => { const shell: NotepadShell = JSON.parse(json); - notepads.push(fromShell(shell, state.notepadPasskeys[shell.title])); + let passkey = state.notepadPasskeys[shell.title]; + if (!passkey && silent) { + passkey = ''; + } + + notepads.push(fromShell(shell, passkey).catch(err => err)); return; }); @@ -290,7 +298,7 @@ async function cleanHangingAssets(notepadStorage: LocalForage, assetStorage: Loc const areNotepadsStillEncrypted = !!resolvedNotepadsOrErrors.find(res => res instanceof Error); - const resolvedNotepads = resolvedNotepadsOrErrors.filter(res => !(res instanceof Error)).map((cryptoInfo: EncryptNotepadAction) => { + const resolvedNotepads = resolvedNotepadsOrErrors.filter((res): res is EncryptNotepadAction => !(res instanceof Error)).map((cryptoInfo: EncryptNotepadAction) => { cryptoPasskeys.push(actions.addCryptoPasskey({ notepadTitle: cryptoInfo.notepad.title, passkey: cryptoInfo.passkey })); return cryptoInfo.notepad; }); diff --git a/app/src/app/epics/SyncEpics.ts b/app/src/app/epics/SyncEpics.ts index faa80a02..669462ad 100644 --- a/app/src/app/epics/SyncEpics.ts +++ b/app/src/app/epics/SyncEpics.ts @@ -15,7 +15,7 @@ import { } from 'rxjs/operators'; import { Action, Success } from 'redux-typescript-actions'; import { AssetList, ISyncedNotepad, SyncLoginRequest, SyncUser } from '../types/SyncTypes'; -import { ASSET_STORAGE, store as STORE, SYNC_STORAGE, TOAST_HANDLER } from '../root'; +import { ASSET_STORAGE, store as STORE, SYNC_STORAGE } from '../root'; import * as DifferenceEngine from '../services/DifferenceEngine'; import { Dialog } from '../services/dialogs'; import { IStoreState, SYNC_NAME } from '../types'; @@ -103,11 +103,11 @@ export const sync$ = (action$: Observable) => filterTruthy() ); -export const requestDownload$ = (action$: Observable) => +export const requestDownload$ = (action$: Observable, _, { getToastEventHandler }: EpicDeps) => action$.pipe( ofType>(actions.requestSyncDownload.type), tap((action: Action) => { - const guid = TOAST_HANDLER.register(() => STORE.dispatch(actions.syncDownload.started(action.payload))); + const guid = getToastEventHandler().register(() => STORE.dispatch(actions.syncDownload.started(action.payload))); Materialize.toast(`A newer copy of your notepad is online DOWNLOAD`); }), noEmit() diff --git a/app/src/app/epics/index.ts b/app/src/app/epics/index.ts index b6db00ff..86e1b21c 100644 --- a/app/src/app/epics/index.ts +++ b/app/src/app/epics/index.ts @@ -9,7 +9,7 @@ import { noteEpics$ } from './NoteEpics'; import { appEpics$ } from './AppEpics'; import { Action } from 'redux-typescript-actions'; import { cryptoEpics$ } from './CryptoEpics'; -import { getStorage, StorageMap } from '../root'; +import { getStorage, StorageMap, TOAST_HANDLER } from '../root'; import { printEpics$ } from './PrintEpics'; import { helpEpics$ } from './HelpEpics'; import { searchEpics$ } from './SearchEpics'; @@ -18,6 +18,7 @@ import { explorerEpics$ } from './ExplorerEpics'; import { dueDatesEpics$ } from './DueDatesEpics'; import { Dispatch, MiddlewareAPI } from 'redux'; import { IStoreState } from '../types'; +import ToastEventHandler from '../services/ToastEventHandler'; const baseEpic$ = combineEpics( notepadEpics$, @@ -35,13 +36,17 @@ const baseEpic$ = combineEpics( export type EpicDeps = { helpNpx: string, - getStorage: () => StorageMap + getStorage: () => StorageMap, + now: () => Date, + getToastEventHandler: () => ToastEventHandler }; export const epicMiddleware = createEpicMiddleware, any, EpicDeps>(baseEpic$, { dependencies: { helpNpx, - getStorage: getStorage + getStorage: getStorage, + now: () => new Date(), + getToastEventHandler: () => TOAST_HANDLER } }); diff --git a/app/src/app/reducers/BaseReducer.ts b/app/src/app/reducers/BaseReducer.ts index 9d8f0992..38f5ed0a 100644 --- a/app/src/app/reducers/BaseReducer.ts +++ b/app/src/app/reducers/BaseReducer.ts @@ -11,8 +11,9 @@ import { SyncReducer } from './SyncReducer'; import { AppReducer } from './AppReducer'; import { IsExportingReducer } from './IsExportingReducer'; import { NotepadPasskeysReducer } from './NotepadPasskeysReducer'; -import { MicroPadAction } from '../actions'; +import { MicroPadAction, READ_ONLY_ACTIONS } from '../actions'; import { Action, Reducer } from 'redux'; +import { isReadOnlyNotebook } from '../ReadOnly'; export const REDUCERS: Array> = [ new AppReducer(), @@ -42,14 +43,27 @@ export class BaseReducer implements ReduxReducer { public reducer(state: IStoreState | undefined, action: MicroPadAction): IStoreState { if (!state) { - return this.initialState; + state = this.initialState; + } + + if (BaseReducer.isReadonlyViolation(state, action)) { + // Skip any state updates if we're in a readonly notebook + return state; } let newState = { ...state }; - REDUCERS.forEach(reducer => newState[reducer.key] = reducer.reducer(state[reducer.key], action)); + REDUCERS.forEach(reducer => newState[reducer.key] = reducer.reducer(state![reducer.key], action)); return isDev() ? deepFreeze(newState) : newState; } + + private static isReadonlyViolation(state: IStoreState, action: MicroPadAction): boolean { + if (!isReadOnlyNotebook(state.notepads?.notepad?.item?.title ?? '')) { + return false; + } + + return READ_ONLY_ACTIONS.has(action.type); + } } diff --git a/app/src/app/reducers/NotepadsReducer.ts b/app/src/app/reducers/NotepadsReducer.ts index e083044e..389105b4 100644 --- a/app/src/app/reducers/NotepadsReducer.ts +++ b/app/src/app/reducers/NotepadsReducer.ts @@ -8,6 +8,7 @@ import { FlatSection } from 'upad-parse/dist/FlatNotepad'; import { FlatNotepad, Note } from 'upad-parse/dist'; import { format } from 'date-fns'; import { DueItem } from '../services/DueDates'; +import { isReadOnlyNotebook } from '../ReadOnly'; export class NotepadsReducer extends MicroPadReducer { public readonly key = 'notepads'; @@ -17,6 +18,15 @@ export class NotepadsReducer extends MicroPadReducer { }; public reducer(state: INotepadsStoreState, action: Action): INotepadsStoreState { + const newState = this.reducerImpl(state, action); + if (newState.notepad && newState.notepad.item) { + newState.notepad.isReadOnly = isReadOnlyNotebook(newState.notepad?.item?.title ?? ''); + } + + return newState; + } + + private reducerImpl(state: INotepadsStoreState, action: Action): INotepadsStoreState { if (isType(action, actions.parseNpx.done)) { const result = action.payload.result; @@ -29,6 +39,7 @@ export class NotepadsReducer extends MicroPadReducer { notepad: { isLoading: false, saving: false, + isReadOnly: isReadOnlyNotebook(result.title), item: result } }; @@ -83,6 +94,7 @@ export class NotepadsReducer extends MicroPadReducer { notepad: { isLoading: false, saving: false, + isReadOnly: isReadOnlyNotebook(notepad.title), item: notepad } }; diff --git a/app/src/app/root.tsx b/app/src/app/root.tsx index 0b4365bf..f1b4296d 100644 --- a/app/src/app/root.tsx +++ b/app/src/app/root.tsx @@ -32,13 +32,13 @@ import * as PasteImage from 'paste-image'; import PrintViewOrAppContainerComponent from './containers/PrintViewContainer'; import WhatsNewModalComponent from './components/WhatsNewModalComponent'; import { SyncUser } from './types/SyncTypes'; -import { INotepadStoreState } from './types/NotepadTypes'; import { SyncProErrorComponent } from './components/sync/SyncProErrorComponent'; import InsertElementComponent from './containers/InsertElementContainer'; import { ThemeName } from './types/Themes'; import AppBodyComponent from './containers/AppBodyContainer'; import ToastEventHandler from './services/ToastEventHandler'; import { LastOpenedNotepad } from './epics/StorageEpics'; +import { noop } from './util'; try { document.domain = MICROPAD_URL.split('//')[1]; @@ -115,8 +115,8 @@ export function getStorage(): StorageMap { document.getElementById('app') as HTMLElement ); - if (!await localforage.getItem('hasRunBefore')) store.dispatch(actions.getHelp.started()); - await localforage.setItem('hasRunBefore', true); + // Some clean up of an old storage item, this line can be deleted at some point in the future + localforage.removeItem('hasRunBefore').then(noop); await displayWhatsNew(); @@ -124,14 +124,14 @@ export function getStorage(): StorageMap { pasteWatcher(); + store.dispatch(actions.clearOldData.started({ silent: true })); + // Show a warning when closing before notepad save or sync is complete store.subscribe(() => { - const isSaving = store.getState().notepads.isLoading || (store.getState().notepads.notepad || {} as INotepadStoreState).isLoading; + const isSaving = store.getState().notepads.isLoading || store.getState().notepads.notepad?.isLoading; const isSyncing = store.getState().sync.isLoading; window.onbeforeunload = (isSyncing || isSaving) ? () => true : null; }); - - store.dispatch(actions.started()); })(); async function hydrateStoreFromLocalforage() { diff --git a/app/src/app/services/CryptoService.ts b/app/src/app/services/CryptoService.ts index 8705f515..8d4eeb1b 100644 --- a/app/src/app/services/CryptoService.ts +++ b/app/src/app/services/CryptoService.ts @@ -9,7 +9,7 @@ export async function fromShell(shell: NotepadShell, key?: string): Promise { delete this.handlers[guid]; }, 5 * 60 * 1000) + return guid; } } diff --git a/app/src/app/services/dialogs.ts b/app/src/app/services/dialogs.ts index a99f7184..2b8b6012 100644 --- a/app/src/app/services/dialogs.ts +++ b/app/src/app/services/dialogs.ts @@ -23,7 +23,7 @@ export class Dialog { }) ) - public static prompt = (message: string, placeholder?: string): Promise => + public static prompt = (message: string, placeholder?: string): Promise => new Promise(resolve => { setTimeout(() => { Vex.dialog.prompt({ @@ -34,7 +34,7 @@ export class Dialog { }, 0); }) - public static promptSecure = (message: string): Promise => + public static promptSecure = (message: string): Promise => new Promise(resolve => { setTimeout(() => { Vex.dialog.open({ diff --git a/app/src/app/types/NotepadTypes.ts b/app/src/app/types/NotepadTypes.ts index e307da5d..0748795e 100644 --- a/app/src/app/types/NotepadTypes.ts +++ b/app/src/app/types/NotepadTypes.ts @@ -16,6 +16,7 @@ export interface INotepadStoreState { saving: boolean; activeSyncId?: string; scribe?: string; + isReadOnly?: boolean; item?: FlatNotepad; } diff --git a/app/src/app/util.ts b/app/src/app/util.ts index 55f193b2..a7325306 100644 --- a/app/src/app/util.ts +++ b/app/src/app/util.ts @@ -137,3 +137,5 @@ export function debounce(callback: (...args: any[]) => void, time: number) { export function unreachable() { return new Error('Unreachable Error!'); } + +export function noop() {}