/* 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, {createRef} from 'react'; import PropTypes from 'prop-types'; import {_t} from "../../../languageHandler"; import * as sdk from "../../../index"; import {MatrixClientPeg} from "../../../MatrixClientPeg"; import {makeRoomPermalink, makeUserPermalink} from "../../../utils/permalinks/Permalinks"; import DMRoomMap from "../../../utils/DMRoomMap"; import {RoomMember} from "matrix-js-sdk/src/matrix"; import SdkConfig from "../../../SdkConfig"; import {getHttpUriForMxc} from "matrix-js-sdk/src/content-repo"; import * as Email from "../../../email"; import {getDefaultIdentityServerUrl, useDefaultIdentityServer} from "../../../utils/IdentityServerUtils"; import {abbreviateUrl} from "../../../utils/UrlUtils"; import dis from "../../../dispatcher"; import IdentityAuthClient from "../../../IdentityAuthClient"; import Modal from "../../../Modal"; import {humanizeTime} from "../../../utils/humanize"; import createRoom from "../../../createRoom"; import {inviteMultipleToRoom} from "../../../RoomInvite"; import SettingsStore from '../../../settings/SettingsStore'; export const KIND_DM = "dm"; export const KIND_INVITE = "invite"; const INITIAL_ROOMS_SHOWN = 3; // Number of rooms to show at first const INCREMENT_ROOMS_SHOWN = 5; // Number of rooms to add when 'show more' is clicked // This is the interface that is expected by various components in this file. It is a bit // awkward because it also matches the RoomMember class from the js-sdk with some extra support // for 3PIDs/email addresses. // // XXX: We should use TypeScript interfaces instead of this weird "abstract" class. class Member { /** * The display name of this Member. For users this should be their profile's display * name or user ID if none set. For 3PIDs this should be the 3PID address (email). */ get name(): string { throw new Error("Member class not implemented"); } /** * The ID of this Member. For users this should be their user ID. For 3PIDs this should * be the 3PID address (email). */ get userId(): string { throw new Error("Member class not implemented"); } /** * Gets the MXC URL of this Member's avatar. For users this should be their profile's * avatar MXC URL or null if none set. For 3PIDs this should always be null. */ getMxcAvatarUrl(): string { throw new Error("Member class not implemented"); } } class DirectoryMember extends Member { _userId: string; _displayName: string; _avatarUrl: string; constructor(userDirResult: {user_id: string, display_name: string, avatar_url: string}) { super(); this._userId = userDirResult.user_id; this._displayName = userDirResult.display_name; this._avatarUrl = userDirResult.avatar_url; } // These next class members are for the Member interface get name(): string { return this._displayName || this._userId; } get userId(): string { return this._userId; } getMxcAvatarUrl(): string { return this._avatarUrl; } } class ThreepidMember extends Member { _id: string; constructor(id: string) { super(); this._id = id; } // This is a getter that would be falsey on all other implementations. Until we have // better type support in the react-sdk we can use this trick to determine the kind // of 3PID we're dealing with, if any. get isEmail(): boolean { return this._id.includes('@'); } // These next class members are for the Member interface get name(): string { return this._id; } get userId(): string { return this._id; } getMxcAvatarUrl(): string { return null; } } class DMUserTile extends React.PureComponent { static propTypes = { member: PropTypes.object.isRequired, // Should be a Member (see interface above) onRemove: PropTypes.func.isRequired, // takes 1 argument, the member being removed }; _onRemove = (e) => { // Stop the browser from highlighting text e.preventDefault(); e.stopPropagation(); this.props.onRemove(this.props.member); }; render() { const BaseAvatar = sdk.getComponent("views.avatars.BaseAvatar"); const AccessibleButton = sdk.getComponent("elements.AccessibleButton"); const avatarSize = 20; const avatar = this.props.member.isEmail ? : ; return ( {avatar} {this.props.member.name} {_t('Remove')} ); } } class DMRoomTile extends React.PureComponent { static propTypes = { member: PropTypes.object.isRequired, // Should be a Member (see interface above) lastActiveTs: PropTypes.number, onToggle: PropTypes.func.isRequired, // takes 1 argument, the member being toggled highlightWord: PropTypes.string, isSelected: PropTypes.bool, }; _onClick = (e) => { // Stop the browser from highlighting text e.preventDefault(); e.stopPropagation(); this.props.onToggle(this.props.member); }; _highlightName(str: string) { if (!this.props.highlightWord) return str; // We convert things to lowercase for index searching, but pull substrings from // the submitted text to preserve case. Note: we don't need to htmlEntities the // string because React will safely encode the text for us. const lowerStr = str.toLowerCase(); const filterStr = this.props.highlightWord.toLowerCase(); const result = []; let i = 0; let ii; while ((ii = lowerStr.indexOf(filterStr, i)) >= 0) { // Push any text we missed (first bit/middle of text) if (ii > i) { // Push any text we aren't highlighting (middle of text match, or beginning of text) result.push({str.substring(i, ii)}); } i = ii; // copy over ii only if we have a match (to preserve i for end-of-text matching) // Highlight the word the user entered const substr = str.substring(i, filterStr.length + i); result.push({substr}); i += substr.length; } // Push any text we missed (end of text) if (i < (str.length - 1)) { result.push({str.substring(i)}); } return result; } render() { const BaseAvatar = sdk.getComponent("views.avatars.BaseAvatar"); let timestamp = null; if (this.props.lastActiveTs) { const humanTs = humanizeTime(this.props.lastActiveTs); timestamp = {humanTs}; } const avatarSize = 36; const avatar = this.props.member.isEmail ? : ; let checkmark = null; if (this.props.isSelected) { // To reduce flickering we put the 'selected' room tile above the real avatar checkmark =
; } // To reduce flickering we put the checkmark on top of the actual avatar (prevents // the browser from reloading the image source when the avatar remounts). const stackedAvatar = ( {avatar} {checkmark} ); return (
{stackedAvatar} {this._highlightName(this.props.member.name)} {this._highlightName(this.props.member.userId)} {timestamp}
); } } export default class InviteDialog extends React.PureComponent { static propTypes = { // Takes an array of user IDs/emails to invite. onFinished: PropTypes.func.isRequired, // The kind of invite being performed. Assumed to be KIND_DM if // not provided. kind: PropTypes.string, // The room ID this dialog is for. Only required for KIND_INVITE. roomId: PropTypes.string, }; static defaultProps = { kind: KIND_DM, }; _debounceTimer: number = null; _editorRef: any = null; constructor(props) { super(props); if (props.kind === KIND_INVITE && !props.roomId) { throw new Error("When using KIND_INVITE a roomId is required for an InviteDialog"); } let alreadyInvited = []; if (props.roomId) { const room = MatrixClientPeg.get().getRoom(props.roomId); if (!room) throw new Error("Room ID given to InviteDialog does not look like a room"); alreadyInvited = [ ...room.getMembersWithMembership('invite'), ...room.getMembersWithMembership('join'), ...room.getMembersWithMembership('ban'), // so we don't try to invite them ].map(m => m.userId); } this.state = { targets: [], // array of Member objects (see interface above) filterText: "", recents: this._buildRecents(alreadyInvited), numRecentsShown: INITIAL_ROOMS_SHOWN, suggestions: this._buildSuggestions(alreadyInvited), numSuggestionsShown: INITIAL_ROOMS_SHOWN, serverResultsMixin: [], // { user: DirectoryMember, userId: string }[], like recents and suggestions threepidResultsMixin: [], // { user: ThreepidMember, userId: string}[], like recents and suggestions canUseIdentityServer: !!MatrixClientPeg.get().getIdentityServerUrl(), tryingIdentityServer: false, // These two flags are used for the 'Go' button to communicate what is going on. busy: false, errorText: null, }; this._editorRef = createRef(); } _buildRecents(excludedTargetIds: string[]): {userId: string, user: RoomMember, lastActive: number} { const rooms = DMRoomMap.shared().getUniqueRoomsWithIndividuals(); const recents = []; for (const userId in rooms) { // Filter out user IDs that are already in the room / should be excluded if (excludedTargetIds.includes(userId)) continue; const room = rooms[userId]; const member = room.getMember(userId); if (!member) continue; // just skip people who don't have memberships for some reason const lastEventTs = room.timeline && room.timeline.length ? room.timeline[room.timeline.length - 1].getTs() : 0; if (!lastEventTs) continue; // something weird is going on with this room recents.push({userId, user: member, lastActive: lastEventTs}); } // Sort the recents by last active to save us time later recents.sort((a, b) => b.lastActive - a.lastActive); return recents; } _buildSuggestions(excludedTargetIds: string[]): {userId: string, user: RoomMember} { const maxConsideredMembers = 200; const client = MatrixClientPeg.get(); const excludedUserIds = [client.getUserId(), SdkConfig.get()['welcomeUserId']]; const joinedRooms = client.getRooms() .filter(r => r.getMyMembership() === 'join') .filter(r => r.getJoinedMemberCount() <= maxConsideredMembers); // Generates { userId: {member, rooms[]} } const memberRooms = joinedRooms.reduce((members, room) => { // Filter out DMs (we'll handle these in the recents section) if (DMRoomMap.shared().getUserIdForRoomId(room.roomId)) { return members; // Do nothing } const joinedMembers = room.getJoinedMembers().filter(u => !excludedUserIds.includes(u.userId)); for (const member of joinedMembers) { // Filter out user IDs that are already in the room / should be excluded if (excludedTargetIds.includes(member.userId)) { continue; } if (!members[member.userId]) { members[member.userId] = { member: member, // Track the room size of the 'picked' member so we can use the profile of // the smallest room (likely a DM). pickedMemberRoomSize: room.getJoinedMemberCount(), rooms: [], }; } members[member.userId].rooms.push(room); if (room.getJoinedMemberCount() < members[member.userId].pickedMemberRoomSize) { members[member.userId].member = member; members[member.userId].pickedMemberRoomSize = room.getJoinedMemberCount(); } } return members; }, {}); // Generates { userId: {member, numRooms, score} } const memberScores = Object.values(memberRooms).reduce((scores, entry) => { const numMembersTotal = entry.rooms.reduce((c, r) => c + r.getJoinedMemberCount(), 0); const maxRange = maxConsideredMembers * entry.rooms.length; scores[entry.member.userId] = { member: entry.member, numRooms: entry.rooms.length, score: Math.max(0, Math.pow(1 - (numMembersTotal / maxRange), 5)), }; return scores; }, {}); // Now that we have scores for being in rooms, boost those people who have sent messages // recently, as a way to improve the quality of suggestions. We do this by checking every // room to see who has sent a message in the last few hours, and giving them a score // which correlates to the freshness of their message. In theory, this results in suggestions // which are closer to "continue this conversation" rather than "this person exists". const trueJoinedRooms = client.getRooms().filter(r => r.getMyMembership() === 'join'); const now = (new Date()).getTime(); const earliestAgeConsidered = now - (60 * 60 * 1000); // 1 hour ago const maxMessagesConsidered = 50; // so we don't iterate over a huge amount of traffic const lastSpoke = {}; // userId: timestamp const lastSpokeMembers = {}; // userId: room member for (const room of trueJoinedRooms) { // Skip low priority rooms and DMs const isDm = DMRoomMap.shared().getUserIdForRoomId(room.roomId); if (Object.keys(room.tags).includes("m.lowpriority") || isDm) { continue; } const events = room.getLiveTimeline().getEvents(); // timelines are most recent last for (let i = events.length - 1; i >= Math.max(0, events.length - maxMessagesConsidered); i--) { const ev = events[i]; if (excludedUserIds.includes(ev.getSender())) { continue; } if (ev.getTs() <= earliestAgeConsidered) { break; // give up: all events from here on out are too old } if (!lastSpoke[ev.getSender()] || lastSpoke[ev.getSender()] < ev.getTs()) { lastSpoke[ev.getSender()] = ev.getTs(); lastSpokeMembers[ev.getSender()] = room.getMember(ev.getSender()); } } } for (const userId in lastSpoke) { const ts = lastSpoke[userId]; const member = lastSpokeMembers[userId]; if (!member) continue; // skip people we somehow don't have profiles for // Scores from being in a room give a 'good' score of about 1.0-1.5, so for our // boost we'll try and award at least +1.0 for making the list, with +4.0 being // an approximate maximum for being selected. const distanceFromNow = Math.abs(now - ts); // abs to account for slight future messages const inverseTime = (now - earliestAgeConsidered) - distanceFromNow; const scoreBoost = Math.max(1, inverseTime / (15 * 60 * 1000)); // 15min segments to keep scores sane let record = memberScores[userId]; if (!record) record = memberScores[userId] = {score: 0}; record.member = member; record.score += scoreBoost; } const members = Object.values(memberScores); members.sort((a, b) => { if (a.score === b.score) { if (a.numRooms === b.numRooms) { return a.member.userId.localeCompare(b.member.userId); } return b.numRooms - a.numRooms; } return b.score - a.score; }); return members.map(m => ({userId: m.member.userId, user: m.member})); } _shouldAbortAfterInviteError(result): boolean { const failedUsers = Object.keys(result.states).filter(a => result.states[a] === 'error'); if (failedUsers.length > 0) { console.log("Failed to invite users: ", result); this.setState({ busy: false, errorText: _t("Failed to invite the following users to chat: %(csvUsers)s", { csvUsers: failedUsers.join(", "), }), }); return true; // abort } return false; } _startDm = async () => { this.setState({busy: true}); const targetIds = this.state.targets.map(t => t.userId); // Check if there is already a DM with these people and reuse it if possible. const existingRoom = DMRoomMap.shared().getDMRoomForIdentifiers(targetIds); if (existingRoom) { dis.dispatch({ action: 'view_room', room_id: existingRoom.roomId, should_peek: false, joining: false, }); this.props.onFinished(); return; } const createRoomOptions = {}; if (SettingsStore.isFeatureEnabled("feature_cross_signing")) { // Check whether all users have uploaded device keys before. // If so, enable encryption in the new room. const client = MatrixClientPeg.get(); const usersToDevicesMap = await client.downloadKeys(targetIds); const allHaveDeviceKeys = Object.values(usersToDevicesMap).every(devices => { // `devices` is an object of the form { deviceId: deviceInfo, ... }. return Object.keys(devices).length > 0; }); if (allHaveDeviceKeys) { createRoomOptions.encryption = true; } } // Check if it's a traditional DM and create the room if required. // TODO: [Canonical DMs] Remove this check and instead just create the multi-person DM let createRoomPromise = Promise.resolve(); if (targetIds.length === 1) { createRoomOptions.dmUserId = targetIds[0]; createRoomPromise = createRoom(createRoomOptions); } else { // Create a boring room and try to invite the targets manually. createRoomPromise = createRoom(createRoomOptions).then(roomId => { return inviteMultipleToRoom(roomId, targetIds); }).then(result => { if (this._shouldAbortAfterInviteError(result)) { return true; // abort } }); } // the createRoom call will show the room for us, so we don't need to worry about that. createRoomPromise.then(abort => { if (abort === true) return; // only abort on true booleans, not roomIds or something this.props.onFinished(); }).catch(err => { console.error(err); this.setState({ busy: false, errorText: _t("We couldn't create your DM. Please check the users you want to invite and try again."), }); }); }; _inviteUsers = () => { this.setState({busy: true}); const targetIds = this.state.targets.map(t => t.userId); const room = MatrixClientPeg.get().getRoom(this.props.roomId); if (!room) { console.error("Failed to find the room to invite users to"); this.setState({ busy: false, errorText: _t("Something went wrong trying to invite the users."), }); return; } inviteMultipleToRoom(this.props.roomId, targetIds).then(result => { if (!this._shouldAbortAfterInviteError(result)) { // handles setting error message too this.props.onFinished(); } }).catch(err => { console.error(err); this.setState({ busy: false, errorText: _t( "We couldn't invite those users. Please check the users you want to invite and try again.", ), }); }); }; _cancel = () => { // We do not want the user to close the dialog while an action is in progress if (this.state.busy) return; this.props.onFinished(); }; _updateFilter = (e) => { const term = e.target.value; this.setState({filterText: term}); // Debounce server lookups to reduce spam. We don't clear the existing server // results because they might still be vaguely accurate, likewise for races which // could happen here. if (this._debounceTimer) { clearTimeout(this._debounceTimer); } this._debounceTimer = setTimeout(async () => { MatrixClientPeg.get().searchUserDirectory({term}).then(async r => { if (term !== this.state.filterText) { // Discard the results - we were probably too slow on the server-side to make // these results useful. This is a race we want to avoid because we could overwrite // more accurate results. return; } if (!r.results) r.results = []; // While we're here, try and autocomplete a search result for the mxid itself // if there's no matches (and the input looks like a mxid). if (term[0] === '@' && term.indexOf(':') > 1 && r.results.length === 0) { try { const profile = await MatrixClientPeg.get().getProfileInfo(term); if (profile) { // If we have a profile, we have enough information to assume that // the mxid can be invited - add it to the list r.results.push({ user_id: term, display_name: profile['displayname'], avatar_url: profile['avatar_url'], }); } } catch (e) { console.warn("Non-fatal error trying to make an invite for a user ID"); console.warn(e); } } this.setState({ serverResultsMixin: r.results.map(u => ({ userId: u.user_id, user: new DirectoryMember(u), })), }); }).catch(e => { console.error("Error searching user directory:"); console.error(e); this.setState({serverResultsMixin: []}); // clear results because it's moderately fatal }); // Whenever we search the directory, also try to search the identity server. It's // all debounced the same anyways. if (!this.state.canUseIdentityServer) { // The user doesn't have an identity server set - warn them of that. this.setState({tryingIdentityServer: true}); return; } if (term.indexOf('@') > 0 && Email.looksValid(term)) { // Start off by suggesting the plain email while we try and resolve it // to a real account. this.setState({ // per above: the userId is a lie here - it's just a regular identifier threepidResultsMixin: [{user: new ThreepidMember(term), userId: term}], }); try { const authClient = new IdentityAuthClient(); const token = await authClient.getAccessToken(); if (term !== this.state.filterText) return; // abandon hope const lookup = await MatrixClientPeg.get().lookupThreePid( 'email', term, undefined, // callback token, ); if (term !== this.state.filterText) return; // abandon hope if (!lookup || !lookup.mxid) { // We weren't able to find anyone - we're already suggesting the plain email // as an alternative, so do nothing. return; } // We append the user suggestion to give the user an option to click // the email anyways, and so we don't cause things to jump around. In // theory, the user would see the user pop up and think "ah yes, that // person!" const profile = await MatrixClientPeg.get().getProfileInfo(lookup.mxid); if (term !== this.state.filterText || !profile) return; // abandon hope this.setState({ threepidResultsMixin: [...this.state.threepidResultsMixin, { user: new DirectoryMember({ user_id: lookup.mxid, display_name: profile.displayname, avatar_url: profile.avatar_url, }), userId: lookup.mxid, }], }); } catch (e) { console.error("Error searching identity server:"); console.error(e); this.setState({threepidResultsMixin: []}); // clear results because it's moderately fatal } } }, 150); // 150ms debounce (human reaction time + some) }; _showMoreRecents = () => { this.setState({numRecentsShown: this.state.numRecentsShown + INCREMENT_ROOMS_SHOWN}); }; _showMoreSuggestions = () => { this.setState({numSuggestionsShown: this.state.numSuggestionsShown + INCREMENT_ROOMS_SHOWN}); }; _toggleMember = (member: Member) => { const targets = this.state.targets.map(t => t); // cheap clone for mutation const idx = targets.indexOf(member); if (idx >= 0) targets.splice(idx, 1); else targets.push(member); this.setState({targets}); }; _removeMember = (member: Member) => { const targets = this.state.targets.map(t => t); // cheap clone for mutation const idx = targets.indexOf(member); if (idx >= 0) { targets.splice(idx, 1); this.setState({targets}); } }; _onPaste = async (e) => { // Prevent the text being pasted into the textarea e.preventDefault(); // Process it as a list of addresses to add instead const text = e.clipboardData.getData("text"); const possibleMembers = [ // If we can avoid hitting the profile endpoint, we should. ...this.state.recents, ...this.state.suggestions, ...this.state.serverResultsMixin, ...this.state.threepidResultsMixin, ]; const toAdd = []; const failed = []; const potentialAddresses = text.split(/[\s,]+/); for (const address of potentialAddresses) { const member = possibleMembers.find(m => m.userId === address); if (member) { toAdd.push(member.user); continue; } if (address.indexOf('@') > 0 && Email.looksValid(address)) { toAdd.push(new ThreepidMember(address)); continue; } if (address[0] !== '@') { failed.push(address); // not a user ID continue; } try { const profile = await MatrixClientPeg.get().getProfileInfo(address); const displayName = profile ? profile.displayname : null; const avatarUrl = profile ? profile.avatar_url : null; toAdd.push(new DirectoryMember({ user_id: address, display_name: displayName, avatar_url: avatarUrl, })); } catch (e) { console.error("Error looking up profile for " + address); console.error(e); failed.push(address); } } if (failed.length > 0) { const QuestionDialog = sdk.getComponent('dialogs.QuestionDialog'); Modal.createTrackedDialog('Invite Paste Fail', '', QuestionDialog, { title: _t('Failed to find the following users'), description: _t( "The following users might not exist or are invalid, and cannot be invited: %(csvNames)s", {csvNames: failed.join(", ")}, ), button: _t('OK'), }); } this.setState({targets: [...this.state.targets, ...toAdd]}); }; _onClickInputArea = (e) => { // Stop the browser from highlighting text e.preventDefault(); e.stopPropagation(); if (this._editorRef && this._editorRef.current) { this._editorRef.current.focus(); } }; _onUseDefaultIdentityServerClick = (e) => { e.preventDefault(); // Update the IS in account data. Actually using it may trigger terms. // eslint-disable-next-line react-hooks/rules-of-hooks useDefaultIdentityServer(); this.setState({canUseIdentityServer: true, tryingIdentityServer: false}); }; _onManageSettingsClick = (e) => { e.preventDefault(); dis.dispatch({ action: 'view_user_settings' }); this._cancel(); }; _renderSection(kind: "recents"|"suggestions") { let sourceMembers = kind === 'recents' ? this.state.recents : this.state.suggestions; let showNum = kind === 'recents' ? this.state.numRecentsShown : this.state.numSuggestionsShown; const showMoreFn = kind === 'recents' ? this._showMoreRecents.bind(this) : this._showMoreSuggestions.bind(this); const lastActive = (m) => kind === 'recents' ? m.lastActive : null; let sectionName = kind === 'recents' ? _t("Recent Conversations") : _t("Suggestions"); if (this.props.kind === KIND_INVITE) { sectionName = kind === 'recents' ? _t("Recently Direct Messaged") : _t("Suggestions"); } // Mix in the server results if we have any, but only if we're searching. We track the additional // members separately because we want to filter sourceMembers but trust the mixin arrays to have // the right members in them. let additionalMembers = []; const hasMixins = this.state.serverResultsMixin || this.state.threepidResultsMixin; if (this.state.filterText && hasMixins && kind === 'suggestions') { // We don't want to duplicate members though, so just exclude anyone we've already seen. const notAlreadyExists = (u: Member): boolean => { return !sourceMembers.some(m => m.userId === u.userId) && !additionalMembers.some(m => m.userId === u.userId); }; const uniqueServerResults = this.state.serverResultsMixin.filter(notAlreadyExists); additionalMembers = additionalMembers.concat(...uniqueServerResults); const uniqueThreepidResults = this.state.threepidResultsMixin.filter(notAlreadyExists); additionalMembers = additionalMembers.concat(...uniqueThreepidResults); } // Hide the section if there's nothing to filter by if (sourceMembers.length === 0 && additionalMembers.length === 0) return null; // Do some simple filtering on the input before going much further. If we get no results, say so. if (this.state.filterText) { const filterBy = this.state.filterText.toLowerCase(); sourceMembers = sourceMembers .filter(m => m.user.name.toLowerCase().includes(filterBy) || m.userId.toLowerCase().includes(filterBy)); if (sourceMembers.length === 0 && additionalMembers.length === 0) { return (

{sectionName}

{_t("No results")}

); } } // Now we mix in the additional members. Again, we presume these have already been filtered. We // also assume they are more relevant than our suggestions and prepend them to the list. sourceMembers = [...additionalMembers, ...sourceMembers]; // If we're going to hide one member behind 'show more', just use up the space of the button // with the member's tile instead. if (showNum === sourceMembers.length - 1) showNum++; // .slice() will return an incomplete array but won't error on us if we go too far const toRender = sourceMembers.slice(0, showNum); const hasMore = toRender.length < sourceMembers.length; const AccessibleButton = sdk.getComponent("elements.AccessibleButton"); let showMore = null; if (hasMore) { showMore = ( {_t("Show more")} ); } const tiles = toRender.map(r => ( t.userId === r.userId)} /> )); return (

{sectionName}

{tiles} {showMore}
); } _renderEditor() { const targets = this.state.targets.map(t => ( )); const input = (