Merge branch '21128-toolbar-context-menu'
[arvados-workbench2.git] / src / views-components / main-app-bar / main-app-bar.tsx
index ee47a0103b86acb6e9502f6b837f2632ea4d45eb..c57d5cd85d51fb75e3a0e507c5ae4d692d75e264 100644 (file)
 //
 // SPDX-License-Identifier: AGPL-3.0
 
-import * as React from "react";
-import { AppBar, Toolbar, Typography, Grid, IconButton, Badge, Button, MenuItem, Tooltip } from "@material-ui/core";
-import { User, getUserFullname } from "~/models/user";
-import { SearchBar } from "~/components/search-bar/search-bar";
-import { DropdownMenu } from "~/components/dropdown-menu/dropdown-menu";
-import { DetailsIcon, NotificationIcon, UserPanelIcon, HelpIcon } from "~/components/icon/icon";
+import React from "react";
+import { AppBar, Toolbar, Typography, Grid } from "@material-ui/core";
+import { StyleRulesCallback, WithStyles, withStyles } from '@material-ui/core/styles';
+import { Link } from "react-router-dom";
+import { User } from "models/user";
+import { SearchBar } from "views-components/search-bar/search-bar";
+import { Routes } from 'routes/routes';
+import { NotificationsMenu } from "views-components/main-app-bar/notifications-menu";
+import { AccountMenu } from "views-components/main-app-bar/account-menu";
+import { HelpMenu } from 'views-components/main-app-bar/help-menu';
+import { ReactNode } from "react";
+import { AdminMenu } from "views-components/main-app-bar/admin-menu";
+import { pluginConfig } from 'plugins';
+import { sanitizeHTML } from "common/html-sanitize";
 
-export interface MainAppBarMenuItem {
-    label: string;
-}
+type CssRules = 'toolbar' | 'link';
 
-export interface MainAppBarMenuItems {
-    accountMenu: MainAppBarMenuItem[];
-    helpMenu: MainAppBarMenuItem[];
-    anonymousMenu: MainAppBarMenuItem[];
-}
+const styles: StyleRulesCallback<CssRules> = () => ({
+    link: {
+        textDecoration: 'none',
+        color: 'inherit'
+    },
+    toolbar: {
+        height: '56px',
+    }
+});
 
 interface MainAppBarDataProps {
-    searchText: string;
-    searchDebounce?: number;
-    breadcrumbs: React.ComponentType<any>;
     user?: User;
-    menuItems: MainAppBarMenuItems;
-    buildInfo: string;
+    buildInfo?: string;
+    children?: ReactNode;
+    uuidPrefix: string;
+    siteBanner: string;
+    sidePanelIsCollapsed: boolean;
 }
 
-export interface MainAppBarActionProps {
-    onSearch: (searchText: string) => void;
-    onMenuItemClick: (menuItem: MainAppBarMenuItem) => void;
-    onDetailsPanelToggle: () => void;
-}
+export type MainAppBarProps = MainAppBarDataProps & WithStyles<CssRules>;
 
-export type MainAppBarProps = MainAppBarDataProps & MainAppBarActionProps;
-
-export const MainAppBar: React.SFC<MainAppBarProps> = (props) => {
-    return <AppBar position="static">
-        <Toolbar>
-            <Grid container justify="space-between">
-                <Grid item xs={3}>
-                    <Typography variant="headline" color="inherit" noWrap>
-                        Arvados 2
-                    </Typography>
-                    <Typography variant="body1" color="inherit" noWrap >
-                        {props.buildInfo}
-                    </Typography>
-                </Grid>
-                <Grid item xs={6} container alignItems="center">
-                    {
-                        props.user && <SearchBar
-                            value={props.searchText}
-                            onSearch={props.onSearch}
-                            debounce={props.searchDebounce}
-                        />
-                    }
+export const MainAppBar = withStyles(styles)(
+    (props: MainAppBarProps) => {
+        return <AppBar position="absolute">
+            <Toolbar className={props.classes.toolbar}>
+                <Grid container justify="space-between">
+                    {pluginConfig.appBarLeft || <Grid container item xs={3} direction="column" justify="center">
+                        <Typography variant='h6' color="inherit" noWrap>
+                            <Link to={Routes.ROOT} className={props.classes.link}>
+                                <span dangerouslySetInnerHTML={{ __html: sanitizeHTML(props.siteBanner) }} /> ({props.uuidPrefix})
+                </Link>
+                        </Typography>
+                        <Typography variant="caption" color="inherit">
+                            
+                            {props.buildInfo}</Typography>
+                    </Grid>}
+                    <Grid
+                        item
+                        xs={6}
+                        container
+                        alignItems="center">
+                        {pluginConfig.appBarMiddle || (props.user && props.user.isActive && <SearchBar />)}
+                    </Grid>
+                    <Grid
+                        item
+                        xs={3}
+                        container
+                        alignItems="center"
+                        justify="flex-end"
+                        wrap="nowrap">
+                        {props.user ? <>
+                            <NotificationsMenu />
+                            <AccountMenu />
+                            {pluginConfig.appBarRight ||
+                                <>
+                                    {props.user.isAdmin && <AdminMenu />}
+                                    <HelpMenu />
+                                </>}
+                        </> :
+                            pluginConfig.appBarRight || <HelpMenu />
+                        }
+                    </Grid>
                 </Grid>
-                <Grid item xs={3} container alignItems="center" justify="flex-end">
-                    {
-                        props.user ? renderMenuForUser(props) : renderMenuForAnonymous(props)
-                    }
-                </Grid>
-            </Grid>
-        </Toolbar>
-        <Toolbar >
-            {props.user && <props.breadcrumbs />}
-            {props.user && <IconButton color="inherit" onClick={props.onDetailsPanelToggle}>
-                <Tooltip title="Additional Info">
-                    <DetailsIcon />
-                </Tooltip>
-            </IconButton>
-            }
-        </Toolbar>
-    </AppBar>;
-};
-
-const renderMenuForUser = ({ user, menuItems, onMenuItemClick }: MainAppBarProps) => {
-    return (
-        <>
-            <IconButton color="inherit">
-                <Tooltip title="Notification">
-                    <Badge badgeContent={3} color="primary">
-                        <NotificationIcon />
-                    </Badge>
-                </Tooltip>
-            </IconButton>
-            <DropdownMenu icon={<UserPanelIcon />} id="account-menu" title="Account Management">
-                <MenuItem>
-                    {getUserFullname(user)}
-                </MenuItem>
-                {renderMenuItems(menuItems.accountMenu, onMenuItemClick)}
-            </DropdownMenu>
-            <DropdownMenu icon={<HelpIcon />} id="help-menu" title="Help">
-                {renderMenuItems(menuItems.helpMenu, onMenuItemClick)}
-            </DropdownMenu>
-        </>
-    );
-};
-
-const renderMenuForAnonymous = ({ onMenuItemClick, menuItems }: MainAppBarProps) => {
-    return menuItems.anonymousMenu.map((item, index) => (
-        <Button key={index} color="inherit" onClick={() => onMenuItemClick(item)}>
-            {item.label}
-        </Button>
-    ));
-};
-
-const renderMenuItems = (menuItems: MainAppBarMenuItem[], onMenuItemClick: (menuItem: MainAppBarMenuItem) => void) => {
-    return menuItems.map((item, index) => (
-        <MenuItem key={index} onClick={() => onMenuItemClick(item)}>
-            {item.label}
-        </MenuItem>
-    ));
-};
+            </Toolbar>
+            {props.children}
+        </AppBar>;
+    }
+);