MessageCard.tsx 2.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879
  1. import { useCallback } from 'react';
  2. import type { LinkProps } from 'next/link';
  3. import { useTranslation } from 'react-i18next';
  4. import ReactMarkdown from 'react-markdown';
  5. import { NextLink } from '~/components/ReactMarkdownComponents/NextLink';
  6. import { useRagSearchModal } from '../../../client/stores/rag-search';
  7. import styles from './MessageCard.module.scss';
  8. const moduleClass = styles['message-card'] ?? '';
  9. const userMessageCardModuleClass = styles['user-message-card'] ?? '';
  10. const UserMessageCard = ({ children }: { children: string }): JSX.Element => (
  11. <div className={`card d-inline-flex align-self-end bg-success-subtle bg-info-subtle ${moduleClass} ${userMessageCardModuleClass}`}>
  12. <div className="card-body">
  13. <ReactMarkdown>{children}</ReactMarkdown>
  14. </div>
  15. </div>
  16. );
  17. const assistantMessageCardModuleClass = styles['assistant-message-card'] ?? '';
  18. const NextLinkWrapper = (props: LinkProps & {children: string, href: string}): JSX.Element => {
  19. const { close: closeRagSearchModal } = useRagSearchModal();
  20. const onClick = useCallback(() => {
  21. closeRagSearchModal();
  22. }, [closeRagSearchModal]);
  23. return (
  24. <NextLink href={props.href} onClick={onClick} className="link-primary">
  25. {props.children}
  26. </NextLink>
  27. );
  28. };
  29. const AssistantMessageCard = ({ children }: { children: string }): JSX.Element => {
  30. const { t } = useTranslation();
  31. return (
  32. <div className={`card border-0 ${moduleClass} ${assistantMessageCardModuleClass}`}>
  33. <div className="card-body d-flex">
  34. <div className="me-2 me-lg-3">
  35. <span className="growi-custom-icons grw-ai-icon rounded-pill">growi_ai</span>
  36. </div>
  37. <div>
  38. { children.length > 0
  39. ? (
  40. <ReactMarkdown components={{ a: NextLinkWrapper }}>{children}</ReactMarkdown>
  41. )
  42. : (
  43. <span className="text-thinking">
  44. {t('modal_aichat.progress_label')} <span className="material-symbols-outlined">more_horiz</span>
  45. </span>
  46. )
  47. }
  48. </div>
  49. </div>
  50. </div>
  51. );
  52. };
  53. type Props = {
  54. role: 'user' | 'assistant',
  55. children: string,
  56. }
  57. export const MessageCard = (props: Props): JSX.Element => {
  58. const { role, children } = props;
  59. return role === 'user'
  60. ? <UserMessageCard>{children}</UserMessageCard>
  61. : <AssistantMessageCard>{children}</AssistantMessageCard>;
  62. };