mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* Making apiBase injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert all of Helm functions to be DI Signed-off-by: Sebastian Malton <sebastian@malton.name> * Make PortForward's use of apiBase fully injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert all metric requests to be injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Replace resource applier with injectables Signed-off-by: Sebastian Malton <sebastian@malton.name> * Switch KubeJsonApi.forCluster to be injectable but do not use Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert the rest of shell sessions to be DI-ed - This is a prerequesit for using the new createKubeJsonApiForClusterInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Use new createKubeJsonApiForClusterInjectable for openNodeShellSession Signed-off-by: Sebastian Malton <sebastian@malton.name> * Make KubeconfigDialog injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove jest-fetch-mock and make fetch injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests with new global override Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add new injectable for create KubeJsonApi and JsonApi instances Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix showing-details-for-helm-release behavioural tests - Remove HelmChartStore in favour of all injectables - Create a model for UpgradeChartDockTab Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix show details and updating helm releases tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix residual typing issues related to metrics Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix crash on load due to circular dependency Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix create resource tab not working Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove legacy apiBase global Signed-off-by: Sebastian Malton <sebastian@malton.name> * Introduce and use isDebuggingInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Introduce and use windowLocationInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove global legacy apiKube Signed-off-by: Sebastian Malton <sebastian@malton.name> * Improve injectable filenames compared to the injectables inside Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove modifying input in requestActivePortForwardInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Introduce and use get(Milli)SecondsFromUnixEpochInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Switch to non-reactive way of gettting possible helm release versions Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix typo Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix bug in KubeApi constructor Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert all KubeApi related tests to use asyncFn Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix unit tests after introducing new injectables that have side effects Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix bad rebase causing tests to fail Signed-off-by: Sebastian Malton <sebastian@malton.name> * Improve expects for multiple field values Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix crash will looking up api refs Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix breaking change on KubeApi.list Signed-off-by: Sebastian Malton <sebastian@malton.name> * Better fix for formatting urls Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove injectable for time since we should just use useMockTime Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add happy path behavioural tests for upgrade chart tab Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove debug message Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix showing-details-for-helm-release tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix installing-helm-chart-from-new-tab tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests relating to hosted cluster id Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots to recent changes in master Co-authored-by: Janne Savolainen <janne.savolainen@live.fi> Signed-off-by: Iku-turso <mikko.aspiala@gmail.com> * Reupdated upgrade chart new tab test snapshots Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix flakiness in unit test when using <Animated> Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix flakiness and improve tests for DeleteClusterDialog Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix kubeconfig-sync tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix <Extensions> tests by removing mockFs and making everything injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix build issues Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix getElectronAppPathInjectable override not returning absolute paths - Also fixes the listing-active-helm-repos-in-prefs tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Replace all uses of getAbsolutePath with joinPaths as it is more correct and less confusing Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix opening application window tests by making override properly absolute Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots relating no longer using getAbsolutePath Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix and add behavioural tests for RenderDelay Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix extension discovery tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix test flakiness because of path side effects, propagate uses to as many places Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix extension-discovery tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add global override to fix some tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Rewrite and fix implementation of KubeconfigManager and its tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests by global override pathExists Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix unit tests failing on windows by using injectable verions of path functions Signed-off-by: Sebastian Malton <sebastian@malton.name> * Attempt to fix test timeout by using runInAction Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots after rebase Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots after rebase Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests after rebase Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix setupIpcMainHandlers usage Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots Signed-off-by: Sebastian Malton <sebastian@malton.name> Signed-off-by: Sebastian Malton <sebastian@malton.name> Signed-off-by: Iku-turso <mikko.aspiala@gmail.com> Co-authored-by: Iku-turso <mikko.aspiala@gmail.com>
188 lines
4.6 KiB
TypeScript
188 lines
4.6 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
|
|
import "./dialog.scss";
|
|
|
|
import React from "react";
|
|
import { createPortal } from "react-dom";
|
|
import { disposeOnUnmount, observer } from "mobx-react";
|
|
import { reaction } from "mobx";
|
|
import { Animate } from "../animate";
|
|
import { cssNames, noop, stopPropagation } from "../../utils";
|
|
import type { ObservableHistory } from "mobx-observable-history";
|
|
import { withInjectables } from "@ogre-tools/injectable-react";
|
|
import observableHistoryInjectable from "../../navigation/observable-history.injectable";
|
|
import requestAnimationFrameInjectable from "../animate/request-animation-frame.injectable";
|
|
|
|
// todo: refactor + handle animation-end in props.onClose()?
|
|
|
|
export interface DialogProps {
|
|
className?: string;
|
|
isOpen?: boolean;
|
|
open?: () => void;
|
|
close?: () => void;
|
|
onOpen?: () => void;
|
|
onClose?: () => void;
|
|
modal?: boolean;
|
|
pinned?: boolean;
|
|
animated?: boolean;
|
|
"data-testid"?: string;
|
|
children?: React.ReactNode | React.ReactNode[];
|
|
}
|
|
|
|
interface DialogState {
|
|
isOpen: boolean;
|
|
}
|
|
|
|
interface Dependencies {
|
|
navigation: ObservableHistory<unknown>;
|
|
requestAnimationFrame: (callback: () => void) => void;
|
|
}
|
|
|
|
@observer
|
|
class NonInjectedDialog extends React.PureComponent<DialogProps & Dependencies & typeof NonInjectedDialog.defaultProps, DialogState> {
|
|
private readonly contentElem = React.createRef<HTMLDivElement>();
|
|
private readonly ref = React.createRef<HTMLDivElement>();
|
|
|
|
static defaultProps = {
|
|
isOpen: false,
|
|
open: noop,
|
|
close: noop,
|
|
onOpen: noop,
|
|
onClose: noop,
|
|
modal: true,
|
|
animated: true,
|
|
pinned: false,
|
|
};
|
|
|
|
public state: DialogState = {
|
|
isOpen: this.props.isOpen ?? false,
|
|
};
|
|
|
|
get elem() {
|
|
return this.ref.current;
|
|
}
|
|
|
|
get isOpen() {
|
|
return this.state.isOpen;
|
|
}
|
|
|
|
componentDidMount() {
|
|
if (this.isOpen) {
|
|
this.onOpen();
|
|
}
|
|
|
|
disposeOnUnmount(this, [
|
|
reaction(() => this.props.navigation.toString(), () => this.close()),
|
|
]);
|
|
}
|
|
|
|
componentDidUpdate(prevProps: DialogProps) {
|
|
const { isOpen } = this.props;
|
|
|
|
if (isOpen !== prevProps.isOpen) {
|
|
this.toggle(isOpen ?? false);
|
|
}
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
if (this.isOpen) this.onClose();
|
|
}
|
|
|
|
toggle(isOpen: boolean) {
|
|
if (isOpen) this.open();
|
|
else this.close();
|
|
}
|
|
|
|
open() {
|
|
this.props.requestAnimationFrame(this.onOpen); // wait for render(), bind close-event to this.elem
|
|
this.setState({ isOpen: true });
|
|
this.props.open?.();
|
|
}
|
|
|
|
close() {
|
|
this.onClose(); // must be first to get access to dialog's content from outside
|
|
this.setState({ isOpen: false });
|
|
this.props.close?.();
|
|
}
|
|
|
|
onOpen = () => {
|
|
this.props.onOpen?.();
|
|
|
|
if (!this.props.pinned) {
|
|
if (this.elem) this.elem.addEventListener("click", this.onClickOutside);
|
|
// Using document.body target to handle keydown event before Drawer does
|
|
document.body.addEventListener("keydown", this.onEscapeKey);
|
|
}
|
|
};
|
|
|
|
onClose = () => {
|
|
this.props.onClose?.();
|
|
|
|
if (!this.props.pinned) {
|
|
if (this.elem) this.elem.removeEventListener("click", this.onClickOutside);
|
|
document.body.removeEventListener("keydown", this.onEscapeKey);
|
|
}
|
|
};
|
|
|
|
onEscapeKey = (evt: KeyboardEvent) => {
|
|
const escapeKey = evt.code === "Escape";
|
|
|
|
if (escapeKey) {
|
|
this.close();
|
|
evt.stopPropagation();
|
|
}
|
|
};
|
|
|
|
onClickOutside = (evt: MouseEvent) => {
|
|
const target = evt.target as HTMLElement;
|
|
|
|
if (!this.contentElem.current?.contains(target)) {
|
|
this.close();
|
|
evt.stopPropagation();
|
|
}
|
|
};
|
|
|
|
render() {
|
|
const { modal, animated, pinned, "data-testid": testId, className } = this.props;
|
|
|
|
let dialog = (
|
|
<div
|
|
className={cssNames("Dialog flex center", className, { modal, pinned })}
|
|
onClick={stopPropagation}
|
|
ref={this.ref}
|
|
data-testid={testId}
|
|
>
|
|
<div
|
|
className="box"
|
|
ref={this.contentElem}
|
|
>
|
|
{this.props.children}
|
|
</div>
|
|
</div>
|
|
);
|
|
|
|
if (animated) {
|
|
dialog = (
|
|
<Animate enter={this.isOpen} name="opacity-scale">
|
|
{dialog}
|
|
</Animate>
|
|
);
|
|
} else if (!this.isOpen) {
|
|
return null;
|
|
}
|
|
|
|
return createPortal(dialog, document.body);
|
|
}
|
|
}
|
|
|
|
export const Dialog = withInjectables<Dependencies, DialogProps>((props) => <NonInjectedDialog {...props} />, {
|
|
getProps: (di, props) => ({
|
|
...props,
|
|
navigation: di.inject(observableHistoryInjectable),
|
|
requestAnimationFrame: di.inject(requestAnimationFrameInjectable),
|
|
}),
|
|
});
|