mirror of
https://github.com/hcengineering/platform.git
synced 2025-04-03 06:21:28 +00:00
433 lines
12 KiB
TypeScript
433 lines
12 KiB
TypeScript
//
|
|
// 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.
|
|
//
|
|
|
|
import type { Employee } from '@anticrm/contact'
|
|
import contact from '@anticrm/contact'
|
|
import { Domain, DOMAIN_MODEL, IndexKind, Markup, Ref, Timestamp, Type } from '@anticrm/core'
|
|
import {
|
|
ArrOf,
|
|
Builder,
|
|
Collection,
|
|
Hidden,
|
|
Index,
|
|
Model,
|
|
Prop,
|
|
TypeDate,
|
|
TypeMarkup,
|
|
TypeNumber,
|
|
TypeRef,
|
|
TypeString,
|
|
UX
|
|
} from '@anticrm/model'
|
|
import attachment from '@anticrm/model-attachment'
|
|
import chunter from '@anticrm/model-chunter'
|
|
import core, { DOMAIN_SPACE, TAttachedDoc, TDoc, TSpace, TType } from '@anticrm/model-core'
|
|
import { createAction } from '@anticrm/model-view'
|
|
import workbench, { createNavigateAction } from '@anticrm/model-workbench'
|
|
import { Asset, IntlString } from '@anticrm/platform'
|
|
import view, { KeyBinding } from '@anticrm/view'
|
|
import {
|
|
Document,
|
|
Issue,
|
|
IssuePriority,
|
|
IssueStatus,
|
|
IssueStatusCategory,
|
|
Project,
|
|
ProjectStatus,
|
|
Team
|
|
} from '@anticrm/tracker'
|
|
import tracker from './plugin'
|
|
|
|
export { trackerOperation } from './migration'
|
|
export { default } from './plugin'
|
|
|
|
export const DOMAIN_TRACKER = 'tracker' as Domain
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
@Model(tracker.class.IssueStatus, core.class.AttachedDoc, DOMAIN_TRACKER)
|
|
export class TIssueStatus extends TAttachedDoc implements IssueStatus {
|
|
name!: string
|
|
description?: string
|
|
color?: number
|
|
|
|
@Prop(TypeRef(tracker.class.IssueStatusCategory), tracker.string.StatusCategory)
|
|
category!: Ref<IssueStatusCategory>
|
|
|
|
@Prop(TypeString(), tracker.string.Rank)
|
|
@Hidden()
|
|
rank!: string
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
@Model(tracker.class.IssueStatusCategory, core.class.Doc, DOMAIN_MODEL)
|
|
export class TIssueStatusCategory extends TDoc implements IssueStatusCategory {
|
|
label!: IntlString
|
|
icon!: Asset
|
|
color!: number
|
|
defaultStatusName!: string
|
|
order!: number
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
export function TypeIssuePriority (): Type<IssuePriority> {
|
|
return { _class: tracker.class.TypeIssuePriority, label: 'TypeIssuePriority' as IntlString }
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
@Model(tracker.class.TypeIssuePriority, core.class.Type, DOMAIN_MODEL)
|
|
export class TTypeIssuePriority extends TType {}
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
@Model(tracker.class.Team, core.class.Space, DOMAIN_SPACE)
|
|
@UX(tracker.string.Team, tracker.icon.Team, tracker.string.Team)
|
|
export class TTeam extends TSpace implements Team {
|
|
@Prop(TypeString(), tracker.string.Title)
|
|
@Index(IndexKind.FullText)
|
|
reamLogo!: IntlString
|
|
|
|
@Prop(TypeString(), tracker.string.Identifier)
|
|
@Index(IndexKind.FullText)
|
|
identifier!: IntlString
|
|
|
|
@Prop(TypeNumber(), tracker.string.Number)
|
|
@Hidden()
|
|
sequence!: number
|
|
|
|
@Prop(Collection(tracker.class.IssueStatus), tracker.string.IssueStatuses)
|
|
issueStatuses!: number
|
|
|
|
@Prop(TypeRef(tracker.class.IssueStatus), tracker.string.DefaultIssueStatus)
|
|
defaultIssueStatus!: Ref<IssueStatus>
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
@Model(tracker.class.Issue, core.class.Doc, DOMAIN_TRACKER)
|
|
@UX(tracker.string.Issue, tracker.icon.Issue, tracker.string.Issue)
|
|
export class TIssue extends TDoc implements Issue {
|
|
@Prop(TypeString(), tracker.string.Title)
|
|
@Index(IndexKind.FullText)
|
|
title!: string
|
|
|
|
@Prop(TypeMarkup(), tracker.string.Description)
|
|
@Index(IndexKind.FullText)
|
|
description!: Markup
|
|
|
|
@Prop(TypeRef(tracker.class.IssueStatus), tracker.string.Status)
|
|
status!: Ref<IssueStatus>
|
|
|
|
@Prop(TypeIssuePriority(), tracker.string.Priority)
|
|
priority!: IssuePriority
|
|
|
|
@Prop(TypeNumber(), tracker.string.Number)
|
|
number!: number
|
|
|
|
@Prop(TypeRef(contact.class.Employee), tracker.string.Assignee)
|
|
assignee!: Ref<Employee> | null
|
|
|
|
@Prop(TypeRef(tracker.class.Issue), tracker.string.Parent)
|
|
parentIssue!: Ref<Issue>
|
|
|
|
@Prop(ArrOf(TypeRef(tracker.class.Issue)), tracker.string.BlockedBy)
|
|
blockedBy!: Ref<Issue>[]
|
|
|
|
@Prop(ArrOf(TypeRef(tracker.class.Issue)), tracker.string.RelatedTo)
|
|
relatedIssue!: Ref<Issue>[]
|
|
|
|
@Prop(Collection(chunter.class.Comment), tracker.string.Comments)
|
|
comments!: number
|
|
|
|
@Prop(Collection(attachment.class.Attachment), tracker.string.Attachments)
|
|
attachments!: number
|
|
|
|
// @Prop(Collection(core.class.TypeString), tracker.string.Labels)
|
|
labels?: string[]
|
|
|
|
declare space: Ref<Team>
|
|
|
|
@Prop(TypeDate(true), tracker.string.DueDate)
|
|
dueDate!: Timestamp | null
|
|
|
|
@Prop(TypeString(), tracker.string.Rank)
|
|
@Hidden()
|
|
rank!: string
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
@Model(tracker.class.Document, core.class.Doc, DOMAIN_TRACKER)
|
|
@UX(tracker.string.Document, tracker.icon.Document, tracker.string.Document)
|
|
export class TDocument extends TDoc implements Document {
|
|
@Prop(TypeString(), tracker.string.Title)
|
|
@Index(IndexKind.FullText)
|
|
title!: string
|
|
|
|
@Prop(TypeString(), tracker.string.DocumentIcon)
|
|
icon!: string | null
|
|
|
|
@Prop(TypeString(), tracker.string.DocumentColor)
|
|
color!: number
|
|
|
|
@Prop(TypeMarkup(), tracker.string.Description)
|
|
@Index(IndexKind.FullText)
|
|
content!: Markup
|
|
|
|
declare space: Ref<Team>
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
@Model(tracker.class.Project, core.class.Doc, DOMAIN_TRACKER)
|
|
@UX(tracker.string.Project, tracker.icon.Project, tracker.string.Project)
|
|
export class TProject extends TDoc implements Project {
|
|
@Prop(TypeString(), tracker.string.Title)
|
|
// @Index(IndexKind.FullText)
|
|
label!: string
|
|
|
|
@Prop(TypeMarkup(), tracker.string.Project)
|
|
description?: Markup
|
|
|
|
@Prop(TypeNumber(), tracker.string.Status)
|
|
status!: ProjectStatus
|
|
|
|
@Prop(TypeRef(contact.class.Employee), tracker.string.ProjectLead)
|
|
lead!: Ref<Employee> | null
|
|
|
|
@Prop(ArrOf(TypeRef(contact.class.Employee)), tracker.string.Members)
|
|
members!: Ref<Employee>[]
|
|
|
|
@Prop(Collection(chunter.class.Comment), chunter.string.Comments)
|
|
comments!: number
|
|
|
|
@Prop(Collection(tracker.class.Document), tracker.string.Document)
|
|
documents!: number
|
|
|
|
@Prop(Collection(attachment.class.Attachment), attachment.string.Attachments)
|
|
attachments?: number
|
|
|
|
@Prop(TypeDate(true), tracker.string.Project)
|
|
startDate!: Timestamp | null
|
|
|
|
@Prop(TypeDate(true), tracker.string.Project)
|
|
targetDate!: Timestamp | null
|
|
|
|
declare space: Ref<Team>
|
|
}
|
|
|
|
export function createModel (builder: Builder): void {
|
|
builder.createModel(TTeam, TProject, TIssue, TIssueStatus, TIssueStatusCategory, TTypeIssuePriority)
|
|
|
|
builder.createDoc(
|
|
tracker.class.IssueStatusCategory,
|
|
core.space.Model,
|
|
{
|
|
label: tracker.string.CategoryBacklog,
|
|
icon: tracker.icon.CategoryBacklog,
|
|
color: 0,
|
|
defaultStatusName: 'Backlog',
|
|
order: 0
|
|
},
|
|
tracker.issueStatusCategory.Backlog
|
|
)
|
|
|
|
builder.createDoc(
|
|
tracker.class.IssueStatusCategory,
|
|
core.space.Model,
|
|
{
|
|
label: tracker.string.CategoryUnstarted,
|
|
icon: tracker.icon.CategoryUnstarted,
|
|
color: 1,
|
|
defaultStatusName: 'Todo',
|
|
order: 1
|
|
},
|
|
tracker.issueStatusCategory.Unstarted
|
|
)
|
|
|
|
builder.createDoc(
|
|
tracker.class.IssueStatusCategory,
|
|
core.space.Model,
|
|
{
|
|
label: tracker.string.CategoryStarted,
|
|
icon: tracker.icon.CategoryStarted,
|
|
color: 2,
|
|
defaultStatusName: 'In Progress',
|
|
order: 2
|
|
},
|
|
tracker.issueStatusCategory.Started
|
|
)
|
|
|
|
builder.createDoc(
|
|
tracker.class.IssueStatusCategory,
|
|
core.space.Model,
|
|
{
|
|
label: tracker.string.CategoryCompleted,
|
|
icon: tracker.icon.CategoryCompleted,
|
|
color: 3,
|
|
defaultStatusName: 'Done',
|
|
order: 3
|
|
},
|
|
tracker.issueStatusCategory.Completed
|
|
)
|
|
|
|
builder.createDoc(
|
|
tracker.class.IssueStatusCategory,
|
|
core.space.Model,
|
|
{
|
|
label: tracker.string.CategoryCanceled,
|
|
icon: tracker.icon.CategoryCanceled,
|
|
color: 4,
|
|
defaultStatusName: 'Canceled',
|
|
order: 4
|
|
},
|
|
tracker.issueStatusCategory.Canceled
|
|
)
|
|
|
|
const issuesId = 'issues'
|
|
const activeId = 'active'
|
|
const backlogId = 'backlog'
|
|
const boardId = 'board'
|
|
const projectsId = 'projects'
|
|
|
|
builder.mixin(tracker.class.TypeIssuePriority, core.class.Class, view.mixin.AttributePresenter, {
|
|
presenter: tracker.component.PriorityPresenter
|
|
})
|
|
|
|
builder.mixin(tracker.class.IssueStatus, core.class.Class, view.mixin.AttributePresenter, {
|
|
presenter: tracker.component.StatusPresenter
|
|
})
|
|
|
|
builder.createDoc(
|
|
workbench.class.Application,
|
|
core.space.Model,
|
|
{
|
|
label: tracker.string.TrackerApplication,
|
|
icon: tracker.icon.Issue,
|
|
hidden: false,
|
|
navigatorModel: {
|
|
specials: [
|
|
{
|
|
id: 'inbox',
|
|
position: 'top',
|
|
label: tracker.string.Inbox,
|
|
icon: tracker.icon.Inbox,
|
|
component: tracker.component.Inbox
|
|
},
|
|
{
|
|
id: 'my-issues',
|
|
position: 'top',
|
|
label: tracker.string.MyIssues,
|
|
icon: tracker.icon.MyIssues,
|
|
component: tracker.component.MyIssues
|
|
},
|
|
{
|
|
id: 'views',
|
|
position: 'top',
|
|
label: tracker.string.Views,
|
|
icon: tracker.icon.Views,
|
|
component: tracker.component.Views
|
|
}
|
|
],
|
|
spaces: [
|
|
{
|
|
label: tracker.string.Teams,
|
|
spaceClass: tracker.class.Team,
|
|
addSpaceLabel: tracker.string.CreateTeam,
|
|
createComponent: tracker.component.CreateTeam,
|
|
icon: tracker.icon.Home,
|
|
specials: [
|
|
{
|
|
id: issuesId,
|
|
label: tracker.string.Issues,
|
|
icon: tracker.icon.Issues,
|
|
component: tracker.component.Issues
|
|
},
|
|
{
|
|
id: activeId,
|
|
label: tracker.string.Active,
|
|
// icon: tracker.icon.TrackerApplication,
|
|
component: tracker.component.Active
|
|
},
|
|
{
|
|
id: backlogId,
|
|
label: tracker.string.Backlog,
|
|
// icon: tracker.icon.TrackerApplication,
|
|
component: tracker.component.Backlog
|
|
},
|
|
{
|
|
id: boardId,
|
|
label: tracker.string.Board,
|
|
// icon: tracker.icon.TrackerApplication,
|
|
component: tracker.component.Board
|
|
},
|
|
{
|
|
id: projectsId,
|
|
label: tracker.string.Projects,
|
|
icon: tracker.icon.Projects,
|
|
component: tracker.component.Projects
|
|
}
|
|
]
|
|
}
|
|
]
|
|
},
|
|
navHeaderComponent: tracker.component.NewIssueHeader
|
|
},
|
|
tracker.app.Tracker
|
|
)
|
|
|
|
function createGotoSpecialAction (builder: Builder, id: string, key: KeyBinding, label: IntlString): void {
|
|
createNavigateAction(builder, key, label, {
|
|
application: tracker.app.Tracker,
|
|
mode: 'space',
|
|
spaceSpecial: id,
|
|
spaceClass: tracker.class.Team
|
|
})
|
|
}
|
|
|
|
createGotoSpecialAction(builder, issuesId, 'g->e', tracker.string.GotoIssues)
|
|
createGotoSpecialAction(builder, activeId, 'g->a', tracker.string.GotoActive)
|
|
createGotoSpecialAction(builder, backlogId, 'g->b', tracker.string.GotoBacklog)
|
|
createGotoSpecialAction(builder, boardId, 'g->d', tracker.string.GotoBoard)
|
|
createGotoSpecialAction(builder, projectsId, 'g->p', tracker.string.GotoProjects)
|
|
|
|
createAction(builder, {
|
|
action: workbench.actionImpl.Navigate,
|
|
actionProps: {
|
|
mode: 'app',
|
|
application: tracker.app.Tracker
|
|
},
|
|
label: tracker.string.GotoTrackerApplication,
|
|
icon: view.icon.ArrowRight,
|
|
input: 'none',
|
|
category: view.category.Navigation,
|
|
target: core.class.Doc,
|
|
context: {
|
|
mode: ['workbench', 'browser', 'editor', 'panel', 'popup']
|
|
}
|
|
})
|
|
}
|