platform/plugins/tracker-resources/src/components/projects/CreateProject.svelte
Vyacheslav Tumanov 090348fc47
UBER-473: show icon for department ()
Signed-off-by: Vyacheslav Tumanov <me@slavatumanov.me>
2023-06-29 21:30:15 +07:00

334 lines
10 KiB
Svelte

<!--
// Copyright © 2022-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 { Employee } from '@hcengineering/contact'
import { AccountArrayEditor, AssigneeBox } from '@hcengineering/contact-resources'
import core, { Account, DocumentUpdate, Ref, generateId, getCurrentAccount } from '@hcengineering/core'
import { Asset } from '@hcengineering/platform'
import presentation, { Card, createQuery, getClient } from '@hcengineering/presentation'
import { StyledTextBox } from '@hcengineering/text-editor'
import { IssueStatus, Project, TimeReportDayType, createStatuses } from '@hcengineering/tracker'
import {
Button,
EditBox,
IconEdit,
IconWithEmoji,
Label,
Toggle,
eventToHTMLElement,
getColorNumberByText,
getPlatformColorDef,
getPlatformColorForTextDef,
showPopup,
themeStore
} from '@hcengineering/ui'
import { createEventDispatcher } from 'svelte'
import tracker from '../../plugin'
import ChangeIdentity from './ChangeIdentity.svelte'
import ProjectIconChooser from './ProjectIconChooser.svelte'
export let project: Project | undefined = undefined
const client = getClient()
const hierarchy = client.getHierarchy()
const projectsQuery = createQuery()
let name: string = project?.name ?? ''
let description: string = project?.description ?? ''
let isPrivate: boolean = project?.private ?? false
let icon: Asset | undefined = project?.icon ?? undefined
let color = project?.color ?? getColorNumberByText(name)
let isColorSelected = false
let defaultAssignee: Ref<Employee> | null | undefined = project?.defaultAssignee ?? null
let members: Ref<Account>[] =
project?.members !== undefined ? hierarchy.clone(project.members) : [getCurrentAccount()._id]
let projectsIdentifiers: Set<string> = new Set()
let isSaving = false
let changeIdentityRef: HTMLElement
const dispatch = createEventDispatcher()
$: isNew = !project
async function handleSave () {
if (isNew) {
await createProject()
} else {
await updateProject()
}
}
let identifier: string = project?.identifier ?? 'TSK'
const defaultStatusId: Ref<IssueStatus> = generateId()
function getProjectData () {
return {
name,
description,
private: isPrivate,
members,
archived: false,
identifier,
sequence: 0,
defaultIssueStatus: defaultStatusId,
defaultAssignee: defaultAssignee ?? undefined,
icon,
color,
defaultTimeReportDay: project?.defaultTimeReportDay ?? TimeReportDayType.PreviousWorkDay
}
}
async function updateProject () {
if (!project) {
return
}
const { sequence, defaultIssueStatus, ...projectData } = getProjectData()
const update: DocumentUpdate<Project> = {}
if (projectData.name !== project?.name) {
update.name = projectData.name
}
if (projectData.description !== project?.description) {
update.description = projectData.description
}
if (projectData.private !== project?.private) {
update.private = projectData.private
}
if (projectData.defaultAssignee !== project?.defaultAssignee) {
update.defaultAssignee = projectData.defaultAssignee
}
if (projectData.icon !== project?.icon) {
update.icon = projectData.icon
}
if (projectData.color !== project?.color) {
update.color = projectData.color
}
if (projectData.defaultTimeReportDay !== project?.defaultTimeReportDay) {
update.defaultTimeReportDay = projectData.defaultTimeReportDay
}
if (projectData.identifier !== project?.identifier) {
update.identifier = projectData.identifier
}
if (projectData.members.length !== project?.members.length) {
update.members = projectData.members
} else {
for (const member of projectData.members) {
if (project.members.findIndex((p) => p === member) === -1) {
update.members = projectData.members
break
}
}
}
if (Object.keys(update).length > 0) {
isSaving = true
await client.update(project, update)
isSaving = false
}
close()
}
async function createProject () {
const projectId = generateId<Project>()
const projectData = getProjectData()
const ops = client.apply(projectId).notMatch(tracker.class.Project, { identifier: projectData.identifier })
isSaving = true
await ops.createDoc(tracker.class.Project, core.space.Space, projectData, projectId)
await createStatuses(ops, projectId, tracker.class.IssueStatus, tracker.attribute.IssueStatus, defaultStatusId)
const succeeded = await ops.commit()
isSaving = false
if (succeeded) {
close()
} else {
changeIdentity(changeIdentityRef)
}
}
function chooseIcon (ev: MouseEvent) {
showPopup(ProjectIconChooser, { icon, color }, 'top', (result) => {
if (result !== undefined && result !== null) {
icon = result.icon
color = result.color
isColorSelected = true
}
})
}
function changeIdentity (element: HTMLElement) {
showPopup(ChangeIdentity, { identifier, projectsIdentifiers }, element, (result) => {
if (result != null) {
identifier = result.toLocaleUpperCase()
}
})
}
function close () {
dispatch('close')
}
$: projectsQuery.query(
tracker.class.Project,
{ _id: { $nin: project ? [project._id] : [] } },
(res) => (projectsIdentifiers = new Set(res.map(({ identifier }) => identifier)))
)
</script>
<Card
label={isNew ? tracker.string.NewProject : tracker.string.EditProject}
okLabel={isNew ? presentation.string.Create : presentation.string.Save}
okAction={handleSave}
canSave={name.length > 0 &&
identifier.length > 0 &&
!projectsIdentifiers.has(identifier) &&
!(members.length === 0 && isPrivate)}
accentHeader
width={'medium'}
gap={'gapV-6'}
onCancel={close}
on:changeContent
>
<div class="antiGrid">
<div class="antiGrid-row">
<div class="antiGrid-row__header">
<Label label={tracker.string.ProjectTitle} />
</div>
<div class="padding">
<EditBox
bind:value={name}
placeholder={tracker.string.ProjectTitlePlaceholder}
kind={'large-style'}
autoFocus
on:input={() => {
if (isNew) {
identifier = name.toLocaleUpperCase().replaceAll(' ', '_').substring(0, 5)
color = isColorSelected ? color : getColorNumberByText(name)
}
}}
/>
</div>
</div>
<div class="antiGrid-row">
<div class="antiGrid-row__header withDesciption">
<Label label={tracker.string.Identifier} />
<span><Label label={tracker.string.UsedInIssueIDs} /></span>
</div>
<div bind:this={changeIdentityRef} class="padding flex-row-center relative">
<EditBox
bind:value={identifier}
disabled={!isNew}
placeholder={tracker.string.ProjectIdentifierPlaceholder}
kind={'large-style'}
uppercase
/>
{#if !isNew}
<div class="ml-1">
<Button size={'small'} icon={IconEdit} on:click={(ev) => changeIdentity(eventToHTMLElement(ev))} />
</div>
{:else if !isSaving && projectsIdentifiers.has(identifier)}
<div class="absolute overflow-label duplicated-identifier">
<Label label={tracker.string.IdentifierExists} />
</div>
{/if}
</div>
</div>
<div class="antiGrid-row">
<div class="antiGrid-row__header topAlign">
<Label label={tracker.string.Description} />
</div>
<div class="padding clear-mins">
<StyledTextBox
alwaysEdit
showButtons={false}
bind:content={description}
placeholder={tracker.string.IssueDescriptionPlaceholder}
/>
</div>
</div>
</div>
<div class="antiGrid">
<div class="antiGrid-row">
<div class="antiGrid-row__header">
<Label label={tracker.string.ChooseIcon} />
</div>
<Button
icon={icon === tracker.component.IconWithEmoji ? IconWithEmoji : icon ?? tracker.icon.Home}
iconProps={icon === tracker.component.IconWithEmoji
? { icon: color }
: {
fill:
color !== undefined
? getPlatformColorDef(color, $themeStore.dark).icon
: getPlatformColorForTextDef(name, $themeStore.dark).icon
}}
size={'large'}
on:click={chooseIcon}
/>
</div>
<div class="antiGrid-row">
<div class="antiGrid-row__header withDesciption">
<Label label={presentation.string.MakePrivate} />
<span><Label label={presentation.string.MakePrivateDescription} /></span>
</div>
<Toggle bind:on={isPrivate} disabled={!isPrivate && members.length === 0} />
</div>
<div class="antiGrid-row">
<div class="antiGrid-row__header">
<Label label={tracker.string.Members} />
</div>
<AccountArrayEditor
value={members}
label={tracker.string.Members}
onChange={(refs) => (members = refs)}
kind={'secondary'}
size={'large'}
/>
</div>
<div class="antiGrid-row">
<div class="antiGrid-row__header">
<Label label={tracker.string.DefaultAssignee} />
</div>
<AssigneeBox
label={tracker.string.Assignee}
placeholder={tracker.string.Assignee}
kind={'secondary'}
size={'large'}
avatarSize={'card'}
bind:value={defaultAssignee}
titleDeselect={tracker.string.Unassigned}
showNavigate={false}
showTooltip={{ label: tracker.string.DefaultAssignee }}
/>
</div>
</div>
</Card>
<style lang="scss">
.duplicated-identifier {
left: 0;
bottom: -0.25rem;
color: var(--theme-warning-color);
}
</style>