2015-12-22 10:00:30 +00:00
|
|
|
/*
|
2016-01-07 04:06:39 +00:00
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
2015-12-22 10:00:30 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
var React = require("react");
|
|
|
|
var sdk = require("./index");
|
|
|
|
|
|
|
|
class Entry {
|
|
|
|
constructor(text) {
|
|
|
|
this.text = text;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return {string} The text to display in this entry.
|
|
|
|
*/
|
|
|
|
getText() {
|
|
|
|
return this.text;
|
|
|
|
}
|
|
|
|
|
2016-01-14 14:39:58 +00:00
|
|
|
/**
|
|
|
|
* @return {string} The text to insert into the input box. Most of the time
|
|
|
|
* this is the same as getText().
|
|
|
|
*/
|
|
|
|
getFillText() {
|
|
|
|
return this.text;
|
|
|
|
}
|
|
|
|
|
2015-12-22 10:00:30 +00:00
|
|
|
/**
|
|
|
|
* @return {ReactClass} Raw JSX
|
|
|
|
*/
|
|
|
|
getImageJsx() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return {?string} The unique key= prop for React dedupe
|
|
|
|
*/
|
|
|
|
getKey() {
|
|
|
|
return null;
|
|
|
|
}
|
2015-12-22 11:14:36 +00:00
|
|
|
|
2016-01-13 17:46:36 +00:00
|
|
|
/**
|
2016-01-14 11:39:24 +00:00
|
|
|
* @return {?string} The suffix to append to the tab-complete, or null to
|
2016-01-13 17:46:36 +00:00
|
|
|
* not do this.
|
|
|
|
*/
|
2016-01-14 11:39:24 +00:00
|
|
|
getSuffix(isFirstWord) {
|
2016-01-13 17:46:36 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2015-12-22 11:14:36 +00:00
|
|
|
/**
|
|
|
|
* Called when this entry is clicked.
|
|
|
|
*/
|
|
|
|
onClick() {
|
|
|
|
// NOP
|
|
|
|
}
|
2015-12-22 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
2016-01-13 17:46:36 +00:00
|
|
|
class CommandEntry extends Entry {
|
2016-01-14 14:39:58 +00:00
|
|
|
constructor(cmd, cmdWithArgs) {
|
|
|
|
super(cmdWithArgs);
|
|
|
|
this.cmd = cmd;
|
|
|
|
}
|
|
|
|
|
|
|
|
getFillText() {
|
|
|
|
return this.cmd;
|
2016-01-13 17:46:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
getKey() {
|
2016-01-14 14:39:58 +00:00
|
|
|
return this.getFillText();
|
2016-01-13 17:46:36 +00:00
|
|
|
}
|
|
|
|
|
2016-01-14 11:39:24 +00:00
|
|
|
getSuffix(isFirstWord) {
|
2016-01-13 17:46:36 +00:00
|
|
|
return " "; // force a space after the command.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-14 14:39:58 +00:00
|
|
|
CommandEntry.fromCommands = function(commandArray) {
|
2016-01-13 17:46:36 +00:00
|
|
|
return commandArray.map(function(cmd) {
|
2016-01-14 14:39:58 +00:00
|
|
|
return new CommandEntry(cmd.getCommand(), cmd.getCommandWithArgs());
|
2016-01-13 17:46:36 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2015-12-22 10:00:30 +00:00
|
|
|
class MemberEntry extends Entry {
|
|
|
|
constructor(member) {
|
|
|
|
super(member.name || member.userId);
|
|
|
|
this.member = member;
|
|
|
|
}
|
|
|
|
|
|
|
|
getImageJsx() {
|
|
|
|
var MemberAvatar = sdk.getComponent("views.avatars.MemberAvatar");
|
|
|
|
return (
|
|
|
|
<MemberAvatar member={this.member} width={24} height={24} />
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
getKey() {
|
|
|
|
return this.member.userId;
|
|
|
|
}
|
2016-01-14 11:39:24 +00:00
|
|
|
|
|
|
|
getSuffix(isFirstWord) {
|
|
|
|
return isFirstWord ? ": " : " ";
|
|
|
|
}
|
2015-12-22 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 10:25:45 +00:00
|
|
|
MemberEntry.fromMemberList = function(room, members) {
|
|
|
|
// build up a dict of when, in the history we have cached,
|
|
|
|
// each member last spoke
|
|
|
|
const lastSpoke = {};
|
|
|
|
const timelineEvents = room.getLiveTimeline().getEvents();
|
2016-07-14 13:06:31 +00:00
|
|
|
for (const ev of room.getLiveTimeline().getEvents()) {
|
2016-07-14 10:40:17 +00:00
|
|
|
lastSpoke[ev.getSender()] = ev.getTs();
|
2016-07-14 10:25:45 +00:00
|
|
|
}
|
|
|
|
|
2015-12-22 15:38:23 +00:00
|
|
|
return members.sort(function(a, b) {
|
2016-07-14 10:25:45 +00:00
|
|
|
const lastSpokeA = lastSpoke[a.userId] || 0;
|
|
|
|
const lastSpokeB = lastSpoke[b.userId] || 0;
|
|
|
|
|
|
|
|
if (lastSpokeA != lastSpokeB) {
|
|
|
|
// B - A here because the highest value
|
|
|
|
// is most recent
|
|
|
|
return lastSpokeB - lastSpokeA;
|
|
|
|
}
|
|
|
|
|
2015-12-22 15:38:23 +00:00
|
|
|
var userA = a.user;
|
|
|
|
var userB = b.user;
|
|
|
|
if (userA && !userB) {
|
|
|
|
return -1; // a comes first
|
|
|
|
}
|
|
|
|
else if (!userA && userB) {
|
|
|
|
return 1; // b comes first
|
|
|
|
}
|
|
|
|
else if (!userA && !userB) {
|
|
|
|
return 0; // don't care
|
|
|
|
}
|
|
|
|
else { // both User objects exist
|
2016-03-16 22:27:09 +00:00
|
|
|
var lastActiveAgoA = userA.lastActiveAgo || Number.MAX_SAFE_INTEGER;
|
|
|
|
var lastActiveAgoB = userB.lastActiveAgo || Number.MAX_SAFE_INTEGER;
|
|
|
|
return lastActiveAgoA - lastActiveAgoB;
|
2015-12-22 15:38:23 +00:00
|
|
|
}
|
|
|
|
}).map(function(m) {
|
|
|
|
return new MemberEntry(m);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2015-12-22 10:00:30 +00:00
|
|
|
module.exports.Entry = Entry;
|
|
|
|
module.exports.MemberEntry = MemberEntry;
|
2016-01-13 17:46:36 +00:00
|
|
|
module.exports.CommandEntry = CommandEntry;
|