mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-06 19:18:22 +01:00
Fix block content sizing (#280)
Make the block content sizing update once when its node moves or becomes magnified. By manually updating this inner sizing rather than letting the block flow in the DOM, the animations of the block frames are much smoother. This also fixes an issue where two scrollbars were being rendered for the Directory Preview widget. This also sets zero padding on nodes when there's only a single node being rendered.
This commit is contained in:
parent
86fc45fc13
commit
c892c39a73
@ -6,7 +6,7 @@ import { PlotView } from "@/app/view/plotview/plotview";
|
|||||||
import { PreviewModel, PreviewView, makePreviewModel } from "@/app/view/preview/preview";
|
import { PreviewModel, PreviewView, makePreviewModel } from "@/app/view/preview/preview";
|
||||||
import { ErrorBoundary } from "@/element/errorboundary";
|
import { ErrorBoundary } from "@/element/errorboundary";
|
||||||
import { CenteredDiv } from "@/element/quickelems";
|
import { CenteredDiv } from "@/element/quickelems";
|
||||||
import { NodeModel } from "@/layout/index";
|
import { NodeModel, useDebouncedNodeInnerRect } from "@/layout/index";
|
||||||
import { counterInc, getViewModel, registerViewModel, unregisterViewModel } from "@/store/global";
|
import { counterInc, getViewModel, registerViewModel, unregisterViewModel } from "@/store/global";
|
||||||
import * as WOS from "@/store/wos";
|
import * as WOS from "@/store/wos";
|
||||||
import * as util from "@/util/util";
|
import * as util from "@/util/util";
|
||||||
@ -122,7 +122,7 @@ const BlockFull = React.memo(({ nodeModel, viewModel }: FullBlockProps) => {
|
|||||||
const [focusedChild, setFocusedChild] = React.useState(null);
|
const [focusedChild, setFocusedChild] = React.useState(null);
|
||||||
const isFocused = jotai.useAtomValue(nodeModel.isFocused);
|
const isFocused = jotai.useAtomValue(nodeModel.isFocused);
|
||||||
const disablePointerEvents = jotai.useAtomValue(nodeModel.disablePointerEvents);
|
const disablePointerEvents = jotai.useAtomValue(nodeModel.disablePointerEvents);
|
||||||
const addlProps = jotai.useAtomValue(nodeModel.additionalProps);
|
const innerRect = useDebouncedNodeInnerRect(nodeModel);
|
||||||
|
|
||||||
React.useLayoutEffect(() => {
|
React.useLayoutEffect(() => {
|
||||||
setBlockClicked(isFocused);
|
setBlockClicked(isFocused);
|
||||||
@ -152,6 +152,32 @@ const BlockFull = React.memo(({ nodeModel, viewModel }: FullBlockProps) => {
|
|||||||
setBlockClicked(true);
|
setBlockClicked(true);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const [blockContentOffset, setBlockContentOffset] = React.useState<Dimensions>();
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (blockRef.current && contentRef.current) {
|
||||||
|
const blockRect = blockRef.current.getBoundingClientRect();
|
||||||
|
const contentRect = contentRef.current.getBoundingClientRect();
|
||||||
|
setBlockContentOffset({
|
||||||
|
top: 0,
|
||||||
|
left: 0,
|
||||||
|
width: blockRect.width - contentRect.width,
|
||||||
|
height: blockRect.height - contentRect.height,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [blockRef, contentRef]);
|
||||||
|
|
||||||
|
const blockContentStyle = React.useMemo<React.CSSProperties>(() => {
|
||||||
|
const retVal: React.CSSProperties = {
|
||||||
|
pointerEvents: disablePointerEvents ? "none" : undefined,
|
||||||
|
};
|
||||||
|
if (innerRect?.width && innerRect.height && blockContentOffset) {
|
||||||
|
retVal.width = `calc(${innerRect?.width} - ${blockContentOffset.width}px)`;
|
||||||
|
retVal.height = `calc(${innerRect?.height} - ${blockContentOffset.height}px)`;
|
||||||
|
}
|
||||||
|
return retVal;
|
||||||
|
}, [innerRect, disablePointerEvents, blockContentOffset]);
|
||||||
|
|
||||||
const viewElem = React.useMemo(
|
const viewElem = React.useMemo(
|
||||||
() => getViewElem(nodeModel.blockId, blockData?.meta?.view, viewModel),
|
() => getViewElem(nodeModel.blockId, blockData?.meta?.view, viewModel),
|
||||||
[nodeModel.blockId, blockData?.meta?.view, viewModel]
|
[nodeModel.blockId, blockData?.meta?.view, viewModel]
|
||||||
@ -195,16 +221,7 @@ const BlockFull = React.memo(({ nodeModel, viewModel }: FullBlockProps) => {
|
|||||||
onChange={() => {}}
|
onChange={() => {}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div key="content" className="block-content" ref={contentRef} style={blockContentStyle}>
|
||||||
key="content"
|
|
||||||
className="block-content"
|
|
||||||
style={{
|
|
||||||
pointerEvents: disablePointerEvents ? "none" : undefined,
|
|
||||||
width: addlProps?.transform?.width,
|
|
||||||
height: addlProps?.transform?.height,
|
|
||||||
}}
|
|
||||||
ref={contentRef}
|
|
||||||
>
|
|
||||||
<ErrorBoundary>
|
<ErrorBoundary>
|
||||||
<React.Suspense fallback={<CenteredDiv>Loading...</CenteredDiv>}>{viewElem}</React.Suspense>
|
<React.Suspense fallback={<CenteredDiv>Loading...</CenteredDiv>}>{viewElem}</React.Suspense>
|
||||||
</ErrorBoundary>
|
</ErrorBoundary>
|
||||||
|
@ -402,7 +402,7 @@ const BlockFrame = React.memo((props: BlockFrameProps) => {
|
|||||||
if (!blockId || !blockData) {
|
if (!blockId || !blockData) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
let FrameElem = BlockFrame_Default;
|
const FrameElem = BlockFrame_Default;
|
||||||
const numBlocks = tabData?.blockids?.length ?? 0;
|
const numBlocks = tabData?.blockids?.length ?? 0;
|
||||||
return <FrameElem {...props} numBlocksInTab={numBlocks} />;
|
return <FrameElem {...props} numBlocksInTab={numBlocks} />;
|
||||||
});
|
});
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
justify-content: center;
|
justify-content: center;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
position: relative;
|
position: relative;
|
||||||
padding: 3px 3px 3px 0;
|
padding-right: 3px;
|
||||||
|
|
||||||
.block-container {
|
.block-container {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
@ -8,6 +8,7 @@ import {
|
|||||||
getLayoutModelForActiveTab,
|
getLayoutModelForActiveTab,
|
||||||
getLayoutModelForTab,
|
getLayoutModelForTab,
|
||||||
getLayoutModelForTabById,
|
getLayoutModelForTabById,
|
||||||
|
useDebouncedNodeInnerRect,
|
||||||
useLayoutModel,
|
useLayoutModel,
|
||||||
} from "./lib/layoutModelHooks";
|
} from "./lib/layoutModelHooks";
|
||||||
import { newLayoutNode } from "./lib/layoutNode";
|
import { newLayoutNode } from "./lib/layoutNode";
|
||||||
@ -44,6 +45,7 @@ export {
|
|||||||
NavigateDirection,
|
NavigateDirection,
|
||||||
newLayoutNode,
|
newLayoutNode,
|
||||||
TileLayout,
|
TileLayout,
|
||||||
|
useDebouncedNodeInnerRect,
|
||||||
useLayoutModel,
|
useLayoutModel,
|
||||||
};
|
};
|
||||||
export type {
|
export type {
|
||||||
|
@ -102,8 +102,12 @@ function TileLayoutComponent({ tabAtom, contents, getCursorPoint }: TileLayoutPr
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const tileStyle = useMemo(
|
const tileStyle = useMemo(
|
||||||
() => ({ "--gap-size-px": `${layoutModel.gapSizePx}px` }) as CSSProperties,
|
() =>
|
||||||
[layoutModel.gapSizePx]
|
({
|
||||||
|
"--gap-size-px": `${layoutModel.gapSizePx}px`,
|
||||||
|
"--animation-time-s": `${layoutModel.animationTimeS}s`,
|
||||||
|
}) as CSSProperties,
|
||||||
|
[layoutModel.gapSizePx, layoutModel.animationTimeS]
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@ -55,6 +55,7 @@ interface ResizeContext {
|
|||||||
|
|
||||||
const DefaultGapSizePx = 5;
|
const DefaultGapSizePx = 5;
|
||||||
const MinNodeSizePx = 40;
|
const MinNodeSizePx = 40;
|
||||||
|
const DefaultAnimationTimeS = 0.15;
|
||||||
|
|
||||||
export class LayoutModel {
|
export class LayoutModel {
|
||||||
/**
|
/**
|
||||||
@ -94,6 +95,11 @@ export class LayoutModel {
|
|||||||
*/
|
*/
|
||||||
gapSizePx: number;
|
gapSizePx: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The time a transition animation takes, in seconds.
|
||||||
|
*/
|
||||||
|
animationTimeS: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* List of nodes that are leafs and should be rendered as a DisplayNode.
|
* List of nodes that are leafs and should be rendered as a DisplayNode.
|
||||||
*/
|
*/
|
||||||
@ -102,6 +108,10 @@ export class LayoutModel {
|
|||||||
* An ordered list of node ids starting from the top left corner to the bottom right corner.
|
* An ordered list of node ids starting from the top left corner to the bottom right corner.
|
||||||
*/
|
*/
|
||||||
leafOrder: PrimitiveAtom<string[]>;
|
leafOrder: PrimitiveAtom<string[]>;
|
||||||
|
/**
|
||||||
|
* Atom representing the number of leaf nodes in a layout.
|
||||||
|
*/
|
||||||
|
numLeafs: Atom<number>;
|
||||||
/**
|
/**
|
||||||
* A map of node models for currently-active leafs.
|
* A map of node models for currently-active leafs.
|
||||||
*/
|
*/
|
||||||
@ -197,7 +207,8 @@ export class LayoutModel {
|
|||||||
renderContent?: ContentRenderer,
|
renderContent?: ContentRenderer,
|
||||||
renderPreview?: PreviewRenderer,
|
renderPreview?: PreviewRenderer,
|
||||||
onNodeDelete?: (data: TabLayoutData) => Promise<void>,
|
onNodeDelete?: (data: TabLayoutData) => Promise<void>,
|
||||||
gapSizePx?: number
|
gapSizePx?: number,
|
||||||
|
animationTimeS?: number
|
||||||
) {
|
) {
|
||||||
this.treeStateAtom = treeStateAtom;
|
this.treeStateAtom = treeStateAtom;
|
||||||
this.getter = getter;
|
this.getter = getter;
|
||||||
@ -208,9 +219,11 @@ export class LayoutModel {
|
|||||||
this.gapSizePx = gapSizePx ?? DefaultGapSizePx;
|
this.gapSizePx = gapSizePx ?? DefaultGapSizePx;
|
||||||
this.halfResizeHandleSizePx = this.gapSizePx > 5 ? this.gapSizePx : DefaultGapSizePx;
|
this.halfResizeHandleSizePx = this.gapSizePx > 5 ? this.gapSizePx : DefaultGapSizePx;
|
||||||
this.resizeHandleSizePx = 2 * this.halfResizeHandleSizePx;
|
this.resizeHandleSizePx = 2 * this.halfResizeHandleSizePx;
|
||||||
|
this.animationTimeS = animationTimeS ?? DefaultAnimationTimeS;
|
||||||
|
|
||||||
this.leafs = atom([]);
|
this.leafs = atom([]);
|
||||||
this.leafOrder = atom([]);
|
this.leafOrder = atom([]);
|
||||||
|
this.numLeafs = atom((get) => get(this.leafOrder).length);
|
||||||
|
|
||||||
this.nodeModels = new Map();
|
this.nodeModels = new Map();
|
||||||
this.additionalProps = atom({});
|
this.additionalProps = atom({});
|
||||||
@ -639,12 +652,27 @@ export class LayoutModel {
|
|||||||
getNodeModel(node: LayoutNode): NodeModel {
|
getNodeModel(node: LayoutNode): NodeModel {
|
||||||
const nodeid = node.id;
|
const nodeid = node.id;
|
||||||
const blockId = node.data.blockId;
|
const blockId = node.data.blockId;
|
||||||
|
const addlPropsAtom = this.getNodeAdditionalPropertiesAtom(nodeid);
|
||||||
if (!this.nodeModels.has(nodeid)) {
|
if (!this.nodeModels.has(nodeid)) {
|
||||||
this.nodeModels.set(nodeid, {
|
this.nodeModels.set(nodeid, {
|
||||||
additionalProps: this.getNodeAdditionalPropertiesAtom(nodeid),
|
additionalProps: addlPropsAtom,
|
||||||
|
animationTimeS: this.animationTimeS,
|
||||||
|
innerRect: atom((get) => {
|
||||||
|
const addlProps = get(addlPropsAtom);
|
||||||
|
const numLeafs = get(this.numLeafs);
|
||||||
|
if (numLeafs > 1 && addlProps?.rect) {
|
||||||
|
return {
|
||||||
|
width: `${addlProps.transform.width} - ${this.gapSizePx}px`,
|
||||||
|
height: `${addlProps.transform.height} - ${this.gapSizePx}px`,
|
||||||
|
} as CSSProperties;
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}),
|
||||||
nodeId: nodeid,
|
nodeId: nodeid,
|
||||||
blockId,
|
blockId,
|
||||||
blockNum: atom((get) => get(this.leafOrder).indexOf(nodeid) + 1),
|
blockNum: atom((get) => get(this.leafOrder).indexOf(nodeid) + 1),
|
||||||
|
isResizing: this.isResizing,
|
||||||
isFocused: atom((get) => {
|
isFocused: atom((get) => {
|
||||||
const treeState = get(this.treeStateAtom);
|
const treeState = get(this.treeStateAtom);
|
||||||
const isFocused = treeState.focusedNodeId === nodeid;
|
const isFocused = treeState.focusedNodeId === nodeid;
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
import { atoms, globalStore, WOS } from "@/app/store/global";
|
import { atoms, globalStore, WOS } from "@/app/store/global";
|
||||||
import useResizeObserver from "@react-hook/resize-observer";
|
import useResizeObserver from "@react-hook/resize-observer";
|
||||||
import { Atom, useAtomValue } from "jotai";
|
import { Atom, useAtomValue } from "jotai";
|
||||||
import { useEffect } from "react";
|
import { CSSProperties, useEffect, useState } from "react";
|
||||||
import { withLayoutTreeStateAtomFromTab } from "./layoutAtom";
|
import { withLayoutTreeStateAtomFromTab } from "./layoutAtom";
|
||||||
import { LayoutModel } from "./layoutModel";
|
import { LayoutModel } from "./layoutModel";
|
||||||
import { LayoutNode, NodeModel, TileLayoutContents } from "./types";
|
import { LayoutNode, NodeModel, TileLayoutContents } from "./types";
|
||||||
@ -59,3 +59,30 @@ export function useTileLayout(tabAtom: Atom<Tab>, tileContent: TileLayoutContent
|
|||||||
export function useNodeModel(layoutModel: LayoutModel, layoutNode: LayoutNode): NodeModel {
|
export function useNodeModel(layoutModel: LayoutModel, layoutNode: LayoutNode): NodeModel {
|
||||||
return layoutModel.getNodeModel(layoutNode);
|
return layoutModel.getNodeModel(layoutNode);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function useDebouncedNodeInnerRect(nodeModel: NodeModel): CSSProperties {
|
||||||
|
const nodeInnerRect = useAtomValue(nodeModel.innerRect);
|
||||||
|
const nodeIsResizing = useAtomValue(nodeModel.isResizing);
|
||||||
|
|
||||||
|
const [debounceTimeout, setDebounceTimeout] = useState<NodeJS.Timeout>();
|
||||||
|
const [innerRect, setInnerRect] = useState<CSSProperties>();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!nodeIsResizing && nodeInnerRect) {
|
||||||
|
if (debounceTimeout) {
|
||||||
|
clearTimeout(debounceTimeout);
|
||||||
|
}
|
||||||
|
setDebounceTimeout(
|
||||||
|
setTimeout(() => {
|
||||||
|
console.log("setting inner rect", nodeInnerRect);
|
||||||
|
setInnerRect(nodeInnerRect);
|
||||||
|
setDebounceTimeout(null);
|
||||||
|
}, nodeModel.animationTimeS * 1000)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
setInnerRect(null);
|
||||||
|
}
|
||||||
|
}, [nodeInnerRect, nodeIsResizing]);
|
||||||
|
|
||||||
|
return innerRect;
|
||||||
|
}
|
||||||
|
@ -99,12 +99,20 @@
|
|||||||
border: 1px solid var(--accent-color);
|
border: 1px solid var(--accent-color);
|
||||||
backdrop-filter: blur(8px);
|
backdrop-filter: blur(8px);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tile-leaf {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:not(:only-child) .tile-leaf {
|
||||||
|
padding: calc(var(--gap-size-px) / 2);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&.animate {
|
&.animate {
|
||||||
.tile-node,
|
.tile-node,
|
||||||
.placeholder {
|
.placeholder {
|
||||||
transition-duration: 0.15s;
|
transition-duration: var(--animation-time-s);
|
||||||
transition-timing-function: ease-in;
|
transition-timing-function: ease-in;
|
||||||
transition-property: transform, width, height, background-color;
|
transition-property: transform, width, height, background-color;
|
||||||
}
|
}
|
||||||
@ -116,11 +124,6 @@
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tile-leaf {
|
|
||||||
overflow: hidden;
|
|
||||||
padding: calc(var(--gap-size-px) / 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
.placeholder {
|
.placeholder {
|
||||||
background-color: var(--accent-color);
|
background-color: var(--accent-color);
|
||||||
opacity: 0.5;
|
opacity: 0.5;
|
||||||
|
@ -316,9 +316,12 @@ export interface LayoutNodeAdditionalProps {
|
|||||||
|
|
||||||
export interface NodeModel {
|
export interface NodeModel {
|
||||||
additionalProps: Atom<LayoutNodeAdditionalProps>;
|
additionalProps: Atom<LayoutNodeAdditionalProps>;
|
||||||
|
animationTimeS: number;
|
||||||
|
innerRect: Atom<CSSProperties>;
|
||||||
blockNum: Atom<number>;
|
blockNum: Atom<number>;
|
||||||
nodeId: string;
|
nodeId: string;
|
||||||
blockId: string;
|
blockId: string;
|
||||||
|
isResizing: Atom<boolean>;
|
||||||
isFocused: Atom<boolean>;
|
isFocused: Atom<boolean>;
|
||||||
isMagnified: Atom<boolean>;
|
isMagnified: Atom<boolean>;
|
||||||
ready: Atom<boolean>;
|
ready: Atom<boolean>;
|
||||||
|
Loading…
Reference in New Issue
Block a user