2020-10-25 15:52:46 +01:00
|
|
|
// 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 {MutableCommentBlock} from '../blocks/commentBlock'
|
|
|
|
import {IBlock} from '../blocks/block'
|
|
|
|
import {Utils} from '../utils'
|
|
|
|
import mutator from '../mutator'
|
|
|
|
|
2020-10-26 17:55:05 +01:00
|
|
|
import Editable from '../widgets/editable'
|
|
|
|
|
2020-10-25 15:52:46 +01:00
|
|
|
import Comment from './comment'
|
|
|
|
|
2020-10-25 19:23:23 +01:00
|
|
|
import './commentsList.scss'
|
|
|
|
|
2020-10-25 15:52:46 +01:00
|
|
|
type Props = {
|
|
|
|
comments: readonly IBlock[]
|
|
|
|
cardId: string
|
|
|
|
intl: IntlShape
|
|
|
|
}
|
|
|
|
|
2020-10-26 17:55:05 +01:00
|
|
|
type State = {
|
|
|
|
newComment: string
|
|
|
|
inputFocused: boolean
|
|
|
|
}
|
|
|
|
|
|
|
|
class CommentsList extends React.Component<Props, State> {
|
|
|
|
public constructor(props: Props) {
|
|
|
|
super(props)
|
|
|
|
this.state = {
|
|
|
|
newComment: '',
|
|
|
|
inputFocused: false,
|
|
|
|
}
|
|
|
|
}
|
2020-10-25 15:52:46 +01:00
|
|
|
|
|
|
|
public shouldComponentUpdate() {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2020-10-26 17:55:05 +01:00
|
|
|
private sendComment = () => {
|
2020-10-25 15:52:46 +01:00
|
|
|
const {cardId} = this.props
|
|
|
|
|
|
|
|
Utils.assertValue(cardId)
|
|
|
|
|
2020-10-26 17:55:05 +01:00
|
|
|
const block = new MutableCommentBlock({parentId: cardId, title: this.state.newComment})
|
2020-10-25 15:52:46 +01:00
|
|
|
mutator.insertBlock(block, 'add comment')
|
2020-10-26 17:55:05 +01:00
|
|
|
this.setState({newComment: ''})
|
2020-10-25 15:52:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public render(): JSX.Element {
|
|
|
|
const {comments, intl} = this.props
|
2020-10-25 19:06:24 +01:00
|
|
|
|
|
|
|
// TODO: Replace this placeholder
|
2020-10-25 15:52:46 +01:00
|
|
|
const username = 'John Smith'
|
|
|
|
const userImageUrl = 'data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100" style="fill: rgb(192, 192, 192);"><rect width="100" height="100" /></svg>'
|
|
|
|
|
|
|
|
return (
|
2020-10-25 19:23:23 +01:00
|
|
|
<div className='CommentsList'>
|
2020-10-25 15:52:46 +01:00
|
|
|
{comments.map((comment) => (
|
|
|
|
<Comment
|
|
|
|
key={comment.id}
|
|
|
|
comment={comment}
|
|
|
|
userImageUrl={userImageUrl}
|
|
|
|
username={username}
|
|
|
|
/>
|
|
|
|
))}
|
|
|
|
|
|
|
|
{/* New comment */}
|
|
|
|
|
|
|
|
<div className='commentrow'>
|
|
|
|
<img
|
|
|
|
className='comment-avatar'
|
|
|
|
src={userImageUrl}
|
|
|
|
/>
|
|
|
|
<Editable
|
|
|
|
className='newcomment'
|
|
|
|
placeholderText={intl.formatMessage({id: 'CardDetail.new-comment-placeholder', defaultMessage: 'Add a comment...'})}
|
2020-10-26 17:55:05 +01:00
|
|
|
onChange={(value: string) => this.setState({newComment: value})}
|
|
|
|
value={this.state.newComment}
|
|
|
|
onFocus={() => this.setState({inputFocused: true})}
|
|
|
|
onBlur={() => this.setState({inputFocused: false})}
|
|
|
|
onKeyDown={(e: React.KeyboardEvent<HTMLInputElement>) => {
|
2020-10-25 15:52:46 +01:00
|
|
|
if (e.keyCode === 13 && !(e.metaKey || e.ctrlKey) && !e.shiftKey && !e.altKey) {
|
2020-10-26 17:55:05 +01:00
|
|
|
this.sendComment()
|
2020-10-25 15:52:46 +01:00
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
|
2020-10-26 17:55:05 +01:00
|
|
|
{this.state.newComment && this.state.inputFocused &&
|
|
|
|
<div
|
|
|
|
className='octo-button filled'
|
|
|
|
onClick={() => {
|
|
|
|
Utils.log(`Send comment: ${this.state.newComment}`)
|
|
|
|
this.sendComment()
|
|
|
|
this.setState({inputFocused: false, newComment: ''})
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<FormattedMessage
|
|
|
|
id='CommentsList.send'
|
|
|
|
defaultMessage='Send'
|
|
|
|
/>
|
|
|
|
</div>}
|
2020-10-25 15:52:46 +01:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default injectIntl(CommentsList)
|