@selfcommunity/react-ui
Version: 
React UI Components to integrate a Community created with SelfCommunity Platform.
243 lines (234 loc) • 12.5 kB
JavaScript
import { __rest } from "tslib";
import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
import { useContext, useEffect, useMemo, useRef, useState } from 'react';
import { styled } from '@mui/material/styles';
import { Button, Card, CardContent, Icon, IconButton, List, TextField, useTheme } from '@mui/material';
import PubSub from 'pubsub-js';
import { SCNotificationTopicType, SCNotificationTypologyType, SCPrivateMessageStatusType, SCPrivateMessageType } from '@selfcommunity/types';
import PrivateMessageSnippetsSkeleton from './Skeleton';
import PrivateMessageSnippetItem from '../PrivateMessageSnippetItem';
import classNames from 'classnames';
import { useThemeProps } from '@mui/system';
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
import { SCUserContext, useSCFetchPrivateMessageSnippets } from '@selfcommunity/react-core';
import { CacheStrategies } from '@selfcommunity/utils';
import PrivateMessageSettingsIconButton from '../PrivateMessageSettingsIconButton';
import useMediaQuery from '@mui/material/useMediaQuery';
import { PREFIX } from './constants';
const messages = defineMessages({
    placeholder: {
        id: 'ui.privateMessage.snippets.searchBar.placeholder',
        defaultMessage: 'ui.privateMessage.snippets.searchBar.placeholder'
    }
});
const classes = {
    root: `${PREFIX}-root`,
    searchBar: `${PREFIX}-search-bar`,
    icon: `${PREFIX}-icon`,
    input: `${PREFIX}-input`,
    clear: `${PREFIX}-clear`,
    newMessageButton: `${PREFIX}-new-message-button`
};
const Root = styled(Card, {
    name: PREFIX,
    slot: 'Root'
})(() => ({}));
/**
 * > API documentation for the Community-JS PrivateMessageSnippets component. Learn about the available props and the CSS API.
 *
 *
 * This component renders the list of conversations preview between users.
 * Take a look at our <strong>demo</strong> component [here](/docs/sdk/community-js/react-ui/Components/Snippets)
 #### Import
 ```jsx
 import {PrivateMessageSnippets} from '@selfcommunity/react-ui';
 ```
 #### Component Name
 The name `SCPrivateMessageSnippets` can be used when providing style overrides in the theme.
 #### CSS
 |Rule Name|Global class|Description|
 |---|---|---|
 |root|.SCPrivateMessageSnippets-root|Styles applied to the root element.|
 |icon|.SCPrivateMessageSnippets-icon|Styles applied to the search icon element.|
 |input|.SCPrivateMessageSnippets-input|Styles applied to the search input element.|
 |clear|.SCPrivateMessageSnippets-clear|Styles applied to the search bar clear icon element.|
 |searchBar|.SCPrivateMessageSnippets-searchBar|Styles applied to the search bar element.|
 |newMessageButton|.SCPrivateMessageSnippets-new-message-button|Styles applied to new message button element.|
 * @param inProps
 */
export default function PrivateMessageSnippets(inProps) {
    // PROPS
    const props = useThemeProps({
        props: inProps,
        name: PREFIX
    });
    const { className = null, threadObj = null, type = null, snippetActions, clearSearch } = props, rest = __rest(props, ["className", "threadObj", "type", "snippetActions", "clearSearch"]);
    // STATE
    const theme = useTheme();
    const isMobile = useMediaQuery(theme.breakpoints.down('md'));
    const { data, updateSnippets } = useSCFetchPrivateMessageSnippets({ cacheStrategy: CacheStrategies.CACHE_FIRST });
    const [search, setSearch] = useState('');
    const isObj = typeof threadObj === 'object';
    const scUserContext = useContext(SCUserContext);
    const authUserId = scUserContext.user ? scUserContext.user.id : null;
    const [_type, _setType] = useState(type);
    // INTL
    const intl = useIntl();
    // REFS
    const refreshSubscription = useRef(null);
    // CONST
    const filteredSnippets = data.snippets.filter((el) => {
        var _a;
        if (search === '') {
            return el;
        }
        else if (el.group) {
            return el.group.slug.includes(search.toLowerCase());
        }
        else if (((_a = el === null || el === void 0 ? void 0 : el.receiver) === null || _a === void 0 ? void 0 : _a.id) === authUserId) {
            return el.sender.username.includes(search.toLowerCase());
        }
        return el.receiver.username.includes(search.toLowerCase());
    });
    const messageReceiver = (item, loggedUserId, obj) => {
        var _a, _b, _c, _d;
        if (obj) {
            return ((_a = item === null || item === void 0 ? void 0 : item.receiver) === null || _a === void 0 ? void 0 : _a.id) !== loggedUserId ? item === null || item === void 0 ? void 0 : item.receiver : item === null || item === void 0 ? void 0 : item.sender;
        }
        return ((_b = item === null || item === void 0 ? void 0 : item.receiver) === null || _b === void 0 ? void 0 : _b.id) !== loggedUserId ? (_c = item === null || item === void 0 ? void 0 : item.receiver) === null || _c === void 0 ? void 0 : _c.id : (_d = item === null || item === void 0 ? void 0 : item.sender) === null || _d === void 0 ? void 0 : _d.id;
    };
    const isSelected = useMemo(() => {
        return (message) => {
            var _a, _b;
            if (threadObj && _type === SCPrivateMessageType.GROUP) {
                return ((_a = message === null || message === void 0 ? void 0 : message.group) === null || _a === void 0 ? void 0 : _a.id) === (isObj ? (_b = threadObj === null || threadObj === void 0 ? void 0 : threadObj.group) === null || _b === void 0 ? void 0 : _b.id : threadObj);
            }
            else if (threadObj && threadObj !== SCPrivateMessageType.NEW) {
                return messageReceiver(message, authUserId) === (isObj ? messageReceiver(threadObj, authUserId) : threadObj);
            }
            return null;
        };
    }, [threadObj, authUserId, _type]);
    //HANDLERS
    const handleChange = (event) => {
        setSearch(event.target.value);
    };
    const handleClear = () => {
        setSearch('');
    };
    const handleOpenNewMessage = () => {
        snippetActions && snippetActions.onNewMessageClick();
        handleClear();
    };
    const handleDeleteConversation = (msg) => {
        snippetActions && snippetActions.onDeleteConfirm(msg);
    };
    function handleOpenThread(msg) {
        const threadType = msg.group !== null ? SCPrivateMessageType.GROUP : SCPrivateMessageType.USER;
        _setType(threadType);
        snippetActions && snippetActions.onSnippetClick(msg, threadType);
        handleClear();
        updateSnippetsParams(msg.id, 'seen');
    }
    /**
     * Updates snippet headline and status or just snippet status
     * @param threadId
     * @param status
     * @param headline
     */
    function updateSnippetsParams(threadId, status, headline) {
        const newSnippets = [...data.snippets];
        const index = newSnippets.findIndex((s) => s.id === threadId);
        if (index !== -1) {
            newSnippets[index].headline = headline !== null && headline !== void 0 ? headline : newSnippets[index].headline;
            // eslint-disable-next-line @typescript-eslint/ban-ts-ignore
            // @ts-ignore
            newSnippets[index].thread_status = status;
            updateSnippets(newSnippets);
        }
    }
    function handleSnippetsUpdate(message, forDeletion, type) {
        const newSnippets = [...data.snippets];
        if (forDeletion && type === SCPrivateMessageType.USER) {
            const _snippets = newSnippets.filter((s) => messageReceiver(s, authUserId) !== message);
            updateSnippets(_snippets);
        }
        else if (forDeletion && SCPrivateMessageType.GROUP) {
            const _snippets = newSnippets.filter((s) => { var _a; return ((_a = s === null || s === void 0 ? void 0 : s.group) === null || _a === void 0 ? void 0 : _a.id) !== message; });
            updateSnippets(_snippets);
        }
        else {
            let temp = [...data.snippets];
            message.map((m) => {
                const idx = newSnippets.findIndex((s) => 
                // eslint-disable-next-line @typescript-eslint/ban-ts-ignore
                // @ts-ignore
                Object.prototype.hasOwnProperty.call(s, 'thread_id') ? s.thread_id === m.thread_id : s.id === m.thread_id);
                if (idx !== -1) {
                    temp[idx].headline = m.message;
                    temp[idx].thread_status = m.status === SCPrivateMessageStatusType.NEW ? m.status : m.thread_status;
                    const element = temp.splice(idx, 1)[0];
                    temp.unshift(element);
                }
                else {
                    temp.unshift(m);
                }
                updateSnippets(temp);
            });
        }
    }
    /**
     * Notification subscriber
     */
    const subscriber = (msg, data) => {
        const res = data.data;
        updateSnippetsParams(res.thread_id, res.notification_obj.snippet.thread_status, res.notification_obj.snippet.headline);
    };
    /**
     * When a ws notification arrives, updates data
     */
    useEffect(() => {
        refreshSubscription.current = PubSub.subscribe(`${SCNotificationTopicType.INTERACTION}.${SCNotificationTypologyType.PRIVATE_MESSAGE}`, subscriber);
        return () => {
            PubSub.unsubscribe(refreshSubscription.current);
        };
    }, [data.snippets]);
    useEffect(() => {
        if (clearSearch)
            handleClear();
    }, [clearSearch]);
    /**
     * Thread/ Private Message Component subscriptions handlers
     */
    useEffect(() => {
        const threadSubscriber = PubSub.subscribe('snippetsChannel', (msg, data) => {
            handleSnippetsUpdate(data);
        });
        const snippetsSubscriber = PubSub.subscribe('snippetsChannelDelete', (msg, data) => {
            handleSnippetsUpdate(data, true, SCPrivateMessageType.USER);
        });
        const snippetsGroupSubscriber = PubSub.subscribe('snippetsChannelDeleteGroup', (msg, data) => {
            handleSnippetsUpdate(data, true, SCPrivateMessageType.GROUP);
        });
        return () => {
            PubSub.unsubscribe(threadSubscriber);
            PubSub.unsubscribe(snippetsSubscriber);
            PubSub.unsubscribe(snippetsGroupSubscriber);
        };
    }, [data.snippets]);
    //RENDERING
    /**
     * Renders snippets skeleton when loading
     */
    if (data.isLoading) {
        return _jsx(PrivateMessageSnippetsSkeleton, { elevation: 0 });
    }
    /**
     * Renders the component
     */
    return (_jsx(Root, Object.assign({}, rest, { className: classNames(classes.root, className) }, { children: _jsxs(CardContent, { children: [_jsx(Button, Object.assign({ variant: "outlined", size: "medium", className: classes.newMessageButton, onClick: handleOpenNewMessage }, { children: _jsx(FormattedMessage, { id: "ui.privateMessage.snippets.button.newMessage", defaultMessage: "ui.privateMessage.snippets.button.newMessage" }) })), data.snippets.length !== 0 && (_jsxs(_Fragment, { children: [_jsx(TextField, { className: classes.searchBar, variant: "outlined", margin: "normal", fullWidth: true, id: `${PREFIX}-search`, placeholder: `${intl.formatMessage(messages.placeholder)}`, size: "small", onChange: handleChange, value: search, InputProps: {
                                className: classes.input,
                                startAdornment: _jsx(Icon, Object.assign({ className: classes.icon }, { children: "search" })),
                                endAdornment: (_jsx(IconButton, Object.assign({ className: classes.clear, disabled: !search, onClick: handleClear, size: "small" }, { children: _jsx(Icon, { children: "close" }) })))
                            } }), _jsx(List, { children: filteredSnippets.map((message) => (_jsx(PrivateMessageSnippetItem, { message: message, onItemClick: () => handleOpenThread(message), secondaryAction: _jsxs(_Fragment, { children: [message.thread_status === SCPrivateMessageStatusType.NEW && _jsx(Icon, Object.assign({ color: "secondary" }, { children: "fiber_manual_record" })), !isMobile && (_jsx(PrivateMessageSettingsIconButton, { threadToDelete: (message === null || message === void 0 ? void 0 : message.group) ? message.group.id : messageReceiver(message, authUserId), onItemDeleteConfirm: () => handleDeleteConversation(messageReceiver(message, authUserId)), user: messageReceiver(message, authUserId, true), group: message === null || message === void 0 ? void 0 : message.group }))] }), selected: isSelected(message) }, message.id))) })] }))] }) })));
}