mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-02-23 23:54:13 +08:00
173 lines
4.5 KiB
TypeScript
173 lines
4.5 KiB
TypeScript
import React from 'react';
|
|
import { CSSTransitionGroup } from 'react-transition-group';
|
|
import { WorkspaceStore } from 'mailspring-exports';
|
|
|
|
import Sheet from './sheet';
|
|
import Toolbar from './sheet-toolbar';
|
|
import { Flexbox } from './components/flexbox';
|
|
import { InjectedComponentSet } from './components/injected-component-set';
|
|
import { SheetDeclaration } from './flux/stores/workspace-store';
|
|
|
|
interface SheetContainerState {
|
|
stack: SheetDeclaration[];
|
|
mode: string;
|
|
error?: string;
|
|
}
|
|
|
|
export default class SheetContainer extends React.Component<{}, SheetContainerState> {
|
|
static displayName = 'SheetContainer';
|
|
|
|
_toolbarComponents = {};
|
|
unsubscribe?: () => void;
|
|
|
|
constructor(props) {
|
|
super(props);
|
|
this.state = this._getStateFromStores();
|
|
}
|
|
|
|
componentDidMount() {
|
|
this.unsubscribe = WorkspaceStore.listen(this._onStoreChange);
|
|
}
|
|
|
|
componentDidCatch(error, info) {
|
|
// We don't currently display the error, but we need to call setState within
|
|
// this function or the component does not re-render after being reset.
|
|
this.setState({ error: error.stack });
|
|
AppEnv.reportError(error);
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
if (this.unsubscribe) {
|
|
this.unsubscribe();
|
|
}
|
|
}
|
|
|
|
_getStateFromStores() {
|
|
return {
|
|
stack: WorkspaceStore.sheetStack(),
|
|
mode: WorkspaceStore.layoutMode(),
|
|
};
|
|
}
|
|
|
|
_onColumnSizeChanged = sheet => {
|
|
const toolbar = this._toolbarComponents[sheet.props.depth];
|
|
if (toolbar) {
|
|
toolbar.recomputeLayout();
|
|
}
|
|
window.dispatchEvent(new Event('resize'));
|
|
};
|
|
|
|
_onStoreChange = () => {
|
|
this.setState(this._getStateFromStores());
|
|
};
|
|
|
|
_lastToolbarClickTime: number = 0;
|
|
|
|
_onToolbarDoubleClick = (e: React.MouseEvent<HTMLDivElement>) => {
|
|
if (process.platform !== 'darwin') return;
|
|
if (e.target instanceof HTMLElement) {
|
|
if (['INPUT', 'A', 'BUTTON'].includes(e.target.tagName)) return;
|
|
if (e.target.hasAttribute('contenteditable')) return;
|
|
}
|
|
|
|
if (Date.now() - this._lastToolbarClickTime < 350) {
|
|
const win = AppEnv.getCurrentWindow();
|
|
if (win.isMaximized()) {
|
|
win.unmaximize();
|
|
} else {
|
|
win.maximize();
|
|
}
|
|
}
|
|
this._lastToolbarClickTime = Date.now();
|
|
};
|
|
|
|
_toolbarContainerElement() {
|
|
const { toolbar } = AppEnv.getLoadSettings();
|
|
if (!toolbar) {
|
|
return [];
|
|
}
|
|
|
|
const components = this.state.stack.map((sheet, index) => (
|
|
<Toolbar
|
|
data={sheet}
|
|
ref={cm => {
|
|
this._toolbarComponents[index] = cm;
|
|
}}
|
|
key={`${index}:${sheet.id}:toolbar`}
|
|
depth={index}
|
|
/>
|
|
));
|
|
|
|
return (
|
|
<div
|
|
className="sheet-toolbar"
|
|
style={{ order: 0, zIndex: 3 }}
|
|
onClick={this._onToolbarDoubleClick}
|
|
>
|
|
{components[0]}
|
|
<CSSTransitionGroup
|
|
transitionLeaveTimeout={125}
|
|
transitionEnterTimeout={125}
|
|
transitionName="opacity-125ms"
|
|
>
|
|
{components.slice(1)}
|
|
</CSSTransitionGroup>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
render() {
|
|
const totalSheets = this.state.stack.length;
|
|
const topSheet = this.state.stack[totalSheets - 1];
|
|
|
|
if (!topSheet) {
|
|
return <div />;
|
|
}
|
|
|
|
const sheetComponents = this.state.stack.map((sheet, index) => (
|
|
<Sheet
|
|
data={sheet}
|
|
depth={index}
|
|
key={index > 0 ? `${index}:${sheet.id}` : `root`}
|
|
onColumnSizeChanged={this._onColumnSizeChanged}
|
|
/>
|
|
));
|
|
|
|
return (
|
|
<Flexbox
|
|
direction="column"
|
|
className={`layout-mode-${this.state.mode}`}
|
|
style={{ overflow: 'hidden' }}
|
|
>
|
|
{this._toolbarContainerElement()}
|
|
|
|
<div style={{ order: 1, zIndex: 2 }}>
|
|
<InjectedComponentSet
|
|
matching={{ locations: [topSheet.Header, WorkspaceStore.Sheet.Global.Header] }}
|
|
direction="column"
|
|
id={topSheet.id}
|
|
/>
|
|
</div>
|
|
|
|
<div style={{ order: 2, flex: 1, position: 'relative', zIndex: 1 }}>
|
|
{sheetComponents[0]}
|
|
<CSSTransitionGroup
|
|
transitionLeaveTimeout={125}
|
|
transitionEnterTimeout={125}
|
|
transitionName="sheet-stack"
|
|
>
|
|
{sheetComponents.slice(1)}
|
|
</CSSTransitionGroup>
|
|
</div>
|
|
|
|
<div style={{ order: 3, zIndex: 4 }}>
|
|
<InjectedComponentSet
|
|
matching={{ locations: [topSheet.Footer, WorkspaceStore.Sheet.Global.Footer] }}
|
|
direction="column"
|
|
id={topSheet.id}
|
|
/>
|
|
</div>
|
|
</Flexbox>
|
|
);
|
|
}
|
|
}
|