mirror of
https://github.com/hcengineering/platform.git
synced 2025-02-07 19:38:46 +00:00
6f7e95e233
Signed-off-by: Andrey Sobolev <haiodo@gmail.com>
190 lines
6.1 KiB
Svelte
190 lines
6.1 KiB
Svelte
<!--
|
|
// Copyright © 2020 Anticrm Platform Contributors.
|
|
//
|
|
// 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 { Contact, EmployeeAccount, Organization, Person } from '@anticrm/contact'
|
|
import contact from '@anticrm/contact'
|
|
import { OrganizationSelector } from '@anticrm/contact-resources'
|
|
import { Account, Class, Client, Doc, generateId, getCurrentAccount, Ref } from '@anticrm/core'
|
|
import { getResource, OK, Resource, Severity, Status } from '@anticrm/platform'
|
|
import { Card, getClient, UserBox } from '@anticrm/presentation'
|
|
import type { Candidate, Review } from '@anticrm/recruit'
|
|
import task, { SpaceWithStates } from '@anticrm/task'
|
|
import { StyledTextBox } from '@anticrm/text-editor'
|
|
import { DatePicker, Grid, Status as StatusControl, StylishEdit } from '@anticrm/ui'
|
|
import view from '@anticrm/view'
|
|
import { createEventDispatcher } from 'svelte'
|
|
import recruit from '../../plugin'
|
|
|
|
export let space: Ref<SpaceWithStates>
|
|
export let candidate: Ref<Person>
|
|
|
|
export let preserveCandidate = false
|
|
|
|
const currentUser = getCurrentAccount() as EmployeeAccount
|
|
|
|
let status: Status = OK
|
|
|
|
let title: string = ''
|
|
let description: string = ''
|
|
let startDate: Date = new Date()
|
|
let dueDate: Date = new Date()
|
|
let location: string = ''
|
|
let company: Ref<Organization> | undefined = undefined
|
|
|
|
const doc: Review = {
|
|
number: 0,
|
|
attachedTo: candidate,
|
|
attachedToClass: recruit.mixin.Candidate,
|
|
_class: recruit.class.Review,
|
|
space: space,
|
|
_id: generateId(),
|
|
collection: 'reviews',
|
|
modifiedOn: Date.now(),
|
|
modifiedBy: '' as Ref<Account>,
|
|
date: 0,
|
|
dueDate: undefined,
|
|
description,
|
|
company,
|
|
verdict: '',
|
|
title,
|
|
participants: [currentUser.employee]
|
|
}
|
|
|
|
const dispatch = createEventDispatcher()
|
|
const client = getClient()
|
|
const hierarchy = client.getHierarchy()
|
|
|
|
export function canClose (): boolean {
|
|
return candidate === undefined
|
|
}
|
|
|
|
let spaceLabel: string = ''
|
|
|
|
$: client.findOne(recruit.class.ReviewCategory, { _id: doc.space }).then((res) => {
|
|
if (res !== undefined) {
|
|
spaceLabel = res.name
|
|
}
|
|
})
|
|
|
|
async function createReview () {
|
|
const sequence = await client.findOne(task.class.Sequence, { attachedTo: recruit.class.Review })
|
|
if (sequence === undefined) {
|
|
throw new Error('sequence object not found')
|
|
}
|
|
|
|
const incResult = await client.update(sequence, { $inc: { sequence: 1 } }, true)
|
|
|
|
const candidateInstance = await client.findOne(contact.class.Person, { _id: doc.attachedTo as Ref<Person> })
|
|
if (candidateInstance === undefined) {
|
|
throw new Error('contact not found')
|
|
}
|
|
if (!client.getHierarchy().hasMixin(candidateInstance, recruit.mixin.Candidate)) {
|
|
await client.createMixin<Contact, Candidate>(
|
|
candidateInstance._id,
|
|
candidateInstance._class,
|
|
candidateInstance.space,
|
|
recruit.mixin.Candidate,
|
|
{}
|
|
)
|
|
}
|
|
|
|
await client.addCollection(recruit.class.Review, doc.space, doc.attachedTo, doc.attachedToClass, 'reviews', {
|
|
number: (incResult as any).object.sequence,
|
|
date: startDate?.getTime() ?? null,
|
|
dueDate: dueDate?.getTime() ?? null,
|
|
description,
|
|
verdict: '',
|
|
title,
|
|
company,
|
|
location
|
|
})
|
|
}
|
|
|
|
async function invokeValidate (
|
|
action: Resource<<T extends Doc>(doc: T, client: Client) => Promise<Status>>
|
|
): Promise<Status> {
|
|
const impl = await getResource(action)
|
|
return await impl(doc, client)
|
|
}
|
|
|
|
async function validate (doc: Review, _class: Ref<Class<Doc>>): Promise<void> {
|
|
const clazz = hierarchy.getClass(_class)
|
|
const validatorMixin = hierarchy.as(clazz, view.mixin.ObjectValidator)
|
|
if (validatorMixin?.validator != null) {
|
|
status = await invokeValidate(validatorMixin.validator)
|
|
} else if (clazz.extends != null) {
|
|
await validate(doc, clazz.extends)
|
|
} else {
|
|
status = OK
|
|
}
|
|
}
|
|
|
|
$: validate(doc, doc._class)
|
|
</script>
|
|
|
|
<Card
|
|
size={'medium'}
|
|
label={recruit.string.CreateReviewParams}
|
|
labelProps={{ label: spaceLabel }}
|
|
okAction={createReview}
|
|
canSave={status.severity === Severity.OK && title.trim().length > 0}
|
|
spaceClass={recruit.class.ReviewCategory}
|
|
spaceQuery={{ archived: false }}
|
|
spaceLabel={recruit.string.ReviewCategory}
|
|
spacePlaceholder={recruit.string.SelectReviewCategory}
|
|
bind:space={doc.space}
|
|
on:close={() => {
|
|
dispatch('close')
|
|
}}
|
|
>
|
|
<StatusControl slot="error" {status} />
|
|
|
|
<Grid column={1} rowGap={1.75}>
|
|
<Grid column={!preserveCandidate ? 2 : 1}>
|
|
<StylishEdit bind:value={title} label={recruit.string.Title} />
|
|
{#if !preserveCandidate}
|
|
<div class="antiComponentBox">
|
|
<UserBox
|
|
_class={contact.class.Person}
|
|
title={recruit.string.Candidate}
|
|
caption={recruit.string.Candidates}
|
|
bind:value={doc.attachedTo}
|
|
/>
|
|
</div>
|
|
{/if}
|
|
</Grid>
|
|
<StyledTextBox
|
|
emphasized
|
|
showButtons={false}
|
|
bind:content={description}
|
|
label={recruit.string.Description}
|
|
alwaysEdit
|
|
placeholder={recruit.string.AddDescription}
|
|
/>
|
|
<Grid column={2}>
|
|
<StylishEdit bind:value={location} label={recruit.string.Location} />
|
|
<div class="antiComponentBox">
|
|
<OrganizationSelector bind:value={company} label={recruit.string.Company} />
|
|
</div>
|
|
</Grid>
|
|
<div class="antiComponentBox">
|
|
<DatePicker title={recruit.string.StartDate} bind:value={startDate} withTime />
|
|
</div>
|
|
<div class="antiComponentBox">
|
|
<DatePicker title={recruit.string.DueDate} bind:value={dueDate} withTime />
|
|
</div>
|
|
</Grid>
|
|
</Card>
|