2024-08-12 09:50:54 +00:00
|
|
|
import { FC, useCallback, useState } from 'react'
|
|
|
|
import {
|
2024-10-10 13:01:30 +00:00
|
|
|
CommentId,
|
2024-08-12 09:50:54 +00:00
|
|
|
ReviewPanelCommentThreadMessage,
|
|
|
|
} from '../../../../../types/review-panel/review-panel'
|
|
|
|
import { useTranslation } from 'react-i18next'
|
|
|
|
import { formatTimeBasedOnYear } from '@/features/utils/format-date'
|
2024-10-18 10:14:45 +00:00
|
|
|
import OLTooltip from '@/features/ui/components/ol/ol-tooltip'
|
2024-08-12 09:50:54 +00:00
|
|
|
import MaterialIcon from '@/shared/components/material-icon'
|
|
|
|
import AutoExpandingTextArea from '@/shared/components/auto-expanding-text-area'
|
|
|
|
import { buildName } from '../utils/build-name'
|
|
|
|
import ReviewPanelCommentOptions from './review-panel-comment-options'
|
|
|
|
import { ExpandableContent } from './review-panel-expandable-content'
|
|
|
|
import ReviewPanelDeleteCommentModal from './review-panel-delete-comment-modal'
|
2024-10-21 08:54:39 +00:00
|
|
|
import { useUserContext } from '@/shared/context/user-context'
|
2024-08-12 09:50:54 +00:00
|
|
|
|
|
|
|
export const ReviewPanelMessage: FC<{
|
|
|
|
message: ReviewPanelCommentThreadMessage
|
|
|
|
hasReplies: boolean
|
|
|
|
isReply: boolean
|
2024-10-10 13:01:30 +00:00
|
|
|
onResolve?: () => Promise<void>
|
|
|
|
onEdit?: (commentId: CommentId, content: string) => Promise<void>
|
|
|
|
onDelete?: (CommentId: CommentId) => Promise<void>
|
2024-08-19 13:16:56 +00:00
|
|
|
isThreadResolved: boolean
|
|
|
|
}> = ({
|
|
|
|
message,
|
|
|
|
isReply,
|
|
|
|
hasReplies,
|
|
|
|
onResolve,
|
2024-10-10 13:01:30 +00:00
|
|
|
onEdit,
|
|
|
|
onDelete,
|
2024-08-19 13:16:56 +00:00
|
|
|
isThreadResolved,
|
|
|
|
}) => {
|
2024-08-12 09:50:54 +00:00
|
|
|
const { t } = useTranslation()
|
|
|
|
const [editing, setEditing] = useState(false)
|
|
|
|
const [deleting, setDeleting] = useState(false)
|
|
|
|
const [content, setContent] = useState(message.content)
|
2024-10-21 08:54:39 +00:00
|
|
|
const user = useUserContext()
|
2024-08-12 09:50:54 +00:00
|
|
|
|
|
|
|
const handleEditOption = useCallback(() => setEditing(true), [])
|
|
|
|
const showDeleteModal = useCallback(() => setDeleting(true), [])
|
|
|
|
const hideDeleteModal = useCallback(() => setDeleting(false), [])
|
|
|
|
|
2024-10-10 13:01:30 +00:00
|
|
|
const handleSubmit = useCallback(() => {
|
|
|
|
onEdit?.(message.id, content)
|
|
|
|
setEditing(false)
|
|
|
|
}, [content, message.id, onEdit])
|
2024-08-12 09:50:54 +00:00
|
|
|
|
2024-10-10 13:01:30 +00:00
|
|
|
const handleDelete = useCallback(() => {
|
|
|
|
onDelete?.(message.id)
|
|
|
|
setDeleting(false)
|
|
|
|
}, [message.id, onDelete])
|
2024-08-12 09:50:54 +00:00
|
|
|
|
|
|
|
if (editing) {
|
|
|
|
return (
|
|
|
|
<div>
|
|
|
|
<AutoExpandingTextArea
|
|
|
|
className="review-panel-comment-input"
|
|
|
|
onBlur={handleSubmit}
|
|
|
|
onChange={e => setContent(e.target.value)}
|
|
|
|
onKeyDown={e => {
|
|
|
|
if (
|
|
|
|
e.key === 'Enter' &&
|
|
|
|
!e.shiftKey &&
|
|
|
|
!e.ctrlKey &&
|
|
|
|
!e.metaKey &&
|
|
|
|
content
|
|
|
|
) {
|
|
|
|
e.preventDefault()
|
|
|
|
handleSubmit()
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
value={content}
|
|
|
|
autoFocus // eslint-disable-line jsx-a11y/no-autofocus
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="review-panel-comment">
|
|
|
|
<div className="review-panel-entry-header">
|
|
|
|
<div>
|
|
|
|
<div className="review-panel-entry-user">
|
|
|
|
{buildName(message.user)}
|
|
|
|
</div>
|
|
|
|
<div className="review-panel-entry-time">
|
|
|
|
{formatTimeBasedOnYear(message.timestamp)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className="review-panel-entry-actions">
|
2024-08-19 13:16:56 +00:00
|
|
|
{!isReply && !isThreadResolved && (
|
2024-10-18 10:14:45 +00:00
|
|
|
<OLTooltip
|
2024-08-12 09:50:54 +00:00
|
|
|
id="resolve-thread"
|
|
|
|
overlayProps={{ placement: 'bottom' }}
|
|
|
|
description={t('resolve_comment')}
|
2024-10-02 11:58:46 +00:00
|
|
|
tooltipProps={{ className: 'review-panel-tooltip' }}
|
2024-08-12 09:50:54 +00:00
|
|
|
>
|
2024-10-18 10:14:45 +00:00
|
|
|
<button type="button" className="btn" onClick={onResolve}>
|
2024-08-12 09:50:54 +00:00
|
|
|
<MaterialIcon
|
|
|
|
type="check"
|
|
|
|
className="review-panel-entry-actions-icon"
|
|
|
|
accessibilityLabel={t('resolve_comment')}
|
|
|
|
/>
|
2024-10-18 10:14:45 +00:00
|
|
|
</button>
|
|
|
|
</OLTooltip>
|
2024-08-12 09:50:54 +00:00
|
|
|
)}
|
|
|
|
|
2024-08-19 13:16:56 +00:00
|
|
|
{!isThreadResolved && (
|
|
|
|
<ReviewPanelCommentOptions
|
2024-10-21 08:54:39 +00:00
|
|
|
belongsToCurrentUser={user.id === message.user.id}
|
2024-08-19 13:16:56 +00:00
|
|
|
onEdit={handleEditOption}
|
|
|
|
onDelete={showDeleteModal}
|
2024-10-18 10:14:45 +00:00
|
|
|
id={message.id}
|
2024-08-19 13:16:56 +00:00
|
|
|
/>
|
|
|
|
)}
|
2024-08-12 09:50:54 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<ExpandableContent className="review-panel-comment-body">
|
|
|
|
{message.content}
|
|
|
|
</ExpandableContent>
|
|
|
|
{deleting && (
|
|
|
|
<ReviewPanelDeleteCommentModal
|
|
|
|
onHide={hideDeleteModal}
|
|
|
|
onDelete={handleDelete}
|
|
|
|
title={hasReplies ? t('delete_comment_thread') : t('delete_comment')}
|
|
|
|
message={
|
|
|
|
hasReplies
|
|
|
|
? t('delete_comment_thread_message')
|
|
|
|
: t('delete_comment_message')
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|