X-Git-Url: https://git.arvados.org/arvados-workbench2.git/blobdiff_plain/5e198d5ecf8dc37ab6d063631b567527c39fb9b5..fc84a3f3932af503d3afd04a58af52270c8fc3b6:/src/views-components/snackbar/snackbar.tsx diff --git a/src/views-components/snackbar/snackbar.tsx b/src/views-components/snackbar/snackbar.tsx index c65d3640..a33b6968 100644 --- a/src/views-components/snackbar/snackbar.tsx +++ b/src/views-components/snackbar/snackbar.tsx @@ -2,26 +2,158 @@ // // SPDX-License-Identifier: AGPL-3.0 -import * as React from "react"; -import { connect } from "react-redux"; -import { RootState } from "../../store/store"; -import MaterialSnackbar, { SnackbarProps } from "@material-ui/core/Snackbar"; +import React from "react"; import { Dispatch } from "redux"; -import { snackbarActions } from "../../store/snackbar/snackbar-actions"; +import { connect } from "react-redux"; +import { RootState } from "store/store"; +import { Button, IconButton, StyleRulesCallback, WithStyles, withStyles, SnackbarContent } from '@material-ui/core'; +import MaterialSnackbar, { SnackbarOrigin } from "@material-ui/core/Snackbar"; +import { snackbarActions, SnackbarKind } from "store/snackbar/snackbar-actions"; +import { navigateTo } from 'store/navigation/navigation-action'; +import WarningIcon from '@material-ui/icons/Warning'; +import CheckCircleIcon from '@material-ui/icons/CheckCircle'; +import ErrorIcon from '@material-ui/icons/Error'; +import InfoIcon from '@material-ui/icons/Info'; +import CloseIcon from '@material-ui/icons/Close'; +import { ArvadosTheme } from "common/custom-theme"; +import { amber, green } from "@material-ui/core/colors"; +import classNames from 'classnames'; -const mapStateToProps = (state: RootState): SnackbarProps => ({ - anchorOrigin: { vertical: "bottom", horizontal: "center" }, - open: state.snackbar.open, - message: {state.snackbar.message}, - autoHideDuration: state.snackbar.hideDuration -}); +interface SnackbarDataProps { + anchorOrigin?: SnackbarOrigin; + autoHideDuration?: number; + open: boolean; + message?: React.ReactElement; + kind: SnackbarKind; + link?: string; +} + +interface SnackbarEventProps { + onClose?: (event: React.SyntheticEvent, reason: string) => void; + onExited: () => void; + onClick: (uuid: string) => void; +} + +const mapStateToProps = (state: RootState): SnackbarDataProps => { + const messages = state.snackbar.messages; + return { + anchorOrigin: { vertical: "bottom", horizontal: "right" }, + open: state.snackbar.open, + message: {messages.length > 0 ? messages[0].message : ""}, + autoHideDuration: messages.length > 0 ? messages[0].hideDuration : 0, + kind: messages.length > 0 ? messages[0].kind : SnackbarKind.INFO, + link: messages.length > 0 ? messages[0].link : '' + }; +}; -const mapDispatchToProps = (dispatch: Dispatch): Pick => ({ +const mapDispatchToProps = (dispatch: Dispatch): SnackbarEventProps => ({ onClose: (event: any, reason: string) => { if (reason !== "clickaway") { dispatch(snackbarActions.CLOSE_SNACKBAR()); } + }, + onExited: () => { + dispatch(snackbarActions.SHIFT_MESSAGES()); + }, + onClick: (uuid: string) => { + dispatch(navigateTo(uuid)); + } +}); + +type CssRules = "success" | "error" | "info" | "warning" | "icon" | "iconVariant" | "message" | "linkButton"; + +const styles: StyleRulesCallback = (theme: ArvadosTheme) => ({ + success: { + backgroundColor: green[600] + }, + error: { + backgroundColor: theme.palette.error.dark + }, + info: { + backgroundColor: theme.palette.primary.main + }, + warning: { + backgroundColor: amber[700] + }, + icon: { + fontSize: 20 + }, + iconVariant: { + opacity: 0.9, + marginRight: theme.spacing.unit + }, + message: { + display: 'flex', + alignItems: 'center' + }, + linkButton: { + fontWeight: 'bolder' } }); -export const Snackbar = connect(mapStateToProps, mapDispatchToProps)(MaterialSnackbar); +type SnackbarProps = SnackbarDataProps & SnackbarEventProps & WithStyles; + +export const Snackbar = withStyles(styles)(connect(mapStateToProps, mapDispatchToProps)( + (props: SnackbarProps) => { + const { classes } = props; + + const variants = { + [SnackbarKind.INFO]: [InfoIcon, classes.info], + [SnackbarKind.WARNING]: [WarningIcon, classes.warning], + [SnackbarKind.SUCCESS]: [CheckCircleIcon, classes.success], + [SnackbarKind.ERROR]: [ErrorIcon, classes.error] + }; + + const [Icon, cssClass] = variants[props.kind]; + + + + return ( + +
+ + {props.message} + + } + action={actions(props)} + />
+
+ ); + } +)); + +const actions = (props: SnackbarProps) => { + const { link, onClose, onClick, classes } = props; + const actions = [ + onClose && onClose(e, '')}> + + + ]; + if (link) { + actions.splice(0, 0, + + ); + } + return actions; +};