// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. import React from 'react' import {BlockIcons} from '../blockIcons' import {MutableCommentBlock} from '../blocks/commentBlock' import {IOrderedBlock} from '../blocks/orderedBlock' import {MutableTextBlock} from '../blocks/textBlock' import {BoardTree} from '../viewModel/boardTree' import {CardTree, MutableCardTree} from '../viewModel/cardTree' import {Menu as OldMenu, MenuOption} from '../menu' import mutator from '../mutator' import {OctoListener} from '../octoListener' import {IBlock} from '../blocks/block' import {OctoUtils} from '../octoUtils' import {PropertyMenu} from '../propertyMenu' import {Utils} from '../utils' import Button from './button' import {Editable} from './editable' import {MarkdownEditor} from './markdownEditor' type Props = { boardTree: BoardTree cardId: string } type State = { isHoverOnCover: boolean cardTree?: CardTree } export default class CardDetail extends React.Component { private titleRef = React.createRef() private cardListener?: OctoListener constructor(props: Props) { super(props) this.state = {isHoverOnCover: false} } componentDidMount() { this.cardListener = new OctoListener() this.cardListener.open([this.props.cardId], async (blockId) => { Utils.log(`cardListener.onChanged: ${blockId}`) await cardTree.sync() this.setState({...this.state, cardTree}) }) const cardTree = new MutableCardTree(this.props.cardId) cardTree.sync().then(() => { this.setState({...this.state, cardTree}) setTimeout(() => { if (this.titleRef.current) { this.titleRef.current.focus() } }, 0) }) } componentWillUnmount() { this.cardListener?.close() this.cardListener = undefined } render() { const {boardTree} = this.props const {cardTree} = this.state const {board} = boardTree if (!cardTree) { return null } const {card, comments} = cardTree const newCommentPlaceholderText = 'Add a comment...' const backgroundRef = React.createRef() const newCommentRef = React.createRef() const sendCommentButtonRef = React.createRef() let contentElements if (cardTree.contents.length > 0) { contentElements = (
{cardTree.contents.map((block) => { if (block.type === 'text') { const cardText = block.title return (
{ this.showContentBlockMenu(e, block) }} >
{ Utils.log(`change text ${block.id}, ${text}`) mutator.changeTitle(block, text, 'edit card text') }} />
) } else if (block.type === 'image') { const url = block.fields.url return (
{ this.showContentBlockMenu(e, block) }} >
{block.title}
) } return
})}
) } else { contentElements = (
{ const block = new MutableTextBlock() block.parentId = card.id block.title = text block.order = cardTree.contents.length * 1000 mutator.insertBlock(block, 'add card text') }} />
) } const icon = card.icon // TODO: Replace this placeholder const username = 'John Smith' const userImageUrl = 'data:image/svg+xml,' return ( <>
{icon ?
{ this.iconClicked(e) }} >{icon}
: undefined }
{ this.setState({...this.state, isHoverOnCover: true}) }} onMouseLeave={() => { this.setState({...this.state, isHoverOnCover: false}) }} >
{ mutator.changeTitle(card, text) }} /> {/* Property list */}
{board.cardProperties.map((propertyTemplate) => { return (
{ const menu = PropertyMenu.shared menu.property = propertyTemplate menu.onNameChanged = (propertyName) => { Utils.log('menu.onNameChanged') mutator.renameProperty(board, propertyTemplate.id, propertyName) } menu.onMenuClicked = async (command) => { switch (command) { case 'type-text': await mutator.changePropertyType(board, propertyTemplate, 'text') break case 'type-number': await mutator.changePropertyType(board, propertyTemplate, 'number') break case 'type-createdTime': await mutator.changePropertyType(board, propertyTemplate, 'createdTime') break case 'type-updatedTime': await mutator.changePropertyType(board, propertyTemplate, 'updatedTime') break case 'type-select': await mutator.changePropertyType(board, propertyTemplate, 'select') break case 'delete': await mutator.deleteProperty(boardTree, propertyTemplate.id) break default: Utils.assertFailure(`Unhandled menu id: ${command}`) } } menu.showAtElement(e.target as HTMLElement) }} >{propertyTemplate.name}
{OctoUtils.propertyValueEditableElement(card, propertyTemplate)}
) })}
{ // TODO: Show UI await mutator.insertPropertyTemplate(boardTree) }} >+ Add a property
{/* Comments */}
{comments.map((comment) => { const optionsButtonRef = React.createRef() const showCommentMenu = (e: React.MouseEvent, activeComment: IBlock) => { OldMenu.shared.options = [ {id: 'delete', name: 'Delete'}, ] OldMenu.shared.onMenuClicked = (id) => { switch (id) { case 'delete': { mutator.deleteBlock(activeComment) break } } } OldMenu.shared.showAtElement(e.target as HTMLElement) } return (
{ optionsButtonRef.current.style.display = null }} onMouseLeave={() => { optionsButtonRef.current.style.display = 'none' }} >
{username}
{(new Date(comment.createAt)).toLocaleTimeString()}
{ showCommentMenu(e, comment) }} >...
{comment.title}
) })} {/* New comment */}
{ }} onFocus={() => { sendCommentButtonRef.current.style.display = null }} onBlur={() => { if (!newCommentRef.current.text) { sendCommentButtonRef.current.style.display = 'none' } }} onKeyDown={(e) => { if (e.keyCode === 13 && !(e.metaKey || e.ctrlKey) && !e.shiftKey && !e.altKey) { sendCommentButtonRef.current.click() } }} />
{ const text = newCommentRef.current.text Utils.log(`Send comment: ${newCommentRef.current.text}`) this.sendComment(text) newCommentRef.current.text = undefined newCommentRef.current.blur() }} >Send

{/* Content blocks */}
{contentElements}
{ OldMenu.shared.options = [ {id: 'text', name: 'Text'}, {id: 'image', name: 'Image'}, ] OldMenu.shared.onMenuClicked = async (optionId: string, type?: string) => { switch (optionId) { case 'text': const block = new MutableTextBlock() block.parentId = card.id block.order = cardTree.contents.length * 1000 await mutator.insertBlock(block, 'add text') break case 'image': Utils.selectLocalFile( (file) => { mutator.createImageBlock(card.id, file, cardTree.contents.length * 1000) }, '.jpg,.jpeg,.png') break } } OldMenu.shared.showAtElement(e.target as HTMLElement) }} >Add content
) } async sendComment(text: string) { const {cardId} = this.props Utils.assertValue(cardId) const block = new MutableCommentBlock({parentId: cardId, title: text}) await mutator.insertBlock(block, 'add comment') } private showContentBlockMenu(e: React.MouseEvent, block: IOrderedBlock) { const {cardTree} = this.state const {cardId} = this.props const index = cardTree.contents.indexOf(block) const options: MenuOption[] = [] if (index > 0) { options.push({id: 'moveUp', name: 'Move up'}) } if (index < cardTree.contents.length - 1) { options.push({id: 'moveDown', name: 'Move down'}) } options.push( {id: 'insertAbove', name: 'Insert above', type: 'submenu'}, {id: 'delete', name: 'Delete'}, ) OldMenu.shared.options = options OldMenu.shared.subMenuOptions.set('insertAbove', [ {id: 'text', name: 'Text'}, {id: 'image', name: 'Image'}, ]) OldMenu.shared.onMenuClicked = (optionId: string, type?: string) => { switch (optionId) { case 'moveUp': { if (index < 1) { Utils.logError(`Unexpected index ${index}`); return } const previousBlock = cardTree.contents[index - 1] const newOrder = OctoUtils.getOrderBefore(previousBlock, cardTree.contents) Utils.log(`moveUp ${newOrder}`) mutator.changeOrder(block, newOrder, 'move up') break } case 'moveDown': { if (index >= cardTree.contents.length - 1) { Utils.logError(`Unexpected index ${index}`); return } const nextBlock = cardTree.contents[index + 1] const newOrder = OctoUtils.getOrderAfter(nextBlock, cardTree.contents) Utils.log(`moveDown ${newOrder}`) mutator.changeOrder(block, newOrder, 'move down') break } case 'insertAbove-text': { const newBlock = new MutableTextBlock() newBlock.parentId = cardId // TODO: Handle need to reorder all blocks newBlock.order = OctoUtils.getOrderBefore(block, cardTree.contents) Utils.log(`insert block ${block.id}, order: ${block.order}`) mutator.insertBlock(newBlock, 'insert card text') break } case 'insertAbove-image': { Utils.selectLocalFile( (file) => { mutator.createImageBlock(cardId, file, OctoUtils.getOrderBefore(block, cardTree.contents)) }, '.jpg,.jpeg,.png') break } case 'delete': { mutator.deleteBlock(block) break } } } OldMenu.shared.showAtElement(e.target as HTMLElement) } private iconClicked(e: React.MouseEvent) { const {cardTree} = this.state const {card} = cardTree OldMenu.shared.options = [ {id: 'random', name: 'Random'}, {id: 'remove', name: 'Remove Icon'}, ] OldMenu.shared.onMenuClicked = (optionId: string, type?: string) => { switch (optionId) { case 'remove': mutator.changeIcon(card, undefined, 'remove icon') break case 'random': const newIcon = BlockIcons.shared.randomIcon() mutator.changeIcon(card, newIcon) break } } OldMenu.shared.showAtElement(e.target as HTMLElement) } close() { OldMenu.shared.hide() PropertyMenu.shared.hide() } }