platform/plugins/contact-resources/src/components/AssigneeBox.svelte
Kristina 3b52779876
Do not show users online status by default (#7046)
Signed-off-by: Kristina Fefelova <kristin.fefelova@gmail.com>
2024-10-27 00:20:51 +07:00

214 lines
6.7 KiB
Svelte

<!--
// Copyright © 2022 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
-->
<script lang="ts">
import contact, { Contact, Employee, Person, getName } from '@hcengineering/contact'
import { Class, DocumentQuery, FindOptions, Ref } from '@hcengineering/core'
import { IntlString, getEmbeddedLabel } from '@hcengineering/platform'
import presentation, { getClient, reduceCalls } from '@hcengineering/presentation'
import {
ActionIcon,
Button,
ButtonKind,
ButtonSize,
Icon,
IconSize,
Label,
LabelAndProps,
getEventPositionElement,
getFocusManager,
showPopup,
tooltip
} from '@hcengineering/ui'
import view from '@hcengineering/view'
import { openDoc } from '@hcengineering/view-resources'
import { createEventDispatcher } from 'svelte'
import { PersonLabelTooltip, personByIdStore } from '..'
import { AssigneeCategory } from '../assignee'
import AssigneePopup from './AssigneePopup.svelte'
import EmployeePresenter from './EmployeePresenter.svelte'
import UserInfo from './UserInfo.svelte'
import IconPerson from './icons/Person.svelte'
export let _class: Ref<Class<Employee>> = contact.mixin.Employee
export let excluded: Ref<Contact>[] | undefined = undefined
export let options: FindOptions<Employee> | undefined = undefined
export let docQuery: DocumentQuery<Employee> | undefined = {
active: true
}
export let label: IntlString
export let placeholder: IntlString = presentation.string.Search
export let value: Ref<Person> | null | undefined
export let categories: AssigneeCategory[] | undefined = undefined
export let allowDeselect = true
export let titleDeselect: IntlString | undefined = undefined
export let readonly = false
export let kind: ButtonKind = 'no-border'
export let size: ButtonSize = 'small'
export let avatarSize: IconSize = kind === 'regular' ? 'small' : 'card'
export let justify: 'left' | 'center' = 'center'
export let width: string | undefined = undefined
export let shrink: number = 0
export let focusIndex = -1
export let showTooltip: LabelAndProps | PersonLabelTooltip | undefined = undefined
export let showNavigate = true
export let shouldShowName: boolean = true
export let id: string | undefined = undefined
export let short: boolean = false
export let showStatus = false
const icon = IconPerson
const dispatch = createEventDispatcher()
let selected: Person | undefined
let container: HTMLElement
const client = getClient()
const updateSelected = reduceCalls(async function (value: Ref<Person> | null | undefined) {
selected = value
? $personByIdStore.get(value) ?? (await client.findOne(contact.class.Person, { _id: value }))
: undefined
})
$: void updateSelected(value)
const mgr = getFocusManager()
function _click (ev: MouseEvent): void {
if (!readonly) {
ev.preventDefault()
ev.stopPropagation()
showPopup(
AssigneePopup,
{
_class,
options,
docQuery,
categories,
ignoreUsers: excluded ?? [],
icon,
selected: value,
placeholder,
allowDeselect,
titleDeselect
},
!$$slots.content ? container : getEventPositionElement(ev),
(result) => {
if (result === null) {
value = null
selected = undefined
dispatch('change', null)
} else if (result !== undefined && result._id !== value) {
value = result._id
dispatch('change', value)
}
if (result !== undefined) {
mgr?.setFocusPos(focusIndex)
}
}
)
}
}
$: hideIcon = size === 'x-large' || (size === 'large' && kind !== 'link')
</script>
<!-- svelte-ignore a11y-click-events-have-key-events -->
<!-- svelte-ignore a11y-no-static-element-interactions -->
<div {id} bind:this={container} class="min-w-0 h-full" class:w-full={width === '100%'} style:flex-shrink={shrink}>
{#if $$slots.content}
<div
class="w-full h-full flex-streatch"
on:click={_click}
use:tooltip={selected !== undefined
? { label: getEmbeddedLabel(getName(client.getHierarchy(), selected)) }
: undefined}
>
<slot name="content" />
</div>
{:else if !shouldShowName}
<EmployeePresenter
value={selected}
{avatarSize}
tooltipLabels={showTooltip}
showPopup={false}
shouldShowName={false}
shouldShowPlaceholder
onEmployeeEdit={_click}
{showStatus}
/>
{:else}
<Button
{focusIndex}
disabled={readonly}
width={width ?? 'min-content'}
{size}
{kind}
{justify}
{showTooltip}
on:click={_click}
>
<span
slot="content"
class="overflow-label flex-grow h-full flex-center"
class:flex-between={showNavigate && selected}
class:dark-color={value == null}
>
<div
class="disabled"
style:width={showNavigate && selected
? `calc(${width ?? 'min-content'} - 1.5rem)`
: `${width ?? 'min-content'}`}
use:tooltip={selected !== undefined
? { label: getEmbeddedLabel(getName(client.getHierarchy(), selected)) }
: undefined}
>
{#if selected}
{#if hideIcon || selected}
<UserInfo value={selected} size={avatarSize} {icon} {short} on:accent-color />
{:else}
{getName(client.getHierarchy(), selected)}
{/if}
{:else}
<div class="flex-presenter not-selected">
{#if icon}
<div class="icon w-4 flex-no-shrink" class:small-gap={size === 'inline' || size === 'small'}>
<Icon {icon} size={'small'} />
</div>
{/if}
<div class="label no-underline">
<Label {label} />
</div>
</div>
{/if}
</div>
{#if selected && showNavigate}
<ActionIcon
icon={view.icon.Open}
size={'small'}
action={() => {
if (selected) {
openDoc(client.getHierarchy(), selected)
}
}}
/>
{/if}
</span>
</Button>
{/if}
</div>