mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
Do not render Tooltip and Menu elements until needed (#5168)
* Clean up Menu DOM elements Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Clean up Tooltip DOM Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Do not render Animate when not in need Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Update snapshots Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * clean up <Animate/> and <Tooltip/> Signed-off-by: Roman <ixrock@gmail.com> Co-authored-by: Roman <ixrock@gmail.com>
This commit is contained in:
parent
ca21e0842a
commit
93a8d0f157
@ -83,19 +83,23 @@ export class Animate extends React.Component<AnimateProps> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
|
if (!this.isVisible) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
const { name, enterDuration, leaveDuration } = this.props;
|
const { name, enterDuration, leaveDuration } = this.props;
|
||||||
const contentElem = this.contentElem;
|
const contentElem = this.contentElem;
|
||||||
const durations = {
|
const cssVarsForAnimation = {
|
||||||
"--enter-duration": `${enterDuration}ms`,
|
"--enter-duration": `${enterDuration}ms`,
|
||||||
"--leave-duration": `${leaveDuration}ms`,
|
"--leave-duration": `${leaveDuration}ms`,
|
||||||
} as React.CSSProperties;
|
} as React.CSSProperties;
|
||||||
|
|
||||||
return React.cloneElement(contentElem, {
|
return React.cloneElement(contentElem, {
|
||||||
className: cssNames("Animate", name, contentElem.props.className, this.statusClassName),
|
className: cssNames("Animate", name, contentElem.props.className, this.statusClassName),
|
||||||
children: this.isVisible ? contentElem.props.children : null,
|
children: contentElem.props.children,
|
||||||
style: {
|
style: {
|
||||||
...contentElem.props.style,
|
...contentElem.props.style,
|
||||||
...durations,
|
...cssVarsForAnimation,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@ -37,10 +37,6 @@ exports[`kube-object-menu given kube object renders 1`] = `
|
|||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div
|
|
||||||
class="Animate opacity-scale Dialog flex center ConfirmDialog modal"
|
|
||||||
style="--enter-duration: 100ms; --leave-duration: 100ms;"
|
|
||||||
/>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
|||||||
@ -16,58 +16,6 @@ exports[`kube-object-status-icon given info and warning statuses are present, wh
|
|||||||
<div />
|
<div />
|
||||||
</i>
|
</i>
|
||||||
</div>
|
</div>
|
||||||
<div
|
|
||||||
class="Tooltip narrow formatter"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="KubeObjectStatusTooltip"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="level warning"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Warning
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
class="status msg"
|
|
||||||
>
|
|
||||||
-
|
|
||||||
Some warning status for some-name
|
|
||||||
|
|
||||||
<span
|
|
||||||
class="age"
|
|
||||||
>
|
|
||||||
·
|
|
||||||
2d
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="level info"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Info
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
class="status msg"
|
|
||||||
>
|
|
||||||
-
|
|
||||||
Some info status for some-name
|
|
||||||
|
|
||||||
<span
|
|
||||||
class="age"
|
|
||||||
>
|
|
||||||
·
|
|
||||||
2d
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
@ -87,36 +35,6 @@ exports[`kube-object-status-icon given level "critical" status, when rendered, r
|
|||||||
<div />
|
<div />
|
||||||
</i>
|
</i>
|
||||||
</div>
|
</div>
|
||||||
<div
|
|
||||||
class="Tooltip narrow formatter"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="KubeObjectStatusTooltip"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="level error"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Critical
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
class="status msg"
|
|
||||||
>
|
|
||||||
-
|
|
||||||
Some critical status for some-name
|
|
||||||
|
|
||||||
<span
|
|
||||||
class="age"
|
|
||||||
>
|
|
||||||
·
|
|
||||||
2d
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
@ -136,36 +54,6 @@ exports[`kube-object-status-icon given level "info" status, when rendered, rende
|
|||||||
<div />
|
<div />
|
||||||
</i>
|
</i>
|
||||||
</div>
|
</div>
|
||||||
<div
|
|
||||||
class="Tooltip narrow formatter"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="KubeObjectStatusTooltip"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="level info"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Info
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
class="status msg"
|
|
||||||
>
|
|
||||||
-
|
|
||||||
Some info status for some-name
|
|
||||||
|
|
||||||
<span
|
|
||||||
class="age"
|
|
||||||
>
|
|
||||||
·
|
|
||||||
2d
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
@ -185,36 +73,6 @@ exports[`kube-object-status-icon given level "warning" status, when rendered, re
|
|||||||
<div />
|
<div />
|
||||||
</i>
|
</i>
|
||||||
</div>
|
</div>
|
||||||
<div
|
|
||||||
class="Tooltip narrow formatter"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="KubeObjectStatusTooltip"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="level warning"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Warning
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
class="status msg"
|
|
||||||
>
|
|
||||||
-
|
|
||||||
Some warning status for some-name
|
|
||||||
|
|
||||||
<span
|
|
||||||
class="age"
|
|
||||||
>
|
|
||||||
·
|
|
||||||
2d
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
@ -254,79 +112,5 @@ exports[`kube-object-status-icon given status for all levels is present, when re
|
|||||||
<div />
|
<div />
|
||||||
</i>
|
</i>
|
||||||
</div>
|
</div>
|
||||||
<div
|
|
||||||
class="Tooltip narrow formatter"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="KubeObjectStatusTooltip"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="level error"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Critical
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
class="status msg"
|
|
||||||
>
|
|
||||||
-
|
|
||||||
Some critical status for some-name
|
|
||||||
|
|
||||||
<span
|
|
||||||
class="age"
|
|
||||||
>
|
|
||||||
·
|
|
||||||
2d
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="level warning"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Warning
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
class="status msg"
|
|
||||||
>
|
|
||||||
-
|
|
||||||
Some warning status for some-name
|
|
||||||
|
|
||||||
<span
|
|
||||||
class="age"
|
|
||||||
>
|
|
||||||
·
|
|
||||||
2d
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="level info"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Info
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
class="status msg"
|
|
||||||
>
|
|
||||||
-
|
|
||||||
Some info status for some-name
|
|
||||||
|
|
||||||
<span
|
|
||||||
class="age"
|
|
||||||
>
|
|
||||||
·
|
|
||||||
2d
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|||||||
@ -93,7 +93,6 @@ export class Menu extends React.Component<MenuProps, State> {
|
|||||||
this.opener.addEventListener(this.props.toggleEvent, this.toggle);
|
this.opener.addEventListener(this.props.toggleEvent, this.toggle);
|
||||||
this.opener.addEventListener("keydown", this.onKeyDown);
|
this.opener.addEventListener("keydown", this.onKeyDown);
|
||||||
}
|
}
|
||||||
this.elem.addEventListener("keydown", this.onKeyDown);
|
|
||||||
window.addEventListener("resize", this.onWindowResize);
|
window.addEventListener("resize", this.onWindowResize);
|
||||||
window.addEventListener("click", this.onClickOutside, true);
|
window.addEventListener("click", this.onClickOutside, true);
|
||||||
window.addEventListener("scroll", this.onScrollOutside, true);
|
window.addEventListener("scroll", this.onScrollOutside, true);
|
||||||
@ -106,7 +105,6 @@ export class Menu extends React.Component<MenuProps, State> {
|
|||||||
this.opener.removeEventListener(this.props.toggleEvent, this.toggle);
|
this.opener.removeEventListener(this.props.toggleEvent, this.toggle);
|
||||||
this.opener.removeEventListener("keydown", this.onKeyDown);
|
this.opener.removeEventListener("keydown", this.onKeyDown);
|
||||||
}
|
}
|
||||||
this.elem.removeEventListener("keydown", this.onKeyDown);
|
|
||||||
window.removeEventListener("resize", this.onWindowResize);
|
window.removeEventListener("resize", this.onWindowResize);
|
||||||
window.removeEventListener("click", this.onClickOutside, true);
|
window.removeEventListener("click", this.onClickOutside, true);
|
||||||
window.removeEventListener("scroll", this.onScrollOutside, true);
|
window.removeEventListener("scroll", this.onScrollOutside, true);
|
||||||
@ -218,7 +216,7 @@ export class Menu extends React.Component<MenuProps, State> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
onKeyDown(evt: KeyboardEvent) {
|
onKeyDown(evt: React.KeyboardEvent | KeyboardEvent) {
|
||||||
if (!this.isOpen) return;
|
if (!this.isOpen) return;
|
||||||
|
|
||||||
switch (evt.code) {
|
switch (evt.code) {
|
||||||
@ -330,6 +328,7 @@ export class Menu extends React.Component<MenuProps, State> {
|
|||||||
left: this.state?.menuStyle?.left,
|
left: this.state?.menuStyle?.left,
|
||||||
top: this.state?.menuStyle?.top,
|
top: this.state?.menuStyle?.top,
|
||||||
}}
|
}}
|
||||||
|
onKeyDown={this.onKeyDown}
|
||||||
>
|
>
|
||||||
{menuItems}
|
{menuItems}
|
||||||
</ul>
|
</ul>
|
||||||
|
|||||||
@ -9,7 +9,7 @@ import React from "react";
|
|||||||
import { createPortal } from "react-dom";
|
import { createPortal } from "react-dom";
|
||||||
import { observer } from "mobx-react";
|
import { observer } from "mobx-react";
|
||||||
import { boundMethod, cssNames, IClassName } from "../../utils";
|
import { boundMethod, cssNames, IClassName } from "../../utils";
|
||||||
import { observable, makeObservable } from "mobx";
|
import { observable, makeObservable, action } from "mobx";
|
||||||
|
|
||||||
export enum TooltipPosition {
|
export enum TooltipPosition {
|
||||||
TOP = "top",
|
TOP = "top",
|
||||||
@ -54,7 +54,8 @@ export class Tooltip extends React.Component<TooltipProps> {
|
|||||||
|
|
||||||
@observable.ref elem: HTMLElement;
|
@observable.ref elem: HTMLElement;
|
||||||
@observable activePosition: TooltipPosition;
|
@observable activePosition: TooltipPosition;
|
||||||
@observable isVisible = false;
|
@observable isVisible = this.props.visible ?? false;
|
||||||
|
@observable isContentVisible = false; // animation manager
|
||||||
|
|
||||||
constructor(props: TooltipProps) {
|
constructor(props: TooltipProps) {
|
||||||
super(props);
|
super(props);
|
||||||
@ -87,15 +88,16 @@ export class Tooltip extends React.Component<TooltipProps> {
|
|||||||
this.hoverTarget.removeEventListener("mouseleave", this.onLeaveTarget);
|
this.hoverTarget.removeEventListener("mouseleave", this.onLeaveTarget);
|
||||||
}
|
}
|
||||||
|
|
||||||
@boundMethod
|
@action.bound
|
||||||
protected onEnterTarget() {
|
protected onEnterTarget() {
|
||||||
this.isVisible = true;
|
this.isVisible = true;
|
||||||
this.refreshPosition();
|
requestAnimationFrame(action(() => this.isContentVisible = true));
|
||||||
}
|
}
|
||||||
|
|
||||||
@boundMethod
|
@action.bound
|
||||||
protected onLeaveTarget() {
|
protected onLeaveTarget() {
|
||||||
this.isVisible = false;
|
this.isVisible = false;
|
||||||
|
this.isContentVisible = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@boundMethod
|
@boundMethod
|
||||||
@ -103,6 +105,10 @@ export class Tooltip extends React.Component<TooltipProps> {
|
|||||||
const { preferredPositions } = this.props;
|
const { preferredPositions } = this.props;
|
||||||
const { elem, targetElem } = this;
|
const { elem, targetElem } = this;
|
||||||
|
|
||||||
|
if (!elem) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
let positions = new Set<TooltipPosition>([
|
let positions = new Set<TooltipPosition>([
|
||||||
TooltipPosition.RIGHT,
|
TooltipPosition.RIGHT,
|
||||||
TooltipPosition.BOTTOM,
|
TooltipPosition.BOTTOM,
|
||||||
@ -150,6 +156,10 @@ export class Tooltip extends React.Component<TooltipProps> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected setPosition(pos: { left: number; top: number }) {
|
protected setPosition(pos: { left: number; top: number }) {
|
||||||
|
if (!this.elem) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const elemStyle = this.elem.style;
|
const elemStyle = this.elem.style;
|
||||||
|
|
||||||
elemStyle.left = `${pos.left}px`;
|
elemStyle.left = `${pos.left}px`;
|
||||||
@ -214,13 +224,17 @@ export class Tooltip extends React.Component<TooltipProps> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { style, formatters, usePortal, children, visible } = this.props;
|
if (!this.isVisible) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { style, formatters, usePortal, children } = this.props;
|
||||||
const className = cssNames("Tooltip", this.props.className, formatters, this.activePosition, {
|
const className = cssNames("Tooltip", this.props.className, formatters, this.activePosition, {
|
||||||
visible: visible ?? this.isVisible,
|
visible: this.isContentVisible,
|
||||||
formatter: !!formatters,
|
formatter: !!formatters,
|
||||||
});
|
});
|
||||||
const tooltip = (
|
const tooltip = (
|
||||||
<div className={className} style={style} ref={this.bindRef}>
|
<div className={className} style={style} ref={this.bindRef} role="tooltip">
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user