// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. import React from 'react' import {injectIntl, IntlShape, FormattedMessage} from 'react-intl' import {Archiver} from '../archiver' import {ISortOption, MutableBoardView} from '../blocks/boardView' import {BlockIcons} from '../blockIcons' import {MutableCard} from '../blocks/card' import {IPropertyTemplate} from '../blocks/board' import {BoardTree} from '../viewModel/boardTree' import ViewMenu from '../components/viewMenu' import {CsvExporter} from '../csvExporter' import {CardFilter} from '../cardFilter' import mutator from '../mutator' import {Utils} from '../utils' import Menu from '../widgets/menu' import MenuWrapper from '../widgets/menuWrapper' import CheckIcon from '../widgets/icons/check' import DropdownIcon from '../widgets/icons/dropdown' import OptionsIcon from '../widgets/icons/options' import SortUpIcon from '../widgets/icons/sortUp' import SortDownIcon from '../widgets/icons/sortDown' import ButtonWithMenu from '../widgets/buttons/buttonWithMenu' import IconButton from '../widgets/buttons/iconButton' import Button from '../widgets/buttons/button' import {Editable} from './editable' import FilterComponent from './filterComponent' import './viewHeader.scss' import {Constants} from '../constants' import DeleteIcon from '../widgets/icons/delete' type Props = { boardTree?: BoardTree showView: (id: string) => void setSearchText: (text: string) => void addCard: () => void addCardFromTemplate: (cardTemplateId?: string) => void addCardTemplate: () => void editCardTemplate: (cardTemplateId: string) => void withGroupBy?: boolean intl: IntlShape } type State = { isSearching: boolean showFilter: boolean } class ViewHeader extends React.Component { private searchFieldRef = React.createRef() shouldComponentUpdate(): boolean { return true } constructor(props: Props) { super(props) this.state = {isSearching: Boolean(this.props.boardTree?.getSearchText()), showFilter: false} } componentDidUpdate(prevPros: Props, prevState: State): void { if (this.state.isSearching && !prevState.isSearching) { this.searchFieldRef.current.focus() } } private filterClicked = () => { this.setState({showFilter: true}) } private hideFilter = () => { this.setState({showFilter: false}) } private onSearchKeyDown = (e: React.KeyboardEvent) => { if (e.keyCode === 27) { // ESC: Clear search this.searchFieldRef.current.text = '' this.setState({isSearching: false}) this.props.setSearchText(undefined) e.preventDefault() } } private searchChanged(text?: string) { this.props.setSearchText(text) } private async testAddCards(count: number) { const {boardTree} = this.props const {board, activeView} = boardTree const startCount = boardTree?.cards?.length let optionIndex = 0 await mutator.performAsUndoGroup(async () => { for (let i = 0; i < count; i++) { const card = new MutableCard() card.parentId = boardTree.board.id card.properties = CardFilter.propertiesThatMeetFilterGroup(activeView.filter, board.cardProperties) card.title = `Test Card ${startCount + i + 1}` card.icon = BlockIcons.shared.randomIcon() if (boardTree.groupByProperty && boardTree.groupByProperty.options.length > 0) { // Cycle through options const option = boardTree.groupByProperty.options[optionIndex] optionIndex = (optionIndex + 1) % boardTree.groupByProperty.options.length card.properties[boardTree.groupByProperty.id] = option.id } await mutator.insertBlock(card, 'test add card') } }) } private async testDistributeCards() { const {boardTree} = this.props if (!boardTree) { return } await mutator.performAsUndoGroup(async () => { let optionIndex = 0 for (const card of boardTree.cards) { if (boardTree.groupByProperty && boardTree.groupByProperty.options.length > 0) { // Cycle through options const option = boardTree.groupByProperty.options[optionIndex] optionIndex = (optionIndex + 1) % boardTree.groupByProperty.options.length const newCard = new MutableCard(card) if (newCard.properties[boardTree.groupByProperty.id] !== option.id) { newCard.properties[boardTree.groupByProperty.id] = option.id await mutator.updateBlock(newCard, card, 'test distribute cards') } } } }) } private async testRandomizeIcons() { const {boardTree} = this.props await mutator.performAsUndoGroup(async () => { for (const card of boardTree.cards) { await mutator.changeIcon(card, BlockIcons.shared.randomIcon(), 'randomize icon') } }) } render(): JSX.Element { const {boardTree, showView, withGroupBy, intl} = this.props const {board, activeView} = boardTree const hasFilter = activeView.filter && activeView.filter.filters?.length > 0 const hasSort = activeView.sortOptions.length > 0 return (
{ mutator.changeTitle(activeView, text) }} /> }/>
{boardTree.board.cardProperties.map((option: IPropertyTemplate) => ( { const property = boardTree.board.cardProperties.find((o: IPropertyTemplate) => o.id === propertyId) Utils.assertValue(property) Utils.log(`Toggle property ${property.name}`) let newVisiblePropertyIds = [] if (activeView.visiblePropertyIds.includes(propertyId)) { newVisiblePropertyIds = activeView.visiblePropertyIds.filter((o: string) => o !== propertyId) } else { newVisiblePropertyIds = [...activeView.visiblePropertyIds, propertyId] } mutator.changeViewVisibleProperties(activeView, newVisiblePropertyIds) }} /> ))} {withGroupBy && {boardTree.board.cardProperties.filter((o: IPropertyTemplate) => o.type === 'select').map((option: IPropertyTemplate) => ( : undefined} onClick={(id) => { if (boardTree.activeView.groupById === id) { return } mutator.changeViewGroupById(boardTree.activeView, id) }} /> ))} }
{(activeView.sortOptions.length > 0) && <> { // This sets the manual card order to the currently displayed order // Note: Perform this as a single update to change both properties correctly const newView = new MutableBoardView(activeView) newView.cardOrder = boardTree.orderedCards().map((o) => o.id) newView.sortOptions = [] mutator.updateBlock(newView, activeView, 'reorder') }} /> { mutator.changeViewSortOptions(activeView, []) }} /> } {this.sortDisplayOptions().map((option) => ( : : undefined} onClick={(propertyId: string) => { let newSortOptions: ISortOption[] = [] if (activeView.sortOptions[0] && activeView.sortOptions[0].propertyId === propertyId) { // Already sorting by name, so reverse it newSortOptions = [ {propertyId, reversed: !activeView.sortOptions[0].reversed}, ] } else { newSortOptions = [ {propertyId, reversed: false}, ] } mutator.changeViewSortOptions(activeView, newSortOptions) }} /> ))} {this.state.isSearching && { this.searchChanged(text) }} onKeyDown={(e) => { this.onSearchKeyDown(e) }} />} {!this.state.isSearching && } }/> CsvExporter.exportTableCsv(boardTree)} /> Archiver.exportBoardTree(boardTree)} /> this.testAddCards(100)} /> this.testAddCards(1000)} /> this.testDistributeCards()} /> this.testRandomizeIcons()} /> { this.props.addCard() }} text={( )} > {boardTree.cardTemplates.map((cardTemplate) => { return ( { this.props.addCardFromTemplate(cardTemplate.id) }} rightIcon={ }/> { this.props.editCardTemplate(cardTemplate.id) }} /> } id='delete' name={intl.formatMessage({id: 'ViewHeader.delete-template', defaultMessage: 'Delete'})} onClick={async () => { await mutator.deleteBlock(cardTemplate, 'delete card template') }} /> } /> ) })} { this.props.addCard() }} /> this.props.addCardTemplate()} />
) } private sortDisplayOptions() { const {boardTree} = this.props const options = boardTree.board.cardProperties.map((o) => ({id: o.id, name: o.name})) options.unshift({id: Constants.titleColumnId, name: 'Name'}) return options } } export default injectIntl(ViewHeader)