import * as React from 'react'; import ResizableLayout from '../ResizableLayout/ResizableLayout'; import findItemByKey from '../ResizableLayout/utils/findItemByKey'; import { MoveButtonClickEvent } from '../ResizableLayout/MoveButtons'; import { move } from '../ResizableLayout/utils/movements'; import { LayoutItem } from '../ResizableLayout/utils/types'; import NoteEditor from '../NoteEditor/NoteEditor'; import NoteContentPropertiesDialog from '../NoteContentPropertiesDialog'; import ShareNoteDialog from '../ShareNoteDialog'; import CommandService from '@joplin/lib/services/CommandService'; import { PluginHtmlContents, PluginStates, utils as pluginUtils } from '@joplin/lib/services/plugins/reducer'; import Sidebar from '../Sidebar/Sidebar'; import UserWebview from '../../services/plugins/UserWebview'; import UserWebviewDialog from '../../services/plugins/UserWebviewDialog'; import { ContainerType } from '@joplin/lib/services/plugins/WebviewController'; import { StateLastDeletion, stateUtils } from '@joplin/lib/reducer'; import InteropServiceHelper from '../../InteropServiceHelper'; import { _ } from '@joplin/lib/locale'; import NoteListWrapper from '../NoteListWrapper/NoteListWrapper'; import { AppState } from '../../app.reducer'; import { saveLayout, loadLayout } from '../ResizableLayout/utils/persist'; import Setting from '@joplin/lib/models/Setting'; import shouldShowMissingPasswordWarning from '@joplin/lib/components/shared/config/shouldShowMissingPasswordWarning'; import produce from 'immer'; import shim from '@joplin/lib/shim'; import bridge from '../../services/bridge'; import time from '@joplin/lib/time'; import styled from 'styled-components'; import { themeStyle, ThemeStyle } from '@joplin/lib/theme'; import validateLayout from '../ResizableLayout/utils/validateLayout'; import iterateItems from '../ResizableLayout/utils/iterateItems'; import removeItem from '../ResizableLayout/utils/removeItem'; import EncryptionService from '@joplin/lib/services/e2ee/EncryptionService'; import ShareFolderDialog from '../ShareFolderDialog/ShareFolderDialog'; import { ShareInvitation } from '@joplin/lib/services/share/reducer'; import removeKeylessItems from '../ResizableLayout/utils/removeKeylessItems'; import { localSyncInfoFromState } from '@joplin/lib/services/synchronizer/syncInfoUtils'; import { isCallbackUrl, parseCallbackUrl } from '@joplin/lib/callbackUrlUtils'; import ElectronAppWrapper from '../../ElectronAppWrapper'; import { showMissingMasterKeyMessage } from '@joplin/lib/services/e2ee/utils'; import { MasterKeyEntity } from '@joplin/lib/services/e2ee/types'; import commands from './commands/index'; import invitationRespond from '@joplin/lib/services/share/invitationRespond'; import restart from '../../services/restart'; const { connect } = require('react-redux'); import PromptDialog from '../PromptDialog'; import NotePropertiesDialog from '../NotePropertiesDialog'; import { NoteListColumns } from '@joplin/lib/services/plugins/api/noteListType'; import validateColumns from '../NoteListHeader/utils/validateColumns'; import TrashNotification from '../TrashNotification/TrashNotification'; import UpdateNotification from '../UpdateNotification/UpdateNotification'; const PluginManager = require('@joplin/lib/services/PluginManager'); const ipcRenderer = require('electron').ipcRenderer; interface LayerModalState { visible: boolean; message: string; } interface Props { plugins: PluginStates; pluginHtmlContents: PluginHtmlContents; pluginsLoaded: boolean; hasNotesBeingSaved: boolean; // eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied dispatch: Function; mainLayout: LayoutItem; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied style: any; layoutMoveMode: boolean; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied editorNoteStatuses: any; customCss: string; shouldUpgradeSyncTarget: boolean; hasDisabledSyncItems: boolean; hasDisabledEncryptionItems: boolean; hasMissingSyncCredentials: boolean; showMissingMasterKeyMessage: boolean; showNeedUpgradingMasterKeyMessage: boolean; showShouldReencryptMessage: boolean; themeId: number; settingEditorCodeView: boolean; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied pluginsLegacy: any; startupPluginsLoaded: boolean; shareInvitations: ShareInvitation[]; isSafeMode: boolean; enableLegacyMarkdownEditor: boolean; needApiAuth: boolean; processingShareInvitationResponse: boolean; isResettingLayout: boolean; listRendererId: string; lastDeletion: StateLastDeletion; lastDeletionNotificationTime: number; selectedFolderId: string; mustUpgradeAppMessage: string; notesSortOrderField: string; notesSortOrderReverse: boolean; notesColumns: NoteListColumns; showInvalidJoplinCloudCredential: boolean; } interface ShareFolderDialogOptions { folderId: string; visible: boolean; } interface State { // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied promptOptions: any; modalLayer: LayerModalState; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied notePropertiesDialogOptions: any; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied noteContentPropertiesDialogOptions: any; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied shareNoteDialogOptions: any; shareFolderDialogOptions: ShareFolderDialogOptions; } const StyledUserWebviewDialogContainer = styled.div` display: flex; position: absolute; top: 0; left: 0; width: 100%; height: 100%; z-index: 1000; box-sizing: border-box; `; const defaultLayout: LayoutItem = { key: 'root', children: [ { key: 'sideBar', width: 250 }, { key: 'noteList', width: 250 }, { key: 'editor' }, ], }; class MainScreenComponent extends React.Component<Props, State> { // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied private waitForNotesSavedIID_: any; private isPrinting_: boolean; private styleKey_: string; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied private styles_: any; // eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied private promptOnClose_: Function; public constructor(props: Props) { super(props); this.state = { promptOptions: null, modalLayer: { visible: false, message: '', }, notePropertiesDialogOptions: {}, noteContentPropertiesDialogOptions: {}, shareNoteDialogOptions: {}, shareFolderDialogOptions: { visible: false, folderId: '', }, }; this.updateMainLayout(this.buildLayout(props.plugins)); this.registerCommands(); this.setupAppCloseHandling(); this.notePropertiesDialog_close = this.notePropertiesDialog_close.bind(this); this.noteContentPropertiesDialog_close = this.noteContentPropertiesDialog_close.bind(this); this.shareNoteDialog_close = this.shareNoteDialog_close.bind(this); this.shareFolderDialog_close = this.shareFolderDialog_close.bind(this); this.resizableLayout_resize = this.resizableLayout_resize.bind(this); this.resizableLayout_renderItem = this.resizableLayout_renderItem.bind(this); this.resizableLayout_moveButtonClick = this.resizableLayout_moveButtonClick.bind(this); this.window_resize = this.window_resize.bind(this); this.rowHeight = this.rowHeight.bind(this); this.layoutModeListenerKeyDown = this.layoutModeListenerKeyDown.bind(this); window.addEventListener('resize', this.window_resize); // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied ipcRenderer.on('asynchronous-message', (_event: any, message: string, args: any) => { if (message === 'openCallbackUrl') { this.openCallbackUrl(args.url); } }); const initialCallbackUrl = (bridge().electronApp() as ElectronAppWrapper).initialCallbackUrl(); if (initialCallbackUrl) { this.openCallbackUrl(initialCallbackUrl); } } private openCallbackUrl(url: string) { if (!isCallbackUrl(url)) throw new Error(`Invalid callback URL: ${url}`); const { command, params } = parseCallbackUrl(url); void CommandService.instance().execute(command.toString(), params.id); } private updateLayoutPluginViews(layout: LayoutItem, plugins: PluginStates) { const infos = pluginUtils.viewInfosByType(plugins, 'webview'); let newLayout = produce(layout, (draftLayout: LayoutItem) => { for (const info of infos) { if (info.view.containerType !== ContainerType.Panel) continue; const viewId = info.view.id; const existingItem = findItemByKey(draftLayout, viewId); if (!existingItem) { draftLayout.children.push({ key: viewId, context: { pluginId: info.plugin.id, }, }); } } }); // Remove layout items that belong to plugins that are no longer // active. const pluginIds = Object.keys(plugins); const itemsToRemove: string[] = []; iterateItems(newLayout, (_itemIndex: number, item: LayoutItem, _parent: LayoutItem) => { if (item.context && item.context.pluginId && !pluginIds.includes(item.context.pluginId)) { itemsToRemove.push(item.key); } return true; }); for (const itemKey of itemsToRemove) { newLayout = removeItem(newLayout, itemKey); } return newLayout !== layout ? validateLayout(newLayout) : layout; } private showShareInvitationNotification(props: Props): boolean { if (props.processingShareInvitationResponse) return false; return !!props.shareInvitations.find(i => i.status === 0); } private buildLayout(plugins: PluginStates): LayoutItem { const rootLayoutSize = this.rootLayoutSize(); const userLayout = Setting.value('ui.layout'); let output = null; try { output = loadLayout(Object.keys(userLayout).length ? userLayout : null, defaultLayout, rootLayoutSize); // For unclear reasons, layout items sometimes end up without a key. // In that case, we can't do anything with them, so remove them // here. It could be due to the deprecated plugin API, which allowed // creating panel without a key, although in this case it should // have been set automatically. // https://github.com/laurent22/joplin/issues/4926 output = removeKeylessItems(output); if (!findItemByKey(output, 'sideBar') || !findItemByKey(output, 'noteList') || !findItemByKey(output, 'editor')) { throw new Error('"sideBar", "noteList" and "editor" must be present in the layout'); } } catch (error) { console.warn('Could not load layout - restoring default layout:', error); console.warn('Layout was:', userLayout); output = loadLayout(null, defaultLayout, rootLayoutSize); } return this.updateLayoutPluginViews(output, plugins); } private window_resize() { this.updateRootLayoutSize(); } public setupAppCloseHandling() { this.waitForNotesSavedIID_ = null; // This event is dispatched from the main process when the app is about // to close. The renderer process must respond with the "appCloseReply" // and tell the main process whether the app can really be closed or not. // For example, it cannot be closed right away if a note is being saved. // If a note is being saved, we wait till it is saved and then call // "appCloseReply" again. ipcRenderer.on('appClose', async () => { if (this.waitForNotesSavedIID_) shim.clearInterval(this.waitForNotesSavedIID_); this.waitForNotesSavedIID_ = null; const sendCanClose = async (canClose: boolean) => { if (canClose) { Setting.setValue('wasClosedSuccessfully', true); await Setting.saveAll(); } ipcRenderer.send('asynchronous-message', 'appCloseReply', { canClose }); }; await sendCanClose(!this.props.hasNotesBeingSaved); if (this.props.hasNotesBeingSaved) { this.waitForNotesSavedIID_ = shim.setInterval(() => { if (!this.props.hasNotesBeingSaved) { shim.clearInterval(this.waitForNotesSavedIID_); this.waitForNotesSavedIID_ = null; void sendCanClose(true); } }, 50); } }); } private notePropertiesDialog_close() { this.setState({ notePropertiesDialogOptions: {} }); } private noteContentPropertiesDialog_close() { this.setState({ noteContentPropertiesDialogOptions: {} }); } private shareNoteDialog_close() { this.setState({ shareNoteDialogOptions: {} }); } private shareFolderDialog_close() { this.setState({ shareFolderDialogOptions: { visible: false, folderId: '' } }); } public updateMainLayout(layout: LayoutItem) { this.props.dispatch({ type: 'MAIN_LAYOUT_SET', value: layout, }); } public updateRootLayoutSize() { // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied this.updateMainLayout(produce(this.props.mainLayout, (draft: any) => { const s = this.rootLayoutSize(); draft.width = s.width; draft.height = s.height; })); } public componentDidUpdate(prevProps: Props, prevState: State) { if (prevProps.style.width !== this.props.style.width || prevProps.style.height !== this.props.style.height || this.messageBoxVisible(prevProps) !== this.messageBoxVisible(this.props) ) { this.updateRootLayoutSize(); } if (prevProps.plugins !== this.props.plugins) { this.updateMainLayout(this.updateLayoutPluginViews(this.props.mainLayout, this.props.plugins)); // this.setState({ layout: this.buildLayout(this.props.plugins) }); } if (this.state.notePropertiesDialogOptions !== prevState.notePropertiesDialogOptions) { this.props.dispatch({ type: this.state.notePropertiesDialogOptions && this.state.notePropertiesDialogOptions.visible ? 'VISIBLE_DIALOGS_ADD' : 'VISIBLE_DIALOGS_REMOVE', name: 'noteProperties', }); } if (this.state.noteContentPropertiesDialogOptions !== prevState.noteContentPropertiesDialogOptions) { this.props.dispatch({ type: this.state.noteContentPropertiesDialogOptions && this.state.noteContentPropertiesDialogOptions.visible ? 'VISIBLE_DIALOGS_ADD' : 'VISIBLE_DIALOGS_REMOVE', name: 'noteContentProperties', }); } if (this.state.shareNoteDialogOptions !== prevState.shareNoteDialogOptions) { this.props.dispatch({ type: this.state.shareNoteDialogOptions && this.state.shareNoteDialogOptions.visible ? 'VISIBLE_DIALOGS_ADD' : 'VISIBLE_DIALOGS_REMOVE', name: 'shareNote', }); } if (this.state.shareFolderDialogOptions !== prevState.shareFolderDialogOptions) { this.props.dispatch({ type: this.state.shareFolderDialogOptions && this.state.shareFolderDialogOptions.visible ? 'VISIBLE_DIALOGS_ADD' : 'VISIBLE_DIALOGS_REMOVE', name: 'shareFolder', }); } if (this.props.mainLayout !== prevProps.mainLayout) { const toSave = saveLayout(this.props.mainLayout); Setting.setValue('ui.layout', toSave); } if (prevState.promptOptions !== this.state.promptOptions) { this.props.dispatch({ type: !prevState.promptOptions ? 'VISIBLE_DIALOGS_ADD' : 'VISIBLE_DIALOGS_REMOVE', name: 'promptDialog', }); } if (this.props.isResettingLayout) { Setting.setValue('ui.layout', null); this.updateMainLayout(this.buildLayout(this.props.plugins)); this.props.dispatch({ type: 'RESET_LAYOUT', value: false, }); } } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public layoutModeListenerKeyDown(event: any) { if (event.key !== 'Escape') return; if (!this.props.layoutMoveMode) return; void CommandService.instance().execute('toggleLayoutMoveMode'); } public componentDidMount() { window.addEventListener('keydown', this.layoutModeListenerKeyDown); } public componentWillUnmount() { this.unregisterCommands(); window.removeEventListener('resize', this.window_resize); window.removeEventListener('keydown', this.layoutModeListenerKeyDown); } public async waitForNoteToSaved(noteId: string) { while (noteId && this.props.editorNoteStatuses[noteId] === 'saving') { // eslint-disable-next-line no-console console.info('Waiting for note to be saved...', this.props.editorNoteStatuses); await time.msleep(100); } } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public async printTo_(target: string, options: any) { // Concurrent print calls are disallowed to avoid incorrect settings being restored upon completion if (this.isPrinting_) { // eslint-disable-next-line no-console console.info(`Printing ${options.path} to ${target} disallowed, already printing.`); return; } this.isPrinting_ = true; // Need to wait for save because the interop service reloads the note from the database await this.waitForNoteToSaved(options.noteId); if (target === 'pdf') { try { const pdfData = await InteropServiceHelper.exportNoteToPdf(options.noteId, { printBackground: true, pageSize: Setting.value('export.pdfPageSize'), landscape: Setting.value('export.pdfPageOrientation') === 'landscape', customCss: this.props.customCss, plugins: this.props.plugins, }); await shim.fsDriver().writeFile(options.path, pdfData, 'buffer'); } catch (error) { console.error(error); bridge().showErrorMessageBox(error.message); } } else if (target === 'printer') { try { await InteropServiceHelper.printNote(options.noteId, { printBackground: true, customCss: this.props.customCss, }); } catch (error) { console.error(error); bridge().showErrorMessageBox(error.message); } } this.isPrinting_ = false; } public rootLayoutSize() { return { width: window.innerWidth, height: this.rowHeight(), }; } public rowHeight() { if (!this.props) return 0; return this.props.style.height - (this.messageBoxVisible() ? this.messageBoxHeight() : 0); } public messageBoxHeight() { return 50; } public styles(themeId: number, width: number, height: number, messageBoxVisible: boolean) { const styleKey = [themeId, width, height, messageBoxVisible].join('_'); if (styleKey === this.styleKey_) return this.styles_; const theme = themeStyle(themeId); this.styleKey_ = styleKey; this.styles_ = {}; this.styles_.header = { width: width, }; this.styles_.messageBox = { width: width, height: this.messageBoxHeight(), display: 'flex', alignItems: 'center', paddingLeft: 10, backgroundColor: theme.warningBackgroundColor, }; const rowHeight = height - (messageBoxVisible ? this.styles_.messageBox.height : 0); this.styles_.rowHeight = rowHeight; this.styles_.resizableLayout = { height: rowHeight, }; this.styles_.prompt = { width: width, height: height, }; this.styles_.modalLayer = { ...theme.textStyle, zIndex: 10000, position: 'absolute', top: 0, left: 0, backgroundColor: theme.backgroundColor, width: width - 20, height: height - 20, padding: 10 }; return this.styles_; } // eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied private renderNotificationMessage(message: string, callForAction: string = null, callForActionHandler: Function = null, callForAction2: string = null, callForActionHandler2: Function = null) { const theme = themeStyle(this.props.themeId); // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied const urlStyle: any = { color: theme.colorWarnUrl, textDecoration: 'underline' }; if (!callForAction) return <span>{message}</span>; const cfa = ( <a href="#" style={urlStyle} onClick={() => callForActionHandler()}> {callForAction} </a> ); const cfa2 = !callForAction2 ? null : ( <a href="#" style={urlStyle} onClick={() => callForActionHandler2()}> {callForAction2} </a> ); return ( <span> {message}{callForAction ? ' ' : ''} {cfa}{callForAction2 ? ' / ' : ''}{cfa2} </span> ); } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public renderNotification(theme: ThemeStyle, styles: any) { if (!this.messageBoxVisible()) return null; const onViewStatusScreen = () => { this.props.dispatch({ type: 'NAV_GO', routeName: 'Status', }); }; const onViewEncryptionConfigScreen = () => { this.props.dispatch({ type: 'NAV_GO', routeName: 'Config', props: { defaultSection: 'encryption', }, }); }; const onViewJoplinCloudLoginScreen = () => { this.props.dispatch({ type: 'NAV_GO', routeName: 'JoplinCloudLogin', }); }; const onViewSyncSettingsScreen = () => { this.props.dispatch({ type: 'NAV_GO', routeName: 'Config', props: { defaultSection: 'sync', }, }); }; const onRestartAndUpgrade = async () => { Setting.setValue('sync.upgradeState', Setting.SYNC_UPGRADE_STATE_MUST_DO); await Setting.saveAll(); await restart(); }; const onDisableSafeModeAndRestart = async () => { Setting.setValue('isSafeMode', false); await Setting.saveAll(); await restart(); }; const onInvitationRespond = async (shareUserId: string, folderId: string, masterKey: MasterKeyEntity, accept: boolean) => { await invitationRespond(shareUserId, folderId, masterKey, accept); }; let msg = null; // When adding something here, don't forget to update the condition in // this.messageBoxVisible() if (this.props.isSafeMode) { msg = this.renderNotificationMessage( _('Safe mode is currently active. Note rendering and all plugins are temporarily disabled.'), _('Disable safe mode and restart'), onDisableSafeModeAndRestart, ); } else if (this.props.hasMissingSyncCredentials) { msg = this.renderNotificationMessage( _('The synchronisation password is missing.'), _('Set the password'), onViewSyncSettingsScreen, ); } else if (this.props.shouldUpgradeSyncTarget) { msg = this.renderNotificationMessage( _('The sync target needs to be upgraded before Joplin can sync. The operation may take a few minutes to complete and the app needs to be restarted. To proceed please click on the link.'), _('Restart and upgrade'), onRestartAndUpgrade, ); } else if (this.props.hasDisabledEncryptionItems) { msg = this.renderNotificationMessage( _('Some items cannot be decrypted.'), _('View them now'), onViewStatusScreen, ); } else if (this.props.showNeedUpgradingMasterKeyMessage) { msg = this.renderNotificationMessage( _('One of your master keys use an obsolete encryption method.'), _('View them now'), onViewEncryptionConfigScreen, ); } else if (this.props.showShouldReencryptMessage) { msg = this.renderNotificationMessage( _('The default encryption method has been changed, you should re-encrypt your data.'), _('More info'), onViewEncryptionConfigScreen, ); } else if (this.showShareInvitationNotification(this.props)) { const invitation = this.props.shareInvitations.find(inv => inv.status === 0); const sharer = invitation.share.user; msg = this.renderNotificationMessage( _('%s (%s) would like to share a notebook with you.', sharer.full_name, sharer.email), _('Accept'), () => onInvitationRespond(invitation.id, invitation.share.folder_id, invitation.master_key, true), _('Reject'), () => onInvitationRespond(invitation.id, invitation.share.folder_id, invitation.master_key, false), ); } else if (this.props.hasDisabledSyncItems) { msg = this.renderNotificationMessage( _('Some items cannot be synchronised.'), _('View them now'), onViewStatusScreen, ); } else if (this.props.showMissingMasterKeyMessage) { msg = this.renderNotificationMessage( _('One or more master keys need a password.'), _('Set the password'), onViewEncryptionConfigScreen, ); } else if (this.props.mustUpgradeAppMessage) { msg = this.renderNotificationMessage(this.props.mustUpgradeAppMessage); } else if (this.props.showInvalidJoplinCloudCredential) { msg = this.renderNotificationMessage( _('Your Joplin Cloud credentials are invalid, please login.'), _('Login to Joplin Cloud.'), onViewJoplinCloudLoginScreen, ); } return ( <div style={styles.messageBox}> <span style={theme.textStyle}>{msg}</span> </div> ); } public messageBoxVisible(props: Props = null) { if (!props) props = this.props; return props.hasDisabledSyncItems || props.showMissingMasterKeyMessage || props.hasMissingSyncCredentials || props.showNeedUpgradingMasterKeyMessage || props.showShouldReencryptMessage || props.hasDisabledEncryptionItems || this.props.shouldUpgradeSyncTarget || props.isSafeMode || this.showShareInvitationNotification(props) || this.props.needApiAuth || !!this.props.mustUpgradeAppMessage || props.showInvalidJoplinCloudCredential; } public registerCommands() { for (const command of commands) { CommandService.instance().registerRuntime(command.declaration.name, command.runtime(this)); } } public unregisterCommands() { for (const command of commands) { CommandService.instance().unregisterRuntime(command.declaration.name); } } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied private resizableLayout_resize(event: any) { this.updateMainLayout(event.layout); } private resizableLayout_moveButtonClick(event: MoveButtonClickEvent) { const newLayout = move(this.props.mainLayout, event.itemKey, event.direction); this.updateMainLayout(newLayout); } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied private resizableLayout_renderItem(key: string, event: any) { // Key should never be undefined but somehow it can happen, also not // clear how. For now in this case render nothing so that the app // doesn't crash. // https://discourse.joplinapp.org/t/rearranging-the-pannels-crushed-the-app-and-generated-fatal-error/14373?u=laurent if (!key) { console.error('resizableLayout_renderItem: Trying to render an item using an empty key. Full layout is:', this.props.mainLayout); return null; } const eventEmitter = event.eventEmitter; // const viewsToRemove:string[] = []; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied const components: any = { sideBar: () => { return <Sidebar key={key} />; }, noteList: () => { return <NoteListWrapper key={key} resizableLayoutEventEmitter={eventEmitter} visible={event.visible} size={event.size} themeId={this.props.themeId} listRendererId={this.props.listRendererId} startupPluginsLoaded={this.props.startupPluginsLoaded} notesSortOrderField={this.props.notesSortOrderField} notesSortOrderReverse={this.props.notesSortOrderReverse} columns={this.props.notesColumns} selectedFolderId={this.props.selectedFolderId} />; }, editor: () => { let bodyEditor = this.props.settingEditorCodeView ? 'CodeMirror6' : 'TinyMCE'; if (this.props.isSafeMode) { bodyEditor = 'PlainText'; } else if (this.props.settingEditorCodeView && this.props.enableLegacyMarkdownEditor) { bodyEditor = 'CodeMirror5'; } return <NoteEditor key={key} bodyEditor={bodyEditor} />; }, }; if (components[key]) return components[key](); const viewsToRemove: string[] = []; if (key.indexOf('plugin-view') === 0) { const viewInfo = pluginUtils.viewInfoByViewId(this.props.plugins, event.item.key); if (!viewInfo) { // Once all startup plugins have loaded, we know that all the // views are ready so we can remove the orphans ones. // // Before they are loaded, there might be views that don't match // any plugins, but that's only because it hasn't loaded yet. if (this.props.startupPluginsLoaded) { console.warn(`Could not find plugin associated with view: ${event.item.key}`); viewsToRemove.push(event.item.key); } } else { const { view, plugin } = viewInfo; const html = this.props.pluginHtmlContents[plugin.id]?.[view.id] ?? ''; return <UserWebview key={view.id} viewId={view.id} themeId={this.props.themeId} html={html} scripts={view.scripts} pluginId={plugin.id} borderBottom={true} fitToContent={false} />; } } else { throw new Error(`Invalid layout component: ${key}`); } if (viewsToRemove.length) { window.requestAnimationFrame(() => { let newLayout = this.props.mainLayout; for (const itemKey of viewsToRemove) { newLayout = removeItem(newLayout, itemKey); } if (newLayout !== this.props.mainLayout) { console.warn('Removed invalid views:', viewsToRemove); this.updateMainLayout(newLayout); } }); } } public renderPluginDialogs() { const output = []; const infos = pluginUtils.viewInfosByType(this.props.plugins, 'webview'); for (const info of infos) { const { plugin, view } = info; if (view.containerType !== ContainerType.Dialog) continue; if (!view.opened) continue; const html = this.props.pluginHtmlContents[plugin.id]?.[view.id] ?? ''; output.push(<UserWebviewDialog key={view.id} viewId={view.id} themeId={this.props.themeId} html={html} scripts={view.scripts} pluginId={plugin.id} buttons={view.buttons} fitToContent={view.fitToContent} />); } if (!output.length) return null; return ( <StyledUserWebviewDialogContainer> {output} </StyledUserWebviewDialogContainer> ); } public render() { const theme = themeStyle(this.props.themeId); const style = { color: theme.color, backgroundColor: theme.backgroundColor, ...this.props.style, }; const promptOptions = this.state.promptOptions; const styles = this.styles(this.props.themeId, style.width, style.height, this.messageBoxVisible()); if (!this.promptOnClose_) { // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied this.promptOnClose_ = (answer: any, buttonType: any) => { return this.state.promptOptions.onClose(answer, buttonType); }; } const messageComp = this.renderNotification(theme, styles); const dialogInfo = PluginManager.instance().pluginDialogToShow(this.props.pluginsLegacy); const pluginDialog = !dialogInfo ? null : <dialogInfo.Dialog {...dialogInfo.props} />; const modalLayerStyle = { ...styles.modalLayer, display: this.state.modalLayer.visible ? 'block' : 'none' }; const notePropertiesDialogOptions = this.state.notePropertiesDialogOptions; const noteContentPropertiesDialogOptions = this.state.noteContentPropertiesDialogOptions; const shareNoteDialogOptions = this.state.shareNoteDialogOptions; const shareFolderDialogOptions = this.state.shareFolderDialogOptions; const layoutComp = this.props.mainLayout ? ( <ResizableLayout height={styles.rowHeight} layout={this.props.mainLayout} onResize={this.resizableLayout_resize} onMoveButtonClick={this.resizableLayout_moveButtonClick} renderItem={this.resizableLayout_renderItem} moveMode={this.props.layoutMoveMode} moveModeMessage={_('Use the arrows to move the layout items. Press "Escape" to exit.')} /> ) : null; return ( <div style={style}> <div style={modalLayerStyle}>{this.state.modalLayer.message}</div> {this.renderPluginDialogs()} {noteContentPropertiesDialogOptions.visible && <NoteContentPropertiesDialog markupLanguage={noteContentPropertiesDialogOptions.markupLanguage} themeId={this.props.themeId} onClose={this.noteContentPropertiesDialog_close} text={noteContentPropertiesDialogOptions.text}/>} {notePropertiesDialogOptions.visible && <NotePropertiesDialog themeId={this.props.themeId} noteId={notePropertiesDialogOptions.noteId} onClose={this.notePropertiesDialog_close} onRevisionLinkClick={notePropertiesDialogOptions.onRevisionLinkClick} />} {shareNoteDialogOptions.visible && <ShareNoteDialog themeId={this.props.themeId} noteIds={shareNoteDialogOptions.noteIds} onClose={this.shareNoteDialog_close} />} {shareFolderDialogOptions.visible && <ShareFolderDialog themeId={this.props.themeId} folderId={shareFolderDialogOptions.folderId} onClose={this.shareFolderDialog_close} />} <PromptDialog autocomplete={promptOptions && 'autocomplete' in promptOptions ? promptOptions.autocomplete : null} defaultValue={promptOptions && promptOptions.value ? promptOptions.value : ''} themeId={this.props.themeId} style={styles.prompt} onClose={this.promptOnClose_} label={promptOptions ? promptOptions.label : ''} description={promptOptions ? promptOptions.description : null} visible={!!this.state.promptOptions} buttons={promptOptions && 'buttons' in promptOptions ? promptOptions.buttons : null} inputType={promptOptions && 'inputType' in promptOptions ? promptOptions.inputType : null} /> <TrashNotification lastDeletion={this.props.lastDeletion} lastDeletionNotificationTime={this.props.lastDeletionNotificationTime} themeId={this.props.themeId} // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied dispatch={this.props.dispatch as any} /> <UpdateNotification themeId={this.props.themeId} /> {messageComp} {layoutComp} {pluginDialog} </div> ); } } const mapStateToProps = (state: AppState) => { const syncInfo = localSyncInfoFromState(state); const showNeedUpgradingEnabledMasterKeyMessage = !!EncryptionService.instance().masterKeysThatNeedUpgrading(syncInfo.masterKeys.filter((k) => !!k.enabled)).length; return { themeId: state.settings.theme, settingEditorCodeView: state.settings['editor.codeView'], hasDisabledSyncItems: state.hasDisabledSyncItems, hasDisabledEncryptionItems: state.hasDisabledEncryptionItems, showMissingMasterKeyMessage: showMissingMasterKeyMessage(syncInfo, state.notLoadedMasterKeys), showNeedUpgradingMasterKeyMessage: showNeedUpgradingEnabledMasterKeyMessage, showShouldReencryptMessage: state.settings['encryption.shouldReencrypt'] >= Setting.SHOULD_REENCRYPT_YES, shouldUpgradeSyncTarget: state.settings['sync.upgradeState'] === Setting.SYNC_UPGRADE_STATE_SHOULD_DO, hasMissingSyncCredentials: shouldShowMissingPasswordWarning(state.settings['sync.target'], state.settings), pluginsLegacy: state.pluginsLegacy, plugins: state.pluginService.plugins, pluginHtmlContents: state.pluginService.pluginHtmlContents, customCss: state.customCss, editorNoteStatuses: state.editorNoteStatuses, hasNotesBeingSaved: stateUtils.hasNotesBeingSaved(state), layoutMoveMode: state.layoutMoveMode, mainLayout: state.mainLayout, startupPluginsLoaded: state.startupPluginsLoaded, shareInvitations: state.shareService.shareInvitations, processingShareInvitationResponse: state.shareService.processingShareInvitationResponse, isSafeMode: state.settings.isSafeMode, enableLegacyMarkdownEditor: state.settings['editor.legacyMarkdown'], needApiAuth: state.needApiAuth, isResettingLayout: state.isResettingLayout, listRendererId: state.settings['notes.listRendererId'], lastDeletion: state.lastDeletion, lastDeletionNotificationTime: state.lastDeletionNotificationTime, selectedFolderId: state.selectedFolderId, mustUpgradeAppMessage: state.mustUpgradeAppMessage, notesSortOrderField: state.settings['notes.sortOrder.field'], notesSortOrderReverse: state.settings['notes.sortOrder.reverse'], notesColumns: validateColumns(state.settings['notes.columns']), showInvalidJoplinCloudCredential: state.settings['sync.target'] === 10 && state.mustAuthenticate, }; }; export default connect(mapStateToProps)(MainScreenComponent);