2019-12-07 13:28:52 +02:00
|
|
|
import React from 'react';
|
2019-12-07 21:43:08 +02:00
|
|
|
import clsx from 'clsx';
|
2019-12-07 21:02:00 +02:00
|
|
|
import { omit } from 'app/functions';
|
2016-08-07 22:18:11 +03:00
|
|
|
|
2016-01-08 15:14:35 +02:00
|
|
|
import styles from './panel.scss';
|
2016-01-09 13:59:42 +02:00
|
|
|
import icons from './icons.scss';
|
2016-01-08 15:14:35 +02:00
|
|
|
|
2017-08-22 21:39:08 +03:00
|
|
|
export function Panel(props: {
|
2019-12-07 13:28:52 +02:00
|
|
|
title?: string;
|
|
|
|
icon?: string;
|
|
|
|
children: React.ReactNode;
|
2017-08-22 21:39:08 +03:00
|
|
|
}) {
|
2019-12-07 13:28:52 +02:00
|
|
|
const { title: titleText, icon: iconType } = props;
|
|
|
|
let icon: React.ReactElement | undefined;
|
|
|
|
let title: React.ReactElement | undefined;
|
2016-01-08 15:14:35 +02:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
if (iconType) {
|
2019-11-27 11:03:32 +02:00
|
|
|
icon = (
|
|
|
|
<button className={styles.headerControl}>
|
2019-12-07 13:28:52 +02:00
|
|
|
<span className={icons[iconType]} />
|
2019-11-27 11:03:32 +02:00
|
|
|
</button>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
if (titleText) {
|
2019-11-27 11:03:32 +02:00
|
|
|
title = (
|
|
|
|
<PanelHeader>
|
|
|
|
{icon}
|
2019-12-07 13:28:52 +02:00
|
|
|
{titleText}
|
2019-11-27 11:03:32 +02:00
|
|
|
</PanelHeader>
|
|
|
|
);
|
|
|
|
}
|
2016-01-08 15:14:35 +02:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
return (
|
|
|
|
<div className={styles.panel}>
|
|
|
|
{title}
|
2016-01-08 15:14:35 +02:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
{props.children}
|
|
|
|
</div>
|
|
|
|
);
|
2016-01-08 15:14:35 +02:00
|
|
|
}
|
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
export function PanelHeader(props: { children: React.ReactNode }) {
|
2019-11-27 11:03:32 +02:00
|
|
|
return (
|
2019-12-26 14:18:58 +02:00
|
|
|
<div className={styles.header} {...props} data-testid="auth-header">
|
2019-11-27 11:03:32 +02:00
|
|
|
{props.children}
|
|
|
|
</div>
|
|
|
|
);
|
2016-01-08 15:14:35 +02:00
|
|
|
}
|
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
export function PanelBody(props: { children: React.ReactNode }) {
|
2019-11-27 11:03:32 +02:00
|
|
|
return (
|
2019-12-26 14:18:58 +02:00
|
|
|
<div className={styles.body} {...props} data-testid="auth-body">
|
2019-11-27 11:03:32 +02:00
|
|
|
{props.children}
|
|
|
|
</div>
|
|
|
|
);
|
2016-01-08 15:14:35 +02:00
|
|
|
}
|
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
export function PanelFooter(props: { children: React.ReactNode }) {
|
2019-11-27 11:03:32 +02:00
|
|
|
return (
|
2019-12-26 14:18:58 +02:00
|
|
|
<div className={styles.footer} {...props} data-testid="auth-controls">
|
2019-11-27 11:03:32 +02:00
|
|
|
{props.children}
|
|
|
|
</div>
|
|
|
|
);
|
2016-01-08 15:14:35 +02:00
|
|
|
}
|
2016-01-09 15:51:55 +02:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
export class PanelBodyHeader extends React.Component<
|
2019-11-27 11:03:32 +02:00
|
|
|
{
|
2019-12-07 13:28:52 +02:00
|
|
|
type?: 'default' | 'error';
|
|
|
|
onClose?: () => void;
|
|
|
|
children: React.ReactNode;
|
2019-11-27 11:03:32 +02:00
|
|
|
},
|
|
|
|
{
|
2019-12-07 13:28:52 +02:00
|
|
|
isClosed: boolean;
|
|
|
|
}
|
2019-11-27 11:03:32 +02:00
|
|
|
> {
|
|
|
|
state: {
|
2019-12-07 13:28:52 +02:00
|
|
|
isClosed: boolean;
|
2019-11-27 11:03:32 +02:00
|
|
|
} = {
|
|
|
|
isClosed: false,
|
|
|
|
};
|
|
|
|
|
|
|
|
render() {
|
|
|
|
const { type = 'default', children } = this.props;
|
|
|
|
|
|
|
|
let close;
|
|
|
|
|
|
|
|
if (type === 'error') {
|
|
|
|
close = <span className={styles.close} onClick={this.onClose} />;
|
2016-01-09 17:47:21 +02:00
|
|
|
}
|
|
|
|
|
2019-12-07 21:43:08 +02:00
|
|
|
const className = clsx(styles[`${type}BodyHeader`], {
|
2019-11-27 11:03:32 +02:00
|
|
|
[styles.isClosed]: this.state.isClosed,
|
|
|
|
});
|
2016-02-13 17:28:47 +02:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
const extraProps = omit(this.props, ['type', 'onClose']);
|
2017-08-22 21:39:08 +03:00
|
|
|
|
|
|
|
return (
|
2019-11-27 11:03:32 +02:00
|
|
|
<div className={className} {...extraProps}>
|
|
|
|
{close}
|
|
|
|
{children}
|
|
|
|
</div>
|
2017-08-22 21:39:08 +03:00
|
|
|
);
|
2019-11-27 11:03:32 +02:00
|
|
|
}
|
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
onClose = (event: React.MouseEvent) => {
|
2019-11-27 11:03:32 +02:00
|
|
|
event.preventDefault();
|
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
const { onClose } = this.props;
|
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
this.setState({ isClosed: true });
|
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
if (onClose) {
|
|
|
|
onClose();
|
|
|
|
}
|
2019-11-27 11:03:32 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function PanelIcon({ icon }: { icon: string }) {
|
|
|
|
return (
|
|
|
|
<div className={styles.panelIcon}>
|
|
|
|
<span className={icons[icon]} />
|
|
|
|
</div>
|
|
|
|
);
|
2017-08-22 21:39:08 +03:00
|
|
|
}
|