@bcrumbs.net/inbox 0.0.24 → 0.0.25
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/index.cjs.js +16 -10
- package/index.esm.js +16 -10
- package/package.json +2 -2
- package/src/app/resources/components/ListResources/Utils.d.ts +1 -0
package/index.cjs.js
CHANGED
|
@@ -36985,7 +36985,7 @@ const Wrapper$3 = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "producti
|
|
|
36985
36985
|
label: "Wrapper"
|
|
36986
36986
|
})("height:100%;", ({
|
|
36987
36987
|
rtl
|
|
36988
|
-
}) => rtl ? `border-left: 1px solid ${bcUi.Color.SECONDARY50};` : `border-right: 1px solid ${bcUi.Color.SECONDARY50};`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AAoB2B","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        var _c;\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            const result = yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            const sent = (_c = result.data) === null || _c === void 0 ? void 0 : _c.addMessage;\n            if (!sent)\n                return;\n            setMessages((prev) => [\n                ...prev,\n                Object.assign(Object.assign({}, sent), { createdAt: moment(sent.createdAt).toDate() }),\n            ]);\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36988
|
+
}) => rtl ? `border-left: 1px solid ${bcUi.Color.SECONDARY50};` : `border-right: 1px solid ${bcUi.Color.SECONDARY50};`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AAoB2B","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            // This code is not causing the message to appear twice, if everything is working as expected without it\n            // we can remove it in the future\n            // const sent = result.data?.addMessage;\n            // if (!sent) return;\n            // setMessages((prev) => [\n            //   ...prev,\n            //   {\n            //     ...sent,\n            //     createdAt: moment(sent.createdAt).toDate(),\n            //   },\n            // ]);\n            // #############################################################\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36989
36989
|
const ChatWrapper = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ? {
|
|
36990
36990
|
target: "e1cywtq1"
|
|
36991
36991
|
} : {
|
|
@@ -36993,7 +36993,7 @@ const ChatWrapper = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "produc
|
|
|
36993
36993
|
label: "ChatWrapper"
|
|
36994
36994
|
})("height:", ({
|
|
36995
36995
|
withHeader
|
|
36996
|
-
}) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)', ";direction:ltr;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AA0B+B","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        var _c;\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            const result = yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            const sent = (_c = result.data) === null || _c === void 0 ? void 0 : _c.addMessage;\n            if (!sent)\n                return;\n            setMessages((prev) => [\n                ...prev,\n                Object.assign(Object.assign({}, sent), { createdAt: moment(sent.createdAt).toDate() }),\n            ]);\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36996
|
+
}) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)', ";direction:ltr;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AA0B+B","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            // This code is not causing the message to appear twice, if everything is working as expected without it\n            // we can remove it in the future\n            // const sent = result.data?.addMessage;\n            // if (!sent) return;\n            // setMessages((prev) => [\n            //   ...prev,\n            //   {\n            //     ...sent,\n            //     createdAt: moment(sent.createdAt).toDate(),\n            //   },\n            // ]);\n            // #############################################################\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36997
36997
|
const OldMessagesCon = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ? {
|
|
36998
36998
|
target: "e1cywtq0"
|
|
36999
36999
|
} : {
|
|
@@ -37001,7 +37001,7 @@ const OldMessagesCon = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "pro
|
|
|
37001
37001
|
label: "OldMessagesCon"
|
|
37002
37002
|
})("position:absolute;top:50px;", ({
|
|
37003
37003
|
withHeader
|
|
37004
|
-
}) => withHeader ? `top: 50px;` : `top: 2px;`, ";left:calc(50% - 75px);z-index:1000;cursor:pointer;text-align:center;width:150px;a{font-size:12px;}.rt-Spinner{margin:0 auto;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AA8BkC","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        var _c;\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            const result = yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            const sent = (_c = result.data) === null || _c === void 0 ? void 0 : _c.addMessage;\n            if (!sent)\n                return;\n            setMessages((prev) => [\n                ...prev,\n                Object.assign(Object.assign({}, sent), { createdAt: moment(sent.createdAt).toDate() }),\n            ]);\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
37004
|
+
}) => withHeader ? `top: 50px;` : `top: 2px;`, ";left:calc(50% - 75px);z-index:1000;cursor:pointer;text-align:center;width:150px;a{font-size:12px;}.rt-Spinner{margin:0 auto;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AA8BkC","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            // This code is not causing the message to appear twice, if everything is working as expected without it\n            // we can remove it in the future\n            // const sent = result.data?.addMessage;\n            // if (!sent) return;\n            // setMessages((prev) => [\n            //   ...prev,\n            //   {\n            //     ...sent,\n            //     createdAt: moment(sent.createdAt).toDate(),\n            //   },\n            // ]);\n            // #############################################################\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
37005
37005
|
// --- Utility Functions ---
|
|
37006
37006
|
const hashStatuses = messages => {
|
|
37007
37007
|
return (messages === null || messages === void 0 ? void 0 : messages.map(m => m.status ? m.status[0] : '0').join(',')) || '';
|
|
@@ -37139,7 +37139,6 @@ const Chat = ({
|
|
|
37139
37139
|
}, [queryMessages, selectedConv, paginationInfo, workspaceId]);
|
|
37140
37140
|
// --- Send Message Logic ---
|
|
37141
37141
|
const sendChatMessage = React.useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {
|
|
37142
|
-
var _c;
|
|
37143
37142
|
if (!selectedConv) return;
|
|
37144
37143
|
const now = moment$2();
|
|
37145
37144
|
const lastMsgTime = moment$2(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);
|
|
@@ -37160,7 +37159,7 @@ const Chat = ({
|
|
|
37160
37159
|
yield autoAssign(selectedConv, myAgent, workspaceId);
|
|
37161
37160
|
}
|
|
37162
37161
|
// Send the actual message
|
|
37163
|
-
|
|
37162
|
+
yield addMessage({
|
|
37164
37163
|
variables: {
|
|
37165
37164
|
input: {
|
|
37166
37165
|
content,
|
|
@@ -37171,11 +37170,18 @@ const Chat = ({
|
|
|
37171
37170
|
}
|
|
37172
37171
|
}
|
|
37173
37172
|
});
|
|
37174
|
-
|
|
37175
|
-
|
|
37176
|
-
|
|
37177
|
-
|
|
37178
|
-
|
|
37173
|
+
// This code is not causing the message to appear twice, if everything is working as expected without it
|
|
37174
|
+
// we can remove it in the future
|
|
37175
|
+
// const sent = result.data?.addMessage;
|
|
37176
|
+
// if (!sent) return;
|
|
37177
|
+
// setMessages((prev) => [
|
|
37178
|
+
// ...prev,
|
|
37179
|
+
// {
|
|
37180
|
+
// ...sent,
|
|
37181
|
+
// createdAt: moment(sent.createdAt).toDate(),
|
|
37182
|
+
// },
|
|
37183
|
+
// ]);
|
|
37184
|
+
// #############################################################
|
|
37179
37185
|
} catch (ex) {
|
|
37180
37186
|
bcUi.triggerErrorToast({
|
|
37181
37187
|
title: t('requestFailed'),
|
package/index.esm.js
CHANGED
|
@@ -36967,7 +36967,7 @@ const Wrapper$3 = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "producti
|
|
|
36967
36967
|
label: "Wrapper"
|
|
36968
36968
|
})("height:100%;", ({
|
|
36969
36969
|
rtl
|
|
36970
|
-
}) => rtl ? `border-left: 1px solid ${Color.SECONDARY50};` : `border-right: 1px solid ${Color.SECONDARY50};`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AAoB2B","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        var _c;\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            const result = yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            const sent = (_c = result.data) === null || _c === void 0 ? void 0 : _c.addMessage;\n            if (!sent)\n                return;\n            setMessages((prev) => [\n                ...prev,\n                Object.assign(Object.assign({}, sent), { createdAt: moment(sent.createdAt).toDate() }),\n            ]);\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36970
|
+
}) => rtl ? `border-left: 1px solid ${Color.SECONDARY50};` : `border-right: 1px solid ${Color.SECONDARY50};`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AAoB2B","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            // This code is not causing the message to appear twice, if everything is working as expected without it\n            // we can remove it in the future\n            // const sent = result.data?.addMessage;\n            // if (!sent) return;\n            // setMessages((prev) => [\n            //   ...prev,\n            //   {\n            //     ...sent,\n            //     createdAt: moment(sent.createdAt).toDate(),\n            //   },\n            // ]);\n            // #############################################################\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36971
36971
|
const ChatWrapper = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ? {
|
|
36972
36972
|
target: "e1cywtq1"
|
|
36973
36973
|
} : {
|
|
@@ -36975,7 +36975,7 @@ const ChatWrapper = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "produc
|
|
|
36975
36975
|
label: "ChatWrapper"
|
|
36976
36976
|
})("height:", ({
|
|
36977
36977
|
withHeader
|
|
36978
|
-
}) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)', ";direction:ltr;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AA0B+B","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        var _c;\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            const result = yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            const sent = (_c = result.data) === null || _c === void 0 ? void 0 : _c.addMessage;\n            if (!sent)\n                return;\n            setMessages((prev) => [\n                ...prev,\n                Object.assign(Object.assign({}, sent), { createdAt: moment(sent.createdAt).toDate() }),\n            ]);\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36978
|
+
}) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)', ";direction:ltr;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AA0B+B","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            // This code is not causing the message to appear twice, if everything is working as expected without it\n            // we can remove it in the future\n            // const sent = result.data?.addMessage;\n            // if (!sent) return;\n            // setMessages((prev) => [\n            //   ...prev,\n            //   {\n            //     ...sent,\n            //     createdAt: moment(sent.createdAt).toDate(),\n            //   },\n            // ]);\n            // #############################################################\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36979
36979
|
const OldMessagesCon = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ? {
|
|
36980
36980
|
target: "e1cywtq0"
|
|
36981
36981
|
} : {
|
|
@@ -36983,7 +36983,7 @@ const OldMessagesCon = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "pro
|
|
|
36983
36983
|
label: "OldMessagesCon"
|
|
36984
36984
|
})("position:absolute;top:50px;", ({
|
|
36985
36985
|
withHeader
|
|
36986
|
-
}) => withHeader ? `top: 50px;` : `top: 2px;`, ";left:calc(50% - 75px);z-index:1000;cursor:pointer;text-align:center;width:150px;a{font-size:12px;}.rt-Spinner{margin:0 auto;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AA8BkC","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        var _c;\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            const result = yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            const sent = (_c = result.data) === null || _c === void 0 ? void 0 : _c.addMessage;\n            if (!sent)\n                return;\n            setMessages((prev) => [\n                ...prev,\n                Object.assign(Object.assign({}, sent), { createdAt: moment(sent.createdAt).toDate() }),\n            ]);\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36986
|
+
}) => withHeader ? `top: 50px;` : `top: 2px;`, ";left:calc(50% - 75px);z-index:1000;cursor:pointer;text-align:center;width:150px;a{font-size:12px;}.rt-Spinner{margin:0 auto;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AA8BkC","file":"index.tsx","sourcesContent":["import { __awaiter } from \"tslib\";\nimport { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from \"@emotion/react/jsx-runtime\";\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport moment from 'moment';\nimport styled from '@emotion/styled';\nimport { useTranslation } from 'react-i18next';\nimport { coreClient } from '@bcrumbs.net/bc-api';\nimport { auth, LangService } from '@bcrumbs.net/bc-shared';\nimport { BCChat } from '@bcrumbs.net/bc-chat';\nimport { Spinner } from '@radix-ui/themes';\nimport { Color, triggerErrorToast } from '@bcrumbs.net/bc-ui';\nimport { useAddMessageMutation, useMessagesLazyQuery, useUsersRolesQuery, useAssignConvMutation, } from '../../../../graphql.autogenerated';\nimport { playSendMessageSound } from '../../utils/sounds';\nimport useCanContribute from '../../hooks/useCanContribute';\nimport { Outside24HourWindowPopup } from '../Popups/Outside24HourWindowPopup';\nimport { getGravatarUrl } from '../../../auth/utils/avatar';\nimport ChatHeader from './ChatHeader';\nimport NoConvSelected from './NoConvSelected';\nimport useAutoAssign from '../../hooks/useAutoAssignment';\n// --- Styled components ---\nconst Wrapper = styled.div `\r\n  height: 100%;\r\n  ${({ rtl }) => rtl\n    ? `border-left: 1px solid ${Color.SECONDARY50};`\n    : `border-right: 1px solid ${Color.SECONDARY50};`};\r\n`;\nconst ChatWrapper = styled.div `\r\n  height: ${({ withHeader }) => withHeader ? 'calc(100% - 50px)' : 'calc(100% - 5px)'};\r\n  direction: ltr;\r\n`;\nconst OldMessagesCon = styled.div `\r\n  position: absolute;\r\n  top: 50px;\r\n  ${({ withHeader }) => (withHeader ? `top: 50px;` : `top: 2px;`)};\r\n  left: calc(50% - 75px);\r\n  z-index: 1000;\r\n  cursor: pointer;\r\n  text-align: center;\r\n  width: 150px;\r\n  a {\r\n    font-size: 12px;\r\n  }\r\n  .rt-Spinner {\r\n    margin: 0 auto;\r\n  }\r\n`;\n// --- Utility Functions ---\nconst hashStatuses = (messages) => {\n    return (messages === null || messages === void 0 ? void 0 : messages.map((m) => (m.status ? m.status[0] : '0')).join(',')) || '';\n};\nconst getLastClientMessage = (messages = []) => {\n    return [...messages]\n        .filter((m) => !m.isAgent)\n        .sort((a, b) => +new Date(b.createdAt) - +new Date(a.createdAt))[0];\n};\n// --- Main Chat Component ---\nconst Chat = ({ selectedConv, onConvEnd, agents, logo, rtl }) => {\n    var _a, _b;\n    const [t] = useTranslation('inbox');\n    const [messages, setMessages] = useState([]);\n    const [olderMessages, setOlderMessages] = useState();\n    const [outside24HourWindowPopup, setOutside24HourWindowPopup] = useState(false);\n    const [paginationInfo, setPaginationInfo] = useState();\n    const [agentsIdsAvatarsMap, setAgentsIdsAvatarsMap] = useState({});\n    const userId = (_a = auth.getUserInfo()) === null || _a === void 0 ? void 0 : _a.id;\n    const workspaceId = auth.getContext();\n    const { autoAssign } = useAutoAssign();\n    // --- Queries ---\n    const { data: userRolesData } = useUsersRolesQuery({\n        client: coreClient,\n        variables: { input: { workspaceId } },\n    });\n    const [queryMessages, { loading: msgsLoading }] = useMessagesLazyQuery({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [addMessage] = useAddMessageMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    const [assignConv] = useAssignConvMutation({\n        fetchPolicy: 'no-cache',\n        client: coreClient,\n    });\n    // --- Derived Data ---\n    const hasAdminOrOwnerRole = useMemo(() => {\n        var _a, _b;\n        return (_b = (_a = userRolesData === null || userRolesData === void 0 ? void 0 : userRolesData.usersRoles) === null || _a === void 0 ? void 0 : _a.nodes) === null || _b === void 0 ? void 0 : _b.some((user) => user.userId === userId &&\n            (user.role === 'ADMIN' || user.role === 'OWNER'));\n    }, [userRolesData, userId]);\n    const myAgent = useMemo(() => agents === null || agents === void 0 ? void 0 : agents.find((m) => m.userId === userId), [agents]);\n    const lastClientMessage = useMemo(() => getLastClientMessage((selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) || []), [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages]);\n    const canContribute = useCanContribute({\n        selectedConv,\n        hasAdminOrOwnerRole,\n        agents,\n    });\n    // --- Effects ---\n    useEffect(() => {\n        if (!agents)\n            return setAgentsIdsAvatarsMap({});\n        const fetchAvatars = () => __awaiter(void 0, void 0, void 0, function* () {\n            const acc = {};\n            yield Promise.all(agents.map((agent) => __awaiter(void 0, void 0, void 0, function* () {\n                if (agent.avatar) {\n                    acc[agent.id] = agent.avatar;\n                }\n                else {\n                    const avatarUrl = getGravatarUrl(agent.email);\n                    try {\n                        const response = yield fetch(avatarUrl);\n                        if (response.ok) {\n                            acc[agent.id] = avatarUrl;\n                        }\n                    }\n                    catch (_a) {\n                        // Do nothing if gravatar fails to load\n                    }\n                }\n            })));\n            setAgentsIdsAvatarsMap(acc);\n        });\n        fetchAvatars();\n    }, [agents]);\n    // Resetting the messages and pagination info when the selected conversation changes\n    useEffect(() => {\n        setOlderMessages(undefined);\n        setPaginationInfo(undefined);\n    }, [selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id]);\n    // Applying new incoming messages to the state\n    useEffect(() => {\n        var _a;\n        setMessages(((_a = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _a === void 0 ? void 0 : _a.map((m) => (Object.assign(Object.assign({}, m), { createdAt: moment(m.createdAt).toDate() })))) || []);\n    }, [\n        selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.id,\n        (_b = selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages) === null || _b === void 0 ? void 0 : _b.length,\n        hashStatuses(selectedConv === null || selectedConv === void 0 ? void 0 : selectedConv.messages),\n    ]);\n    // --- Load Older Messages ---\n    const loadOlderMessages = useCallback(() => {\n        if (!selectedConv || (paginationInfo && !paginationInfo.hasNextPage))\n            return;\n        queryMessages({\n            variables: {\n                input: {\n                    clientId: selectedConv.clientId,\n                    endDate: moment(selectedConv.createdAt)\n                        .subtract(1, 'minute')\n                        .toDate(),\n                    workspaceId,\n                },\n                pageInfo: {\n                    cursor: paginationInfo === null || paginationInfo === void 0 ? void 0 : paginationInfo.endCursor,\n                    limit: 50,\n                },\n            },\n        }).then(({ data }) => {\n            var _a;\n            if ((_a = data === null || data === void 0 ? void 0 : data.messages) === null || _a === void 0 ? void 0 : _a.nodes) {\n                const newMessages = data.messages.nodes.reverse();\n                const separator = {\n                    type: 'seperator',\n                    content: moment(selectedConv.createdAt).format('LLLL'),\n                };\n                setOlderMessages((prev) => prev ? [...newMessages, ...prev] : [...newMessages, separator]);\n                if (data === null || data === void 0 ? void 0 : data.messages.pageInfo) {\n                    setPaginationInfo({\n                        endCursor: data === null || data === void 0 ? void 0 : data.messages.pageInfo.endCursor,\n                        hasNextPage: data === null || data === void 0 ? void 0 : data.messages.pageInfo.hasNextPage,\n                    });\n                }\n            }\n        });\n    }, [queryMessages, selectedConv, paginationInfo, workspaceId]);\n    // --- Send Message Logic ---\n    const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {\n        if (!selectedConv)\n            return;\n        const now = moment();\n        const lastMsgTime = moment(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);\n        const isOutside24h = now.diff(lastMsgTime, 'hours') >= 24;\n        if (selectedConv.ended) {\n            return triggerErrorToast({\n                title: t('requestFailed'),\n                description: t('inbox.noMessagesForEndedConv'),\n            });\n        }\n        if (isOutside24h) {\n            return setOutside24HourWindowPopup(true);\n        }\n        playSendMessageSound();\n        try {\n            // Auto-assign if needed\n            if (!hasAdminOrOwnerRole && !selectedConv.assigneeId && myAgent) {\n                yield autoAssign(selectedConv, myAgent, workspaceId);\n            }\n            // Send the actual message\n            yield addMessage({\n                variables: {\n                    input: {\n                        content,\n                        convId: selectedConv.id,\n                        integrationId: selectedConv.integrationId,\n                        workspaceId: selectedConv.workspaceId,\n                        type,\n                    },\n                },\n            });\n            // This code is not causing the message to appear twice, if everything is working as expected without it\n            // we can remove it in the future\n            // const sent = result.data?.addMessage;\n            // if (!sent) return;\n            // setMessages((prev) => [\n            //   ...prev,\n            //   {\n            //     ...sent,\n            //     createdAt: moment(sent.createdAt).toDate(),\n            //   },\n            // ]);\n            // #############################################################\n        }\n        catch (ex) {\n            triggerErrorToast({\n                title: t('requestFailed'),\n                description: `${t('inbox.failedToSendMessage')}: ${ex.message}`,\n            });\n        }\n    }), [\n        selectedConv,\n        hasAdminOrOwnerRole,\n        myAgent,\n        addMessage,\n        autoAssign,\n        lastClientMessage,\n        t,\n        workspaceId,\n    ]);\n    // --- Render ---\n    return (_jsxs(Wrapper, { rtl: rtl, children: [selectedConv && messages ? (_jsxs(_Fragment, { children: [canContribute && onConvEnd && (_jsx(ChatHeader, { selectedConv: selectedConv, onConvEnd: onConvEnd })), _jsxs(ChatWrapper, { withHeader: canContribute, children: [msgsLoading ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(Spinner, { size: '3' }) })) : !olderMessages ? (_jsx(OldMessagesCon, { withHeader: canContribute && !!onConvEnd, children: _jsx(\"a\", { onClick: loadOlderMessages, children: t('inbox.showOlderMessages') }) })) : null, _jsx(BCChat, { config: {\n                                    hideHeader: true,\n                                    language: LangService.getLang().toUpperCase(),\n                                    backgroundColor: '#f9f9f9',\n                                    inputDisabled: !canContribute,\n                                    brandLogo: logo,\n                                }, messages: olderMessages ? [...olderMessages, ...messages] : messages, minimize: () => null, sendMessage: sendChatMessage, onReachingUp: loadOlderMessages, agentsIdsAvatarsMap: agentsIdsAvatarsMap })] })] })) : (_jsx(NoConvSelected, {})), outside24HourWindowPopup && selectedConv && (_jsx(Outside24HourWindowPopup, { onClose: () => setOutside24HourWindowPopup(false), selectedConv: selectedConv }))] }));\n};\nexport default Chat;\n//# sourceMappingURL=index.js.map"]} */"));
|
|
36987
36987
|
// --- Utility Functions ---
|
|
36988
36988
|
const hashStatuses = messages => {
|
|
36989
36989
|
return (messages === null || messages === void 0 ? void 0 : messages.map(m => m.status ? m.status[0] : '0').join(',')) || '';
|
|
@@ -37121,7 +37121,6 @@ const Chat = ({
|
|
|
37121
37121
|
}, [queryMessages, selectedConv, paginationInfo, workspaceId]);
|
|
37122
37122
|
// --- Send Message Logic ---
|
|
37123
37123
|
const sendChatMessage = useCallback((type, content) => __awaiter(void 0, void 0, void 0, function* () {
|
|
37124
|
-
var _c;
|
|
37125
37124
|
if (!selectedConv) return;
|
|
37126
37125
|
const now = moment$2();
|
|
37127
37126
|
const lastMsgTime = moment$2(lastClientMessage === null || lastClientMessage === void 0 ? void 0 : lastClientMessage.createdAt);
|
|
@@ -37142,7 +37141,7 @@ const Chat = ({
|
|
|
37142
37141
|
yield autoAssign(selectedConv, myAgent, workspaceId);
|
|
37143
37142
|
}
|
|
37144
37143
|
// Send the actual message
|
|
37145
|
-
|
|
37144
|
+
yield addMessage({
|
|
37146
37145
|
variables: {
|
|
37147
37146
|
input: {
|
|
37148
37147
|
content,
|
|
@@ -37153,11 +37152,18 @@ const Chat = ({
|
|
|
37153
37152
|
}
|
|
37154
37153
|
}
|
|
37155
37154
|
});
|
|
37156
|
-
|
|
37157
|
-
|
|
37158
|
-
|
|
37159
|
-
|
|
37160
|
-
|
|
37155
|
+
// This code is not causing the message to appear twice, if everything is working as expected without it
|
|
37156
|
+
// we can remove it in the future
|
|
37157
|
+
// const sent = result.data?.addMessage;
|
|
37158
|
+
// if (!sent) return;
|
|
37159
|
+
// setMessages((prev) => [
|
|
37160
|
+
// ...prev,
|
|
37161
|
+
// {
|
|
37162
|
+
// ...sent,
|
|
37163
|
+
// createdAt: moment(sent.createdAt).toDate(),
|
|
37164
|
+
// },
|
|
37165
|
+
// ]);
|
|
37166
|
+
// #############################################################
|
|
37161
37167
|
} catch (ex) {
|
|
37162
37168
|
triggerErrorToast({
|
|
37163
37169
|
title: t('requestFailed'),
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@bcrumbs.net/inbox",
|
|
3
3
|
"description": "Inbox widget for Bread Crumbs portals",
|
|
4
|
-
"version": "0.0.
|
|
4
|
+
"version": "0.0.25",
|
|
5
5
|
"keyword": [
|
|
6
6
|
"bcrumbs",
|
|
7
7
|
"bc-ui",
|
|
@@ -25,7 +25,7 @@
|
|
|
25
25
|
"dependencies": {
|
|
26
26
|
"@bcrumbs.net/bc-shared": "^0.0.5",
|
|
27
27
|
"@bcrumbs.net/bc-ui": "^0.0.5",
|
|
28
|
-
"@bcrumbs.net/bc-api": "^0.0.
|
|
28
|
+
"@bcrumbs.net/bc-api": "^0.0.42",
|
|
29
29
|
"@azure/storage-blob": "^12.25.0"
|
|
30
30
|
},
|
|
31
31
|
"module": "./index.esm.js",
|