UNPKG

matrix-react-sdk

Version:
375 lines (314 loc) 38.9 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 _react = _interopRequireDefault(require("react")); var sdk = _interopRequireWildcard(require("../../../index")); var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher")); var _classnames = _interopRequireDefault(require("classnames")); var _room = require("matrix-js-sdk/src/models/room"); var _roomMember = require("matrix-js-sdk/src/models/room-member"); var _propTypes = _interopRequireDefault(require("prop-types")); var _MatrixClientPeg = require("../../../MatrixClientPeg"); var _FlairStore = _interopRequireDefault(require("../../../stores/FlairStore")); var _Permalinks = require("../../../utils/permalinks/Permalinks"); var _MatrixClientContext = _interopRequireDefault(require("../../../contexts/MatrixClientContext")); var _actions = require("../../../dispatcher/actions"); var _Media = require("../../../customisations/Media"); var _Tooltip = _interopRequireDefault(require("./Tooltip")); var _replaceableComponent = require("../../../utils/replaceableComponent"); var _dec, _class, _class2, _temp; let Pill = (_dec = (0, _replaceableComponent.replaceableComponent)("views.elements.Pill"), _dec(_class = (_temp = _class2 = class Pill extends _react.default.Component { constructor(...args) { super(...args); (0, _defineProperty2.default)(this, "state", { // ID/alias of the room/user resourceId: null, // Type of pill pillType: null, // The member related to the user pill member: null, // The group related to the group pill group: null, // The room related to the room pill room: null, // Is the user hovering the pill hover: false }); (0, _defineProperty2.default)(this, "onMouseOver", () => { this.setState({ hover: true }); }); (0, _defineProperty2.default)(this, "onMouseLeave", () => { this.setState({ hover: false }); }); (0, _defineProperty2.default)(this, "onUserPillClicked", () => { _dispatcher.default.dispatch({ action: _actions.Action.ViewUser, member: this.state.member }); }); } static roomNotifPos(text) { return text.indexOf("@room"); } static roomNotifLen() { return "@room".length; } // TODO: [REACT-WARNING] Replace with appropriate lifecycle event // eslint-disable-next-line camelcase async UNSAFE_componentWillReceiveProps(nextProps) { let resourceId; let prefix; if (nextProps.url) { if (nextProps.inMessage) { const parts = (0, _Permalinks.parseAppLocalLink)(nextProps.url); resourceId = parts.primaryEntityId; // The room/user ID prefix = parts.sigil; // The first character of prefix } else { resourceId = (0, _Permalinks.getPrimaryPermalinkEntity)(nextProps.url); prefix = resourceId ? resourceId[0] : undefined; } } const pillType = this.props.type || { '@': Pill.TYPE_USER_MENTION, '#': Pill.TYPE_ROOM_MENTION, '!': Pill.TYPE_ROOM_MENTION, '+': Pill.TYPE_GROUP_MENTION }[prefix]; let member; let group; let room; switch (pillType) { case Pill.TYPE_AT_ROOM_MENTION: { room = nextProps.room; } break; case Pill.TYPE_USER_MENTION: { const localMember = nextProps.room ? nextProps.room.getMember(resourceId) : undefined; member = localMember; if (!localMember) { member = new _roomMember.RoomMember(null, resourceId); this.doProfileLookup(resourceId, member); } } break; case Pill.TYPE_ROOM_MENTION: { const localRoom = resourceId[0] === '#' ? _MatrixClientPeg.MatrixClientPeg.get().getRooms().find(r => { return r.getCanonicalAlias() === resourceId || r.getAltAliases().includes(resourceId); }) : _MatrixClientPeg.MatrixClientPeg.get().getRoom(resourceId); room = localRoom; if (!localRoom) {// TODO: This would require a new API to resolve a room alias to // a room avatar and name. // this.doRoomProfileLookup(resourceId, member); } } break; case Pill.TYPE_GROUP_MENTION: { const cli = _MatrixClientPeg.MatrixClientPeg.get(); try { group = await _FlairStore.default.getGroupProfileCached(cli, resourceId); } catch (e) { // if FlairStore failed, fall back to just groupId group = { groupId: resourceId, avatarUrl: null, name: null }; } } } this.setState({ resourceId, pillType, member, group, room }); } componentDidMount() { this._unmounted = false; this._matrixClient = _MatrixClientPeg.MatrixClientPeg.get(); // eslint-disable-next-line new-cap this.UNSAFE_componentWillReceiveProps(this.props); // HACK: We shouldn't be calling lifecycle functions ourselves. } componentWillUnmount() { this._unmounted = true; } doProfileLookup(userId, member) { _MatrixClientPeg.MatrixClientPeg.get().getProfileInfo(userId).then(resp => { if (this._unmounted) { return; } member.name = resp.displayname; member.rawDisplayName = resp.displayname; member.events.member = { getContent: () => { return { avatar_url: resp.avatar_url }; }, getDirectionalContent: function () { return this.getContent(); } }; this.setState({ member }); }).catch(err => { console.error('Could not retrieve profile data for ' + userId + ':', err); }); } render() { const BaseAvatar = sdk.getComponent('views.avatars.BaseAvatar'); const MemberAvatar = sdk.getComponent('avatars.MemberAvatar'); const RoomAvatar = sdk.getComponent('avatars.RoomAvatar'); const resource = this.state.resourceId; let avatar = null; let linkText = resource; let pillClass; let userId; let href = this.props.url; let onClick; switch (this.state.pillType) { case Pill.TYPE_AT_ROOM_MENTION: { const room = this.props.room; if (room) { linkText = "@room"; if (this.props.shouldShowPillAvatar) { avatar = /*#__PURE__*/_react.default.createElement(RoomAvatar, { room: room, width: 16, height: 16, "aria-hidden": "true" }); } pillClass = 'mx_AtRoomPill'; } } break; case Pill.TYPE_USER_MENTION: { // If this user is not a member of this room, default to the empty member const member = this.state.member; if (member) { userId = member.userId; member.rawDisplayName = member.rawDisplayName || ''; linkText = member.rawDisplayName; if (this.props.shouldShowPillAvatar) { avatar = /*#__PURE__*/_react.default.createElement(MemberAvatar, { member: member, width: 16, height: 16, "aria-hidden": "true" }); } pillClass = 'mx_UserPill'; href = null; onClick = this.onUserPillClicked; } } break; case Pill.TYPE_ROOM_MENTION: { const room = this.state.room; if (room) { linkText = room.name || resource; if (this.props.shouldShowPillAvatar) { avatar = /*#__PURE__*/_react.default.createElement(RoomAvatar, { room: room, width: 16, height: 16, "aria-hidden": "true" }); } } pillClass = 'mx_RoomPill'; } break; case Pill.TYPE_GROUP_MENTION: { if (this.state.group) { const { avatarUrl, groupId, name } = this.state.group; linkText = groupId; if (this.props.shouldShowPillAvatar) { avatar = /*#__PURE__*/_react.default.createElement(BaseAvatar, { name: name || groupId, width: 16, height: 16, "aria-hidden": "true", url: avatarUrl ? (0, _Media.mediaFromMxc)(avatarUrl).getSquareThumbnailHttp(16) : null }); } pillClass = 'mx_GroupPill'; } } break; } const classes = (0, _classnames.default)("mx_Pill", pillClass, { "mx_UserPill_me": userId === _MatrixClientPeg.MatrixClientPeg.get().getUserId(), "mx_UserPill_selected": this.props.isSelected }); if (this.state.pillType) { const { yOffset } = this.props; let tip; if (this.state.hover && resource) { tip = /*#__PURE__*/_react.default.createElement(_Tooltip.default, { label: resource, yOffset: yOffset }); } return /*#__PURE__*/_react.default.createElement(_MatrixClientContext.default.Provider, { value: this._matrixClient }, this.props.inMessage ? /*#__PURE__*/_react.default.createElement("a", { className: classes, href: href, onClick: onClick, "data-offset-key": this.props.offsetKey, onMouseOver: this.onMouseOver, onMouseLeave: this.onMouseLeave }, avatar, linkText, tip) : /*#__PURE__*/_react.default.createElement("span", { className: classes, "data-offset-key": this.props.offsetKey, onMouseOver: this.onMouseOver, onMouseLeave: this.onMouseLeave }, avatar, linkText, tip)); } else { // Deliberately render nothing if the URL isn't recognised return null; } } }, (0, _defineProperty2.default)(_class2, "TYPE_USER_MENTION", 'TYPE_USER_MENTION'), (0, _defineProperty2.default)(_class2, "TYPE_ROOM_MENTION", 'TYPE_ROOM_MENTION'), (0, _defineProperty2.default)(_class2, "TYPE_GROUP_MENTION", 'TYPE_GROUP_MENTION'), (0, _defineProperty2.default)(_class2, "TYPE_AT_ROOM_MENTION", 'TYPE_AT_ROOM_MENTION'), (0, _defineProperty2.default)(_class2, "propTypes", { // The Type of this Pill. If url is given, this is auto-detected. type: _propTypes.default.string, // The URL to pillify (no validation is done) url: _propTypes.default.string, // Whether the pill is in a message inMessage: _propTypes.default.bool, // The room in which this pill is being rendered room: _propTypes.default.instanceOf(_room.Room), // Whether to include an avatar in the pill shouldShowPillAvatar: _propTypes.default.bool, // Whether to render this pill as if it were highlit by a selection isSelected: _propTypes.default.bool }), _temp)) || _class); var _default = Pill; exports.default = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/elements/Pill.js"],"names":["Pill","React","Component","resourceId","pillType","member","group","room","hover","setState","dis","dispatch","action","Action","ViewUser","state","roomNotifPos","text","indexOf","roomNotifLen","length","UNSAFE_componentWillReceiveProps","nextProps","prefix","url","inMessage","parts","primaryEntityId","sigil","undefined","props","type","TYPE_USER_MENTION","TYPE_ROOM_MENTION","TYPE_GROUP_MENTION","TYPE_AT_ROOM_MENTION","localMember","getMember","RoomMember","doProfileLookup","localRoom","MatrixClientPeg","get","getRooms","find","r","getCanonicalAlias","getAltAliases","includes","getRoom","cli","FlairStore","getGroupProfileCached","e","groupId","avatarUrl","name","componentDidMount","_unmounted","_matrixClient","componentWillUnmount","userId","getProfileInfo","then","resp","displayname","rawDisplayName","events","getContent","avatar_url","getDirectionalContent","catch","err","console","error","render","BaseAvatar","sdk","getComponent","MemberAvatar","RoomAvatar","resource","avatar","linkText","pillClass","href","onClick","shouldShowPillAvatar","onUserPillClicked","getSquareThumbnailHttp","classes","getUserId","isSelected","yOffset","tip","offsetKey","onMouseOver","onMouseLeave","PropTypes","string","bool","instanceOf","Room"],"mappings":";;;;;;;;;;;;;AAeA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;;;IAGMA,I,WADL,gDAAqB,qBAArB,C,mCAAD,MACMA,IADN,SACmBC,eAAMC,SADzB,CACmC;AAAA;AAAA;AAAA,iDA6BvB;AACJ;AACAC,MAAAA,UAAU,EAAE,IAFR;AAGJ;AACAC,MAAAA,QAAQ,EAAE,IAJN;AAMJ;AACAC,MAAAA,MAAM,EAAE,IAPJ;AAQJ;AACAC,MAAAA,KAAK,EAAE,IATH;AAUJ;AACAC,MAAAA,IAAI,EAAE,IAXF;AAYJ;AACAC,MAAAA,KAAK,EAAE;AAbH,KA7BuB;AAAA,uDAiIjB,MAAM;AAChB,WAAKC,QAAL,CAAc;AACVD,QAAAA,KAAK,EAAE;AADG,OAAd;AAGH,KArI8B;AAAA,wDAuIhB,MAAM;AACjB,WAAKC,QAAL,CAAc;AACVD,QAAAA,KAAK,EAAE;AADG,OAAd;AAGH,KA3I8B;AAAA,6DAkKX,MAAM;AACtBE,0BAAIC,QAAJ,CAAa;AACTC,QAAAA,MAAM,EAAEC,gBAAOC,QADN;AAETT,QAAAA,MAAM,EAAE,KAAKU,KAAL,CAAWV;AAFV,OAAb;AAIH,KAvK8B;AAAA;;AAC/B,SAAOW,YAAP,CAAoBC,IAApB,EAA0B;AACtB,WAAOA,IAAI,CAACC,OAAL,CAAa,OAAb,CAAP;AACH;;AAED,SAAOC,YAAP,GAAsB;AAClB,WAAO,QAAQC,MAAf;AACH;;AAsCD;AACA;AACA,QAAMC,gCAAN,CAAuCC,SAAvC,EAAkD;AAC9C,QAAInB,UAAJ;AACA,QAAIoB,MAAJ;;AAEA,QAAID,SAAS,CAACE,GAAd,EAAmB;AACf,UAAIF,SAAS,CAACG,SAAd,EAAyB;AACrB,cAAMC,KAAK,GAAG,mCAAkBJ,SAAS,CAACE,GAA5B,CAAd;AACArB,QAAAA,UAAU,GAAGuB,KAAK,CAACC,eAAnB,CAFqB,CAEe;;AACpCJ,QAAAA,MAAM,GAAGG,KAAK,CAACE,KAAf,CAHqB,CAGC;AACzB,OAJD,MAIO;AACHzB,QAAAA,UAAU,GAAG,2CAA0BmB,SAAS,CAACE,GAApC,CAAb;AACAD,QAAAA,MAAM,GAAGpB,UAAU,GAAGA,UAAU,CAAC,CAAD,CAAb,GAAmB0B,SAAtC;AACH;AACJ;;AAED,UAAMzB,QAAQ,GAAG,KAAK0B,KAAL,CAAWC,IAAX,IAAmB;AAChC,WAAK/B,IAAI,CAACgC,iBADsB;AAEhC,WAAKhC,IAAI,CAACiC,iBAFsB;AAGhC,WAAKjC,IAAI,CAACiC,iBAHsB;AAIhC,WAAKjC,IAAI,CAACkC;AAJsB,MAKlCX,MALkC,CAApC;AAOA,QAAIlB,MAAJ;AACA,QAAIC,KAAJ;AACA,QAAIC,IAAJ;;AACA,YAAQH,QAAR;AACI,WAAKJ,IAAI,CAACmC,oBAAV;AAAgC;AAC5B5B,UAAAA,IAAI,GAAGe,SAAS,CAACf,IAAjB;AACH;AACG;;AACJ,WAAKP,IAAI,CAACgC,iBAAV;AAA6B;AACzB,gBAAMI,WAAW,GAAGd,SAAS,CAACf,IAAV,GAAiBe,SAAS,CAACf,IAAV,CAAe8B,SAAf,CAAyBlC,UAAzB,CAAjB,GAAwD0B,SAA5E;AACAxB,UAAAA,MAAM,GAAG+B,WAAT;;AACA,cAAI,CAACA,WAAL,EAAkB;AACd/B,YAAAA,MAAM,GAAG,IAAIiC,sBAAJ,CAAe,IAAf,EAAqBnC,UAArB,CAAT;AACA,iBAAKoC,eAAL,CAAqBpC,UAArB,EAAiCE,MAAjC;AACH;AACJ;AACG;;AACJ,WAAKL,IAAI,CAACiC,iBAAV;AAA6B;AACzB,gBAAMO,SAAS,GAAGrC,UAAU,CAAC,CAAD,CAAV,KAAkB,GAAlB,GACdsC,iCAAgBC,GAAhB,GAAsBC,QAAtB,GAAiCC,IAAjC,CAAuCC,CAAD,IAAO;AACzC,mBAAOA,CAAC,CAACC,iBAAF,OAA0B3C,UAA1B,IACA0C,CAAC,CAACE,aAAF,GAAkBC,QAAlB,CAA2B7C,UAA3B,CADP;AAEH,WAHD,CADc,GAITsC,iCAAgBC,GAAhB,GAAsBO,OAAtB,CAA8B9C,UAA9B,CAJT;AAKAI,UAAAA,IAAI,GAAGiC,SAAP;;AACA,cAAI,CAACA,SAAL,EAAgB,CACZ;AACA;AACA;AACH;AACJ;AACG;;AACJ,WAAKxC,IAAI,CAACkC,kBAAV;AAA8B;AAC1B,gBAAMgB,GAAG,GAAGT,iCAAgBC,GAAhB,EAAZ;;AAEA,cAAI;AACApC,YAAAA,KAAK,GAAG,MAAM6C,oBAAWC,qBAAX,CAAiCF,GAAjC,EAAsC/C,UAAtC,CAAd;AACH,WAFD,CAEE,OAAOkD,CAAP,EAAU;AAAE;AACV/C,YAAAA,KAAK,GAAG;AACJgD,cAAAA,OAAO,EAAEnD,UADL;AAEJoD,cAAAA,SAAS,EAAE,IAFP;AAGJC,cAAAA,IAAI,EAAE;AAHF,aAAR;AAKH;AACJ;AAxCL;;AA0CA,SAAK/C,QAAL,CAAc;AAACN,MAAAA,UAAD;AAAaC,MAAAA,QAAb;AAAuBC,MAAAA,MAAvB;AAA+BC,MAAAA,KAA/B;AAAsCC,MAAAA;AAAtC,KAAd;AACH;;AAEDkD,EAAAA,iBAAiB,GAAG;AAChB,SAAKC,UAAL,GAAkB,KAAlB;AACA,SAAKC,aAAL,GAAqBlB,iCAAgBC,GAAhB,EAArB,CAFgB,CAIhB;;AACA,SAAKrB,gCAAL,CAAsC,KAAKS,KAA3C,EALgB,CAKmC;AACtD;;AAED8B,EAAAA,oBAAoB,GAAG;AACnB,SAAKF,UAAL,GAAkB,IAAlB;AACH;;AAcDnB,EAAAA,eAAe,CAACsB,MAAD,EAASxD,MAAT,EAAiB;AAC5BoC,qCAAgBC,GAAhB,GAAsBoB,cAAtB,CAAqCD,MAArC,EAA6CE,IAA7C,CAAmDC,IAAD,IAAU;AACxD,UAAI,KAAKN,UAAT,EAAqB;AACjB;AACH;;AACDrD,MAAAA,MAAM,CAACmD,IAAP,GAAcQ,IAAI,CAACC,WAAnB;AACA5D,MAAAA,MAAM,CAAC6D,cAAP,GAAwBF,IAAI,CAACC,WAA7B;AACA5D,MAAAA,MAAM,CAAC8D,MAAP,CAAc9D,MAAd,GAAuB;AACnB+D,QAAAA,UAAU,EAAE,MAAM;AACd,iBAAO;AAACC,YAAAA,UAAU,EAAEL,IAAI,CAACK;AAAlB,WAAP;AACH,SAHkB;AAInBC,QAAAA,qBAAqB,EAAE,YAAW;AAC9B,iBAAO,KAAKF,UAAL,EAAP;AACH;AANkB,OAAvB;AAQA,WAAK3D,QAAL,CAAc;AAACJ,QAAAA;AAAD,OAAd;AACH,KAfD,EAeGkE,KAfH,CAeUC,GAAD,IAAS;AACdC,MAAAA,OAAO,CAACC,KAAR,CAAc,yCAAyCb,MAAzC,GAAkD,GAAhE,EAAqEW,GAArE;AACH,KAjBD;AAkBH;;AASDG,EAAAA,MAAM,GAAG;AACL,UAAMC,UAAU,GAAGC,GAAG,CAACC,YAAJ,CAAiB,0BAAjB,CAAnB;AACA,UAAMC,YAAY,GAAGF,GAAG,CAACC,YAAJ,CAAiB,sBAAjB,CAArB;AACA,UAAME,UAAU,GAAGH,GAAG,CAACC,YAAJ,CAAiB,oBAAjB,CAAnB;AAEA,UAAMG,QAAQ,GAAG,KAAKlE,KAAL,CAAWZ,UAA5B;AAEA,QAAI+E,MAAM,GAAG,IAAb;AACA,QAAIC,QAAQ,GAAGF,QAAf;AACA,QAAIG,SAAJ;AACA,QAAIvB,MAAJ;AACA,QAAIwB,IAAI,GAAG,KAAKvD,KAAL,CAAWN,GAAtB;AACA,QAAI8D,OAAJ;;AACA,YAAQ,KAAKvE,KAAL,CAAWX,QAAnB;AACI,WAAKJ,IAAI,CAACmC,oBAAV;AAAgC;AAC5B,gBAAM5B,IAAI,GAAG,KAAKuB,KAAL,CAAWvB,IAAxB;;AACA,cAAIA,IAAJ,EAAU;AACN4E,YAAAA,QAAQ,GAAG,OAAX;;AACA,gBAAI,KAAKrD,KAAL,CAAWyD,oBAAf,EAAqC;AACjCL,cAAAA,MAAM,gBAAG,6BAAC,UAAD;AAAY,gBAAA,IAAI,EAAE3E,IAAlB;AAAwB,gBAAA,KAAK,EAAE,EAA/B;AAAmC,gBAAA,MAAM,EAAE,EAA3C;AAA+C,+BAAY;AAA3D,gBAAT;AACH;;AACD6E,YAAAA,SAAS,GAAG,eAAZ;AACH;AACJ;AACG;;AACJ,WAAKpF,IAAI,CAACgC,iBAAV;AAA6B;AACzB;AACA,gBAAM3B,MAAM,GAAG,KAAKU,KAAL,CAAWV,MAA1B;;AACA,cAAIA,MAAJ,EAAY;AACRwD,YAAAA,MAAM,GAAGxD,MAAM,CAACwD,MAAhB;AACAxD,YAAAA,MAAM,CAAC6D,cAAP,GAAwB7D,MAAM,CAAC6D,cAAP,IAAyB,EAAjD;AACAiB,YAAAA,QAAQ,GAAG9E,MAAM,CAAC6D,cAAlB;;AACA,gBAAI,KAAKpC,KAAL,CAAWyD,oBAAf,EAAqC;AACjCL,cAAAA,MAAM,gBAAG,6BAAC,YAAD;AAAc,gBAAA,MAAM,EAAE7E,MAAtB;AAA8B,gBAAA,KAAK,EAAE,EAArC;AAAyC,gBAAA,MAAM,EAAE,EAAjD;AAAqD,+BAAY;AAAjE,gBAAT;AACH;;AACD+E,YAAAA,SAAS,GAAG,aAAZ;AACAC,YAAAA,IAAI,GAAG,IAAP;AACAC,YAAAA,OAAO,GAAG,KAAKE,iBAAf;AACH;AACJ;AACG;;AACJ,WAAKxF,IAAI,CAACiC,iBAAV;AAA6B;AACzB,gBAAM1B,IAAI,GAAG,KAAKQ,KAAL,CAAWR,IAAxB;;AACA,cAAIA,IAAJ,EAAU;AACN4E,YAAAA,QAAQ,GAAG5E,IAAI,CAACiD,IAAL,IAAayB,QAAxB;;AACA,gBAAI,KAAKnD,KAAL,CAAWyD,oBAAf,EAAqC;AACjCL,cAAAA,MAAM,gBAAG,6BAAC,UAAD;AAAY,gBAAA,IAAI,EAAE3E,IAAlB;AAAwB,gBAAA,KAAK,EAAE,EAA/B;AAAmC,gBAAA,MAAM,EAAE,EAA3C;AAA+C,+BAAY;AAA3D,gBAAT;AACH;AACJ;;AACD6E,UAAAA,SAAS,GAAG,aAAZ;AACH;AACG;;AACJ,WAAKpF,IAAI,CAACkC,kBAAV;AAA8B;AAC1B,cAAI,KAAKnB,KAAL,CAAWT,KAAf,EAAsB;AAClB,kBAAM;AAACiD,cAAAA,SAAD;AAAYD,cAAAA,OAAZ;AAAqBE,cAAAA;AAArB,gBAA6B,KAAKzC,KAAL,CAAWT,KAA9C;AAEA6E,YAAAA,QAAQ,GAAG7B,OAAX;;AACA,gBAAI,KAAKxB,KAAL,CAAWyD,oBAAf,EAAqC;AACjCL,cAAAA,MAAM,gBAAG,6BAAC,UAAD;AACL,gBAAA,IAAI,EAAE1B,IAAI,IAAIF,OADT;AACkB,gBAAA,KAAK,EAAE,EADzB;AAC6B,gBAAA,MAAM,EAAE,EADrC;AACyC,+BAAY,MADrD;AAEL,gBAAA,GAAG,EAAEC,SAAS,GAAG,yBAAaA,SAAb,EAAwBkC,sBAAxB,CAA+C,EAA/C,CAAH,GAAwD;AAFjE,gBAAT;AAGH;;AACDL,YAAAA,SAAS,GAAG,cAAZ;AACH;AACJ;AACG;AApDR;;AAuDA,UAAMM,OAAO,GAAG,yBAAW,SAAX,EAAsBN,SAAtB,EAAiC;AAC7C,wBAAkBvB,MAAM,KAAKpB,iCAAgBC,GAAhB,GAAsBiD,SAAtB,EADgB;AAE7C,8BAAwB,KAAK7D,KAAL,CAAW8D;AAFU,KAAjC,CAAhB;;AAKA,QAAI,KAAK7E,KAAL,CAAWX,QAAf,EAAyB;AACrB,YAAM;AAACyF,QAAAA;AAAD,UAAY,KAAK/D,KAAvB;AAEA,UAAIgE,GAAJ;;AACA,UAAI,KAAK/E,KAAL,CAAWP,KAAX,IAAoByE,QAAxB,EAAkC;AAC9Ba,QAAAA,GAAG,gBAAG,6BAAC,gBAAD;AAAS,UAAA,KAAK,EAAEb,QAAhB;AAA0B,UAAA,OAAO,EAAEY;AAAnC,UAAN;AACH;;AAED,0BAAO,6BAAC,4BAAD,CAAqB,QAArB;AAA8B,QAAA,KAAK,EAAE,KAAKlC;AAA1C,SACD,KAAK7B,KAAL,CAAWL,SAAX,gBACE;AACI,QAAA,SAAS,EAAEiE,OADf;AAEI,QAAA,IAAI,EAAEL,IAFV;AAGI,QAAA,OAAO,EAAEC,OAHb;AAII,2BAAiB,KAAKxD,KAAL,CAAWiE,SAJhC;AAKI,QAAA,WAAW,EAAE,KAAKC,WALtB;AAMI,QAAA,YAAY,EAAE,KAAKC;AANvB,SAQMf,MARN,EASMC,QATN,EAUMW,GAVN,CADF,gBAaE;AACI,QAAA,SAAS,EAAEJ,OADf;AAEI,2BAAiB,KAAK5D,KAAL,CAAWiE,SAFhC;AAGI,QAAA,WAAW,EAAE,KAAKC,WAHtB;AAII,QAAA,YAAY,EAAE,KAAKC;AAJvB,SAMMf,MANN,EAOMC,QAPN,EAQMW,GARN,CAdD,CAAP;AAyBH,KAjCD,MAiCO;AACH;AACA,aAAO,IAAP;AACH;AACJ;;AAvR8B,C,8DASJ,mB,+DACA,mB,gEACC,oB,kEACE,sB,uDAEX;AACf;AACA/D,EAAAA,IAAI,EAAEmE,mBAAUC,MAFD;AAGf;AACA3E,EAAAA,GAAG,EAAE0E,mBAAUC,MAJA;AAKf;AACA1E,EAAAA,SAAS,EAAEyE,mBAAUE,IANN;AAOf;AACA7F,EAAAA,IAAI,EAAE2F,mBAAUG,UAAV,CAAqBC,UAArB,CARS;AASf;AACAf,EAAAA,oBAAoB,EAAEW,mBAAUE,IAVjB;AAWf;AACAR,EAAAA,UAAU,EAAEM,mBAAUE;AAZP,C;eA4QRpG,I","sourcesContent":["/*\nCopyright 2017 - 2019, 2021 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*/\nimport React from 'react';\nimport * as sdk from '../../../index';\nimport dis from '../../../dispatcher/dispatcher';\nimport classNames from 'classnames';\nimport { Room } from 'matrix-js-sdk/src/models/room';\nimport { RoomMember } from 'matrix-js-sdk/src/models/room-member';\nimport PropTypes from 'prop-types';\nimport {MatrixClientPeg} from '../../../MatrixClientPeg';\nimport FlairStore from \"../../../stores/FlairStore\";\nimport {getPrimaryPermalinkEntity, parseAppLocalLink} from \"../../../utils/permalinks/Permalinks\";\nimport MatrixClientContext from \"../../../contexts/MatrixClientContext\";\nimport {Action} from \"../../../dispatcher/actions\";\nimport {mediaFromMxc} from \"../../../customisations/Media\";\nimport Tooltip from './Tooltip';\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\n\n@replaceableComponent(\"views.elements.Pill\")\nclass Pill extends React.Component {\n    static roomNotifPos(text) {\n        return text.indexOf(\"@room\");\n    }\n\n    static roomNotifLen() {\n        return \"@room\".length;\n    }\n\n    static TYPE_USER_MENTION = 'TYPE_USER_MENTION';\n    static TYPE_ROOM_MENTION = 'TYPE_ROOM_MENTION';\n    static TYPE_GROUP_MENTION = 'TYPE_GROUP_MENTION';\n    static TYPE_AT_ROOM_MENTION = 'TYPE_AT_ROOM_MENTION'; // '@room' mention\n\n    static propTypes = {\n        // The Type of this Pill. If url is given, this is auto-detected.\n        type: PropTypes.string,\n        // The URL to pillify (no validation is done)\n        url: PropTypes.string,\n        // Whether the pill is in a message\n        inMessage: PropTypes.bool,\n        // The room in which this pill is being rendered\n        room: PropTypes.instanceOf(Room),\n        // Whether to include an avatar in the pill\n        shouldShowPillAvatar: PropTypes.bool,\n        // Whether to render this pill as if it were highlit by a selection\n        isSelected: PropTypes.bool,\n    };\n\n    state = {\n        // ID/alias of the room/user\n        resourceId: null,\n        // Type of pill\n        pillType: null,\n\n        // The member related to the user pill\n        member: null,\n        // The group related to the group pill\n        group: null,\n        // The room related to the room pill\n        room: null,\n        // Is the user hovering the pill\n        hover: false,\n    };\n\n    // TODO: [REACT-WARNING] Replace with appropriate lifecycle event\n    // eslint-disable-next-line camelcase\n    async UNSAFE_componentWillReceiveProps(nextProps) {\n        let resourceId;\n        let prefix;\n\n        if (nextProps.url) {\n            if (nextProps.inMessage) {\n                const parts = parseAppLocalLink(nextProps.url);\n                resourceId = parts.primaryEntityId; // The room/user ID\n                prefix = parts.sigil; // The first character of prefix\n            } else {\n                resourceId = getPrimaryPermalinkEntity(nextProps.url);\n                prefix = resourceId ? resourceId[0] : undefined;\n            }\n        }\n\n        const pillType = this.props.type || {\n            '@': Pill.TYPE_USER_MENTION,\n            '#': Pill.TYPE_ROOM_MENTION,\n            '!': Pill.TYPE_ROOM_MENTION,\n            '+': Pill.TYPE_GROUP_MENTION,\n        }[prefix];\n\n        let member;\n        let group;\n        let room;\n        switch (pillType) {\n            case Pill.TYPE_AT_ROOM_MENTION: {\n                room = nextProps.room;\n            }\n                break;\n            case Pill.TYPE_USER_MENTION: {\n                const localMember = nextProps.room ? nextProps.room.getMember(resourceId) : undefined;\n                member = localMember;\n                if (!localMember) {\n                    member = new RoomMember(null, resourceId);\n                    this.doProfileLookup(resourceId, member);\n                }\n            }\n                break;\n            case Pill.TYPE_ROOM_MENTION: {\n                const localRoom = resourceId[0] === '#' ?\n                    MatrixClientPeg.get().getRooms().find((r) => {\n                        return r.getCanonicalAlias() === resourceId ||\n                               r.getAltAliases().includes(resourceId);\n                    }) : MatrixClientPeg.get().getRoom(resourceId);\n                room = localRoom;\n                if (!localRoom) {\n                    // TODO: This would require a new API to resolve a room alias to\n                    // a room avatar and name.\n                    // this.doRoomProfileLookup(resourceId, member);\n                }\n            }\n                break;\n            case Pill.TYPE_GROUP_MENTION: {\n                const cli = MatrixClientPeg.get();\n\n                try {\n                    group = await FlairStore.getGroupProfileCached(cli, resourceId);\n                } catch (e) { // if FlairStore failed, fall back to just groupId\n                    group = {\n                        groupId: resourceId,\n                        avatarUrl: null,\n                        name: null,\n                    };\n                }\n            }\n        }\n        this.setState({resourceId, pillType, member, group, room});\n    }\n\n    componentDidMount() {\n        this._unmounted = false;\n        this._matrixClient = MatrixClientPeg.get();\n\n        // eslint-disable-next-line new-cap\n        this.UNSAFE_componentWillReceiveProps(this.props); // HACK: We shouldn't be calling lifecycle functions ourselves.\n    }\n\n    componentWillUnmount() {\n        this._unmounted = true;\n    }\n\n    onMouseOver = () => {\n        this.setState({\n            hover: true,\n        });\n    };\n\n    onMouseLeave = () => {\n        this.setState({\n            hover: false,\n        });\n    };\n\n    doProfileLookup(userId, member) {\n        MatrixClientPeg.get().getProfileInfo(userId).then((resp) => {\n            if (this._unmounted) {\n                return;\n            }\n            member.name = resp.displayname;\n            member.rawDisplayName = resp.displayname;\n            member.events.member = {\n                getContent: () => {\n                    return {avatar_url: resp.avatar_url};\n                },\n                getDirectionalContent: function() {\n                    return this.getContent();\n                },\n            };\n            this.setState({member});\n        }).catch((err) => {\n            console.error('Could not retrieve profile data for ' + userId + ':', err);\n        });\n    }\n\n    onUserPillClicked = () => {\n        dis.dispatch({\n            action: Action.ViewUser,\n            member: this.state.member,\n        });\n    };\n\n    render() {\n        const BaseAvatar = sdk.getComponent('views.avatars.BaseAvatar');\n        const MemberAvatar = sdk.getComponent('avatars.MemberAvatar');\n        const RoomAvatar = sdk.getComponent('avatars.RoomAvatar');\n\n        const resource = this.state.resourceId;\n\n        let avatar = null;\n        let linkText = resource;\n        let pillClass;\n        let userId;\n        let href = this.props.url;\n        let onClick;\n        switch (this.state.pillType) {\n            case Pill.TYPE_AT_ROOM_MENTION: {\n                const room = this.props.room;\n                if (room) {\n                    linkText = \"@room\";\n                    if (this.props.shouldShowPillAvatar) {\n                        avatar = <RoomAvatar room={room} width={16} height={16} aria-hidden=\"true\" />;\n                    }\n                    pillClass = 'mx_AtRoomPill';\n                }\n            }\n                break;\n            case Pill.TYPE_USER_MENTION: {\n                // If this user is not a member of this room, default to the empty member\n                const member = this.state.member;\n                if (member) {\n                    userId = member.userId;\n                    member.rawDisplayName = member.rawDisplayName || '';\n                    linkText = member.rawDisplayName;\n                    if (this.props.shouldShowPillAvatar) {\n                        avatar = <MemberAvatar member={member} width={16} height={16} aria-hidden=\"true\" />;\n                    }\n                    pillClass = 'mx_UserPill';\n                    href = null;\n                    onClick = this.onUserPillClicked;\n                }\n            }\n                break;\n            case Pill.TYPE_ROOM_MENTION: {\n                const room = this.state.room;\n                if (room) {\n                    linkText = room.name || resource;\n                    if (this.props.shouldShowPillAvatar) {\n                        avatar = <RoomAvatar room={room} width={16} height={16} aria-hidden=\"true\" />;\n                    }\n                }\n                pillClass = 'mx_RoomPill';\n            }\n                break;\n            case Pill.TYPE_GROUP_MENTION: {\n                if (this.state.group) {\n                    const {avatarUrl, groupId, name} = this.state.group;\n\n                    linkText = groupId;\n                    if (this.props.shouldShowPillAvatar) {\n                        avatar = <BaseAvatar\n                            name={name || groupId} width={16} height={16} aria-hidden=\"true\"\n                            url={avatarUrl ? mediaFromMxc(avatarUrl).getSquareThumbnailHttp(16) : null} />;\n                    }\n                    pillClass = 'mx_GroupPill';\n                }\n            }\n                break;\n        }\n\n        const classes = classNames(\"mx_Pill\", pillClass, {\n            \"mx_UserPill_me\": userId === MatrixClientPeg.get().getUserId(),\n            \"mx_UserPill_selected\": this.props.isSelected,\n        });\n\n        if (this.state.pillType) {\n            const {yOffset} = this.props;\n\n            let tip;\n            if (this.state.hover && resource) {\n                tip = <Tooltip label={resource} yOffset={yOffset} />;\n            }\n\n            return <MatrixClientContext.Provider value={this._matrixClient}>\n                { this.props.inMessage ?\n                    <a\n                        className={classes}\n                        href={href}\n                        onClick={onClick}\n                        data-offset-key={this.props.offsetKey}\n                        onMouseOver={this.onMouseOver}\n                        onMouseLeave={this.onMouseLeave}\n                    >\n                        { avatar }\n                        { linkText }\n                        { tip }\n                    </a> :\n                    <span\n                        className={classes}\n                        data-offset-key={this.props.offsetKey}\n                        onMouseOver={this.onMouseOver}\n                        onMouseLeave={this.onMouseLeave}\n                    >\n                        { avatar }\n                        { linkText }\n                        { tip }\n                    </span> }\n            </MatrixClientContext.Provider>;\n        } else {\n            // Deliberately render nothing if the URL isn't recognised\n            return null;\n        }\n    }\n}\n\nexport default Pill;\n"]}