2020-10-20 12:50:53 -07:00
|
|
|
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
|
|
// See LICENSE.txt for license information.
|
2020-10-20 12:52:56 -07:00
|
|
|
import {Block} from './blocks/block'
|
|
|
|
import {Board, IPropertyOption, IPropertyTemplate} from './blocks/board'
|
|
|
|
import {BoardView} from './blocks/boardView'
|
|
|
|
import {Card} from './blocks/card'
|
|
|
|
import {CardFilter} from './cardFilter'
|
|
|
|
import octoClient from './octoClient'
|
|
|
|
import {IBlock} from './octoTypes'
|
|
|
|
import {OctoUtils} from './octoUtils'
|
|
|
|
import {Utils} from './utils'
|
2020-10-08 09:21:27 -07:00
|
|
|
|
2020-10-14 17:35:15 -07:00
|
|
|
type Group = { option: IPropertyOption, cards: Card[] }
|
2020-10-08 09:21:27 -07:00
|
|
|
|
|
|
|
class BoardTree {
|
2020-10-20 12:50:53 -07:00
|
|
|
board!: Board
|
|
|
|
views: BoardView[] = []
|
|
|
|
cards: Card[] = []
|
|
|
|
emptyGroupCards: Card[] = []
|
|
|
|
groups: Group[] = []
|
|
|
|
|
|
|
|
activeView?: BoardView
|
|
|
|
groupByProperty?: IPropertyTemplate
|
|
|
|
|
|
|
|
private searchText?: string
|
|
|
|
private allCards: Card[] = []
|
|
|
|
get allBlocks(): IBlock[] {
|
|
|
|
return [this.board, ...this.views, ...this.allCards]
|
|
|
|
}
|
|
|
|
|
|
|
|
constructor(private boardId: string) {
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
async sync(): Promise<void> {
|
2020-10-20 12:50:53 -07:00
|
|
|
const blocks = await octoClient.getSubtree(this.boardId)
|
|
|
|
this.rebuild(OctoUtils.hydrateBlocks(blocks))
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
private rebuild(blocks: Block[]): void {
|
|
|
|
this.board = blocks.find((block) => block.type === 'board') as Board
|
2020-10-20 12:50:53 -07:00
|
|
|
this.views = blocks.filter((block) => block.type === 'view') as BoardView[]
|
|
|
|
this.allCards = blocks.filter((block) => block.type === 'card') as Card[]
|
2020-10-20 13:36:54 -07:00
|
|
|
this.cards = []
|
2020-10-20 12:50:53 -07:00
|
|
|
|
|
|
|
this.ensureMinimumSchema()
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
private async ensureMinimumSchema(): Promise<boolean> {
|
2020-10-20 12:50:53 -07:00
|
|
|
const {board} = this
|
|
|
|
|
|
|
|
let didChange = false
|
|
|
|
|
|
|
|
// At least one select property
|
|
|
|
const selectProperties = board.cardProperties.find((o) => o.type === 'select')
|
|
|
|
if (!selectProperties) {
|
|
|
|
const property: IPropertyTemplate = {
|
|
|
|
id: Utils.createGuid(),
|
|
|
|
name: 'Status',
|
2020-10-20 13:36:54 -07:00
|
|
|
type: 'select',
|
|
|
|
options: [],
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
board.cardProperties.push(property)
|
|
|
|
didChange = true
|
2020-10-20 13:36:54 -07:00
|
|
|
}
|
2020-10-20 12:50:53 -07:00
|
|
|
|
|
|
|
// At least one view
|
|
|
|
if (this.views.length < 1) {
|
|
|
|
const view = new BoardView()
|
2020-10-20 13:36:54 -07:00
|
|
|
view.parentId = board.id
|
2020-10-20 12:50:53 -07:00
|
|
|
view.groupById = board.cardProperties.find((o) => o.type === 'select')?.id
|
|
|
|
this.views.push(view)
|
|
|
|
didChange = true
|
|
|
|
}
|
|
|
|
|
|
|
|
return didChange
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
setActiveView(viewId: string): void {
|
|
|
|
this.activeView = this.views.find((o) => o.id === viewId)
|
|
|
|
if (!this.activeView) {
|
2020-10-20 12:50:53 -07:00
|
|
|
Utils.logError(`Cannot find BoardView: ${viewId}`)
|
2020-10-20 13:36:54 -07:00
|
|
|
this.activeView = this.views[0]
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
// Fix missing group by (e.g. for new views)
|
|
|
|
if (this.activeView.viewType === 'board' && !this.activeView.groupById) {
|
|
|
|
this.activeView.groupById = this.board.cardProperties.find((o) => o.type === 'select')?.id
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
2020-10-20 13:36:54 -07:00
|
|
|
this.applyFilterSortAndGroup()
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
getSearchText(): string | undefined {
|
2020-10-20 13:36:54 -07:00
|
|
|
return this.searchText
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
setSearchText(text?: string): void {
|
|
|
|
this.searchText = text
|
2020-10-20 12:50:53 -07:00
|
|
|
this.applyFilterSortAndGroup()
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
applyFilterSortAndGroup(): void {
|
2020-10-20 12:50:53 -07:00
|
|
|
Utils.assert(this.allCards !== undefined)
|
|
|
|
|
|
|
|
this.cards = this.filterCards(this.allCards)
|
|
|
|
Utils.assert(this.cards !== undefined)
|
|
|
|
this.cards = this.searchFilterCards(this.cards)
|
2020-10-20 13:36:54 -07:00
|
|
|
Utils.assert(this.cards !== undefined)
|
2020-10-20 12:50:53 -07:00
|
|
|
this.cards = this.sortCards(this.cards)
|
|
|
|
Utils.assert(this.cards !== undefined)
|
|
|
|
|
|
|
|
if (this.activeView.groupById) {
|
|
|
|
this.setGroupByProperty(this.activeView.groupById)
|
|
|
|
} else {
|
|
|
|
Utils.assert(this.activeView.viewType !== 'board')
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
Utils.assert(this.cards !== undefined)
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private searchFilterCards(cards: Card[]): Card[] {
|
|
|
|
const searchText = this.searchText?.toLocaleLowerCase()
|
|
|
|
if (!searchText) {
|
|
|
|
return cards.slice()
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
return cards.filter((card) => {
|
|
|
|
return (card.title?.toLocaleLowerCase().indexOf(searchText) !== -1)
|
|
|
|
})
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private setGroupByProperty(propertyId: string) {
|
2020-10-20 13:36:54 -07:00
|
|
|
const {board} = this
|
2020-10-20 12:50:53 -07:00
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
let property = board.cardProperties.find((o) => o.id === propertyId)
|
2020-10-20 12:50:53 -07:00
|
|
|
|
|
|
|
// TODO: Handle multi-select
|
|
|
|
if (!property || property.type !== 'select') {
|
2020-10-20 13:36:54 -07:00
|
|
|
Utils.logError(`this.view.groupById card property not found: ${propertyId}`)
|
|
|
|
property = board.cardProperties.find((o) => o.type === 'select')
|
|
|
|
Utils.assertValue(property)
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
this.groupByProperty = property
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
this.groupCards()
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private groupCards() {
|
|
|
|
this.groups = []
|
|
|
|
|
|
|
|
const groupByPropertyId = this.groupByProperty.id
|
|
|
|
|
|
|
|
this.emptyGroupCards = this.cards.filter((o) => {
|
|
|
|
const propertyValue = o.properties[groupByPropertyId]
|
2020-10-20 13:36:54 -07:00
|
|
|
return !propertyValue || !this.groupByProperty.options.find((option) => option.value === propertyValue)
|
2020-10-20 12:52:56 -07:00
|
|
|
})
|
2020-10-20 12:50:53 -07:00
|
|
|
|
|
|
|
const propertyOptions = this.groupByProperty.options || []
|
|
|
|
for (const option of propertyOptions) {
|
2020-10-20 13:36:54 -07:00
|
|
|
const cards = this.cards.
|
2020-10-20 12:50:53 -07:00
|
|
|
filter((o) => {
|
2020-10-20 13:36:54 -07:00
|
|
|
const propertyValue = o.properties[groupByPropertyId]
|
|
|
|
return propertyValue && propertyValue === option.value
|
|
|
|
})
|
2020-10-20 12:50:53 -07:00
|
|
|
|
|
|
|
const group: Group = {
|
|
|
|
option,
|
2020-10-20 13:36:54 -07:00
|
|
|
cards,
|
|
|
|
}
|
2020-10-20 12:50:53 -07:00
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
this.groups.push(group)
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private filterCards(cards: Card[]): Card[] {
|
|
|
|
const {board} = this
|
2020-10-20 13:36:54 -07:00
|
|
|
const filterGroup = this.activeView?.filter
|
2020-10-20 12:50:53 -07:00
|
|
|
if (!filterGroup) {
|
|
|
|
return cards.slice()
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
return CardFilter.applyFilterGroup(filterGroup, board.cardProperties, cards)
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private sortCards(cards: Card[]): Card[] {
|
2020-10-20 13:36:54 -07:00
|
|
|
if (!this.activeView) {
|
|
|
|
Utils.assertFailure()
|
|
|
|
return cards
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
const {board} = this
|
2020-10-20 13:36:54 -07:00
|
|
|
const {sortOptions} = this.activeView
|
|
|
|
let sortedCards: Card[] = []
|
2020-10-20 12:50:53 -07:00
|
|
|
|
|
|
|
if (sortOptions.length < 1) {
|
|
|
|
Utils.log('Default sort')
|
|
|
|
sortedCards = cards.sort((a, b) => {
|
2020-10-20 13:36:54 -07:00
|
|
|
const aValue = a.title || ''
|
|
|
|
const bValue = b.title || ''
|
2020-10-20 12:50:53 -07:00
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
// Always put empty values at the bottom
|
2020-10-20 12:50:53 -07:00
|
|
|
if (aValue && !bValue) {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
if (bValue && !aValue) {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
if (!aValue && !bValue) {
|
|
|
|
return a.createAt - b.createAt
|
|
|
|
}
|
|
|
|
|
|
|
|
return a.createAt - b.createAt
|
2020-10-20 12:52:56 -07:00
|
|
|
})
|
2020-10-20 13:36:54 -07:00
|
|
|
} else {
|
|
|
|
for (const sortOption of sortOptions) {
|
|
|
|
if (sortOption.propertyId === '__name') {
|
|
|
|
Utils.log('Sort by name')
|
|
|
|
sortedCards = cards.sort((a, b) => {
|
|
|
|
const aValue = a.title || ''
|
|
|
|
const bValue = b.title || ''
|
|
|
|
|
|
|
|
// Always put empty values at the bottom, newest last
|
2020-10-20 12:50:53 -07:00
|
|
|
if (aValue && !bValue) {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
if (bValue && !aValue) {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
if (!aValue && !bValue) {
|
|
|
|
return a.createAt - b.createAt
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
let result = aValue.localeCompare(bValue)
|
2020-10-20 12:50:53 -07:00
|
|
|
if (sortOption.reversed) {
|
|
|
|
result = -result
|
|
|
|
}
|
2020-10-20 13:36:54 -07:00
|
|
|
return result
|
2020-10-20 12:52:56 -07:00
|
|
|
})
|
2020-10-20 13:36:54 -07:00
|
|
|
} else {
|
|
|
|
const sortPropertyId = sortOption.propertyId
|
2020-10-20 12:50:53 -07:00
|
|
|
const template = board.cardProperties.find((o) => o.id === sortPropertyId)
|
2020-10-20 13:36:54 -07:00
|
|
|
if (!template) {
|
|
|
|
Utils.logError(`Missing template for property id: ${sortPropertyId}`)
|
2020-10-20 12:50:53 -07:00
|
|
|
return cards.slice()
|
2020-10-20 13:36:54 -07:00
|
|
|
}
|
|
|
|
Utils.log(`Sort by ${template?.name}`)
|
|
|
|
sortedCards = cards.sort((a, b) => {
|
2020-10-20 12:50:53 -07:00
|
|
|
// Always put cards with no titles at the bottom
|
2020-10-20 13:36:54 -07:00
|
|
|
if (a.title && !b.title) {
|
2020-10-20 12:50:53 -07:00
|
|
|
return -1
|
|
|
|
}
|
2020-10-20 13:36:54 -07:00
|
|
|
if (b.title && !a.title) {
|
2020-10-20 12:50:53 -07:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
if (!a.title && !b.title) {
|
|
|
|
return a.createAt - b.createAt
|
|
|
|
}
|
|
|
|
|
2020-10-20 12:52:56 -07:00
|
|
|
const aValue = a.properties[sortPropertyId] || ''
|
2020-10-20 13:36:54 -07:00
|
|
|
const bValue = b.properties[sortPropertyId] || ''
|
|
|
|
let result = 0
|
2020-10-20 12:50:53 -07:00
|
|
|
if (template.type === 'select') {
|
2020-10-20 13:36:54 -07:00
|
|
|
// Always put empty values at the bottom
|
2020-10-20 12:50:53 -07:00
|
|
|
if (aValue && !bValue) {
|
|
|
|
return -1
|
|
|
|
}
|
2020-10-20 13:36:54 -07:00
|
|
|
if (bValue && !aValue) {
|
2020-10-20 12:50:53 -07:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
if (!aValue && !bValue) {
|
|
|
|
return a.createAt - b.createAt
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort by the option order (not alphabetically by value)
|
2020-10-20 13:36:54 -07:00
|
|
|
const aOrder = template.options.findIndex((o) => o.value === aValue)
|
|
|
|
const bOrder = template.options.findIndex((o) => o.value === bValue)
|
2020-10-20 12:50:53 -07:00
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
result = aOrder - bOrder
|
2020-10-20 12:50:53 -07:00
|
|
|
} else if (template.type === 'number' || template.type === 'date') {
|
|
|
|
// Always put empty values at the bottom
|
2020-10-20 13:36:54 -07:00
|
|
|
if (aValue && !bValue) {
|
2020-10-20 12:50:53 -07:00
|
|
|
return -1
|
|
|
|
}
|
2020-10-20 13:36:54 -07:00
|
|
|
if (bValue && !aValue) {
|
2020-10-20 12:50:53 -07:00
|
|
|
return 1
|
|
|
|
}
|
2020-10-20 13:36:54 -07:00
|
|
|
if (!aValue && !bValue) {
|
2020-10-20 12:50:53 -07:00
|
|
|
return a.createAt - b.createAt
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
result = Number(aValue) - Number(bValue)
|
2020-10-20 12:50:53 -07:00
|
|
|
} else if (template.type === 'createdTime') {
|
|
|
|
result = a.createAt - b.createAt
|
2020-10-20 13:36:54 -07:00
|
|
|
} else if (template.type === 'updatedTime') {
|
|
|
|
result = a.updateAt - b.updateAt
|
2020-10-20 12:50:53 -07:00
|
|
|
} else {
|
|
|
|
// Text-based sort
|
|
|
|
|
|
|
|
// Always put empty values at the bottom
|
2020-10-20 13:36:54 -07:00
|
|
|
if (aValue && !bValue) {
|
2020-10-20 12:50:53 -07:00
|
|
|
return -1
|
|
|
|
}
|
2020-10-20 13:36:54 -07:00
|
|
|
if (bValue && !aValue) {
|
2020-10-20 12:50:53 -07:00
|
|
|
return 1
|
|
|
|
}
|
2020-10-20 13:36:54 -07:00
|
|
|
if (!aValue && !bValue) {
|
2020-10-20 12:50:53 -07:00
|
|
|
return a.createAt - b.createAt
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
result = aValue.localeCompare(bValue)
|
|
|
|
}
|
2020-10-20 12:50:53 -07:00
|
|
|
|
2020-10-20 13:36:54 -07:00
|
|
|
if (sortOption.reversed) {
|
2020-10-20 12:50:53 -07:00
|
|
|
result = -result
|
|
|
|
}
|
|
|
|
return result
|
2020-10-20 12:52:56 -07:00
|
|
|
})
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
2020-10-20 13:36:54 -07:00
|
|
|
}
|
2020-10-20 12:50:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return sortedCards
|
|
|
|
}
|
2020-10-08 09:21:27 -07:00
|
|
|
}
|
|
|
|
|
2020-10-20 12:50:53 -07:00
|
|
|
export {BoardTree}
|