* Add a Jest snapshot of AppPermission * Move the test inside 'for a pinned widget' category * Make only spinner message bold * Set font size specified with "mx_AppPermission_smallText" by default - Add "mx_AppPermission_largeText" for elements whose size has not been specified with mx_AppPermission_smallText - Create _AppWarning.pcss for AppWarning * Make AppPermission panel scrollable, keeping the content at the center * Run prettier * Use Heading component * Use Icon component * Fix the test
165 lines
6.3 KiB
TypeScript
165 lines
6.3 KiB
TypeScript
/*
|
|
Copyright 2017 Vector Creations Ltd
|
|
Copyright 2018, 2019 New Vector Ltd
|
|
Copyright 2019, 2020 The Matrix.org Foundation C.I.C.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
import React from "react";
|
|
import { RoomMember } from "matrix-js-sdk/src/models/room-member";
|
|
|
|
import { _t } from "../../../languageHandler";
|
|
import SdkConfig from "../../../SdkConfig";
|
|
import WidgetUtils from "../../../utils/WidgetUtils";
|
|
import { MatrixClientPeg } from "../../../MatrixClientPeg";
|
|
import MemberAvatar from "../avatars/MemberAvatar";
|
|
import BaseAvatar from "../avatars/BaseAvatar";
|
|
import Heading from "../typography/Heading";
|
|
import AccessibleButton from "./AccessibleButton";
|
|
import TextWithTooltip from "./TextWithTooltip";
|
|
import { parseUrl } from "../../../utils/UrlUtils";
|
|
import { Icon as HelpIcon } from "../../../../res/img/feather-customised/help-circle.svg";
|
|
|
|
interface IProps {
|
|
url: string;
|
|
creatorUserId: string;
|
|
roomId: string;
|
|
onPermissionGranted: () => void;
|
|
isRoomEncrypted?: boolean;
|
|
}
|
|
|
|
interface IState {
|
|
roomMember: RoomMember | null;
|
|
isWrapped: boolean;
|
|
widgetDomain: string | null;
|
|
}
|
|
|
|
export default class AppPermission extends React.Component<IProps, IState> {
|
|
public static defaultProps: Partial<IProps> = {
|
|
onPermissionGranted: () => {},
|
|
};
|
|
|
|
public constructor(props: IProps) {
|
|
super(props);
|
|
|
|
// The first step is to pick apart the widget so we can render information about it
|
|
const urlInfo = this.parseWidgetUrl();
|
|
|
|
// The second step is to find the user's profile so we can show it on the prompt
|
|
const room = MatrixClientPeg.get().getRoom(this.props.roomId);
|
|
let roomMember: RoomMember | null = null;
|
|
if (room) roomMember = room.getMember(this.props.creatorUserId);
|
|
|
|
// Set all this into the initial state
|
|
this.state = {
|
|
roomMember,
|
|
...urlInfo,
|
|
};
|
|
}
|
|
|
|
private parseWidgetUrl(): { isWrapped: boolean; widgetDomain: string | null } {
|
|
const widgetUrl = parseUrl(this.props.url);
|
|
|
|
// HACK: We're relying on the query params when we should be relying on the widget's `data`.
|
|
// This is a workaround for Scalar.
|
|
if (WidgetUtils.isScalarUrl(this.props.url) && widgetUrl.searchParams.has("url")) {
|
|
const unwrappedUrl = parseUrl(widgetUrl.searchParams.get("url")!);
|
|
return {
|
|
widgetDomain: unwrappedUrl.host || unwrappedUrl.hostname,
|
|
isWrapped: true,
|
|
};
|
|
} else {
|
|
return {
|
|
widgetDomain: widgetUrl.host || widgetUrl.hostname,
|
|
isWrapped: false,
|
|
};
|
|
}
|
|
}
|
|
|
|
public render(): React.ReactNode {
|
|
const brand = SdkConfig.get().brand;
|
|
|
|
const displayName = this.state.roomMember ? this.state.roomMember.name : this.props.creatorUserId;
|
|
const userId = displayName === this.props.creatorUserId ? null : this.props.creatorUserId;
|
|
|
|
const avatar = this.state.roomMember ? (
|
|
<MemberAvatar member={this.state.roomMember} width={38} height={38} />
|
|
) : (
|
|
<BaseAvatar name={this.props.creatorUserId} width={38} height={38} />
|
|
);
|
|
|
|
const warningTooltipText = (
|
|
<div>
|
|
{_t("Any of the following data may be shared:")}
|
|
<ul>
|
|
<li>{_t("Your display name")}</li>
|
|
<li>{_t("Your avatar URL")}</li>
|
|
<li>{_t("Your user ID")}</li>
|
|
<li>{_t("Your device ID")}</li>
|
|
<li>{_t("Your theme")}</li>
|
|
<li>{_t("Your language")}</li>
|
|
<li>{_t("%(brand)s URL", { brand })}</li>
|
|
<li>{_t("Room ID")}</li>
|
|
<li>{_t("Widget ID")}</li>
|
|
</ul>
|
|
</div>
|
|
);
|
|
const warningTooltip = (
|
|
<TextWithTooltip
|
|
tooltip={warningTooltipText}
|
|
tooltipClass="mx_Tooltip--appPermission mx_Tooltip--appPermission--dark"
|
|
class="mx_TextWithTooltip_target--helpIcon"
|
|
>
|
|
<HelpIcon className="mx_Icon mx_Icon_12" />
|
|
</TextWithTooltip>
|
|
);
|
|
|
|
// Due to i18n limitations, we can't dedupe the code for variables in these two messages.
|
|
const warning = this.state.isWrapped
|
|
? _t(
|
|
"Using this widget may share data <helpIcon /> with %(widgetDomain)s & your integration manager.",
|
|
{ widgetDomain: this.state.widgetDomain },
|
|
{ helpIcon: () => warningTooltip },
|
|
)
|
|
: _t(
|
|
"Using this widget may share data <helpIcon /> with %(widgetDomain)s.",
|
|
{ widgetDomain: this.state.widgetDomain },
|
|
{ helpIcon: () => warningTooltip },
|
|
);
|
|
|
|
const encryptionWarning = this.props.isRoomEncrypted ? _t("Widgets do not use message encryption.") : null;
|
|
|
|
return (
|
|
<div className="mx_AppPermission">
|
|
<div className="mx_AppPermission_content">
|
|
<div className="mx_AppPermission_content_bolder">{_t("Widget added by")}</div>
|
|
<div>
|
|
{avatar}
|
|
<Heading size="h4">{displayName}</Heading>
|
|
<div>{userId}</div>
|
|
</div>
|
|
<div>{warning}</div>
|
|
<div>
|
|
{_t("This widget may use cookies.")} {encryptionWarning}
|
|
</div>
|
|
<div>
|
|
<AccessibleButton kind="primary_sm" onClick={this.props.onPermissionGranted}>
|
|
{_t("Continue")}
|
|
</AccessibleButton>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
}
|