platform/plugins/view-resources/src/components/ValueSelector.svelte
Andrey Sobolev 9ccf593737
UBER-958: Fix query updates (#4742)
Signed-off-by: Andrey Sobolev <haiodo@gmail.com>
2024-02-21 22:11:41 +06:00

184 lines
5.3 KiB
Svelte

<script lang="ts">
import { Class, Doc, DocumentQuery, FindOptions, Hierarchy, Mixin, Ref, generateId } from '@hcengineering/core'
import { Asset, IntlString } from '@hcengineering/platform'
import { ObjectPopup, getClient, updateAttribute } from '@hcengineering/presentation'
import { Label, SelectPopup, resizeObserver } from '@hcengineering/ui'
import { createEventDispatcher } from 'svelte'
import view from '../plugin'
import ObjectPresenter from './ObjectPresenter.svelte'
export let value: Doc | Doc[]
export let isEditable: boolean = true
export let _class: Ref<Class<Doc>> | undefined
export let query: DocumentQuery<Doc> | undefined
export let queryOptions: FindOptions<Doc> | undefined
export let castRequest: Ref<Mixin<Doc>> | undefined = undefined
export let attribute: string
export let searchField: string
export let values:
| Array<{
icon?: Asset
label: IntlString
id: string | number
}>
| undefined = undefined
export let fillQuery: Record<string, string> | undefined
export let docMatches: string[] | undefined
export let placeholder: IntlString | undefined
export let width: 'medium' | 'large' | 'full' = 'medium'
export let size: 'small' | 'medium' | 'large' = 'small'
export let embedded: boolean = false
let progress = false
const dispatch = createEventDispatcher()
const client = getClient()
const hierarchy = client.getHierarchy()
const changeValue = async (newStatus: any): Promise<void> => {
if (newStatus === '#null') {
newStatus = null
return
}
if (!isEditable || newStatus === undefined) {
dispatch('close', undefined)
return
}
progress = true
const docs = [...(Array.isArray(value) ? value : [value])]
const changed = (d: Doc) => (d as any)[attribute] !== newStatus
const ops = client.apply('value-selector:' + generateId())
// We need to sort docs by modified date, to have same modified order impacted.
docs.sort((a: Doc, b: Doc) => b._id.localeCompare(a._id))
for (const it of docs.filter(changed)) {
const cl = Hierarchy.mixinOrClass(it)
const attr =
castRequest !== undefined
? hierarchy.getAttribute(castRequest, attribute)
: hierarchy.getAttribute(cl, attribute)
if (attr === undefined) {
throw new Error('attribute not found')
}
await updateAttribute(ops, it, cl, { key: attribute, attr }, newStatus)
}
await ops.commit()
progress = false
dispatch('close', newStatus)
}
$: current = Array.isArray(value)
? value.every((v) => (v as any)[attribute] === (value as any[])[0][attribute])
? (value as any[])[0][attribute]
: value
: (value as any)[attribute]
let finalQuery: DocumentQuery<Doc> = {}
let docMatch = true
function updateQuery (
query: DocumentQuery<Doc> | undefined,
value: Doc | Doc[],
fillQuery: Record<string, string> | undefined
): void {
// Check if docMatches is applied.
if (docMatches !== undefined && Array.isArray(value)) {
for (const k of docMatches) {
const v = (value[0] as any)[k]
for (const d of value) {
if (v !== (d as any)[k]) {
docMatch = false
return
}
}
}
}
const q = { ...query }
const docs = Array.isArray(value) ? value : [value]
for (const [docKey, queryKey] of Object.entries(fillQuery ?? {})) {
const vs: any[] = []
for (const dv of docs) {
const dvv = (dv as any)[docKey]
if (dvv !== undefined) {
if (!vs.includes(dvv)) {
vs.push(dvv)
}
}
}
;(q as any)[queryKey] = docs.length === 1 ? vs[0] : { $in: vs }
if (docKey === '_object') {
;(q as any)[queryKey] = docs[0]
}
}
finalQuery = q
docMatch = true
}
$: updateQuery(query, value, fillQuery)
$: valuesToShow = values !== undefined ? values.map((it) => ({ ...it, isSelected: it.id === current })) : []
</script>
{#if docMatch}
{#if values}
<SelectPopup
value={valuesToShow}
on:close={(evt) => {
changeValue(evt.detail)
}}
placeholder={placeholder ?? view.string.Filter}
searchable
{width}
{size}
{embedded}
loading={progress}
on:changeContent
/>
{:else if _class !== undefined}
<ObjectPopup
{_class}
docQuery={finalQuery}
options={queryOptions ?? {}}
{searchField}
allowDeselect={true}
selected={current}
on:close={(evt) => {
changeValue(evt.detail === null ? null : evt.detail?._id)
}}
placeholder={placeholder ?? view.string.Filter}
{width}
{size}
{embedded}
loading={progress}
on:changeContent
>
<svelte:fragment slot="item" let:item>
<div class="flex-row-center flex-grow overflow-label">
<ObjectPresenter
objectId={item._id}
_class={item._class}
value={item}
inline={false}
noUnderline
props={{ disabled: true, inline: false, size, avatarSize: 'smaller' }}
/>
</div>
</svelte:fragment>
</ObjectPopup>
{/if}
{:else}
<div class="selectPopup" use:resizeObserver={() => dispatch('changeContent')}>
<div class="flex-center w-60 h-18">
<Label label={view.string.DontMatchCriteria} />
</div>
</div>
{/if}