1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import React from 'react';
6 import { Card, CardHeader, WithStyles, withStyles, Typography, CardContent, Tooltip } from '@material-ui/core';
7 import { StyleRulesCallback } from '@material-ui/core';
8 import { ArvadosTheme } from 'common/custom-theme';
9 import { RootState } from 'store/store';
10 import { connect } from 'react-redux';
11 import { getResource } from 'store/resources/resources';
12 import { MultiselectToolbar } from 'components/multiselect-toolbar/MultiselectToolbar';
13 import { getPropertyChip } from '../resource-properties-form/property-chip';
14 import { ProjectResource } from 'models/project';
15 import { ResourceKind } from 'models/resource';
16 import { UserResource } from 'models/user';
17 import { UserResourceAccountStatus } from 'views-components/data-explorer/renderers';
18 import { FavoriteStar, PublicFavoriteStar } from 'views-components/favorite-star/favorite-star';
19 import { FreezeIcon } from 'components/icon/icon';
20 import { Resource } from 'models/resource';
21 import { MoreVerticalIcon } from 'components/icon/icon';
22 import { IconButton } from '@material-ui/core';
23 import { ContextMenuResource } from 'store/context-menu/context-menu-actions';
24 import { resourceUuidToContextMenuKind } from 'store/context-menu/context-menu-actions';
25 import { openContextMenu } from 'store/context-menu/context-menu-actions';
26 import { CollectionResource } from 'models/collection';
27 import { RichTextEditorLink } from 'components/rich-text-editor-link/rich-text-editor-link';
44 const styles: StyleRulesCallback<CssRules> = (theme: ArvadosTheme) => ({
51 color: theme.palette.primary.main,
59 margin: '0.3rem auto auto 1rem',
66 flexDirection: 'column',
75 margin: 'auto 0 0.5rem 0.3rem',
76 color: theme.palette.text.primary,
83 color: theme.palette.text.primary,
90 marginBottom: '0.5rem',
105 const mapStateToProps = (state: RootState) => {
106 const currentRoute = state.router.location?.pathname.split('/') || [];
107 const currentItemUuid = currentRoute[currentRoute.length - 1];
108 const currentResource = getResource(currentItemUuid)(state.resources);
109 const frozenByUser = currentResource && getResource((currentResource as ProjectResource).frozenByUuid as string)(state.resources);
110 const frozenByFullName = frozenByUser && (frozenByUser as Resource & { fullName: string }).fullName;
113 isAdmin: state.auth.user?.isAdmin,
119 const mapDispatchToProps = (dispatch: any) => ({
120 handleContextMenu: (event: React.MouseEvent<HTMLElement>, resource: any, isAdmin: boolean) => {
121 // When viewing the contents of a filter group, all contents should be treated as read only.
122 let readOnly = false;
123 if (resource.groupClass === 'filter') {
127 const menuKind = dispatch(resourceUuidToContextMenuKind(resource.uuid, readOnly));
128 if (menuKind && resource) {
130 openContextMenu(event, {
133 ownerUuid: resource.ownerUuid,
134 isTrashed: 'isTrashed' in resource ? resource.isTrashed : false,
138 isFrozen: !!resource.frozenByUuid,
139 description: resource.description,
140 storageClassesDesired: (resource as CollectionResource).storageClassesDesired,
141 properties: 'properties' in resource ? resource.properties : {},
149 type DetailsCardProps = WithStyles<CssRules> & {
150 currentResource: ProjectResource | UserResource;
151 frozenByFullName?: string;
153 handleContextMenu: (event: React.MouseEvent<HTMLElement>, resource: ContextMenuResource, isAdmin: boolean) => void;
156 type UserCardProps = WithStyles<CssRules> & {
157 currentResource: UserResource;
160 type ProjectCardProps = WithStyles<CssRules> & {
161 currentResource: ProjectResource;
162 frozenByFullName: string | undefined;
164 handleContextMenu: (event: React.MouseEvent<HTMLElement>, resource: ContextMenuResource, isAdmin: boolean) => void;
167 export const ProjectDetailsCard = connect(
171 withStyles(styles)((props: DetailsCardProps) => {
172 const { classes, currentResource, frozenByFullName, handleContextMenu, isAdmin } = props;
173 switch (currentResource.kind as string) {
174 case ResourceKind.USER:
178 currentResource={currentResource as UserResource}
181 case ResourceKind.PROJECT:
185 currentResource={currentResource as ProjectResource}
186 frozenByFullName={frozenByFullName}
188 handleContextMenu={(ev) => handleContextMenu(ev, currentResource as any, isAdmin)}
197 const UserCard: React.FC<UserCardProps> = ({ classes, currentResource }) => {
198 const { fullName, uuid } = currentResource as UserResource & { fullName: string };
201 <Card className={classes.root}>
203 className={classes.cardHeader}
205 <section className={classes.nameContainer}>
212 {!currentResource.isActive && (
213 <Typography className={classes.activeIndicator}>
214 <UserResourceAccountStatus uuid={uuid} />
219 action={<MultiselectToolbar inputSelectedUuid={uuid} />}
225 const ProjectCard: React.FC<ProjectCardProps> = ({ classes, currentResource, frozenByFullName, handleContextMenu, isAdmin }) => {
226 const { name, description } = currentResource as ProjectResource;
229 <Card className={classes.root}>
231 className={classes.cardHeader}
234 <section className={classes.namePlate}>
238 style={{ marginRight: '1rem' }}
243 className={classes.faveIcon}
244 resourceUuid={currentResource.uuid}
247 className={classes.faveIcon}
248 resourceUuid={currentResource.uuid}
250 {!!frozenByFullName && <Tooltip
251 className={classes.frozenIcon}
252 title={<span>Project was frozen by {frozenByFullName}</span>}
254 <FreezeIcon style={{ fontSize: 'inherit' }} />
257 <section className={classes.chipSection}>
258 <Typography component='div'>
259 {typeof currentResource.properties === 'object' &&
260 Object.keys(currentResource.properties).map((k) =>
261 Array.isArray(currentResource.properties[k])
262 ? currentResource.properties[k].map((v: string) => getPropertyChip(k, v, undefined, classes.tag))
263 : getPropertyChip(k, currentResource.properties[k], undefined, classes.tag)
276 aria-label='More options'
277 onClick={(ev) => handleContextMenu(ev, currentResource as any, isAdmin)}
283 <CardContent className={classes.cardContent}>
286 <div className={classes.showMore}>
288 title={`Description of ${name}`}
289 content={description}
290 label='Show full description'