2022-08-29 21:24:38 +02:00
|
|
|
import { useState, useRef } from 'react'
|
2022-01-01 20:42:25 +01:00
|
|
|
import Image from 'next/image'
|
|
|
|
import Link from 'next/link'
|
|
|
|
import date from 'date-and-time'
|
2022-08-31 20:44:33 +01:00
|
|
|
import type { MotionProps } from 'framer-motion'
|
|
|
|
import { motion } from 'framer-motion'
|
2022-01-01 20:42:25 +01:00
|
|
|
|
2022-08-31 20:44:33 +01:00
|
|
|
import type { MessageWithMember } from '../../../../models/Message'
|
2022-01-07 21:21:38 +01:00
|
|
|
import { MessageText } from './MessageText'
|
|
|
|
import { Loader } from '../../../design/Loader'
|
|
|
|
import { MessageFile } from './MessageFile'
|
2022-08-29 21:24:38 +02:00
|
|
|
import { useAuthentication } from '../../../../tools/authentication'
|
2022-08-31 20:44:33 +01:00
|
|
|
import { MessageOptions } from './MessageOptions'
|
|
|
|
import { EditMessage } from './EditMessage'
|
2022-01-01 20:42:25 +01:00
|
|
|
|
2022-08-31 20:44:33 +01:00
|
|
|
export interface MessageProps extends MotionProps {
|
2022-01-01 20:42:25 +01:00
|
|
|
message: MessageWithMember
|
|
|
|
}
|
|
|
|
|
|
|
|
export const Message: React.FC<MessageProps> = (props) => {
|
|
|
|
const { message } = props
|
|
|
|
|
2022-08-29 21:24:38 +02:00
|
|
|
const textareaReference = useRef<HTMLTextAreaElement>(null)
|
|
|
|
const [isEditing, setIsEditing] = useState(false)
|
|
|
|
const { authentication, user } = useAuthentication()
|
|
|
|
|
|
|
|
const handleTextareaKeyDown: React.KeyboardEventHandler<HTMLFormElement> = (
|
|
|
|
event
|
|
|
|
) => {
|
|
|
|
if (event.key === 'Enter' && !event.shiftKey) {
|
|
|
|
event.preventDefault()
|
|
|
|
event.currentTarget.dispatchEvent(
|
|
|
|
new Event('submit', { cancelable: true, bubbles: true })
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleEdit = async (): Promise<void> => {
|
|
|
|
const newMessage = textareaReference.current?.value ?? message.value
|
|
|
|
if (
|
|
|
|
typeof newMessage === 'string' &&
|
|
|
|
newMessage.length > 0 &&
|
|
|
|
newMessage !== message.value
|
|
|
|
) {
|
|
|
|
try {
|
|
|
|
await authentication.api.put(`/messages/${message.id}`, {
|
|
|
|
value: newMessage
|
|
|
|
})
|
|
|
|
} catch {}
|
|
|
|
}
|
|
|
|
handleEditMode()
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleEditMode = (): void => {
|
|
|
|
setIsEditing((oldIsEditing) => {
|
|
|
|
return !oldIsEditing
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-01-01 20:42:25 +01:00
|
|
|
return (
|
2022-08-31 20:44:33 +01:00
|
|
|
<motion.div
|
|
|
|
layout
|
|
|
|
initial='initial'
|
|
|
|
animate='animate'
|
|
|
|
exit='exit'
|
2022-01-07 21:21:38 +01:00
|
|
|
data-cy={`message-${message.id}`}
|
2022-08-31 20:44:33 +01:00
|
|
|
className='group flex w-full p-4 transition hover:bg-gray-200 dark:hover:bg-gray-900'
|
|
|
|
transition={{ type: 'spring', stiffness: 500, damping: 60 }}
|
|
|
|
variants={{
|
|
|
|
initial: { x: -100, opacity: 0 },
|
|
|
|
animate: { x: 0, opacity: 1 },
|
|
|
|
exit: { opacity: 0 }
|
|
|
|
}}
|
2022-01-07 21:21:38 +01:00
|
|
|
>
|
2022-01-01 20:42:25 +01:00
|
|
|
<Link href={`/application/users/${message.member.user.id}`}>
|
2022-12-13 11:38:07 +01:00
|
|
|
<div className='mr-4 flex h-12 w-12 flex-shrink-0 items-center justify-center'>
|
|
|
|
<div className='h-10 w-10 drop-shadow-md'>
|
|
|
|
<Image
|
|
|
|
quality={100}
|
|
|
|
className='rounded-full'
|
|
|
|
src={
|
|
|
|
message.member.user.logo == null
|
|
|
|
? '/images/data/user-default.png'
|
|
|
|
: message.member.user.logo
|
|
|
|
}
|
|
|
|
alt={"Users's profil picture"}
|
|
|
|
width={50}
|
|
|
|
height={50}
|
|
|
|
draggable={false}
|
|
|
|
/>
|
2022-01-01 20:42:25 +01:00
|
|
|
</div>
|
2022-12-13 11:38:07 +01:00
|
|
|
</div>
|
2022-01-01 20:42:25 +01:00
|
|
|
</Link>
|
2022-08-29 21:24:38 +02:00
|
|
|
<div className='relative w-full whitespace-pre-wrap break-words break-all'>
|
2022-02-19 23:20:33 +01:00
|
|
|
<div className='flex w-max items-center'>
|
2022-01-01 20:42:25 +01:00
|
|
|
<Link href={`/application/users/${message.member.user.id}`}>
|
2022-12-13 11:38:07 +01:00
|
|
|
<span
|
|
|
|
data-cy='message-member-user-name'
|
|
|
|
className='font-bold text-gray-900 dark:text-gray-200'
|
|
|
|
>
|
|
|
|
{message.member.user.name}
|
|
|
|
</span>
|
2022-01-01 20:42:25 +01:00
|
|
|
</Link>
|
|
|
|
<span
|
|
|
|
data-cy='message-date'
|
2022-02-19 23:20:33 +01:00
|
|
|
className='ml-4 select-none text-xs text-gray-500 dark:text-gray-200'
|
2022-01-01 20:42:25 +01:00
|
|
|
>
|
|
|
|
{date.format(new Date(message.createdAt), 'DD/MM/YYYY - HH:mm:ss')}
|
|
|
|
</span>
|
|
|
|
</div>
|
2022-08-31 20:44:33 +01:00
|
|
|
|
|
|
|
{message.member.userId === user.id && (
|
|
|
|
<MessageOptions
|
|
|
|
message={message}
|
|
|
|
editMode={isEditing ? ':white_check_mark:' : ':pencil2:'}
|
|
|
|
handleEdit={isEditing ? handleEdit : handleEditMode}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
|
2022-01-07 21:21:38 +01:00
|
|
|
{message.type === 'text' ? (
|
2022-08-29 21:24:38 +02:00
|
|
|
<>
|
|
|
|
{isEditing ? (
|
2022-08-31 20:44:33 +01:00
|
|
|
<EditMessage
|
|
|
|
message={message}
|
|
|
|
textareaRef={textareaReference}
|
|
|
|
handleEdit={handleEdit}
|
|
|
|
handleKeyDown={handleTextareaKeyDown}
|
|
|
|
/>
|
2022-08-29 21:24:38 +02:00
|
|
|
) : (
|
|
|
|
<MessageText message={message} />
|
|
|
|
)}
|
|
|
|
</>
|
2022-01-07 21:21:38 +01:00
|
|
|
) : message.type === 'file' ? (
|
|
|
|
<MessageFile message={message} />
|
|
|
|
) : (
|
|
|
|
<Loader />
|
|
|
|
)}
|
2022-01-01 20:42:25 +01:00
|
|
|
</div>
|
2022-08-31 20:44:33 +01:00
|
|
|
</motion.div>
|
2022-01-01 20:42:25 +01:00
|
|
|
)
|
|
|
|
}
|