mirror of
https://github.com/hcengineering/platform.git
synced 2025-04-24 09:16:43 +00:00
188 lines
5.7 KiB
Svelte
188 lines
5.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 { Class, Doc, DocumentQuery, Ref, Space, WithLookup } from '@hcengineering/core'
|
|
import { IntlString, Asset, getResource } from '@hcengineering/platform'
|
|
import { getClient } from '@hcengineering/presentation'
|
|
import {
|
|
AnyComponent,
|
|
Button,
|
|
Component,
|
|
IconAdd,
|
|
IModeSelector,
|
|
Loading,
|
|
ModeSelector,
|
|
SearchInput,
|
|
showPopup,
|
|
Header,
|
|
Breadcrumb
|
|
} from '@hcengineering/ui'
|
|
import {
|
|
ViewOptionModel,
|
|
ViewOptions,
|
|
ViewQueryOption,
|
|
Viewlet,
|
|
ViewletDescriptor,
|
|
ViewletPreference
|
|
} from '@hcengineering/view'
|
|
import { FilterBar, FilterButton, ViewletSelector, ViewletSettingButton } from '@hcengineering/view-resources'
|
|
|
|
export let _class: Ref<Class<Doc>>
|
|
export let space: Ref<Space> | undefined = undefined
|
|
export let icon: Asset
|
|
export let label: IntlString
|
|
export let createEvent: string | undefined
|
|
export let createLabel: IntlString | undefined
|
|
export let createComponent: AnyComponent | undefined
|
|
export let createComponentProps: Record<string, any> = {}
|
|
export let isCreationDisabled = false
|
|
export let descriptors: Array<Ref<ViewletDescriptor>> | undefined = undefined
|
|
export let baseQuery: DocumentQuery<Doc> | undefined = undefined
|
|
export let modes: IModeSelector<any> | undefined = undefined
|
|
|
|
const client = getClient()
|
|
const hierarchy = client.getHierarchy()
|
|
|
|
let search = ''
|
|
let viewlet: WithLookup<Viewlet> | undefined
|
|
let filterVisible: boolean = false
|
|
|
|
let preference: ViewletPreference | undefined
|
|
let viewlets: Array<WithLookup<Viewlet>> = []
|
|
let viewOptions: ViewOptions | undefined
|
|
|
|
$: _baseQuery = { ...(baseQuery ?? {}), ...(viewlet?.baseQuery ?? {}) }
|
|
$: query = { ..._baseQuery }
|
|
$: searchQuery = search === '' ? query : { ...query, $search: search }
|
|
$: resultQuery = searchQuery
|
|
|
|
$: void updateQuery(_baseQuery, viewOptions, viewlet)
|
|
|
|
async function updateQuery (
|
|
initialQuery: DocumentQuery<Doc>,
|
|
viewOptions: ViewOptions | undefined,
|
|
viewlet: Viewlet | undefined
|
|
): Promise<void> {
|
|
query =
|
|
viewOptions !== undefined && viewlet !== undefined
|
|
? await getViewQuery(initialQuery, viewOptions, viewlet.viewOptions?.other)
|
|
: initialQuery
|
|
}
|
|
|
|
async function getViewQuery (
|
|
query: DocumentQuery<Doc>,
|
|
viewOptions: ViewOptions,
|
|
viewOptionsModel: ViewOptionModel[] | undefined
|
|
): Promise<DocumentQuery<Doc>> {
|
|
if (viewOptionsModel === undefined) return query
|
|
let result: DocumentQuery<Doc> = hierarchy.clone(query)
|
|
for (const viewOption of viewOptionsModel) {
|
|
if (viewOption.actionTarget !== 'query') continue
|
|
const queryOption = viewOption as ViewQueryOption
|
|
const f = await getResource(queryOption.action)
|
|
const resultP = f(viewOptions[queryOption.key] ?? queryOption.defaultValue, result)
|
|
if (resultP instanceof Promise) {
|
|
result = await resultP
|
|
} else {
|
|
result = resultP
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
function showCreateDialog (): void {
|
|
if (createComponent === undefined) return
|
|
showPopup(createComponent, createComponentProps, 'top')
|
|
}
|
|
</script>
|
|
|
|
<Header
|
|
adaptive={modes !== undefined ? 'doubleRow' : filterVisible ? 'freezeActions' : 'disabled'}
|
|
hideActions={!(createLabel && createComponent)}
|
|
hideExtra={modes === undefined}
|
|
freezeBefore
|
|
>
|
|
<svelte:fragment slot="beforeTitle">
|
|
<ViewletSelector
|
|
bind:viewlet
|
|
bind:preference
|
|
bind:viewlets
|
|
viewletQuery={{
|
|
attachTo: _class,
|
|
variant: { $exists: false },
|
|
...(descriptors !== undefined ? { descriptor: { $in: descriptors } } : {})
|
|
}}
|
|
/>
|
|
<ViewletSettingButton bind:viewOptions bind:viewlet />
|
|
</svelte:fragment>
|
|
|
|
<Breadcrumb {icon} {label} size={'large'} isCurrent />
|
|
|
|
<svelte:fragment slot="search">
|
|
<SearchInput bind:value={search} collapsed />
|
|
<FilterButton {_class} bind:visible={filterVisible} />
|
|
</svelte:fragment>
|
|
<svelte:fragment slot="actions">
|
|
{#if createLabel && createComponent}
|
|
<Button
|
|
icon={IconAdd}
|
|
label={createLabel}
|
|
kind={'primary'}
|
|
disabled={isCreationDisabled}
|
|
event={createEvent}
|
|
on:click={() => {
|
|
showCreateDialog()
|
|
}}
|
|
/>
|
|
{/if}
|
|
</svelte:fragment>
|
|
<svelte:fragment slot="extra">
|
|
{#if modes !== undefined}
|
|
<ModeSelector kind={'subtle'} props={modes} />
|
|
{/if}
|
|
</svelte:fragment>
|
|
</Header>
|
|
|
|
{#if !viewlet?.$lookup?.descriptor?.component || viewlet?.attachTo !== _class || (preference !== undefined && viewlet?._id !== preference.attachedTo)}
|
|
<Loading />
|
|
{:else if viewOptions && viewlet}
|
|
<FilterBar
|
|
{_class}
|
|
{space}
|
|
query={searchQuery}
|
|
{viewOptions}
|
|
on:change={(e) => {
|
|
resultQuery = { ...query, ...e.detail }
|
|
}}
|
|
/>
|
|
<Component
|
|
is={viewlet.$lookup.descriptor.component}
|
|
props={{
|
|
_class,
|
|
space,
|
|
options: viewlet.options,
|
|
config: preference?.config ?? viewlet.config,
|
|
viewlet,
|
|
viewOptions,
|
|
viewOptionsConfig: viewlet.viewOptions?.other,
|
|
createItemDialog: createComponent,
|
|
createItemLabel: createLabel,
|
|
query: resultQuery,
|
|
totalQuery: query,
|
|
...viewlet.props
|
|
}}
|
|
/>
|
|
{/if}
|