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 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 {
|
||||||
|
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%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.table-body { width: 100%; }
|
||||||
|
|
||||||
.firstCell {
|
.firstCell {
|
||||||
display: flex;
|
display: flex;
|
||||||
// justify-content: space-between;
|
// justify-content: space-between;
|
||||||
|
@ -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, {
|
||||||
|
Loading…
Reference in New Issue
Block a user