mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
159 lines
3.6 KiB
TypeScript
159 lines
3.6 KiB
TypeScript
import "./dialog.scss";
|
|
|
|
import * as React from "react";
|
|
import { createPortal, findDOMNode } from "react-dom";
|
|
import { disposeOnUnmount, observer } from "mobx-react";
|
|
import { reaction } from "mobx";
|
|
import { Animate } from "../animate";
|
|
import { cssNames, noop, stopPropagation } from "../../utils";
|
|
import { navigation } from "../../navigation";
|
|
|
|
export interface DialogProps {
|
|
className?: string;
|
|
isOpen?: boolean;
|
|
open?: () => void;
|
|
close?: () => void;
|
|
onOpen?: () => void;
|
|
onClose?: () => void;
|
|
modal?: boolean;
|
|
pinned?: boolean;
|
|
animated?: boolean;
|
|
}
|
|
|
|
interface DialogState {
|
|
isOpen: boolean;
|
|
}
|
|
|
|
// fixme: handle animation end props.onClose() (await props.close()?)
|
|
@observer
|
|
export class Dialog extends React.PureComponent<DialogProps, DialogState> {
|
|
private contentElem: HTMLElement;
|
|
|
|
static defaultProps: DialogProps = {
|
|
isOpen: false,
|
|
open: noop,
|
|
close: noop,
|
|
onOpen: noop,
|
|
onClose: noop,
|
|
modal: true,
|
|
animated: true,
|
|
pinned: false,
|
|
};
|
|
|
|
@disposeOnUnmount
|
|
closeOnNavigate = reaction(() => navigation.getPath(), () => this.close())
|
|
|
|
public state: DialogState = {
|
|
isOpen: this.props.isOpen,
|
|
}
|
|
|
|
get elem(): HTMLElement {
|
|
// eslint-disable-next-line react/no-find-dom-node
|
|
return findDOMNode(this) as HTMLElement;
|
|
}
|
|
|
|
get isOpen(): boolean {
|
|
return this.state.isOpen;
|
|
}
|
|
|
|
componentDidMount(): void {
|
|
if (this.isOpen) {
|
|
this.onOpen();
|
|
}
|
|
}
|
|
|
|
componentDidUpdate(prevProps: DialogProps): void {
|
|
const { isOpen } = this.props;
|
|
if (isOpen !== prevProps.isOpen) {
|
|
this.toggle(isOpen);
|
|
}
|
|
}
|
|
|
|
componentWillUnmount(): void {
|
|
if (this.isOpen) {
|
|
this.onClose();
|
|
}
|
|
}
|
|
|
|
toggle(isOpen: boolean): void {
|
|
if (isOpen) {
|
|
this.open();
|
|
} else {
|
|
this.close();
|
|
}
|
|
}
|
|
|
|
open(): void {
|
|
requestAnimationFrame(this.onOpen); // wait for render(), bind close-event to this.elem
|
|
this.setState({ isOpen: true });
|
|
this.props.open();
|
|
}
|
|
|
|
close(): void {
|
|
this.onClose(); // must be first to get access to dialog's content from outside
|
|
this.setState({ isOpen: false });
|
|
this.props.close();
|
|
}
|
|
|
|
onOpen = (): void => {
|
|
this.props.onOpen();
|
|
if (!this.props.pinned) {
|
|
if (this.elem) {
|
|
this.elem.addEventListener('click', this.onClickOutside);
|
|
}
|
|
window.addEventListener('keydown', this.onEscapeKey);
|
|
}
|
|
}
|
|
|
|
onClose = (): void => {
|
|
this.props.onClose();
|
|
if (!this.props.pinned) {
|
|
if (this.elem) {
|
|
this.elem.removeEventListener('click', this.onClickOutside);
|
|
}
|
|
window.removeEventListener('keydown', this.onEscapeKey);
|
|
}
|
|
}
|
|
|
|
onEscapeKey = (evt: KeyboardEvent): void=> {
|
|
const escapeKey = evt.code === "Escape";
|
|
if (escapeKey) {
|
|
this.close();
|
|
evt.stopPropagation();
|
|
}
|
|
}
|
|
|
|
onClickOutside = (evt: MouseEvent): void => {
|
|
const target = evt.target as HTMLElement;
|
|
if (!this.contentElem.contains(target)) {
|
|
this.close();
|
|
evt.stopPropagation();
|
|
}
|
|
}
|
|
|
|
render(): JSX.Element {
|
|
const { modal, animated, pinned } = this.props;
|
|
let { className } = this.props;
|
|
className = cssNames("Dialog flex center", className, { modal, pinned });
|
|
let dialog = (
|
|
<div className={className} onClick={stopPropagation}>
|
|
<div className="box" ref={(e): void => {
|
|
this.contentElem = e;
|
|
}}>
|
|
{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);
|
|
}
|
|
}
|