mirror of
https://github.com/hcengineering/platform.git
synced 2025-04-14 20:39:03 +00:00
218 lines
5.9 KiB
Svelte
218 lines
5.9 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 { Ref, SortingOrder, WithLookup } from '@anticrm/core'
|
|
import { createQuery } from '@anticrm/presentation'
|
|
import { Issue, IssueStatus } from '@anticrm/tracker'
|
|
import {
|
|
Icon,
|
|
tooltip,
|
|
showPanel,
|
|
IconForward,
|
|
IconDetails,
|
|
showPopup,
|
|
SelectPopup,
|
|
closeTooltip,
|
|
Spinner,
|
|
getPlatformColor
|
|
} from '@anticrm/ui'
|
|
import tracker from '../../../plugin'
|
|
import { getIssueId } from '../../../issues'
|
|
import IssueStatusIcon from '../IssueStatusIcon.svelte'
|
|
|
|
export let issue: WithLookup<Issue>
|
|
|
|
const subIssuesQeury = createQuery()
|
|
|
|
let subIssues: WithLookup<Issue>[] | undefined
|
|
let subIssuesElement: Element
|
|
|
|
function openIssue (target: Ref<Issue>) {
|
|
if (target !== issue._id) {
|
|
showPanel(tracker.component.EditIssue, target, issue._class, 'content')
|
|
}
|
|
}
|
|
|
|
function openParentIssue () {
|
|
if (parentIssue) {
|
|
closeTooltip()
|
|
openIssue(parentIssue._id)
|
|
}
|
|
}
|
|
|
|
function showSubIssues () {
|
|
if (subIssues) {
|
|
closeTooltip()
|
|
showPopup(
|
|
SelectPopup,
|
|
{
|
|
value: subIssues.map((iss) => {
|
|
const team = iss.$lookup?.space
|
|
const status = iss.$lookup?.status as WithLookup<IssueStatus>
|
|
const icon = status.$lookup?.category?.icon
|
|
const color = status.color ?? status.$lookup?.category?.color
|
|
|
|
return {
|
|
id: iss._id,
|
|
icon,
|
|
isSelected: iss._id === issue._id,
|
|
...(team !== undefined ? { text: `${getIssueId(team, iss)} ${iss.title}` } : undefined),
|
|
...(color !== undefined ? { iconColor: getPlatformColor(color) } : undefined)
|
|
}
|
|
}),
|
|
width: 'large'
|
|
},
|
|
{
|
|
getBoundingClientRect: () => {
|
|
const rect = subIssuesElement.getBoundingClientRect()
|
|
const offsetX = 5
|
|
const offsetY = -1
|
|
|
|
return DOMRect.fromRect({ width: 1, height: 1, x: rect.right + offsetX, y: rect.top + offsetY })
|
|
}
|
|
},
|
|
(selectedIssue) => selectedIssue !== undefined && openIssue(selectedIssue)
|
|
)
|
|
}
|
|
}
|
|
|
|
$: areSubIssuesLoading = !subIssues
|
|
$: parentIssue = issue.$lookup?.attachedTo ? (issue.$lookup?.attachedTo as Issue) : null
|
|
$: if (parentIssue) {
|
|
subIssuesQeury.query(
|
|
tracker.class.Issue,
|
|
{ space: issue.space, attachedTo: parentIssue._id },
|
|
(res) => (subIssues = res),
|
|
{
|
|
sort: { modifiedOn: SortingOrder.Descending },
|
|
lookup: {
|
|
space: tracker.class.Team,
|
|
status: [tracker.class.IssueStatus, { category: tracker.class.IssueStatusCategory }]
|
|
}
|
|
}
|
|
)
|
|
} else {
|
|
subIssuesQeury.unsubscribe()
|
|
}
|
|
</script>
|
|
|
|
{#if parentIssue}
|
|
<div class="flex root">
|
|
<div class="item clear-mins">
|
|
<div
|
|
class="flex-center parent-issue cursor-pointer"
|
|
use:tooltip={{ label: tracker.string.OpenParent, direction: 'bottom' }}
|
|
on:click={openParentIssue}
|
|
>
|
|
{#if issue?.$lookup?.status}
|
|
<div class="pr-2">
|
|
<IssueStatusIcon value={issue.$lookup.status} size="small" />
|
|
</div>
|
|
{/if}
|
|
{#if issue.$lookup?.space}
|
|
<span class="overflow-label flex-no-shrink mr-2">{getIssueId(issue.$lookup.space, parentIssue)}</span>
|
|
{/if}
|
|
<span class="overflow-label issue-title">{parentIssue.title}</span>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="item">
|
|
{#if areSubIssuesLoading}
|
|
<div class="flex-center spinner">
|
|
<Spinner size="small" />
|
|
</div>
|
|
{:else}
|
|
<div
|
|
bind:this={subIssuesElement}
|
|
class="flex-center sub-issues cursor-pointer"
|
|
use:tooltip={{ label: tracker.string.OpenSubIssues, direction: 'bottom' }}
|
|
on:click|preventDefault={areSubIssuesLoading ? undefined : showSubIssues}
|
|
>
|
|
<span class="overflow-label">{subIssues?.length}</span>
|
|
<div class="ml-2">
|
|
<!-- TODO: fix icon -->
|
|
<Icon icon={IconDetails} size="small" />
|
|
</div>
|
|
<div class="ml-1-5">
|
|
<Icon icon={IconForward} size="small" />
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
|
|
<style lang="scss">
|
|
$padding: 0.375rem 0.75rem;
|
|
$border: 1px solid var(--button-border-color);
|
|
|
|
.root {
|
|
max-width: fit-content;
|
|
line-height: 150%;
|
|
border: $border;
|
|
border-radius: 0.25rem;
|
|
box-shadow: var(--primary-shadow);
|
|
|
|
.item {
|
|
position: relative;
|
|
|
|
&:not(:first-child)::before {
|
|
position: absolute;
|
|
content: '';
|
|
border-left: $border;
|
|
inset: 0.375rem auto;
|
|
}
|
|
}
|
|
}
|
|
|
|
.spinner {
|
|
padding: $padding;
|
|
height: 100%;
|
|
}
|
|
|
|
.parent-issue {
|
|
padding: $padding;
|
|
|
|
.issue-title {
|
|
color: var(--theme-content-accent-color);
|
|
transition: color 0.15s;
|
|
}
|
|
|
|
&:hover {
|
|
.issue-title {
|
|
color: var(--theme-caption-color);
|
|
}
|
|
}
|
|
&:active {
|
|
.issue-title {
|
|
color: var(--theme-content-accent-color);
|
|
}
|
|
}
|
|
}
|
|
|
|
.sub-issues {
|
|
padding: $padding;
|
|
color: var(--theme-content-color);
|
|
transition: color 0.15s;
|
|
|
|
&:hover {
|
|
color: var(--theme-caption-color);
|
|
}
|
|
&:active {
|
|
color: var(--theme-content-accent-color);
|
|
}
|
|
}
|
|
</style>
|