mirror of
https://github.com/hcengineering/platform.git
synced 2024-11-27 01:13:27 +03:00
parent
3eec4d03ae
commit
3f3953de3a
@ -28,7 +28,7 @@
|
||||
import Edit from './icons/Edit.svelte'
|
||||
import SocialEditor from './SocialEditor.svelte'
|
||||
import AttributesBar from './AttributesBar.svelte'
|
||||
import { TableView } from '@anticrm/view-resources'
|
||||
import { Table } from '@anticrm/view-resources'
|
||||
|
||||
import core from '@anticrm/core'
|
||||
|
||||
@ -89,24 +89,25 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
Applications
|
||||
|
||||
<TableView
|
||||
_class={recruit.class.Applicant}
|
||||
config={['$lookup.candidate', '$lookup.state', '$lookup.candidate.city', '$lookup.space.name']}
|
||||
options={
|
||||
{
|
||||
lookup: {
|
||||
candidate: recruit.class.Candidate,
|
||||
state: core.class.State,
|
||||
space: core.class.Space
|
||||
<div class="group">
|
||||
<div class="caption">Applications</div>
|
||||
<Table
|
||||
_class={recruit.class.Applicant}
|
||||
config={['$lookup.candidate', '$lookup.state', '$lookup.candidate.city', '$lookup.space.name']}
|
||||
options={
|
||||
{
|
||||
lookup: {
|
||||
candidate: recruit.class.Candidate,
|
||||
state: core.class.State,
|
||||
space: core.class.Space
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
search=""
|
||||
/>
|
||||
search=""
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="attachments">
|
||||
<div class="group">
|
||||
<Attachments objectId={object._id} _class={object._class} space={object.space} {object}/>
|
||||
</div>
|
||||
|
||||
@ -162,8 +163,14 @@
|
||||
span { margin-left: .5rem; }
|
||||
}
|
||||
|
||||
.attachments {
|
||||
.group {
|
||||
margin-top: 3.5rem;
|
||||
|
||||
.caption {
|
||||
font-weight: 500;
|
||||
font-size: 1.25rem;
|
||||
color: var(--theme-caption-color);
|
||||
}
|
||||
}
|
||||
|
||||
// .container {
|
||||
|
194
plugins/view-resources/src/components/Table.svelte
Normal file
194
plugins/view-resources/src/components/Table.svelte
Normal file
@ -0,0 +1,194 @@
|
||||
<!--
|
||||
// Copyright © 2020, 2021 Anticrm Platform Contributors.
|
||||
// Copyright © 2021 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 type { Ref, Class, Doc, Space, FindOptions } from '@anticrm/core'
|
||||
import { SortingOrder } from '@anticrm/core'
|
||||
import { buildModel } from '../utils'
|
||||
import { getClient } from '@anticrm/presentation'
|
||||
import { Label, showPopup, Loading, CheckBox, IconDown, IconUp } from '@anticrm/ui'
|
||||
import MoreV from './icons/MoreV.svelte'
|
||||
import Menu from './Menu.svelte'
|
||||
|
||||
import { createQuery } from '@anticrm/presentation'
|
||||
|
||||
export let _class: Ref<Class<Doc>>
|
||||
export let space: Ref<Space>
|
||||
export let options: FindOptions<Doc> | undefined
|
||||
export let config: string[]
|
||||
export let search: string
|
||||
|
||||
let sortKey = 'modifiedOn'
|
||||
let sortOrder = SortingOrder.Descending
|
||||
let selectRow: number | undefined = undefined
|
||||
|
||||
let objects: Doc[]
|
||||
|
||||
const query = createQuery()
|
||||
$: query.query(_class, search === '' ? (space ? { space } : {}) : { $search: search }, result => { objects = result }, { sort: { [sortKey]: sortOrder }, ...options })
|
||||
|
||||
function getValue(doc: Doc, key: string): any {
|
||||
if (key.length === 0)
|
||||
return doc
|
||||
const path = key.split('.')
|
||||
const len = path.length
|
||||
let obj = doc as any
|
||||
for (let i=0; i<len; i++){
|
||||
obj = obj?.[path[i]]
|
||||
}
|
||||
return obj ?? ''
|
||||
}
|
||||
|
||||
const client = getClient()
|
||||
let checking: boolean = false
|
||||
|
||||
const showMenu = (ev: MouseEvent, object: Doc, row: number): void => {
|
||||
selectRow = row
|
||||
showPopup(Menu, { object }, ev.target as HTMLElement, (() => { selectRow = undefined }))
|
||||
}
|
||||
|
||||
function changeSorting(key: string) {
|
||||
if (key === '')
|
||||
return
|
||||
if (key !== sortKey) {
|
||||
sortKey = key
|
||||
sortOrder = SortingOrder.Ascending
|
||||
} else {
|
||||
sortOrder = (sortOrder === SortingOrder.Ascending) ? SortingOrder.Descending : SortingOrder.Ascending
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
{#await buildModel(client, _class, config, options)}
|
||||
<Loading/>
|
||||
{:then model}
|
||||
<table class="table-body">
|
||||
<thead>
|
||||
<tr class="tr-head">
|
||||
{#each model as attribute, cellHead}
|
||||
{#if !cellHead}
|
||||
<th>
|
||||
<div class="checkCell" class:checkall={checking}>
|
||||
<CheckBox symbol={'minus'} />
|
||||
</div>
|
||||
</th>
|
||||
{/if}
|
||||
<th class="sortable" class:sorted={attribute.key === sortKey} on:click={() => changeSorting(attribute.key)}>
|
||||
<div class="flex-row-center">
|
||||
<Label label = {attribute.label}/>
|
||||
{#if attribute.key === sortKey}
|
||||
<div class="icon">
|
||||
{#if sortOrder === SortingOrder.Ascending}
|
||||
<IconUp size={'small'} />
|
||||
{:else}
|
||||
<IconDown size={'small'} />
|
||||
{/if}
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
</th>
|
||||
{/each}
|
||||
</tr>
|
||||
</thead>
|
||||
{#if objects}
|
||||
<tbody>
|
||||
{#each objects as object, row (object._id)}
|
||||
<tr class="tr-body" class:checking class:fixed={row === selectRow}>
|
||||
{#each model as attribute, cell}
|
||||
{#if !cell}
|
||||
<td><div class="checkCell"><CheckBox bind:checked={checking} /></div></td>
|
||||
<td><div class="firstCell">
|
||||
<svelte:component this={attribute.presenter} value={getValue(object, attribute.key)}/>
|
||||
<div class="menuRow" on:click={(ev) => showMenu(ev, object, row)}><MoreV size={'small'} /></div>
|
||||
</div></td>
|
||||
{:else}
|
||||
<td><svelte:component this={attribute.presenter} value={getValue(object, attribute.key)}/></td>
|
||||
{/if}
|
||||
{/each}
|
||||
</tr>
|
||||
{/each}
|
||||
</tbody>
|
||||
{/if}
|
||||
</table>
|
||||
{/await}
|
||||
|
||||
<style lang="scss">
|
||||
.table-body { width: 100%; }
|
||||
|
||||
.firstCell {
|
||||
display: flex;
|
||||
// justify-content: space-between;
|
||||
align-items: center;
|
||||
.menuRow {
|
||||
visibility: hidden;
|
||||
margin-left: .5rem;
|
||||
opacity: .6;
|
||||
cursor: pointer;
|
||||
&:hover { opacity: 1; }
|
||||
}
|
||||
}
|
||||
.checkCell {
|
||||
visibility: hidden;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
th, td {
|
||||
padding: .5rem 1.5rem;
|
||||
text-align: left;
|
||||
&:first-child {
|
||||
padding: 0 .75rem;
|
||||
width: 2.5rem;
|
||||
}
|
||||
&:nth-child(2) {
|
||||
padding-left: 0;
|
||||
// padding-right: 0;
|
||||
}
|
||||
}
|
||||
|
||||
th {
|
||||
height: 2.5rem;
|
||||
font-weight: 500;
|
||||
font-size: .75rem;
|
||||
color: var(--theme-content-dark-color);
|
||||
box-shadow: inset 0 -1px 0 0 var(--theme-bg-focused-color);
|
||||
user-select: none;
|
||||
|
||||
&.sortable { cursor: pointer; }
|
||||
&.sorted .icon {
|
||||
margin-left: .25rem;
|
||||
opacity: .6;
|
||||
}
|
||||
.checkall { visibility: visible; }
|
||||
}
|
||||
|
||||
.tr-body {
|
||||
height: 3.25rem;
|
||||
color: var(--theme-caption-color);
|
||||
border-bottom: 1px solid var(--theme-button-border-hovered);
|
||||
&:hover, &.checking {
|
||||
background-color: var(--theme-table-bg-hover);
|
||||
.checkCell { visibility: visible; }
|
||||
}
|
||||
&:hover .firstCell .menuRow { visibility: visible; }
|
||||
}
|
||||
|
||||
.fixed {
|
||||
background-color: var(--theme-table-bg-hover);
|
||||
.checkCell { visibility: visible; }
|
||||
.menuRow { visibility: visible; }
|
||||
}
|
||||
</style>
|
@ -138,6 +138,8 @@
|
||||
height: 100%;
|
||||
}
|
||||
|
||||
.table-body { width: 100%; }
|
||||
|
||||
.firstCell {
|
||||
display: flex;
|
||||
// justify-content: space-between;
|
||||
|
@ -21,12 +21,13 @@ import BooleanEditor from './components/BooleanEditor.svelte'
|
||||
import StatePresenter from './components/StatePresenter.svelte'
|
||||
import TimestampPresenter from './components/TimestampPresenter.svelte'
|
||||
import TableView from './components/TableView.svelte'
|
||||
import Table from './components/Table.svelte'
|
||||
import KanbanView from './components/KanbanView.svelte'
|
||||
|
||||
import { getClient, MessageBox } from '@anticrm/presentation'
|
||||
import { showPopup } from '@anticrm/ui'
|
||||
|
||||
export { TableView }
|
||||
export { Table }
|
||||
|
||||
function Delete(object: Doc): void {
|
||||
showPopup(MessageBox, {
|
||||
|
Loading…
Reference in New Issue
Block a user