focalboard/webapp/src/cardFilter.ts

161 lines
5.8 KiB
TypeScript
Raw Normal View History

2020-10-20 21:50:53 +02:00
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
2020-10-20 21:52:56 +02:00
import {IPropertyTemplate} from './blocks/board'
import {Card} from './blocks/card'
import {FilterClause} from './filterClause'
import {FilterGroup} from './filterGroup'
import {Utils} from './utils'
2020-10-08 18:21:27 +02:00
class CardFilter {
2020-10-21 03:28:55 +02:00
static applyFilterGroup(filterGroup: FilterGroup, templates: readonly IPropertyTemplate[], cards: Card[]): Card[] {
2020-10-20 21:50:53 +02:00
return cards.filter((card) => this.isFilterGroupMet(filterGroup, templates, card))
}
2020-10-08 18:21:27 +02:00
2020-10-21 03:28:55 +02:00
static isFilterGroupMet(filterGroup: FilterGroup, templates: readonly IPropertyTemplate[], card: Card): boolean {
2020-10-20 21:50:53 +02:00
const {filters} = filterGroup
2020-10-08 18:21:27 +02:00
2020-10-20 21:50:53 +02:00
if (filterGroup.filters.length < 1) {
return true // No filters = always met
}
2020-10-08 18:21:27 +02:00
2020-10-20 21:50:53 +02:00
if (filterGroup.operation === 'or') {
for (const filter of filters) {
if (FilterGroup.isAnInstanceOf(filter)) {
if (this.isFilterGroupMet(filter, templates, card)) {
return true
}
} else if (this.isClauseMet(filter, templates, card)) {
return true
}
}
return false
}
Utils.assert(filterGroup.operation === 'and')
for (const filter of filters) {
if (FilterGroup.isAnInstanceOf(filter)) {
if (!this.isFilterGroupMet(filter, templates, card)) {
return false
}
} else if (!this.isClauseMet(filter, templates, card)) {
return false
}
}
return true
}
2020-10-08 18:21:27 +02:00
2020-10-21 03:28:55 +02:00
static isClauseMet(filter: FilterClause, templates: readonly IPropertyTemplate[], card: Card): boolean {
2020-10-20 21:50:53 +02:00
const value = card.properties[filter.propertyId]
switch (filter.condition) {
case 'includes': {
if (filter.values.length < 1) {
break
} // No values = ignore clause (always met)
return (filter.values.find((cValue) => cValue === value) !== undefined)
}
case 'notIncludes': {
if (filter.values.length < 1) {
break
} // No values = ignore clause (always met)
return (filter.values.find((cValue) => cValue === value) === undefined)
}
case 'isEmpty': {
return !value
}
case 'isNotEmpty': {
return Boolean(value)
}
default: {
Utils.assertFailure(`Invalid filter condition ${filter.condition}`)
}
}
return true
}
2020-10-08 18:21:27 +02:00
2020-11-12 23:06:02 +01:00
static propertiesThatMeetFilterGroup(filterGroup: FilterGroup | undefined, templates: readonly IPropertyTemplate[]): Record<string, string> {
2020-10-20 21:50:53 +02:00
// TODO: Handle filter groups
2020-11-12 23:06:02 +01:00
if (!filterGroup) {
return {}
}
2020-10-20 21:50:53 +02:00
const filters = filterGroup.filters.filter((o) => !FilterGroup.isAnInstanceOf(o))
if (filters.length < 1) {
return {}
}
2020-10-08 18:21:27 +02:00
2020-10-20 21:50:53 +02:00
if (filterGroup.operation === 'or') {
// Just need to meet the first clause
const property = this.propertyThatMeetsFilterClause(filters[0] as FilterClause, templates)
const result: Record<string, string> = {}
2020-11-12 23:06:02 +01:00
if (property.value) {
result[property.id] = property.value
}
return result
} else {
// And: Need to meet all clauses
const result: Record<string, string> = {}
filters.forEach((filterClause) => {
const property = this.propertyThatMeetsFilterClause(filterClause as FilterClause, templates)
if (property.value) {
result[property.id] = property.value
}
})
2020-10-20 21:50:53 +02:00
return result
}
}
2020-10-08 18:21:27 +02:00
2020-10-21 03:28:55 +02:00
static propertyThatMeetsFilterClause(filterClause: FilterClause, templates: readonly IPropertyTemplate[]): { id: string, value?: string } {
2020-10-20 21:50:53 +02:00
const template = templates.find((o) => o.id === filterClause.propertyId)
2020-11-12 23:06:02 +01:00
if (!template) {
Utils.assertFailure(`propertyThatMeetsFilterClause. Cannot find template: ${filterClause.propertyId}`)
return {id: filterClause.propertyId}
}
2020-10-20 21:50:53 +02:00
switch (filterClause.condition) {
case 'includes': {
if (filterClause.values.length < 1) {
return {id: filterClause.propertyId}
}
return {id: filterClause.propertyId, value: filterClause.values[0]}
}
case 'notIncludes': {
if (filterClause.values.length < 1) {
return {id: filterClause.propertyId}
}
if (template.type === 'select') {
2020-10-23 21:59:09 +02:00
const option = template.options.find((o) => !filterClause.values.includes(o.id))
2020-11-12 23:06:02 +01:00
if (option) {
return {id: filterClause.propertyId, value: option.id}
} else {
// No other options exist
return {id: filterClause.propertyId}
}
2020-10-20 21:50:53 +02:00
}
// TODO: Handle non-select types
return {id: filterClause.propertyId}
}
case 'isEmpty': {
return {id: filterClause.propertyId}
}
case 'isNotEmpty': {
if (template.type === 'select') {
if (template.options.length > 0) {
const option = template.options[0]
2020-10-23 21:59:09 +02:00
return {id: filterClause.propertyId, value: option.id}
2020-10-20 21:50:53 +02:00
}
return {id: filterClause.propertyId}
}
// TODO: Handle non-select types
return {id: filterClause.propertyId}
}
2020-10-20 22:36:54 +02:00
default: {
Utils.assertFailure(`Unexpected filter condition: ${filterClause.condition}`)
return {id: filterClause.propertyId}
}
2020-10-20 21:50:53 +02:00
}
}
2020-10-08 18:21:27 +02:00
}
2020-10-20 21:50:53 +02:00
export {CardFilter}