fchat-rising/fchat/characters.ts

196 lines
7.9 KiB
TypeScript
Raw Normal View History

2019-07-07 01:37:15 +00:00
import core from '../chat/core';
import { methods } from '../site/character_page/data_store';
2017-09-02 01:50:31 +00:00
import {decodeHTML} from './common';
import {Character as Interfaces, Connection} from './interfaces';
2019-07-07 01:37:15 +00:00
import { Character as CharacterProfile } from '../site/character_page/interfaces';
2024-01-30 23:59:55 +00:00
import Vue from 'vue';
2017-09-02 01:50:31 +00:00
class Character implements Interfaces.Character {
gender: Interfaces.Gender = 'None';
2017-09-02 01:50:31 +00:00
status: Interfaces.Status = 'offline';
statusText = '';
isFriend = false;
isBookmarked = false;
isChatOp = false;
isIgnored = false;
2024-01-27 03:45:59 +00:00
overrides: CharacterOverrides = {};
2017-09-02 01:50:31 +00:00
constructor(public name: string) {
2017-09-02 01:50:31 +00:00
}
}
2024-01-27 03:45:59 +00:00
export interface CharacterOverrides {
avatarUrl?: string;
gender?: Interfaces.Gender;
status?: Interfaces.Status;
}
2017-09-02 01:50:31 +00:00
class State implements Interfaces.State {
characters: {[key: string]: Character | undefined} = {};
2019-07-07 01:37:15 +00:00
2017-09-02 01:50:31 +00:00
ownCharacter: Character = <any>undefined; /*tslint:disable-line:no-any*///hack
2019-07-07 01:37:15 +00:00
ownProfile: CharacterProfile = <any>undefined; /*tslint:disable-line:no-any*///hack
2017-09-02 01:50:31 +00:00
friends: Character[] = [];
bookmarks: Character[] = [];
ignoreList: string[] = [];
opList: string[] = [];
friendList: string[] = [];
bookmarkList: string[] = [];
get(name: string): Character {
const key = name.toLowerCase();
let char = this.characters[key];
if(char === undefined) {
char = new Character(name);
char.isFriend = this.friendList.indexOf(name) !== -1;
char.isBookmarked = this.bookmarkList.indexOf(name) !== -1;
char.isChatOp = this.opList.indexOf(name) !== -1;
char.isIgnored = this.ignoreList.indexOf(key) !== -1;
this.characters[key] = char;
}
return char;
}
setStatus(character: Character, status: Interfaces.Status, text: string): void {
if(character.status === 'offline' && status !== 'offline') {
if(character.isFriend) this.friends.push(character);
if(character.isBookmarked) this.bookmarks.push(character);
} else if(status === 'offline' && character.status !== 'offline') {
if(character.isFriend) this.friends.splice(this.friends.indexOf(character), 1);
if(character.isBookmarked) this.bookmarks.splice(this.bookmarks.indexOf(character), 1);
}
character.status = status;
character.statusText = decodeHTML(text);
}
2019-07-07 01:37:15 +00:00
2024-01-27 03:45:59 +00:00
setOverride(name: string, type: 'avatarUrl', value: string | undefined): void;
setOverride(name: string, type: 'gender', value: Interfaces.Gender | undefined): void;
setOverride(name: string, type: 'status', value: Interfaces.Status | undefined): void;
setOverride(name: string, type: keyof CharacterOverrides, value: any): void {
const char = this.get(name);
2024-01-30 23:59:55 +00:00
Vue.set(char.overrides, type, value);
2024-01-27 03:45:59 +00:00
}
2019-07-07 01:37:15 +00:00
async resolveOwnProfile(): Promise<void> {
await methods.fieldsGet();
this.ownProfile = await methods.characterData(this.ownCharacter.name, -1, false);
}
2017-09-02 01:50:31 +00:00
}
let state: State;
export default function(this: void, connection: Connection): Interfaces.State {
state = new State();
let reconnectStatus: Connection.ClientCommands['STA'];
connection.onEvent('connecting', async(isReconnect) => {
state.friends = [];
state.bookmarks = [];
2019-01-03 17:38:17 +00:00
state.bookmarkList = (await connection.queryApi<{characters: string[]}>('bookmark-list.php')).characters;
state.friendList = (await connection.queryApi<{friends: {source: string, dest: string, last_online: number}[]}>('friend-list.php'))
.friends.map((x) => x.dest);
2018-03-28 13:51:05 +00:00
if(isReconnect && (<Character | undefined>state.ownCharacter) !== undefined)
reconnectStatus = {status: state.ownCharacter.status, statusmsg: state.ownCharacter.statusText};
2017-09-02 01:50:31 +00:00
for(const key in state.characters) {
const character = state.characters[key]!;
character.isFriend = state.friendList.indexOf(character.name) !== -1;
character.isBookmarked = state.bookmarkList.indexOf(character.name) !== -1;
character.status = 'offline';
character.statusText = '';
}
});
connection.onEvent('connected', async(isReconnect) => {
if(!isReconnect) return;
connection.send('STA', reconnectStatus);
for(const key in state.characters) {
const char = state.characters[key]!;
char.isIgnored = state.ignoreList.indexOf(key) !== -1;
char.isChatOp = state.opList.indexOf(char.name) !== -1;
}
});
connection.onMessage('IGN', (data) => {
switch(data.action) {
case 'init':
state.ignoreList = data.characters.slice();
break;
case 'add':
state.ignoreList.push(data.character.toLowerCase());
state.get(data.character).isIgnored = true;
break;
case 'delete':
state.ignoreList.splice(state.ignoreList.indexOf(data.character.toLowerCase()), 1);
state.get(data.character).isIgnored = false;
}
});
2018-01-06 16:14:21 +00:00
connection.onMessage('ADL', (data) => {
state.opList = data.ops.slice();
});
2017-09-02 01:50:31 +00:00
connection.onMessage('LIS', (data) => {
for(const char of data.characters) {
const character = state.get(char[0]);
character.gender = char[1];
state.setStatus(character, char[2], char[3]);
}
});
connection.onMessage('FLN', (data) => {
state.setStatus(state.get(data.character), 'offline', '');
});
2019-07-07 01:37:15 +00:00
connection.onMessage('NLN', async(data) => {
2017-09-02 01:50:31 +00:00
const character = state.get(data.identity);
2019-07-07 01:37:15 +00:00
if(data.identity === connection.character) {
state.ownCharacter = character;
await state.resolveOwnProfile();
// tslint:disable-next-line no-unnecessary-type-assertion
core.cache.setProfile(state.ownProfile as CharacterProfile);
}
character.name = data.identity;
2017-09-02 01:50:31 +00:00
character.gender = data.gender;
state.setStatus(character, data.status, '');
});
connection.onMessage('STA', (data) => {
state.setStatus(state.get(data.character), data.status, data.statusmsg);
});
connection.onMessage('AOP', (data) => {
state.opList.push(data.character);
const char = state.get(data.character);
char.isChatOp = true;
});
connection.onMessage('DOP', (data) => {
state.opList.splice(state.opList.indexOf(data.character), 1);
const char = state.get(data.character);
char.isChatOp = false;
});
connection.onMessage('RTB', (data) => {
2017-10-18 23:29:28 +00:00
if(data.type !== 'trackadd' && data.type !== 'trackrem' && data.type !== 'friendadd' && data.type !== 'friendremove') return;
const character = state.get(data.name);
2017-09-02 01:50:31 +00:00
switch(data.type) {
case 'trackadd':
state.bookmarkList.push(data.name);
2017-10-18 23:29:28 +00:00
character.isBookmarked = true;
if(character.status !== 'offline') state.bookmarks.push(character);
2017-09-02 01:50:31 +00:00
break;
case 'trackrem':
state.bookmarkList.splice(state.bookmarkList.indexOf(data.name), 1);
2017-10-18 23:29:28 +00:00
character.isBookmarked = false;
if(character.status !== 'offline') state.bookmarks.splice(state.bookmarks.indexOf(character), 1);
2017-09-02 01:50:31 +00:00
break;
case 'friendadd':
2018-01-06 16:14:21 +00:00
if(character.isFriend) return;
2017-09-02 01:50:31 +00:00
state.friendList.push(data.name);
2017-10-18 23:29:28 +00:00
character.isFriend = true;
if(character.status !== 'offline') state.friends.push(character);
2017-09-02 01:50:31 +00:00
break;
case 'friendremove':
state.friendList.splice(state.friendList.indexOf(data.name), 1);
2017-10-18 23:29:28 +00:00
character.isFriend = false;
if(character.status !== 'offline') state.friends.splice(state.friends.indexOf(character), 1);
2017-09-02 01:50:31 +00:00
}
});
return state;
}