2020-10-24 09:39:01 +02:00
|
|
|
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React from 'react'
|
2020-10-24 10:08:22 +02:00
|
|
|
import {injectIntl, IntlShape, FormattedMessage} from 'react-intl'
|
2020-10-24 09:39:01 +02:00
|
|
|
|
|
|
|
import {Archiver} from '../archiver'
|
|
|
|
import {ISortOption} from '../blocks/boardView'
|
|
|
|
import {BlockIcons} from '../blockIcons'
|
|
|
|
import {MutableCard} from '../blocks/card'
|
2020-10-24 10:08:22 +02:00
|
|
|
import {IPropertyTemplate} from '../blocks/board'
|
2020-10-24 09:39:01 +02:00
|
|
|
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 {Editable} from './editable'
|
2020-10-25 11:10:59 +01:00
|
|
|
import FilterComponent from './filterComponent'
|
2020-10-24 09:39:01 +02:00
|
|
|
|
2020-10-25 22:29:20 +01:00
|
|
|
import './viewHeader.scss'
|
|
|
|
|
2020-10-24 09:39:01 +02:00
|
|
|
type Props = {
|
|
|
|
boardTree?: BoardTree
|
|
|
|
showView: (id: string) => void
|
|
|
|
setSearchText: (text: string) => void
|
|
|
|
addCard: (show: boolean) => void
|
|
|
|
withGroupBy?: boolean
|
2020-10-24 10:08:22 +02:00
|
|
|
intl: IntlShape
|
2020-10-24 09:39:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type State = {
|
|
|
|
isSearching: boolean
|
|
|
|
showFilter: boolean
|
|
|
|
}
|
|
|
|
|
2020-10-24 10:08:22 +02:00
|
|
|
class ViewHeader extends React.Component<Props, State> {
|
2020-10-24 09:39:01 +02:00
|
|
|
private searchFieldRef = React.createRef<Editable>()
|
|
|
|
|
|
|
|
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({...this.state, 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
|
|
|
|
|
|
|
|
for (let i = 0; i < count; i++) {
|
|
|
|
const card = new MutableCard()
|
|
|
|
card.parentId = boardTree.board.id
|
|
|
|
card.properties = CardFilter.propertiesThatMeetFilterGroup(activeView.filter, board.cardProperties)
|
|
|
|
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
|
|
|
|
card.title = `Test Card ${startCount + i + 1}`
|
|
|
|
card.icon = BlockIcons.shared.randomIcon()
|
|
|
|
}
|
|
|
|
await mutator.insertBlock(card, 'test add card')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private async testRandomizeIcons() {
|
|
|
|
const {boardTree} = this.props
|
|
|
|
|
|
|
|
for (const card of boardTree.cards) {
|
|
|
|
mutator.changeIcon(card, BlockIcons.shared.randomIcon(), 'randomize icon')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
render(): JSX.Element {
|
2020-10-24 10:08:22 +02:00
|
|
|
const {boardTree, showView, withGroupBy, intl} = this.props
|
2020-10-24 09:39:01 +02:00
|
|
|
const {board, activeView} = boardTree
|
|
|
|
|
|
|
|
const hasFilter = activeView.filter && activeView.filter.filters?.length > 0
|
|
|
|
const hasSort = activeView.sortOptions.length > 0
|
|
|
|
|
|
|
|
return (
|
2020-10-25 22:29:20 +01:00
|
|
|
<div className='ViewHeader'>
|
2020-10-24 09:39:01 +02:00
|
|
|
<Editable
|
|
|
|
style={{color: '#000000', fontWeight: 600}}
|
|
|
|
text={activeView.title}
|
|
|
|
placeholderText='Untitled View'
|
|
|
|
onChanged={(text) => {
|
|
|
|
mutator.changeTitle(activeView, text)
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
<MenuWrapper>
|
|
|
|
<div
|
|
|
|
className='octo-button'
|
|
|
|
style={{color: '#000000', fontWeight: 600}}
|
|
|
|
>
|
|
|
|
<div className='imageDropdown'/>
|
|
|
|
</div>
|
|
|
|
<ViewMenu
|
|
|
|
board={board}
|
|
|
|
boardTree={boardTree}
|
|
|
|
showView={showView}
|
|
|
|
/>
|
|
|
|
</MenuWrapper>
|
|
|
|
<div className='octo-spacer'/>
|
|
|
|
<MenuWrapper>
|
|
|
|
<div className={'octo-button'}>
|
|
|
|
<FormattedMessage
|
2020-10-24 09:45:06 +02:00
|
|
|
id='ViewHeader.properties'
|
2020-10-24 09:39:01 +02:00
|
|
|
defaultMessage='Properties'
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<Menu>
|
2020-10-24 10:08:22 +02:00
|
|
|
{boardTree.board.cardProperties.map((option: IPropertyTemplate) => (
|
2020-10-24 09:39:01 +02:00
|
|
|
<Menu.Switch
|
|
|
|
key={option.id}
|
|
|
|
id={option.id}
|
|
|
|
name={option.name}
|
|
|
|
isOn={activeView.visiblePropertyIds.includes(option.id)}
|
|
|
|
onClick={(propertyId: string) => {
|
2020-10-24 10:08:22 +02:00
|
|
|
const property = boardTree.board.cardProperties.find((o: IPropertyTemplate) => o.id === propertyId)
|
2020-10-24 09:39:01 +02:00
|
|
|
Utils.assertValue(property)
|
|
|
|
Utils.log(`Toggle property ${property.name}`)
|
|
|
|
|
|
|
|
let newVisiblePropertyIds = []
|
|
|
|
if (activeView.visiblePropertyIds.includes(propertyId)) {
|
2020-10-24 10:08:22 +02:00
|
|
|
newVisiblePropertyIds = activeView.visiblePropertyIds.filter((o: string) => o !== propertyId)
|
2020-10-24 09:39:01 +02:00
|
|
|
} else {
|
|
|
|
newVisiblePropertyIds = [...activeView.visiblePropertyIds, propertyId]
|
|
|
|
}
|
|
|
|
mutator.changeViewVisibleProperties(activeView, newVisiblePropertyIds)
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
))}
|
|
|
|
</Menu>
|
|
|
|
</MenuWrapper>
|
|
|
|
{withGroupBy &&
|
|
|
|
<MenuWrapper>
|
|
|
|
<div
|
|
|
|
className='octo-button'
|
|
|
|
id='groupByButton'
|
|
|
|
>
|
2020-10-24 09:45:06 +02:00
|
|
|
<FormattedMessage
|
|
|
|
id='ViewHeader.group-by'
|
|
|
|
defaultMessage='Group by {property}'
|
|
|
|
values={{
|
|
|
|
property: (
|
|
|
|
<span
|
|
|
|
style={{color: '#000000'}}
|
|
|
|
id='groupByLabel'
|
|
|
|
>
|
|
|
|
{boardTree.groupByProperty?.name}
|
|
|
|
</span>
|
|
|
|
),
|
|
|
|
}}
|
|
|
|
/>
|
2020-10-24 09:39:01 +02:00
|
|
|
</div>
|
|
|
|
<Menu>
|
2020-10-24 10:08:22 +02:00
|
|
|
{boardTree.board.cardProperties.filter((o: IPropertyTemplate) => o.type === 'select').map((option: IPropertyTemplate) => (
|
2020-10-24 09:39:01 +02:00
|
|
|
<Menu.Text
|
|
|
|
key={option.id}
|
|
|
|
id={option.id}
|
|
|
|
name={option.name}
|
|
|
|
onClick={(id) => {
|
|
|
|
if (boardTree.activeView.groupById === id) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
mutator.changeViewGroupById(boardTree.activeView, id)
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
))}
|
|
|
|
</Menu>
|
|
|
|
</MenuWrapper>}
|
|
|
|
<div
|
|
|
|
className={hasFilter ? 'octo-button active' : 'octo-button'}
|
|
|
|
style={{position: 'relative', overflow: 'unset'}}
|
|
|
|
onClick={this.filterClicked}
|
|
|
|
>
|
|
|
|
<FormattedMessage
|
2020-10-24 09:45:06 +02:00
|
|
|
id='ViewHeader.filter'
|
2020-10-24 09:39:01 +02:00
|
|
|
defaultMessage='Filter'
|
|
|
|
/>
|
|
|
|
{this.state.showFilter &&
|
|
|
|
<FilterComponent
|
|
|
|
boardTree={boardTree}
|
|
|
|
onClose={this.hideFilter}
|
|
|
|
/>}
|
|
|
|
</div>
|
|
|
|
<MenuWrapper>
|
|
|
|
<div className={hasSort ? 'octo-button active' : 'octo-button'}>
|
|
|
|
<FormattedMessage
|
2020-10-24 09:45:06 +02:00
|
|
|
id='ViewHeader.sort'
|
2020-10-24 09:39:01 +02:00
|
|
|
defaultMessage='Sort'
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<Menu>
|
2020-10-26 09:56:12 -07:00
|
|
|
<Menu.Text
|
|
|
|
id='none'
|
|
|
|
name='None'
|
|
|
|
onClick={() => {
|
|
|
|
mutator.changeViewSortOptions(activeView, [])
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
|
|
|
|
<Menu.Separator />
|
|
|
|
|
2020-10-24 10:08:22 +02:00
|
|
|
{boardTree.board.cardProperties.map((option: IPropertyTemplate) => (
|
2020-10-24 09:39:01 +02:00
|
|
|
<Menu.Text
|
|
|
|
key={option.id}
|
|
|
|
id={option.id}
|
|
|
|
name={option.name}
|
|
|
|
icon={(activeView.sortOptions[0]?.propertyId === option.id) ? activeView.sortOptions[0].reversed ? 'sortUp' : 'sortDown' : 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)
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
))}
|
|
|
|
</Menu>
|
|
|
|
</MenuWrapper>
|
|
|
|
{this.state.isSearching &&
|
2020-10-24 10:08:22 +02:00
|
|
|
<Editable
|
|
|
|
ref={this.searchFieldRef}
|
|
|
|
text={boardTree.getSearchText()}
|
|
|
|
placeholderText={intl.formatMessage({id: 'ViewHeader.search-text', defaultMessage: 'Search text'})}
|
|
|
|
style={{color: '#000000'}}
|
|
|
|
onChanged={(text) => {
|
|
|
|
this.searchChanged(text)
|
|
|
|
}}
|
|
|
|
onKeyDown={(e) => {
|
|
|
|
this.onSearchKeyDown(e)
|
|
|
|
}}
|
|
|
|
/>}
|
2020-10-24 09:39:01 +02:00
|
|
|
{!this.state.isSearching &&
|
|
|
|
<div
|
|
|
|
className='octo-button'
|
|
|
|
onClick={() => {
|
|
|
|
this.setState({...this.state, isSearching: true})
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<FormattedMessage
|
2020-10-24 09:45:06 +02:00
|
|
|
id='ViewHeader.search'
|
2020-10-24 09:39:01 +02:00
|
|
|
defaultMessage='Search'
|
|
|
|
/>
|
|
|
|
</div>}
|
|
|
|
<MenuWrapper>
|
|
|
|
<div className='imageOptions'/>
|
|
|
|
<Menu>
|
|
|
|
<Menu.Text
|
|
|
|
id='exportCsv'
|
2020-10-24 10:08:22 +02:00
|
|
|
name={intl.formatMessage({id: 'ViewHeader.export-csv', defaultMessage: 'Export to CSV'})}
|
2020-10-24 09:39:01 +02:00
|
|
|
onClick={() => CsvExporter.exportTableCsv(boardTree)}
|
|
|
|
/>
|
|
|
|
<Menu.Text
|
|
|
|
id='exportBoardArchive'
|
2020-10-24 10:08:22 +02:00
|
|
|
name={intl.formatMessage({id: 'ViewHeader.export-board-archive', defaultMessage: 'Export Board Archive'})}
|
2020-10-24 09:39:01 +02:00
|
|
|
onClick={() => Archiver.exportBoardTree(boardTree)}
|
|
|
|
/>
|
|
|
|
<Menu.Text
|
|
|
|
id='testAdd100Cards'
|
2020-10-24 10:08:22 +02:00
|
|
|
name={intl.formatMessage({id: 'ViewHeader.test-add-100-cards', defaultMessage: 'TEST: Add 100 cards'})}
|
2020-10-24 09:39:01 +02:00
|
|
|
onClick={() => this.testAddCards(100)}
|
|
|
|
/>
|
|
|
|
<Menu.Text
|
|
|
|
id='testAdd1000Cards'
|
2020-10-24 10:08:22 +02:00
|
|
|
name={intl.formatMessage({id: 'ViewHeader.test-add-1000-cards', defaultMessage: 'TEST: Add 1,000 cards'})}
|
2020-10-24 09:39:01 +02:00
|
|
|
onClick={() => this.testAddCards(1000)}
|
|
|
|
/>
|
|
|
|
<Menu.Text
|
|
|
|
id='testRandomizeIcons'
|
2020-10-24 10:08:22 +02:00
|
|
|
name={intl.formatMessage({id: 'ViewHeader.test-randomize-icons', defaultMessage: 'TEST: Randomize icons'})}
|
2020-10-24 09:39:01 +02:00
|
|
|
onClick={() => this.testRandomizeIcons()}
|
|
|
|
/>
|
|
|
|
</Menu>
|
|
|
|
</MenuWrapper>
|
|
|
|
<div
|
|
|
|
className='octo-button filled'
|
|
|
|
onClick={() => {
|
|
|
|
this.props.addCard(true)
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<FormattedMessage
|
2020-10-24 09:45:06 +02:00
|
|
|
id='ViewHeader.new'
|
2020-10-24 09:39:01 +02:00
|
|
|
defaultMessage='New'
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2020-10-24 10:08:22 +02:00
|
|
|
|
|
|
|
export default injectIntl(ViewHeader)
|