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';
45 const styles: StyleRulesCallback<CssRules> = (theme: ArvadosTheme) => ({
56 WebkitMaskImage: '-webkit-gradient(linear, left bottom, right bottom, from(rgba(0,0,0,1)), to(rgba(0,0,0,0)))',
59 color: theme.palette.primary.main,
67 margin: '0.3rem auto auto 1rem',
74 flexDirection: 'column',
83 margin: 'auto 0 0.5rem 0.3rem',
84 color: theme.palette.text.primary,
91 color: theme.palette.text.primary,
98 marginBottom: '0.5rem',
113 const mapStateToProps = (state: RootState) => {
114 const currentRoute = state.router.location?.pathname.split('/') || [];
115 const currentItemUuid = currentRoute[currentRoute.length - 1];
116 const currentResource = getResource(currentItemUuid)(state.resources);
117 const frozenByUser = currentResource && getResource((currentResource as ProjectResource).frozenByUuid as string)(state.resources);
118 const frozenByFullName = frozenByUser && (frozenByUser as Resource & { fullName: string }).fullName;
121 isAdmin: state.auth.user?.isAdmin,
127 const mapDispatchToProps = (dispatch: any) => ({
128 handleContextMenu: (event: React.MouseEvent<HTMLElement>, resource: any, isAdmin: boolean) => {
129 // When viewing the contents of a filter group, all contents should be treated as read only.
130 let readOnly = false;
131 if (resource.groupClass === 'filter') {
135 const menuKind = dispatch(resourceUuidToContextMenuKind(resource.uuid, readOnly));
136 if (menuKind && resource) {
138 openContextMenu(event, {
141 ownerUuid: resource.ownerUuid,
142 isTrashed: 'isTrashed' in resource ? resource.isTrashed : false,
146 isFrozen: !!resource.frozenByUuid,
147 description: resource.description,
148 storageClassesDesired: (resource as CollectionResource).storageClassesDesired,
149 properties: 'properties' in resource ? resource.properties : {},
157 type DetailsCardProps = WithStyles<CssRules> & {
158 currentResource: ProjectResource | UserResource;
159 frozenByFullName?: string;
161 handleContextMenu: (event: React.MouseEvent<HTMLElement>, resource: ContextMenuResource, isAdmin: boolean) => void;
164 type UserCardProps = WithStyles<CssRules> & {
165 currentResource: UserResource;
168 type ProjectCardProps = WithStyles<CssRules> & {
169 currentResource: ProjectResource;
170 frozenByFullName: string | undefined;
172 handleContextMenu: (event: React.MouseEvent<HTMLElement>, resource: ContextMenuResource, isAdmin: boolean) => void;
175 export const ProjectDetailsCard = connect(
179 withStyles(styles)((props: DetailsCardProps) => {
180 const { classes, currentResource, frozenByFullName, handleContextMenu, isAdmin } = props;
181 switch (currentResource.kind as string) {
182 case ResourceKind.USER:
186 currentResource={currentResource as UserResource}
189 case ResourceKind.PROJECT:
193 currentResource={currentResource as ProjectResource}
194 frozenByFullName={frozenByFullName}
196 handleContextMenu={(ev) => handleContextMenu(ev, currentResource as any, isAdmin)}
205 const UserCard: React.FC<UserCardProps> = ({ classes, currentResource }) => {
206 const { fullName, uuid } = currentResource as UserResource & { fullName: string };
209 <Card className={classes.root}>
211 className={classes.cardheader}
213 <section className={classes.nameContainer}>
220 {!currentResource.isActive && (
221 <Typography className={classes.activeIndicator}>
222 <UserResourceAccountStatus uuid={uuid} />
227 action={<MultiselectToolbar inputSelectedUuid={uuid} />}
233 const ProjectCard: React.FC<ProjectCardProps> = ({ classes, currentResource, frozenByFullName, handleContextMenu, isAdmin }) => {
234 const { name, uuid, description } = currentResource as ProjectResource;
237 <Card className={classes.root}>
239 className={classes.cardheader}
242 <section className={classes.namePlate}>
246 style={{ marginRight: '1rem' }}
251 className={classes.faveIcon}
252 resourceUuid={currentResource.uuid}
255 className={classes.faveIcon}
256 resourceUuid={currentResource.uuid}
258 {!!frozenByFullName && <Tooltip
259 className={classes.frozenIcon}
260 title={<span>Project was frozen by {frozenByFullName}</span>}
262 <FreezeIcon style={{ fontSize: 'inherit' }} />
265 <section className={classes.chipsection}>
266 <Typography component='div'>
267 {typeof currentResource.properties === 'object' &&
268 Object.keys(currentResource.properties).map((k) =>
269 Array.isArray(currentResource.properties[k])
270 ? currentResource.properties[k].map((v: string) => getPropertyChip(k, v, undefined, classes.tag))
271 : getPropertyChip(k, currentResource.properties[k], undefined, classes.tag)
284 aria-label='More options'
285 onClick={(ev) => handleContextMenu(ev, currentResource as any, isAdmin)}
291 <CardContent className={classes.cardcontent}>
294 {/* <Typography className={classes.fadeout}>{description.replace(/<[^>]*>/g, '').slice(0, 45)}...</Typography> */}
295 <div className={classes.showmore}>
297 title={`Description of ${name}`}
298 content={description}
299 label='Show full description'