matrix-react-sdk
Version:
SDK for matrix.org using React
227 lines (182 loc) • 23.7 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 _classnames = _interopRequireDefault(require("classnames"));
var _RoomAvatar = _interopRequireDefault(require("./RoomAvatar"));
var _NotificationBadge = _interopRequireDefault(require("../rooms/NotificationBadge"));
var _RoomNotificationStateStore = require("../../../stores/notifications/RoomNotificationStateStore");
var _presence = require("../../../utils/presence");
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _languageHandler = require("../../../languageHandler");
var _TextWithTooltip = _interopRequireDefault(require("../elements/TextWithTooltip"));
var _DMRoomMap = _interopRequireDefault(require("../../../utils/DMRoomMap"));
var _replaceableComponent = require("../../../utils/replaceableComponent");
var _dec, _class, _temp;
var Icon;
(function (Icon) {
Icon["None"] = "NONE";
Icon["Globe"] = "GLOBE";
Icon["PresenceOnline"] = "ONLINE";
Icon["PresenceAway"] = "AWAY";
Icon["PresenceOffline"] = "OFFLINE";
})(Icon || (Icon = {}));
function tooltipText(variant
/*: Icon*/
) {
switch (variant) {
case Icon.Globe:
return (0, _languageHandler._t)("This room is public");
case Icon.PresenceOnline:
return (0, _languageHandler._t)("Online");
case Icon.PresenceAway:
return (0, _languageHandler._t)("Away");
case Icon.PresenceOffline:
return (0, _languageHandler._t)("Offline");
}
}
let DecoratedRoomAvatar = (_dec = (0, _replaceableComponent.replaceableComponent)("views.avatars.DecoratedRoomAvatar"), _dec(_class = (_temp = class DecoratedRoomAvatar extends _react.default.PureComponent
/*:: <IProps, IState>*/
{
constructor(props
/*: IProps*/
) {
super(props);
(0, _defineProperty2.default)(this, "_dmUser", void 0);
(0, _defineProperty2.default)(this, "isUnmounted", false);
(0, _defineProperty2.default)(this, "isWatchingTimeline", false);
(0, _defineProperty2.default)(this, "onRoomTimeline", (ev
/*: MatrixEvent*/
, room
/*: Room*/
) => {
if (this.isUnmounted) return; // apparently these can happen?
if (!room) return;
if (this.props.room.roomId !== room.roomId) return;
if (ev.getType() === 'm.room.join_rules' || ev.getType() === 'm.room.member') {
this.setState({
icon: this.calculateIcon()
});
}
});
(0, _defineProperty2.default)(this, "onPresenceUpdate", () => {
if (this.isUnmounted) return;
const newIcon = this.getPresenceIcon();
if (newIcon !== this.state.icon) this.setState({
icon: newIcon
});
});
this.state = {
notificationState: _RoomNotificationStateStore.RoomNotificationStateStore.instance.getRoomState(this.props.room),
icon: this.calculateIcon()
};
}
componentWillUnmount() {
this.isUnmounted = true;
if (this.isWatchingTimeline) this.props.room.off('Room.timeline', this.onRoomTimeline);
this.dmUser = null; // clear listeners, if any
}
get isPublicRoom()
/*: boolean*/
{
const joinRules = this.props.room.currentState.getStateEvents("m.room.join_rules", "");
const joinRule = joinRules && joinRules.getContent().join_rule;
return joinRule === 'public';
}
get dmUser()
/*: User*/
{
return this._dmUser;
}
set dmUser(val
/*: User*/
) {
const oldUser = this._dmUser;
this._dmUser = val;
if (oldUser && oldUser !== this._dmUser) {
oldUser.off('User.currentlyActive', this.onPresenceUpdate);
oldUser.off('User.presence', this.onPresenceUpdate);
}
if (this._dmUser && oldUser !== this._dmUser) {
this._dmUser.on('User.currentlyActive', this.onPresenceUpdate);
this._dmUser.on('User.presence', this.onPresenceUpdate);
}
}
getPresenceIcon()
/*: Icon*/
{
if (!this.dmUser) return Icon.None;
let icon = Icon.None;
const isOnline = this.dmUser.currentlyActive || this.dmUser.presence === 'online';
if (isOnline) {
icon = Icon.PresenceOnline;
} else if (this.dmUser.presence === 'offline') {
icon = Icon.PresenceOffline;
} else if (this.dmUser.presence === 'unavailable') {
icon = Icon.PresenceAway;
}
return icon;
}
calculateIcon()
/*: Icon*/
{
let icon = Icon.None; // We look at the DMRoomMap and not the tag here so that we don't exclude DMs in Favourites
const otherUserId = _DMRoomMap.default.shared().getUserIdForRoomId(this.props.room.roomId);
if (otherUserId && this.props.room.getJoinedMemberCount() === 2) {
// Track presence, if available
if ((0, _presence.isPresenceEnabled)()) {
if (otherUserId) {
this.dmUser = _MatrixClientPeg.MatrixClientPeg.get().getUser(otherUserId);
icon = this.getPresenceIcon();
}
}
} else {
// Track publicity
icon = this.isPublicRoom ? Icon.Globe : Icon.None;
if (!this.isWatchingTimeline) {
this.props.room.on('Room.timeline', this.onRoomTimeline);
this.isWatchingTimeline = true;
}
}
return icon;
}
render()
/*: React.ReactNode*/
{
let badge
/*: React.ReactNode*/
;
if (this.props.displayBadge) {
badge = /*#__PURE__*/_react.default.createElement(_NotificationBadge.default, {
notification: this.state.notificationState,
forceCount: this.props.forceCount,
roomId: this.props.room.roomId
});
}
let icon;
if (this.state.icon !== Icon.None) {
icon = /*#__PURE__*/_react.default.createElement(_TextWithTooltip.default, {
tooltip: tooltipText(this.state.icon),
class: `mx_DecoratedRoomAvatar_icon mx_DecoratedRoomAvatar_icon_${this.state.icon.toLowerCase()}`
});
}
const classes = (0, _classnames.default)("mx_DecoratedRoomAvatar", {
mx_DecoratedRoomAvatar_cutout: icon
});
return /*#__PURE__*/_react.default.createElement("div", {
className: classes
}, /*#__PURE__*/_react.default.createElement(_RoomAvatar.default, {
room: this.props.room,
width: this.props.avatarSize,
height: this.props.avatarSize,
oobData: this.props.oobData,
viewAvatarOnClick: this.props.viewAvatarOnClick
}), icon, badge);
}
}, _temp)) || _class);
exports.default = DecoratedRoomAvatar;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/avatars/DecoratedRoomAvatar.tsx"],"names":["Icon","tooltipText","variant","Globe","PresenceOnline","PresenceAway","PresenceOffline","DecoratedRoomAvatar","React","PureComponent","constructor","props","ev","room","isUnmounted","roomId","getType","setState","icon","calculateIcon","newIcon","getPresenceIcon","state","notificationState","RoomNotificationStateStore","instance","getRoomState","componentWillUnmount","isWatchingTimeline","off","onRoomTimeline","dmUser","isPublicRoom","joinRules","currentState","getStateEvents","joinRule","getContent","join_rule","_dmUser","val","oldUser","onPresenceUpdate","on","None","isOnline","currentlyActive","presence","otherUserId","DMRoomMap","shared","getUserIdForRoomId","getJoinedMemberCount","MatrixClientPeg","get","getUser","render","badge","displayBadge","forceCount","toLowerCase","classes","mx_DecoratedRoomAvatar_cutout","avatarSize","oobData","viewAvatarOnClick"],"mappings":";;;;;;;;;;;AAgBA;;AACA;;AAKA;;AACA;;AACA;;AAEA;;AACA;;AACA;;AACA;;AACA;;AACA;;;;IAgBKA,I;;WAAAA,I;AAAAA,EAAAA,I;AAAAA,EAAAA,I;AAAAA,EAAAA,I;AAAAA,EAAAA,I;AAAAA,EAAAA,I;GAAAA,I,KAAAA,I;;AASL,SAASC,WAAT,CAAqBC;AAArB;AAAA,EAAoC;AAChC,UAAQA,OAAR;AACI,SAAKF,IAAI,CAACG,KAAV;AACI,aAAO,yBAAG,qBAAH,CAAP;;AACJ,SAAKH,IAAI,CAACI,cAAV;AACI,aAAO,yBAAG,QAAH,CAAP;;AACJ,SAAKJ,IAAI,CAACK,YAAV;AACI,aAAO,yBAAG,MAAH,CAAP;;AACJ,SAAKL,IAAI,CAACM,eAAV;AACI,aAAO,yBAAG,SAAH,CAAP;AARR;AAUH;;IAGoBC,mB,WADpB,gDAAqB,mCAArB,C,yBAAD,MACqBA,mBADrB,SACiDC,eAAMC;AADvD;AACqF;AAKjFC,EAAAA,WAAW,CAACC;AAAD;AAAA,IAAgB;AACvB,UAAMA,KAAN;AADuB;AAAA,uDAHL,KAGK;AAAA,8DAFE,KAEF;AAAA,0DAsCF,CAACC;AAAD;AAAA,MAAkBC;AAAlB;AAAA,SAAiC;AACtD,UAAI,KAAKC,WAAT,EAAsB,OADgC,CAGtD;;AACA,UAAI,CAACD,IAAL,EAAW;AACX,UAAI,KAAKF,KAAL,CAAWE,IAAX,CAAgBE,MAAhB,KAA2BF,IAAI,CAACE,MAApC,EAA4C;;AAE5C,UAAIH,EAAE,CAACI,OAAH,OAAiB,mBAAjB,IAAwCJ,EAAE,CAACI,OAAH,OAAiB,eAA7D,EAA8E;AAC1E,aAAKC,QAAL,CAAc;AAACC,UAAAA,IAAI,EAAE,KAAKC,aAAL;AAAP,SAAd;AACH;AACJ,KAhD0B;AAAA,4DAkDA,MAAM;AAC7B,UAAI,KAAKL,WAAT,EAAsB;AAEtB,YAAMM,OAAO,GAAG,KAAKC,eAAL,EAAhB;AACA,UAAID,OAAO,KAAK,KAAKE,KAAL,CAAWJ,IAA3B,EAAiC,KAAKD,QAAL,CAAc;AAACC,QAAAA,IAAI,EAAEE;AAAP,OAAd;AACpC,KAvD0B;AAGvB,SAAKE,KAAL,GAAa;AACTC,MAAAA,iBAAiB,EAAEC,uDAA2BC,QAA3B,CAAoCC,YAApC,CAAiD,KAAKf,KAAL,CAAWE,IAA5D,CADV;AAETK,MAAAA,IAAI,EAAE,KAAKC,aAAL;AAFG,KAAb;AAIH;;AAEMQ,EAAAA,oBAAP,GAA8B;AAC1B,SAAKb,WAAL,GAAmB,IAAnB;AACA,QAAI,KAAKc,kBAAT,EAA6B,KAAKjB,KAAL,CAAWE,IAAX,CAAgBgB,GAAhB,CAAoB,eAApB,EAAqC,KAAKC,cAA1C;AAC7B,SAAKC,MAAL,GAAc,IAAd,CAH0B,CAGN;AACvB;;AAED,MAAYC,YAAZ;AAAA;AAAoC;AAChC,UAAMC,SAAS,GAAG,KAAKtB,KAAL,CAAWE,IAAX,CAAgBqB,YAAhB,CAA6BC,cAA7B,CAA4C,mBAA5C,EAAiE,EAAjE,CAAlB;AACA,UAAMC,QAAQ,GAAGH,SAAS,IAAIA,SAAS,CAACI,UAAV,GAAuBC,SAArD;AACA,WAAOF,QAAQ,KAAK,QAApB;AACH;;AAED,MAAYL,MAAZ;AAAA;AAA2B;AACvB,WAAO,KAAKQ,OAAZ;AACH;;AAED,MAAYR,MAAZ,CAAmBS;AAAnB;AAAA,IAA8B;AAC1B,UAAMC,OAAO,GAAG,KAAKF,OAArB;AACA,SAAKA,OAAL,GAAeC,GAAf;;AACA,QAAIC,OAAO,IAAIA,OAAO,KAAK,KAAKF,OAAhC,EAAyC;AACrCE,MAAAA,OAAO,CAACZ,GAAR,CAAY,sBAAZ,EAAoC,KAAKa,gBAAzC;AACAD,MAAAA,OAAO,CAACZ,GAAR,CAAY,eAAZ,EAA6B,KAAKa,gBAAlC;AACH;;AACD,QAAI,KAAKH,OAAL,IAAgBE,OAAO,KAAK,KAAKF,OAArC,EAA8C;AAC1C,WAAKA,OAAL,CAAaI,EAAb,CAAgB,sBAAhB,EAAwC,KAAKD,gBAA7C;;AACA,WAAKH,OAAL,CAAaI,EAAb,CAAgB,eAAhB,EAAiC,KAAKD,gBAAtC;AACH;AACJ;;AAqBOrB,EAAAA,eAAR;AAAA;AAAgC;AAC5B,QAAI,CAAC,KAAKU,MAAV,EAAkB,OAAO/B,IAAI,CAAC4C,IAAZ;AAElB,QAAI1B,IAAI,GAAGlB,IAAI,CAAC4C,IAAhB;AAEA,UAAMC,QAAQ,GAAG,KAAKd,MAAL,CAAYe,eAAZ,IAA+B,KAAKf,MAAL,CAAYgB,QAAZ,KAAyB,QAAzE;;AACA,QAAIF,QAAJ,EAAc;AACV3B,MAAAA,IAAI,GAAGlB,IAAI,CAACI,cAAZ;AACH,KAFD,MAEO,IAAI,KAAK2B,MAAL,CAAYgB,QAAZ,KAAyB,SAA7B,EAAwC;AAC3C7B,MAAAA,IAAI,GAAGlB,IAAI,CAACM,eAAZ;AACH,KAFM,MAEA,IAAI,KAAKyB,MAAL,CAAYgB,QAAZ,KAAyB,aAA7B,EAA4C;AAC/C7B,MAAAA,IAAI,GAAGlB,IAAI,CAACK,YAAZ;AACH;;AAED,WAAOa,IAAP;AACH;;AAEOC,EAAAA,aAAR;AAAA;AAA8B;AAC1B,QAAID,IAAI,GAAGlB,IAAI,CAAC4C,IAAhB,CAD0B,CAG1B;;AACA,UAAMI,WAAW,GAAGC,mBAAUC,MAAV,GAAmBC,kBAAnB,CAAsC,KAAKxC,KAAL,CAAWE,IAAX,CAAgBE,MAAtD,CAApB;;AACA,QAAIiC,WAAW,IAAI,KAAKrC,KAAL,CAAWE,IAAX,CAAgBuC,oBAAhB,OAA2C,CAA9D,EAAiE;AAC7D;AACA,UAAI,kCAAJ,EAAyB;AACrB,YAAIJ,WAAJ,EAAiB;AACb,eAAKjB,MAAL,GAAcsB,iCAAgBC,GAAhB,GAAsBC,OAAtB,CAA8BP,WAA9B,CAAd;AACA9B,UAAAA,IAAI,GAAG,KAAKG,eAAL,EAAP;AACH;AACJ;AACJ,KARD,MAQO;AACH;AACAH,MAAAA,IAAI,GAAG,KAAKc,YAAL,GAAoBhC,IAAI,CAACG,KAAzB,GAAiCH,IAAI,CAAC4C,IAA7C;;AACA,UAAI,CAAC,KAAKhB,kBAAV,EAA8B;AAC1B,aAAKjB,KAAL,CAAWE,IAAX,CAAgB8B,EAAhB,CAAmB,eAAnB,EAAoC,KAAKb,cAAzC;AACA,aAAKF,kBAAL,GAA0B,IAA1B;AACH;AACJ;;AACD,WAAOV,IAAP;AACH;;AAEMsC,EAAAA,MAAP;AAAA;AAAiC;AAC7B,QAAIC;AAAsB;AAA1B;;AACA,QAAI,KAAK9C,KAAL,CAAW+C,YAAf,EAA6B;AACzBD,MAAAA,KAAK,gBAAG,6BAAC,0BAAD;AACJ,QAAA,YAAY,EAAE,KAAKnC,KAAL,CAAWC,iBADrB;AAEJ,QAAA,UAAU,EAAE,KAAKZ,KAAL,CAAWgD,UAFnB;AAGJ,QAAA,MAAM,EAAE,KAAKhD,KAAL,CAAWE,IAAX,CAAgBE;AAHpB,QAAR;AAKH;;AAED,QAAIG,IAAJ;;AACA,QAAI,KAAKI,KAAL,CAAWJ,IAAX,KAAoBlB,IAAI,CAAC4C,IAA7B,EAAmC;AAC/B1B,MAAAA,IAAI,gBAAG,6BAAC,wBAAD;AACH,QAAA,OAAO,EAAEjB,WAAW,CAAC,KAAKqB,KAAL,CAAWJ,IAAZ,CADjB;AAEH,QAAA,KAAK,EAAG,2DAA0D,KAAKI,KAAL,CAAWJ,IAAX,CAAgB0C,WAAhB,EAA8B;AAF7F,QAAP;AAIH;;AAED,UAAMC,OAAO,GAAG,yBAAW,wBAAX,EAAqC;AACjDC,MAAAA,6BAA6B,EAAE5C;AADkB,KAArC,CAAhB;AAIA,wBAAO;AAAK,MAAA,SAAS,EAAE2C;AAAhB,oBACH,6BAAC,mBAAD;AACI,MAAA,IAAI,EAAE,KAAKlD,KAAL,CAAWE,IADrB;AAEI,MAAA,KAAK,EAAE,KAAKF,KAAL,CAAWoD,UAFtB;AAGI,MAAA,MAAM,EAAE,KAAKpD,KAAL,CAAWoD,UAHvB;AAII,MAAA,OAAO,EAAE,KAAKpD,KAAL,CAAWqD,OAJxB;AAKI,MAAA,iBAAiB,EAAE,KAAKrD,KAAL,CAAWsD;AALlC,MADG,EAQF/C,IARE,EASFuC,KATE,CAAP;AAWH;;AAxIgF,C","sourcesContent":["/*\nCopyright 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 from \"react\";\nimport classNames from \"classnames\";\nimport { Room } from \"matrix-js-sdk/src/models/room\";\nimport { User } from \"matrix-js-sdk/src/models/user\";\nimport { MatrixEvent } from \"matrix-js-sdk/src/models/event\";\n\nimport RoomAvatar from \"./RoomAvatar\";\nimport NotificationBadge from '../rooms/NotificationBadge';\nimport { RoomNotificationStateStore } from \"../../../stores/notifications/RoomNotificationStateStore\";\nimport { NotificationState } from \"../../../stores/notifications/NotificationState\";\nimport {isPresenceEnabled} from \"../../../utils/presence\";\nimport {MatrixClientPeg} from \"../../../MatrixClientPeg\";\nimport {_t} from \"../../../languageHandler\";\nimport TextWithTooltip from \"../elements/TextWithTooltip\";\nimport DMRoomMap from \"../../../utils/DMRoomMap\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\n\ninterface IProps {\n    room: Room;\n    avatarSize: number;\n    displayBadge?: boolean;\n    forceCount?: boolean;\n    oobData?: object;\n    viewAvatarOnClick?: boolean;\n}\n\ninterface IState {\n    notificationState?: NotificationState;\n    icon: Icon;\n}\n\nenum Icon {\n    // Note: the names here are used in CSS class names\n    None = \"NONE\", // ... except this one\n    Globe = \"GLOBE\",\n    PresenceOnline = \"ONLINE\",\n    PresenceAway = \"AWAY\",\n    PresenceOffline = \"OFFLINE\",\n}\n\nfunction tooltipText(variant: Icon) {\n    switch (variant) {\n        case Icon.Globe:\n            return _t(\"This room is public\");\n        case Icon.PresenceOnline:\n            return _t(\"Online\");\n        case Icon.PresenceAway:\n            return _t(\"Away\");\n        case Icon.PresenceOffline:\n            return _t(\"Offline\");\n    }\n}\n\n@replaceableComponent(\"views.avatars.DecoratedRoomAvatar\")\nexport default class DecoratedRoomAvatar extends React.PureComponent<IProps, IState> {\n    private _dmUser: User;\n    private isUnmounted = false;\n    private isWatchingTimeline = false;\n\n    constructor(props: IProps) {\n        super(props);\n\n        this.state = {\n            notificationState: RoomNotificationStateStore.instance.getRoomState(this.props.room),\n            icon: this.calculateIcon(),\n        };\n    }\n\n    public componentWillUnmount() {\n        this.isUnmounted = true;\n        if (this.isWatchingTimeline) this.props.room.off('Room.timeline', this.onRoomTimeline);\n        this.dmUser = null; // clear listeners, if any\n    }\n\n    private get isPublicRoom(): boolean {\n        const joinRules = this.props.room.currentState.getStateEvents(\"m.room.join_rules\", \"\");\n        const joinRule = joinRules && joinRules.getContent().join_rule;\n        return joinRule === 'public';\n    }\n\n    private get dmUser(): User {\n        return this._dmUser;\n    }\n\n    private set dmUser(val: User) {\n        const oldUser = this._dmUser;\n        this._dmUser = val;\n        if (oldUser && oldUser !== this._dmUser) {\n            oldUser.off('User.currentlyActive', this.onPresenceUpdate);\n            oldUser.off('User.presence', this.onPresenceUpdate);\n        }\n        if (this._dmUser && oldUser !== this._dmUser) {\n            this._dmUser.on('User.currentlyActive', this.onPresenceUpdate);\n            this._dmUser.on('User.presence', this.onPresenceUpdate);\n        }\n    }\n\n    private onRoomTimeline = (ev: MatrixEvent, room: Room) => {\n        if (this.isUnmounted) return;\n\n        // apparently these can happen?\n        if (!room) return;\n        if (this.props.room.roomId !== room.roomId) return;\n\n        if (ev.getType() === 'm.room.join_rules' || ev.getType() === 'm.room.member') {\n            this.setState({icon: this.calculateIcon()});\n        }\n    };\n\n    private onPresenceUpdate = () => {\n        if (this.isUnmounted) return;\n\n        const newIcon = this.getPresenceIcon();\n        if (newIcon !== this.state.icon) this.setState({icon: newIcon});\n    };\n\n    private getPresenceIcon(): Icon {\n        if (!this.dmUser) return Icon.None;\n\n        let icon = Icon.None;\n\n        const isOnline = this.dmUser.currentlyActive || this.dmUser.presence === 'online';\n        if (isOnline) {\n            icon = Icon.PresenceOnline;\n        } else if (this.dmUser.presence === 'offline') {\n            icon = Icon.PresenceOffline;\n        } else if (this.dmUser.presence === 'unavailable') {\n            icon = Icon.PresenceAway;\n        }\n\n        return icon;\n    }\n\n    private calculateIcon(): Icon {\n        let icon = Icon.None;\n\n        // We look at the DMRoomMap and not the tag here so that we don't exclude DMs in Favourites\n        const otherUserId = DMRoomMap.shared().getUserIdForRoomId(this.props.room.roomId);\n        if (otherUserId && this.props.room.getJoinedMemberCount() === 2) {\n            // Track presence, if available\n            if (isPresenceEnabled()) {\n                if (otherUserId) {\n                    this.dmUser = MatrixClientPeg.get().getUser(otherUserId);\n                    icon = this.getPresenceIcon();\n                }\n            }\n        } else {\n            // Track publicity\n            icon = this.isPublicRoom ? Icon.Globe : Icon.None;\n            if (!this.isWatchingTimeline) {\n                this.props.room.on('Room.timeline', this.onRoomTimeline);\n                this.isWatchingTimeline = true;\n            }\n        }\n        return icon;\n    }\n\n    public render(): React.ReactNode {\n        let badge: React.ReactNode;\n        if (this.props.displayBadge) {\n            badge = <NotificationBadge\n                notification={this.state.notificationState}\n                forceCount={this.props.forceCount}\n                roomId={this.props.room.roomId}\n            />;\n        }\n\n        let icon;\n        if (this.state.icon !== Icon.None) {\n            icon = <TextWithTooltip\n                tooltip={tooltipText(this.state.icon)}\n                class={`mx_DecoratedRoomAvatar_icon mx_DecoratedRoomAvatar_icon_${this.state.icon.toLowerCase()}`}\n            />;\n        }\n\n        const classes = classNames(\"mx_DecoratedRoomAvatar\", {\n            mx_DecoratedRoomAvatar_cutout: icon,\n        });\n\n        return <div className={classes}>\n            <RoomAvatar\n                room={this.props.room}\n                width={this.props.avatarSize}\n                height={this.props.avatarSize}\n                oobData={this.props.oobData}\n                viewAvatarOnClick={this.props.viewAvatarOnClick}\n            />\n            {icon}\n            {badge}\n        </div>;\n    }\n}\n"]}