platform/plugins/tracker-resources/src/components/issues/Move.svelte
Denis Bykhov 2422baa108
UBER-808 (#3656)
Signed-off-by: Denis Bykhov <bykhov.denis@gmail.com>
2023-09-04 23:06:34 +06:00

396 lines
13 KiB
Svelte

<!--
// Copyright © 2020, 2021 Anticrm Platform Contributors.
// Copyright © 2021, 2023 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 { createEventDispatcher } from 'svelte'
import { Ref, Status } from '@hcengineering/core'
import { SpaceSelector, createQuery, getClient, Card } from '@hcengineering/presentation'
import { Component, Issue, IssueStatus, Milestone, Project } from '@hcengineering/tracker'
import ui, { Button, IconForward, Label, Spinner, Toggle, tooltip } from '@hcengineering/ui'
import view from '@hcengineering/view'
import { statusStore } from '@hcengineering/view-resources'
import tracker from '../../plugin'
import {
ComponentToUpdate,
IssueToUpdate,
StatusToUpdate,
collectIssues,
findTargetStatus,
moveIssueToSpace
} from '../../utils'
import { componentStore } from '../../component'
import ProjectPresenter from '../projects/ProjectPresenter.svelte'
import IssuePresenter from './IssuePresenter.svelte'
import TitlePresenter from './TitlePresenter.svelte'
import ComponentMovePresenter from './move/ComponentMovePresenter.svelte'
import StatusMovePresenter from './move/StatusMovePresenter.svelte'
import SelectReplacement from './move/SelectReplacement.svelte'
import PriorityEditor from './PriorityEditor.svelte'
import { getStates } from '@hcengineering/task'
export let selected: Issue | Issue[]
$: docs = Array.isArray(selected) ? selected : [selected]
const client = getClient()
const dispatch = createEventDispatcher()
const hierarchy = client.getHierarchy()
let targetProject: Project | undefined
let space: Ref<Project>
$: {
const doc = docs[0]
if (space === undefined) {
space = doc.space
}
}
let processing = false
async function createMissingStatus (st: Ref<Status>): Promise<void> {
if (targetProject) {
await client.update(targetProject, { $push: { states: st } })
}
}
async function createMissingComponent (c: Ref<Component>): Promise<void> {
const cur = $componentStore.get(c)
const components = $componentStore.filter((it) => it.space === targetProject?._id)
if (
cur === undefined ||
targetProject === undefined ||
components.find((c) => c.label === cur.label) !== undefined
) {
return
}
await client.createDoc(cur._class, targetProject._id, {
label: cur.label,
attachments: 0,
description: cur.description,
comments: 0,
lead: cur.lead
})
}
const moveAll = async () => {
if (targetProject === undefined) {
return
}
processing = true
for (const issue of toMove) {
const upd = issueToUpdate.get(issue._id) ?? {}
if (issue.status !== undefined) {
if (!upd.useStatus) {
const newStatus = statusToUpdate[issue.status]
if (newStatus !== undefined) {
if (newStatus.create) {
await createMissingStatus(newStatus.ref)
}
upd.status = newStatus.ref
}
} else if (upd.createStatus) {
await createMissingStatus(issue.status)
}
}
if (issue.component !== undefined && issue.component !== null) {
if (!upd.useComponent) {
const newComponent = componentToUpdate[issue.component]
if (newComponent !== undefined) {
if (newComponent.create) {
await createMissingComponent(newComponent.ref)
}
upd.component = newComponent.ref
}
} else if (upd.createComponent) {
await createMissingComponent(issue.component)
}
}
issueToUpdate.set(issue._id, upd)
}
await moveIssueToSpace(client, docs, targetProject, issueToUpdate)
processing = false
dispatch('close')
}
const targetSpaceQuery = createQuery()
let issueToUpdate: Map<Ref<Issue>, IssueToUpdate> = new Map()
let statusToUpdate: Record<Ref<IssueStatus>, StatusToUpdate | undefined> = {}
let componentToUpdate: Record<Ref<Component>, ComponentToUpdate | undefined> = {}
$: targetSpaceQuery.query(tracker.class.Project, { _id: space }, (res) => {
;[targetProject] = res
})
let toMove: Issue[] = []
let loading = true
$: {
collectIssues(client, docs).then((res) => {
toMove = res
loading = false
})
}
$: if (keepOriginalAttribytes) {
setOriginalAttributes()
} else if (targetProject !== undefined) {
setReplacementAttributres(targetProject)
}
const componentQuery = createQuery()
let components: Component[] = []
$: componentQuery.query(tracker.class.Component, {}, (res) => {
components = res
})
const milestoneQuery = createQuery()
let milestones: Milestone[] = []
$: milestoneQuery.query(tracker.class.Milestone, {}, (res) => {
milestones = res
})
$: statuses = getStates(targetProject, $statusStore)
let keepOriginalAttribytes: boolean = false
let showManageAttributes: boolean = false
$: isManageAttributesAvailable = issueToUpdate.size > 0 && docs[0]?.space !== targetProject?._id
function setOriginalAttributes () {
for (const issue of toMove) {
const upd = issueToUpdate.get(issue._id) ?? {}
upd.createStatus = false
upd.useStatus = false
upd.createComponent = false
upd.useComponent = false
issueToUpdate.set(issue._id, upd)
}
for (const status in Object.keys(statusToUpdate)) {
statusToUpdate[status] = { ref: status, create: true }
}
for (const component in Object.keys(componentToUpdate)) {
componentToUpdate[component] = { ref: component, create: true }
}
for (const issue of toMove) {
let upd = issueToUpdate.get(issue._id) ?? {}
if (issue.status !== undefined) {
upd = {
...upd,
status: issue.status
}
}
if (issue.component !== undefined && issue.component !== null) {
upd = {
...upd,
component: issue.component
}
}
issueToUpdate.set(issue._id, upd)
}
}
function setReplacementAttributres (currentSpace: Project) {
for (const issue of toMove) {
const upd = issueToUpdate.get(issue._id) ?? {}
// In case of target space change
if (upd.status !== undefined && $statusStore.get(upd.status)?.space !== currentSpace._id && !upd.createStatus) {
upd.status = undefined
}
if (upd.status === undefined) {
upd.status = findTargetStatus(issue.status, currentSpace, $statusStore, true) ?? currentSpace.defaultIssueStatus
}
if (issue.component !== undefined) {
const cur = components.find((it) => it._id === issue.component)
if (cur !== undefined) {
if (
upd.component !== undefined &&
components.find((it) => it._id === upd.component)?.space !== currentSpace._id
) {
upd.component = undefined
}
if (upd.component === undefined) {
upd.component = components.find((it) => it.space === currentSpace?._id && it.label === cur.label)?._id
}
}
}
if (issue.milestone != null) {
const currentMilestone = milestones.find((it) => it._id === issue.milestone)
if (currentMilestone !== undefined) {
if (upd.milestone === undefined) {
upd.milestone = milestones.find(
(it) => it.space === currentSpace?._id && it.label === currentMilestone.label
)?._id
}
}
}
if (issue.attachedTo !== tracker.ids.NoParent && toMove.find((it) => it._id === issue.attachedTo) === undefined) {
upd.attachedTo = tracker.ids.NoParent
upd.attachedToClass = tracker.class.Issue
}
issueToUpdate.set(issue._id, upd)
}
}
</script>
<Card
label={showManageAttributes ? tracker.string.ManageAttributes : tracker.string.MoveIssues}
okLabel={view.string.Move}
okAction={moveAll}
canSave={docs[0]?.space !== targetProject?._id}
onCancel={() => dispatch('close')}
backAction={() => {
showManageAttributes = !showManageAttributes
}}
isBack={showManageAttributes}
thinHeader
accentHeader
hideSubheader={showManageAttributes}
hideContent={showManageAttributes}
hideAttachments
numberOfBlocks={showManageAttributes
? toMove.length
: targetProject !== undefined && !keepOriginalAttribytes && docs[0]?.space !== targetProject?._id
? 1
: 0}
on:changeContent
>
<svelte:fragment slot="title" let:label>
{#if !showManageAttributes}
<Label {label} />
{#if Array.isArray(selected) && selected.length}
<span class="content-dark-color ml-1-5">{selected.length}</span>
{/if}
{:else}
<Label {label} />
{/if}
</svelte:fragment>
<svelte:fragment slot="subheader">
{#if !showManageAttributes}
<span class="content-halfcontent-color overflow-label" style:margin-top={'-.5rem'}>
<Label label={tracker.string.MoveIssuesDescription} />
</span>
{/if}
</svelte:fragment>
{#if !showManageAttributes}
<div class="flex-between">
{#if targetProject !== undefined}
<SpaceSelector
_class={targetProject._class}
label={hierarchy.getClass(tracker.class.Project).label}
bind:space
kind={'regular'}
size={'large'}
component={ProjectPresenter}
iconWithEmoji={tracker.component.IconWithEmoji}
defaultIcon={tracker.icon.Home}
/>
<Button
label={tracker.string.ManageAttributes}
iconRight={IconForward}
kind={'ghost'}
size={'small'}
padding={'0 1rem'}
disabled={!isManageAttributesAvailable}
on:click={() => {
if (!isManageAttributesAvailable) {
return
}
showManageAttributes = !showManageAttributes
}}
/>
{/if}
</div>
{:else if loading}<Spinner />{/if}
<svelte:fragment slot="blocks" let:block>
{#if !showManageAttributes}
{#if targetProject !== undefined && !keepOriginalAttribytes}
<SelectReplacement
currentProject={space}
{statuses}
{components}
{targetProject}
issues={toMove}
bind:statusToUpdate
bind:componentToUpdate
/>
{/if}
{:else if toMove.length > 0 && targetProject}
{@const issue = toMove[block]}
{@const upd = issueToUpdate.get(issue._id) ?? {}}
{@const originalComponent = components.find((it) => it._id === issue.component)}
{@const targetComponent = components.find(
(it) => it.space === targetProject?._id && it.label === originalComponent?.label
)}
{#key keepOriginalAttribytes}
{#if issue.space !== targetProject._id && (upd.status !== undefined || upd.component !== undefined)}
<div class="flex-row-center min-h-9 gap-1-5 content-color">
<PriorityEditor value={issue} isEditable={false} kind={'list'} size={'small'} shouldShowLabel={false} />
<IssuePresenter value={issue} disabled kind={'list'} />
<TitlePresenter disabled value={issue} showParent={false} maxWidth={'7.5rem'} />
</div>
{#key upd.status}
<StatusMovePresenter currentProject={space} {issue} bind:issueToUpdate {targetProject} {statuses} />
{/key}
{#if targetComponent === undefined}
{#key upd.component}
<ComponentMovePresenter {issue} bind:issueToUpdate {targetProject} {components} />
{/key}
{/if}
{/if}
{/key}
{/if}
</svelte:fragment>
<svelte:fragment slot="footer">
<div
class="flex-row-center gap-2"
use:tooltip={{
component: Label,
props: { label: tracker.string.KeepOriginalAttributesTooltip }
}}
>
<Toggle
disabled={!isManageAttributesAvailable}
on:change={() => {
keepOriginalAttribytes = !keepOriginalAttribytes
if (!keepOriginalAttribytes) {
statusToUpdate = {}
componentToUpdate = {}
}
}}
/>
<span class="lines-limit-2">
<Label label={tracker.string.KeepOriginalAttributes} />
</span>
</div>
</svelte:fragment>
<svelte:fragment slot="buttons">
<Button label={ui.string.Cancel} size={'large'} disabled={processing} on:click={() => dispatch('close')} />
</svelte:fragment>
</Card>