File size: 3,474 Bytes
246d201
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
import { useState, useEffect } from "react";
import { useTranslation } from "react-i18next";
import Markdown from "react-markdown";
import remarkGfm from "remark-gfm";
import { code } from "../markdown/code";
import { ol, ul } from "../markdown/list";
import ArrowUp from "#/icons/angle-up-solid.svg?react";
import ArrowDown from "#/icons/angle-down-solid.svg?react";
import CheckCircle from "#/icons/check-circle-solid.svg?react";
import XCircle from "#/icons/x-circle-solid.svg?react";
import { cn } from "#/utils/utils";

interface ExpandableMessageProps {
  id?: string;
  message: string;
  type: string;
  success?: boolean;
}

export function ExpandableMessage({

  id,

  message,

  type,

  success,

}: ExpandableMessageProps) {
  const { t, i18n } = useTranslation();
  const [showDetails, setShowDetails] = useState(true);
  const [headline, setHeadline] = useState("");
  const [details, setDetails] = useState(message);

  useEffect(() => {
    if (id && i18n.exists(id)) {
      setHeadline(t(id));
      setDetails(message);
      setShowDetails(false);
    }
  }, [id, message, i18n.language]);

  const statusIconClasses = "h-4 w-4 ml-2 inline";

  return (
    <div

      className={cn(

        "flex gap-2 items-center justify-start border-l-2 pl-2 my-2 py-2",

        type === "error" ? "border-danger" : "border-neutral-300",

      )}

    >

      <div className="text-sm w-full">

        {headline && (

          <div className="flex flex-row justify-between items-center w-full">

            <span

              className={cn(

                "font-bold",

                type === "error" ? "text-danger" : "text-neutral-300",

              )}

            >

              {headline}

              <button

                type="button"

                onClick={() => setShowDetails(!showDetails)}

                className="cursor-pointer text-left"

              >

                {showDetails ? (

                  <ArrowUp

                    className={cn(

                      "h-4 w-4 ml-2 inline",

                      type === "error" ? "fill-danger" : "fill-neutral-300",

                    )}

                  />

                ) : (

                  <ArrowDown

                    className={cn(

                      "h-4 w-4 ml-2 inline",

                      type === "error" ? "fill-danger" : "fill-neutral-300",

                    )}

                  />

                )}

              </button>

            </span>

            {type === "action" && success !== undefined && (

              <span className="flex-shrink-0">

                {success ? (

                  <CheckCircle

                    data-testid="status-icon"

                    className={cn(statusIconClasses, "fill-success")}

                  />

                ) : (

                  <XCircle

                    data-testid="status-icon"

                    className={cn(statusIconClasses, "fill-danger")}

                  />

                )}

              </span>

            )}

          </div>

        )}

        {showDetails && (

          <Markdown

            className="text-sm overflow-auto"

            components={{

              code,

              ul,

              ol,

            }}

            remarkPlugins={[remarkGfm]}

          >

            {details}

          </Markdown>

        )}

      </div>

    </div>
  );
}