1
0
mirror of https://github.com/mattermost/focalboard.git synced 2025-04-04 21:14:22 +02:00
focalboard/webapp/src/boardTree.ts

325 lines
11 KiB
TypeScript
Raw Normal View History

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}