mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
Fix flakiness in unit test when using <Animated>
Signed-off-by: Sebastian Malton <sebastian@malton.name>
This commit is contained in:
parent
b8538e87b5
commit
7e47c633bf
@ -3116,60 +3116,6 @@ exports[`New Upgrade Helm Chart Dock Tab given a namespace is selected when navi
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ul
|
|
||||||
class="Animate opacity Menu MenuActions flex HelmReleaseMenu right bottom portal enter leave"
|
|
||||||
id="menu-actions-for-release-menu-for-my-second-namespace/some-name"
|
|
||||||
style="--enter-duration: 100ms; --leave-duration: 100ms;"
|
|
||||||
>
|
|
||||||
<li
|
|
||||||
class="MenuItem"
|
|
||||||
data-testid="upgrade-chart-menu-item-for-my-second-namespace/some-name"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<i
|
|
||||||
class="Icon material focusable"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="icon"
|
|
||||||
data-icon-name="refresh"
|
|
||||||
>
|
|
||||||
refresh
|
|
||||||
</span>
|
|
||||||
</i>
|
|
||||||
<div>
|
|
||||||
Upgrade
|
|
||||||
</div>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Upgrade
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
<li
|
|
||||||
class="MenuItem"
|
|
||||||
data-testid="menu-action-remove"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<i
|
|
||||||
class="Icon material focusable"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="icon"
|
|
||||||
data-icon-name="delete"
|
|
||||||
>
|
|
||||||
delete
|
|
||||||
</span>
|
|
||||||
</i>
|
|
||||||
<div>
|
|
||||||
Delete
|
|
||||||
</div>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Delete
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
@ -4167,60 +4113,6 @@ exports[`New Upgrade Helm Chart Dock Tab given a namespace is selected when navi
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ul
|
|
||||||
class="Animate opacity Menu MenuActions flex HelmReleaseMenu bottom right portal enter leave"
|
|
||||||
id="menu-actions-for-release-menu-for-my-second-namespace/some-name"
|
|
||||||
style="--enter-duration: 100ms; --leave-duration: 100ms; left: 0px; top: 8px;"
|
|
||||||
>
|
|
||||||
<li
|
|
||||||
class="MenuItem"
|
|
||||||
data-testid="upgrade-chart-menu-item-for-my-second-namespace/some-name"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<i
|
|
||||||
class="Icon material focusable"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="icon"
|
|
||||||
data-icon-name="refresh"
|
|
||||||
>
|
|
||||||
refresh
|
|
||||||
</span>
|
|
||||||
</i>
|
|
||||||
<div>
|
|
||||||
Upgrade
|
|
||||||
</div>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Upgrade
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
<li
|
|
||||||
class="MenuItem"
|
|
||||||
data-testid="menu-action-remove"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<i
|
|
||||||
class="Icon material focusable"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="icon"
|
|
||||||
data-icon-name="delete"
|
|
||||||
>
|
|
||||||
delete
|
|
||||||
</span>
|
|
||||||
</i>
|
|
||||||
<div>
|
|
||||||
Delete
|
|
||||||
</div>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Delete
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
@ -5220,60 +5112,6 @@ exports[`New Upgrade Helm Chart Dock Tab given a namespace is selected when navi
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ul
|
|
||||||
class="Animate opacity Menu MenuActions flex HelmReleaseMenu bottom right portal enter leave"
|
|
||||||
id="menu-actions-for-release-menu-for-my-second-namespace/some-name"
|
|
||||||
style="--enter-duration: 100ms; --leave-duration: 100ms; left: 0px; top: 8px;"
|
|
||||||
>
|
|
||||||
<li
|
|
||||||
class="MenuItem"
|
|
||||||
data-testid="upgrade-chart-menu-item-for-my-second-namespace/some-name"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<i
|
|
||||||
class="Icon material focusable"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="icon"
|
|
||||||
data-icon-name="refresh"
|
|
||||||
>
|
|
||||||
refresh
|
|
||||||
</span>
|
|
||||||
</i>
|
|
||||||
<div>
|
|
||||||
Upgrade
|
|
||||||
</div>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Upgrade
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
<li
|
|
||||||
class="MenuItem"
|
|
||||||
data-testid="menu-action-remove"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<i
|
|
||||||
class="Icon material focusable"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="icon"
|
|
||||||
data-icon-name="delete"
|
|
||||||
>
|
|
||||||
delete
|
|
||||||
</span>
|
|
||||||
</i>
|
|
||||||
<div>
|
|
||||||
Delete
|
|
||||||
</div>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Delete
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
@ -6041,59 +5879,5 @@ exports[`New Upgrade Helm Chart Dock Tab given a namespace is selected when navi
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ul
|
|
||||||
class="Animate opacity Menu MenuActions flex HelmReleaseMenu bottom right portal enter leave"
|
|
||||||
id="menu-actions-for-release-menu-for-my-second-namespace/some-name"
|
|
||||||
style="--enter-duration: 100ms; --leave-duration: 100ms; left: 0px; top: 8px;"
|
|
||||||
>
|
|
||||||
<li
|
|
||||||
class="MenuItem"
|
|
||||||
data-testid="upgrade-chart-menu-item-for-my-second-namespace/some-name"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<i
|
|
||||||
class="Icon material focusable"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="icon"
|
|
||||||
data-icon-name="refresh"
|
|
||||||
>
|
|
||||||
refresh
|
|
||||||
</span>
|
|
||||||
</i>
|
|
||||||
<div>
|
|
||||||
Upgrade
|
|
||||||
</div>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Upgrade
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
<li
|
|
||||||
class="MenuItem"
|
|
||||||
data-testid="menu-action-remove"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<i
|
|
||||||
class="Icon material focusable"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="icon"
|
|
||||||
data-icon-name="delete"
|
|
||||||
>
|
|
||||||
delete
|
|
||||||
</span>
|
|
||||||
</i>
|
|
||||||
<div>
|
|
||||||
Delete
|
|
||||||
</div>
|
|
||||||
<span
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
Delete
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</body>
|
</body>
|
||||||
`;
|
`;
|
||||||
|
|||||||
@ -17,6 +17,7 @@ import type { RequestHelmReleaseConfiguration } from "../../../common/k8s-api/en
|
|||||||
import requestHelmReleaseConfigurationInjectable from "../../../common/k8s-api/endpoints/helm-releases.api/request-configuration.injectable";
|
import requestHelmReleaseConfigurationInjectable from "../../../common/k8s-api/endpoints/helm-releases.api/request-configuration.injectable";
|
||||||
import type { RequestHelmReleases } from "../../../common/k8s-api/endpoints/helm-releases.api/request-releases.injectable";
|
import type { RequestHelmReleases } from "../../../common/k8s-api/endpoints/helm-releases.api/request-releases.injectable";
|
||||||
import requestHelmReleasesInjectable from "../../../common/k8s-api/endpoints/helm-releases.api/request-releases.injectable";
|
import requestHelmReleasesInjectable from "../../../common/k8s-api/endpoints/helm-releases.api/request-releases.injectable";
|
||||||
|
import { advanceFakeTime, useFakeTime } from "../../../common/test-utils/use-fake-time";
|
||||||
import dockStoreInjectable from "../../../renderer/components/dock/dock/store.injectable";
|
import dockStoreInjectable from "../../../renderer/components/dock/dock/store.injectable";
|
||||||
import type { ApplicationBuilder } from "../../../renderer/components/test-utils/get-application-builder";
|
import type { ApplicationBuilder } from "../../../renderer/components/test-utils/get-application-builder";
|
||||||
import { getApplicationBuilder } from "../../../renderer/components/test-utils/get-application-builder";
|
import { getApplicationBuilder } from "../../../renderer/components/test-utils/get-application-builder";
|
||||||
@ -50,6 +51,8 @@ describe("New Upgrade Helm Chart Dock Tab", () => {
|
|||||||
navigateToHelmReleases = windowDi.inject(navigateToHelmReleasesInjectable);
|
navigateToHelmReleases = windowDi.inject(navigateToHelmReleasesInjectable);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
useFakeTime("2020-01-12 12:00:00");
|
||||||
|
|
||||||
builder.namespaces.add("my-first-namespace");
|
builder.namespaces.add("my-first-namespace");
|
||||||
builder.namespaces.add("my-second-namespace");
|
builder.namespaces.add("my-second-namespace");
|
||||||
|
|
||||||
@ -114,6 +117,7 @@ describe("New Upgrade Helm Chart Dock Tab", () => {
|
|||||||
const upgradeHelmChartMenuItem = renderResult.getByTestId("upgrade-chart-menu-item-for-my-second-namespace/some-name");
|
const upgradeHelmChartMenuItem = renderResult.getByTestId("upgrade-chart-menu-item-for-my-second-namespace/some-name");
|
||||||
|
|
||||||
upgradeHelmChartMenuItem.click();
|
upgradeHelmChartMenuItem.click();
|
||||||
|
advanceFakeTime(100);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("renders", () => {
|
it("renders", () => {
|
||||||
|
|||||||
@ -4,12 +4,12 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import "./animate.scss";
|
import "./animate.scss";
|
||||||
import React from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { observable, makeObservable } from "mobx";
|
|
||||||
import { observer } from "mobx-react";
|
|
||||||
import { cssNames, noop } from "../../utils";
|
import { cssNames, noop } from "../../utils";
|
||||||
import { withInjectables } from "@ogre-tools/injectable-react";
|
import { withInjectables } from "@ogre-tools/injectable-react";
|
||||||
import requestAnimationFrameInjectable from "./request-animation-frame.injectable";
|
import requestAnimationFrameInjectable from "./request-animation-frame.injectable";
|
||||||
|
import defaultEnterDurationForAnimatedInjectable from "./default-enter-duration.injectable";
|
||||||
|
import defaultLeaveDurationForAnimatedInjectable from "./default-leave-duration.injectable";
|
||||||
|
|
||||||
export type AnimateName = "opacity" | "slide-right" | "opacity-scale" | string;
|
export type AnimateName = "opacity" | "slide-right" | "opacity-scale" | string;
|
||||||
|
|
||||||
@ -25,114 +25,92 @@ export interface AnimateProps {
|
|||||||
|
|
||||||
interface Dependencies {
|
interface Dependencies {
|
||||||
requestAnimationFrame: (callback: () => void) => void;
|
requestAnimationFrame: (callback: () => void) => void;
|
||||||
|
defaultEnterDuration: number;
|
||||||
|
defaultLeaveDuration: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@observer
|
const NonInjectedAnimate = (propsAndDeps: AnimateProps & Dependencies) => {
|
||||||
class DefaultedAnimate extends React.Component<AnimateProps & Dependencies & typeof DefaultedAnimate.defaultProps> {
|
const {
|
||||||
static defaultProps = {
|
requestAnimationFrame,
|
||||||
name: "opacity",
|
defaultEnterDuration,
|
||||||
enter: true,
|
defaultLeaveDuration,
|
||||||
onEnter: noop,
|
...props
|
||||||
onLeave: noop,
|
} = propsAndDeps;
|
||||||
enterDuration: 100,
|
const {
|
||||||
leaveDuration: 100,
|
children,
|
||||||
|
enter = true,
|
||||||
|
enterDuration = defaultEnterDuration,
|
||||||
|
leaveDuration = defaultLeaveDuration,
|
||||||
|
name = "opacity",
|
||||||
|
onEnter: onEnterHandler = noop<[]>,
|
||||||
|
onLeave: onLeaveHandler = noop<[]>,
|
||||||
|
} = props;
|
||||||
|
|
||||||
|
const [isVisible, setIsVisible] = useState(enter);
|
||||||
|
const [showClassNameEnter, setShowClassNameEnter] = useState(false);
|
||||||
|
const [showClassNameLeave, setShowClassNameLeave] = useState(false);
|
||||||
|
|
||||||
|
const contentElem = React.Children.only(children) as React.ReactElement<React.HTMLAttributes<any>>;
|
||||||
|
const onEnter = () => {
|
||||||
|
setIsVisible(true);
|
||||||
|
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
setShowClassNameEnter(true);
|
||||||
|
onEnterHandler();
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
const onLeave = () => {
|
||||||
|
if (isVisible) {
|
||||||
|
setShowClassNameLeave(true);
|
||||||
|
onLeaveHandler();
|
||||||
|
|
||||||
@observable isVisible = !!this.props.enter;
|
// Cleanup after duration
|
||||||
@observable statusClassName = {
|
setTimeout(() => {
|
||||||
enter: false,
|
setIsVisible(false);
|
||||||
leave: false,
|
setShowClassNameEnter(false);
|
||||||
|
setShowClassNameLeave(false);
|
||||||
|
}, leaveDuration);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
const toggle = (entering: boolean) => {
|
||||||
constructor(props: AnimateProps & Dependencies & typeof DefaultedAnimate.defaultProps) {
|
if (entering) {
|
||||||
super(props);
|
onEnter();
|
||||||
makeObservable(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
get contentElem() {
|
|
||||||
return React.Children.only(this.props.children) as React.ReactElement<React.HTMLAttributes<any>>;
|
|
||||||
}
|
|
||||||
|
|
||||||
private toggle(enter: boolean) {
|
|
||||||
if (enter) {
|
|
||||||
this.enter();
|
|
||||||
} else {
|
} else {
|
||||||
this.leave();
|
onLeave();
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => toggle(enter), [enter]);
|
||||||
|
|
||||||
|
if (!isVisible) {
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount() {
|
const cssVarsForAnimation = {
|
||||||
this.toggle(this.props.enter);
|
"--enter-duration": `${enterDuration}ms`,
|
||||||
}
|
"--leave-duration": `${leaveDuration}ms`,
|
||||||
|
} as React.CSSProperties;
|
||||||
|
|
||||||
componentDidUpdate(prevProps: Readonly<AnimateProps>): void {
|
return React.cloneElement(contentElem, {
|
||||||
const { enter } = this.props;
|
className: cssNames("Animate", name, contentElem.props.className, {
|
||||||
|
enter: showClassNameEnter,
|
||||||
if (prevProps.enter !== enter) {
|
leave: showClassNameLeave,
|
||||||
this.toggle(enter);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
enter() {
|
|
||||||
this.isVisible = true; // triggers render() to apply css-animation in existing dom
|
|
||||||
|
|
||||||
this.props.requestAnimationFrame(() => {
|
|
||||||
this.statusClassName.enter = true;
|
|
||||||
this.props.onEnter();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
leave() {
|
|
||||||
if (!this.isVisible) return;
|
|
||||||
this.statusClassName.leave = true;
|
|
||||||
this.props.onLeave();
|
|
||||||
this.resetAfterLeaveDuration();
|
|
||||||
}
|
|
||||||
|
|
||||||
resetAfterLeaveDuration() {
|
|
||||||
setTimeout(() => this.reset(), this.props.leaveDuration);
|
|
||||||
}
|
|
||||||
|
|
||||||
reset() {
|
|
||||||
this.isVisible = false;
|
|
||||||
this.statusClassName.enter = false;
|
|
||||||
this.statusClassName.leave = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
|
||||||
if (!this.isVisible) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { name, enterDuration, leaveDuration } = this.props;
|
|
||||||
const contentElem = this.contentElem;
|
|
||||||
const cssVarsForAnimation = {
|
|
||||||
"--enter-duration": `${enterDuration}ms`,
|
|
||||||
"--leave-duration": `${leaveDuration}ms`,
|
|
||||||
} as React.CSSProperties;
|
|
||||||
|
|
||||||
return React.cloneElement(contentElem, {
|
|
||||||
className: cssNames("Animate", name, contentElem.props.className, this.statusClassName),
|
|
||||||
children: contentElem.props.children,
|
|
||||||
style: {
|
|
||||||
...contentElem.props.style,
|
|
||||||
...cssVarsForAnimation,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export const NonInjectedAnimate = (props: AnimateProps & Dependencies) => <DefaultedAnimate {...props} />;
|
|
||||||
|
|
||||||
export const Animate = withInjectables<Dependencies, AnimateProps>(
|
|
||||||
NonInjectedAnimate,
|
|
||||||
|
|
||||||
{
|
|
||||||
getProps: (di, props) => ({
|
|
||||||
requestAnimationFrame: di.inject(requestAnimationFrameInjectable),
|
|
||||||
...props,
|
|
||||||
}),
|
}),
|
||||||
},
|
children: contentElem.props.children,
|
||||||
);
|
style: {
|
||||||
|
...contentElem.props.style,
|
||||||
|
...cssVarsForAnimation,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Animate = withInjectables<Dependencies, AnimateProps>(NonInjectedAnimate, {
|
||||||
|
getProps: (di, props) => ({
|
||||||
|
...props,
|
||||||
|
requestAnimationFrame: di.inject(requestAnimationFrameInjectable),
|
||||||
|
defaultEnterDuration: di.inject(defaultEnterDurationForAnimatedInjectable),
|
||||||
|
defaultLeaveDuration: di.inject(defaultLeaveDurationForAnimatedInjectable),
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@ -0,0 +1,12 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
||||||
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
||||||
|
*/
|
||||||
|
import { getInjectable } from "@ogre-tools/injectable";
|
||||||
|
|
||||||
|
const defaultEnterDurationForAnimatedInjectable = getInjectable({
|
||||||
|
id: "default-enter-duration-for-animated",
|
||||||
|
instantiate: () => 100,
|
||||||
|
});
|
||||||
|
|
||||||
|
export default defaultEnterDurationForAnimatedInjectable;
|
||||||
@ -0,0 +1,12 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
||||||
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
||||||
|
*/
|
||||||
|
import { getInjectable } from "@ogre-tools/injectable";
|
||||||
|
|
||||||
|
const defaultLeaveDurationForAnimatedInjectable = getInjectable({
|
||||||
|
id: "default-leave-duration-for-animated",
|
||||||
|
instantiate: () => 100,
|
||||||
|
});
|
||||||
|
|
||||||
|
export default defaultLeaveDurationForAnimatedInjectable;
|
||||||
@ -4,9 +4,12 @@
|
|||||||
*/
|
*/
|
||||||
import { getInjectable } from "@ogre-tools/injectable";
|
import { getInjectable } from "@ogre-tools/injectable";
|
||||||
|
|
||||||
|
export type RequestAnimationFrame = (callback: () => void) => void;
|
||||||
|
|
||||||
const requestAnimationFrameInjectable = getInjectable({
|
const requestAnimationFrameInjectable = getInjectable({
|
||||||
id: "request-animation-frame",
|
id: "request-animation-frame",
|
||||||
instantiate: () => (callback: () => void) => requestAnimationFrame(callback),
|
// NOTE: this cannot be simplified to just `=> requestAnimationFrame` else an Illegal Invocation error will be thrown
|
||||||
|
instantiate: (): RequestAnimationFrame => (callback) => requestAnimationFrame(callback),
|
||||||
causesSideEffects: true,
|
causesSideEffects: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@ -13,9 +13,15 @@ import { Animate } from "../animate";
|
|||||||
import type { IconProps } from "../icon";
|
import type { IconProps } from "../icon";
|
||||||
import { Icon } from "../icon";
|
import { Icon } from "../icon";
|
||||||
import isEqual from "lodash/isEqual";
|
import isEqual from "lodash/isEqual";
|
||||||
|
import type { RequestAnimationFrame } from "../animate/request-animation-frame.injectable";
|
||||||
|
import { withInjectables } from "@ogre-tools/injectable-react";
|
||||||
|
import requestAnimationFrameInjectable from "../animate/request-animation-frame.injectable";
|
||||||
|
|
||||||
export const MenuContext = React.createContext<MenuContextValue | null>(null);
|
export const MenuContext = React.createContext<MenuContextValue | null>(null);
|
||||||
export type MenuContextValue = Menu;
|
export interface MenuContextValue {
|
||||||
|
readonly props: Readonly<MenuProps>;
|
||||||
|
close: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
export interface MenuPosition {
|
export interface MenuPosition {
|
||||||
left?: boolean;
|
left?: boolean;
|
||||||
@ -62,10 +68,14 @@ const defaultPropsMenu: Partial<MenuProps> = {
|
|||||||
animated: true,
|
animated: true,
|
||||||
};
|
};
|
||||||
|
|
||||||
export class Menu extends React.Component<MenuProps, State> {
|
interface Dependencies {
|
||||||
|
requestAnimationFrame: RequestAnimationFrame;
|
||||||
|
}
|
||||||
|
|
||||||
|
class NonInjectedMenu extends React.Component<MenuProps & Dependencies, State> {
|
||||||
static defaultProps = defaultPropsMenu as object;
|
static defaultProps = defaultPropsMenu as object;
|
||||||
|
|
||||||
constructor(props: MenuProps) {
|
constructor(props: MenuProps & Dependencies) {
|
||||||
super(props);
|
super(props);
|
||||||
autoBind(this);
|
autoBind(this);
|
||||||
}
|
}
|
||||||
@ -372,6 +382,13 @@ export class Menu extends React.Component<MenuProps, State> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const Menu = withInjectables<Dependencies, MenuProps>(NonInjectedMenu, {
|
||||||
|
getProps: (di, props) => ({
|
||||||
|
...props,
|
||||||
|
requestAnimationFrame: di.inject(requestAnimationFrameInjectable),
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
export function SubMenu(props: Partial<MenuProps>) {
|
export function SubMenu(props: Partial<MenuProps>) {
|
||||||
const { className, ...menuProps } = props;
|
const { className, ...menuProps } = props;
|
||||||
|
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user