mirror of
https://github.com/irssi/irssi.git
synced 2024-12-04 14:46:39 -05:00
Allow storing multiple "other" prefixes such as +q and +a.
Original patch by JasonX, somewhat changed by exg and me. git-svn-id: file:///var/www/svn.irssi.org/SVN/irssi/trunk@4922 dbcabf3a-b0e7-0310-adc4-f8d773084564
This commit is contained in:
parent
aefa7b47c1
commit
89cd47bf3a
@ -19,7 +19,7 @@ unsigned int send_massjoin:1; /* Waiting to be sent in massjoin signal */
|
|||||||
unsigned int op:1;
|
unsigned int op:1;
|
||||||
unsigned int halfop:1;
|
unsigned int halfop:1;
|
||||||
unsigned int voice:1;
|
unsigned int voice:1;
|
||||||
char other;
|
char prefixes[MAX_USER_PREFIXES+1];
|
||||||
|
|
||||||
/*GHashTable *module_data;*/
|
/*GHashTable *module_data;*/
|
||||||
|
|
||||||
|
@ -359,45 +359,31 @@ GSList *nicklist_get_same_unique(SERVER_REC *server, void *id)
|
|||||||
/* nick record comparision for sort functions */
|
/* nick record comparision for sort functions */
|
||||||
int nicklist_compare(NICK_REC *p1, NICK_REC *p2, const char *nick_prefix)
|
int nicklist_compare(NICK_REC *p1, NICK_REC *p2, const char *nick_prefix)
|
||||||
{
|
{
|
||||||
int status1, status2;
|
int i;
|
||||||
|
|
||||||
if (p1 == NULL) return -1;
|
if (p1 == NULL) return -1;
|
||||||
if (p2 == NULL) return 1;
|
if (p2 == NULL) return 1;
|
||||||
|
|
||||||
/* we assign each status (op, halfop, voice, normal) a number
|
if (p1->prefixes[0] == p2->prefixes[0])
|
||||||
* and compare them. this is easier than 100,000 if's and
|
return g_strcasecmp(p1->nick, p2->nick);
|
||||||
* returns :-)
|
|
||||||
* -- yath */
|
|
||||||
|
|
||||||
if (p1->other) {
|
if (!p1->prefixes[0])
|
||||||
const char *other = (nick_prefix == NULL) ? NULL : strchr(nick_prefix, p1->other);
|
|
||||||
status1 = (other == NULL) ? 5 : 1000 - (other - nick_prefix);
|
|
||||||
} else if (p1->op)
|
|
||||||
status1 = 4;
|
|
||||||
else if (p1->halfop)
|
|
||||||
status1 = 3;
|
|
||||||
else if (p1->voice)
|
|
||||||
status1 = 2;
|
|
||||||
else
|
|
||||||
status1 = 1;
|
|
||||||
|
|
||||||
if (p2->other) {
|
|
||||||
const char *other = (nick_prefix == NULL) ? NULL : strchr(nick_prefix, p2->other);
|
|
||||||
status2 = (other == NULL) ? 5 : 1000 - (other - nick_prefix);
|
|
||||||
} else if (p2->op)
|
|
||||||
status2 = 4;
|
|
||||||
else if (p2->halfop)
|
|
||||||
status2 = 3;
|
|
||||||
else if (p2->voice)
|
|
||||||
status2 = 2;
|
|
||||||
else
|
|
||||||
status2 = 1;
|
|
||||||
|
|
||||||
if (status1 < status2)
|
|
||||||
return 1;
|
return 1;
|
||||||
else if (status1 > status2)
|
if (!p2->prefixes[0])
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
|
/* They aren't equal. We've taken care of that already.
|
||||||
|
* The first one we encounter in this list is the greater.
|
||||||
|
*/
|
||||||
|
|
||||||
|
for (i = 0; nick_prefix[i] != '\0'; i++) {
|
||||||
|
if (p1->prefixes[0] == nick_prefix[i])
|
||||||
|
return -1;
|
||||||
|
if (p2->prefixes[0] == nick_prefix[i])
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* we should never have gotten here... */
|
||||||
return g_strcasecmp(p1->nick, p2->nick);
|
return g_strcasecmp(p1->nick, p2->nick);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8,6 +8,8 @@
|
|||||||
#define IS_NICK(server) \
|
#define IS_NICK(server) \
|
||||||
(NICK(server) ? TRUE : FALSE)
|
(NICK(server) ? TRUE : FALSE)
|
||||||
|
|
||||||
|
#define MAX_USER_PREFIXES 7 /* Max prefixes kept for any user-in-chan. 7+1 is a memory unit */
|
||||||
|
|
||||||
struct _NICK_REC {
|
struct _NICK_REC {
|
||||||
#include "nick-rec.h"
|
#include "nick-rec.h"
|
||||||
};
|
};
|
||||||
|
@ -91,7 +91,6 @@ static void cmd_upgrade(const char *data)
|
|||||||
static void session_save_nick(CHANNEL_REC *channel, NICK_REC *nick,
|
static void session_save_nick(CHANNEL_REC *channel, NICK_REC *nick,
|
||||||
CONFIG_REC *config, CONFIG_NODE *node)
|
CONFIG_REC *config, CONFIG_NODE *node)
|
||||||
{
|
{
|
||||||
static char other[2];
|
|
||||||
node = config_node_section(node, NULL, NODE_TYPE_BLOCK);
|
node = config_node_section(node, NULL, NODE_TYPE_BLOCK);
|
||||||
|
|
||||||
config_node_set_str(config, node, "nick", nick->nick);
|
config_node_set_str(config, node, "nick", nick->nick);
|
||||||
@ -99,9 +98,7 @@ static void session_save_nick(CHANNEL_REC *channel, NICK_REC *nick,
|
|||||||
config_node_set_bool(config, node, "halfop", nick->halfop);
|
config_node_set_bool(config, node, "halfop", nick->halfop);
|
||||||
config_node_set_bool(config, node, "voice", nick->voice);
|
config_node_set_bool(config, node, "voice", nick->voice);
|
||||||
|
|
||||||
other[0] = nick->other;
|
config_node_set_str(config, node, "prefixes", nick->prefixes);
|
||||||
other[1] = '\0';
|
|
||||||
config_node_set_str(config, node, "other", other);
|
|
||||||
|
|
||||||
signal_emit("session save nick", 4, channel, nick, config, node);
|
signal_emit("session save nick", 4, channel, nick, config, node);
|
||||||
}
|
}
|
||||||
|
@ -413,14 +413,8 @@ static void display_sorted_nicks(CHANNEL_REC *channel, GSList *nicklist)
|
|||||||
for (tmp = nicklist; tmp != NULL; tmp = tmp->next) {
|
for (tmp = nicklist; tmp != NULL; tmp = tmp->next) {
|
||||||
NICK_REC *rec = tmp->data;
|
NICK_REC *rec = tmp->data;
|
||||||
|
|
||||||
if (rec->other)
|
if (rec->prefixes[0])
|
||||||
nickmode[0] = rec->other;
|
nickmode[0] = rec->prefixes[0];
|
||||||
else if (rec->op)
|
|
||||||
nickmode[0] = '@';
|
|
||||||
else if (rec->halfop)
|
|
||||||
nickmode[0] = '%';
|
|
||||||
else if (rec->voice)
|
|
||||||
nickmode[0] = '+';
|
|
||||||
else
|
else
|
||||||
nickmode[0] = ' ';
|
nickmode[0] = ' ';
|
||||||
|
|
||||||
|
@ -140,17 +140,13 @@ static char *channel_get_nickmode_rec(NICK_REC *nickrec)
|
|||||||
|
|
||||||
emptystr = settings_get_bool("show_nickmode_empty") ? " " : "";
|
emptystr = settings_get_bool("show_nickmode_empty") ? " " : "";
|
||||||
|
|
||||||
if (nickrec == NULL)
|
if (nickrec == NULL || nickrec->prefixes[0] == '\0')
|
||||||
nickmode = g_strdup(emptystr);
|
nickmode = g_strdup(emptystr);
|
||||||
else if (nickrec->other) {
|
else {
|
||||||
nickmode = g_malloc(2);
|
nickmode = g_malloc(2);
|
||||||
nickmode[0] = nickrec->other;
|
nickmode[0] = nickrec->prefixes[0];
|
||||||
nickmode[1] = '\0';
|
nickmode[1] = '\0';
|
||||||
} else
|
}
|
||||||
nickmode = g_strdup(nickrec->op ? "@" :
|
|
||||||
nickrec->halfop ? "%" :
|
|
||||||
nickrec->voice ? "+" :
|
|
||||||
emptystr);
|
|
||||||
return nickmode;
|
return nickmode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -86,18 +86,12 @@ static char *expando_usermode(SERVER_REC *server, void *item, int *free_ret)
|
|||||||
static char *expando_cumode(SERVER_REC *server, void *item, int *free_ret)
|
static char *expando_cumode(SERVER_REC *server, void *item, int *free_ret)
|
||||||
{
|
{
|
||||||
if (IS_IRC_CHANNEL(item) && CHANNEL(item)->ownnick) {
|
if (IS_IRC_CHANNEL(item) && CHANNEL(item)->ownnick) {
|
||||||
char other = NICK(CHANNEL(item)->ownnick)->other;
|
char prefix = NICK(CHANNEL(item)->ownnick)->prefixes[0];
|
||||||
if (other != '\0') {
|
|
||||||
char *cumode = g_malloc(2);
|
char *cumode = g_malloc(2);
|
||||||
cumode[0] = other;
|
cumode[0] = prefix;
|
||||||
cumode[1] = '\0';
|
cumode[1] = '\0';
|
||||||
*free_ret = TRUE;
|
*free_ret = TRUE;
|
||||||
return cumode;
|
return cumode; /* will be "\0\0" = "" if there is no prefix */
|
||||||
}
|
|
||||||
|
|
||||||
return NICK(CHANNEL(item)->ownnick)->op ? "@" :
|
|
||||||
NICK(CHANNEL(item)->ownnick)->halfop ? "%" :
|
|
||||||
NICK(CHANNEL(item)->ownnick)->voice ? "+" : "";
|
|
||||||
}
|
}
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
@ -99,7 +99,8 @@ static void event_names_list(IRC_SERVER_REC *server, const char *data)
|
|||||||
IRC_CHANNEL_REC *chanrec;
|
IRC_CHANNEL_REC *chanrec;
|
||||||
NICK_REC *rec;
|
NICK_REC *rec;
|
||||||
char *params, *type, *channel, *names, *ptr;
|
char *params, *type, *channel, *names, *ptr;
|
||||||
int op, halfop, voice, other;
|
int op, halfop, voice;
|
||||||
|
char prefixes[MAX_USER_PREFIXES+1];
|
||||||
|
|
||||||
g_return_if_fail(data != NULL);
|
g_return_if_fail(data != NULL);
|
||||||
|
|
||||||
@ -137,9 +138,11 @@ static void event_names_list(IRC_SERVER_REC *server, const char *data)
|
|||||||
/* some servers show ".@nick", there's also been talk about
|
/* some servers show ".@nick", there's also been talk about
|
||||||
showing "@+nick" and since none of these chars are valid
|
showing "@+nick" and since none of these chars are valid
|
||||||
nick chars, just check them until a non-nickflag char is
|
nick chars, just check them until a non-nickflag char is
|
||||||
found. FIXME: we just ignore owner char now. */
|
found. */
|
||||||
op = halfop = voice = other = FALSE;
|
op = halfop = voice = FALSE;
|
||||||
|
prefixes[0] = '\0';
|
||||||
while (isnickflag(server, *ptr)) {
|
while (isnickflag(server, *ptr)) {
|
||||||
|
prefix_add(prefixes, *ptr, (SERVER_REC *) server);
|
||||||
switch (*ptr) {
|
switch (*ptr) {
|
||||||
case '@':
|
case '@':
|
||||||
op = TRUE;
|
op = TRUE;
|
||||||
@ -150,8 +153,6 @@ static void event_names_list(IRC_SERVER_REC *server, const char *data)
|
|||||||
case '+':
|
case '+':
|
||||||
voice = TRUE;
|
voice = TRUE;
|
||||||
break;
|
break;
|
||||||
default:
|
|
||||||
other = *ptr;
|
|
||||||
}
|
}
|
||||||
ptr++;
|
ptr++;
|
||||||
}
|
}
|
||||||
@ -159,8 +160,7 @@ static void event_names_list(IRC_SERVER_REC *server, const char *data)
|
|||||||
if (nicklist_find((CHANNEL_REC *) chanrec, ptr) == NULL) {
|
if (nicklist_find((CHANNEL_REC *) chanrec, ptr) == NULL) {
|
||||||
rec = irc_nicklist_insert(chanrec, ptr, op, halfop,
|
rec = irc_nicklist_insert(chanrec, ptr, op, halfop,
|
||||||
voice, FALSE);
|
voice, FALSE);
|
||||||
if (other)
|
memcpy(rec->prefixes, prefixes, sizeof(rec->prefixes));
|
||||||
rec->other = other;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -114,7 +114,6 @@ static void sig_session_restore_nick(IRC_CHANNEL_REC *channel,
|
|||||||
CONFIG_NODE *node)
|
CONFIG_NODE *node)
|
||||||
{
|
{
|
||||||
const char *nick;
|
const char *nick;
|
||||||
char *other;
|
|
||||||
int op, halfop, voice;
|
int op, halfop, voice;
|
||||||
NICK_REC *nickrec;
|
NICK_REC *nickrec;
|
||||||
|
|
||||||
@ -129,8 +128,9 @@ static void sig_session_restore_nick(IRC_CHANNEL_REC *channel,
|
|||||||
voice = config_node_get_bool(node, "voice", FALSE);
|
voice = config_node_get_bool(node, "voice", FALSE);
|
||||||
halfop = config_node_get_bool(node, "halfop", FALSE);
|
halfop = config_node_get_bool(node, "halfop", FALSE);
|
||||||
nickrec = irc_nicklist_insert(channel, nick, op, halfop, voice, FALSE);
|
nickrec = irc_nicklist_insert(channel, nick, op, halfop, voice, FALSE);
|
||||||
other = config_node_get_str(node, "other", NULL);
|
strocpy(nickrec->prefixes,
|
||||||
nickrec->other = other == NULL ? '\0' : other[0];
|
config_node_get_str(node, "prefixes", ""),
|
||||||
|
sizeof(nickrec->prefixes));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void session_restore_channel(IRC_CHANNEL_REC *channel)
|
static void session_restore_channel(IRC_CHANNEL_REC *channel)
|
||||||
|
@ -46,8 +46,12 @@ static void nick_mode_change(IRC_CHANNEL_REC *channel, const char *nick,
|
|||||||
if (mode == '@') nickrec->op = type == '+';
|
if (mode == '@') nickrec->op = type == '+';
|
||||||
else if (mode == '+') nickrec->voice = type == '+';
|
else if (mode == '+') nickrec->voice = type == '+';
|
||||||
else if (mode == '%') nickrec->halfop = type == '+';
|
else if (mode == '%') nickrec->halfop = type == '+';
|
||||||
else if (channel->server->prefix[(unsigned char) mode] != '\0')
|
if (channel->server->prefix[(unsigned char) mode] != '\0') {
|
||||||
nickrec->other = (type == '+' ? mode : '\0');
|
if (type == '+')
|
||||||
|
prefix_add(nickrec->prefixes, mode, (SERVER_REC *) channel->server);
|
||||||
|
else
|
||||||
|
prefix_del(nickrec->prefixes, mode);
|
||||||
|
}
|
||||||
|
|
||||||
modestr[0] = mode; modestr[1] = '\0';
|
modestr[0] = mode; modestr[1] = '\0';
|
||||||
typestr[0] = type; typestr[1] = '\0';
|
typestr[0] = type; typestr[1] = '\0';
|
||||||
@ -55,6 +59,57 @@ static void nick_mode_change(IRC_CHANNEL_REC *channel, const char *nick,
|
|||||||
channel, nickrec, setby, modestr, typestr);
|
channel, nickrec, setby, modestr, typestr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void prefix_add(char *prefixes, char newprefix, SERVER_REC *server)
|
||||||
|
{
|
||||||
|
const char *prefixlst;
|
||||||
|
char newprefixes[MAX_USER_PREFIXES+1]; /* to hold the new prefixes */
|
||||||
|
unsigned int newpos = 0; /* to hold our position in the new prefixes */
|
||||||
|
unsigned int oldpos = 0; /* to hold our position in the old prefixes */
|
||||||
|
|
||||||
|
prefixlst = server->get_nick_flags(server);
|
||||||
|
|
||||||
|
/* go through the possible prefixes, copy higher ones, and find this one's place
|
||||||
|
* always leave room for the current prefix to be added, though.
|
||||||
|
*/
|
||||||
|
while (*prefixlst != '\0' && prefixes[oldpos] != '\0' &&
|
||||||
|
newpos < MAX_USER_PREFIXES - 1) {
|
||||||
|
if (prefixes[oldpos] == newprefix)
|
||||||
|
return; /* already inserted. why are we here? */
|
||||||
|
|
||||||
|
if (*prefixlst == newprefix)
|
||||||
|
break; /* insert the new prefix here */
|
||||||
|
|
||||||
|
if (*prefixlst == prefixes[oldpos]) {
|
||||||
|
/* this prefix is present.
|
||||||
|
* the one we are inserting goes after it.
|
||||||
|
* copy it over, and continue searching.
|
||||||
|
*/
|
||||||
|
newprefixes[newpos++] = prefixes[oldpos++];
|
||||||
|
}
|
||||||
|
prefixlst++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* newpos is now the position in which we wish to insert the prefix */
|
||||||
|
newprefixes[newpos++] = newprefix;
|
||||||
|
|
||||||
|
/* finish copying the remaining prefixes */
|
||||||
|
while (prefixes[oldpos] != '\0' && newpos < MAX_USER_PREFIXES)
|
||||||
|
newprefixes[newpos++] = prefixes[oldpos++];
|
||||||
|
|
||||||
|
newprefixes[newpos] = '\0';
|
||||||
|
|
||||||
|
memcpy(prefixes, newprefixes, sizeof(prefixes));
|
||||||
|
}
|
||||||
|
|
||||||
|
void prefix_del(char *prefixes, char oldprefix)
|
||||||
|
{
|
||||||
|
char *todel;
|
||||||
|
|
||||||
|
todel = strchr(prefixes, oldprefix);
|
||||||
|
if (todel)
|
||||||
|
memmove(todel, todel+1, strlen(todel));
|
||||||
|
}
|
||||||
|
|
||||||
static int mode_is_set(const char *str, char mode)
|
static int mode_is_set(const char *str, char mode)
|
||||||
{
|
{
|
||||||
char *end, *pos;
|
char *end, *pos;
|
||||||
|
@ -54,6 +54,9 @@ void channel_set_singlemode(IRC_CHANNEL_REC *channel, const char *nicks,
|
|||||||
void channel_set_mode(IRC_SERVER_REC *server, const char *channel,
|
void channel_set_mode(IRC_SERVER_REC *server, const char *channel,
|
||||||
const char *mode);
|
const char *mode);
|
||||||
|
|
||||||
|
void prefix_add(char *prefixes, char newprefix, SERVER_REC *server);
|
||||||
|
void prefix_del(char *prefixes, char oldprefix);
|
||||||
|
|
||||||
mode_func_t modes_type_a;
|
mode_func_t modes_type_a;
|
||||||
mode_func_t modes_type_b;
|
mode_func_t modes_type_b;
|
||||||
mode_func_t modes_type_c;
|
mode_func_t modes_type_c;
|
||||||
|
@ -135,7 +135,7 @@ static NETSPLIT_REC *netsplit_add(IRC_SERVER_REC *server, const char *nick,
|
|||||||
splitchan->op = nickrec->op;
|
splitchan->op = nickrec->op;
|
||||||
splitchan->halfop = nickrec->halfop;
|
splitchan->halfop = nickrec->halfop;
|
||||||
splitchan->voice = nickrec->voice;
|
splitchan->voice = nickrec->voice;
|
||||||
splitchan->other = nickrec->other;
|
memcpy(splitchan->prefixes, nickrec->prefixes, sizeof(splitchan->prefixes));
|
||||||
|
|
||||||
rec->channels = g_slist_append(rec->channels, splitchan);
|
rec->channels = g_slist_append(rec->channels, splitchan);
|
||||||
}
|
}
|
||||||
|
@ -28,7 +28,7 @@ typedef struct {
|
|||||||
unsigned int op:1;
|
unsigned int op:1;
|
||||||
unsigned int halfop:1;
|
unsigned int halfop:1;
|
||||||
unsigned int voice:1;
|
unsigned int voice:1;
|
||||||
unsigned int other:7;
|
char prefixes[MAX_USER_PREFIXES+1];
|
||||||
} NETSPLIT_CHAN_REC;
|
} NETSPLIT_CHAN_REC;
|
||||||
|
|
||||||
void netsplit_init(void);
|
void netsplit_init(void);
|
||||||
|
@ -180,14 +180,8 @@ static void dump_join(IRC_CHANNEL_REC *channel, CLIENT_REC *client)
|
|||||||
else
|
else
|
||||||
g_string_append_c(str, ' ');
|
g_string_append_c(str, ' ');
|
||||||
|
|
||||||
if (nick->other)
|
if (nick->prefixes[0])
|
||||||
g_string_append_c(str, nick->other);
|
g_string_append_c(str, nick->prefixes[0]);
|
||||||
else if (nick->op)
|
|
||||||
g_string_append_c(str, '@');
|
|
||||||
else if (nick->halfop)
|
|
||||||
g_string_append_c(str, '%');
|
|
||||||
else if (nick->voice)
|
|
||||||
g_string_append_c(str, '+');
|
|
||||||
g_string_append(str, nick->nick);
|
g_string_append(str, nick->nick);
|
||||||
}
|
}
|
||||||
g_slist_free(nicks);
|
g_slist_free(nicks);
|
||||||
|
@ -432,7 +432,8 @@ void perl_nick_fill_hash(HV *hv, NICK_REC *nick)
|
|||||||
hv_store(hv, "op", 2, newSViv(nick->op), 0);
|
hv_store(hv, "op", 2, newSViv(nick->op), 0);
|
||||||
hv_store(hv, "halfop", 6, newSViv(nick->halfop), 0);
|
hv_store(hv, "halfop", 6, newSViv(nick->halfop), 0);
|
||||||
hv_store(hv, "voice", 5, newSViv(nick->voice), 0);
|
hv_store(hv, "voice", 5, newSViv(nick->voice), 0);
|
||||||
hv_store(hv, "other", 5, newSViv(nick->other), 0);
|
hv_store(hv, "other", 5, newSViv(nick->prefixes[0]), 0);
|
||||||
|
hv_store(hv, "prefixes", 8, new_pv(nick->prefixes), 0);
|
||||||
|
|
||||||
hv_store(hv, "last_check", 10, newSViv(nick->last_check), 0);
|
hv_store(hv, "last_check", 10, newSViv(nick->last_check), 0);
|
||||||
hv_store(hv, "send_massjoin", 13, newSViv(nick->send_massjoin), 0);
|
hv_store(hv, "send_massjoin", 13, newSViv(nick->send_massjoin), 0);
|
||||||
|
Loading…
Reference in New Issue
Block a user