/******************************************************************************
 Functions coded by Flier (THANX to Sheik!!)

 CheckInvite         Checks signon user against list and invites if possible
 BuildPrivs          Returns privilege as flags
 AutoNickComplete    Completes the nick at the beginning of the input line
 RemoveFromDCCList   Removes nick from DCC list
 CheckPermBans       Checks if permanent ban isn't there
 TraceKill           Does trace then kills users matching filter
 Map                 Generates map of IRC servers
 AddToMap            Stores map info to memory
 PrintMap            Prints map info
 FindShit            Updates pointer to shit list
 CheckTimeMinute     Checks for things every minute
 AddFriendPrivs      Adds/removes flags for userlist entries
 AddFriendChannel    Adds/removes channels for userlist entries
 MassKick            Kickc multiple nicks at the same time
 ServerPing	     Pings a server for precise lag time accros net - Zakath
 OnOffCommand        Sets ScrollZ settings (on/off)
 NumberCommand       Sets ScrollZ settings (number)
 SetAutoCompletion   Sets auto completion on/auto string/off)
 SetIdleKick         Sets idle kick on/auto channels/off)
 ChannelCommand      Sets ScrollZ settings (on channels/off)
 Usage               Prints usage for command
 ConvertmIRC         Converts mIRC colors to ANSI
 ScrollZTrace        Sends TRACE command, allows for switches - by Zakath
 HandleTrace         Formats TRACE - by Zakath
 ColorUserHost       Colorizes user@host
 AddChannel          Adds/removes channels to ScrollZ settings
 HashFunc            Used to calculate hash value
 ScrollZInfo         Shows some ScrollZ related information
 ChannelCreateTime   Channel creation time (numeric 329)
 ExecUptime          Displays your systems uptime
 CTCPFing            CTCP Finger's someone.
 NoSuchServer4SPing  Checks whether we tried to ping non existing server
 GetNetsplitServer   Returns server from netsplit info
 HandleDelayOp       Handles delayed opping
 HandleDelayNotify   Handles delayed notify
 ShowIdle            Shows idle time for users
 SwitchNick          Try to switch nick
 MassKill	     Kills multiple nicks at once - by acidflash
 CheckServer         Check if server is valid
 CleanUpScrollZVars  Clean up ScrollZ allocated variables
 CleanUp             Clean up memory on exit
 EncryptMsg          Support for encrypted communication sessions
 EncryptMessage      Encrypt message
 DecryptMessage      Decrypt message
 FilterTrace         Filtered trace
 DoFilterTrace       Actual filtered trace
 FormatTime          Format time in compressed format
 AddJoinKey          Store key for later join
 CheckJoinKey        Check key for join
 TryChannelJoin      Try to join a channel
 ChangePassword      Change user's password
 StatsIFilter        Does /STATS i with filter
 HandleStatsI        Handles STATS i reply from server
 ARinWindowToggle    Toggle value of ARinWindow
 IsIrcNetOperChannel Figure out if we're dealing with IrcNet's oper channel
 RedrawAll           Force redraw of status bars and input prompt
 CompareAddr         Compare two List pointers (for list.c)
 TopicLocked         Topic locking
 TimeStamp           Returns time stamp in static buffer
 CJoin               CJOIN and KNOCK commands
 ChanLogCommand      Configures per channel logging
 UpdateChanLogFName  Update channel logging filename
 ChannelLogReport    Report channel logging event (start, stop)
 ChannelLogReportAll Report channel logging event for all channels
 ChannelLogSave      Save channel logging event (join, kick, ...)
 CdExceptions        Clears channel of ban exceptions
 SendToServer        Sends data to server (used from timer)
 RateLimitJoin       Should we rate limit the join
 ExtendTopic         Extends current topic with given text
 TopicDelimiter      Sets topic delimiter
******************************************************************************/

/*
 * $Id: edit6.c,v 1.159 2006/04/30 14:15:43 f Exp $
 */

#include "irc.h"
#include "crypt.h"
#include "vars.h"
#include "ircaux.h"
#include "window.h"
#include "whois.h"
#include "hook.h"
#include "input.h"
#include "ignore.h"
#include "keys.h"
#include "list.h"
#include "names.h"
#include "alias.h"
#include "history.h"
#include "funny.h"
#include "exec.h"
#include "ctcp.h"
#include "dcc.h"
#include "translat.h"
#include "output.h"
#include "notify.h"
#include "numbers.h"
#include "status.h"
#include "screen.h"
#include "server.h"
#include "edit.h"
#include "struct.h"
#include "parse.h"
#include "myvars.h" 
#include "whowas.h"

#include <sys/stat.h> /* for umask() */

#define SZMAXCRYPTSIZE 304

void PrintUsage _((char *));
#ifdef EXTRAS
void CheckTopic _((char *, int, ChannelList *));
#endif
void EncryptAdd _((char *, char *));
void EncryptMasterPass _((char *, char *));
void EncryptMasterDelUser _((char *, char *));
void EncryptDelUser _((char *));
void EncryptMasterList _((char *, char *));
void EncryptList _((char *));
void UpdateChanLogFName _((ChannelList *));
void ChannelLogReport _((char *, ChannelList *));
int CheckServer _((int));

extern NickList *CheckJoiners _((char *, char *, int , ChannelList *));
extern struct friends *CheckUsers _((char *, char *));
extern void PrintSetting _((char *, char *, char *, char *));
extern int  CheckChannel _((char *, char *));
extern void FServer _((char *, char *, char *));
extern void SetAway _((char *, char *, char *));
extern void NotChanOp _((char *));
extern void NoWindowChannel _((void));
extern void AutoJoinOnInvToggle _((char *, char *, char *));
extern void NHProtToggle _((char *, char *, char *));
extern int  matchmcommand _((char *, int));
extern int  AddLast _((List *, List *));
extern void AddJoinChannel _((void));
extern void HandleUserhost _((WhoisStuff *, char *, char *));
extern void CreateBan _((char *, char *, char *));
extern void ColorizeJoiner _((NickList *, char *));
extern void CleanUpCdcc _((void));
extern void CleanUpIgnore _((void));
extern void CleanUpTimer _((void));
extern void CleanUpLists _((void));
extern void CleanUpWhowas _((void));
extern void CleanUpWindows _((void));
extern void CleanUpFlood _((void));
extern void CleanUpVars _((void));
extern void Dump _((char *, char *, char *));
extern int  EncryptString _((char *, char *, char *, int, int));
extern int  DecryptString _((char *, char *, char *, int, int));
extern void queuemcommand _((char *));
extern void CheckDCCSpeed _((DCC_list *, time_t));
/* Patched by Zakath */
extern void CeleAway _((int));
extern char *OpenCreateFile _((char *, int));
extern void StripAnsi _((char *, char *, int));

extern void e_channel _((char *, char *, char *));
extern void e_nick _((char *, char *, char *));
extern void timercmd _((char *, char *, char *));
extern void clear_channel _((ChannelList *));

/* Patched by Zakath */
#ifdef CELE
static int TraceAll=1;
static int TraceOper=0;
static int TraceServer=0;
static int TraceClass=0;
static int TraceUser=0;
#endif
/* ***************** */
static struct mapstr *maplist=NULL;

#ifdef OPER
int StatsiNumber;
int StatscNumber;
int StatslNumber;
int StatsdNumber;
static int  FilterKillNum;
static char *tkillreason=(char *) 0;
static char *tkillpattern=(char *) 0;
/* for filtered trace */
static char *ftpattern=(char *) 0;
int  tottcount=0;
int  mattcount=0;
#endif

static struct commands {
    char *command;
    int  *var;
    char **strvar;
    char *setting;
    char *setting2;
} command_list[]= {
    { "AJOIN"       , &AutoJoinOnInv  , &AutoJoinChannels      , NULL                         , NULL },
    { "AREJOIN"     , &AutoRejoin     , &AutoRejoinChannels    , "Auto rejoin"                , NULL },
    { "BITCH"       , &Bitch          , &BitchChannels         , "Bitch mode"                 , NULL },
    { "DPROT"       , &MDopWatch      , &MDopWatchChannels     , "Mass deop watch"            , NULL },
    { "FAKE"        , &ShowFakes      , &ShowFakesChannels     , "Fake modes display"         , NULL },
    { "FRLIST"      , &FriendList     , &FriendListChannels    , "Friend list"                , NULL },
    { "KICKONBAN"   , &KickOnBan      , &KickOnBanChannels     , "Kick on ban"                , NULL },
    { "KICKONFLOOD" , &KickOnFlood    , &KickOnFloodChannels   , "Kick on flood"              , NULL },
    { "KICKOPS"     , &KickOps        , &KickOpsChannels       , "Kick channel operators"     , NULL },
    { "KPROT"       , &KickWatch      , &KickWatchChannels     , "Mass kick watch"            , NULL },
    { "NHPROT"      , &NHProt         , &NHProtChannels        , NULL                         , NULL },
    { "NPROT"       , &NickWatch      , &NickWatchChannels     , "Nick flood watch"           , NULL },
    { "ORIGNICK"    , &OrigNickChange , &OrigNick              , "Reverting to original nick" , ", wanted nick :" },
    { "SHOWAWAY"    , &ShowAway       , &ShowAwayChannels      , "Notifying on away/back"     , NULL },
    { "COMPRESS"    , &CompressModes  , &CompressModesChannels , "Compress modes"             , NULL },
    { "BKLIST"      , &BKList         , &BKChannels            , "Shit list"                  , NULL },
    { "CHANLOG"     , &ChanLog        , &ChanLogChannels       , "Per channel logging"        , NULL },
#ifdef EXTRAS
    { "CHSIGNOFF"   , &ShowSignoffChan, &SignoffChannels       , "Show channels in signoff"   , NULL },
#endif
#if defined(EXTRAS) || defined(FLIER)
    { "AUTOINV"     , &AutoInv        , &AutoInvChannels       , "Auto invite on notify"      , NULL },
#endif
#ifdef ACID
    { "FORCEJOIN"   , &ForceJoin      , &ForceJoinChannels     , "Force channel join"         , NULL },
#endif
    { NULL          , NULL            , NULL                   , NULL                         , NULL }
};

#define NUM_JOIN_KEYS 10
static int joinkeycount=0;
static struct joinkeystr {
    char *key;
    char *channel;
} joinkeys[NUM_JOIN_KEYS];

extern char *ScrollZver1;
extern char *HelpPathVar;
extern char *CelerityNtfy;
extern char *TimeStampString;

extern time_t start_time;

#if defined(EXTRAS) || defined(FLIER)
/* Checks if signed on user should be invited */
void CheckInvite(nick,userhost,server)
char *nick;
char *userhost;
int  server;
{
    char tmpbuf[mybufsize/4];
    NickList *tmpnick;
    ChannelList *tmpchan;
    struct friends *tmpfriend;

    if (AutoInv) {
        snprintf(tmpbuf,sizeof(tmpbuf),"%s!%s",nick,userhost);
        for (tmpchan=server_list[server].chan_list;tmpchan;tmpchan=tmpchan->next) {
            if (!CheckChannel(tmpchan->channel,AutoInvChannels)) continue;
#if defined(ACID) || defined(FLIER)
            if (HAS_OPS(tmpchan->status))
#else
            if (HAS_OPS(tmpchan->status) && ((tmpchan->mode)&MODE_INVITE))
#endif /* ACID || FLIER */
            {
                tmpnick=find_in_hash(tmpchan,nick);
                tmpfriend=CheckUsers(tmpbuf,tmpchan->channel);
                if (!tmpnick && (tmpfriend=CheckUsers(tmpbuf,tmpchan->channel)) &&
                    !(tmpfriend->passwd) &&
#ifdef ACID
                    ((tmpfriend->privs)&FLINVITE)
#else
                    ((tmpfriend->privs)&(FLINVITE | FLCHOPS | FLOP | FLUNBAN))==
                     (FLINVITE | FLCHOPS | FLOP | FLUNBAN)
#endif /* ACID */
                     ) {
#ifndef VILAS
                         if (tmpchan->key)
                             send_to_server("NOTICE %s :You have been ctcp invited to %s (key is %s) -ScrollZ-",nick,tmpchan->channel,tmpchan->key);
#else
                         if (tmpchan->key)
                             send_to_server("NOTICE %s :The channel %s key is %s",nick,
                                            tmpchan->channel,tmpchan->key);
#endif /* VILAS */
                         send_to_server("INVITE %s %s",nick,tmpchan->channel);
                     }
            }
        }
            
    }
}
#endif /* EXTRAS || FLIER */

/* Returns privilege as flags */
void BuildPrivs(user,buffer)
struct friends *user;
char *buffer;
{
    if (user && user->privs) {
        if ((user->privs)&FLINVITE) strcat(buffer,"I");
        if ((user->privs)&FLCHOPS) strcat(buffer,"C");
        if ((user->privs)&FLVOICE) strcat(buffer,"V");
        if ((user->privs)&FLOP) strcat(buffer,"O");
	if ((user->privs)&FLHOP) strcat(buffer,"H");
        if ((user->privs)&FLAUTOOP) strcat(buffer,"A");
        if ((user->privs)&FLUNBAN) strcat(buffer,"U");
        if ((user->privs)&FLPROT) strcat(buffer,"P");
        if ((user->privs)&FLCDCC) strcat(buffer,"D");
        if ((user->privs)&FLGOD) strcat(buffer,"G");
        if ((user->privs)&FLJOIN) strcat(buffer,"J");
        if ((user->privs)&FLNOFLOOD) strcat(buffer,"F");
        if ((user->privs)&FLINSTANT) strcat(buffer,"X");
        if ((user->privs)&FLWHOWAS) strcat(buffer,"Z");
    }
}

/* Completes the nick at the beginning of the input line */
void AutoNickComplete(line,result,tmpchan)
char *line;
char *result;
ChannelList *tmpchan;
{
    int  len;
    int  count=0;
    char *tmp=AutoReplyString;
    char *mynick;
    char *tmpstr=result;
    char tmpbuf[mybufsize];
    NickList *nick=NULL;
    NickList *tmpnick;

    strcpy(result,line);
    if (*tmpstr==':') return;
    while (*tmpstr && *tmpstr>' ' && *tmpstr!=':') tmpstr++;
    if (*tmpstr==':') {
        *tmpstr='\0';
        tmpstr++;
        len=strlen(result);
        mynick=get_server_nickname(tmpchan->server);
        /* count number of matches and save first match */
        for (tmpnick=tmpchan->nicks;tmpnick;tmpnick=tmpnick->next)
            if (!my_strnicmp(tmpnick->nick,result,len) &&
                my_stricmp(mynick,tmpnick->nick)) {
                count++;
                if (!nick) nick=tmpnick;
                /* if it is exact match stop */
                if (!my_stricmp(tmpnick->nick,result)) {
                    nick=tmpnick;
                    break;
                }
            }
        /* stop if set to AUTO and we have a match, same if there
           was only one match found */
        if (count==1 || (AutoNickCompl==2 && count)) tmpnick=NULL;
        if (!tmpnick && nick) {
            if (count==1 || AutoNickCompl==2) strmcpy(tmpbuf,nick->nick,sizeof(tmpbuf));
            else if (AutoNickCompl==1) strmcpy(tmpbuf,result,sizeof(tmpbuf));
            if (*tmpstr==' ' && tmp && *(tmp+1)==' ') tmpstr++;
            if (tmp && *tmp) strmcat(tmpbuf,tmp,sizeof(tmpbuf));
            strmcat(tmpbuf,tmpstr,sizeof(tmpbuf));
            strcpy(result,tmpbuf);
            return;
        }
    }
    strcpy(result,line);
}

/* Removes nick from DCC list */
void RemoveFromDCCList(nick)
char *nick;
{
    DCC_list *Client;
    DCC_list *tmp;

    for (Client=ClientList;Client;Client=tmp) {
        tmp=Client->next;
        if (Client->user && !my_stricmp(nick,Client->user) &&
            !((Client->flags)&DCC_ACTIVE))
            dcc_erase(Client);
    }

}

/* Checks if permanent ban isn't there */
void CheckPermBans(chan)
ChannelList *chan;
{
    int  max=get_int_var(MAX_MODES_VAR);
    int  count=0;
    char *modes=(char *) 0;
    char tmpbuf[mybufsize/2];
    char modebuf[mybufsize/32];
    struct bans *tmpban;
    struct autobankicks *tmpabk;

    if (chan) {
        strcpy(modebuf,"+bbbbbbbb");
        for (tmpabk=abklist;tmpabk;tmpabk=tmpabk->next) {
            if ((tmpabk->shit)&8) {
                if (!CheckChannel(chan->channel,tmpabk->channels)) continue;
                for (tmpban=chan->banlist;tmpban;tmpban=tmpban->next) {
                    if (tmpban->exception) continue;
                    if (wild_match(tmpban->ban,tmpabk->userhost) ||
                        wild_match(tmpabk->userhost,tmpban->ban))
                        break;
                }
                if (!tmpban) {
                    snprintf(tmpbuf,sizeof(tmpbuf)," %s",tmpabk->userhost);
                    malloc_strcat(&modes,tmpbuf);
                    count++;
                    if (count==max) {
                        modebuf[count+1]='\0';
                        send_to_server("MODE %s %s %s",chan->channel,modebuf,modes);
                        new_free(&modes);
                        count=0;
                    }
                }
            }
        }
        if (count) {
            modebuf[count+1]='\0';
            send_to_server("MODE %s %s %s",chan->channel,modebuf,modes);
            new_free(&modes);
        }
    }
}

/* Does trace then kills users matching filter */
#ifdef OPER
void TraceKill(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *filter;

    if (inSZFKill) {
        say("Already doing filter kill or trace kill");
        return;
    }
    if (args && *args) {
        filter=new_next_arg(args,&args);
        malloc_strcpy(&tkillpattern,filter);
        if (args && *args) malloc_strcpy(&tkillreason,args);
        else new_free(&tkillreason);
        FilterKillNum=0;
        inSZFKill=1;
        inSZTrace=1;
        send_to_server("TRACE");
    }
    else PrintUsage("TKILL filter [reason]");
}

/* Does the actual killing */
void DoTraceKill(user)
char *user;
{
    char *nick;
    char *host=(char *) 0;
    char *tmpstr;
    char tmpbuf[mybufsize/4];

    nick=user;
    if ((host=index(user,'['))) {
        *host++='\0';
        if ((tmpstr=index(host,']'))) *tmpstr='\0';
        else host=NULL;
    }
    if (nick && *nick && host && *host) {
        if (!my_stricmp(nick,get_server_nickname(from_server))) return;
        snprintf(tmpbuf,sizeof(tmpbuf),"%s!%s",nick,host);
        if (wild_match(tmpbuf,tkillpattern) || wild_match(tkillpattern,tmpbuf)) {
            FilterKillNum++;
            if (tkillreason) strmcpy(tmpbuf,tkillreason,sizeof(tmpbuf));
            else strmcpy(tmpbuf,nick,sizeof(tmpbuf));
            send_to_server("KILL %s :%s (%d)",nick,tmpbuf,FilterKillNum);
        }
    }
}

/* Reports statistics for filter kill */
void HandleEndOfTraceKill() {
    if (inSZFKill) {
        say("Total of %d users were killed",FilterKillNum);
        inSZFKill=0;
        new_free(&tkillreason);
        new_free(&tkillpattern);
    }
}
#endif /* OPER */

/* Generates map of IRC servers */
#ifndef LITE
void Map(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *server;
    char servbuf[mybufsize/16+4];
    time_t timenow=time((time_t *) 0);

    if (timenow-LastLinks>=120 || !inSZLinks) {
        server=new_next_arg(args,&args);
        LastLinks=timenow;
        inSZLinks=4;
        strcpy(servbuf,"LINKS ");
        if (server) {
            strmcat(servbuf,server,sizeof(servbuf));
            strmcat(servbuf," *",sizeof(servbuf));
        }
        send_to_server("%s",servbuf);
        say("Generating map of IRC servers");
    }
    else say("Wait till previous LINKS, LLOOK, LLOOKUP or MAP completes");
}

/* This stores map info to memory */
void AddToMap(server,distance)
char *server;
char *distance;
{
    int dist=atoi(distance);
    struct mapstr *tmpmap;
    struct mapstr *insmap;
    struct mapstr *prevmap;

    tmpmap=(struct mapstr *) new_malloc(sizeof(struct mapstr));
    if (tmpmap) {
        tmpmap->server=(char *) 0;
        tmpmap->distance=dist;
        tmpmap->next=NULL;
        if (server) malloc_strcpy(&tmpmap->server,server);
        if (!maplist) {
            maplist=tmpmap;
            return;
        }
        for (insmap=maplist,prevmap=maplist;insmap && insmap->distance<dist;) {
            prevmap=insmap;
            insmap=insmap->next;
        }
        if (insmap && insmap->distance>=dist) {
            tmpmap->next=insmap;
            if (insmap==maplist) maplist=tmpmap;
            else prevmap->next=tmpmap;
        }
        else prevmap->next=tmpmap;
    }
}

/* This prints map */
void PrintMap() {
    int  prevdist=0;
    char *ascii;
    char tmpbuf1[mybufsize/2];
    char tmpbuf2[mybufsize/4];
    struct mapstr *tmpmap;

#ifdef HAVE_ICONV_H
    if (get_int_var(HIGH_ASCII_VAR)) ascii="\342\224\224\342\224\200> ";
#else
    if (get_int_var(HIGH_ASCII_VAR)) ascii="ÀÄ> ";
#endif /* HAVE_ICONV_H */
    else ascii="`-> ";
    if (maplist) prevdist=maplist->distance;
    for (tmpmap=maplist;maplist;tmpmap=maplist) {
        maplist=maplist->next;
#ifdef WANTANSI
        if (!tmpmap->distance || prevdist!=tmpmap->distance)
            snprintf(tmpbuf2,sizeof(tmpbuf2),"[%s%d%s]",
                    CmdsColors[COLLINKS].color3,tmpmap->distance,Colors[COLOFF]);
        else snprintf(tmpbuf2,sizeof(tmpbuf2),empty_string);
        snprintf(tmpbuf1,sizeof(tmpbuf1),"%%s%%%ds%%s%s%s%s %s",tmpmap->distance*4,
                CmdsColors[COLLINKS].color1,tmpmap->server,Colors[COLOFF],tmpbuf2);
        say(tmpbuf1,CmdsColors[COLLINKS].color4,
            prevdist!=tmpmap->distance?ascii:empty_string,Colors[COLOFF]);
#else
        if (!tmpmap->distance || prevdist!=tmpmap->distance)
            snprintf(tmpbuf2,sizeof(tmpbuf2),"[%d]",tmpmap->distance);
        else *tmpbuf2='\0';
        snprintf(tmpbuf1,sizeof(tmpbuf1),"%%%ds%%s %s",tmpmap->distance*3,tmpbuf2);
        say(tmpbuf1,prevdist!=tmpmap->distance?ascii:empty_string,
            tmpmap->server);
#endif
        prevdist=tmpmap->distance;
        new_free(&(tmpmap->server));
        new_free(&tmpmap);
    }
}
#endif

/* Updates pointer to shit list */
struct autobankicks *FindShit(userhost,channel)
char *userhost;
char *channel;
{
    struct autobankicks *tmpabk;

    for (tmpabk=abklist;tmpabk;tmpabk=tmpabk->next)
        if ((wild_match(tmpabk->userhost,userhost) ||
             wild_match(userhost,tmpabk->userhost)) &&
            CheckChannel(channel,tmpabk->channels))
                return(tmpabk);
    return(NULL);
}

/* Checks for things every minute */
void CheckTimeMinute() {
    int  i;
#ifdef EXTRAS
    int  max=get_int_var(MAX_MODES_VAR);
#endif
    int  found;
    int  old_server;
    int  wildcards;
    char *tmpstr;
    char *tmpstr1;
    char tmpbuf[mybufsize/2];
#ifdef EXTRAS
    char tmpbuf2[mybufsize/4];
#endif
    time_t timenow=time((time_t *) 0);
    static time_t lastqueuechk=0;
    struct wholeftch  *tmpch;
    struct wholeftch  *wholch;
    struct wholeftstr *wholeft;
    struct wholeftstr *tmpwholeft;
    struct list *tmplist;
    DCC_list *Client;
#ifdef EXTRAS
    NickList *tmpnick;
#endif
    ChannelList *tmpchan;

    if (curr_scr_win->server!=-1) {
        if (AutoAwayTime>0 && timenow-idle_time>AutoAwayTime*60 && !away_set) {
#ifdef CELE
            snprintf(tmpbuf,sizeof(tmpbuf),"Inactive - %d mins (Auto SetAway)",AutoAwayTime);
#else
            snprintf(tmpbuf,sizeof(tmpbuf),"Automatically set away");
#endif
            say("Setting you away after being idle for %d minutes",AutoAwayTime);
            SetAway(NULL,tmpbuf,NULL);
            idle_time=timenow;
        }
#ifdef CELE
        else if (away_set) CeleAway(1); /* Updates idle counter in away */
#endif /* CELE */
        if (away_set && AutoJoinOnInv==2 && AutoJoinChannels && timenow-LastCheck>500) {
            strmcpy(tmpbuf,AutoJoinChannels,sizeof(tmpbuf));
            tmpstr=tmpbuf;
            tmpstr1=tmpbuf;
            wildcards=0;
            while (*tmpstr) {
                if (*tmpstr=='?' || *tmpstr=='*') wildcards=1;
                if (*tmpstr==',') {
                    *tmpstr='\0';
                    if (!wildcards) {
                        for (tmpchan=server_list[curr_scr_win->server].chan_list;tmpchan;
                             tmpchan=tmpchan->next)
                            if (!my_stricmp(tmpstr1,tmpchan->channel)) break;
                        if (!tmpchan) e_channel("JOIN",tmpstr1,tmpstr1);
                    }
                    tmpstr1=tmpstr+1;
                    wildcards=0;
                }
                tmpstr++;
            }
            if (!wildcards) {
                for (tmpchan=server_list[curr_scr_win->server].chan_list;tmpchan;
                     tmpchan=tmpchan->next)
                    if (!my_stricmp(tmpstr1,tmpchan->channel)) break;
                if (!tmpchan) e_channel("JOIN",tmpstr1,tmpstr1);
            }
            LastCheck=timenow;
        }
#ifdef EXTRAS
        i=0;
        for (tmpchan=server_list[curr_scr_win->server].chan_list;tmpchan;tmpchan=tmpchan->next)
            if (tmpchan->IdleKick && HAS_OPS(tmpchan->status)) {
                if (i==max) break;
                for (tmpnick=tmpchan->nicks;tmpnick;tmpnick=tmpnick->next) {
                    if (i==max) break;
                    if (tmpnick->frlist || tmpnick->chanop) continue;
                    if (tmpnick->hasvoice && tmpchan->IdleKick==1) continue;
                    if (timenow-tmpnick->lastmsg>IdleTime*60) {
                        i++;
                        CreateBan(tmpnick->nick,tmpnick->userhost,tmpbuf);
                        send_to_server("MODE %s -o+b %s %s",tmpchan->channel,tmpnick->nick,
                                       tmpbuf);
                        send_to_server("KICK %s %s :Idle user",tmpchan->channel,tmpnick->nick);
                        snprintf(tmpbuf2,sizeof(tmpbuf2),"30 MODE %s -b %s",tmpchan->channel,tmpbuf);
                        timercmd("TIMER",tmpbuf2,NULL);
                    }
                }
            }
#endif /* EXTRAS */
    }
    wholeft=wholist;
    tmpwholeft=wholist;
    while (wholeft) {
        if (timenow-wholeft->time>600) {
            wholch=wholeft->channels;
            while (wholch) {
                while (wholch->nicklist) {
                    tmplist=wholch->nicklist;
                    wholch->nicklist=tmplist->next;
                    new_free(&(tmplist->nick));
                    new_free(&(tmplist->userhost));
                    new_free(&tmplist);
                }
                tmpch=wholch->next;
                new_free(&(wholch->channel));
                new_free(&(wholch));
                wholch=tmpch;
            }
            if (wholeft==wholist) wholist=wholeft->next;
            else tmpwholeft->next=wholeft->next;
            new_free(&(wholeft->splitserver));
            new_free(&wholeft);
        }
        tmpwholeft=wholeft;
        if (wholeft) wholeft=wholeft->next;
    }
    if (get_int_var(AUTO_RECONNECT_VAR) && LastServer+117<timenow) {
        found=0;
        for (i=0;i<number_of_servers;i++) found|=server_list[i].connected;
        if (!found && number_of_servers>0) {
            say("None of the servers is connected, connecting to next server in list");
            FServer(NULL,"+",NULL);
        }
    }
    if (lastqueuechk+300<timenow) {
        strcpy(tmpbuf,"flush");
        queuemcommand(tmpbuf);
        lastqueuechk=timenow;
    }
    for (Client=ClientList;Client;Client=Client->next) {
        if (Client->minspeed>=0.001 && timenow>=Client->CdccTime) {
            old_server=from_server;
            from_server=Client->server;
            CheckDCCSpeed(Client,timenow);
            from_server=old_server;
        }
    }
    clean_whowas_list();
    clean_whowas_chan_list();
#ifdef CELE
    build_status((char *) 0);
#endif
}

/* Adds/removes flags for userlist entries */
void AddFriendPrivs(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int  i;
    int  add=!strcmp(command,"ADD")?1:0;
    int  count=0;
    int  privs=0;
    char *filter;
    char *flags;
    char tmpbuf[mybufsize/4];
    struct friends *tmpfriend;

    filter=new_next_arg(args,&args);
    flags=new_next_arg(args,&args);
    if (filter && flags) {
        while (*flags) {
            if (*flags=='I' || *flags=='i') privs|=FLINVITE;
            else if (*flags=='C' || *flags=='c') privs|=FLCHOPS;
            else if (*flags=='V' || *flags=='v') privs|=FLVOICE;
            else if (*flags=='O' || *flags=='o') privs|=FLOP;
	    else if (*flags=='H' || *flags=='h') privs|=FLHOP;
            else if (*flags=='A' || *flags=='a') privs|=FLAUTOOP;
            else if (*flags=='U' || *flags=='u') privs|=FLUNBAN;
            else if (*flags=='P' || *flags=='p') privs|=FLPROT;
            else if (*flags=='D' || *flags=='d') privs|=FLCDCC;
            else if (*flags=='G' || *flags=='g') privs|=FLGOD;
            else if (*flags=='J' || *flags=='j') privs|=FLJOIN;
            else if (*flags=='F' || *flags=='f') privs|=FLNOFLOOD;
            else if (*flags=='X' || *flags=='x') privs|=FLINSTANT;
            else if (*flags=='Z' || *flags=='z') privs|=FLWHOWAS;
            flags++;
        }
        for (i=1,tmpfriend=frlist;tmpfriend;i++,tmpfriend=tmpfriend->next) {
            if ((*filter=='#' && matchmcommand(filter,i)) ||
                    wild_match(tmpfriend->userhost,filter) ||
                    wild_match(filter,tmpfriend->userhost)) {
                if (add) {
                    if (((tmpfriend->privs)&privs)!=privs) {
                        tmpfriend->privs|=privs;
                        count++;
                    }
                }
                else {
                    if ((tmpfriend->privs)&privs) {
                        tmpfriend->privs&=(~privs);
                        count++;
                    }
                }
            }
        }
        say("%d out of %d userlist entries changed",count,i-1);
    }
    else {
        snprintf(tmpbuf,sizeof(tmpbuf),"%sFFLAG filter|#number flaglist",command);
        PrintUsage(tmpbuf);
    }
}

/* Adds/removes channel from a list separated by , */
int AddRemoveChannel(setting,channel,add)
char **setting;
char *channel;
int  add;
{
    int  change=0;
    char *tmpstr;
    char *tmpchan;
    char tmpbuf1[mybufsize/4];
    char tmpbuf2[mybufsize/4];

    if (add) {
        if (!CheckChannel(*setting,channel)) {
            change=1;
            snprintf(tmpbuf2,sizeof(tmpbuf2),",%s",channel);
            malloc_strcat(setting,tmpbuf2);
        }
    }
    else {
        tmpstr=tmpbuf2;
        strmcpy(tmpstr,*setting,sizeof(tmpbuf2));
        *tmpbuf1='\0';
        if ((tmpchan=strtok(tmpstr,","))) {
            do {
                if (my_stricmp(tmpchan,channel)) {
                    if (change) strmcat(tmpbuf1,",",sizeof(tmpbuf1));
                    strmcat(tmpbuf1,tmpchan,sizeof(tmpbuf1));
                    change=1;
                }
            }
            while ((tmpchan=strtok(NULL,",")));
            if (tmpbuf1[0] && my_stricmp(tmpbuf1,*setting)) {
                malloc_strcpy(setting,tmpbuf1);
                change=1;
            }
            else change=0;
        }
    }
    return(change?add+1:0);
}

/* Adds/removes channels for userlist entries */
void AddFriendChannel(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int  i,j;
    int  add=!strcmp(command,"ADD")?1:0;
    int  count=0;
    int  change;
    int  ischan=0;
    char *filter;
    char *channel;
    char tmpbuf[mybufsize/4];
    NickList *tmpnick;
    ChannelList *chan;
    struct friends *tmpfriend;

    filter=new_next_arg(args,&args);
    channel=new_next_arg(args,&args);
    if (filter && channel) {
        if (*filter=='#' && isalpha(*(filter+1))) ischan=1;
        for (i=0,tmpfriend=frlist;tmpfriend;i++,tmpfriend=tmpfriend->next)
            if ((*filter=='#' && matchmcommand(filter,i+1)) || ischan ||
                wild_match(tmpfriend->userhost,filter) ||
                wild_match(filter,tmpfriend->userhost)) {
                if (ischan && !CheckChannel(filter,tmpfriend->channels)) continue;
                if ((change=AddRemoveChannel(&(tmpfriend->channels),channel,add))) {
                    count++;
                    for (j=0;j<number_of_servers;j++)
                        for (chan=server_list[j].chan_list;chan;chan=chan->next)
                            if (CheckChannel(chan->channel,channel))
                                for (tmpnick=chan->nicks;tmpnick;tmpnick=tmpnick->next) {
                                    if (tmpnick->userhost)
                                        snprintf(tmpbuf,sizeof(tmpbuf),"%s!%s",tmpnick->nick,tmpnick->userhost);
                                    else strmcpy(tmpbuf,tmpnick->nick,sizeof(tmpbuf));
                                    if (change>1) {
                                        if (wild_match(tmpfriend->userhost,tmpbuf))
                                            tmpnick->frlist=tmpfriend;
                                    }
                                    else if (tmpnick->frlist==tmpfriend)
                                        tmpnick->frlist=(struct friends *) 0;
                                }
                }
            }
        say("%d out of %d userlist entries changed",count,i);
    }
    else {
        snprintf(tmpbuf,sizeof(tmpbuf),"%sFCHAN filter|#number channel",command);
        PrintUsage(tmpbuf);
    }
}

/* Kicks multiple nicks at the same time */
void MassKick(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *channel;
    char *comment;
    char *tmpnick;
    NickList *joiner;
    ChannelList *chan;

    if (args && *args) {
        if (is_channel(args)) channel=new_next_arg(args,&args);
        else if (!(channel=get_channel_by_refnum(0))) {
            NoWindowChannel();
            return;
        }
        if (!(comment=index(args,':'))) comment=DefaultK;
        else *comment++='\0';
        if (args && *args) {
            chan=lookup_channel(channel,curr_scr_win->server,0);
            if (chan && HAS_OPS(chan->status)) {
                while ((tmpnick=new_next_arg(args,&args))) {
                    joiner=CheckJoiners(tmpnick,channel,curr_scr_win->server,chan);
                    if (joiner) send_to_server("KICK %s %s :%s",channel,tmpnick,comment);
                    else say("Can't find %s on %s",tmpnick,channel);
                }
            }
            else NotChanOp(channel);
        }
        else PrintUsage("MK [#channel] nick1 [nick2] ... [:comment]");
    }
    else PrintUsage("MK [#channel] nick1 [nick2] ... [:comment]");
}

/* Pings a server for precise lag time accros net. - Zakath */
void ServerPing(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *server;
    struct spingstr *spingnew;
    struct spingstr *spingnext;
    struct spingstr *spingold=NULL;
    struct timeval timenow;

    server=new_next_arg(args,&args);
    if (server) {
        gettimeofday(&timenow,NULL);
        for (spingnew=spinglist;spingnew;spingnew=spingnext) {
            spingnext=spingnew->next;
            if (timenow.tv_sec-spingnew->sec>=600)
            {
                say("Removed server %s from list after 10 minutes",
                    spingnew->servername);
                if (spingold) spingold->next=spingnew->next;
                else spinglist=spingnew->next;
                new_free(&(spingnew->servername));
                new_free(&spingnew);
            }
            else spingold=spingnew;
        }
        spingnew=(struct spingstr *) new_malloc(sizeof(struct spingstr));
        spingnew->servername=(char *) 0;
        spingnew->usec=0L;
        spingnew->next=NULL;
        malloc_strcpy(&(spingnew->servername),server);
        add_to_list_ext((List **) &spinglist,(List *) spingnew,
                        (int (*) _((List *, List *))) AddLast);
        send_to_server("PING %s :%s",server_list[from_server].itsname,server);
#ifdef WANTANSI
        say("Sent server ping to %s%s%s",CmdsColors[COLCSCAN].color1,server,Colors[COLOFF]);
#else
        say("Sent server ping to %c%s%c",bold,server,bold);
#endif
        spingnew->sec=timenow.tv_sec;
        spingnew->usec=timenow.tv_usec;
    }
    else PrintUsage("SPING <Server to ping>");
}

/* Sets ScrollZ settings (on/off) */
void OnOffCommand(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int  i;
    char *tmpstr;
    char tmpbuf[mybufsize/8];
    struct commands {
        char *command;
        int  *var;
        char *setting;
    } command_list[]= {
        { "AUTOGET"     , &AutoGet         , "Cdcc auto-get" },
        { "AWAYENCR"    , &AwayEncrypt     , "Encryption of ScrollZ.away" },
        { "EGO"         , &Ego             , "Ego" },
        { "EXTMES"      , &ExtMes          , "Extended messages display" },
        { "EXTPUB"      , &ExtPub          , "Extended public messages display" },
        { "LOGON"       , &LogOn           , "Logging if not away" },
        { "LONGSTATUS"  , &LongStatus      , "Cdcc long status" },
#ifdef EXTRA_STUFF
        { "M"           , &RenameFiles     , "Cdcc M" },
#endif
#ifdef WANTANSI
        { "MIRC"        , &DisplaymIRC     , "Convert mIRC colors to ANSI" },
#endif
#ifdef EXTRAS
        { "NICKCHAN"    , &ShowNickAllChan , "Show nick change in all channels" },
#endif
        { "OVERWRITE"   , &CdccOverWrite   , "Cdcc overwrite" },
        { "SECURE"      , &Security        , "Cdcc security" },
        { "SERVNOTICE"  , &ServerNotice    , "Server notices display" },
        { "SHOWNICK"    , &ShowNick        , "Showing nick on public messages" },
        { "STAMP"       , &Stamp           , "Time stamp events" },
        { "STATS"       , &CdccStats       , "Cdcc stats in plist" },
        { "STATUS"      , &ShowDCCStatus   , "Cdcc showing on status bar" },
        { "VERBOSE"     , &CdccVerbose     , "Cdcc verbose mode" },
        { "WARNING"     , &DCCWarning      , "Check incoming DCCs" },
#ifdef EXTRAS
        { "SHOWSIGN"    , &ShowSignAllChan , "Show signoff message in all channels" },
#endif
        { "SHOWWALLOP"  , &ShowWallop      , "Show included/excluded nicks with wallops" },
        { NULL          , NULL             , NULL }
    };

    upper(command);
    for (i=0;command_list[i].command;i++) {
        if (!strcmp(command_list[i].command,command)) break;
    }
    if (!(command_list[i].command)) return;
    tmpstr=new_next_arg(args,&args);
    if (tmpstr) {
        if (!my_stricmp("ON",tmpstr)) *(command_list[i].var)=1;
        else if (!my_stricmp("OFF",tmpstr)) *(command_list[i].var)=0;
        else if (!strcmp("MIRC",command_list[i].command) &&
                 !my_stricmp("STRIP",tmpstr)) 
            *(command_list[i].var)=2;
        else if (!strcmp("VERBOSE",command_list[i].command) &&
                 !my_stricmp("QUIET",tmpstr)) 
            *(command_list[i].var)=2;
        else if (!strcmp("STAMP",command_list[i].command) &&
                 !my_stricmp("MAX",tmpstr)) 
            *(command_list[i].var)=2;
        else if (!strcmp("AUTOGET",command_list[i].command) &&
                 !my_stricmp("ALWAYS",tmpstr)) 
            *(command_list[i].var)=2;
        else {
            if (!strcmp("MIRC",command_list[i].command))
                snprintf(tmpbuf,sizeof(tmpbuf),"%s on/off/strip",command_list[i].command);
            else if (!strcmp("VERBOSE",command_list[i].command))
                snprintf(tmpbuf,sizeof(tmpbuf),"CDCC %s on/off/quiet",command_list[i].command);
            else if (!strcmp("AUTOGET",command_list[i].command))
                snprintf(tmpbuf,sizeof(tmpbuf),"CDCC %s on/off/always",command_list[i].command);
            else if (!strcmp("LONGSTATUS",command_list[i].command) ||
                     !strcmp("OVERWRITE",command_list[i].command)  ||
                     !strcmp("SECURE",command_list[i].command)     ||
                     !strcmp("STATS",command_list[i].command)      ||
                     !strcmp("WARNING",command_list[i].command))
                snprintf(tmpbuf,sizeof(tmpbuf),"CDCC %s on/off",command_list[i].command);
            else if (!strcmp("STAMP",command_list[i].command))
                snprintf(tmpbuf,sizeof(tmpbuf),"%s on/off/max",command_list[i].command);
            else snprintf(tmpbuf,sizeof(tmpbuf),"%s on/off",command_list[i].command);
            PrintUsage(tmpbuf);
            return;
        }
    }
    if (!strcmp("MIRC",command_list[i].command) && *(command_list[i].var)==2)
        PrintSetting(command_list[i].setting,"STRIP",empty_string,empty_string);
    else if (!strcmp("VERBOSE",command_list[i].command) && *(command_list[i].var)==2)
        PrintSetting(command_list[i].setting,"QUIET",empty_string,empty_string);
    else if (!strcmp("AUTOGET",command_list[i].command) && *(command_list[i].var)==2)
        PrintSetting(command_list[i].setting,"ALWAYS",empty_string,empty_string);
    else if (!strcmp("STAMP",command_list[i].command) && *(command_list[i].var)==2)
        PrintSetting(command_list[i].setting,"MAX",empty_string,empty_string);
    else if (*(command_list[i].var))
        PrintSetting(command_list[i].setting,"ON",empty_string,empty_string);
    else PrintSetting(command_list[i].setting,"OFF",empty_string,empty_string);
}

/* Sets ScrollZ settings (numbers) */
void NumberCommand(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int  i;
    int  number;
    int  isnumber=1;
    char *tmpstr;
    char tmpbuf[mybufsize/8];
    struct commands {
        char *command;
        int  *var;
        char *setting;
    } command_list[]= {
        { "AWAYT"       , &AutoAwayTime   , "Minutes before automatically setting you away" },
        { "DEOPS"       , &DeopSensor     , "Mass deop sensor" },
        { "DEOPT"       , &MDopTimer      , "Mass deop timer" },
        { "IDLE"        , &CdccIdle       , "Cdcc auto-close time" },
        { "IGTIME"      , &IgnoreTime     , "Ignore time" },
        { "KICKS"       , &KickSensor     , "Mass kick sensor" },
        { "KICKT"       , &KickTimer      , "Mass kick timer" },
        { "NICKS"       , &NickSensor     , "Nick flood sensor" },
        { "NICKT"       , &NickTimer      , "Nick flood timer" },
        { "PTIME"       , &PlistTime      , "Cdcc plist time" },
        { "NTIME"       , &NlistTime      , "Cdcc notice time" },
        { "ORIGNTIME"   , &OrigNickDelay  , "Delay before next attempt to switch to orignick" },
        { "AUTOOPDELAY" , &AutoOpDelay    , "Delay before automatically oping user" },
#ifdef EXTRAS
        { "IDLETIME"    , &IdleTime       , "Minutes till client starts kicking idlers" },
#endif
        { NULL          , NULL            , NULL }
    };

    upper(command);
    for (i=0;command_list[i].command;i++) {
        if (!strcmp(command_list[i].command,command)) break;
    }
    if (!(command_list[i].command)) return;
    if (*args) {
        for (tmpstr=args;*tmpstr;tmpstr++) isnumber&=isdigit(*tmpstr)?1:0;
        number=atoi(args);
        if (isnumber && number>-1) {
            *(command_list[i].var)=number;
            /* reset orignick timer */
            if (!strcmp(command_list[i].command,"ORIGNTIME")) LastNick=time((time_t *) 0);
        }
        else {
            snprintf(tmpbuf,sizeof(tmpbuf),"%s number",command_list[i].command);
            PrintUsage(tmpbuf);
            return;
        }
    }
    snprintf(tmpbuf,sizeof(tmpbuf),"%d",*(command_list[i].var));
    PrintSetting(command_list[i].setting,tmpbuf,empty_string,empty_string);
}

/* Sets channels' settings */
void SetChannels(setting)
int setting;
{
    int i;
    int oldlog;
    ChannelList *chan;
    WhowasChanList *whowas;

    for (i = 0; i < number_of_servers; i++) {
        for (chan = server_list[i].chan_list; chan; chan = chan->next) {
            switch (setting) {
                case 1: chan->AutoRejoin =
                    AutoRejoin ? CheckChannel(chan->channel, AutoRejoinChannels) : 0;
                    break;
                case 2: chan->Bitch =
                    Bitch ? CheckChannel(chan->channel, BitchChannels) : 0;
                    break;
                case 3: chan->MDopWatch =
                    MDopWatch ? CheckChannel(chan->channel, MDopWatchChannels) : 0;
                    break;
                case 4: chan->ShowFakes =
                    ShowFakes ? CheckChannel(chan->channel, ShowFakesChannels) : 0;
                    break;
                case 5: chan->FriendList =
                    FriendList ? CheckChannel(chan->channel, FriendListChannels) : 0;
                    break;
                case 6: chan->KickOnBan =
                    KickOnBan ? CheckChannel(chan->channel, KickOnBanChannels) : 0;
                    break;
                case 7: chan->KickOnFlood =
                    KickOnFlood ? CheckChannel(chan->channel, KickOnFloodChannels) : 0;
                    break;
                case 8: chan->KickOps =
                    KickOps ? CheckChannel(chan->channel, KickOpsChannels) : 0;
                    break;
                case 9: chan->KickWatch =
                    KickWatch ? CheckChannel(chan->channel, KickWatchChannels) : 0;
                    break;
                case 11: chan->NickWatch =
                    NickWatch ? CheckChannel(chan->channel, NickWatchChannels) : 0;
                    break;
                case 13: chan->ShowAway =
                    ShowAway ? CheckChannel(chan->channel, ShowAwayChannels) : 0;
                    break;
                case 14: chan->CompressModes =
                    CompressModes ? CheckChannel(chan->channel, CompressModesChannels) : 0;
                    break;
                case 15: chan->BKList =
                    BKList ? CheckChannel(chan->channel, BKChannels) : 0;
                    break;
                case 16:
                    oldlog = chan->ChanLog;
                    chan->ChanLog = ChanLog ? CheckChannel(chan->channel, ChanLogChannels) : 0;
                    if (!oldlog && chan->ChanLog) ChannelLogReport("started", chan);
                    break;
#ifdef EXTRAS
                case 99: chan->IdleKick=
                    IdleKick ? CheckChannel(chan->channel, IdleKickChannels) : 0;
                    break;
#endif
                case 100:
                    UpdateChanLogFName(chan);
                    break;
            }
#ifdef EXTRAS
            if (chan->IdleKick) chan->IdleKick = IdleKick;
#endif
        }
    }
    for (whowas = whowas_chan_list; whowas; whowas = whowas->next) {
        switch (setting) {
            case 1: whowas->channellist->AutoRejoin =
                AutoRejoin ? CheckChannel(whowas->channellist->channel, AutoRejoinChannels) : 0;
                break;
            case 2: whowas->channellist->Bitch =
                Bitch ? CheckChannel(whowas->channellist->channel, BitchChannels) : 0;
                break;
            case 3: whowas->channellist->MDopWatch =
                MDopWatch ? CheckChannel(whowas->channellist->channel, MDopWatchChannels) : 0;
                break;
            case 4: whowas->channellist->ShowFakes =
                ShowFakes ? CheckChannel(whowas->channellist->channel, ShowFakesChannels) : 0;
                break;
            case 5: whowas->channellist->FriendList =
                FriendList ? CheckChannel(whowas->channellist->channel, FriendListChannels) : 0;
                break;
            case 6: whowas->channellist->KickOnBan =
                KickOnBan ? CheckChannel(whowas->channellist->channel, KickOnBanChannels) : 0;
                break;
            case 7: whowas->channellist->KickOnFlood =
                KickOnFlood ? CheckChannel(whowas->channellist->channel, KickOnFloodChannels) : 0;
                break;
            case 8: whowas->channellist->KickOps =
                KickOps ? CheckChannel(whowas->channellist->channel, KickOpsChannels) : 0;
                break;
            case 9: whowas->channellist->KickWatch =
                KickWatch ? CheckChannel(whowas->channellist->channel, KickWatchChannels) : 0;
                break;
            case 11: whowas->channellist->NickWatch =
                NickWatch ? CheckChannel(whowas->channellist->channel, NickWatchChannels) : 0;
                break;
            case 13: whowas->channellist->ShowAway =
                ShowAway ? CheckChannel(whowas->channellist->channel, ShowAwayChannels) : 0;
                break;
            case 14: whowas->channellist->CompressModes =
                CompressModes ? CheckChannel(whowas->channellist->channel, CompressModesChannels) : 0;
                break;
            case 15: whowas->channellist->BKList =
                BKList ? CheckChannel(whowas->channellist->channel, BKChannels) : 0;
                break;
            case 16: whowas->channellist->ChanLog =
                ChanLog ? CheckChannel(whowas->channellist->channel, ChanLogChannels) : 0;
                break;
#ifdef EXTRAS
            case 99: whowas->channellist->IdleKick =
                IdleKick ? CheckChannel(whowas->channellist->channel, IdleKickChannels) : 0;
                break;
#endif
            case 100:
                UpdateChanLogFName(whowas->channellist);
                break;
        }
#ifdef EXTRAS
        if (whowas->channellist->IdleKick) whowas->channellist->IdleKick = IdleKick;
#endif
    }
    update_all_status();
}

/* Sets auto completion on/off */
void SetAutoCompletion(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int newset=0;
    char *tmpstr;

    if (args && *args) {
        if (!my_strnicmp("ON",args,2) || !my_strnicmp("AUTO",args,4)) {
            if (!my_strnicmp("ON",args,2)) {
                newset=1;
                tmpstr=args+2;
            }
            else {
                newset=2;
                tmpstr=args+4;
            }
            if (*tmpstr) *tmpstr++='\0';
            if (*tmpstr) malloc_strcpy(&AutoReplyString,tmpstr);
            else {
                PrintUsage("AUTOCOMPL on/auto string/off");
                return;
            }
            AutoNickCompl=newset;
        }
        else if (!my_stricmp("OFF",args)) {
            AutoNickCompl=0;
            new_free(&AutoReplyString);
        }
        else {
            PrintUsage("AUTOCOMPL on/auto string/off");
            return;
        }
    }
    if (AutoNickCompl)
        PrintSetting("Auto nick completion",AutoNickCompl==1?"ON":"AUTO",
                     ", right side string is",AutoReplyString);
    else PrintSetting("Auto nick completion","OFF",empty_string,empty_string);
}

#ifdef EXTRAS
/* Sets idle kick on/off */
void SetIdleKick(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int newset=0;
    char *tmpstr;

    if (args && *args) {
        tmpstr=new_next_arg(args,&args);
        if (!my_stricmp("ON",tmpstr) || !my_stricmp("AUTO",tmpstr)) {
            if (!my_stricmp("ON",tmpstr)) newset=1;
            else newset=2;
            tmpstr=new_next_arg(args,&args);
            if (tmpstr && *tmpstr) malloc_strcpy(&IdleKickChannels,tmpstr);
            else {
                PrintUsage("IDLEKICK on/auto channels/off");
                return;
            }
            IdleKick=newset;
        }
        else if (!my_stricmp("OFF",tmpstr)) {
            IdleKick=0;
            new_free(&IdleKickChannels);
        }
        else {
            PrintUsage("IDLEKICK on/auto channels/off");
            return;
        }
    }
    if (IdleKick)
        PrintSetting("Idle kick",IdleKick==1?"ON":"AUTO",
                     " for channels :",IdleKickChannels);
    else PrintSetting("Idle kick","OFF",empty_string,empty_string);
    /* 99 is IdleKick, look at SetChannels() */
    SetChannels(99);
}
#endif /* EXTRAS */

/* Sets ScrollZ settings (on channels/off) */
void ChannelCommand(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int  i;
    int  isorignick;
    char *tmpstr;
    char *tmpchan;
    char *quietstr;
    char *cmdchars;
    char tmpbuf[mybufsize / 8];

    upper(command);
    for (i = 0; command_list[i].command; i++)
        if (!strcmp(command_list[i].command, command)) break;
    if (!(command_list[i].command)) return;
    isorignick = !strcmp(command_list[i].command, "ORIGNICK");
    tmpstr = new_next_arg(args, &args);
    if (tmpstr) {
        if (!my_stricmp("ON", tmpstr)) {
            tmpchan = new_next_arg(args, &args);
            quietstr = new_next_arg(args, &args);
            if (tmpchan && *tmpchan) malloc_strcpy(command_list[i].strvar, tmpchan);
            else {
                snprintf(tmpbuf, sizeof(tmpbuf), "%s on %s%s/off",
                         command_list[i].command, isorignick ? "nick" : "channels",
                         isorignick ? " [quiet]" : "");
                PrintUsage(tmpbuf);
                return;
            }
            if (isorignick) {
                int x;

                if (quietstr && *quietstr && !my_stricmp(quietstr,"QUIET"))
                    OrigNickQuiet = 1;
                else OrigNickQuiet = 0;
                x = get_int_var(NO_ASK_NICKNAME_VAR);
                if (x == 0) {
                    if (!(cmdchars = get_string_var(CMDCHARS_VAR)))
                        cmdchars = DEFAULT_CMDCHARS;
                    say("You might want to %cSET NO_ASK_NICKNAME ON", *cmdchars);
                }
                OrigNickNumber = 0; /* start from scratch */
                OrigNickSent = 0;
                LastNick = time((time_t *) 0);
            }
            *(command_list[i].var) = 1;
        }
        else if (!my_stricmp("OFF", tmpstr)) {
            *(command_list[i].var) = 0;
            new_free(command_list[i].strvar);
        }
        else {
            snprintf(tmpbuf, sizeof(tmpbuf), "%s on %s%s/off", command_list[i].command,
                     isorignick ? "nick" : "channels", isorignick ? " [quiet]" : "");
            PrintUsage(tmpbuf);
            return;
        }
    }
    tmpstr = command_list[i].setting2;
    if (!tmpstr) tmpstr = " for channels :";
    if (*(command_list[i].var)) PrintSetting(command_list[i].setting, "ON", tmpstr,
                                             *(command_list[i].strvar));
    else PrintSetting(command_list[i].setting, "OFF", empty_string, empty_string);
    /* first update channel log filepath */
    if (!strcmp(command_list[i].command, "CHANLOG")) SetChannels(100);
    SetChannels(i);
}

/* Prints usage for command */
void PrintUsage(line)
char *line;
{
    say("Usage: /%s",line);
}

#ifdef WANTANSI
/* Converts mIRC colors to ANSI, by Ananda - fix by ddd */
void ConvertmIRC(buffer,newbuf)
char *buffer;
char *newbuf;
{
    struct {
        char *fg, *bg;
    } codes[16] = {
        { "",   ""        },      /* white                */
        { "",      ""        },      /* black (grey for us)  */
        { "",   ""        },      /* blue                 */
        { "",   ""        },      /* green                */
        { "",   ""        },      /* red                  */
        { "",   ""        },      /* brown                */
        { "",   ""        },      /* magenta              */
        { "",   ""        },      /* bright red           */
        { "",   ""        },      /* yellow               */
        { "",   ""        },      /* bright green         */
        { "",   ""        },      /* cyan                 */
        { "",   ""        },      /* bright cyan          */
        { "",   ""        },      /* bright blue          */
        { "",   ""        },      /* bright magenta       */
        { "",   ""        },      /* dark grey            */
        { "",   ""        }       /* grey                 */
    };
    register char *sptr=buffer;
    register char *dptr=newbuf;
    register short code;

    *dptr='\0';
    while (*sptr) {
        if (*sptr=='' && isdigit(*(sptr+1))) {
            sptr++;
	    code=(*sptr++)-'0';
	    if (isdigit(*sptr)) code=code*10+(*sptr++)-'0';
            if (code>15 || code<0) continue;
            if (DisplaymIRC==1) {
                strcpy(dptr,codes[code].fg);
                while (*dptr) dptr++;
            }
            if (*sptr==',') {
                sptr++;
	        code=(*sptr++)-'0';
	        if (isdigit(*sptr)) code=code*10+(*sptr++)-'0';
                if (code>0 && code<15 && DisplaymIRC==1) {
                    strcpy(dptr,codes[code].bg);
                    while (*dptr) dptr++;
                }
            }
        }
        else if (*sptr=='') {
            if (DisplaymIRC==1) {
                strcpy(dptr,Colors[COLOFF]);
                while (*dptr) dptr++;
            }
            sptr++;
        }
        else *dptr++=*sptr++;
    }
    *dptr='\0';
}
#endif

#ifdef CELE
/* Formats TRACE - by Zakath */
void HandleTrace(trnum,type,sclass,arg1,arg2,arg3,arg4)
int  trnum;
char *type;
char *sclass;
char *arg1;
char *arg2;
char *arg3;
char *arg4;
{
    if ((TraceOper || TraceAll) && trnum==204)
        put_it("%sOper: %c%s%c (%s)",numeric_banner(),bold,arg1,bold,sclass);
    else if ((TraceUser || TraceAll) && trnum==205)
        put_it("%sUser: %c%s%c (%s)",numeric_banner(),bold,arg1,bold,sclass);
    else if ((TraceServer || TraceAll) && trnum==206)
        put_it("%sServ: %c%s%c - by %s (%s)",numeric_banner(),bold,arg3,bold,arg4,sclass);
    else if (TraceAll) put_it("%s%c%s%c %s %s",numeric_banner(),bold,type,bold,sclass,arg1);
}

/* Sends TRACE command, allows for switches - by Zakath */
void ScrollZTrace(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *tswitch;
    char *server=(char *) 0;

    TraceAll=1; /* Unless changed w/ a switch, we want all info */
    TraceOper=0;
    TraceUser=0;
    TraceClass=0;
    TraceServer=0;
    if (args && *args) {
        tswitch=new_next_arg(args,&args);
        if (index(tswitch, '-')==NULL) server=tswitch;
        else {
            if (args && *args) server=next_arg(args,&args);
            else
                if (!my_stricmp(tswitch,"-u")) {
                    TraceAll=0;
                    TraceUser=1;
                    say("Tracing server for all %cusers%c...",bold,bold);
                }
            if (!my_stricmp(tswitch,"-o")) {
                TraceAll=0;
                TraceOper=1;
                say("Tracing server for all %copers%c...",bold,bold);
            }
            if (!my_stricmp(tswitch,"-s")) {
                TraceAll=0;
                TraceServer=1;
                say("Tracing server for all %cserver connections%c...",bold,bold);
            }
        }
    }
    if (server) send_to_server("TRACE %s",server);
    else send_to_server("TRACE");
}
#endif

#ifdef WANTANSI
/* Colorizes user@host */
void ColorUserHost(userhost,color,buffer,parentheses)
char *userhost;
char *color;
char *buffer;
int  parentheses;
{
    char *tmpstr;
    char tmpbuf[mybufsize/4+1];

    if (!userhost) {
        *buffer='\0';
        return;
    }
    strmcpy(tmpbuf,userhost,mybufsize/4);
    tmpstr=index(tmpbuf,'@');
    if (tmpstr) {
        *tmpstr='\0';
        tmpstr++;
    }
    else {
        *buffer='\0';
        return;
    }
    if (parentheses) sprintf(buffer,"%s(%s%s%s%s%s@%s%s%s%s%s)%s",
                             CmdsColors[COLMISC].color2,Colors[COLOFF],
                             color,tmpbuf,Colors[COLOFF],
                             CmdsColors[COLMISC].color1,Colors[COLOFF],
                             color,tmpstr,Colors[COLOFF],
                             CmdsColors[COLMISC].color2,Colors[COLOFF]);
    else sprintf(buffer,"%s%s%s%s@%s%s%s%s",
                 color,tmpbuf,Colors[COLOFF],
                 CmdsColors[COLMISC].color1,Colors[COLOFF],
                 color,tmpstr,Colors[COLOFF]);
}
#endif

/* Adds/removes channels to ScrollZ settings */
void AddChannel(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int  i;
    int  add=!strcmp(command,"ADD")?1:0;
    int  len;
    char *channel;
    char *szsetting;
    char tmpbuf[mybufsize/4];

    szsetting=new_next_arg(args,&args);
    channel=new_next_arg(args,&args);
    if (!szsetting || !channel) {
        snprintf(tmpbuf,sizeof(tmpbuf),"%sCHAN setting channel",command);
        PrintUsage(tmpbuf);
        return;
    }
    upper(szsetting);
    len=strlen(szsetting);
    for (i=0;command_list[i].command;i++)
        if (!strncmp(command_list[i].command,szsetting,len)) break;
    if (!(command_list[i].command) || !strcmp(szsetting,"ORIGNICK")) {
        say("Illegal command, try /HELP %sCHAN",command);
        return;
    }
    if (!(*(command_list[i].var))) {
        say("%s is set to OFF, aborting",command_list[i].command);
        return;
    }
    AddRemoveChannel(command_list[i].strvar,channel,add);
    if (!strcmp(command_list[i].command,"AJOIN")) AutoJoinOnInvToggle(NULL,NULL,NULL);
    else if (!strcmp(command_list[i].command,"NHPROT")) NHProtToggle(NULL,NULL,NULL);
    else PrintSetting(command_list[i].setting,"ON"," for channels :",
                      *(command_list[i].strvar));
    SetChannels(i);
}

/* Used to calculate hash value */
int HashFunc(nick)
char *nick;
{
    int  sum=0;
    unsigned char *tmp;

    for (tmp=nick;*tmp;tmp++)
        sum+=(*tmp>='a' && *tmp<='z'?*tmp-' ':*tmp);
    return(sum%HASHTABLESIZE);
}

/* Shows some ScrollZ related information */
void ScrollZInfo(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int  timediff = time(NULL) - start_time;
    char *szver;
    char *tmpstr1;
    char tmpbuf[mybufsize / 4 + 1];

    strmcpy(tmpbuf, ScrollZver, sizeof(tmpbuf));
    tmpstr1 = index(tmpbuf, ' ');
    szver = index(tmpstr1 + 1, ' ') + 1;
    tmpstr1 = index(szver,' ');
    *tmpstr1 = '\0';
    say("This is ScrollZ %s (client base ircII %s)", szver, irc_version);
    say("Client uptime: %dd %02dh %02dm", timediff / 86400, (timediff / 3600) % 24, (timediff / 60) % 60);
    if (CheckServer(from_server)) {
        timediff = time(NULL) - server_list[from_server].ConnectTime;
        say("Connected to the current server (%s): %dd %02dh %02dm",
                get_server_name(from_server), timediff / 86400, (timediff / 3600) % 24, (timediff / 60) % 60);
    }
    say("Support channel: #ScrollZ on Efnet");
    say("Mailing list: scrollz@ahnberg.pp.se");
    say("Home page: http://www.scrollz.com/");
    say("FTP: ftp://ftp.scrollz.com/");
}

/* Handles reply number 329 from server */
void ChannelCreateTime(from,ArgList)
char *from;
char **ArgList;
{
    char *channel;
    time_t createtime;

    if (ArgList[1] && is_channel(ArgList[0])) {
        channel=ArgList[0];
        createtime=atoi(ArgList[1]);
        save_message_from();
        message_from(channel,LOG_CRAP);
        if (createtime) {
#ifdef WANTANSI
            put_it("%sChannel %s%s%s created on %s%.24s%s",numeric_banner(),
                    CmdsColors[COLJOIN].color3,channel,Colors[COLOFF],
                    CmdsColors[COLJOIN].color4,ctime(&createtime),Colors[COLOFF]);
#else
            put_it("%sChannel %s created on %.24s",numeric_banner(),channel,
                    ctime(&createtime));
#endif
        }
        else {
#ifdef WANTANSI
            put_it("%sTime stamping is off for channel %s%s%s",numeric_banner(),
                    CmdsColors[COLJOIN].color3,channel,Colors[COLOFF]);
#else
            put_it("%sTime stamping is off for channel %s",numeric_banner(),channel);
#endif
        }
        restore_message_from();
    }
}

#ifdef CELE
/* displays your system's uptime */
void ExecUptime(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char tmpbuf[mybufsize/4];

    snprintf(tmpbuf,sizeof(tmpbuf),"uptime");
    execcmd(NULL,tmpbuf,NULL);
}

/* Does /CTCP nick finger */
void CTCPFing(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *target=NULL;

    if (args && *args && *args!='*') target=new_next_arg(args,&args);
    if (!target && !(target=get_channel_by_refnum(0))) {
        NoWindowChannel();
        return;
    }
    send_to_server("PRIVMSG %s :%cFINGER%c",target,CTCP_DELIM_CHAR,CTCP_DELIM_CHAR);
}
#endif

/* Checks whether we tried to ping non existing server */
void NoSuchServer4SPing(from,ArgList)
char *from;
char **ArgList;
{
    char *server=ArgList[0];
    struct spingstr *spingtmp;

    if (server && (spingtmp=(struct spingstr *) list_lookup((List **) &spinglist,server,
                                                            !USE_WILDCARDS,REMOVE_FROM_LIST))) {
        new_free(&(spingtmp->servername));
        new_free(&spingtmp);
#ifdef WANTANSI
        say("No such server to ping: %s%s%s",
            CmdsColors[COLCSCAN].color1,server,Colors[COLOFF]);
#else
        say("No such server to ping: %s",server);
#endif
    }
    else if (server && ArgList[1]) put_it("%s%s %s",numeric_banner(),server,ArgList[1]);
}

/* Returns server from netsplit info */
char *GetNetsplitServer(channel,nick)
char *channel;
char *nick;
{
    char *server;
    struct wholeftch  *wholch;
    struct wholeftstr *wholeft;

    for (wholeft=wholist;wholeft;wholeft=wholeft->next) {
        for (wholch=wholeft->channels;wholch;wholch=wholch->next)
            if (!my_stricmp(channel,wholch->channel)) break;
        if (wholch && (server=index(wholeft->splitserver,' '))) {
            server++;
            return(server);
        }
    }
    return(empty_string);
}

/* Handles delayed opping */
void HandleDelayOp(stuff)
char *stuff;
{
    char *flag;
    char mode=' ';
    char *nick;
    char *channel;
    NickList *tmpnick;
    ChannelList *chan;

    channel=new_next_arg(stuff,&stuff);
    flag=new_next_arg(stuff,&stuff);
    nick=new_next_arg(stuff,&stuff);
    chan=lookup_channel(channel,from_server,0);
    if ((chan=lookup_channel(channel,from_server,0)) && chan->FriendList && HAS_OPS(chan->status) &&
        (tmpnick=CheckJoiners(nick,channel,from_server,chan)) && tmpnick->frlist) {
	switch (*flag) {
	    case 'v':
		if ((tmpnick->frlist->privs&FLVOICE) && !tmpnick->chanop && !tmpnick->halfop)
		    mode='v';
		break;
	    case 'h':
		if ((chan->status&CHAN_CHOP) && (tmpnick->frlist->privs&FLHOP) && !tmpnick->chanop)
		    mode='h';
		break;
	    case 'o':
		if ((chan->status&CHAN_CHOP) && (tmpnick->frlist->privs&FLOP) && !tmpnick->chanop)
		    mode='o';
		break;
	}
	if (mode!=' ') send_to_server("MODE %s +%c %s",channel,mode,nick);
    }
}

/* Handles delayed notify */
void HandleDelayNotify(stuff)
char *stuff;
{
    int old_server=from_server;
    void (*func)()=(void(*)()) HandleUserhost;

    if (inSZNotify) inSZNotify++;
    else inSZNotify=2;
    from_server=primary_server;
    add_userhost_to_whois(stuff,func);
    from_server=old_server;
}

#ifdef EXTRAS
/* Show idle time for users */
void ShowIdle(command, args, subargs)
char *command;
char *args;
char *subargs;
{
#ifdef WANTANSI
    int  len = 0;
#endif
    int  days;
    int  hours;
    int  mins;
    int  secs;
    int  origidle;
    int  showops = 0;
    int  shownonops = 0;
    int  chanusers = 0;
    int  totalidle = 0;
    char *filter = (char *) 0;
    char *channel = (char *) 0;
    char tmpbuf1[mybufsize / 4];
    char tmpbuf2[mybufsize / 4];
    time_t timenow = time((time_t *) 0);
    NickList *joiner;
    ChannelList *chan;

    channel = new_next_arg(args, &args);
    if (channel && *channel == '-') {
        upper(channel);
        channel++;
        if (*channel && *channel == 'O') showops = 1;
        else if (*channel && *channel == 'N') shownonops = 1;
        channel = new_next_arg(args, &args);
    }
    if (channel) {
        if (is_channel(channel)) strmcpy(tmpbuf1, channel, sizeof(tmpbuf1));
        else snprintf(tmpbuf1, sizeof(tmpbuf1), "#%s", channel);
        channel = tmpbuf1;
    }
    if (!channel && (channel = get_channel_by_refnum(0)) == NULL) {
        NoWindowChannel();
        return;
    }
    if (!(chan = lookup_channel(channel, curr_scr_win->server, 0))) return;
    if (!(chan->IdleKick)) say("Idle kick is off for channel %s", chan->channel);
    if (!(filter = new_next_arg(args, &args))) filter = "*";
    say("%-42s  Idle time","User");
    for (joiner = chan->nicks; joiner; joiner = joiner->next) {
        snprintf(tmpbuf2, sizeof(tmpbuf2), "%s!%s", joiner->nick,
                joiner->userhost ? joiner->userhost : empty_string);
        if (!wild_match(filter, tmpbuf2)) continue;
        if (showops && !(joiner->chanop)) continue;
        if (shownonops && joiner->chanop) continue;
        origidle = timenow-joiner->lastmsg;
        days = origidle / 86400;
        hours = (origidle - (days * 86400)) / 3600;
        mins = (origidle - (days * 86400) - (hours * 3600)) / 60;
        secs = origidle - (days * 86400) - (hours * 3600) - (mins * 60);
#ifdef WANTANSI
        len = strlen(joiner->nick) + 1;
        if (joiner->userhost) {
            ColorUserHost(joiner->userhost, CmdsColors[COLWHO].color2, tmpbuf2, 1);
            len += strlen(joiner->userhost);
        }
        else *tmpbuf2 = '\0';
        for ( ;len < 40; len++) strmcat(tmpbuf2, " ", sizeof(tmpbuf2));
        say("%s%s%s %s  %dd %dh %dm %ds",
            CmdsColors[COLWHO].color1, joiner->nick, Colors[COLOFF],
            tmpbuf2, days, hours, mins, secs);
#else
        say("%-42s  %dd %dh %dm %ds", tmpbuf2, days, hours, mins, secs);
#endif
        chanusers++;
        totalidle += origidle;
    }
    say("Average idle time in channel %s is %.2f seconds", chan->channel,
        (float) totalidle / chanusers);
}
#endif

/* Try to switch nick to orignick */
void SwitchNick() {
    int i=0;
    char savechar='\0';
    char *curnick=(char *) 0;
    char *realnick=OrigNick;
    time_t timenow=time((time_t *) 0);

    if (timenow>=LastNick+OrigNickDelay) {
        for (;i<OrigNickNumber;i++) {
            if (realnick) realnick=index(realnick,',');
            if (realnick) realnick++;
        }
        if (!realnick) { /* we have exhausted all possible nicks */
            realnick=OrigNick;
            OrigNickNumber=0;
        }
        else if (realnick && *realnick) {
            if ((curnick=index(realnick,','))) {
                OrigNickNumber++;
                savechar=',';
                *curnick='\0';
            }
            else OrigNickNumber=0;
        }
        if (realnick && *realnick) {
            if (my_stricmp(get_server_nickname(from_server),realnick))
                e_nick(NULL,realnick,NULL);
            LastNick=timenow+1;
            OrigNickSent++;
            if (savechar && curnick) *curnick=savechar;
        }
    }
}

#ifdef OPER
/* Kills multiple nicks at the same time by acidflash */
void MassKill(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int	 meep=1;
    char *comment;
    char *tmpnick;

    if (args && *args) {
        if (!(comment=index(args,':'))) comment=DefaultKill;
        else *comment++='\0';
        if (args && *args) {
            while ((tmpnick=new_next_arg(args,&args))) {
                send_to_server("KILL %s :%s (%d)",tmpnick,comment,meep);
                meep++;
            }
        }
    }
    else PrintUsage("MKILL nick1 nick2 nick3 ... [:reason]");
}
#endif /* OPER */

/* Check if given server is valid, ie. it still exists and it's connected */
int CheckServer(server)
int server;
{
    if (server>=0 && server<number_of_servers && server_list[server].connected)
        return(1);
    return(0);
}

/* Clean up ScrollZ allocated variables */
void CleanUpScrollZVars() {
    new_free(&DefaultServer);
    new_free(&ScrollZstr);
    new_free(&ScrollZver1);
    new_free(&DefaultSignOff);
    new_free(&DefaultSetAway);
    new_free(&DefaultSetBack);
    new_free(&DefaultUserinfo);
    new_free(&DefaultFinger);
    new_free(&AutoJoinChannels);
    new_free(&CdccUlDir);
    new_free(&CdccDlDir);
    new_free(&WhoKilled);
    new_free(&CdccChannels);
    new_free(&AutoRejoinChannels);
    new_free(&MDopWatchChannels);
    new_free(&ShowFakesChannels);
    new_free(&KickOnFloodChannels);
    new_free(&KickWatchChannels);
    new_free(&NHProtChannels);
    new_free(&NickWatchChannels);
    new_free(&ShowAwayChannels);
    new_free(&KickOpsChannels);
    new_free(&KickOnBanChannels);
    new_free(&BitchChannels);
    new_free(&FriendListChannels);
#ifdef EXTRAS
    new_free(&IdleKickChannels);
    new_free(&SignoffChannels);
#endif
    new_free(&CompressModesChannels);
    new_free(&BKChannels);
    new_free(&EncryptPassword);
#ifdef OPER
    new_free(&StatskFilter);
    new_free(&StatsiFilter);
    new_free(&StatslFilter);
    new_free(&StatscFilter);
    new_free(&StatsdFilter);
#endif
    new_free(&AutoReplyBuffer);
    new_free(&OrigNick);
    new_free(&HelpPathVar);
    new_free(&CelerityNtfy);
    new_free(&URLBuffer);
    new_free(&LastChat);
    new_free(&CurrentDCC);
    new_free(&DefaultK);
    new_free(&DefaultBK);
    new_free(&DefaultBKI);
    new_free(&DefaultBKT);
    new_free(&DefaultFK);
    new_free(&DefaultLK);
    new_free(&DefaultABK);
    new_free(&DefaultSK);
#ifdef OPER
    new_free(&DefaultKill);
#endif
    new_free(&PermUserMode);
    new_free(&AutoReplyString);
#ifdef ACID
    new_free(&ForceJoinChannels);
#endif
    new_free(&ChanLogChannels);
}

/* Clean up all stuff from memory on exit */
void CleanUp() {
    int i;
    char tmpbuf[mybufsize/32];
    struct list *tmplist;
    struct nicks *tmpnick,*tmpnickfree;
    struct mapstr *tmpmap;
    struct urlstr *tmpurl;
    struct encrstr *tmpencr;
    struct splitstr *tmpsplit;
    struct spingstr *tmpsping;
    struct wholeftch *tmpchan,*tmpchanfree;
    struct wholeftstr *tmpwho;
    DCC_list *tmpdcc,*tmpdccfree;
    ChannelList *tmpch,*tmpchfree;

    for (tmpdcc=ClientList;tmpdcc;) {
        tmpdccfree=tmpdcc;
        tmpdcc=tmpdcc->next;
        dcc_erase(tmpdccfree);
    }
    CleanUpCdcc();
    CleanUpFlood();
    CleanUpIgnore();
    CleanUpTimer();
    CleanUpLists();
    CleanUpWhowas();
    CleanUpWindows();
    CleanUpVars();
    CleanUpScrollZVars();
    inSZNotify=1;
    strcpy(tmpbuf,"-");
    notify(NULL,tmpbuf,NULL);
    strcpy(tmpbuf,"ALL");
    Dump(NULL,tmpbuf,NULL);
    for (i=0;i<number_of_servers;i++) {
	if (server_list[i].name) new_free(&server_list[i].name);
	if (server_list[i].itsname) new_free(&server_list[i].itsname);
	if (server_list[i].password) new_free(&server_list[i].password);
	if (server_list[i].away) new_free(&server_list[i].away);
	if (server_list[i].version_string) new_free(&server_list[i].version_string);
	if (server_list[i].nickname) new_free(&server_list[i].nickname);
	if (server_list[i].whois_stuff.nick) new_free(&server_list[i].whois_stuff.nick);
	if (server_list[i].whois_stuff.user) new_free(&server_list[i].whois_stuff.user);
	if (server_list[i].whois_stuff.host) new_free(&server_list[i].whois_stuff.host);
	if (server_list[i].whois_stuff.channel) new_free(&server_list[i].whois_stuff.channel);
	if (server_list[i].whois_stuff.channels) new_free(&server_list[i].whois_stuff.channels);
	if (server_list[i].whois_stuff.name) new_free(&server_list[i].whois_stuff.name);
	if (server_list[i].whois_stuff.server) new_free(&server_list[i].whois_stuff.server);
	if (server_list[i].whois_stuff.server_stuff) new_free(&server_list[i].whois_stuff.server_stuff);
	if (server_list[i].ctcp_send_size) new_free(&server_list[i].ctcp_send_size);
	if (server_list[i].LastJoin) new_free(&(server_list[i].LastJoin));
        for (tmpnick=server_list[i].arlist;tmpnick;) {
            tmpnickfree=tmpnick;
            tmpnick=tmpnick->next;
            new_free(&(tmpnickfree->nick));
            new_free(&tmpnickfree);
        }
        for (tmpnick=server_list[i].nicklist;tmpnick;) {
            tmpnickfree=tmpnick;
            tmpnick=tmpnick->next;
            new_free(&(tmpnickfree->nick));
            new_free(&tmpnickfree);
        }
        for (tmpch=server_list[i].chan_list;tmpch;) {
            tmpchfree=tmpch;
            tmpch=tmpch->next;
            clear_channel(tmpchfree);
            new_free(&(tmpchfree->key));
            new_free(&(tmpchfree->s_mode));
            new_free(&(tmpchfree->modelock));
            new_free(&(tmpchfree->topiclock));
            new_free(&(tmpchfree->topicstr));
            new_free(&(tmpchfree->topicwho));
            new_free(&(tmpchfree->channel));
            new_free(&tmpchfree);
        }
	from_server=i;
	clean_whois_queue();
    }
    free(server_list);
    while (splitlist) {
        tmpsplit=splitlist;
        splitlist=splitlist->next;
        new_free(&(tmpsplit->servers));
        new_free(&tmpsplit);
    }
    while (spinglist) {
        tmpsping=spinglist;
        spinglist=spinglist->next;
        new_free(&(tmpsping->servername));
        new_free(&tmpsping);
    }
    while (encrlist) {
        tmpencr=encrlist;
        encrlist=encrlist->next;
        new_free(&(tmpencr->user));
        new_free(&(tmpencr->key));
        new_free(&tmpencr);
    }
    while (maplist) {
        tmpmap=maplist;
        maplist=maplist->next;
        new_free(&(tmpmap->server));
        new_free(&(tmpmap->uplink));
        new_free(&tmpmap);
    }
    while (urllist) {
        tmpurl=urllist;
        urllist=urllist->next;
        new_free(&(tmpurl->urls));
        new_free(&(tmpurl->source));
        new_free(&tmpurl);
    }
    while (wholist) {
        tmpwho=wholist;
        wholist=wholist->next;
        tmpchan=tmpwho->channels;
        while (tmpchan) {
            while (tmpchan->nicklist) {
                tmplist=tmpchan->nicklist;
                tmpchan->nicklist=tmplist->next;
                new_free(&(tmplist->nick));
                new_free(&(tmplist->userhost));
                new_free(&tmplist);
            }
            tmpchanfree=tmpchan;
            tmpchan=tmpchan->next;
            new_free(&(tmpchanfree->channel));
            new_free(&(tmpchanfree));
        }
        new_free(&(tmpwho->splitserver));
        new_free(&tmpwho);
    }
#ifdef WANTANSI
    for (i=0;i<NUMCMDCOLORS;i++) {
        new_free(&(CmdsColors[i].color1));
        new_free(&(CmdsColors[i].color2));
        new_free(&(CmdsColors[i].color3));
        new_free(&(CmdsColors[i].color4));
        new_free(&(CmdsColors[i].color5));
        new_free(&(CmdsColors[i].color6));
    }
#endif
#ifdef HAVE_SSL
    gnutls_global_deinit();
#endif
}

/* Support for encrypted sessions */
void EncryptMsg(command, args, subargs)
char *command;
char *args;
char *subargs;
{
    int show_keys = 0;
    int clear_all = 0;
    int pwlen;
    char *user;
    char *key;
    char *pass = NULL;
    char *mastpass;
    char *cmdchars;
    char tmpbuf[mybufsize / 4 + 1];

    if (!EncryptPassword) {
        if (!(cmdchars = get_string_var(CMDCHARS_VAR))) cmdchars = DEFAULT_CMDCHARS;
        say("Set master password first with %cPASSWD", *cmdchars);
        return;
    }
    user = new_next_arg(args,&args);
    if (user && !my_stricmp(user, "-PASS")) {
        pass = new_next_arg(args, &args);
        if (!pass) {
            say("No master password given");
            return;
        }
        user = new_next_arg(args, &args);
    }
    if (user && !my_stricmp(user, "-SHOW")) {
        show_keys = 1;
        user = new_next_arg(args, &args);
    }
    if (user && !my_stricmp(user, "-CLEAR")) {
        clear_all = 1;
        user = NULL;
    }
    if (!user) user = new_next_arg(args, &args);
    key = new_next_arg(args, &args);
    if (pass) {
        pwlen = 2 * strlen(pass) + 16;
        mastpass = (char *) new_malloc(pwlen + 1);
        EncryptString(mastpass, pass, pass, pwlen, 0);
        if (strcmp(EncryptPassword, mastpass)) {
            say("Incorrect master password!");
            new_free(&mastpass);
            return;
        }
        new_free(&mastpass);
    }
    if (user) {
        *tmpbuf = '\0';
        if (show_keys) strmcat(tmpbuf, ":", mybufsize / 4);
        strmcat(tmpbuf, user, mybufsize / 4);
        if (key) EncryptAdd(tmpbuf, key);
        else if (*user == '-') {
            if (pass) EncryptDelUser(user + 1);
            else add_wait_prompt("Master Password:", EncryptMasterDelUser, user + 1, WAIT_PROMPT_LINE);
        }
        else {
            if (pass) add_wait_prompt("Password:", EncryptAdd, tmpbuf, WAIT_PROMPT_LINE);
            else add_wait_prompt("Master Password:", EncryptMasterPass, tmpbuf, WAIT_PROMPT_LINE);
        }
    }
    else {
        if (!encrlist) {
            say("No users in encryption list");
            return;
        }
        if (show_keys) *tmpbuf = '1';
        else *tmpbuf = '0';
        if (clear_all) *(tmpbuf + 1) = '1';
        else *(tmpbuf + 1) = '0';
        *(tmpbuf + 2) = '\0';
        if (pass || (!show_keys && !clear_all)) EncryptList(tmpbuf);
        else add_wait_prompt("Master Password:", EncryptMasterList, tmpbuf, WAIT_PROMPT_LINE);
    }
}

/* Handle master password */
void EncryptMasterPass(char *user, char *pass)
{
    int pwlen;
    char *mastpass;

    pwlen = 2 * strlen(pass) + 16;
    mastpass = (char *) new_malloc(pwlen + 1);
    EncryptString(mastpass, pass, pass, pwlen, 0);
    if (strcmp(EncryptPassword, mastpass)) {
        say("Incorrect master password!");
        new_free(&mastpass);
        return;
    }
    new_free(&mastpass);
    add_wait_prompt("Password:", EncryptAdd, user, WAIT_PROMPT_LINE);
}

/* Handle encryption list */
void EncryptAdd(user, key)
char *user;
char *key;
{
    int show_keys = 0;
    struct encrstr *tmp;

    if (*user == ':') {
        show_keys++;
        user++;
    }
    tmp = (struct encrstr *) list_lookup((List **) &encrlist, user, !USE_WILDCARDS, !REMOVE_FROM_LIST);
    if (tmp) malloc_strcpy(&(tmp->key), key);
    else {
        tmp = (struct encrstr *) new_malloc(sizeof(struct encrstr));
        tmp->next = (struct encrstr *) 0;
        tmp->user = (char *) 0;
        tmp->key = (char *) 0;
        malloc_strcpy(&(tmp->user), user);
        malloc_strcpy(&(tmp->key), key);
        add_to_list((List **) &encrlist, (List *) tmp);
    }
    if (show_keys) say("Communication with %s will be encrypted using key %s", tmp->user, tmp->key);
    else say("Communication with %s will be encrypted", tmp->user);
}

/* Handle master password */
void EncryptMasterDelUser(char *user, char *pass)
{
    int pwlen;
    char *mastpass;

    pwlen = 2 * strlen(pass) + 16;
    mastpass = (char *) new_malloc(pwlen + 1);
    EncryptString(mastpass, pass, pass, pwlen, 0);
    if (strcmp(EncryptPassword, mastpass)) {
        say("Incorrect master password!");
        new_free(&mastpass);
        return;
    }
    new_free(&mastpass);
    EncryptDelUser(user);
}

/* Remove user from encryption list */
void EncryptDelUser(char *user)
{
    struct encrstr *tmp;

    tmp = (struct encrstr *) list_lookup((List **) &encrlist, user, !USE_WILDCARDS, REMOVE_FROM_LIST);
    if (!tmp) {
        say("User %s not found in encryption list", user);
        return;
    }
    new_free(&(tmp->user));
    new_free(&(tmp->key));
    new_free(&tmp);
    say("Communication with %s will no longer be encrypted", user);
}

/* Handle master password */
void EncryptMasterList(char *flags, char *pass)
{
    int pwlen;
    char *mastpass;

    pwlen = 2 * strlen(pass) + 16;
    mastpass = (char *) new_malloc(pwlen + 1);
    EncryptString(mastpass, pass, pass, pwlen, 0);
    if (strcmp(EncryptPassword, mastpass)) {
        say("Incorrect master password!");
        new_free(&mastpass);
        return;
    }
    new_free(&mastpass);
    EncryptList(flags);
}

/* Display encryption list */
void EncryptList(char *flags)
{
    int show_keys = 0;
    int clear_all = 0;
    struct encrstr *tmp, *tmpnext;

    if (*flags == '1') show_keys = 1;
    if (*(flags + 1) == '1') clear_all = 1;
    for (tmp = encrlist; tmp;) {
        tmpnext = tmp->next;
        if (clear_all) {
            if (tmp == encrlist) say("Removing all users from encryption list");
            say("Communication with %s will no longer be encrypted", tmp->user);
            new_free(&(tmp->user));
            new_free(&(tmp->key));
            new_free(&tmp);
        }
        else {
            if (tmp == encrlist) say("Listing all users on encryption list");
            if (show_keys) say("User %s with key %s", tmp->user, tmp->key);
            else say("User %s", tmp->user);
        }
        tmp = tmpnext;
    }
    if (clear_all) encrlist = (struct encrstr *) 0;
}

/* Encrypt message */
int EncryptMessage(message,user)
char *message;
char *user;
{
    struct encrstr *tmp;

    if ((tmp=(struct encrstr *) list_lookup((List **) &encrlist,user,!USE_WILDCARDS,
                                            !REMOVE_FROM_LIST))) {
        /* cut long messages before irc server does it so we don't have
           problems at the other end during decryption */
        if (strlen(message)>SZMAXCRYPTSIZE) message[SZMAXCRYPTSIZE]='\0';
        return(EncryptString(message,message,tmp->key,BIG_BUFFER_SIZE-16,1));
    }
    return(0);
}

/* Decrypt message */
int DecryptMessage(message,user)
char *message;
char *user;
{
    struct encrstr *tmp;

    if ((tmp=(struct encrstr *) list_lookup((List **) &encrlist,user,!USE_WILDCARDS,
                                            !REMOVE_FROM_LIST))) {
        return(DecryptString(message,message,tmp->key,BIG_BUFFER_SIZE-16,1));
    }
    return(0);
}

#ifdef OPER
/* Filtered trace */
void FilterTrace(command, args, subargs)
char *command;
char *args;
char *subargs;
{
    int etrace = 0;
    int countonly = 0;
    int haveserver = 0;
    char *filter;
    char *server;
    char cmdbuf[mybufsize];

    if (inSZTrace) {
        say("Already doing filtered trace");
        return;
    }
    filter = new_next_arg(args, &args);
    if (!my_stricmp(filter,"-E")) {
        etrace = 3;
        filter = new_next_arg(args, &args);
    }
    else if (!my_stricmp(filter,"-G")) {
        etrace = 4;
        filter = new_next_arg(args, &args);
    }
    if (!my_stricmp(filter,"-COUNT")) {
        countonly = 1;
        filter = new_next_arg(args, &args);
    }
    if (filter) {
        server = new_next_arg(args, &args);
        if (server && *server) haveserver = 1;
        tottcount = 0;
        mattcount = 0;
        new_free(&ftpattern);
        if (countonly) malloc_strcpy(&ftpattern, ":");
        malloc_strcat(&ftpattern, filter);
        if (etrace) {
            snprintf(cmdbuf, sizeof(cmdbuf), "ETRACE %s", haveserver ? server : "");
            inSZTrace = etrace;
        }
        else {
            snprintf(cmdbuf, sizeof(cmdbuf), "TRACE %s", haveserver ? server : "");
            inSZTrace = 2;
        }
        send_to_server("%s", cmdbuf);
        say("Finding all users matching %s on %sserver%s%s", filter,
            haveserver ? "" : "local ", haveserver ? " " : "", haveserver ? server : "");
    }
    else PrintUsage("FTRACE [-E|-G] [-COUNT] filter [server]");
}

/* Actual filtered trace */
void DoFilterTrace(stuff)
char *stuff;
{
    char *nick;
    char *host;
    char *prevhost = stuff;
    char *tmpstr;
    char *pattern = ftpattern;
    char tmpbuf[mybufsize / 4];

    nick = stuff;
    while ((host = index(prevhost, '['))) prevhost = host + 1;
    if (prevhost && !host) {
        tottcount++;
        host = prevhost - 1;
        *host++ = '\0';
        if ((tmpstr = index(host, ']'))) *tmpstr = '\0';
        else host = (char *) 0;
        snprintf(tmpbuf, sizeof(tmpbuf), "%s!%s", nick, host);
        if (*pattern == ':') pattern++;
        if (wild_match(pattern, tmpbuf)) {
            mattcount++;
            if (*ftpattern == ':') return;
#ifdef WANTANSI
            ColorUserHost(host, CmdsColors[COLWHO].color2, tmpbuf, 0);
            say("%s%-9s%s %s", CmdsColors[COLWHO].color1, nick, Colors[COLOFF], tmpbuf);
#else
            say("%-9s %s", nick, tmpbuf);
#endif
        }
    }
}
#endif /* OPER */

/* Format time in compressed format */
char *FormatTime(timediff)
int timediff;
{
    static char timebuf[mybufsize/32];

    snprintf(timebuf,sizeof(timebuf),"%dd%dh%dm%ds",timediff/86400,(timediff/3600)%24,(timediff/60)%60,
            timediff%60);
    return(timebuf);
}

/* Store key for later join */
void AddJoinKey(clienttype,notice)
int clienttype;
char *notice;
{
    int i;
    static int joinkeyinit=0;
    char *key=(char *) 0;
    char *tmpstr=(char *) 0;
    char *channel=(char *) 0;
    char tmpbuf[mybufsize/4+1];

    /* initalize */
    if (!joinkeyinit) {
        for (i=0;i<NUM_JOIN_KEYS;i++) {
            joinkeys[i].key=(char *) 0;
            joinkeys[i].channel=(char *) 0;
        }
        joinkeyinit=1;
    }
    strmcpy(tmpbuf,notice,mybufsize/4);
    notice=tmpbuf;
    switch (clienttype) {
        case 1: /* ScrollZ */
            channel=notice+30;
            tmpstr=channel;
            while (*tmpstr && !isspace(*tmpstr)) tmpstr++;
            if (*tmpstr) {
                *tmpstr++='\0';
                if (!strncmp(tmpstr,"(key is ",8)) {
                    tmpstr+=8;
                    key=tmpstr;
                    while (*tmpstr && !isspace(*tmpstr)) tmpstr++;
                    tmpstr--;
                    if (*tmpstr==')') {
                        *tmpstr++='\0';
                        if (strcmp(tmpstr," -ScrollZ-")) channel=(char *) 0;
                    }
                    else channel=(char *) 0;
                }
            }
            break;
        case 2: /* Pupette */
            key=notice+16;
            tmpstr=key;
            while (*tmpstr && !isspace(*tmpstr)) tmpstr++;
            if (*tmpstr) {
                *tmpstr++='\0';
                if (!strncmp(tmpstr,"to join ",8)) {
                    tmpstr+=8;
                    channel=tmpstr;
                }
            }
            break;
        case 3: /* C-ToolZ */
            channel=notice+17;
            tmpstr=channel;
            while (*tmpstr && *tmpstr!='') tmpstr++;
            if (*tmpstr) {
                *tmpstr++='\0';
                if (!strncmp(tmpstr," is ",5)) {
                    tmpstr+=5;
                    key=tmpstr;
                    while (*tmpstr && *tmpstr!='.') tmpstr++;
                    if (*tmpstr=='.') tmpstr--;
                    if (*tmpstr=='') {
                        *tmpstr++='\0';
                        if (strcmp(tmpstr,". <C-ToolZ>")) channel=(char *) 0;
                    }
                    else channel=(char *) 0;
                }
            }
            break;
        case 4: /* BitchX */
            channel=notice+22;
            tmpstr=channel;
            while (*tmpstr && !isspace(*tmpstr)) tmpstr++;
            if (*tmpstr) tmpstr--;
            if (*tmpstr=='.') {
                *tmpstr++='\0';
                if (!strncmp(tmpstr," Key is [",9)) {
                    tmpstr+=9;
                    key=tmpstr;
                    while (*tmpstr) tmpstr++;
                    tmpstr--;
                    if (*tmpstr=='') {
                        *tmpstr--='\0';
                        if (*tmpstr==']') *tmpstr='\0';
                        else channel=(char *) 0;
                    }
                }
            }
            break;
        case 5: /* ScrollZ with -DVILAS */
            channel=notice+12;
            tmpstr=channel;
            while (*tmpstr && !isspace(*tmpstr)) tmpstr++;
            if (*tmpstr) {
                *tmpstr++='\0';
                if (!strncmp(tmpstr,"key is ",7)) {
                    tmpstr+=7;
                    key=tmpstr;
                }
                else channel=(char *) 0;
            }
            break;
        default:
            return;
    }
    if (!key || !(*key) || !channel || !(*channel)) return;
    for (i=0;i<joinkeycount;i++)
        if (!my_stricmp(joinkeys[i].channel,channel)) break;
    if (i<joinkeycount) {
        new_free(&(joinkeys[i].key));
        new_free(&(joinkeys[i].channel));
        joinkeycount=i;
    }
    malloc_strcpy(&joinkeys[joinkeycount].key,key);
    malloc_strcpy(&joinkeys[joinkeycount].channel,channel);
    joinkeycount++;
    if (joinkeycount>=NUM_JOIN_KEYS) joinkeycount=0;
    say("Added channel %s and key %s to join list",channel,key);
}

/* Check key for join */
char *CheckJoinKey(channel)
char *channel;
{
    int i;
    char *key=empty_string;

    if (channel) {
        for (i=0;i<joinkeycount;i++)
            if (joinkeys[i].channel && !my_stricmp(joinkeys[i].channel,channel))
                break;
        if (i<joinkeycount && joinkeys[i].key)
            key=joinkeys[i].key;
    }
    return(key);
}

/* Try to join a channel */
#ifdef ACID
void TryChannelJoin() {
    int  found;
    int  lasttimer=0;
    char *tmpstr;
    char *tmpstr1;
    char tmpbuf[mybufsize/2+2];
    void (*func)()=(void(*)()) TryChannelJoin;
    TimerList *tmptimer;
    WhowasList *whowas;
    ChannelList *tmpchan;

    if (curr_scr_win->server!=-1 && ForceJoin && ForceJoinChannels) {
        strmcpy(tmpbuf,ForceJoinChannels,mybufsize/2);
        tmpstr=tmpbuf;
        tmpstr1=tmpbuf;
        while (1) {
            if (*tmpstr==',' || !(*tmpstr)) {
                if (!(*tmpstr)) *(tmpstr+1)='\0';
                *tmpstr='\0';
                for (tmpchan=server_list[curr_scr_win->server].chan_list;tmpchan;
                        tmpchan=tmpchan->next)
                    if (!my_stricmp(tmpstr1,tmpchan->channel)) break;
                /* 
                 * ask up to 5 users with +z flag set to open the channel for us
                 */
                if (!tmpchan) {
                    found=0;
                    for (whowas=whowas_userlist_list;whowas;whowas=whowas->next) {
                        if (whowas->nicklist && whowas->nicklist->frlist &&
                                ((whowas->nicklist->frlist->privs)&FLWHOWAS) &&
                                !my_stricmp(whowas->channel,tmpstr1)) {
                            found++;
                            if (found>5) break;
                            send_to_server("PRIVMSG %s :OPEN %sINVITE %s",
                                    whowas->nicklist->nick,tmpstr1,tmpstr1);
                        }
                    }
                }
                tmpstr1=tmpstr+1;
            }
            if (!(*tmpstr) && !(*(tmpstr+1))) break;
            tmpstr++;
        }
        /*
         * if there is one or less entries for TryChannelJoin() in
         * the timer list, add one
         */
        found=0;
        for (tmptimer=PendingTimers;tmptimer;tmptimer=tmptimer->next) {
            if (tmptimer->command && tmptimer->func==func &&
                !strcmp(tmptimer->command,"rejoin")) {
                found++;
                lasttimer=tmptimer->time;
            }
        }
        if (found<2) {
            lasttimer-=time((time_t *) 0);
            lasttimer+=30;
            if (lasttimer<=0) lasttimer=30;
            snprintf(tmpbuf,sizeof(tmpbuf),"-INV %d rejoin",lasttimer);
            timercmd("FTIMER",tmpbuf,(char *) func);
        }
    }
}
#endif /* ACID */

/* Change user's password */
void ChangePassword(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int count=0;
    int isfilt=0;
    int countall=0;
    char *filter;
    char *passwd;
    char tmpbuf1[mybufsize/8+1];
    struct friends *tmpfriend;

    filter=new_next_arg(args,&args);
    passwd=new_next_arg(args,&args);
    if (!(filter && passwd)) {
        PrintUsage("CHPASS filter password");
        return;
    }
    if (*filter=='#') {
        isfilt=1;
        filter++;
    }
    for (tmpfriend=frlist;tmpfriend;tmpfriend=tmpfriend->next) {
        countall++;
        if ((isfilt && matchmcommand(filter,countall)) ||
            wild_match(filter,tmpfriend->userhost) ||
            wild_match(tmpfriend->userhost,filter)) {
            count++;
            /* if password is set to minus remove it */
            if (!strcmp(passwd,"-")) new_free(&(tmpfriend->passwd));
            else {
                EncryptString(tmpbuf1,passwd,passwd,mybufsize/16,0);
                malloc_strcpy(&(tmpfriend->passwd),tmpbuf1);
            }
        }
    }
    say("Changed %d out of %d entries",count,countall);
}

/* Does /STATS i with filter */
#ifdef OPER
void StatsIFilter(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *tmpstr;

    tmpstr=new_next_arg(args,&args);
    if (tmpstr) {
        malloc_strcpy(&StatsiFilter,tmpstr);
        StatsiNumber=0;
        send_to_server("STATS I %s",args);
    }
    else PrintUsage("FILINE filter [server]");
}

/* Parses STATS i reply from server */
void HandleStatsI(statschar, ipiline, uhiline)
char *statschar;
char *ipiline;
char *uhiline;
{
    char *tmpstr;
    char tmpbuf1[mybufsize / 2 + 1];

    if (!StatsiNumber) say("I-Line");
    if (!uhiline) uhiline = empty_string;
    strmcpy(tmpbuf1, uhiline, mybufsize / 2);
    if (StatsiFilter) {
        tmpstr = new_next_arg(uhiline, &uhiline);
        tmpstr = new_next_arg(uhiline, &uhiline);
        if (!(tmpstr && *tmpstr)) tmpstr = empty_string;
    }
    if (!StatsiFilter || (StatsiFilter && 
        (wild_match(StatsiFilter, tmpstr) || wild_match(StatsiFilter, ipiline))))
        say("%s %s %s",statschar, ipiline, tmpbuf1);
    StatsiNumber++;
}

/* Does /STATS c with filter */
void StatsCFilter(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *tmpstr;

    tmpstr=new_next_arg(args,&args);
    if (tmpstr) {
        malloc_strcpy(&StatscFilter,tmpstr);
        StatscNumber=0;
        send_to_server("STATS c %s",args);
    }
    else PrintUsage("FCLINE filter [server]");
}

/* Parses STATS c reply from server */
void HandleStatsC(statschar,str1,str2)
char *statschar;
char *str1;
char *str2;
{
    char *tmpstr;
    char tmpbuf1[mybufsize/2+1];

    if (!StatscNumber) say("C-Line");
    strmcpy(tmpbuf1,str2,mybufsize/2);
    if (StatscFilter) {
        tmpstr=new_next_arg(str2,&str2);
        tmpstr=new_next_arg(str2,&str2);
        if (!(tmpstr && *tmpstr)) tmpstr=empty_string;
    }
    if (!StatscFilter || (StatscFilter && 
        (wild_match(StatscFilter,str1) || wild_match(StatscFilter,tmpstr))))
        say("%s %s %s",statschar,str1,tmpbuf1);
    StatscNumber++;
}

/* Does /STATS L with filter */
void StatsLFilter(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char statschar='L';
    char *tmpstr;

    tmpstr=new_next_arg(args,&args);
    if (tmpstr) {
        if (!my_stricmp(tmpstr,"-NAME")) {
            statschar='l';
            tmpstr=new_next_arg(args,&args);
        }
        if (tmpstr) {
            malloc_strcpy(&StatslFilter,tmpstr);
            StatslNumber=0;
            send_to_server("STATS %c * %s",statschar,args);
            return;
        }
    }
    PrintUsage("FLLINE [-NAME] filter");
}

/* Parses STATS L reply from server */
void HandleStatsL(statschar,str1,str2)
char *statschar;
char *str1;
char *str2;
{
    if (!StatslNumber) say("L-Line");
    if (!StatslFilter || (StatslFilter && (wild_match(StatslFilter,statschar))))
        say("%s %s %s",statschar,str1,str2);
    StatslNumber++;
}
#endif /* OPER */

/* Toggle value of ARinWindow */
void ARinWindowToggle(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *tmpstr;

    tmpstr=new_next_arg(args,&args);
    if (tmpstr) {
        if (!my_stricmp("OFF",tmpstr)) ARinWindow=0;
        else if (!my_stricmp("ON",tmpstr)) ARinWindow=1;
        else if (!my_stricmp("USER",tmpstr)) ARinWindow=2;
        else if (!my_stricmp("BOTH",tmpstr)) ARinWindow=3;
        else {
            PrintUsage("ARINWIN on/user/both/off");
            return;
        }
    }
    switch (ARinWindow) {
        case 0:
            tmpstr="OFF";
            break;
        case 1:
            tmpstr="ON";
            break;
        case 2:
            tmpstr="USER";
            break;
        case 3:
            tmpstr="BOTH";
            break;
    }
    PrintSetting("Auto reply in window",tmpstr,empty_string,empty_string);
}

/* Figure out if we're dealing with IrcNet's oper channel */
int IsIrcNetOperChannel(channel)
char *channel;
{
    if (!my_stricmp(channel,"&servers") || !my_stricmp(channel,"&errors")   ||
        !my_stricmp(channel,"&notices") || !my_stricmp(channel,"&local")    ||
        !my_stricmp(channel,"&channel") || !my_stricmp(channel,"&numerics") ||
        !my_stricmp(channel,"&kills")   || !my_stricmp(channel,"&clients")  ||
        !my_stricmp(channel,"&oper"))
        return(1);
    return(0);
}

/* Force redraw of status bars and input prompt */
void RedrawAll(void) {
    build_status(NULL);
    update_input(UPDATE_ALL);
    refresh_screen(0, NULL);
}

/* Compare two pointers for list.c */
int CompareAddr(element1,element2)
    List *element1;
    List *element2;
{
    typedef struct winlist_str Winlist;
    struct winlist_str {
        Winlist *next;
        Window *window;
    };
    Winlist *winelem=(Winlist *) element1;

    if (winelem->window==(Window *) element2) return(0);
    return(1);
}

/* Topic locking */
#ifdef EXTRAS
void TopicLocked(command, args, subargs)
char *command;
char *args;
char *subargs;
{
    int first = 1;
    int remove;
    char *topic = NULL;
    char *tmpchan = NULL;
    char tmpbuf[mybufsize / 4];
    ChannelList *chan;
    WhowasChanList *whowas;

    remove = !strcmp(command, "TOPICUNLOCK");
    tmpchan = new_next_arg(args, &args);
    topic = args;
    if (tmpchan && !is_channel(tmpchan)) {
        snprintf(tmpbuf, sizeof(tmpbuf), "#%s", tmpchan);
        tmpchan = tmpbuf;
    }
    switch (remove) {
        case 0:
            if (tmpchan && topic) {
                int didset = 0;
                char *tmpstr;

                for (tmpstr = topic; *tmpstr; tmpstr++)
                    ;
                if (tmpstr == topic) {
                    PrintUsage("TOPICLOCK [channel topic]");
                    return;
                }
                for (chan = server_list[curr_scr_win->server].chan_list; chan; chan = chan->next) {
                    if (CheckChannel(chan->channel, tmpchan)) {
                        didset = 1;
                        malloc_strcpy(&(chan->topiclock), topic);
                        CheckTopic(chan->channel, curr_scr_win->server, chan);
                        say("Topic %s is now locked for channel %s", topic, chan->channel);
                    }
                }
                if (!didset) say("You are not on channel %s", tmpchan);
            }
            else {
                for (chan = server_list[curr_scr_win->server].chan_list; chan; chan = chan->next) {
                    if (chan->topiclock) {
                        if (first) {
                            say("%-25.25s %s", "Channel", "topic");
                            first = 0;
                        }
                        say("%-25.25s %s", chan->channel, chan->topiclock);
                    }
                }
                for (whowas = whowas_chan_list; whowas; whowas = whowas->next) {
                    if (whowas->channellist->topiclock) {
                        if (first) {
                            say("%-25.25s %s", "Channel", "topic");
                            first = 0;
                        }
                        say("%-25.25s %s", whowas->channellist->channel, whowas->channellist->topiclock);
                    }
                }
                if (first) say("No channels are topic locked");
            }
            break;
        case 1:
            if (!tmpchan && (tmpchan = get_channel_by_refnum(0)) == NULL) {
                NoWindowChannel();
                return;
            }
            for (chan = server_list[curr_scr_win->server].chan_list; chan; chan = chan->next) {
                if (CheckChannel(chan->channel, tmpchan)) {
                    if (chan->topiclock) {
                        say("Topic lock has been removed for channel %s", tmpchan);
                        new_free(&(chan->topiclock));
                    }
                }
            }
            break;
    }
}

/* Check topic lock */
void CheckTopic(channel,server,tmpchan)
char *channel;
int server;
ChannelList *tmpchan;
{
    int changeit=0;
    ChannelList *chan;

    if (!tmpchan) chan=lookup_channel(channel,server,0);
    else chan=tmpchan;
    if (chan->topiclock) {
        if (!chan->topicstr) changeit=1;
        else if (strcmp(chan->topicstr,chan->topiclock)) changeit=1;
        if (changeit && HAS_OPS(chan->status)) send_to_server("TOPIC %s :%s",channel,chan->topiclock);
    }
}
#endif

/* Return time stamp in static buffer */
char *TimeStamp(when)
int when;
{
    static char stampbuf[mybufsize / 4 + 1];

    *stampbuf = '\0';
    if (Stamp >= when && TimeStampString) strmcpy(stampbuf, TimeStampString, mybufsize / 4);
    return(stampbuf);
}

/* CJOIN and KNOCK by braneded */
void CJoin(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    char *tmpchan=(char *) 0;
    char tmpbuf[mybufsize/4];

    tmpchan=new_next_arg(args,&args);
    if (tmpchan && !is_channel(tmpchan)) {
        snprintf(tmpbuf,sizeof(tmpbuf),"#%s",tmpchan);
        tmpchan=tmpbuf;
    }
    if (!tmpchan) {
        snprintf(tmpbuf,sizeof(tmpbuf),"%s [#]channel",command);
        PrintUsage(tmpbuf);
        return;
    }
    send_to_server("%s %s %s",command,tmpchan,args);
}

/* Configures per channel logging */
void ChanLogCommand(command, args, subargs)
char *command;
char *args;
char *subargs;
{
    char *setting = (char *) 0;

    setting = new_next_arg(args,&args);
    if (setting) {
        if (!strcmp(command, "CHANLOGDIR")) {
            if (!strcmp(setting, "-")) new_free(&ChanLogDir);
            else {
                char *path = expand_twiddle(setting);

                malloc_strcpy(&ChanLogDir, path);
                new_free(&path);
            }
        }
        else if (!strcmp(command, "CHANLOGPREFIX")) {
            if (!strcmp(setting, "-")) new_free(&ChanLogPrefix);
            else malloc_strcpy(&ChanLogPrefix, setting);
        }
        else if (!strcmp(command, "CHANLOGPOST")) {
            if (!strcmp(setting, "-")) new_free(&ChanLogPostfix);
            else malloc_strcpy(&ChanLogPostfix, setting);
        }
        SetChannels(100);
    }
    if (!strcmp(command, "CHANLOGDIR")) {
        if (ChanLogDir) PrintSetting("Channel logging directory", ChanLogDir, empty_string, empty_string);
        else PrintSetting("Channel logging directory", OpenCreateFile("", 0), empty_string, empty_string);
    }
    else if (!strcmp(command, "CHANLOGPREFIX")) {
        if (ChanLogPrefix) PrintSetting("Channel logging prefix", ChanLogPrefix, empty_string, empty_string);
        else PrintSetting("Channel logging prefix", "OFF", empty_string, empty_string);
    }
    else if (!strcmp(command, "CHANLOGPOST")) {
        if (ChanLogPostfix) PrintSetting("Channel logging postix", ChanLogPostfix, empty_string, empty_string);
        else PrintSetting("Channel logging postix", "OFF", empty_string, empty_string);
    }
}

/* Update channel logging filename */
void UpdateChanLogFName(chan)
ChannelList *chan;
{
    char *filename = NULL;
    char *filepath = NULL;
    char *channame = NULL;

    if (!chan) return;
    if (ChanLogPrefix) malloc_strcpy(&filename, ChanLogPrefix);
    malloc_strcpy(&channame, chan->channel);
    lower(channame);
    malloc_strcat(&filename, channame);
    new_free(&channame);
    if (ChanLogPostfix) malloc_strcat(&filename, ChanLogPostfix);
    if (ChanLogDir) malloc_strcpy(&filepath, ChanLogDir);
    else {
        char *path;

        path = OpenCreateFile(filename, 1);
        if (path) malloc_strcpy(&filepath, path);
        new_free(&filename);
    }
    malloc_strcpy(&(chan->chanlogfpath), filepath);
    if (filename) {
        malloc_strcat(&(chan->chanlogfpath), "/");
        malloc_strcat(&(chan->chanlogfpath), filename);
        new_free(&filename);
    }
    new_free(&filepath);
}

/* Report channel logging event (start, stop) */
void ChannelLogReport(event, chan)
char *event;
ChannelList *chan;
{
    char *tmpbuf = (char *) 0;
    char tmpbuf2[mybufsize / 16];
    FILE *fp;
    time_t timenow = time(NULL);

    if (!chan || !chan->ChanLog) return;
    fp = fopen(chan->chanlogfpath, "a");
    if (fp) {
        snprintf(tmpbuf2,sizeof(tmpbuf2), "[%.24s] ", ctime(&timenow));
        malloc_strcpy(&tmpbuf, tmpbuf2);
        malloc_strcat(&tmpbuf, "Log for ");
        malloc_strcat(&tmpbuf, chan->channel);
        malloc_strcat(&tmpbuf, " ");
        malloc_strcat(&tmpbuf, event);
        fprintf(fp, "%s\n", tmpbuf);
        fclose(fp);
        new_free(&tmpbuf);
    }
}

/* Report channel logging event for all channels */
void ChannelLogReportAll(event, chan)
char *event;
ChannelList *chan;
{
    ChannelList *tmpchan;

    for (tmpchan = chan; tmpchan; tmpchan = tmpchan->next) {
        if (tmpchan->ChanLog) ChannelLogReport(event, tmpchan);
    }
}

/* Save channel logging event (join, kick, ...) */
void ChannelLogSave(message, chan)
char *message;
ChannelList *chan;
{
    int  oldumask;
    char tmpbuf1[2 * mybufsize];
    char tmpbuf2[2 * mybufsize];
    FILE *logfile;
    time_t now;

    if (!chan || !chan->chanlogfpath) return;
    oldumask = umask(0177);
    if ((logfile = fopen(chan->chanlogfpath, "a")) != NULL) {
        now = time((time_t *) 0);
        snprintf(tmpbuf2, sizeof(tmpbuf2), "[%.24s] %s", ctime(&now), message);
        StripAnsi(tmpbuf2, tmpbuf1, 2);
        if (AwayEncrypt && EncryptPassword) EncryptString(tmpbuf2, tmpbuf1, EncryptPassword, mybufsize, 0);
        else strmcpy(tmpbuf2, tmpbuf1, sizeof(tmpbuf2));
        fprintf(logfile, "%s\n", tmpbuf2);
        fclose(logfile);
    }
    umask(oldumask);
}

#ifdef OPER
/* Does /STATS D with filter */
void StatsDFilter(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int doall=1;
    char *tmpstr;
    char *server=NULL;

    tmpstr=new_next_arg(args,&args);
    if (tmpstr && *tmpstr) {
        if (!my_strnicmp(tmpstr,"-TEMP",5)) {
            doall=0;
            tmpstr=new_next_arg(args,&args);
        }
        if (tmpstr && *tmpstr) {
            malloc_strcpy(&StatsdFilter,tmpstr);
            StatsdNumber=0;
            server=new_next_arg(args,&args);
            send_to_server("STATS %c %s",doall?'D':'d',server?server:"");
            return;
        }
    }
    PrintUsage("FDLINE [-TEMP] filter [server]");
}

/* Parses STATS d reply from server */
void HandleStatsD(statschar,str1,str2)
char *statschar;
char *str1;
char *str2;
{
    char *tmpstr;
    char tmpbuf1[mybufsize/2+1];

    if (!StatsdNumber) say("D-Line");
    strmcpy(tmpbuf1,str2,mybufsize/2);
    if (StatsdFilter) {
        tmpstr=new_next_arg(str2,&str2);
        tmpstr=new_next_arg(str2,&str2);
        if (!(tmpstr && *tmpstr)) tmpstr=empty_string;
    }
    if (!StatsdFilter || (StatsdFilter && 
        (wild_match(StatsdFilter,str1) || wild_match(StatsdFilter,tmpstr))))
        say("%s %s %s",statschar,str1,tmpbuf1);
    StatsdNumber++;
}
#endif /* OPER */

/* Clears channel of ban exceptions */
void CdExceptions(command,args,subargs)
char *command;
char *args;
char *subargs;
{
    int  count = 0;
    int  send = 0;
    int  max = get_int_var(MAX_MODES_VAR);
    char *tmpchan = (char *) 0;
    char tmpbuf1[mybufsize];
    char tmpbuf2[mybufsize];
    char tmpbuf3[mybufsize / 4];
    char tmpbuf4[mybufsize];
    ChannelList *chan;
    struct bans *tmpban;

    if (*args) {
        tmpchan = new_next_arg(args, &args);
        if (is_channel(tmpchan)) strmcpy(tmpbuf3, tmpchan, sizeof(tmpbuf3));
        else snprintf(tmpbuf3, sizeof(tmpbuf3), "#%s", tmpchan);
        tmpchan = tmpbuf3;
    }
    else if ((tmpchan = get_channel_by_refnum(0)) == NULL) {
        NoWindowChannel();
        return;
    }
    chan = lookup_channel(tmpchan, from_server, 0);
    if (!chan) return;
    if (HAS_OPS(chan->status)) {
        *tmpbuf1 = '\0';
        *tmpbuf2 = '\0';
        *tmpbuf4 = '\0';
        snprintf(tmpbuf2, sizeof(tmpbuf2), "MODE %s -", chan->channel);
        for (tmpban = chan->banlist; tmpban; tmpban = tmpban->next) {
            if (!tmpban->exception) continue;
            strmcat(tmpbuf2, "b", sizeof(tmpbuf2));
            strmcat(tmpbuf4, " ", sizeof(tmpbuf2));
            strmcat(tmpbuf4, tmpban->ban, sizeof(tmpbuf2));
            count++;
            send = 1;
            if (count == max) {
                strmcat(tmpbuf1, tmpbuf2, sizeof(tmpbuf1));
                strmcat(tmpbuf1, tmpbuf4, sizeof(tmpbuf1));
                strmcat(tmpbuf1, "\r\n", sizeof(tmpbuf1));
                snprintf(tmpbuf2, sizeof(tmpbuf2), "MODE %s -", chan->channel);
                *tmpbuf4 = '\0';
                count = 0;
            }
            if (strlen(tmpbuf1) >= IRCD_BUFFER_SIZE - 150) {
                send_to_server("%s", tmpbuf1);
                *tmpbuf1 = '\0';
                send = 0;
            }
        }
        if (count) {
            strmcat(tmpbuf1, tmpbuf2, sizeof(tmpbuf1));
            strmcat(tmpbuf1, tmpbuf4, sizeof(tmpbuf1));
            strmcat(tmpbuf1, "\r\n", sizeof(tmpbuf1));
            send=1;
        }
        if (send) send_to_server("%s", tmpbuf1);
    }
    else NotChanOp(tmpchan);
}

/* Send data to server */
void SendToServer(stuff)
char *stuff;
{
    send_to_server("%s", stuff);
}

/* Should we rate limit the join */
/* only on hybrid7 / ircd-ratbox for now */
int RateLimitJoin(server_index)
int server_index;
{
    char *verstr = server_list[server_index].version_string;
    if (wild_match("ircd-ratbox-*", verstr) || wild_match("hybrid-7*", verstr))
        return 1;
    return 0;
}

/* Extends current topic */
void ExtendTopic(command, args, subargs)
char *command;
char *args;
char *subargs;
{
    char *tmpchan = NULL;
    ChannelList *chan;

    if (args && *args) {
        if (is_channel(args)) tmpchan = new_next_arg(args, &args);
        else if (!(tmpchan = get_channel_by_refnum(0))) {
            NoWindowChannel();
            return;
        }
        chan = lookup_channel(tmpchan, curr_scr_win->server, 0);
        if (!chan) return;
        if (chan->topicstr) send_to_server("TOPIC %s :%s %s %s", chan->channel,
                                           chan->topicstr, ExtTopicDelimiter, args);
        else send_to_server("TOPIC %s :%s", chan->channel, args);
    }
    else PrintUsage("ETOPIC [#channel] text");
}

/* Sets topic delimiter */
void TopicDelimiter(command, args, subargs)
char *command;
char *args;
char *subargs;
{
    char *delim;
    char tmpbuf[mybufsize / 32];

    delim = new_next_arg(args, &args);
    if (delim) {
        snprintf(tmpbuf, sizeof(tmpbuf), "%s", delim);
        tmpbuf[1] = '\0';
        malloc_strcpy(&ExtTopicDelimiter, tmpbuf);
    }
    say("Topic delimiter is %s", ExtTopicDelimiter);
}


syntax highlighted by Code2HTML, v. 0.9.1