matrix-react-sdk
Version:
SDK for matrix.org using React
125 lines (121 loc) • 18.6 kB
JavaScript
;
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 = _interopRequireWildcard(require("react"));
var _logger = require("matrix-js-sdk/src/logger");
var _languageHandler = require("../../../languageHandler");
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
/*
Copyright 2024 New Vector Ltd.
Copyright 2015, 2016 OpenMarket Ltd
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
const DIV_ID = "mx_recaptcha";
/**
* A pure UI component which displays a captcha form.
*/
class CaptchaForm extends _react.default.Component {
constructor(props) {
super(props);
(0, _defineProperty2.default)(this, "captchaWidgetId", void 0);
(0, _defineProperty2.default)(this, "recaptchaContainer", /*#__PURE__*/(0, _react.createRef)());
this.state = {
errorText: undefined
};
}
componentDidMount() {
// Just putting a script tag into the returned jsx doesn't work, annoyingly,
// so we do this instead.
if (this.isRecaptchaReady()) {
// already loaded
this.onCaptchaLoaded();
} else {
_logger.logger.log("Loading recaptcha script...");
window.mxOnRecaptchaLoaded = () => {
this.onCaptchaLoaded();
};
const scriptTag = document.createElement("script");
scriptTag.setAttribute("src", `https://www.recaptcha.net/recaptcha/api.js?onload=mxOnRecaptchaLoaded&render=explicit`);
this.recaptchaContainer.current?.appendChild(scriptTag);
}
}
componentWillUnmount() {
this.resetRecaptcha();
// Resettting the captcha does not clear the challenge overlay from the body in android webviews.
// Search for an iframe with the challenge src and remove it's topmost ancestor from the body.
// TODO: Remove this when the "mobile_register" page is retired.
const iframes = document.querySelectorAll("iframe");
for (const iframe of iframes) {
if (iframe.src.includes("https://www.recaptcha.net/recaptcha/api2/bframe")) {
let parentBeforeBody = iframe;
do {
parentBeforeBody = parentBeforeBody.parentElement;
} while (parentBeforeBody?.parentElement && parentBeforeBody.parentElement != document.body);
parentBeforeBody?.remove();
}
}
}
// Borrowed directly from: https://github.com/codeep/react-recaptcha-google/commit/e118fa5670fa268426969323b2e7fe77698376ba
isRecaptchaReady() {
return typeof window !== "undefined" && typeof global.grecaptcha !== "undefined" && typeof global.grecaptcha.render === "function";
}
renderRecaptcha(divId) {
if (!this.isRecaptchaReady()) {
_logger.logger.error("grecaptcha not loaded!");
throw new Error("Recaptcha did not load successfully");
}
const publicKey = this.props.sitePublicKey;
if (!publicKey) {
_logger.logger.error("No public key for recaptcha!");
throw new Error("This server has not supplied enough information for Recaptcha authentication");
}
_logger.logger.info(`Rendering to ${divId}`);
this.captchaWidgetId = global.grecaptcha?.render(divId, {
sitekey: publicKey,
callback: this.props.onCaptchaResponse
});
}
resetRecaptcha() {
if (this.captchaWidgetId) {
global?.grecaptcha?.reset(this.captchaWidgetId);
}
}
onCaptchaLoaded() {
_logger.logger.log("Loaded recaptcha script.");
try {
this.renderRecaptcha(DIV_ID);
// clear error if re-rendered
this.setState({
errorText: undefined
});
} catch (e) {
this.setState({
errorText: e instanceof Error ? e.message : String(e)
});
}
}
render() {
let error;
if (this.state.errorText) {
error = /*#__PURE__*/_react.default.createElement("div", {
className: "error"
}, this.state.errorText);
}
return /*#__PURE__*/_react.default.createElement("div", {
ref: this.recaptchaContainer
}, /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("auth|captcha_description")), /*#__PURE__*/_react.default.createElement("div", {
id: DIV_ID
}), error);
}
}
exports.default = CaptchaForm;
(0, _defineProperty2.default)(CaptchaForm, "defaultProps", {
onCaptchaResponse: () => {}
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_logger","_languageHandler","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","DIV_ID","CaptchaForm","React","Component","constructor","props","_defineProperty2","createRef","state","errorText","undefined","componentDidMount","isRecaptchaReady","onCaptchaLoaded","logger","log","window","mxOnRecaptchaLoaded","scriptTag","document","createElement","setAttribute","recaptchaContainer","current","appendChild","componentWillUnmount","resetRecaptcha","iframes","querySelectorAll","iframe","src","includes","parentBeforeBody","parentElement","body","remove","global","grecaptcha","render","renderRecaptcha","divId","error","Error","publicKey","sitePublicKey","info","captchaWidgetId","sitekey","callback","onCaptchaResponse","reset","setState","message","String","className","ref","_t","id","exports"],"sources":["../../../../src/components/views/auth/CaptchaForm.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2015, 2016 OpenMarket 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, { createRef } from \"react\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\n\nimport { _t } from \"../../../languageHandler\";\n\nconst DIV_ID = \"mx_recaptcha\";\n\ninterface ICaptchaFormProps {\n    sitePublicKey: string;\n    onCaptchaResponse: (response: string) => void;\n}\n\ninterface ICaptchaFormState {\n    errorText?: string;\n}\n\n/**\n * A pure UI component which displays a captcha form.\n */\nexport default class CaptchaForm extends React.Component<ICaptchaFormProps, ICaptchaFormState> {\n    public static defaultProps = {\n        onCaptchaResponse: () => {},\n    };\n\n    private captchaWidgetId?: string;\n    private recaptchaContainer = createRef<HTMLDivElement>();\n\n    public constructor(props: ICaptchaFormProps) {\n        super(props);\n\n        this.state = {\n            errorText: undefined,\n        };\n    }\n\n    public componentDidMount(): void {\n        // Just putting a script tag into the returned jsx doesn't work, annoyingly,\n        // so we do this instead.\n        if (this.isRecaptchaReady()) {\n            // already loaded\n            this.onCaptchaLoaded();\n        } else {\n            logger.log(\"Loading recaptcha script...\");\n            window.mxOnRecaptchaLoaded = () => {\n                this.onCaptchaLoaded();\n            };\n            const scriptTag = document.createElement(\"script\");\n            scriptTag.setAttribute(\n                \"src\",\n                `https://www.recaptcha.net/recaptcha/api.js?onload=mxOnRecaptchaLoaded&render=explicit`,\n            );\n            this.recaptchaContainer.current?.appendChild(scriptTag);\n        }\n    }\n\n    public componentWillUnmount(): void {\n        this.resetRecaptcha();\n        // Resettting the captcha does not clear the challenge overlay from the body in android webviews.\n        // Search for an iframe with the challenge src and remove it's topmost ancestor from the body.\n        // TODO: Remove this when the \"mobile_register\" page is retired.\n        const iframes = document.querySelectorAll(\"iframe\");\n        for (const iframe of iframes) {\n            if (iframe.src.includes(\"https://www.recaptcha.net/recaptcha/api2/bframe\")) {\n                let parentBeforeBody: HTMLElement | null = iframe;\n                do {\n                    parentBeforeBody = parentBeforeBody.parentElement;\n                } while (parentBeforeBody?.parentElement && parentBeforeBody.parentElement != document.body);\n                parentBeforeBody?.remove();\n            }\n        }\n    }\n\n    // Borrowed directly from: https://github.com/codeep/react-recaptcha-google/commit/e118fa5670fa268426969323b2e7fe77698376ba\n    private isRecaptchaReady(): boolean {\n        return (\n            typeof window !== \"undefined\" &&\n            typeof global.grecaptcha !== \"undefined\" &&\n            typeof global.grecaptcha.render === \"function\"\n        );\n    }\n\n    private renderRecaptcha(divId: string): void {\n        if (!this.isRecaptchaReady()) {\n            logger.error(\"grecaptcha not loaded!\");\n            throw new Error(\"Recaptcha did not load successfully\");\n        }\n\n        const publicKey = this.props.sitePublicKey;\n        if (!publicKey) {\n            logger.error(\"No public key for recaptcha!\");\n            throw new Error(\"This server has not supplied enough information for Recaptcha authentication\");\n        }\n\n        logger.info(`Rendering to ${divId}`);\n        this.captchaWidgetId = global.grecaptcha?.render(divId, {\n            sitekey: publicKey,\n            callback: this.props.onCaptchaResponse,\n        });\n    }\n\n    private resetRecaptcha(): void {\n        if (this.captchaWidgetId) {\n            global?.grecaptcha?.reset(this.captchaWidgetId);\n        }\n    }\n\n    private onCaptchaLoaded(): void {\n        logger.log(\"Loaded recaptcha script.\");\n        try {\n            this.renderRecaptcha(DIV_ID);\n            // clear error if re-rendered\n            this.setState({\n                errorText: undefined,\n            });\n        } catch (e) {\n            this.setState({\n                errorText: e instanceof Error ? e.message : String(e),\n            });\n        }\n    }\n\n    public render(): React.ReactNode {\n        let error: JSX.Element | undefined;\n        if (this.state.errorText) {\n            error = <div className=\"error\">{this.state.errorText}</div>;\n        }\n\n        return (\n            <div ref={this.recaptchaContainer}>\n                <p>{_t(\"auth|captcha_description\")}</p>\n                <div id={DIV_ID} />\n                {error}\n            </div>\n        );\n    }\n}\n"],"mappings":"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","ignoreList":[]}