fchat-rising/fchat/channels.ts

273 lines
11 KiB
TypeScript
Raw Normal View History

2017-09-02 01:50:31 +00:00
import {decodeHTML} from './common';
import {Channel as Interfaces, Character, Connection} from './interfaces';
interface SortableMember extends Interfaces.Member {
rank: Interfaces.Rank,
key: string
}
2017-09-02 01:50:31 +00:00
export function queuedJoin(this: void, channels: string[]): void {
const timer: NodeJS.Timer = setInterval(() => {
const channel = channels.shift();
if(channel === undefined) return clearInterval(timer);
state.join(channel);
}, 100);
}
function sortMember(this: void | never, array: SortableMember[], member: SortableMember): void {
2017-09-02 01:50:31 +00:00
let i = 0;
for(; i < array.length; ++i) {
const other = array[i];
if(other.character.isChatOp && !member.character.isChatOp) continue;
if(member.character.isChatOp && !other.character.isChatOp) break;
if(other.rank > member.rank) continue;
if(member.rank > other.rank) break;
2017-10-18 23:29:28 +00:00
if(other.character.isFriend && !member.character.isFriend) continue;
if(member.character.isFriend && !other.character.isFriend) break;
if(other.character.isBookmarked && !member.character.isBookmarked) continue;
if(member.character.isBookmarked && !other.character.isBookmarked) break;
if(member.key < other.key) break;
2017-09-02 01:50:31 +00:00
}
array.splice(i, 0, member);
}
class Channel implements Interfaces.Channel {
description = '';
opList: string[];
owner = '';
mode: Interfaces.Mode = 'both';
members: {[key: string]: SortableMember | undefined} = {};
sortedMembers: SortableMember[] = [];
2017-09-02 01:50:31 +00:00
constructor(readonly id: string, readonly name: string) {
}
addMember(member: SortableMember): void {
2017-09-02 01:50:31 +00:00
this.members[member.character.name] = member;
sortMember(this.sortedMembers, member);
2017-10-18 23:29:28 +00:00
for(const handler of state.handlers) handler('join', this, member);
2017-09-02 01:50:31 +00:00
}
removeMember(name: string): void {
const member = this.members[name];
if(member !== undefined) {
delete this.members[name];
this.sortedMembers.splice(this.sortedMembers.indexOf(member), 1);
2017-10-18 23:29:28 +00:00
for(const handler of state.handlers) handler('leave', this, member);
2017-09-02 01:50:31 +00:00
}
}
reSortMember(member: SortableMember): void {
2017-09-02 01:50:31 +00:00
this.sortedMembers.splice(this.sortedMembers.indexOf(member), 1);
sortMember(this.sortedMembers, member);
}
createMember(character: Character): SortableMember {
2017-09-02 01:50:31 +00:00
return {
character,
rank: this.owner === character.name ? Interfaces.Rank.Owner :
this.opList.indexOf(character.name) !== -1 ? Interfaces.Rank.Op : Interfaces.Rank.Member,
key: character.name.toLowerCase()
2017-09-02 01:50:31 +00:00
};
}
}
class ListItem implements Interfaces.ListItem {
isJoined = false;
constructor(readonly id: string, readonly name: string, public memberCount: number) {
}
}
class State implements Interfaces.State {
officialChannels: {readonly [key: string]: ListItem | undefined} = {};
openRooms: {readonly [key: string]: ListItem | undefined} = {};
joinedChannels: Channel[] = [];
2017-10-16 23:58:57 +00:00
joinedMap: {[key: string]: Channel | undefined} = {};
2017-09-02 01:50:31 +00:00
handlers: Interfaces.EventHandler[] = [];
constructor(private connection: Connection) {
}
join(channel: string): void {
this.connection.send('JCH', {channel});
}
leave(channel: string): void {
this.connection.send('LCH', {channel});
}
getChannelItem(id: string): ListItem | undefined {
id = id.toLowerCase();
return (id.substr(0, 4) === 'adh-' ? this.openRooms : this.officialChannels)[id];
}
onEvent(handler: Interfaces.EventHandler): void {
this.handlers.push(handler);
}
getChannel(id: string): Channel | undefined {
2017-10-16 23:58:57 +00:00
return this.joinedMap[id.toLowerCase()];
2017-09-02 01:50:31 +00:00
}
}
let state: State;
export default function(this: void, connection: Connection, characters: Character.State): Interfaces.State {
state = new State(connection);
let getChannelTimer: NodeJS.Timer | undefined;
2017-10-18 23:29:28 +00:00
let rejoin: string[] | undefined;
connection.onEvent('connecting', (isReconnect) => {
if(isReconnect && rejoin === undefined) rejoin = Object.keys(state.joinedMap);
2017-09-02 01:50:31 +00:00
state.joinedChannels = [];
2017-10-16 23:58:57 +00:00
state.joinedMap = {};
2017-09-02 01:50:31 +00:00
});
2017-10-18 23:29:28 +00:00
connection.onEvent('connected', () => {
if(rejoin !== undefined) {
queuedJoin(rejoin);
rejoin = undefined;
}
2017-09-02 01:50:31 +00:00
const getChannels = () => {
connection.send('CHA');
connection.send('ORS');
};
getChannels();
if(getChannelTimer !== undefined) clearInterval(getChannelTimer);
getChannelTimer = setInterval(getChannels, 60000);
});
2017-10-16 23:58:57 +00:00
connection.onEvent('closed', () => {
if(getChannelTimer !== undefined) clearInterval(getChannelTimer);
});
2017-09-02 01:50:31 +00:00
connection.onMessage('CHA', (data) => {
const channels: {[key: string]: ListItem} = {};
for(const channel of data.channels) {
const id = channel.name.toLowerCase();
const item = new ListItem(id, channel.name, channel.characters);
2017-10-16 23:58:57 +00:00
if(state.joinedMap[id] !== undefined) item.isJoined = true;
2017-09-02 01:50:31 +00:00
channels[id] = item;
}
state.officialChannels = channels;
});
connection.onMessage('ORS', (data) => {
const channels: {[key: string]: ListItem} = {};
for(const channel of data.channels) {
const id = channel.name.toLowerCase();
const item = new ListItem(id, decodeHTML(channel.title), channel.characters);
2017-10-16 23:58:57 +00:00
if(state.joinedMap[id] !== undefined) item.isJoined = true;
2017-09-02 01:50:31 +00:00
channels[id] = item;
}
state.openRooms = channels;
});
connection.onMessage('JCH', (data) => {
const item = state.getChannelItem(data.channel);
if(data.character.identity === connection.character) {
2017-10-16 23:58:57 +00:00
const id = data.channel.toLowerCase();
const channel = state.joinedMap[id] = new Channel(id, decodeHTML(data.title));
state.joinedChannels.push(channel);
2017-09-02 01:50:31 +00:00
if(item !== undefined) item.isJoined = true;
} else {
const channel = state.getChannel(data.channel);
if(channel === undefined) return state.leave(data.channel);
2017-10-18 23:29:28 +00:00
const member = channel.createMember(characters.get(data.character.identity));
channel.addMember(member);
2017-09-02 01:50:31 +00:00
if(item !== undefined) item.memberCount++;
}
});
connection.onMessage('ICH', (data) => {
const channel = state.getChannel(data.channel);
if(channel === undefined) return state.leave(data.channel);
2017-09-02 01:50:31 +00:00
channel.mode = data.mode;
const members: {[key: string]: SortableMember} = {};
const sorted: SortableMember[] = [];
2017-09-02 01:50:31 +00:00
for(const user of data.users) {
const name = user.identity;
const member = channel.createMember(characters.get(name));
members[name] = member;
sortMember(sorted, member);
}
channel.members = members;
channel.sortedMembers = sorted;
const item = state.getChannelItem(data.channel);
if(item !== undefined) item.memberCount = data.users.length;
2017-10-16 23:58:57 +00:00
for(const handler of state.handlers) handler('join', channel);
2017-09-02 01:50:31 +00:00
});
connection.onMessage('CDS', (data) => {
const channel = state.getChannel(data.channel);
if(channel === undefined) return state.leave(data.channel);
channel.description = decodeHTML(data.description);
});
2017-09-02 01:50:31 +00:00
connection.onMessage('LCH', (data) => {
const channel = state.getChannel(data.channel);
if(channel === undefined) return;
const item = state.getChannelItem(data.channel);
if(data.character === connection.character) {
2017-10-16 23:58:57 +00:00
state.joinedChannels.splice(state.joinedChannels.indexOf(channel), 1);
delete state.joinedMap[channel.id];
for(const handler of state.handlers) handler('leave', channel);
2017-09-02 01:50:31 +00:00
if(item !== undefined) item.isJoined = false;
} else {
channel.removeMember(data.character);
if(item !== undefined) item.memberCount--;
}
});
connection.onMessage('COA', (data) => {
const channel = state.getChannel(data.channel);
if(channel === undefined) return state.leave(data.channel);
2017-09-02 01:50:31 +00:00
channel.opList.push(data.character);
const member = channel.members[data.character];
if(member === undefined || member.rank === Interfaces.Rank.Owner) return;
member.rank = Interfaces.Rank.Op;
channel.reSortMember(member);
});
connection.onMessage('COL', (data) => {
const channel = state.getChannel(data.channel);
if(channel === undefined) return state.leave(data.channel);
2017-09-02 01:50:31 +00:00
channel.owner = data.oplist[0];
channel.opList = data.oplist.slice(1);
});
connection.onMessage('COR', (data) => {
const channel = state.getChannel(data.channel);
if(channel === undefined) return state.leave(data.channel);
2017-09-02 01:50:31 +00:00
channel.opList.splice(channel.opList.indexOf(data.character), 1);
const member = channel.members[data.character];
if(member === undefined || member.rank === Interfaces.Rank.Owner) return;
member.rank = Interfaces.Rank.Member;
channel.reSortMember(member);
});
connection.onMessage('CSO', (data) => {
const channel = state.getChannel(data.channel);
if(channel === undefined) return state.leave(data.channel);
2017-09-02 01:50:31 +00:00
const oldOwner = channel.members[channel.owner];
if(oldOwner !== undefined) {
oldOwner.rank = Interfaces.Rank.Member;
channel.reSortMember(oldOwner);
}
channel.owner = data.character;
const newOwner = channel.members[data.character];
if(newOwner !== undefined) {
newOwner.rank = Interfaces.Rank.Owner;
channel.reSortMember(newOwner);
}
});
connection.onMessage('RMO', (data) => {
const channel = state.getChannel(data.channel);
if(channel === undefined) return state.leave(data.channel);
channel.mode = data.mode;
});
2017-09-02 01:50:31 +00:00
connection.onMessage('FLN', (data) => {
2017-10-16 23:58:57 +00:00
for(const key in state.joinedMap)
state.joinedMap[key]!.removeMember(data.character);
2017-09-02 01:50:31 +00:00
});
const globalHandler = (data: Connection.ServerCommands['AOP'] | Connection.ServerCommands['DOP']) => {
//tslint:disable-next-line:forin
2017-10-16 23:58:57 +00:00
for(const key in state.joinedMap) {
const channel = state.joinedMap[key]!;
2017-09-02 01:50:31 +00:00
const member = channel.members[data.character];
if(member !== undefined) channel.reSortMember(member);
}
};
connection.onMessage('AOP', globalHandler);
connection.onMessage('DOP', globalHandler);
return state;
}