You've already forked joplin
mirror of
https://github.com/laurent22/joplin.git
synced 2025-11-23 22:36:32 +02:00
186 lines
6.1 KiB
TypeScript
186 lines
6.1 KiB
TypeScript
import * as React from 'react';
|
|
import { useRef, useState, useEffect, useCallback } from 'react';
|
|
import useWindowResizeEvent from './utils/useWindowResizeEvent';
|
|
import setLayoutItemProps from './utils/setLayoutItemProps';
|
|
import useLayoutItemSizes, { LayoutItemSizes, itemSize, calculateMaxSizeAvailableForItem, itemMinWidth, itemMinHeight } from './utils/useLayoutItemSizes';
|
|
import validateLayout from './utils/validateLayout';
|
|
import { Size, LayoutItem } from './utils/types';
|
|
import { canMove, MoveDirection } from './utils/movements';
|
|
import MoveButtons, { MoveButtonClickEvent } from './MoveButtons';
|
|
import { StyledWrapperRoot, StyledMoveOverlay, MoveModeRootMessage } from './utils/style';
|
|
import type { ResizeCallback, ResizeStartCallback } from 're-resizable';
|
|
import Dialog from '../Dialog';
|
|
import EventEmitter = require('events');
|
|
import LayoutItemContainer from './LayoutItemContainer';
|
|
|
|
interface OnResizeEvent {
|
|
layout: LayoutItem;
|
|
}
|
|
|
|
interface ResizedItem {
|
|
key: string;
|
|
initialWidth: number;
|
|
initialHeight: number;
|
|
maxSize: Size;
|
|
}
|
|
|
|
interface Props {
|
|
layout: LayoutItem;
|
|
layoutKeyToLabel: (key: string)=> string;
|
|
onResize(event: OnResizeEvent): void;
|
|
width?: number;
|
|
height?: number;
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
|
renderItem: Function;
|
|
onMoveButtonClick(event: MoveButtonClickEvent): void;
|
|
moveMode: boolean;
|
|
moveModeMessage: string;
|
|
}
|
|
|
|
function itemVisible(item: LayoutItem, moveMode: boolean) {
|
|
if (moveMode) return true;
|
|
if (item.children && !item.children.length) return false;
|
|
return item.visible !== false;
|
|
}
|
|
|
|
function ResizableLayout(props: Props) {
|
|
const eventEmitter = useRef(new EventEmitter());
|
|
|
|
const [resizedItem, setResizedItem] = useState<ResizedItem|null>(null);
|
|
const lastUsedMoveButtonKey = useRef<string|null>(null);
|
|
|
|
const onMoveButtonClick = useCallback((event: MoveButtonClickEvent) => {
|
|
lastUsedMoveButtonKey.current = event.buttonKey;
|
|
props.onMoveButtonClick(event);
|
|
}, [props.onMoveButtonClick]);
|
|
|
|
const renderMoveControls = (item: LayoutItem, parent: LayoutItem | null, size: Size) => {
|
|
return (
|
|
<StyledWrapperRoot key={item.key} size={size}>
|
|
<StyledMoveOverlay>
|
|
<MoveButtons
|
|
autoFocusKey={lastUsedMoveButtonKey.current}
|
|
itemKey={item.key}
|
|
itemLabel={props.layoutKeyToLabel(item.key)}
|
|
onClick={onMoveButtonClick}
|
|
canMoveLeft={canMove(MoveDirection.Left, item, parent)}
|
|
canMoveRight={canMove(MoveDirection.Right, item, parent)}
|
|
canMoveUp={canMove(MoveDirection.Up, item, parent)}
|
|
canMoveDown={canMove(MoveDirection.Down, item, parent)}
|
|
/>
|
|
</StyledMoveOverlay>
|
|
</StyledWrapperRoot>
|
|
);
|
|
};
|
|
|
|
function renderItemWrapper(comp: React.ReactNode, item: LayoutItem, size: Size) {
|
|
return (
|
|
<StyledWrapperRoot key={item.key} size={size}>
|
|
{comp}
|
|
</StyledWrapperRoot>
|
|
);
|
|
}
|
|
|
|
function renderLayoutItem(
|
|
item: LayoutItem, parent: LayoutItem | null, sizes: LayoutItemSizes, isVisible: boolean, isLastChild: boolean, onlyMoveControls: boolean,
|
|
): React.ReactNode {
|
|
const onResizeStart: ResizeStartCallback = () => {
|
|
setResizedItem({
|
|
key: item.key,
|
|
initialWidth: sizes[item.key].width,
|
|
initialHeight: sizes[item.key].height,
|
|
maxSize: calculateMaxSizeAvailableForItem(item, parent, sizes),
|
|
});
|
|
};
|
|
|
|
const onResize: ResizeCallback = (_event, direction, _refToElement, delta) => {
|
|
const newWidth = Math.max(itemMinWidth, resizedItem.initialWidth + delta.width);
|
|
const newHeight = Math.max(itemMinHeight, resizedItem.initialHeight + delta.height);
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const newSize: any = {};
|
|
|
|
if (item.width) newSize.width = item.width;
|
|
if (item.height) newSize.height = item.height;
|
|
|
|
if (direction === 'bottom') {
|
|
newSize.height = newHeight;
|
|
} else {
|
|
newSize.width = newWidth;
|
|
}
|
|
|
|
const newLayout = setLayoutItemProps(props.layout, item.key, newSize);
|
|
|
|
props.onResize({ layout: newLayout });
|
|
eventEmitter.current.emit('resize');
|
|
};
|
|
|
|
const onResizeStop: ResizeCallback = (_event, _direction, _refToElement, delta) => {
|
|
onResize(_event, _direction, _refToElement, delta);
|
|
setResizedItem(null);
|
|
};
|
|
|
|
const resizedItemMaxSize = resizedItem && item.key === resizedItem.key ? resizedItem.maxSize : null;
|
|
const visible = itemVisible(item, props.moveMode);
|
|
const itemContainerProps = {
|
|
key: item.key, item, parent, sizes, resizedItemMaxSize, onResizeStart, onResizeStop, onResize, isLastChild, visible,
|
|
};
|
|
if (!item.children) {
|
|
const size = itemSize(item, parent, sizes, false);
|
|
|
|
const comp = props.renderItem(item.key, {
|
|
item: item,
|
|
eventEmitter: eventEmitter.current,
|
|
size: size,
|
|
visible: isVisible,
|
|
});
|
|
|
|
const wrapper = onlyMoveControls ? renderMoveControls(item, parent, size) : renderItemWrapper(comp, item, size);
|
|
return <LayoutItemContainer {...itemContainerProps}>
|
|
{wrapper}
|
|
</LayoutItemContainer>;
|
|
} else {
|
|
const childrenComponents = [];
|
|
for (let i = 0; i < item.children.length; i++) {
|
|
const child = item.children[i];
|
|
childrenComponents.push(
|
|
renderLayoutItem(child, item, sizes, isVisible && itemVisible(child, props.moveMode), i === item.children.length - 1, onlyMoveControls),
|
|
);
|
|
}
|
|
|
|
return <LayoutItemContainer {...itemContainerProps}>
|
|
{childrenComponents}
|
|
</LayoutItemContainer>;
|
|
}
|
|
}
|
|
|
|
useEffect(() => {
|
|
validateLayout(props.layout);
|
|
}, [props.layout]);
|
|
|
|
useWindowResizeEvent(eventEmitter);
|
|
const sizes = useLayoutItemSizes(props.layout, props.moveMode);
|
|
|
|
const renderRoot = (moveControlsOnly: boolean) => {
|
|
return renderLayoutItem(props.layout, null, sizes, itemVisible(props.layout, props.moveMode), true, moveControlsOnly);
|
|
};
|
|
|
|
function renderMoveModeBox() {
|
|
return <div>
|
|
<Dialog contentFillsScreen={true} className='change-app-layout-dialog'>
|
|
<MoveModeRootMessage>{props.moveModeMessage}</MoveModeRootMessage>
|
|
{renderRoot(true)}
|
|
</Dialog>
|
|
{renderRoot(false)}
|
|
</div>;
|
|
}
|
|
|
|
if (props.moveMode) {
|
|
return renderMoveModeBox();
|
|
} else {
|
|
return renderRoot(false);
|
|
}
|
|
}
|
|
|
|
export default ResizableLayout;
|