// Copyright (C) The Arvados Authors. All rights reserved. // // SPDX-License-Identifier: AGPL-3.0 import React from 'react'; import { Autocomplete, AutocompleteCat } from 'components/autocomplete/autocomplete'; import { connect, DispatchProp } from 'react-redux'; import { ServiceRepository } from 'services/services'; import { FilterBuilder } from '../../services/api/filter-builder'; import { debounce } from 'debounce'; import { ListItemText, Typography } from '@mui/material'; import { noop } from 'lodash/fp'; import { GroupClass, GroupResource } from 'models/group'; import { getUserDetailsString, getUserDisplayName, UserResource } from 'models/user'; import { Resource, ResourceKind } from 'models/resource'; import { ListResults } from 'services/common-service/common-service'; export interface Participant { name: string; tooltip: string; uuid: string; } type ParticipantResource = GroupResource | UserResource; interface ParticipantSelectProps { items: Participant[]; excludedParticipants?: string[]; label?: string; autofocus?: boolean; onlyPeople?: boolean; onlyActive?: boolean; disabled?: boolean; category?: AutocompleteCat; onBlur?: (event: React.FocusEvent) => void; onFocus?: (event: React.FocusEvent) => void; onCreate?: (person: Participant) => void; onDelete?: (index: number) => void; onSelect?: (person: Participant) => void; } interface ParticipantSelectState { isWorking: boolean; value: string; suggestions: ParticipantResource[]; cachedSuggestions: ParticipantResource[]; } const getDisplayName = (item: GroupResource | UserResource, detailed: boolean) => { switch (item.kind) { case ResourceKind.USER: return getUserDisplayName(item, detailed, detailed); case ResourceKind.GROUP: return item.name + `(${`(${(item as Resource).uuid})`})`; default: return (item as Resource).uuid; } }; const getSharingDisplayName = (item: GroupResource | UserResource, detailed: boolean = false) => { switch (item.kind) { case ResourceKind.USER: return `${getUserDisplayName(item, detailed, detailed)} (${item.email})`; case ResourceKind.GROUP: return item.name; default: return (item as Resource).uuid; } }; const getDisplayTooltip = (item: GroupResource | UserResource) => { switch (item.kind) { case ResourceKind.USER: return getUserDetailsString(item); case ResourceKind.GROUP: return item.name + `(${`(${(item as Resource).uuid})`})`; default: return (item as Resource).uuid; } }; export const ParticipantSelect = connect()( class ParticipantSelect extends React.Component { state: ParticipantSelectState = { isWorking: false, value: '', suggestions: [], cachedSuggestions: [], }; componentDidUpdate(prevProps: ParticipantSelectProps & DispatchProp, prevState: ParticipantSelectState) { if (prevState.suggestions.length === 0 && this.state.suggestions.length > 0 && this.state.value.length === 0) { this.setState({ cachedSuggestions: this.state.suggestions }); } } render() { const { label = 'Add people and groups' } = this.props; return ( ); } onFocus = (e) => { this.setState({ isWorking: true }); this.getSuggestions(); } onBlur = (e) => { if (this.props.onBlur) { this.props.onBlur(e); } setTimeout(() => this.setState({ value: '', suggestions: [] }), 200); } renderChipValue(chipValue: Participant) { const { name, uuid } = chipValue; return name || uuid; } renderChipTooltip(item: Participant) { return item.tooltip; } renderSuggestion(item: ParticipantResource) { return ( {getDisplayName(item, true)} ); } handleDelete = (_: Participant, index: number) => { const { onDelete = noop } = this.props; onDelete(index); } handleCreate = () => { const { onCreate } = this.props; if (onCreate) { this.setState({ value: '', suggestions: [] }); onCreate({ name: '', tooltip: '', uuid: this.state.value, }); } } handleSelect = (selection: ParticipantResource) => { if (!selection) return; const { uuid } = selection; const { onSelect = noop } = this.props; this.setState({ value: '', suggestions: this.state.cachedSuggestions }); onSelect({ name: this.props.category === AutocompleteCat.SHARING ? getSharingDisplayName(selection) : getDisplayName(selection, false), tooltip: getDisplayTooltip(selection), uuid, }); } handleChange = (event: React.ChangeEvent) => { this.setState({ value: event.target.value }, this.getSuggestions); } getSuggestions = debounce(() => this.props.dispatch(this.requestSuggestions), 500); requestSuggestions = async (_: void, __: void, { userService, groupsService }: ServiceRepository) => { this.setState({ isWorking: true }); const { value } = this.state; // +1 to see if there are more than 10 results const limit = 11; const filterUsers = new FilterBuilder() .addILike('any', value) .addEqual('is_active', this.props.onlyActive || undefined) .addNotIn('uuid', this.props.excludedParticipants) .getFilters(); const userItems: ListResults = await userService.list({ filters: filterUsers, limit, count: "none" }); const filterGroups = new FilterBuilder() .addNotIn('group_class', [GroupClass.PROJECT, GroupClass.FILTER]) .addNotIn('uuid', this.props.excludedParticipants) .addILike('name', value) .getFilters(); const groupItems: ListResults = await groupsService.list({ filters: filterGroups, limit, count: "none" }); this.setState({ suggestions: this.props.onlyPeople ? userItems.items : userItems.items.concat(groupItems.items), isWorking: false, }); } });