matrix-react-sdk
Version:
SDK for matrix.org using React
124 lines (120 loc) • 20.8 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _reactFocusLock = _interopRequireDefault(require("react-focus-lock"));
var _classnames = _interopRequireDefault(require("classnames"));
var _AccessibleButton = _interopRequireDefault(require("../elements/AccessibleButton"));
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _languageHandler = require("../../../languageHandler");
var _MatrixClientContext = _interopRequireDefault(require("../../../contexts/MatrixClientContext"));
var _Heading = _interopRequireDefault(require("../typography/Heading"));
var _PosthogTrackers = require("../../../PosthogTrackers");
var _KeyBindingsManager = require("../../../KeyBindingsManager");
var _KeyboardShortcuts = require("../../../accessibility/KeyboardShortcuts");
/*
Copyright 2024 New Vector Ltd.
Copyright 2019 The Matrix.org Foundation C.I.C.
Copyright 2018, 2019 New Vector Ltd
Copyright 2017 Vector Creations Ltd
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
/*
* Basic container for modal dialogs.
*
* Includes a div for the title, and a keypress handler which cancels the
* dialog on escape.
*/
class BaseDialog extends _react.default.Component {
constructor(props) {
super(props);
// XXX: The contract on MatrixClientContext says it is only available within a LoggedInView subtree,
// given that modals function outside the MatrixChat React tree this simulates that. We don't want to
// use safeGet as it throwing would mean we cannot use modals whilst the user isn't logged in.
// The longer term solution is to move our ModalManager into the React tree to inherit contexts properly.
(0, _defineProperty2.default)(this, "matrixClient", void 0);
(0, _defineProperty2.default)(this, "onKeyDown", e => {
this.props.onKeyDown?.(e);
const action = (0, _KeyBindingsManager.getKeyBindingsManager)().getAccessibilityAction(e);
switch (action) {
case _KeyboardShortcuts.KeyBindingAction.Escape:
if (!this.props.hasCancel) break;
e.stopPropagation();
e.preventDefault();
this.props.onFinished();
break;
}
});
(0, _defineProperty2.default)(this, "onCancelClick", () => {
this.props.onFinished();
});
this.matrixClient = _MatrixClientPeg.MatrixClientPeg.get();
}
render() {
let cancelButton;
if (this.props.hasCancel) {
cancelButton = /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
onClick: this.onCancelClick,
className: "mx_Dialog_cancelButton",
"aria-label": (0, _languageHandler._t)("dialog_close_label"),
title: (0, _languageHandler._t)("action|close"),
placement: "bottom"
});
}
let headerImage;
if (this.props.headerImage) {
headerImage = /*#__PURE__*/_react.default.createElement("img", {
className: "mx_Dialog_titleImage",
src: this.props.headerImage,
alt: ""
});
}
const lockProps = {
"onKeyDown": this.onKeyDown,
"role": "dialog",
// This should point to a node describing the dialog.
// If we were about to completely follow this recommendation we'd need to
// make all the components relying on BaseDialog to be aware of it.
// So instead we will use the whole content as the description.
// Description comes first and if the content contains more text,
// AT users can skip its presentation.
"aria-describedby": this.props.contentId
};
if (this.props["aria-label"]) {
lockProps["aria-label"] = this.props["aria-label"];
} else {
lockProps["aria-labelledby"] = "mx_BaseDialog_title";
}
return /*#__PURE__*/_react.default.createElement(_MatrixClientContext.default.Provider, {
value: this.matrixClient
}, this.props.screenName && /*#__PURE__*/_react.default.createElement(_PosthogTrackers.PosthogScreenTracker, {
screenName: this.props.screenName
}), /*#__PURE__*/_react.default.createElement(_reactFocusLock.default, {
returnFocus: true,
lockProps: lockProps,
className: (0, _classnames.default)(this.props.className, {
mx_Dialog_fixedWidth: this.props.fixedWidth
})
}, this.props.top, /*#__PURE__*/_react.default.createElement("div", {
className: (0, _classnames.default)("mx_Dialog_header", {
mx_Dialog_headerWithButton: !!this.props.headerButton
})
}, !!(this.props.title || headerImage) && /*#__PURE__*/_react.default.createElement(_Heading.default, {
size: "3",
as: "h1",
className: (0, _classnames.default)("mx_Dialog_title", this.props.titleClass),
id: "mx_BaseDialog_title"
}, headerImage, this.props.title), this.props.headerButton), this.props.children, cancelButton));
}
}
exports.default = BaseDialog;
(0, _defineProperty2.default)(BaseDialog, "defaultProps", {
hasCancel: true,
fixedWidth: true
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_reactFocusLock","_classnames","_AccessibleButton","_MatrixClientPeg","_languageHandler","_MatrixClientContext","_Heading","_PosthogTrackers","_KeyBindingsManager","_KeyboardShortcuts","BaseDialog","React","Component","constructor","props","_defineProperty2","default","e","onKeyDown","action","getKeyBindingsManager","getAccessibilityAction","KeyBindingAction","Escape","hasCancel","stopPropagation","preventDefault","onFinished","matrixClient","MatrixClientPeg","get","render","cancelButton","createElement","onClick","onCancelClick","className","_t","title","placement","headerImage","src","alt","lockProps","contentId","Provider","value","screenName","PosthogScreenTracker","returnFocus","classNames","mx_Dialog_fixedWidth","fixedWidth","top","mx_Dialog_headerWithButton","headerButton","size","as","titleClass","id","children","exports"],"sources":["../../../../src/components/views/dialogs/BaseDialog.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2019 The Matrix.org Foundation C.I.C.\nCopyright 2018, 2019 New Vector Ltd\nCopyright 2017 Vector Creations Ltd\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport React from \"react\";\nimport FocusLock from \"react-focus-lock\";\nimport classNames from \"classnames\";\nimport { MatrixClient } from \"matrix-js-sdk/src/matrix\";\n\nimport AccessibleButton from \"../elements/AccessibleButton\";\nimport { MatrixClientPeg } from \"../../../MatrixClientPeg\";\nimport { _t } from \"../../../languageHandler\";\nimport MatrixClientContext from \"../../../contexts/MatrixClientContext\";\nimport Heading from \"../typography/Heading\";\nimport { PosthogScreenTracker, ScreenName } from \"../../../PosthogTrackers\";\nimport { getKeyBindingsManager } from \"../../../KeyBindingsManager\";\nimport { KeyBindingAction } from \"../../../accessibility/KeyboardShortcuts\";\n\ninterface IProps {\n    // Whether the dialog should have a 'close' button that will\n    // cause the dialog to be cancelled. This should only be set\n    // to false if there is nothing the app can sensibly do if the\n    // dialog is cancelled, eg. \"We can't restore your session and\n    // the app cannot work\". Default: true.\n    \"hasCancel\"?: boolean;\n\n    // called when a key is pressed\n    \"onKeyDown\"?: (e: KeyboardEvent | React.KeyboardEvent) => void;\n\n    // CSS class to apply to dialog div\n    \"className\"?: string;\n\n    // if true, dialog container is 60% of the viewport width. Otherwise,\n    // the container will have no fixed size, allowing its contents to\n    // determine its size. Default: true.\n    \"fixedWidth\"?: boolean;\n\n    // To be displayed at the top of the dialog. Even above the title.\n    \"top\"?: React.ReactNode;\n\n    // Title for the dialog.\n    \"title\"?: React.ReactNode;\n    // Specific aria label to use, if not provided will set aria-labelledBy to mx_Dialog_title\n    \"aria-label\"?: string;\n\n    // Path to an icon to put in the header\n    \"headerImage\"?: string;\n\n    // children should be the content of the dialog\n    \"children\"?: React.ReactNode;\n\n    // Id of content element\n    // If provided, this is used to add a aria-describedby attribute\n    \"contentId\"?: string;\n\n    // optional additional class for the title element (basically anything that can be passed to classnames)\n    \"titleClass\"?: string | string[];\n\n    \"headerButton\"?: JSX.Element;\n\n    // optional Posthog ScreenName to supply during the lifetime of this dialog\n    \"screenName\"?: ScreenName;\n    onFinished(): void;\n}\n\n/*\n * Basic container for modal dialogs.\n *\n * Includes a div for the title, and a keypress handler which cancels the\n * dialog on escape.\n */\nexport default class BaseDialog extends React.Component<IProps> {\n    private matrixClient: MatrixClient;\n\n    public static defaultProps: Partial<IProps> = {\n        hasCancel: true,\n        fixedWidth: true,\n    };\n\n    public constructor(props: IProps) {\n        super(props);\n\n        // XXX: The contract on MatrixClientContext says it is only available within a LoggedInView subtree,\n        // given that modals function outside the MatrixChat React tree this simulates that. We don't want to\n        // use safeGet as it throwing would mean we cannot use modals whilst the user isn't logged in.\n        // The longer term solution is to move our ModalManager into the React tree to inherit contexts properly.\n        this.matrixClient = MatrixClientPeg.get()!;\n    }\n\n    private onKeyDown = (e: KeyboardEvent | React.KeyboardEvent): void => {\n        this.props.onKeyDown?.(e);\n\n        const action = getKeyBindingsManager().getAccessibilityAction(e);\n        switch (action) {\n            case KeyBindingAction.Escape:\n                if (!this.props.hasCancel) break;\n\n                e.stopPropagation();\n                e.preventDefault();\n                this.props.onFinished();\n                break;\n        }\n    };\n\n    private onCancelClick = (): void => {\n        this.props.onFinished();\n    };\n\n    public render(): React.ReactNode {\n        let cancelButton;\n        if (this.props.hasCancel) {\n            cancelButton = (\n                <AccessibleButton\n                    onClick={this.onCancelClick}\n                    className=\"mx_Dialog_cancelButton\"\n                    aria-label={_t(\"dialog_close_label\")}\n                    title={_t(\"action|close\")}\n                    placement=\"bottom\"\n                />\n            );\n        }\n\n        let headerImage;\n        if (this.props.headerImage) {\n            headerImage = <img className=\"mx_Dialog_titleImage\" src={this.props.headerImage} alt=\"\" />;\n        }\n\n        const lockProps: Record<string, any> = {\n            \"onKeyDown\": this.onKeyDown,\n            \"role\": \"dialog\",\n            // This should point to a node describing the dialog.\n            // If we were about to completely follow this recommendation we'd need to\n            // make all the components relying on BaseDialog to be aware of it.\n            // So instead we will use the whole content as the description.\n            // Description comes first and if the content contains more text,\n            // AT users can skip its presentation.\n            \"aria-describedby\": this.props.contentId,\n        };\n\n        if (this.props[\"aria-label\"]) {\n            lockProps[\"aria-label\"] = this.props[\"aria-label\"];\n        } else {\n            lockProps[\"aria-labelledby\"] = \"mx_BaseDialog_title\";\n        }\n\n        return (\n            <MatrixClientContext.Provider value={this.matrixClient}>\n                {this.props.screenName && <PosthogScreenTracker screenName={this.props.screenName} />}\n                <FocusLock\n                    returnFocus={true}\n                    lockProps={lockProps}\n                    className={classNames(this.props.className, {\n                        mx_Dialog_fixedWidth: this.props.fixedWidth,\n                    })}\n                >\n                    {this.props.top}\n                    <div\n                        className={classNames(\"mx_Dialog_header\", {\n                            mx_Dialog_headerWithButton: !!this.props.headerButton,\n                        })}\n                    >\n                        {!!(this.props.title || headerImage) && (\n                            <Heading\n                                size=\"3\"\n                                as=\"h1\"\n                                className={classNames(\"mx_Dialog_title\", this.props.titleClass)}\n                                id=\"mx_BaseDialog_title\"\n                            >\n                                {headerImage}\n                                {this.props.title}\n                            </Heading>\n                        )}\n                        {this.props.headerButton}\n                    </div>\n                    {this.props.children}\n                    {cancelButton}\n                </FocusLock>\n            </MatrixClientContext.Provider>\n        );\n    }\n}\n"],"mappings":"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","ignoreList":[]}