focalboard/webapp/src/components/viewMenu.tsx
2021-01-04 13:36:16 -08:00

187 lines
6.3 KiB
TypeScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import React from 'react'
import {injectIntl, IntlShape} from 'react-intl'
import {Board} from '../blocks/board'
import {IViewType, MutableBoardView} from '../blocks/boardView'
import {Constants} from '../constants'
import mutator from '../mutator'
import {Utils} from '../utils'
import {BoardTree} from '../viewModel/boardTree'
import AddIcon from '../widgets/icons/add'
import BoardIcon from '../widgets/icons/board'
import DeleteIcon from '../widgets/icons/delete'
import DuplicateIcon from '../widgets/icons/duplicate'
import TableIcon from '../widgets/icons/table'
import Menu from '../widgets/menu'
type Props = {
boardTree: BoardTree
board: Board,
showView: (id: string) => void
intl: IntlShape
readonly: boolean
}
export class ViewMenu extends React.PureComponent<Props> {
private handleDuplicateView = async () => {
const {boardTree, showView} = this.props
Utils.log('duplicateView')
const currentViewId = boardTree.activeView.id
const newView = boardTree.activeView.duplicate()
newView.title = `Copy of ${boardTree.activeView.title}`
await mutator.insertBlock(
newView,
'duplicate view',
async () => {
// This delay is needed because OctoListener has a default 100 ms notification delay before updates
setTimeout(() => {
showView(newView.id)
}, 120)
},
async () => {
showView(currentViewId)
},
)
}
private handleDeleteView = async () => {
const {boardTree, showView} = this.props
Utils.log('deleteView')
const view = boardTree.activeView
const nextView = boardTree.views.find((o) => o !== view)
await mutator.deleteBlock(view, 'delete view')
if (nextView) {
showView(nextView.id)
}
}
private handleViewClick = (id: string) => {
const {boardTree, showView} = this.props
Utils.log('view ' + id)
const view = boardTree.views.find((o) => o.id === id)
Utils.assert(view, `view not found: ${id}`)
if (view) {
showView(view.id)
}
}
private handleAddViewBoard = async () => {
const {board, boardTree, showView, intl} = this.props
Utils.log('addview-board')
const view = new MutableBoardView()
view.title = intl.formatMessage({id: 'View.NewBoardTitle', defaultMessage: 'Board view'})
view.viewType = 'board'
view.parentId = board.id
view.rootId = board.rootId
const oldViewId = boardTree.activeView.id
await mutator.insertBlock(
view,
'add view',
async () => {
// This delay is needed because OctoListener has a default 100 ms notification delay before updates
setTimeout(() => {
showView(view.id)
}, 120)
},
async () => {
showView(oldViewId)
})
}
private handleAddViewTable = async () => {
const {board, boardTree, showView, intl} = this.props
Utils.log('addview-table')
const view = new MutableBoardView()
view.title = intl.formatMessage({id: 'View.NewTableTitle', defaultMessage: 'Table view'})
view.viewType = 'table'
view.parentId = board.id
view.rootId = board.rootId
view.visiblePropertyIds = board.cardProperties.map((o) => o.id)
view.columnWidths = {}
view.columnWidths[Constants.titleColumnId] = Constants.defaultTitleColumnWidth
const oldViewId = boardTree.activeView.id
await mutator.insertBlock(
view,
'add view',
async () => {
// This delay is needed because OctoListener has a default 100 ms notification delay before updates
setTimeout(() => {
Utils.log(`showView: ${view.id}`)
showView(view.id)
}, 120)
},
async () => {
showView(oldViewId)
})
}
render(): JSX.Element {
const {boardTree} = this.props
return (
<Menu>
{boardTree.views.map((view) => (
<Menu.Text
key={view.id}
id={view.id}
name={view.title}
icon={this.iconForViewType(view.viewType)}
onClick={this.handleViewClick}
/>))}
<Menu.Separator/>
{!this.props.readonly &&
<Menu.Text
id='__duplicateView'
name='Duplicate View'
icon={<DuplicateIcon/>}
onClick={this.handleDuplicateView}
/>
}
{!this.props.readonly && boardTree.views.length > 1 &&
<Menu.Text
id='__deleteView'
name='Delete View'
icon={<DeleteIcon/>}
onClick={this.handleDeleteView}
/>
}
{!this.props.readonly &&
<Menu.SubMenu
id='__addView'
name='Add View'
icon={<AddIcon/>}
>
<Menu.Text
id='board'
name='Board'
icon={<BoardIcon/>}
onClick={this.handleAddViewBoard}
/>
<Menu.Text
id='table'
name='Table'
icon={<TableIcon/>}
onClick={this.handleAddViewTable}
/>
</Menu.SubMenu>
}
</Menu>
)
}
private iconForViewType(viewType: IViewType) {
switch (viewType) {
case 'board': return <BoardIcon/>
case 'table': return <TableIcon/>
default: return <div/>
}
}
}
export default injectIntl(ViewMenu)