I found this Script a while ago and worked with it and wanted to make sure it gets seen by everyone since it's usefull at all.
Please note: I'm not the author of the orginial script! - The credit goes to wiebe, that's why I also let his credits in it.
Have fun.
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; PUSHMODE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;info:
;
; by wiebe @ QuakeNet
; version 1.31 (written and tested on mIRC 6.15)
;
; last edit: Fri Apr 30 2004
;
;
;What does this script do?
;
; Queues and dumps modes together as much as possible.
;
;
;Some examples:
;
; Example 1: from within same command
; voicing some users with the following command:
; //var %x = $nick($chan,0,r) | while (%x) { mode $chan +v $nick($chan,%x,r) | dec %x }
; and this happens:
;
; [18:45:32] * dev-mirc sets mode: +v zyrtepf
; [18:45:32] * dev-mirc sets mode: +v zcttchv
; [18:45:32] * dev-mirc sets mode: +v yznzko
; [18:45:32] * dev-mirc sets mode: +v yuxf
; [18:45:32] * dev-mirc sets mode: +v twyqh
; [18:45:34] * dev-mirc sets mode: +v pitqxpg
; [18:45:36] * dev-mirc sets mode: +v nsln
; [18:45:38] * dev-mirc sets mode: +v moypfnk
; [18:45:40] * dev-mirc sets mode: +v gibuhee
; [18:45:42] * dev-mirc sets mode: +v evuwntm
;
; This sends a mode command +v <nick> for each nick
;
; Now we do the same but with pushmode instead of the normal mode command:
; //var %x = $nick($chan,0,r) | while (%x) { pushmode $chan +v $nick($chan,%x,r) | dec %x }
; and this happens:
;
; [18:48:03] * dev-mirc sets mode: +vvvvvv zyrtepf zcttchv yznzko yuxf twyqh pitqxpg
; [18:48:04] * dev-mirc sets mode: +vvvv nsln moypfnk gibuhee evuwntm
;
; The modes were now pushed together, so only 2 mode commands are needed instead of 10
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
; Example 2: the same event repeated
; An normal anti-flood script for example does this:
;
; [18:23:32] <zcttchv> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:32] <evuwntm> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:32] <moypfnk> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:33] <zyrtepf> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:33] <gibuhee> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:33] <yznzko> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:33] <yuxf> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:33] <pitqxpg> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:33] <nsln> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:33] <twyqh> AVERAGE ANTI FLOOD SCRIPT WITHOUT PUSHMODE
; [18:23:33] * dev-mirc sets mode: +b *!*qgeqd@8V5Ry4.8vtU30.virtual
; [18:23:33] * dev-mirc sets mode: +b *!*ihgkzu@8V5Ry4.8vtU30.virtual
; [18:23:33] * dev-mirc sets mode: +b *!*dqqyex@8V5Ry4.8vtU30.virtual
; [18:23:36] * dev-mirc sets mode: +b *!*qfzzq@8V5Ry4.8vtU30.virtual
; [18:23:40] * dev-mirc sets mode: +b *!*alavec@8V5Ry4.8vtU30.virtual
; [18:23:44] * dev-mirc sets mode: +b *!*avwy@8V5Ry4.8vtU30.virtual
; [18:23:48] * dev-mirc sets mode: +b *!*tqkukhp@8V5Ry4.8vtU30.virtual
; [18:23:52] * dev-mirc sets mode: +b *!*xrvdohn@8V5Ry4.8vtU30.virtual
; [18:23:56] * dev-mirc sets mode: +b *!*ujundvh@8V5Ry4.8vtU30.virtual
; [18:24:00] * dev-mirc sets mode: +b *!*kid@8V5Ry4.8vtU30.virtual
;
; The anti-flood script sends each ban with a mode command
;
; The same script using pushmode does the following:
;
; [18:25:11] <zcttchv> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:11] <evuwntm> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:11] <moypfnk> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:11] <zyrtepf> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:11] <gibuhee> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:11] <yznzko> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:11] <yuxf> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:11] <pitqxpg> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:11] <nsln> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:12] <twyqh> AVERAGE ANTI FLOOD SCRIPT WITH PUSHMODE
; [18:25:12] * dev-mirc sets mode: +bbbbbb *!*qgeqd@8V5Ry4.8vtU30.virtual *!*ihgkzu@8V5Ry4.8vtU30.virtual *!*dqqyex@8V5Ry4.8vtU30.virtual *!*qfzzq@8V5Ry4.8vtU30.virtual *!*alavec@8V5Ry4.8vtU30.virtual *!*avwy@8V5Ry4.8vtU30.virtual
; [18:25:13] * dev-mirc sets mode: +bbbb *!*tqkukhp@8V5Ry4.8vtU30.virtual *!*xrvdohn@8V5Ry4.8vtU30.virtual *!*ujundvh@8V5Ry4.8vtU30.virtual *!*kid@8V5Ry4.8vtU30.virtual
;
; The anti-flood script sends the modes to pushmode, which turns 10 mode commands into 2
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
; Example 3: multiple events
; Some form of a bitchmode script:
;
; on @!*:op:#channel:{ if ($opnick != $me) { mode $chan -o $opnick } }
; on @!*:voice:#channel:{ if ($vnick != $me) { mode $chan -v $vnick } }
; on @!*:help:#channel:{ if ($hnick != $me) { mode $chan -h $hnick } }
; on @!*:mode:#channel:{
; if (n isincs $1) && (n !isincs $gettok($chan($chan).mode,1,32)) { mode $chan +n }
; if (t isincs $1) && (t !isincs $gettok($chan($chan).mode,1,32)) { mode $chan +t }
; }
;
; [13:54:31] * wiebe sets mode: -nt+ovh nick1 nick2 nick3
; [13:54:31] * dev-mirc sets mode: +n
; [13:54:31] * dev-mirc sets mode: +t
; [13:54:31] * dev-mirc sets mode: -o nick1
; [13:54:31] * dev-mirc sets mode: -v nick2
; [13:54:31] * dev-mirc sets mode: -h nick3
;
; With pushmode:
;
; on @!*:op:#channel:{ if ($opnick != $me) { pushmode $chan -o $opnick } }
; on @!*:voice:#channel:{ if ($vnick != $me) { pushmode $chan -v $vnick } }
; on @!*:help:#channel:{ if ($hnick != $me) { pushmode $chan -h $hnick } }
; on @!*:mode:#channel:{
; if (n isincs $1) && (n !isincs $gettok($chan($chan).mode,1,32)) { pushmode $chan +n }
; if (t isincs $1) && (t !isincs $gettok($chan($chan).mode,1,32)) { pushmode $chan +t }
; }
;
; [13:55:36] * wiebe sets mode: -nt+ohv nick1 nick2 nick3
; [13:55:36] * dev-mirc sets mode: -ohv+nt nick1 nick2 nick3
;
; only 1 mode line needed for the 4 events
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
;
;How to use this script?
;
; pushmode [-cdflnprswuNeM] <channel> <+/-mode> [arg]
;
; can be used to push channel modes together, see below
;
;
; The script has 4 different queue's,
; a queue for single modes (like +m) is emptied everytime pushmode is sending modes
; a faster queue, a normal queue (default) and a slower queue for modes with a parameter (like +b banmask)
; first the faster queue is emptied, then the default queue and last the slower queue
;
; The -n switch (next) queues the mode in the faster queue, the -l switch (low) can be used for the slower queue
; this is only for modes with a parameter
; pushmode #channel +v nick adds +v nick to the default queue
; pushmode -l #channel -v nick adds -v nick to the slower queue
; pushmode -n #channel +o nick adds +o nick to the faster queue
;
; The -c switch (clear) can be used to clear the queues
; the -l (low), -d (default), -n (next) and -s (single) switch
; can be used to clear a specific queue or multiple queues
; pushmode -c clears all queues for all channels
; pushmode -dc clears the default queue for all channels
; pushmode -nc #channel clears the faster queue for #channel
; pushmode -cs #channel clears the single queue for #channel
; pushmode -cdl #channel clears the default and slower queue for #channel
;
; The -f switch (flush) starts to empty the queue at once without waiting for the timer to start it
; pushmode -f starts pushmode for all channels
; pushmode -f #channel starts pushmode for #channel
; pushmode -f #channel +o nick starts pushmode for #channel after adding +o nick to the queue
;
; The -p switch (passive) can be used to only add a mode to the queue,
; but it will not cause pushmode to empty the queue,
; this mode will be send the next time pushmode is dumping modes
; or when the number of modes in the queues equals $modespl
; pushmode -p #channel +b queues in the default queue and
; sends +b (request banlist) the next time pushmode is dumping modes
; pushmode -p #channel -b *!*@host.com queues in the default queue and
; sends -b *!*@host.com the next chance it gets
;
; The -r switch (remove) removes a mode from the queue,
; the -l (low), -d (default) and -n (next) switch
; can be used to remove a mode from a specific queue or from multiple queues
; when combined with the -w a wildcardmatch is done
; if you want to clear the entire queue, it is better (read faster) to use the -c switch
; usefull when you want to set a key with pushmode and you first clear any +k and -k modes
; pushmode -dr #channel +b *!*@host.com removes +b *!*@host.com from the default queue
; pushmode -r #channel +v nick removes +v nick from all queues
; pushmode -rw #channel +b * removes all bans from all queues
; pushmode -rw #channel +? removes all single + modes from the queue
; pushmode -rdl #channel +k key removes +k key from the default and slower queue
;
; The -uN switch can be used to remove a ban after N seconds using pushmode
; using 0 for N, makes the script remove the ban from the internal tempban list
; this will not remove the ban from the channel
; pushmode -u120 #channel +b *!*@host.com removes the ban *!*@host.com after 120 seconds
; pushmode -u0 #channel +b *!*@leave.this.ban removes *!*@leave.this.ban from the internal tempban list
;
; The -eM switch (expire) removes the mode from the queue after M seconds,
; so if the mode is not send within M seconds, it will not be send
; pushmode -e10 #channel +m mode +m will only be send within 10 seconds or not
; pushmode -e60 #channel +l 100 mode +l 100 will only be send within 60 seconds or not
;
; Combinations are possible, for example:
; pushmode -nfpe60u600 #channel +b *!*user@*.host.com
;
; The script can voice/devoice users even if they changed nick since the mode was in the queue
; this can be done by giving nick!user@host as parameter
; the default settings makes the script understand that @%+ is ohv, if you are on a server with additional
; modes which can be set on users on a channel, you have to edit the prefix setting below
; the target needs to be in your IAL list for this to work
; the script adds/removes a tag to the IAL using ialmark and finds the nick when sending the mode
; other scripts using ialmark may break this part
;
; pushmode #channel +v goober!~bla@123.abc.isp.com will voice the user goober, even if he would
; change nick before the mode is send
;
; pushmode #channel -h somenick!someuser@abc.users.undernet.org dehalfops somenick, even if he would
; change nick before the mode is send
;
; pushmode #channel +o dev-mirc!dev-mirc@def.users.quakenet.org will op dev-mirc, even if he would
; change nick before the mode is send
;
;
; pushuser <+/-mode> [arg]
;
; can be used for usermodes, usefull if you have several scripts setting usermodes on connect
; pushuser will send them all at once instead of N times a mode command
;
;
; Pushmode (pushuser) can only take 1 mode at a time
; The - or + needs to be included
; The script does not allow duplicates in the same queue
; Temp bans work only if nick!user@host format is used (like *!*@host.com and not *host.com or just host.com)
; this is because of the isban operator
;
; Script is uses $modespl (MODES= setting),
; here meaning how many parameter modes (like +b banmask) can be put into 1 line
; the number of modes without parameter (like +m) is unlimited
; if you want to use this script on a server where this is different, you have to change the script a bit
; see the pushmode.dump alias
;
; Some checks are done on the modes, like if the mode exists, if the mode makes "sense" etc.
; it uses $chanmodes (with b,k,l,imnpst as default modes), checks op/halfop status.
; you can see the checks and change them in the pushmode.dump alias
;
;
;What use has this script?
;
; not only can you simply 'queue' mode changes from within the same event or script
; but mode changes by all scripts can be pushed together
;
;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHMODE.DELAY ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; when first called, the script will wait N seconds before sending a mode change
alias -l pushmode.delay {
return 1
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHMODE.RESTART ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; set here the delay (seconds) pushmode should have between multiple mode lines
; after sending a mode change and there are items left in the queue,
; it will wait N seconds before sending the next mode change
alias -l pushmode.restart {
return 2
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHMODE.BAN ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; set here the delay (seconds) for checking which bans are to be unset
; a timer runs with this interval
; it checks the temp bans set by this script for all channels and unbans them with pushmode
; settings this to 600 (10 min) for example, makes a temp ban set for 5min being unset after 5~15 minutes
; so this setting defines how accurate the time of a temp ban is
alias -l pushmode.ban {
return 600
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHMODE.PREFIX ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; set here what modes can be set on a user on a channel (this is NOT b or any other list mode)
; default @ = o, % = h, + = v
; leave empty if you dont want to use this
; if you are not sure what this is, leave it
; replace each char with their mode char, @ is o for example
alias -l pushmode.prefix {
return $nickmode
; should $nickmode not work, remove it and use the following line
return $replace($prefix,@,o,%,h,+,v)
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHMODE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; pushmode [-cdflnprswuNeM] <channel> <+/-mode> [arg]
alias pushmode {
var %f, %q, %p, %e, %u
; part for switches
if ($left($1,1) == -) {
; c switch is there
if (c isin $1) {
; no channel
if (!$2) {
; d switch is there, clear all default queues for this connection
if (d isin $1) { $pushmode.queue($+($cid,.pushmode.*.default)).clear }
; n switch is there, clear all next queues for this connection
if (n isin $1) { $pushmode.queue($+($cid,.pushmode.*.next)).clear }
; l switch is there, clear all low queues for this connection
if (l isin $1) { $pushmode.queue($+($cid,.pushmode.*.low)).clear }
; s switch is there, clear all single queues for this connection
if (s isin $1) { $pushmode.queue($+($cid,.pushmode.*.single)).clear }
; no other switches, clear all queues for this connection
if (n !isin $1) && (d !isin $1) && (l !isin $1) && (s !isin $1) { $pushmode.queue($+($cid,.pushmode.*)).clear }
}
else {
; d switch is there, clear the default queue for the channel
if (d isin $1) { $pushmode.queue($+($cid,.pushmode.,$hash($2,32),.default)).clear }
; n switch is there, clear the next queue for the channel
if (n isin $1) { $pushmode.queue($+($cid,.pushmode.,$hash($2,32),.next)).clear }
; l switch is there, clear the low queue for the channel
if (l isin $1) { $pushmode.queue($+($cid,.pushmode.,$hash($2,32),.low)).clear }
; s switch is there, clear the single queue for this connection
if (s isin $1) { $pushmode.queue($+($cid,.pushmode.,$hash($2,32),.single)).clear }
; no other switches, clear all queues for the channel
if (n !isin $1) && (d !isin $1) && (l !isin $1) && (s !isin $1) {
$pushmode.queue($+($cid,.pushmode.,$hash($2,32),.next)).clear
$pushmode.queue($+($cid,.pushmode.,$hash($2,32),.default)).clear
$pushmode.queue($+($cid,.pushmode.,$hash($2,32),.low)).clear
$pushmode.queue($+($cid,.pushmode.,$hash($2,32),.single)).clear
}
}
return
}
; f switch is there, set a var
if (f isin $1) { var %f = 1 }
; n switch is there, set a var
if (n isin $1) { var %q = next }
; l switch is there, set a var
elseif (l isin $1) { var %q = low }
; p switch is there, set a var
if (p isin $1) { var %p = 1 }
; u or e switch is used, set a var
if (u isin $1) || (e isin $1) { var %string = $remove($1,-,d,c,f,l,n,p,r,s,w) , %x = 1
; loop through all chars
while (%x <= $len(%string)) {
; found u, set a var
if ($mid(%string,%x,1) == u) { var %y = $calc(%x +1)
; as long as it is numbers, set a var, next char
while ($mid(%string,%y,1) isnum) { var %u = %u $+ $mid(%string,%y,1) | inc %y }
}
; found e, set a var
if ($mid(%string,%x,1) == e) { var %y = $calc(%x +1)
; as long as it is numbers, set a var, next char
while ($mid(%string,%y,1) isnum) { var %e = %e $+ $mid(%string,%y,1) | inc %y }
}
inc %x
}
; '%e' is a number, set a var
if (%e isnum) { var %e = -e $+ %e }
}
; u switch is there with a number, +b and a banmask, add to the hash table, make it decrease each second
if (u isin $1) && ($3 == +b) && ($4) && (%u > 0) {
hadd -m $+($cid,.pushmode.,$hash($2,32),.bans) $4 %u | hdec -c $+($cid,.pushmode.,$hash($2,32),.bans) $4
; check the timer, start the timer
if (!$timer($+($cid,.pushmode.bans))) {
.timer $+ $cid $+ .pushmode.bans 1 $$pushmode.ban pushmode.tempban
}
}
; number with u is 0
if (%u == 0) && ($3 == +b) && ($4) {
; check if it is already in the hash table as temp ban, delete it
if ($hget($+($cid,.pushmode.,$hash($2,32),.bans))) && ($hget($+($cid,.pushmode.,$hash($2,32),.bans),$4)) {
hdel $+($cid,.pushmode.,$hash($2,32),.bans) $4
; check if the hash table is empty, free the hash table
if ($hget($+($cid,.pushmode.,$hash($2,32),.bans),0).item == 0) {
hfree $+($cid,.pushmode.,$hash($2,32),.bans)
}
}
return
}
; r switch is there and mode '$3'
if (r isin $1) && ($3) {
; w switch is there
if (w isin $1) {
; no parameter, remove the mode from the single queue
if ($4 == $null) { pushmode.queue -rw $+($cid,.pushmode.,$hash($2,32),.single) $3 }
else {
; d switch is there, remove the mode from the default queue
if (d isin $1) { pushmode.queue -rw $+($cid,.pushmode.,$hash($2,32),.default) $3-4 }
; n switch is there, remove the mode from the next queue
if (n isin $1) { pushmode.queue -rw $+($cid,.pushmode.,$hash($2,32),.next) $3-4 }
; l switch is there, remove the mode from the low queue
if (l isin $1) { pushmode.queue -rw $+($cid,.pushmode.,$hash($2,32),.low) $3-4 }
; no other switch is there, remove the mode from all queues
if (n !isin $1) && (d !isin $1) && (l !isin $1) {
pushmode.queue -rw $+($cid,.pushmode.,$hash($2,32),.next) $3-4 | pushmode.queue -rw $+($cid,.pushmode.,$hash($2,32),.low) $3-4 | pushmode.queue -rw $+($cid,.pushmode.,$hash($2,32),.default) $3-4
}
}
}
else {
; no parameter, remove mode from the single queue
if ($4 == $null) { pushmode.queue -r $+($cid,.pushmode.,$hash($2,32),.single) $3 }
else {
; d switch is there, remove mode form the default queue
if (d isin $1) { pushmode.queue -r $+($cid,.pushmode.,$hash($2,32),.default) $3-4 }
; n switch is there, remove mode form next queue
if (n isin $1) { pushmode.queue -r $+($cid,.pushmode.,$hash($2,32),.next) $3-4 }
; l switch is there, remove mode form the low queue
if (l isin $1) { pushmode.queue -r $+($cid,.pushmode.,$hash($2,32),.low) $3-4 }
; no other switch is there, remove mode form all queues
if (n !isin $1) && (d !isin $1) && (l !isin $1) && (s !isin $1) {
pushmode.queue -r $+($cid,.pushmode.,$hash($2,32),.next) $3-4 | pushmode.queue -r $+($cid,.pushmode.,$hash($2,32),.low) $3-4 | pushmode.queue -r $+($cid,.pushmode.,$hash($2,32),.default) $3-4
}
}
}
return
}
tokenize 32 $2-
}
; leave multiple modes out
tokenize 32 $1 $left($2,2) $3
; '$2' starts with '+' or with '-', we are on channel '$1'
if ($istok(+ -,$left($2,1),32)) && ($me ison $1) {
; '$3' does exist
if ($3 != $null) {
; -n switch was used, queue mode '$2 $3' in the next queue
if (%q == next) { pushmode.queue %e $+($cid,.pushmode.,$hash($1,32),.next) $2-3 }
; -l switch was used, queue mode '$2 $3' in the low queue
elseif (%q == low) { pushmode.queue %e $+($cid,.pushmode.,$hash($1,32),.low) $2-3 }
; use default queue, queue mode '$2 $3' in the 'default' queue
else { pushmode.queue %e $+($cid,.pushmode.,$hash($1,32),.default) $2-3 }
}
; the mode does not have a parameter, queue mode '$2' it in the single queue
else { pushmode.queue $+($cid,.pushmode.,$hash($1,32),.single) $2 }
; check if the timer is not already running and check the -p switch, start the timer
if (!$timer($+($cid,.pushmode.,$hash($1,32)))) && (%p != 1) {
.timer $+ $cid $+ .pushmode. $+ $hash($1,32) 1 $$pushmode.delay pushmode.dump $1
}
; no timer, and passive switch, set a var
if (!$timer($+($cid,.pushmode.,$hash($1,32)))) && (%p == 1) {
var %x = 1, %queue = .next .default .low, %q = 1, %t = 0
; loop through each queue, inc var, next queue
while ($gettok(%queue,%q,32)) {
inc %t $pushmode.queue($+($cid,.pushmode.,$hash($1,32),$gettok(%queue,%q,32))).size | inc %q
}
; at least $modespl modes in the queues, start the timer
if (%t >= $modespl) {
.timer $+ $cid $+ .pushmode. $+ $hash($1,32) 1 $$pushmode.delay pushmode.dump $1
}
}
; f switch was used
if (%f == 1) {
; no channel, set a var
if (!$1) { var %x = $chan(0)
; loop through all the channels, stop the timer, run 'pushmode.dump chan', decrease '%x' and go on to the next channel
while (%x) {
.timer $+ $+($cid,.pushmode.,$hash($chan(%x),32)) off | pushmode.dump $chan(%x) | dec %x
}
}
; we are on '$1', stop the timer, run 'pushmode.dump $1'
elseif ($me ison $1) {
.timer $+ $+($cid,.pushmode.,$hash($1,32)) off | pushmode.dump $1
}
}
}
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHMODE.DUMP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; $1 = #channel
alias -l pushmode.dump {
; we are on channel '$1', set vars
if ($me ison $1) { var %x = 1, %queue = .next .default .low, %q = 1, %mode, %check, %smode
; loop through each queue
while ($gettok(%queue,%q,32)) {
; we loop as long as '%x' <= '$modespl' and as long as the size of the queue is greater then 0
while (%x <= $modespl) && ($pushmode.queue($+($cid,.pushmode.,$hash($1,32),$gettok(%queue,%q,32))).size > 0) {
; save the next item in a var
var %next = $pushmode.queue($+($cid,.pushmode.,$hash($1,32),$gettok(%queue,%q,32))).next
; check if mode is not already going to be set
if (!$istokcs(%check,%next,44)) && (%next) { var %1 = $gettok(%next,1,32), %2 = $gettok(%next,2,32)
; some checks / examples of checks
if (%1 === +o) && (%2 isop $1) { }
elseif (%1 === -o) && (%2 !isop $1) { }
elseif (%1 === +h) && (%2 ishop $1) { }
elseif (%1 === -h) && (%2 !ison $1) { }
elseif (%1 === +v) && (%2 isvoice $1) { }
elseif (%1 === -v) && (%2 !ison $1) { }
elseif (%1 === +b) && (%2 isban $1) { }
elseif (%1 === +l) && ((%2 !isnum) || (%2 < 1)) { }
elseif (%1 === +l) && (%2 == $chan($1).limit) { }
elseif (%1 === +k) && ($chan($1).key) { }
elseif (%1 === -k) && (%2 !=== $chan($1).key) { }
elseif ($right(%1,1) === o) && ($me !isop $1) { }
elseif ($right(%1,1) === h) && ($me !isop $1) { }
elseif ($me !isop $1) && ($me !ishop $1) { }
; add '%next' to '%check', add the mode in the var
else { var %check = $addtok(%check,%next,44) | var %mode = $+($gettok(%mode,1,32),$gettok(%next,1,32)) $gettok(%mode,2-,32) $gettok(%next,2,32) }
inc %x
}
}
inc %q
}
; we loop as long as the size of the single queue is greater then 0, set vars
while ($pushmode.queue($+($cid,.pushmode.,$hash($1,32),.single)).size > 0) {
var %next = $pushmode.queue($+($cid,.pushmode.,$hash($1,32),.single)).next
var %a = $+($gettok($chanmodes,1,44),b), %b = $+($gettok($chanmodes,2,44),k)
var %c = $+($gettok($chanmodes,3,44),l), %d = $+($gettok($chanmodes,4,44),imnpst)
; some checks / examples of checks
if ($right(%next,1) isincs %b) { }
elseif ($right(%next,1) isincs %c) && ($left(%next,1) == +) { }
elseif ($right(%next,1) isincs %b) { }
elseif ($me !isop $1) && ($me !ishop $1) && ($right(%next,1) !isincs %a) { }
; add next mode to the var
else { var %smode = $+(%smode,%next) }
}
; if the ibl isnt filled for that channel, add +b to request the banlist
; if you want this uncomment the following line
;if (b !isincs %smode) && (!$chan($1).ibl) { var %smode = $+(%smode,+b) }
; add the mode in the var
var %mode = $+($gettok(%mode,1,32),%smode) $gettok(%mode,2-,32)
; there are modes in '%mode', send the modes
if (%mode) { .quote MODE $1 %mode }
var %q = 1
; loop through the queues
while ($gettok(%queue,%q,32)) {
; items left, break
if ($pushmode.queue($+($cid,.pushmode.,$hash($1,32),$gettok(%queue,%q,32))).size > 0) { break }
inc %q
}
; loop was ended with break, start the timer
if (%q <= $numtok(%queue,32)) {
.timer $+ $cid $+ .pushmode. $+ $hash($1,32) 1 $$pushmode.restart pushmode.dump $1
}
}
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHUSER ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; $1 = +/-mode, $2 = param
alias pushuser {
; remove multiple modes, only use the 1st one given
tokenize 32 $left($1,2) $2
; '$2' is 2 chars long and '$1' starts with '+' or with '-'
if ($len($1) == 2) && ($istok(+ -, $left($1,1),32)) {
; '$2' exists, add the mode to the user queue
if ($2 != $null) { pushmode.queue $+($cid,.user) $1-2 }
; '$2' does not exist, add the mode to the user queue
else { pushmode.queue $+($cid,.user) $1 }
; check if the timer already runs, start the timer
if (!$timer($+($cid,.pushuser))) {
.timer $+ $cid $+ .pushuser 1 $$pushmode.delay pushuser.dump
}
}
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHUSER.DUMP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
alias -l pushuser.dump {
; loop as long as there are items in the user queue, set a var with the next item, add the mode to the var
while ($pushmode.queue($+($cid,.user)).size > 0) { var %next = $pushmode.queue($+($cid,.user)).next
var %mode = $+($gettok(%mode,1,32),$gettok(%next,1,32)) $gettok(%mode,2-,32) $gettok(%next,2,32) }
; if '%mode' exists, send the modes
if (%mode) { .quote MODE $me %mode }
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHMODE.QUEUE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; pushmode.queue [-eNrw] table mode param
alias -l pushmode.queue {
var %e
; switch is used
if ($left($1,1) == -) {
; got a switch, there are numbers, set a var
if (e isin $1) { if ($right($1,-2) isnum) { var %e = $+(-u,$right($1,-2)) } }
; r switch, w switch, set var
if (r isin $1) { if (w isin $1) { var %x = $hfind($2,$3-4,0,w).data
while (%x) {
; check hash table, delete
if ($gettok($hget($2,$hfind($2,$3-4,%x,w).data),1,32) === $3) || ($3 iswm $gettok($hget($2,$hfind($2,$3-4,%x,w).data),1,32)) {
hdel $2 $hfind($2,$3-4,%x,w).data
}
dec %x
}
}
; no w switch, set vars
else {
var %param = $replace($4,\,\\,|,\|,$chr(40),$+(\,$chr(40)),$chr(41),$+(\,$chr(41)),$chr(91),$+(\,$chr(91)),$chr(93),$+(\,$chr(93)),^,\^,$chr(123),$+(\,$chr(123)),$chr(125),$+(\,$chr(125)),.,\.,$,\$,+,\+,*,\*,?,\?)
var %mode = \ $+ $left($3,1) $+ $chr(91) $+ $replace($left($3,2),\,\\,|,\|,$chr(40),$+(\,$chr(40)),$chr(41),$+(\,$chr(41)),$chr(91),$+(\,$chr(91)),$chr(93),$+(\,$chr(93)),^,\^,$chr(123),$+(\,$chr(123)),$chr(125),$+(\,$chr(125)),.,\.,$,\$,+,\+,*,\*,?,\?) $+ $chr(93)
; check hash table, delete
if ($hfind($2,%mode %param,1,r).data) { hdel $2 $hfind($2,%mode %param,1,r).data }
}
; 1 item left, free hash table
if ($hget($2,0).item == 1) { hfree $2 }
return
}
tokenize 32 $2-4
}
; mode is a usermode, it matches *!*@* and no * or ? are there, and its not for the umode queue
if ($right($2,1) isin $pushmode.prefix) && (*!*@* iswm $3) && (* !isin $3) && (? !isin $3) && (*.user !iswm $1) {
; the user is in the ial, add tag to ialmark, tokenize
if ($ial($gettok($3,1,33))) {
.ialmark $gettok($3,1,33) $ial($gettok($3,1,33)).mark $+($1,.,$iif($hget($1,last),$calc($ifmatch +1),1))
tokenize 32 $1 $2 $+(*!,$gettok($3,2,33))
}
; user is not in ial, tokenize
else { tokenize 32 $1 $2 $gettok($3,1,33) }
}
; set a var, prefix special chars in a regex with a \
var %param = $replace($3,\,\\,|,\|,$chr(40),$+(\,$chr(40)),$chr(41),$+(\,$chr(41)),$chr(91),$+(\,$chr(91)),$chr(93),$+(\,$chr(93)),^,\^,$chr(123),$+(\,$chr(123)),$chr(125),$+(\,$chr(125)),.,\.,$,\$,+,\+,*,\*,?,\?)
var %mode = \ $+ $left($2,1) $+ $chr(91) $+ $replace($left($2,2),\,\\,|,\|,$chr(40),$+(\,$chr(40)),$chr(41),$+(\,$chr(41)),$chr(91),$+(\,$chr(91)),$chr(93),$+(\,$chr(93)),^,\^,$chr(123),$+(\,$chr(123)),$chr(125),$+(\,$chr(125)),.,\.,$,\$,+,\+,*,\*,?,\?) $+ $chr(93)
; there is a 2nd parameter and mode '%mode %param' is not already in the queue, where '%mode' is case sensitive
if ($2 != $null) && ($hfind($1,%mode %param,0,r).data == 0) {
; increase item 'last', add mode '$2-' to the hashtable with item name that 'last' has
hinc -m $1 last | hadd $+(-m,%e) $1 $hget($1,last) $2-
}
; propertie is next and hash table '$1' exists, increase item 'first'
elseif ($isid) && ($prop == next) && ($hget($1)) { hinc -m $1 first
; 'first' is smaller or equal to 'last', set vars
if ($hget($1,first) <= $hget($1,last)) {
var %next = $hget($1,$hget($1,first)), %number = $hget($1,first)
var %mode = $gettok(%next,1,32), %param = $gettok(%next,2,32)
; mode is a usermode, parameter matches *!*@*, set var
if ($right(%mode,1) isin $pushmode.prefix) && (*!*@* iswm %param) {
var %x = $ial(%param,0)
while (%x) { var %nick = $ial(%param,%x).nick
; the tag is there, remove it, set var, stop loop
if ($wildtok($ial(%nick).mark,$+($1,.,%number),1,32)) {
.ialmark %nick $remove($ial(%nick).mark,$ifmatch)
var %next = %mode %nick | break
}
dec %x
}
; no matches found, clear var
if (%x == 0) { var %next = $null }
}
; delete this item from the hashtable
hdel $1 $hget($1,first)
; this is the last item, free the hash table
if ($hget($1,first) >= $hget($1,last)) { hfree $1 }
return %next
}
}
; called as identifier ($alias) and propertie is size ($alias().size)
; decrease number of items with 1, (1 item in queue, and last is there)
elseif ($isid) && ($prop == size) { return $iif($calc($hget($1,0).item -1) >= 0,$ifmatch,0) }
; called as identifier ($alias) and propertie is clear ($alias().clear), free hashtables that match $1
elseif ($isid) && ($prop == clear) { hfree -w $1 }
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ALIAS PUSHMODE.TEMPBAN ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
alias -l pushmode.tempban {
var %x = 1
; loop through the channels
while (%x <= $chan(0)) {
; loop as long as a match is found, items with values 0 or lower
while ($hfind($+($cid,.pushmode.,$hash($chan(%x),32),.bans),/^[0-]/,1,r).data) {
; ibl is not filled, or the ban is set, use pushmode to remove the ban
if (!$chan(%x).ibl) || ($hfind($+($cid,.pushmode.,$hash($chan(%x),32),.bans),/^[0-]/,1,r).data isban $chan(%x)) {
pushmode $chan(%x) -b $hfind($+($cid,.pushmode.,$hash($chan(%x),32),.bans),/^[0-]/,1,r).data
}
; del the item from the hash table
hdel $+($cid,.pushmode.,$hash($chan(%x),32),.bans) $hfind($+($cid,.pushmode.,$hash($chan(%x),32),.bans),/^[0-]/,1,r).data
}
; check if hash table is empty and the hash table exists, remove hash table
if ($hget($+($cid,.pushmode.,$hash($chan(%x),32),.bans),0).item == 0) && ($hget($+($cid,.pushmode.,$hash($chan(%x),32),.bans))) {
hfree $+($cid,.pushmode.,$hash($chan(%x),32),.bans)
}
inc %x
}
; start the timer
.timer $+ $cid $+ .pushmode.bans 1 $$pushmode.ban pushmode.tempban
}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; DISCONNECT EVENT ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
on *:disconnect: { pushmode -c }