platform/plugins/tracker-resources/src/components/issues/EditIssue.svelte
Artyom Grigorovich 84844b65a9
Tracker: Project - Project selector (#1740)
Signed-off-by: Artyom Grigorovich <grigorovichartyom@gmail.com>
2022-05-16 17:38:51 +07:00

259 lines
7.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 contact from '@anticrm/contact'
import { Class, Ref, SortingOrder, WithLookup } from '@anticrm/core'
import { Panel } from '@anticrm/panel'
import { createQuery, getClient, UserBox } from '@anticrm/presentation'
import { StyledTextBox } from '@anticrm/text-editor'
import type { Issue, IssueStatus, Team } from '@anticrm/tracker'
import {
Button,
DatePresenter,
EditBox,
IconDownOutline,
IconEdit,
IconMoreH,
IconUpOutline,
Label
} from '@anticrm/ui'
import { createEventDispatcher, onMount } from 'svelte'
import tracker from '../../plugin'
import IssuePresenter from './IssuePresenter.svelte'
import PriorityEditor from './PriorityEditor.svelte'
import ProjectEditor from './ProjectEditor.svelte'
import StatusEditor from './StatusEditor.svelte'
export let _id: Ref<Issue>
export let _class: Ref<Class<Issue>>
const dispatch = createEventDispatcher()
const client = getClient()
const query = createQuery()
const statusesQuery = createQuery()
let issue: Issue | undefined
let currentTeam: Team | undefined
let issueStatuses: WithLookup<IssueStatus>[] | undefined
let innerWidth: number
$: _id &&
_class &&
query.query(_class, { _id }, async (result) => {
issue = result[0]
})
$: if (issue !== undefined) {
client.findOne(tracker.class.Team, { _id: issue.space }).then((r) => {
currentTeam = r
})
}
$: currentTeam &&
statusesQuery.query(
tracker.class.IssueStatus,
{ attachedTo: currentTeam._id },
(statuses) => {
issueStatuses = statuses
},
{
lookup: { category: tracker.class.IssueStatusCategory },
sort: { rank: SortingOrder.Ascending }
}
)
$: issueLabel = currentTeam && issue && `${currentTeam.identifier}-${issue.number}`
function change (field: string, value: any) {
if (issue !== undefined) {
client.update(issue, { [field]: value })
}
}
function copy (text: string): void {
navigator.clipboard.writeText(text)
}
onMount(() => {
dispatch('open', { ignoreKeys: ['comments', 'name', 'description', 'number'] })
})
</script>
{#if issue !== undefined}
<Panel
object={issue}
isHeader
isAside={true}
isSub={false}
bind:innerWidth
on:close={() => {
dispatch('close')
}}
>
<svelte:fragment slot="custom-title">Custom Title</svelte:fragment>
<svelte:fragment slot="subtitle">
<div class="flex-between flex-grow">
{#if currentTeam}
<IssuePresenter value={issue} {currentTeam} />
{/if}
<div class="buttons-group xsmall-gap">
<Button icon={IconEdit} kind={'transparent'} size={'medium'} />
{#if innerWidth < 900}
<Button icon={IconMoreH} kind={'transparent'} size={'medium'} />
{/if}
</div>
</div>
</svelte:fragment>
<svelte:fragment slot="navigator">
<Button icon={IconDownOutline} kind={'secondary'} size={'medium'} />
<Button icon={IconUpOutline} kind={'secondary'} size={'medium'} />
</svelte:fragment>
<svelte:fragment slot="header">
<span class="fs-title">{issueLabel}</span>
</svelte:fragment>
<svelte:fragment slot="tools">
<Button icon={IconEdit} kind={'transparent'} size={'medium'} />
<Button icon={IconMoreH} kind={'transparent'} size={'medium'} />
</svelte:fragment>
<div class="mt-6">
<EditBox
label={tracker.string.Title}
bind:value={issue.title}
placeholder={tracker.string.IssueTitlePlaceholder}
maxWidth={'16rem'}
focus
on:change={() => change('title', issue?.title)}
/>
</div>
<div class="mt-6 mb-6">
<StyledTextBox
alwaysEdit
bind:content={issue.description}
placeholder={tracker.string.IssueDescriptionPlaceholder}
on:value={(evt) => evt.detail !== issue?.description && change('description', evt.detail)}
/>
</div>
<span slot="actions-label">{issueLabel}</span>
<svelte:fragment slot="actions">
<Button
icon={tracker.icon.Issue}
title={tracker.string.CopyIssueUrl}
width="min-content"
size="small"
kind="transparent"
on:click={() => copy(window.location.href)}
/>
<Button
icon={tracker.icon.Views}
title={tracker.string.CopyIssueId}
width="min-content"
size="small"
kind="transparent"
on:click={() => issueLabel && copy(issueLabel)}
/>
</svelte:fragment>
<svelte:fragment slot="custom-attributes" let:direction>
{#if issue && currentTeam && issueStatuses && direction === 'column'}
<div class="content">
<span class="label">
<Label label={tracker.string.Status} />
</span>
<StatusEditor value={issue} statuses={issueStatuses} shouldShowLabel />
<span class="label">
<Label label={tracker.string.Priority} />
</span>
<PriorityEditor value={issue} shouldShowLabel />
<span class="label">
<Label label={tracker.string.Assignee} />
</span>
<UserBox
_class={contact.class.Employee}
label={tracker.string.Assignee}
placeholder={tracker.string.Assignee}
bind:value={issue.assignee}
allowDeselect
titleDeselect={tracker.string.Unassigned}
size={'large'}
kind={'link'}
width={'100%'}
justify={'left'}
on:change={() => change('assignee', issue?.assignee)}
/>
<span class="label">
<Label label={tracker.string.Labels} />
</span>
<Button
label={tracker.string.Labels}
icon={tracker.icon.Labels}
size={'large'}
kind={'link'}
width={'100%'}
justify={'left'}
/>
<div class="divider" />
<span class="label">
<Label label={tracker.string.Project} />
</span>
<ProjectEditor value={issue} />
{#if issue.dueDate !== null}
<div class="divider" />
<span class="label">
<Label label={tracker.string.DueDate} />
</span>
<DatePresenter bind:value={issue.dueDate} editable on:change={({ detail }) => change('dueDate', detail)} />
{/if}
</div>
{:else}
<div class="buttons-group small-gap">
<Button
label={tracker.string.Labels}
icon={tracker.icon.Labels}
width="min-content"
size="small"
kind="no-border"
/>
<ProjectEditor value={issue} size={'small'} kind={'no-border'} width={'min-content'} />
</div>
{/if}
</svelte:fragment>
</Panel>
{/if}
<style lang="scss">
.content {
display: grid;
grid-template-columns: 1fr 1.5fr;
grid-auto-flow: row;
justify-content: start;
align-items: center;
gap: 1rem;
margin-top: 1rem;
width: 100%;
height: min-content;
}
.divider {
grid-column: 1 / 3;
height: 1px;
background-color: var(--divider-color);
}
</style>