2017-09-02 01:50:31 +00:00
import { queuedJoin } from '../fchat/channels' ;
import { decodeHTML } from '../fchat/common' ;
2020-06-16 16:27:31 +00:00
// import { CharacterCacheRecord } from '../learn/profile-cache';
2020-03-15 16:23:39 +00:00
import { AdManager } from './ads/ad-manager' ;
2019-06-07 19:31:42 +00:00
import { characterImage , ConversationSettings , EventMessage , Message , messageToString } from './common' ;
2017-09-02 01:50:31 +00:00
import core from './core' ;
2022-01-01 00:06:08 +00:00
import { Channel , Character , Conversation as Interfaces } from './interfaces' ;
2017-09-02 01:50:31 +00:00
import l from './localize' ;
2018-03-28 13:51:05 +00:00
import { CommandContext , isAction , isCommand , isWarn , parse as parseCommand } from './slash_commands' ;
2017-09-02 01:50:31 +00:00
import MessageType = Interfaces . Message . Type ;
2020-03-15 16:23:39 +00:00
import { EventBus } from './preview/event-bus' ;
2020-04-19 17:59:49 +00:00
import throat from 'throat' ;
import Bluebird from 'bluebird' ;
2022-01-01 00:06:08 +00:00
import log from 'electron-log' ;
import isChannel = Interfaces . isChannel ;
2017-10-16 23:58:57 +00:00
2020-03-15 14:02:31 +00:00
function createMessage ( this : any , type : MessageType , sender : Character , text : string , time? : Date ) : Message {
2018-03-28 13:51:05 +00:00
if ( type === MessageType . Message && isAction ( text ) ) {
2017-09-02 01:50:31 +00:00
type = MessageType . Action ;
text = text . substr ( text . charAt ( 4 ) === ' ' ? 4 : 3 ) ;
}
return new Message ( type , sender , text , time ) ;
}
2020-03-15 14:02:31 +00:00
function safeAddMessage ( this : any , messages : Interfaces.Message [ ] , message : Interfaces.Message , max : number ) : void {
2017-09-02 01:50:31 +00:00
if ( messages . length >= max ) messages . shift ( ) ;
messages . push ( message ) ;
}
abstract class Conversation implements Interfaces . Conversation {
abstract enteredText : string ;
abstract readonly name : string ;
messages : Interfaces.Message [ ] = [ ] ;
errorText = '' ;
unread = Interfaces . UnreadState . None ;
lastRead : Interfaces.Message | undefined = undefined ;
infoText = '' ;
abstract readonly maxMessageLength : number | undefined ;
2018-03-04 02:32:26 +00:00
_settings : Interfaces.Settings | undefined ;
2017-09-02 01:50:31 +00:00
protected abstract context : CommandContext ;
2018-04-08 00:22:32 +00:00
protected maxMessages = 50 ;
2021-02-05 01:08:51 +00:00
protected insertCount = 0 ;
2018-03-04 02:32:26 +00:00
protected allMessages : Interfaces.Message [ ] = [ ] ;
2018-04-08 00:22:32 +00:00
readonly reportMessages : Interfaces.Message [ ] = [ ] ;
2017-09-02 01:50:31 +00:00
private lastSent = '' ;
2021-02-05 01:08:51 +00:00
// private loadedMore = false;
2019-06-07 19:31:42 +00:00
adManager : AdManager ;
2017-09-02 01:50:31 +00:00
2022-01-01 00:06:08 +00:00
public static readonly conversationThroat = throat ( 1 ) ; // make sure user posting and ad posting won't get in each others' way
2020-04-19 17:59:49 +00:00
2017-09-02 01:50:31 +00:00
constructor ( readonly key : string , public _isPinned : boolean ) {
2019-06-07 19:31:42 +00:00
this . adManager = new AdManager ( this ) ;
2017-09-02 01:50:31 +00:00
}
get settings ( ) : Interfaces . Settings {
//tslint:disable-next-line:strict-boolean-expressions
return this . _settings || ( this . _settings = state . settings [ this . key ] || new ConversationSettings ( ) ) ;
}
set settings ( value : Interfaces.Settings ) {
this . _settings = value ;
2018-01-06 16:14:21 +00:00
state . setSettings ( this . key , value ) ; //tslint:disable-line:no-floating-promises
2017-09-02 01:50:31 +00:00
}
get isPinned ( ) : boolean {
return this . _isPinned ;
}
set isPinned ( value : boolean ) {
if ( value === this . _isPinned ) return ;
this . _isPinned = value ;
2018-01-06 16:14:21 +00:00
state . savePinned ( ) ; //tslint:disable-line:no-floating-promises
2017-09-02 01:50:31 +00:00
}
2018-08-18 19:37:53 +00:00
clearText ( ) : void {
setImmediate ( ( ) = > this . enteredText = '' ) ;
}
2018-01-06 16:14:21 +00:00
async send ( ) : Promise < void > {
2017-09-02 01:50:31 +00:00
if ( this . enteredText . length === 0 ) return ;
2019-06-02 23:57:32 +00:00
2017-09-02 01:50:31 +00:00
if ( isCommand ( this . enteredText ) ) {
const parsed = parseCommand ( this . enteredText , this . context ) ;
if ( typeof parsed === 'string' ) this . errorText = parsed ;
else {
parsed . call ( this ) ;
this . lastSent = this . enteredText ;
2018-08-18 19:37:53 +00:00
this . clearText ( ) ;
2017-09-02 01:50:31 +00:00
}
} else {
this . lastSent = this . enteredText ;
2018-01-06 16:14:21 +00:00
await this . doSend ( ) ;
2017-09-02 01:50:31 +00:00
}
}
2019-09-17 17:14:14 +00:00
//tslint:disable-next-line:no-async-without-await
2018-01-06 16:14:21 +00:00
abstract async addMessage ( message : Interfaces.Message ) : Promise < void > ;
2017-09-02 01:50:31 +00:00
loadLastSent ( ) : void {
this . enteredText = this . lastSent ;
}
2018-08-10 16:59:37 +00:00
loadMore ( ) : boolean {
if ( this . messages . length >= this . allMessages . length ) return false ;
2018-04-08 00:22:32 +00:00
this . maxMessages += 50 ;
2021-02-05 01:08:51 +00:00
// this.loadedMore = true;
2017-09-02 01:50:31 +00:00
this . messages = this . allMessages . slice ( - this . maxMessages ) ;
2020-06-16 19:18:04 +00:00
EventBus . $emit ( 'conversation-load-more' , { conversation : this } ) ;
2018-08-10 16:59:37 +00:00
return true ;
2017-09-02 01:50:31 +00:00
}
show ( ) : void {
state . show ( this ) ;
}
onHide ( ) : void {
this . errorText = '' ;
this . lastRead = this . messages [ this . messages . length - 1 ] ;
2018-04-08 00:22:32 +00:00
this . maxMessages = 50 ;
2017-09-02 01:50:31 +00:00
this . messages = this . allMessages . slice ( - this . maxMessages ) ;
2021-02-05 01:08:51 +00:00
// this.loadedMore = false;
this . insertCount = 0 ;
}
// Keeps the message-list from re-rendering every time when full, cleaning up after itself every 200 messages
stretch ( ) : void {
if ( ( core . conversations . selectedConversation !== this ) || ( this . messages . length < this . maxMessages ) ) {
return ;
}
if ( this . insertCount < 200 ) {
this . maxMessages += 1 ;
this . insertCount += 1 ;
} else {
const removed = this . insertCount ;
this . maxMessages -= removed ;
this . insertCount = 0 ;
this . messages = this . allMessages . slice ( - this . maxMessages ) ;
log . debug ( 'conversation.view.cleanup' , { channel : this.name , removed , left : this.messages.length , limit : this.maxMessages } ) ;
}
2017-09-02 01:50:31 +00:00
}
2018-04-08 00:22:32 +00:00
clear ( ) : void {
this . allMessages = [ ] ;
this . messages = [ ] ;
}
2017-09-02 01:50:31 +00:00
abstract close ( ) : void ;
protected safeAddMessage ( message : Interfaces.Message ) : void {
2018-04-08 00:22:32 +00:00
safeAddMessage ( this . reportMessages , message , 500 ) ;
2017-09-02 01:50:31 +00:00
safeAddMessage ( this . allMessages , message , 500 ) ;
safeAddMessage ( this . messages , message , this . maxMessages ) ;
}
2020-03-15 14:02:31 +00:00
protected abstract doSend ( ) : void | Promise < void > ;
2020-04-19 17:59:49 +00:00
protected static readonly POST_DELAY = 1250 ;
2022-01-01 00:06:08 +00:00
public static async testPostDelay ( ) : Promise < void > {
2020-04-19 17:59:49 +00:00
const lastPostDelta = Date . now ( ) - core . cache . getLastPost ( ) . getTime ( ) ;
// console.log('Last Post Delta', lastPostDelta, ((lastPostDelta < Conversation.POST_DELAY) && (lastPostDelta > 0)));
if ( ( lastPostDelta < Conversation . POST_DELAY ) && ( lastPostDelta > 0 ) ) {
await Bluebird . delay ( Conversation . POST_DELAY - lastPostDelta ) ;
}
}
2020-10-20 23:19:43 +00:00
isSendingAutomatedAds ( ) : boolean {
return this . adManager . isActive ( ) ;
}
toggleAutomatedAds ( ) : void {
this . adManager . isActive ( ) ? this . adManager . stop ( ) : this . adManager . start ( ) ;
}
hasAutomatedAds ( ) : boolean {
return ( this . adManager . getAds ( ) . length > 0 ) ;
}
2017-09-02 01:50:31 +00:00
}
2020-06-30 00:59:00 +00:00
2017-09-02 01:50:31 +00:00
class PrivateConversation extends Conversation implements Interfaces . PrivateConversation {
readonly name = this . character . name ;
readonly context = CommandContext . Private ;
typingStatus : Interfaces.TypingStatus = 'clear' ;
readonly maxMessageLength = core . connection . vars . priv_max ;
private _enteredText = '' ;
private ownTypingStatus : Interfaces.TypingStatus = 'clear' ;
private timer : number | undefined ;
private logPromise = core . logs . getBacklog ( this ) . then ( ( messages ) = > {
this . allMessages . unshift ( . . . messages ) ;
2018-04-08 00:22:32 +00:00
this . reportMessages . unshift ( . . . messages ) ;
2017-09-02 01:50:31 +00:00
this . messages = this . allMessages . slice ( ) ;
} ) ;
constructor ( readonly character : Character ) {
super ( character . name . toLowerCase ( ) , state . pinned . private . indexOf ( character . name ) !== - 1 ) ;
this . lastRead = this . messages [ this . messages . length - 1 ] ;
}
get enteredText ( ) : string {
return this . _enteredText ;
}
set enteredText ( value : string ) {
this . _enteredText = value ;
if ( this . timer !== undefined ) clearTimeout ( this . timer ) ;
if ( value . length > 0 ) {
if ( this . ownTypingStatus !== 'typing' ) this . setOwnTyping ( 'typing' ) ;
this . timer = window . setTimeout ( ( ) = > this . setOwnTyping ( 'paused' ) , 5000 ) ;
} else if ( this . ownTypingStatus !== 'clear' ) this . setOwnTyping ( 'clear' ) ;
}
2018-01-06 16:14:21 +00:00
async addMessage ( message : Interfaces.Message ) : Promise < void > {
await this . logPromise ;
2021-02-04 22:05:40 +00:00
2021-02-05 01:08:51 +00:00
this . stretch ( ) ;
2021-02-04 22:05:40 +00:00
2017-09-02 01:50:31 +00:00
this . safeAddMessage ( message ) ;
if ( message . type !== Interfaces . Message . Type . Event ) {
2018-01-06 16:14:21 +00:00
if ( core . state . settings . logMessages ) await core . logs . logMessage ( this , message ) ;
2017-12-05 01:47:27 +00:00
if ( this . settings . notify !== Interfaces . Setting . False && message . sender !== core . characters . ownCharacter )
2018-07-20 01:12:26 +00:00
await core . notifications . notify ( this , message . sender . name , message . text , characterImage ( message . sender . name ) , 'attention' ) ;
2018-01-06 16:14:21 +00:00
if ( this !== state . selectedConversation || ! state . windowFocused )
2017-09-02 01:50:31 +00:00
this . unread = Interfaces . UnreadState . Mention ;
this . typingStatus = 'clear' ;
}
}
2018-01-06 16:14:21 +00:00
async close ( ) : Promise < void > {
2017-09-02 01:50:31 +00:00
state . privateConversations . splice ( state . privateConversations . indexOf ( this ) , 1 ) ;
delete state . privateMap [ this . character . name . toLowerCase ( ) ] ;
2018-01-06 16:14:21 +00:00
await state . savePinned ( ) ;
2017-09-02 01:50:31 +00:00
if ( state . selectedConversation === this ) state . show ( state . consoleTab ) ;
}
2018-01-06 16:14:21 +00:00
async sort ( newIndex : number ) : Promise < void > {
2017-09-02 01:50:31 +00:00
state . privateConversations . splice ( state . privateConversations . indexOf ( this ) , 1 ) ;
state . privateConversations . splice ( newIndex , 0 , this ) ;
2018-01-06 16:14:21 +00:00
return state . savePinned ( ) ;
2017-09-02 01:50:31 +00:00
}
2018-01-06 16:14:21 +00:00
protected async doSend ( ) : Promise < void > {
await this . logPromise ;
2017-09-02 01:50:31 +00:00
if ( this . character . status === 'offline' ) {
this . errorText = l ( 'chat.errorOffline' , this . character . name ) ;
return ;
2019-09-17 17:14:14 +00:00
}
if ( this . character . isIgnored ) {
2017-09-02 01:50:31 +00:00
this . errorText = l ( 'chat.errorIgnored' , this . character . name ) ;
return ;
}
2019-06-02 23:57:32 +00:00
2019-06-07 19:31:42 +00:00
if ( this . adManager . isActive ( ) ) {
2019-06-02 23:57:32 +00:00
this . errorText = 'Cannot send ads manually while ad auto-posting is active' ;
return ;
}
2020-04-19 17:59:49 +00:00
const messageText = this . enteredText ;
2018-08-18 19:37:53 +00:00
this . clearText ( ) ;
2020-04-19 17:59:49 +00:00
await Conversation . conversationThroat (
async ( ) = > {
await Conversation . testPostDelay ( ) ;
core . connection . send ( 'PRI' , { recipient : this.name , message : messageText } ) ;
core . cache . markLastPostTime ( ) ;
const message = createMessage ( MessageType . Message , core . characters . ownCharacter , messageText ) ;
this . safeAddMessage ( message ) ;
if ( core . state . settings . logMessages ) await core . logs . logMessage ( this , message ) ;
}
) ;
2017-09-02 01:50:31 +00:00
}
private setOwnTyping ( status : Interfaces.TypingStatus ) : void {
this . ownTypingStatus = status ;
core . connection . send ( 'TPN' , { character : this.name , status } ) ;
}
}
class ChannelConversation extends Conversation implements Interfaces . ChannelConversation {
readonly context = CommandContext . Channel ;
readonly name = this . channel . name ;
isSendingAds = this . channel . mode === 'ads' ;
2018-08-10 16:59:37 +00:00
nextAd = 0 ;
2017-09-02 01:50:31 +00:00
private chat : Interfaces.Message [ ] = [ ] ;
private ads : Interfaces.Message [ ] = [ ] ;
private both : Interfaces.Message [ ] = [ ] ;
2018-03-04 02:32:26 +00:00
private _mode ! : Channel . Mode ;
2017-09-02 01:50:31 +00:00
private adEnteredText = '' ;
private chatEnteredText = '' ;
private logPromise = core . logs . getBacklog ( this ) . then ( ( messages ) = > {
this . both . unshift ( . . . messages ) ;
this . chat . unshift ( . . . this . both . filter ( ( x ) = > x . type !== MessageType . Ad ) ) ;
this . ads . unshift ( . . . this . both . filter ( ( x ) = > x . type === MessageType . Ad ) ) ;
2018-04-08 00:22:32 +00:00
this . reportMessages . unshift ( . . . messages ) ;
2017-09-02 01:50:31 +00:00
this . lastRead = this . messages [ this . messages . length - 1 ] ;
2017-10-16 23:58:57 +00:00
this . messages = this . allMessages . slice ( - this . maxMessages ) ;
2017-09-02 01:50:31 +00:00
} ) ;
constructor ( readonly channel : Channel ) {
super ( ` # ${ channel . id . replace ( /[^\w- ]/gi , '' ) } ` , state . pinned . channels . indexOf ( channel . id ) !== - 1 ) ;
2017-10-17 18:21:57 +00:00
core . watch < Channel.Mode | undefined > ( function ( ) : Channel . Mode | undefined {
2017-09-02 01:50:31 +00:00
const c = this . channels . getChannel ( channel . id ) ;
return c !== undefined ? c.mode : undefined ;
2017-12-05 01:47:27 +00:00
} , ( value : Channel.Mode | undefined ) = > {
2017-09-02 01:50:31 +00:00
if ( value === undefined ) return ;
this . mode = value ;
if ( value !== 'both' ) this . isSendingAds = value === 'ads' ;
} ) ;
2018-03-04 02:32:26 +00:00
this . mode = channel . mode === 'both' && channel . id in state . modes ? state . modes [ channel . id ] ! : channel . mode ;
2017-09-02 01:50:31 +00:00
}
get maxMessageLength ( ) : number {
return core . connection . vars [ this . isSendingAds ? 'lfrp_max' : 'chat_max' ] ;
}
get mode ( ) : Channel . Mode {
return this . _mode ;
}
set mode ( mode : Channel.Mode ) {
this . _mode = mode ;
2018-04-08 00:22:32 +00:00
this . maxMessages = 50 ;
2017-09-02 01:50:31 +00:00
this . allMessages = this [ mode ] ;
this . messages = this . allMessages . slice ( - this . maxMessages ) ;
2018-03-04 02:32:26 +00:00
if ( mode === this . channel . mode && this . channel . id in state . modes ) delete state . modes [ this . channel . id ] ;
else if ( mode !== this . channel . mode && mode !== state . modes [ this . channel . id ] ) state . modes [ this . channel . id ] = mode ;
else return ;
state . saveModes ( ) ; //tslint:disable-line:no-floating-promises
2017-09-02 01:50:31 +00:00
}
get enteredText ( ) : string {
return this . isSendingAds ? this . adEnteredText : this.chatEnteredText ;
}
set enteredText ( value : string ) {
if ( this . isSendingAds ) this . adEnteredText = value ;
else this . chatEnteredText = value ;
}
addModeMessage ( mode : Channel.Mode , message : Interfaces.Message ) : void {
2018-08-18 19:37:53 +00:00
safeAddMessage ( this [ mode ] , message , 500 ) ;
if ( this . _mode === mode ) safeAddMessage ( this . messages , message , this . maxMessages ) ;
2017-09-02 01:50:31 +00:00
}
2018-01-06 16:14:21 +00:00
async addMessage ( message : Interfaces.Message ) : Promise < void > {
await this . logPromise ;
2021-02-04 22:05:40 +00:00
2021-02-05 01:08:51 +00:00
this . stretch ( ) ;
2021-02-04 22:05:40 +00:00
2018-03-28 13:51:05 +00:00
if ( ( message . type === MessageType . Message || message . type === MessageType . Ad ) && isWarn ( message . text ) ) {
2017-12-05 01:47:27 +00:00
const member = this . channel . members [ message . sender . name ] ;
if ( member !== undefined && member . rank > Channel . Rank . Member || message . sender . isChatOp )
message = new Message ( MessageType . Warn , message . sender , message . text . substr ( 6 ) , message . time ) ;
}
2017-09-02 01:50:31 +00:00
if ( message . type === MessageType . Ad ) {
this . addModeMessage ( 'ads' , message ) ;
2018-01-06 16:14:21 +00:00
if ( core . state . settings . logAds ) await core . logs . logMessage ( this , message ) ;
2017-09-02 01:50:31 +00:00
} else {
this . addModeMessage ( 'chat' , message ) ;
if ( message . type !== Interfaces . Message . Type . Event ) {
if ( message . type === Interfaces . Message . Type . Warn ) this . addModeMessage ( 'ads' , message ) ;
2018-01-06 16:14:21 +00:00
if ( core . state . settings . logMessages ) await core . logs . logMessage ( this , message ) ;
2018-03-04 02:32:26 +00:00
if ( this . unread === Interfaces . UnreadState . None && ( this !== state . selectedConversation || ! state . windowFocused )
&& this . mode !== 'ads' )
2017-09-02 01:50:31 +00:00
this . unread = Interfaces . UnreadState . Unread ;
} else this . addModeMessage ( 'ads' , message ) ;
}
this . addModeMessage ( 'both' , message ) ;
2018-08-18 19:37:53 +00:00
if ( message . type !== Interfaces . Message . Type . Event )
safeAddMessage ( this . reportMessages , message , 500 ) ;
2017-09-02 01:50:31 +00:00
}
2018-08-10 16:59:37 +00:00
clear ( ) : void {
this . messages = [ ] ;
this . chat . length = 0 ;
this . ads . length = 0 ;
this . both . length = 0 ;
}
2017-09-02 01:50:31 +00:00
close ( ) : void {
core . connection . send ( 'LCH' , { channel : this.channel.id } ) ;
}
2018-01-06 16:14:21 +00:00
async sort ( newIndex : number ) : Promise < void > {
2017-09-02 01:50:31 +00:00
state . channelConversations . splice ( state . channelConversations . indexOf ( this ) , 1 ) ;
state . channelConversations . splice ( newIndex , 0 , this ) ;
2018-01-06 16:14:21 +00:00
return state . savePinned ( ) ;
2017-09-02 01:50:31 +00:00
}
2018-01-06 16:14:21 +00:00
protected async doSend ( ) : Promise < void > {
2017-09-02 01:50:31 +00:00
const isAd = this . isSendingAds ;
2019-06-02 23:57:32 +00:00
2021-09-07 06:02:31 +00:00
if ( isAd && this . adManager . isActive ( ) ) {
2019-06-02 23:57:32 +00:00
this . errorText = 'Cannot post ads manually while ad auto-posting is active' ;
return ;
}
if ( isAd && Date . now ( ) < this . nextAd ) {
this . errorText = 'You must wait at least ten minutes between ad posts on this channel' ;
return ;
}
2020-04-19 17:59:49 +00:00
const message = this . enteredText ;
if ( ! isAd ) {
this . clearText ( ) ;
}
await Conversation . conversationThroat (
async ( ) = > {
await Conversation . testPostDelay ( ) ;
core . connection . send ( isAd ? 'LRP' : 'MSG' , { channel : this.channel.id , message } ) ;
core . cache . markLastPostTime ( ) ;
await this . addMessage (
createMessage ( isAd ? MessageType.Ad : MessageType.Message , core . characters . ownCharacter , message , new Date ( ) )
) ;
2019-10-21 22:55:05 +00:00
2020-04-19 17:59:49 +00:00
if ( isAd )
this . nextAd = Date . now ( ) + core . connection . vars . lfrp_flood * 1000 ;
}
) ;
2017-09-02 01:50:31 +00:00
}
2019-06-02 23:57:32 +00:00
2020-04-19 17:59:49 +00:00
2020-10-24 19:43:06 +00:00
hasAutomatedAds ( ) : boolean {
return ( ( this . mode === 'both' ) || ( this . mode === 'ads' ) )
&& super . hasAutomatedAds ( ) ;
}
2019-06-02 23:57:32 +00:00
async sendAd ( text : string ) : Promise < void > {
if ( text . length < 1 )
return ;
2020-06-29 19:30:08 +00:00
const initTime = Date . now ( ) ;
2020-04-19 17:59:49 +00:00
await Conversation . conversationThroat (
async ( ) = > {
2020-06-29 19:30:08 +00:00
const throatTime = Date . now ( ) ;
2020-06-30 00:59:00 +00:00
await Promise . all (
[
await Conversation . testPostDelay ( ) ,
await core . adCoordinator . requestTurnToPostAd ( )
]
) ;
2019-06-07 21:42:25 +00:00
2020-06-29 19:30:08 +00:00
const delayTime = Date . now ( ) ;
2020-04-19 17:59:49 +00:00
core . connection . send ( 'LRP' , { channel : this.channel.id , message : text } ) ;
core . cache . markLastPostTime ( ) ;
2020-06-30 15:45:29 +00:00
log . debug (
'conversation.sendAd' ,
{
character : core.characters.ownCharacter?.name ,
channel : this.channel.name ,
throatDelta : throatTime - initTime ,
delayDelta : delayTime - throatTime ,
totalWait : delayTime - initTime ,
text
}
) ;
2020-06-29 19:30:08 +00:00
2020-04-19 17:59:49 +00:00
await this . addMessage (
createMessage ( MessageType . Ad , core . characters . ownCharacter , text , new Date ( ) )
) ;
2019-06-02 23:57:32 +00:00
2020-04-19 17:59:49 +00:00
this . nextAd = Date . now ( ) + core . connection . vars . lfrp_flood * 1000 ;
}
) ;
2019-06-02 23:57:32 +00:00
}
2017-09-02 01:50:31 +00:00
}
2020-04-19 17:59:49 +00:00
2017-09-02 01:50:31 +00:00
class ConsoleConversation extends Conversation {
readonly context = CommandContext . Console ;
readonly name = l ( 'chat.consoleTab' ) ;
readonly maxMessageLength = undefined ;
enteredText = '' ;
constructor ( ) {
super ( '_' , false ) ;
this . allMessages = [ ] ;
}
//tslint:disable-next-line:no-empty
close ( ) : void {
}
2018-01-06 16:14:21 +00:00
async addMessage ( message : Interfaces.Message ) : Promise < void > {
2017-09-02 01:50:31 +00:00
this . safeAddMessage ( message ) ;
2018-01-06 16:14:21 +00:00
if ( core . state . settings . logMessages ) await core . logs . logMessage ( this , message ) ;
if ( this !== state . selectedConversation || ! state . windowFocused ) this . unread = Interfaces . UnreadState . Unread ;
2017-09-02 01:50:31 +00:00
}
protected doSend ( ) : void {
this . errorText = l ( 'chat.consoleChat' ) ;
}
}
class State implements Interfaces . State {
privateConversations : PrivateConversation [ ] = [ ] ;
channelConversations : ChannelConversation [ ] = [ ] ;
privateMap : { [ key : string ] : PrivateConversation | undefined } = { } ;
channelMap : { [ key : string ] : ChannelConversation | undefined } = { } ;
2018-03-04 02:32:26 +00:00
consoleTab ! : ConsoleConversation ;
2017-09-02 01:50:31 +00:00
selectedConversation : Conversation = this . consoleTab ;
2019-01-03 17:38:17 +00:00
recent : Interfaces.RecentPrivateConversation [ ] = [ ] ;
recentChannels : Interfaces.RecentChannelConversation [ ] = [ ] ;
2018-03-04 02:32:26 +00:00
pinned ! : { channels : string [ ] , private : string [ ] } ;
settings ! : { [ key : string ] : Interfaces . Settings } ;
modes ! : { [ key : string ] : Channel . Mode | undefined } ;
windowFocused = document . hasFocus ( ) ;
2018-01-06 16:14:21 +00:00
get hasNew ( ) : boolean {
return this . privateConversations . some ( ( x ) = > x . unread === Interfaces . UnreadState . Mention ) ||
this . channelConversations . some ( ( x ) = > x . unread === Interfaces . UnreadState . Mention ) ;
}
2017-09-02 01:50:31 +00:00
2022-01-02 22:37:57 +00:00
getPrivate ( character : Character ) : PrivateConversation ;
getPrivate ( character : Character , noCreate : boolean = false ) : PrivateConversation | undefined {
2017-09-02 01:50:31 +00:00
const key = character . name . toLowerCase ( ) ;
let conv = state . privateMap [ key ] ;
if ( conv !== undefined ) return conv ;
2022-01-02 22:37:57 +00:00
if ( noCreate ) {
return ;
}
2017-09-02 01:50:31 +00:00
conv = new PrivateConversation ( character ) ;
this . privateConversations . push ( conv ) ;
this . privateMap [ key ] = conv ;
2019-01-03 17:38:17 +00:00
const index = this . recent . findIndex ( ( c ) = > c . character === conv ! . name ) ;
if ( index !== - 1 ) this . recent . splice ( index , 1 ) ;
if ( this . recent . length >= 50 ) this . recent . pop ( ) ;
this . recent . unshift ( { character : conv.name } ) ;
core . settingsStore . set ( 'recent' , this . recent ) ; //tslint:disable-line:no-floating-promises
2017-09-02 01:50:31 +00:00
return conv ;
}
byKey ( key : string ) : Conversation | undefined {
if ( key === '_' ) return this . consoleTab ;
2019-01-03 17:38:17 +00:00
key = key . toLowerCase ( ) ;
return key [ 0 ] === '#' ? this . channelMap [ key . substr ( 1 ) ] : this . privateMap [ key ] ;
2017-09-02 01:50:31 +00:00
}
2018-01-06 16:14:21 +00:00
async savePinned ( ) : Promise < void > {
2017-09-02 01:50:31 +00:00
this . pinned . channels = this . channelConversations . filter ( ( x ) = > x . isPinned ) . map ( ( x ) = > x . channel . id ) ;
this . pinned . private = this . privateConversations . filter ( ( x ) = > x . isPinned ) . map ( ( x ) = > x . name ) ;
2018-01-06 16:14:21 +00:00
await core . settingsStore . set ( 'pinned' , this . pinned ) ;
2017-09-02 01:50:31 +00:00
}
2018-03-04 02:32:26 +00:00
async saveModes ( ) : Promise < void > {
await core . settingsStore . set ( 'modes' , this . modes ) ;
}
2018-01-06 16:14:21 +00:00
async setSettings ( key : string , value : Interfaces.Settings ) : Promise < void > {
2017-09-02 01:50:31 +00:00
this . settings [ key ] = value ;
2018-01-06 16:14:21 +00:00
await core . settingsStore . set ( 'conversationSettings' , this . settings ) ;
2017-09-02 01:50:31 +00:00
}
show ( conversation : Conversation ) : void {
2019-09-17 17:14:14 +00:00
if ( conversation === this . selectedConversation ) return ;
2017-09-02 01:50:31 +00:00
this . selectedConversation . onHide ( ) ;
conversation . unread = Interfaces . UnreadState . None ;
this . selectedConversation = conversation ;
2020-06-16 16:27:31 +00:00
EventBus . $emit ( 'select-conversation' , { conversation } ) ;
2017-09-02 01:50:31 +00:00
}
async reloadSettings ( ) : Promise < void > {
//tslint:disable:strict-boolean-expressions
this . pinned = await core . settingsStore . get ( 'pinned' ) || { private : [ ] , channels : [ ] } ;
2018-03-04 02:32:26 +00:00
this . modes = await core . settingsStore . get ( 'modes' ) || { } ;
2017-09-02 01:50:31 +00:00
for ( const conversation of this . channelConversations )
conversation . _isPinned = this . pinned . channels . indexOf ( conversation . channel . id ) !== - 1 ;
for ( const conversation of this . privateConversations )
conversation . _isPinned = this . pinned . private . indexOf ( conversation . name ) !== - 1 ;
this . recent = await core . settingsStore . get ( 'recent' ) || [ ] ;
2019-01-03 17:38:17 +00:00
this . recentChannels = await core . settingsStore . get ( 'recentChannels' ) || [ ] ;
2017-09-02 01:50:31 +00:00
const settings = < { [ key : string ] : ConversationSettings } > await core . settingsStore . get ( 'conversationSettings' ) || { } ;
for ( const key in settings ) {
2019-09-17 17:14:14 +00:00
settings [ key ] = Object . assign ( new ConversationSettings ( ) , settings [ key ] ) ;
2019-01-03 17:38:17 +00:00
const conv = this . byKey ( key ) ;
2019-09-17 17:14:14 +00:00
if ( conv !== undefined ) conv . _settings = settings [ key ] ;
2017-09-02 01:50:31 +00:00
}
this . settings = settings ;
//tslint:enable
}
}
let state : State ;
2020-03-15 14:02:31 +00:00
async function addEventMessage ( this : any , message : Interfaces.Message ) : Promise < void > {
2018-01-06 16:14:21 +00:00
await state . consoleTab . addMessage ( message ) ;
if ( core . state . settings . eventMessages && state . selectedConversation !== state . consoleTab )
await state . selectedConversation . addMessage ( message ) ;
2017-09-02 01:50:31 +00:00
}
2020-03-15 14:02:31 +00:00
function isOfInterest ( this : any , character : Character ) : boolean {
2017-09-02 01:50:31 +00:00
return character . isFriend || character . isBookmarked || state . privateMap [ character . name . toLowerCase ( ) ] !== undefined ;
}
2022-01-02 22:37:57 +00:00
async function testSmartFilterForPrivateMessage ( fromChar : Character.Character ) : Promise < boolean > {
const cachedProfile = core . cache . profileCache . getSync ( fromChar . name ) || await core . cache . profileCache . get ( fromChar . name ) ;
if (
cachedProfile &&
cachedProfile . match . isFiltered &&
core . state . settings . risingFilter . autoReply &&
! cachedProfile . match . autoResponded
) {
cachedProfile . match . autoResponded = true ;
log . debug ( 'filter.autoresponse' , { name : fromChar.name } ) ;
void Conversation . conversationThroat (
async ( ) = > {
await Conversation . testPostDelay ( ) ;
// tslint:disable-next-line:prefer-template
const m = '[Automated message] Sorry, the player of this character has indicated that they are not interested in characters matching your profile. They will not see your message.\n\n' +
'Need a filter for yourself? Try out [url=https://mrstallion.github.io/fchat-rising/]F-Chat Rising[/url]' ;
core . connection . send ( 'PRI' , { recipient : fromChar.name , message : m } ) ;
core . cache . markLastPostTime ( ) ;
}
) ;
}
if ( cachedProfile && cachedProfile . match . isFiltered && core . state . settings . risingFilter . hidePrivateMessages ) {
return true ;
}
return false ;
}
async function testSmartFilterForChannel ( fromChar : Character.Character , conversation : ChannelConversation ) : Promise < boolean > {
if (
( isChannel ( conversation ) && conversation . channel . owner === '' && core . state . settings . risingFilter . hidePublicChannelMessages ) ||
( isChannel ( conversation ) && conversation . channel . owner !== '' && core . state . settings . risingFilter . hidePrivateChannelMessages )
) {
const cachedProfile = core . cache . profileCache . getSync ( fromChar . name ) || await core . cache . profileCache . get ( fromChar . name ) ;
if ( cachedProfile && cachedProfile . match . isFiltered && ! fromChar . isChatOp ) {
return true ;
}
}
return false ;
}
2020-03-15 14:02:31 +00:00
export default function ( this : any ) : Interfaces . State {
2017-09-02 01:50:31 +00:00
state = new State ( ) ;
2018-01-06 16:14:21 +00:00
window . addEventListener ( 'focus' , ( ) = > {
state . windowFocused = true ;
if ( state . selectedConversation !== undefined ! ) state . selectedConversation . unread = Interfaces . UnreadState . None ;
} ) ;
2018-03-04 02:32:26 +00:00
window . addEventListener ( 'blur' , ( ) = > {
state . windowFocused = false ;
if ( state . selectedConversation !== undefined ! )
state . selectedConversation . lastRead = state . selectedConversation . messages [ state . selectedConversation . messages . length - 1 ] ;
} ) ;
2017-09-02 01:50:31 +00:00
const connection = core . connection ;
connection . onEvent ( 'connecting' , async ( isReconnect ) = > {
state . channelConversations = [ ] ;
state . channelMap = { } ;
2017-12-05 01:47:27 +00:00
if ( ! isReconnect ) {
state . consoleTab = new ConsoleConversation ( ) ;
state . privateConversations = [ ] ;
state . privateMap = { } ;
} else state . consoleTab . unread = Interfaces . UnreadState . None ;
2017-09-02 01:50:31 +00:00
state . selectedConversation = state . consoleTab ;
2020-06-16 16:27:31 +00:00
EventBus . $emit ( 'select-conversation' , { conversation : state.selectedConversation } ) ;
2017-09-02 01:50:31 +00:00
await state . reloadSettings ( ) ;
} ) ;
connection . onEvent ( 'connected' , ( isReconnect ) = > {
if ( isReconnect ) return ;
for ( const item of state . pinned . private ) state . getPrivate ( core . characters . get ( item ) ) ;
queuedJoin ( state . pinned . channels . slice ( ) ) ;
} ) ;
2018-01-06 16:14:21 +00:00
core . channels . onEvent ( async ( type , channel , member ) = > {
2017-10-18 23:29:28 +00:00
if ( type === 'join' )
if ( member === undefined ) {
const conv = new ChannelConversation ( channel ) ;
2017-12-05 01:47:27 +00:00
state . channelMap [ channel . id ] = conv ;
2017-10-18 23:29:28 +00:00
state . channelConversations . push ( conv ) ;
2019-01-03 17:38:17 +00:00
const index = state . recentChannels . findIndex ( ( c ) = > c . channel === channel . id ) ;
if ( index !== - 1 ) state . recentChannels . splice ( index , 1 ) ;
if ( state . recentChannels . length >= 50 ) state . recentChannels . pop ( ) ;
state . recentChannels . unshift ( { channel : channel.id , name : conv.channel.name } ) ;
core . settingsStore . set ( 'recentChannels' , state . recentChannels ) ; //tslint:disable-line:no-floating-promises
2020-04-11 14:48:09 +00:00
AdManager . onNewChannelAvailable ( conv ) ;
2017-10-18 23:29:28 +00:00
} else {
2018-03-04 02:32:26 +00:00
const conv = state . channelMap [ channel . id ] ;
if ( conv === undefined ) return ;
2017-10-18 23:29:28 +00:00
if ( conv . settings . joinMessages === Interfaces . Setting . False || conv . settings . joinMessages === Interfaces . Setting . Default &&
! core . state . settings . joinMessages ) return ;
const text = l ( 'events.channelJoin' , ` [user] ${ member . character . name } [/user] ` ) ;
2018-01-06 16:14:21 +00:00
await conv . addMessage ( new EventMessage ( text ) ) ;
2017-10-18 23:29:28 +00:00
}
else if ( member === undefined ) {
2018-03-04 02:32:26 +00:00
const conv = state . channelMap [ channel . id ] ;
if ( conv === undefined ) return ;
2017-09-02 01:50:31 +00:00
state . channelConversations . splice ( state . channelConversations . indexOf ( conv ) , 1 ) ;
2017-12-05 01:47:27 +00:00
delete state . channelMap [ channel . id ] ;
2018-01-06 16:14:21 +00:00
await state . savePinned ( ) ;
2017-09-02 01:50:31 +00:00
if ( state . selectedConversation === conv ) state . show ( state . consoleTab ) ;
2017-10-18 23:29:28 +00:00
} else {
2018-03-04 02:32:26 +00:00
const conv = state . channelMap [ channel . id ] ;
if ( conv === undefined ) return ;
2017-10-18 23:29:28 +00:00
if ( conv . settings . joinMessages === Interfaces . Setting . False || conv . settings . joinMessages === Interfaces . Setting . Default &&
! core . state . settings . joinMessages ) return ;
const text = l ( 'events.channelLeave' , ` [user] ${ member . character . name } [/user] ` ) ;
2018-01-06 16:14:21 +00:00
await conv . addMessage ( new EventMessage ( text ) ) ;
2017-09-02 01:50:31 +00:00
}
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'PRI' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
const char = core . characters . get ( data . character ) ;
if ( char . isIgnored ) return connection . send ( 'IGN' , { action : 'notify' , character : data.character } ) ;
const message = createMessage ( MessageType . Message , char , decodeHTML ( data . message ) , time ) ;
2022-01-02 22:37:57 +00:00
if ( await testSmartFilterForPrivateMessage ( char ) === true ) {
return ;
}
2019-07-07 01:37:15 +00:00
EventBus . $emit ( 'private-message' , { message } ) ;
2022-01-02 22:37:57 +00:00
2017-09-02 01:50:31 +00:00
const conv = state . getPrivate ( char ) ;
2018-01-06 16:14:21 +00:00
await conv . addMessage ( message ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'MSG' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
const char = core . characters . get ( data . character ) ;
2017-12-05 01:47:27 +00:00
const conversation = state . channelMap [ data . channel . toLowerCase ( ) ] ;
if ( conversation === undefined ) return core . channels . leave ( data . channel ) ;
2019-09-17 17:14:14 +00:00
if ( char . isIgnored ) return ;
2022-01-01 00:06:08 +00:00
2017-09-02 01:50:31 +00:00
const message = createMessage ( MessageType . Message , char , decodeHTML ( data . message ) , time ) ;
2022-01-01 00:06:08 +00:00
2022-01-02 22:37:57 +00:00
if ( await testSmartFilterForChannel ( char , conversation ) === true ) {
return ;
2022-01-01 00:06:08 +00:00
}
2017-09-02 01:50:31 +00:00
2022-01-02 22:37:57 +00:00
await conversation . addMessage ( message ) ;
EventBus . $emit ( 'channel-message' , { message , channel : conversation } ) ;
2018-07-20 01:12:26 +00:00
const words = conversation . settings . highlightWords . slice ( ) ;
2017-10-18 23:29:28 +00:00
if ( conversation . settings . defaultHighlights ) words . push ( . . . core . state . settings . highlightWords ) ;
if ( conversation . settings . highlight === Interfaces . Setting . Default && core . state . settings . highlight ||
conversation . settings . highlight === Interfaces . Setting . True ) words . push ( core . connection . character ) ;
2018-07-20 01:12:26 +00:00
for ( let i = 0 ; i < words . length ; ++ i )
words [ i ] = words [ i ] . replace ( /[^\w]/gi , '\\$&' ) ;
2017-09-02 01:50:31 +00:00
//tslint:disable-next-line:no-null-keyword
const results = words . length > 0 ? message . text . match ( new RegExp ( ` \\ b( ${ words . join ( '|' ) } ) \\ b ` , 'i' ) ) : null ;
if ( results !== null ) {
2018-07-20 01:12:26 +00:00
await core . notifications . notify ( conversation , data . character , l ( 'chat.highlight' , results [ 0 ] , conversation . name , message . text ) ,
2017-09-02 01:50:31 +00:00
characterImage ( data . character ) , 'attention' ) ;
2018-01-06 16:14:21 +00:00
if ( conversation !== state . selectedConversation || ! state . windowFocused ) conversation . unread = Interfaces . UnreadState . Mention ;
2017-09-02 01:50:31 +00:00
message . isHighlight = true ;
2019-01-03 17:38:17 +00:00
await state . consoleTab . addMessage ( new EventMessage ( l ( 'events.highlight' , ` [user] ${ data . character } [/user] ` , results [ 0 ] ,
` [session= ${ conversation . name } ] ${ data . channel } [/session] ` ) , time ) ) ;
2018-03-04 02:32:26 +00:00
} else if ( conversation . settings . notify === Interfaces . Setting . True ) {
2018-07-20 01:12:26 +00:00
await core . notifications . notify ( conversation , conversation . name , messageToString ( message ) ,
2017-09-02 01:50:31 +00:00
characterImage ( data . character ) , 'attention' ) ;
2018-03-04 02:32:26 +00:00
if ( conversation !== state . selectedConversation || ! state . windowFocused ) conversation . unread = Interfaces . UnreadState . Mention ;
}
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'LRP' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
const char = core . characters . get ( data . character ) ;
2017-12-05 01:47:27 +00:00
const conv = state . channelMap [ data . channel . toLowerCase ( ) ] ;
if ( conv === undefined ) return core . channels . leave ( data . channel ) ;
2019-09-17 17:14:14 +00:00
if ( char . isIgnored || core . state . hiddenUsers . indexOf ( char . name ) !== - 1 ) return ;
2020-03-14 19:28:14 +00:00
2019-07-07 01:37:15 +00:00
const msg = new Message ( MessageType . Ad , char , decodeHTML ( data . message ) , time ) ;
2020-06-16 16:27:31 +00:00
const p = await core . cache . resolvePScore (
( core . conversations . selectedConversation !== conv ) ,
char ,
conv ,
msg
) ;
2019-07-07 01:37:15 +00:00
2019-07-08 19:08:16 +00:00
EventBus . $emit ( 'channel-ad' , { message : msg , channel : conv , profile : p } ) ;
2020-03-14 19:28:14 +00:00
2019-07-07 01:37:15 +00:00
await conv . addMessage ( msg ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'RLL' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
const sender = core . characters . get ( data . character ) ;
let text : string ;
if ( data . type === 'bottle' )
text = l ( 'chat.bottle' , ` [user] ${ data . target } [/user] ` ) ;
else {
const results = data . results . length > 1 ? ` ${ data . results . join ( '+' ) } = ${ data . endresult } ` : data . endresult . toString ( ) ;
text = l ( 'chat.roll' , data . rolls . join ( '+' ) , results ) ;
}
const message = new Message ( MessageType . Roll , sender , text , time ) ;
if ( 'channel' in data ) {
2017-12-05 01:47:27 +00:00
const channel = ( < { channel : string } > data ) . channel . toLowerCase ( ) ;
const conversation = state . channelMap [ channel ] ;
if ( conversation === undefined ) return core . channels . leave ( channel ) ;
2019-09-17 17:14:14 +00:00
if ( sender . isIgnored ) return ;
2018-03-04 02:32:26 +00:00
if ( data . type === 'bottle' && data . target === core . connection . character ) {
2018-07-20 01:12:26 +00:00
await core . notifications . notify ( conversation , conversation . name , messageToString ( message ) ,
2017-09-02 01:50:31 +00:00
characterImage ( data . character ) , 'attention' ) ;
2018-03-04 02:32:26 +00:00
if ( conversation !== state . selectedConversation || ! state . windowFocused )
conversation . unread = Interfaces . UnreadState . Mention ;
message . isHighlight = true ;
}
await conversation . addMessage ( message ) ;
2017-09-02 01:50:31 +00:00
} else {
2018-03-04 02:32:26 +00:00
if ( sender . isIgnored ) return ;
2017-09-02 01:50:31 +00:00
const char = core . characters . get (
data . character === connection . character ? ( < { recipient : string } > data ) . recipient : data.character ) ;
if ( char . isIgnored ) return connection . send ( 'IGN' , { action : 'notify' , character : data.character } ) ;
const conversation = state . getPrivate ( char ) ;
2018-01-06 16:14:21 +00:00
await conversation . addMessage ( message ) ;
2017-09-02 01:50:31 +00:00
}
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'NLN' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
const message = new EventMessage ( l ( 'events.login' , ` [user] ${ data . identity } [/user] ` ) , time ) ;
2018-01-06 16:14:21 +00:00
if ( isOfInterest ( core . characters . get ( data . identity ) ) ) await addEventMessage ( message ) ;
2017-09-02 01:50:31 +00:00
const conv = state . privateMap [ data . identity . toLowerCase ( ) ] ;
2018-01-06 16:14:21 +00:00
if ( conv !== undefined && ( ! core . state . settings . eventMessages || conv !== state . selectedConversation ) )
await conv . addMessage ( message ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'FLN' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
const message = new EventMessage ( l ( 'events.logout' , ` [user] ${ data . character } [/user] ` ) , time ) ;
2018-01-06 16:14:21 +00:00
if ( isOfInterest ( core . characters . get ( data . character ) ) ) await addEventMessage ( message ) ;
2017-09-02 01:50:31 +00:00
const conv = state . privateMap [ data . character . toLowerCase ( ) ] ;
if ( conv === undefined ) return ;
conv . typingStatus = 'clear' ;
2018-01-06 16:14:21 +00:00
if ( ! core . state . settings . eventMessages || conv !== state . selectedConversation ) await conv . addMessage ( message ) ;
2017-09-02 01:50:31 +00:00
} ) ;
connection . onMessage ( 'TPN' , ( data ) = > {
const conv = state . privateMap [ data . character . toLowerCase ( ) ] ;
if ( conv !== undefined ) conv . typingStatus = data . status ;
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'CBU' , async ( data , time ) = > {
2017-12-05 01:47:27 +00:00
const conv = state . channelMap [ data . channel . toLowerCase ( ) ] ;
if ( conv === undefined ) return core . channels . leave ( data . channel ) ;
2018-09-28 00:08:10 +00:00
const text = l ( 'events.ban' , conv . name , data . character , data . operator ) ;
2017-12-05 01:47:27 +00:00
conv . infoText = text ;
2018-01-06 16:14:21 +00:00
return addEventMessage ( new EventMessage ( text , time ) ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'CKU' , async ( data , time ) = > {
2017-12-05 01:47:27 +00:00
const conv = state . channelMap [ data . channel . toLowerCase ( ) ] ;
if ( conv === undefined ) return core . channels . leave ( data . channel ) ;
2018-09-28 00:08:10 +00:00
const text = l ( 'events.kick' , conv . name , data . character , data . operator ) ;
2017-12-05 01:47:27 +00:00
conv . infoText = text ;
2018-01-06 16:14:21 +00:00
return addEventMessage ( new EventMessage ( text , time ) ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'CTU' , async ( data , time ) = > {
2017-12-05 01:47:27 +00:00
const conv = state . channelMap [ data . channel . toLowerCase ( ) ] ;
if ( conv === undefined ) return core . channels . leave ( data . channel ) ;
2018-09-28 00:08:10 +00:00
const text = l ( 'events.timeout' , conv . name , data . character , data . operator , data . length . toString ( ) ) ;
2017-12-05 01:47:27 +00:00
conv . infoText = text ;
2018-01-06 16:14:21 +00:00
return addEventMessage ( new EventMessage ( text , time ) ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'BRO' , async ( data , time ) = > {
2018-09-28 00:08:10 +00:00
if ( data . character !== undefined ) {
const content = decodeHTML ( data . message . substr ( data . character . length + 24 ) ) ;
const message = new EventMessage ( l ( 'events.broadcast' , ` [user] ${ data . character } [/user] ` , content ) , time ) ;
await state . consoleTab . addMessage ( message ) ;
await core . notifications . notify ( state . consoleTab , l ( 'events.broadcast.notification' , data . character ) , content ,
characterImage ( data . character ) , 'attention' ) ;
for ( const conv of ( < Conversation [ ] > state . channelConversations ) . concat ( state . privateConversations ) )
await conv . addMessage ( message ) ;
} else return addEventMessage ( new EventMessage ( decodeHTML ( data . message ) , time ) ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'CIU' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
const text = l ( 'events.invite' , ` [user] ${ data . sender } [/user] ` , ` [session= ${ data . title } ] ${ data . name } [/session] ` ) ;
2018-01-06 16:14:21 +00:00
return addEventMessage ( new EventMessage ( text , time ) ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'ERR' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
state . selectedConversation . errorText = data . message ;
2018-01-06 16:14:21 +00:00
return addEventMessage ( new EventMessage ( ` [color=red] ${ l ( 'events.error' , data . message ) } [/color] ` , time ) ) ;
} ) ;
connection . onMessage ( 'IGN' , async ( data , time ) = > {
if ( data . action !== 'add' && data . action !== 'delete' ) return ;
2019-01-03 17:38:17 +00:00
const text = l ( ` events.ignore_ ${ data . action } ` , data . character ) ;
state . selectedConversation . infoText = text ;
return addEventMessage ( new EventMessage ( text , time ) ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'RTB' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
let url = 'https://www.f-list.net/' ;
let text : string , character : string ;
if ( data . type === 'comment' ) { //tslint:disable-line:prefer-switch
switch ( data . target_type ) {
case 'newspost' :
url += ` newspost/ ${ data . target_id } /#Comment ${ data . id } ` ;
break ;
case 'bugreport' :
2018-07-20 01:12:26 +00:00
url += ` view_bugreport.php?id= ${ data . target_id } /# ${ data . id } ` ;
2017-09-02 01:50:31 +00:00
break ;
case 'changelog' :
2018-07-20 01:12:26 +00:00
url += ` log.php?id= ${ data . target_id } /# ${ data . id } ` ;
2017-09-02 01:50:31 +00:00
break ;
case 'feature' :
2018-07-20 01:12:26 +00:00
url += ` vote.php?id= ${ data . target_id } /# ${ data . id } ` ;
2017-09-02 01:50:31 +00:00
}
const key = ` events.rtbComment ${ ( data . parent_id !== 0 ? 'Reply' : '' ) } ` ;
text = l ( key , ` [user] ${ data . name } [/user] ` , l ( ` events.rtbComment_ ${ data . target_type } ` ) , ` [url= ${ url } ] ${ data . target } [/url] ` ) ;
character = data . name ;
} else if ( data . type === 'note' ) {
2020-11-27 20:58:14 +00:00
// tslint:disable-next-line:no-unsafe-any
core . siteSession . interfaces . notes . incrementNotes ( ) ;
2017-09-02 01:50:31 +00:00
text = l ( 'events.rtb_note' , ` [user] ${ data . sender } [/user] ` , ` [url= ${ url } view_note.php?note_id= ${ data . id } ] ${ data . subject } [/url] ` ) ;
character = data . sender ;
} else if ( data . type === 'friendrequest' ) {
2020-11-27 20:58:14 +00:00
// tslint:disable-next-line:no-unsafe-any
core . siteSession . interfaces . notes . incrementMessages ( ) ;
2017-09-02 01:50:31 +00:00
text = l ( ` events.rtb_friendrequest ` , ` [user] ${ data . name } [/user] ` ) ;
character = data . name ;
} else {
switch ( data . type ) {
case 'grouprequest' :
url += 'panel/group_requests.php' ;
break ;
case 'bugreport' :
url += ` view_bugreport.php?id= ${ data . id } ` ;
break ;
case 'helpdeskticket' :
url += ` view_ticket.php?id= ${ data . id } ` ;
break ;
case 'helpdeskreply' :
url += ` view_ticket.php?id= ${ data . id } ` ;
break ;
case 'featurerequest' :
url += ` vote.php?fid= ${ data . id } ` ;
break ;
default : //TODO
return ;
}
text = l ( ` events.rtb_ ${ data . type } ` , ` [user] ${ data . name } [/user] ` ,
data . title !== undefined ? ` [url= ${ url } ] ${ data . title } [/url] ` : url ) ;
character = data . name ;
}
2018-01-06 16:14:21 +00:00
await addEventMessage ( new EventMessage ( text , time ) ) ;
2017-09-02 01:50:31 +00:00
if ( data . type === 'note' )
2018-07-20 01:12:26 +00:00
await core . notifications . notify ( state . consoleTab , character , text , characterImage ( character ) , 'newnote' ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2019-01-03 17:38:17 +00:00
const sfcList : Interfaces.SFCMessage [ ] = [ ] ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'SFC' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
let text : string , message : Interfaces.Message ;
if ( data . action === 'report' ) {
text = l ( 'events.report' , ` [user] ${ data . character } [/user] ` , decodeHTML ( data . tab ) , decodeHTML ( data . report ) ) ;
2018-07-20 01:12:26 +00:00
if ( ! data . old )
await core . notifications . notify ( state . consoleTab , data . character , text , characterImage ( data . character ) , 'modalert' ) ;
2017-09-02 01:50:31 +00:00
message = new EventMessage ( text , time ) ;
safeAddMessage ( sfcList , message , 500 ) ;
2019-01-03 17:38:17 +00:00
( < Interfaces.SFCMessage > message ) . sfc = data ;
2017-09-02 01:50:31 +00:00
} else {
text = l ( 'events.report.confirmed' , ` [user] ${ data . moderator } [/user] ` , ` [user] ${ data . character } [/user] ` ) ;
for ( const item of sfcList )
if ( item . sfc . logid === data . logid ) {
item . sfc . confirmed = true ;
break ;
}
message = new EventMessage ( text , time ) ;
}
2018-01-06 16:14:21 +00:00
return addEventMessage ( message ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'STA' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
if ( data . character === core . connection . character ) {
2018-01-06 16:14:21 +00:00
await addEventMessage ( new EventMessage ( l ( data . statusmsg . length > 0 ? 'events.status.ownMessage' : 'events.status.own' ,
2017-09-02 01:50:31 +00:00
l ( ` status. ${ data . status } ` ) , decodeHTML ( data . statusmsg ) ) , time ) ) ;
return ;
}
const char = core . characters . get ( data . character ) ;
if ( ! isOfInterest ( char ) ) return ;
const status = l ( ` status. ${ data . status } ` ) ;
const key = data . statusmsg . length > 0 ? 'events.status.message' : 'events.status' ;
const message = new EventMessage ( l ( key , ` [user] ${ data . character } [/user] ` , status , decodeHTML ( data . statusmsg ) ) , time ) ;
2018-01-06 16:14:21 +00:00
await addEventMessage ( message ) ;
2017-09-02 01:50:31 +00:00
const conv = state . privateMap [ data . character . toLowerCase ( ) ] ;
2018-04-16 23:14:13 +00:00
if ( conv !== undefined && ( ! core . state . settings . eventMessages || conv !== state . selectedConversation ) )
await conv . addMessage ( message ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'SYS' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
state . selectedConversation . infoText = data . message ;
2018-01-06 16:14:21 +00:00
return addEventMessage ( new EventMessage ( data . message , time ) ) ;
2017-09-02 01:50:31 +00:00
} ) ;
2018-03-04 02:32:26 +00:00
connection . onMessage ( 'UPT' , async ( data , time ) = > addEventMessage ( new EventMessage ( l ( 'events.uptime' ,
data . startstring , data . channels . toString ( ) , data . users . toString ( ) , data . accepted . toString ( ) , data . maxusers . toString ( ) ) , time ) ) ) ;
2018-01-06 16:14:21 +00:00
connection . onMessage ( 'ZZZ' , async ( data , time ) = > {
2017-09-02 01:50:31 +00:00
state . selectedConversation . infoText = data . message ;
2018-01-06 16:14:21 +00:00
return addEventMessage ( new EventMessage ( data . message , time ) ) ;
2017-09-02 01:50:31 +00:00
} ) ;
return state ;
2020-03-14 19:28:14 +00:00
}