mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-14 18:27:44 +02:00
807 lines
27 KiB
TypeScript
807 lines
27 KiB
TypeScript
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 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 { defaultWindowId, StateLastDeletion, stateUtils } from '@joplin/lib/reducer';
|
|
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 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 { 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 invitationRespond from '@joplin/lib/services/share/invitationRespond';
|
|
import restart from '../services/restart';
|
|
import { connect } from 'react-redux';
|
|
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';
|
|
import NoteEditor from './NoteEditor/NoteEditor';
|
|
|
|
const ipcRenderer = require('electron').ipcRenderer;
|
|
|
|
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;
|
|
shouldUpgradeSyncTarget: boolean;
|
|
hasDisabledSyncItems: boolean;
|
|
hasDisabledEncryptionItems: boolean;
|
|
hasMissingSyncCredentials: boolean;
|
|
showMissingMasterKeyMessage: boolean;
|
|
showNeedUpgradingMasterKeyMessage: boolean;
|
|
showShouldReencryptMessage: boolean;
|
|
themeId: number;
|
|
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;
|
|
// 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 styleKey_: string;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
private styles_: any;
|
|
|
|
public constructor(props: Props) {
|
|
super(props);
|
|
|
|
this.state = {
|
|
promptOptions: null,
|
|
notePropertiesDialogOptions: {},
|
|
noteContentPropertiesDialogOptions: {},
|
|
shareNoteDialogOptions: {},
|
|
shareFolderDialogOptions: {
|
|
visible: false,
|
|
folderId: '',
|
|
},
|
|
};
|
|
|
|
this.updateMainLayout(this.buildLayout(props.plugins));
|
|
|
|
this.setupAppCloseHandling();
|
|
|
|
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);
|
|
}
|
|
});
|
|
}
|
|
|
|
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.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() {
|
|
window.removeEventListener('resize', this.window_resize);
|
|
window.removeEventListener('keydown', this.layoutModeListenerKeyDown);
|
|
}
|
|
|
|
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,
|
|
};
|
|
|
|
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;
|
|
}
|
|
|
|
// 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: () => {
|
|
return <div className='note-editor-wrapper' role='main' aria-label={_('Note')}>
|
|
<NoteEditor
|
|
windowId={defaultWindowId}
|
|
key={key}
|
|
startupPluginsLoaded={this.props.startupPluginsLoaded}
|
|
/>
|
|
</div>;
|
|
},
|
|
};
|
|
|
|
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 styles = this.styles(this.props.themeId, style.width, style.height, this.messageBoxVisible());
|
|
|
|
const messageComp = this.renderNotification(theme, styles);
|
|
|
|
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}>
|
|
<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}
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
const mapStateToProps = (state: AppState) => {
|
|
const syncInfo = localSyncInfoFromState(state);
|
|
const showNeedUpgradingEnabledMasterKeyMessage = !!EncryptionService.instance().masterKeysThatNeedUpgrading(syncInfo.masterKeys.filter((k) => !!k.enabled)).length;
|
|
const windowState = stateUtils.windowStateById(state, defaultWindowId);
|
|
|
|
return {
|
|
themeId: state.settings.theme,
|
|
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),
|
|
plugins: state.pluginService.plugins,
|
|
pluginHtmlContents: state.pluginService.pluginHtmlContents,
|
|
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: windowState.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);
|