UNPKG

matrix-react-sdk

Version:
308 lines (257 loc) 39.2 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends")); var _react = _interopRequireWildcard(require("react")); var _propTypes = _interopRequireDefault(require("prop-types")); var _event = require("matrix-js-sdk/src/models/event"); var _languageHandler = require("../../../languageHandler"); var sdk = _interopRequireWildcard(require("../../../index")); var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher")); var _ContextMenu = require("../../structures/ContextMenu"); var _EventUtils = require("../../../utils/EventUtils"); var _RoomContext = _interopRequireDefault(require("../../../contexts/RoomContext")); var _Toolbar = _interopRequireDefault(require("../../../accessibility/Toolbar")); var _RovingTabIndex = require("../../../accessibility/RovingTabIndex"); var _replaceableComponent = require("../../../utils/replaceableComponent"); var _MessageContextMenu = require("../context_menus/MessageContextMenu"); var _Resend = _interopRequireDefault(require("../../../Resend")); var _MatrixClientPeg = require("../../../MatrixClientPeg"); var _dec, _class, _class2, _temp; const OptionsButton = ({ mxEvent, getTile, getReplyThread, permalinkCreator, onFocusChange }) => { const [menuDisplayed, button, openMenu, closeMenu] = (0, _ContextMenu.useContextMenu)(); const [onFocus, isActive, ref] = (0, _RovingTabIndex.useRovingTabIndex)(button); (0, _react.useEffect)(() => { onFocusChange(menuDisplayed); }, [onFocusChange, menuDisplayed]); let contextMenu; if (menuDisplayed) { const MessageContextMenu = sdk.getComponent('context_menus.MessageContextMenu'); const tile = getTile && getTile(); const replyThread = getReplyThread && getReplyThread(); const buttonRect = button.current.getBoundingClientRect(); contextMenu = /*#__PURE__*/_react.default.createElement(_ContextMenu.ContextMenu, (0, _extends2.default)({}, (0, _ContextMenu.aboveLeftOf)(buttonRect), { onFinished: closeMenu }), /*#__PURE__*/_react.default.createElement(MessageContextMenu, { mxEvent: mxEvent, permalinkCreator: permalinkCreator, eventTileOps: tile && tile.getEventTileOps ? tile.getEventTileOps() : undefined, collapseReplyThread: replyThread && replyThread.canCollapse() ? replyThread.collapse : undefined, onFinished: closeMenu })); } return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_ContextMenu.ContextMenuTooltipButton, { className: "mx_MessageActionBar_maskButton mx_MessageActionBar_optionsButton", title: (0, _languageHandler._t)("Options"), onClick: openMenu, isExpanded: menuDisplayed, inputRef: ref, onFocus: onFocus, tabIndex: isActive ? 0 : -1 }), contextMenu); }; const ReactButton = ({ mxEvent, reactions, onFocusChange }) => { const [menuDisplayed, button, openMenu, closeMenu] = (0, _ContextMenu.useContextMenu)(); const [onFocus, isActive, ref] = (0, _RovingTabIndex.useRovingTabIndex)(button); (0, _react.useEffect)(() => { onFocusChange(menuDisplayed); }, [onFocusChange, menuDisplayed]); let contextMenu; if (menuDisplayed) { const buttonRect = button.current.getBoundingClientRect(); const ReactionPicker = sdk.getComponent('emojipicker.ReactionPicker'); contextMenu = /*#__PURE__*/_react.default.createElement(_ContextMenu.ContextMenu, (0, _extends2.default)({}, (0, _ContextMenu.aboveLeftOf)(buttonRect), { onFinished: closeMenu, managed: false }), /*#__PURE__*/_react.default.createElement(ReactionPicker, { mxEvent: mxEvent, reactions: reactions, onFinished: closeMenu })); } return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_ContextMenu.ContextMenuTooltipButton, { className: "mx_MessageActionBar_maskButton mx_MessageActionBar_reactButton", title: (0, _languageHandler._t)("React"), onClick: openMenu, isExpanded: menuDisplayed, inputRef: ref, onFocus: onFocus, tabIndex: isActive ? 0 : -1 }), contextMenu); }; let MessageActionBar = (_dec = (0, _replaceableComponent.replaceableComponent)("views.messages.MessageActionBar"), _dec(_class = (_temp = _class2 = class MessageActionBar extends _react.default.PureComponent { constructor(...args) { super(...args); (0, _defineProperty2.default)(this, "onDecrypted", () => { // When an event decrypts, it is likely to change the set of available // actions, so we force an update to check again. this.forceUpdate(); }); (0, _defineProperty2.default)(this, "onBeforeRedaction", () => { // When an event is redacted, we can't edit it so update the available actions. this.forceUpdate(); }); (0, _defineProperty2.default)(this, "onSent", () => { // When an event is sent and echoed the possible actions change. this.forceUpdate(); }); (0, _defineProperty2.default)(this, "onFocusChange", focused => { if (!this.props.onFocusChange) { return; } this.props.onFocusChange(focused); }); (0, _defineProperty2.default)(this, "onReplyClick", ev => { _dispatcher.default.dispatch({ action: 'reply_to_event', event: this.props.mxEvent }); }); (0, _defineProperty2.default)(this, "onEditClick", ev => { _dispatcher.default.dispatch({ action: 'edit_event', event: this.props.mxEvent }); }); (0, _defineProperty2.default)(this, "onResendClick", ev => { this.runActionOnFailedEv(tarEv => _Resend.default.resend(tarEv)); }); (0, _defineProperty2.default)(this, "onCancelClick", ev => { this.runActionOnFailedEv(tarEv => _Resend.default.removeFromQueue(tarEv), testEv => (0, _MessageContextMenu.canCancel)(testEv.status)); }); } componentDidMount() { if (this.props.mxEvent.status && this.props.mxEvent.status !== _event.EventStatus.SENT) { this.props.mxEvent.on("Event.status", this.onSent); } const client = _MatrixClientPeg.MatrixClientPeg.get(); client.decryptEventIfNeeded(this.props.mxEvent); if (this.props.mxEvent.isBeingDecrypted()) { this.props.mxEvent.once("Event.decrypted", this.onDecrypted); } this.props.mxEvent.on("Event.beforeRedaction", this.onBeforeRedaction); } componentWillUnmount() { this.props.mxEvent.off("Event.status", this.onSent); this.props.mxEvent.off("Event.decrypted", this.onDecrypted); this.props.mxEvent.off("Event.beforeRedaction", this.onBeforeRedaction); } /** * Runs a given fn on the set of possible events to test. The first event * that passes the checkFn will have fn executed on it. Both functions take * a MatrixEvent object. If no particular conditions are needed, checkFn can * be null/undefined. If no functions pass the checkFn, no action will be * taken. * @param {Function} fn The execution function. * @param {Function} checkFn The test function. */ runActionOnFailedEv(fn, checkFn) { if (!checkFn) checkFn = () => true; const mxEvent = this.props.mxEvent; const editEvent = mxEvent.replacingEvent(); const redactEvent = mxEvent.localRedactionEvent(); const tryOrder = [redactEvent, editEvent, mxEvent]; for (const ev of tryOrder) { if (ev && checkFn(ev)) { fn(ev); break; } } } render() { const toolbarOpts = []; if ((0, _EventUtils.canEditContent)(this.props.mxEvent)) { toolbarOpts.push( /*#__PURE__*/_react.default.createElement(_RovingTabIndex.RovingAccessibleTooltipButton, { className: "mx_MessageActionBar_maskButton mx_MessageActionBar_editButton", title: (0, _languageHandler._t)("Edit"), onClick: this.onEditClick, key: "edit" })); } const cancelSendingButton = /*#__PURE__*/_react.default.createElement(_RovingTabIndex.RovingAccessibleTooltipButton, { className: "mx_MessageActionBar_maskButton mx_MessageActionBar_cancelButton", title: (0, _languageHandler._t)("Delete"), onClick: this.onCancelClick, key: "cancel" }); // We show a different toolbar for failed events, so detect that first. const mxEvent = this.props.mxEvent; const editStatus = mxEvent.replacingEvent() && mxEvent.replacingEvent().status; const redactStatus = mxEvent.localRedactionEvent() && mxEvent.localRedactionEvent().status; const allowCancel = (0, _MessageContextMenu.canCancel)(mxEvent.status) || (0, _MessageContextMenu.canCancel)(editStatus) || (0, _MessageContextMenu.canCancel)(redactStatus); const isFailed = [mxEvent.status, editStatus, redactStatus].includes("not_sent"); if (allowCancel && isFailed) { // The resend button needs to appear ahead of the edit button, so insert to the // start of the opts toolbarOpts.splice(0, 0, /*#__PURE__*/_react.default.createElement(_RovingTabIndex.RovingAccessibleTooltipButton, { className: "mx_MessageActionBar_maskButton mx_MessageActionBar_resendButton", title: (0, _languageHandler._t)("Retry"), onClick: this.onResendClick, key: "resend" })); // The delete button should appear last, so we can just drop it at the end toolbarOpts.push(cancelSendingButton); } else { if ((0, _EventUtils.isContentActionable)(this.props.mxEvent)) { // Like the resend button, the react and reply buttons need to appear before the edit. // The only catch is we do the reply button first so that we can make sure the react // button is the very first button without having to do length checks for `splice()`. if (this.context.canReply) { toolbarOpts.splice(0, 0, /*#__PURE__*/_react.default.createElement(_RovingTabIndex.RovingAccessibleTooltipButton, { className: "mx_MessageActionBar_maskButton mx_MessageActionBar_replyButton", title: (0, _languageHandler._t)("Reply"), onClick: this.onReplyClick, key: "reply" })); } if (this.context.canReact) { toolbarOpts.splice(0, 0, /*#__PURE__*/_react.default.createElement(ReactButton, { mxEvent: this.props.mxEvent, reactions: this.props.reactions, onFocusChange: this.onFocusChange, key: "react" })); } } if (allowCancel) { toolbarOpts.push(cancelSendingButton); } // The menu button should be last, so dump it there. toolbarOpts.push( /*#__PURE__*/_react.default.createElement(OptionsButton, { mxEvent: this.props.mxEvent, getReplyThread: this.props.getReplyThread, getTile: this.props.getTile, permalinkCreator: this.props.permalinkCreator, onFocusChange: this.onFocusChange, key: "menu" })); } // aria-live=off to not have this read out automatically as navigating around timeline, gets repetitive. return /*#__PURE__*/_react.default.createElement(_Toolbar.default, { className: "mx_MessageActionBar", "aria-label": (0, _languageHandler._t)("Message Actions"), "aria-live": "off" }, toolbarOpts); } }, (0, _defineProperty2.default)(_class2, "propTypes", { mxEvent: _propTypes.default.object.isRequired, // The Relations model from the JS SDK for reactions to `mxEvent` reactions: _propTypes.default.object, permalinkCreator: _propTypes.default.object, getTile: _propTypes.default.func, getReplyThread: _propTypes.default.func, onFocusChange: _propTypes.default.func }), (0, _defineProperty2.default)(_class2, "contextType", _RoomContext.default), _temp)) || _class); exports.default = MessageActionBar; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/messages/MessageActionBar.js"],"names":["OptionsButton","mxEvent","getTile","getReplyThread","permalinkCreator","onFocusChange","menuDisplayed","button","openMenu","closeMenu","onFocus","isActive","ref","contextMenu","MessageContextMenu","sdk","getComponent","tile","replyThread","buttonRect","current","getBoundingClientRect","getEventTileOps","undefined","canCollapse","collapse","ReactButton","reactions","ReactionPicker","MessageActionBar","React","PureComponent","forceUpdate","focused","props","ev","dis","dispatch","action","event","runActionOnFailedEv","tarEv","Resend","resend","removeFromQueue","testEv","status","componentDidMount","EventStatus","SENT","on","onSent","client","MatrixClientPeg","get","decryptEventIfNeeded","isBeingDecrypted","once","onDecrypted","onBeforeRedaction","componentWillUnmount","off","fn","checkFn","editEvent","replacingEvent","redactEvent","localRedactionEvent","tryOrder","render","toolbarOpts","push","onEditClick","cancelSendingButton","onCancelClick","editStatus","redactStatus","allowCancel","isFailed","includes","splice","onResendClick","context","canReply","onReplyClick","canReact","PropTypes","object","isRequired","func","RoomContext"],"mappings":"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","sourcesContent":["/*\nCopyright 2019 New Vector Ltd\nCopyright 2019 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2019, 2020 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport React, {useEffect} from 'react';\nimport PropTypes from 'prop-types';\nimport { EventStatus } from 'matrix-js-sdk/src/models/event';\n\nimport { _t } from '../../../languageHandler';\nimport * as sdk from '../../../index';\nimport dis from '../../../dispatcher/dispatcher';\nimport {aboveLeftOf, ContextMenu, ContextMenuTooltipButton, useContextMenu} from '../../structures/ContextMenu';\nimport { isContentActionable, canEditContent } from '../../../utils/EventUtils';\nimport RoomContext from \"../../../contexts/RoomContext\";\nimport Toolbar from \"../../../accessibility/Toolbar\";\nimport {RovingAccessibleTooltipButton, useRovingTabIndex} from \"../../../accessibility/RovingTabIndex\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\nimport {canCancel} from \"../context_menus/MessageContextMenu\";\nimport Resend from \"../../../Resend\";\nimport { MatrixClientPeg } from \"../../../MatrixClientPeg\";\n\nconst OptionsButton = ({mxEvent, getTile, getReplyThread, permalinkCreator, onFocusChange}) => {\n    const [menuDisplayed, button, openMenu, closeMenu] = useContextMenu();\n    const [onFocus, isActive, ref] = useRovingTabIndex(button);\n    useEffect(() => {\n        onFocusChange(menuDisplayed);\n    }, [onFocusChange, menuDisplayed]);\n\n    let contextMenu;\n    if (menuDisplayed) {\n        const MessageContextMenu = sdk.getComponent('context_menus.MessageContextMenu');\n\n        const tile = getTile && getTile();\n        const replyThread = getReplyThread && getReplyThread();\n\n        const buttonRect = button.current.getBoundingClientRect();\n        contextMenu = <ContextMenu {...aboveLeftOf(buttonRect)} onFinished={closeMenu}>\n            <MessageContextMenu\n                mxEvent={mxEvent}\n                permalinkCreator={permalinkCreator}\n                eventTileOps={tile && tile.getEventTileOps ? tile.getEventTileOps() : undefined}\n                collapseReplyThread={replyThread && replyThread.canCollapse() ? replyThread.collapse : undefined}\n                onFinished={closeMenu}\n            />\n        </ContextMenu>;\n    }\n\n    return <React.Fragment>\n        <ContextMenuTooltipButton\n            className=\"mx_MessageActionBar_maskButton mx_MessageActionBar_optionsButton\"\n            title={_t(\"Options\")}\n            onClick={openMenu}\n            isExpanded={menuDisplayed}\n            inputRef={ref}\n            onFocus={onFocus}\n            tabIndex={isActive ? 0 : -1}\n        />\n\n        { contextMenu }\n    </React.Fragment>;\n};\n\nconst ReactButton = ({mxEvent, reactions, onFocusChange}) => {\n    const [menuDisplayed, button, openMenu, closeMenu] = useContextMenu();\n    const [onFocus, isActive, ref] = useRovingTabIndex(button);\n    useEffect(() => {\n        onFocusChange(menuDisplayed);\n    }, [onFocusChange, menuDisplayed]);\n\n    let contextMenu;\n    if (menuDisplayed) {\n        const buttonRect = button.current.getBoundingClientRect();\n        const ReactionPicker = sdk.getComponent('emojipicker.ReactionPicker');\n        contextMenu = <ContextMenu {...aboveLeftOf(buttonRect)} onFinished={closeMenu} managed={false}>\n            <ReactionPicker mxEvent={mxEvent} reactions={reactions} onFinished={closeMenu} />\n        </ContextMenu>;\n    }\n\n    return <React.Fragment>\n        <ContextMenuTooltipButton\n            className=\"mx_MessageActionBar_maskButton mx_MessageActionBar_reactButton\"\n            title={_t(\"React\")}\n            onClick={openMenu}\n            isExpanded={menuDisplayed}\n            inputRef={ref}\n            onFocus={onFocus}\n            tabIndex={isActive ? 0 : -1}\n        />\n\n        { contextMenu }\n    </React.Fragment>;\n};\n\n@replaceableComponent(\"views.messages.MessageActionBar\")\nexport default class MessageActionBar extends React.PureComponent {\n    static propTypes = {\n        mxEvent: PropTypes.object.isRequired,\n        // The Relations model from the JS SDK for reactions to `mxEvent`\n        reactions: PropTypes.object,\n        permalinkCreator: PropTypes.object,\n        getTile: PropTypes.func,\n        getReplyThread: PropTypes.func,\n        onFocusChange: PropTypes.func,\n    };\n\n    static contextType = RoomContext;\n\n    componentDidMount() {\n        if (this.props.mxEvent.status && this.props.mxEvent.status !== EventStatus.SENT) {\n            this.props.mxEvent.on(\"Event.status\", this.onSent);\n        }\n\n        const client = MatrixClientPeg.get();\n        client.decryptEventIfNeeded(this.props.mxEvent);\n\n        if (this.props.mxEvent.isBeingDecrypted()) {\n            this.props.mxEvent.once(\"Event.decrypted\", this.onDecrypted);\n        }\n        this.props.mxEvent.on(\"Event.beforeRedaction\", this.onBeforeRedaction);\n    }\n\n    componentWillUnmount() {\n        this.props.mxEvent.off(\"Event.status\", this.onSent);\n        this.props.mxEvent.off(\"Event.decrypted\", this.onDecrypted);\n        this.props.mxEvent.off(\"Event.beforeRedaction\", this.onBeforeRedaction);\n    }\n\n    onDecrypted = () => {\n        // When an event decrypts, it is likely to change the set of available\n        // actions, so we force an update to check again.\n        this.forceUpdate();\n    };\n\n    onBeforeRedaction = () => {\n        // When an event is redacted, we can't edit it so update the available actions.\n        this.forceUpdate();\n    };\n\n    onSent = () => {\n        // When an event is sent and echoed the possible actions change.\n        this.forceUpdate();\n    };\n\n    onFocusChange = (focused) => {\n        if (!this.props.onFocusChange) {\n            return;\n        }\n        this.props.onFocusChange(focused);\n    };\n\n    onReplyClick = (ev) => {\n        dis.dispatch({\n            action: 'reply_to_event',\n            event: this.props.mxEvent,\n        });\n    };\n\n    onEditClick = (ev) => {\n        dis.dispatch({\n            action: 'edit_event',\n            event: this.props.mxEvent,\n        });\n    };\n\n    /**\n     * Runs a given fn on the set of possible events to test. The first event\n     * that passes the checkFn will have fn executed on it. Both functions take\n     * a MatrixEvent object. If no particular conditions are needed, checkFn can\n     * be null/undefined. If no functions pass the checkFn, no action will be\n     * taken.\n     * @param {Function} fn The execution function.\n     * @param {Function} checkFn The test function.\n     */\n    runActionOnFailedEv(fn, checkFn) {\n        if (!checkFn) checkFn = () => true;\n\n        const mxEvent = this.props.mxEvent;\n        const editEvent = mxEvent.replacingEvent();\n        const redactEvent = mxEvent.localRedactionEvent();\n        const tryOrder = [redactEvent, editEvent, mxEvent];\n        for (const ev of tryOrder) {\n            if (ev && checkFn(ev)) {\n                fn(ev);\n                break;\n            }\n        }\n    }\n\n    onResendClick = (ev) => {\n        this.runActionOnFailedEv((tarEv) => Resend.resend(tarEv));\n    };\n\n    onCancelClick = (ev) => {\n        this.runActionOnFailedEv(\n            (tarEv) => Resend.removeFromQueue(tarEv),\n            (testEv) => canCancel(testEv.status),\n        );\n    };\n\n    render() {\n        const toolbarOpts = [];\n        if (canEditContent(this.props.mxEvent)) {\n            toolbarOpts.push(<RovingAccessibleTooltipButton\n                className=\"mx_MessageActionBar_maskButton mx_MessageActionBar_editButton\"\n                title={_t(\"Edit\")}\n                onClick={this.onEditClick}\n                key=\"edit\"\n            />);\n        }\n\n        const cancelSendingButton = <RovingAccessibleTooltipButton\n            className=\"mx_MessageActionBar_maskButton mx_MessageActionBar_cancelButton\"\n            title={_t(\"Delete\")}\n            onClick={this.onCancelClick}\n            key=\"cancel\"\n        />;\n\n        // We show a different toolbar for failed events, so detect that first.\n        const mxEvent = this.props.mxEvent;\n        const editStatus = mxEvent.replacingEvent() && mxEvent.replacingEvent().status;\n        const redactStatus = mxEvent.localRedactionEvent() && mxEvent.localRedactionEvent().status;\n        const allowCancel = canCancel(mxEvent.status) || canCancel(editStatus) || canCancel(redactStatus);\n        const isFailed = [mxEvent.status, editStatus, redactStatus].includes(\"not_sent\");\n        if (allowCancel && isFailed) {\n            // The resend button needs to appear ahead of the edit button, so insert to the\n            // start of the opts\n            toolbarOpts.splice(0, 0, <RovingAccessibleTooltipButton\n                className=\"mx_MessageActionBar_maskButton mx_MessageActionBar_resendButton\"\n                title={_t(\"Retry\")}\n                onClick={this.onResendClick}\n                key=\"resend\"\n            />);\n\n            // The delete button should appear last, so we can just drop it at the end\n            toolbarOpts.push(cancelSendingButton);\n        } else {\n            if (isContentActionable(this.props.mxEvent)) {\n                // Like the resend button, the react and reply buttons need to appear before the edit.\n                // The only catch is we do the reply button first so that we can make sure the react\n                // button is the very first button without having to do length checks for `splice()`.\n                if (this.context.canReply) {\n                    toolbarOpts.splice(0, 0, <RovingAccessibleTooltipButton\n                        className=\"mx_MessageActionBar_maskButton mx_MessageActionBar_replyButton\"\n                        title={_t(\"Reply\")}\n                        onClick={this.onReplyClick}\n                        key=\"reply\"\n                    />);\n                }\n                if (this.context.canReact) {\n                    toolbarOpts.splice(0, 0, <ReactButton\n                        mxEvent={this.props.mxEvent}\n                        reactions={this.props.reactions}\n                        onFocusChange={this.onFocusChange}\n                        key=\"react\"\n                    />);\n                }\n            }\n\n            if (allowCancel) {\n                toolbarOpts.push(cancelSendingButton);\n            }\n\n            // The menu button should be last, so dump it there.\n            toolbarOpts.push(<OptionsButton\n                mxEvent={this.props.mxEvent}\n                getReplyThread={this.props.getReplyThread}\n                getTile={this.props.getTile}\n                permalinkCreator={this.props.permalinkCreator}\n                onFocusChange={this.onFocusChange}\n                key=\"menu\"\n            />);\n        }\n\n        // aria-live=off to not have this read out automatically as navigating around timeline, gets repetitive.\n        return <Toolbar className=\"mx_MessageActionBar\" aria-label={_t(\"Message Actions\")} aria-live=\"off\">\n            {toolbarOpts}\n        </Toolbar>;\n    }\n}\n"]}