Add Table (#282)

Signed-off-by: Alexander Platov <sas_lord@mail.ru>
This commit is contained in:
Alexander Platov 2021-10-23 17:49:11 +03:00 committed by GitHub
parent 3eec4d03ae
commit 3f3953de3a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 222 additions and 18 deletions

View File

@ -28,7 +28,7 @@
import Edit from './icons/Edit.svelte' import Edit from './icons/Edit.svelte'
import SocialEditor from './SocialEditor.svelte' import SocialEditor from './SocialEditor.svelte'
import AttributesBar from './AttributesBar.svelte' import AttributesBar from './AttributesBar.svelte'
import { TableView } from '@anticrm/view-resources' import { Table } from '@anticrm/view-resources'
import core from '@anticrm/core' import core from '@anticrm/core'
@ -89,24 +89,25 @@
</div> </div>
</div> </div>
Applications <div class="group">
<div class="caption">Applications</div>
<TableView <Table
_class={recruit.class.Applicant} _class={recruit.class.Applicant}
config={['$lookup.candidate', '$lookup.state', '$lookup.candidate.city', '$lookup.space.name']} config={['$lookup.candidate', '$lookup.state', '$lookup.candidate.city', '$lookup.space.name']}
options={ options={
{ {
lookup: { lookup: {
candidate: recruit.class.Candidate, candidate: recruit.class.Candidate,
state: core.class.State, state: core.class.State,
space: core.class.Space space: core.class.Space
}
} }
} }
} search=""
search="" />
/> </div>
<div class="attachments"> <div class="group">
<Attachments objectId={object._id} _class={object._class} space={object.space} {object}/> <Attachments objectId={object._id} _class={object._class} space={object.space} {object}/>
</div> </div>
@ -162,8 +163,14 @@
span { margin-left: .5rem; } span { margin-left: .5rem; }
} }
.attachments { .group {
margin-top: 3.5rem; margin-top: 3.5rem;
.caption {
font-weight: 500;
font-size: 1.25rem;
color: var(--theme-caption-color);
}
} }
// .container { // .container {

View 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>

View File

@ -138,6 +138,8 @@
height: 100%; height: 100%;
} }
.table-body { width: 100%; }
.firstCell { .firstCell {
display: flex; display: flex;
// justify-content: space-between; // justify-content: space-between;

View File

@ -21,12 +21,13 @@ import BooleanEditor from './components/BooleanEditor.svelte'
import StatePresenter from './components/StatePresenter.svelte' import StatePresenter from './components/StatePresenter.svelte'
import TimestampPresenter from './components/TimestampPresenter.svelte' import TimestampPresenter from './components/TimestampPresenter.svelte'
import TableView from './components/TableView.svelte' import TableView from './components/TableView.svelte'
import Table from './components/Table.svelte'
import KanbanView from './components/KanbanView.svelte' import KanbanView from './components/KanbanView.svelte'
import { getClient, MessageBox } from '@anticrm/presentation' import { getClient, MessageBox } from '@anticrm/presentation'
import { showPopup } from '@anticrm/ui' import { showPopup } from '@anticrm/ui'
export { TableView } export { Table }
function Delete(object: Doc): void { function Delete(object: Doc): void {
showPopup(MessageBox, { showPopup(MessageBox, {