Redesign the Ban List window. Closes Issues #303, #342, #427

This is a combination of 18 commits.
 The first commit's message is:
Here is the initial banlist branch of RichardHitt/hexchat.
Changed files are only src/fe-gtk/{banlist.c,fe-gtk.h}.
This version works and contains my first efforts at selective
sensitization of radio buttons and control buttons.

From this point I intend to undertake a stepwise redesign.

Step 1 will be to make the existing banlist code work for
multiple simultaneous banlist windows (for different channels,
obviously).  It will be a hackathon with the only goal of getting
it working.

Step 2 will be the objectization and alpha-stage tidying-up of
all the terrible looking stuff I will have done in Step 1.

 This is the 2nd commit message:

Here's the post-Step-1 commit.  It works for multiple banlist windows.

Note particularly what I've done to banlist.h.
Note that for many functions in banlist.c the argument is now
a banlist_info *, rather than a session *.
Note in banlist.c the initialization of array modes[] which
contains driving information for the checkboxes.

Of course those checkboxes aren't yet implemented.  Maybe in Step 2
I will change to checkboxes from radio buttons; but definitely I will
change to letting modes[] drive processing.

 This is the 3rd commit message:

Converted to checkboxes.  Much additional work.
Note that the infrastructure for Auto-invite is not yet
present in the hexchat tree.  I'm nearly done with banlist,
I think!

 This is the 4th commit message:

Fleshed out 'invite'.  Tagged masks uniformly, e.g. (b) (e) (I).
General cleanup, nearly at the point of beta quality.

 This is the 5th commit message:

Added fourth mode type: quiet.  Did lots and lots of cleanup.  Beta-ready?

 This is the 6th commit message:

Get the banlist timestamps properly sortable.

 This is the 7th commit message:

Redesign the supports_foo() routines.  Now they're responsible for setting
the flags in ->capable, ->readable, ->writeable.

 This is the 8th commit message:

Deleted a couple of RBH comments.

 This is the 9th commit message:

Now the ESC key will close the banlist window.

 This is the 10th commit message:

Fix the fe-text occurrence of fe_add_ban_list().

 This is the 11th commit message:

Fixed also fe_ban_list_end() and removed fe_is_banwindow().

 This is the 12th commit message:

Use old-style initialization for array of structures modes[]

 This is the 13th commit message:

Oops, incomplete regression of modes[] initialization.  This fixes.

 This is the 14th commit message:

Fixed strptime buy implementing a special version here.
Fixed column width concerns by setting resizable and autosize.

 This is the 15th commit message:

Get rid of testing line.

 This is the 16th commit message:

Changed to gtkutil_destroy_on_esc ()

 This is the 17th commit message:

Remove no-longer-used functnion

 This is the 18th commit message:

Minor cleanups to banlist.c, banlist.h

 Please enter the commit message for your changes. Lines starting
 with '#' will be ignored, and an empty message aborts the commit.

 Author:    RichardHitt <rbh00@netcom.com>
 Committer: Richard Hitt <rbh00@f17.rbh00.pacbell.net>

 Not currently on any branch.
 Changes to be committed:
   (use "git reset HEAD <file>..." to unstage)

	modified:   src/common/fe.h
	modified:   src/common/hexchat.h
	modified:   src/common/inbound.c
	modified:   src/common/modes.c
	modified:   src/common/proto-irc.c
	modified:   src/common/server.c
	modified:   src/fe-gtk/banlist.c
	modified:   src/fe-gtk/banlist.h
	modified:   src/fe-gtk/fe-gtk.c
	modified:   src/fe-gtk/fe-gtk.h
	modified:   src/fe-gtk/maingui.c
	modified:   src/fe-text/fe-text.c
This commit is contained in:
RichardHitt 2013-03-05 00:13:51 -08:00 committed by Richard Hitt
parent c5404b8e25
commit f5631b2e22
12 changed files with 583 additions and 150 deletions

View File

@ -59,9 +59,8 @@ int fe_is_chanwindow (struct server *serv);
void fe_add_chan_list (struct server *serv, char *chan, char *users, void fe_add_chan_list (struct server *serv, char *chan, char *users,
char *topic); char *topic);
void fe_chan_list_end (struct server *serv); void fe_chan_list_end (struct server *serv);
int fe_is_banwindow (struct session *sess); gboolean fe_add_ban_list (struct session *sess, char *mask, char *who, char *when, int rplcode);
void fe_add_ban_list (struct session *sess, char *mask, char *who, char *when, int is_exemption); gboolean fe_ban_list_end (struct session *sess, int rplcode);
void fe_ban_list_end (struct session *sess, int is_exemption);
void fe_notify_update (char *name); void fe_notify_update (char *name);
void fe_notify_ask (char *name, char *networks); void fe_notify_ask (char *name, char *networks);
void fe_text_clear (struct session *sess, int lines); void fe_text_clear (struct session *sess, int lines);

View File

@ -575,6 +575,7 @@ typedef struct server
unsigned int have_idmsg:1; /* freenode's IDENTIFY-MSG */ unsigned int have_idmsg:1; /* freenode's IDENTIFY-MSG */
unsigned int have_sasl:1; /* SASL capability */ unsigned int have_sasl:1; /* SASL capability */
unsigned int have_except:1; /* ban exemptions +e */ unsigned int have_except:1; /* ban exemptions +e */
unsigned int have_invite:1; /* invite exemptions +I */
unsigned int using_cp1255:1; /* encoding is CP1255/WINDOWS-1255? */ unsigned int using_cp1255:1; /* encoding is CP1255/WINDOWS-1255? */
unsigned int using_irc:1; /* encoding is "IRC" (CP1252/UTF-8 hybrid)? */ unsigned int using_irc:1; /* encoding is "IRC" (CP1252/UTF-8 hybrid)? */
unsigned int use_who:1; /* whether to use WHO command to get dcc_ip */ unsigned int use_who:1; /* whether to use WHO command to get dcc_ip */

View File

@ -1272,12 +1272,14 @@ inbound_user_info (session *sess, char *chan, char *user, char *host,
} }
int int
inbound_banlist (session *sess, time_t stamp, char *chan, char *mask, char *banner, int is_exemption) inbound_banlist (session *sess, time_t stamp, char *chan, char *mask, char *banner, int rplcode)
{ {
char *time_str = ctime (&stamp); char *time_str = ctime (&stamp);
server *serv = sess->server; server *serv = sess->server;
char *nl;
time_str[19] = 0; /* get rid of the \n */ if ((nl = strchr (time_str, '\n')))
*nl = 0;
if (stamp == 0) if (stamp == 0)
time_str = ""; time_str = "";
@ -1288,18 +1290,17 @@ inbound_banlist (session *sess, time_t stamp, char *chan, char *mask, char *bann
goto nowindow; goto nowindow;
} }
if (!fe_is_banwindow (sess)) if (!fe_add_ban_list (sess, mask, banner, time_str, rplcode))
{ {
nowindow: nowindow:
/* let proto-irc.c do the 'goto def' for exemptions */ /* let proto-irc.c do the 'goto def' for exemptions */
if (is_exemption) if (rplcode == 348) /* RPL_EXCEPTLIST */
return FALSE; return FALSE;
EMIT_SIGNAL (XP_TE_BANLIST, sess, chan, mask, banner, time_str, 0); EMIT_SIGNAL (XP_TE_BANLIST, sess, chan, mask, banner, time_str, 0);
return TRUE; return TRUE;
} }
fe_add_ban_list (sess, mask, banner, time_str, is_exemption);
return TRUE; return TRUE;
} }

View File

@ -824,6 +824,10 @@ inbound_005 (server * serv, char *word[])
#ifndef WIN32 #ifndef WIN32
serv->have_except = TRUE; serv->have_except = TRUE;
#endif #endif
} else if (strcmp (word[w], "INVEX") == 0)
{
/* supports mode letter +I, default channel invite */
serv->have_invite = TRUE;
} else if (strncmp (word[w], "ELIST=", 6) == 0) } else if (strncmp (word[w], "ELIST=", 6) == 0)
{ {
/* supports LIST >< min/max user counts? */ /* supports LIST >< min/max user counts? */

View File

@ -778,8 +778,18 @@ process_numeric (session * sess, int n,
} }
break; break;
case 346: /* +I-list entry */
if (!inbound_banlist (sess, atol (word[7]), word[4], word[5], word[6], 346))
goto def;
break;
case 347: /* end of invite list */
if (!fe_ban_list_end (sess, 347))
goto def;
break;
case 348: /* +e-list entry */ case 348: /* +e-list entry */
if (!inbound_banlist (sess, atol (word[7]), word[4], word[5], word[6], TRUE)) if (!inbound_banlist (sess, atol (word[7]), word[4], word[5], word[6], 348))
goto def; goto def;
break; break;
@ -790,9 +800,8 @@ process_numeric (session * sess, int n,
sess = serv->front_session; sess = serv->front_session;
goto def; goto def;
} }
if (!fe_is_banwindow (sess)) if (!fe_ban_list_end (sess, 349))
goto def; goto def;
fe_ban_list_end (sess, TRUE);
break; break;
case 353: /* NAMES */ case 353: /* NAMES */
@ -806,7 +815,8 @@ process_numeric (session * sess, int n,
break; break;
case 367: /* banlist entry */ case 367: /* banlist entry */
inbound_banlist (sess, atol (word[7]), word[4], word[5], word[6], FALSE); if (!inbound_banlist (sess, atol (word[7]), word[4], word[5], word[6], 367))
goto def;
break; break;
case 368: case 368:
@ -816,9 +826,8 @@ process_numeric (session * sess, int n,
sess = serv->front_session; sess = serv->front_session;
goto def; goto def;
} }
if (!fe_is_banwindow (sess)) if (!fe_ban_list_end (sess, 368))
goto def; goto def;
fe_ban_list_end (sess, FALSE);
break; break;
case 369: /* WHOWAS end */ case 369: /* WHOWAS end */
@ -881,6 +890,18 @@ process_numeric (session * sess, int n,
notify_set_online (serv, word[4]); notify_set_online (serv, word[4]);
break; break;
case 728: /* +q-list entry */
/* NOTE: FREENODE returns these results inconsistent with e.g. +b */
/* Who else has imlemented MODE_QUIET, I wonder? */
if (!inbound_banlist (sess, atol (word[8]), word[4], word[6], word[7], 728))
goto def;
break;
case 729: /* end of quiet list */
if (!fe_ban_list_end (sess, 729))
goto def;
break;
case 903: /* successful SASL auth */ case 903: /* successful SASL auth */
case 904: /* aborted SASL auth */ case 904: /* aborted SASL auth */
case 905: /* failed SASL auth */ case 905: /* failed SASL auth */

View File

@ -1890,6 +1890,7 @@ server_set_defaults (server *serv)
serv->have_idmsg = FALSE; serv->have_idmsg = FALSE;
serv->have_sasl = FALSE; serv->have_sasl = FALSE;
serv->have_except = FALSE; serv->have_except = FALSE;
serv->have_invite = FALSE;
} }
char * char *

View File

@ -36,6 +36,7 @@
#include <gtk/gtkmessagedialog.h> #include <gtk/gtkmessagedialog.h>
#include <gtk/gtktreeview.h> #include <gtk/gtktreeview.h>
#include <gtk/gtktreeselection.h> #include <gtk/gtktreeselection.h>
#include <glib.h>
#include "../common/hexchat.h" #include "../common/hexchat.h"
#include "../common/fe.h" #include "../common/fe.h"
@ -46,6 +47,52 @@
#include "maingui.h" #include "maingui.h"
#include "banlist.h" #include "banlist.h"
/*
* These supports_* routines set capable, readable, writable bits */
static void supports_bans (banlist_info *, int);
static void supports_exempt (banlist_info *, int);
static void supports_invite (banlist_info *, int);
static void supports_quiet (banlist_info *, int);
static mode_info modes[MODE_CT] = {
{
"Bans",
"(b) ",
'b',
RPL_BANLIST,
RPL_ENDOFBANLIST,
1<<MODE_BAN,
supports_bans
}
,{
"Exempts",
"(e) ",
'e',
RPL_EXCEPTLIST,
RPL_ENDOFEXCEPTLIST,
1<<MODE_EXEMPT,
supports_exempt
}
,{
"Invites",
"(I) ",
'I',
RPL_INVITELIST,
RPL_ENDOFINVITELIST,
1<<MODE_INVITE,
supports_invite
}
,{
"Quiets",
"(q) ",
'q',
RPL_QUIETLIST,
RPL_ENDOFQUIETLIST,
1<<MODE_QUIET,
supports_quiet
}
};
/* model for the banlist tree */ /* model for the banlist tree */
enum enum
{ {
@ -58,7 +105,7 @@ enum
static GtkTreeView * static GtkTreeView *
get_view (struct session *sess) get_view (struct session *sess)
{ {
return GTK_TREE_VIEW (sess->res->banlist_treeview); return GTK_TREE_VIEW (sess->res->banlist->treeview);
} }
static GtkListStore * static GtkListStore *
@ -67,91 +114,281 @@ get_store (struct session *sess)
return GTK_LIST_STORE (gtk_tree_view_get_model (get_view (sess))); return GTK_LIST_STORE (gtk_tree_view_get_model (get_view (sess)));
} }
static gboolean static void
supports_exempt (server *serv) supports_bans (banlist_info *banl, int i)
{ {
int bit = 1<<i;
banl->capable |= bit;
banl->readable |= bit;
banl->writeable |= bit;
return;
}
static void
supports_exempt (banlist_info *banl, int i)
{
server *serv = banl->sess->server;
char *cm = serv->chanmodes; char *cm = serv->chanmodes;
int bit = 1<<i;
if (serv->have_except) if (serv->have_except)
return TRUE; goto yes;
if (!cm) if (!cm)
return FALSE; return;
while (*cm) while (*cm)
{ {
if (*cm == ',') if (*cm == ',')
break; break;
if (*cm == 'e') if (*cm == 'e')
return TRUE; goto yes;
cm++; cm++;
} }
return;
return FALSE; yes:
banl->capable |= bit;
banl->writeable |= bit;
} }
void static void
fe_add_ban_list (struct session *sess, char *mask, char *who, char *when, int is_exempt) supports_invite (banlist_info *banl, int i)
{ {
server *serv = banl->sess->server;
char *cm = serv->chanmodes;
int bit = 1<<i;
if (serv->have_invite)
goto yes;
if (!cm)
return;
while (*cm)
{
if (*cm == ',')
break;
if (*cm == 'I')
goto yes;
cm++;
}
return;
yes:
banl->capable |= bit;
banl->writeable |= bit;
}
static void
supports_quiet (banlist_info *banl, int i)
{
server *serv = banl->sess->server;
char *cm = serv->chanmodes;
int bit = 1<<i;
if (!cm)
return;
while (*cm)
{
if (*cm == ',')
break;
if (*cm == modes[i].letter)
goto yes;
cm++;
}
return;
yes:
banl->capable |= bit;
banl->readable |= bit;
banl->writeable |= bit;
}
/* fe_add_ban_list() and fe_ban_list_end() return TRUE if consumed, FALSE otherwise */
gboolean
fe_add_ban_list (struct session *sess, char *mask, char *who, char *when, int rplcode)
{
banlist_info *banl = sess->res->banlist;
int i;
GtkListStore *store; GtkListStore *store;
GtkTreeIter iter; GtkTreeIter iter;
char buf[512]; char buf[512];
if (!banl)
return FALSE;
for (i = 0; i < MODE_CT; i++)
if (modes[i].code == rplcode)
break;
if (i == MODE_CT)
{
/* printf ("Unexpected value in fe_add_ban_list: %d\n", rplcode); */
return FALSE;
}
if (banl->pending & 1<<i)
{
store = get_store (sess); store = get_store (sess);
gtk_list_store_append (store, &iter); gtk_list_store_append (store, &iter);
if (is_exempt) g_snprintf (buf, sizeof buf, "%s%s", modes[i].tag, mask);
{
snprintf (buf, sizeof (buf), "(EX) %s", mask);
gtk_list_store_set (store, &iter, 0, buf, 1, who, 2, when, -1); gtk_list_store_set (store, &iter, 0, buf, 1, who, 2, when, -1);
} else
{ banl->line_ct++;
gtk_list_store_set (store, &iter, 0, mask, 1, who, 2, when, -1); return TRUE;
} }
else return FALSE;
} }
void /* Sensitize checkboxes and buttons as appropriate for the moment */
fe_ban_list_end (struct session *sess, int is_exemption) static void
banlist_sensitize (banlist_info *banl)
{ {
gtk_widget_set_sensitive (sess->res->banlist_butRefresh, TRUE); int checkable, i;
/* CHECKBOXES -- */
checkable = banl->sess->me->op? banl->writeable: banl->readable;
for (i = 0; i < MODE_CT; i++)
{
if (banl->checkboxes[i] == NULL)
continue;
if ((checkable & 1<<i) == 0)
/* Checkbox is not checkable. Grey it and uncheck it. */
{
gtk_widget_set_sensitive (banl->checkboxes[i], FALSE);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (banl->checkboxes[i]), FALSE);
}
else
/* Checkbox is checkable. Be sure it's sensitive. */
{
gtk_widget_set_sensitive (banl->checkboxes[i], TRUE);
}
}
/* BUTTONS --- */
if (banl->sess->me->op == 0 || banl->line_ct == 0)
{
/* If user is not op or list is empty, buttons should be all greyed */
gtk_widget_set_sensitive (banl->but_clear, FALSE);
gtk_widget_set_sensitive (banl->but_crop, FALSE);
gtk_widget_set_sensitive (banl->but_remove, FALSE);
}
else
{
/* If no lines are selected, only the CLEAR button should be sensitive */
if (banl->select_ct == 0)
{
gtk_widget_set_sensitive (banl->but_clear, TRUE);
gtk_widget_set_sensitive (banl->but_crop, FALSE);
gtk_widget_set_sensitive (banl->but_remove, FALSE);
}
/* If any lines are selected, only the REMOVE and CROP buttons should be sensitive */
else
{
gtk_widget_set_sensitive (banl->but_clear, FALSE);
gtk_widget_set_sensitive (banl->but_crop, TRUE);
gtk_widget_set_sensitive (banl->but_remove, TRUE);
}
}
/* Set "Refresh" sensitvity */
gtk_widget_set_sensitive (banl->but_refresh, banl->pending? FALSE: banl->checked? TRUE: FALSE);
}
/* fe_ban_list_end() returns TRUE if consumed, FALSE otherwise */
gboolean
fe_ban_list_end (struct session *sess, int rplcode)
{
banlist_info *banl = sess->res->banlist;
int i;
if (!banl)
return FALSE;
for (i = 0; i < MODE_CT; i++)
if (modes[i].endcode == rplcode)
break;
if (i == MODE_CT)
{
/* printf ("Unexpected rplcode value in fe_ban_list_end: %d\n", rplcode); */
return FALSE;
}
if (banl->pending & modes[i].bit)
{
banl->pending &= ~modes[i].bit;
if (!banl->pending)
{
gtk_widget_set_sensitive (banl->but_refresh, TRUE);
banlist_sensitize (banl);
}
return TRUE;
}
else return FALSE;
}
static void
banlist_select_changed (GtkWidget *item, banlist_info *banl)
{
GList *list;
if (banl->line_ct == 0)
banl->select_ct = 0;
else
{
list = gtk_tree_selection_get_selected_rows (GTK_TREE_SELECTION (item), NULL);
banl->select_ct = list? 1: 0;
g_list_foreach (list, (GFunc) gtk_tree_path_free, NULL);
g_list_free (list);
}
banlist_sensitize (banl);
} }
/** /**
* * Performs the actual refresh operations. * * Performs the actual refresh operations.
* */ * */
static void static void
banlist_do_refresh (struct session *sess) banlist_do_refresh (banlist_info *banl)
{ {
session *sess = banl->sess;
char tbuf[256]; char tbuf[256];
int i;
char *tbufp;
banlist_sensitize (banl);
if (sess->server->connected) if (sess->server->connected)
{ {
GtkListStore *store; GtkListStore *store;
gtk_widget_set_sensitive (sess->res->banlist_butRefresh, FALSE); g_snprintf (tbuf, sizeof tbuf, DISPLAY_NAME": Ban List (%s, %s)",
snprintf (tbuf, sizeof tbuf, DISPLAY_NAME": Ban List (%s, %s)",
sess->channel, sess->server->servername); sess->channel, sess->server->servername);
mg_set_title (sess->res->banlist_window, tbuf); mg_set_title (banl->window, tbuf);
store = get_store (sess); store = get_store (sess);
gtk_list_store_clear (store); gtk_list_store_clear (store);
banl->line_ct = 0;
handle_command (sess, "ban", FALSE); banl->pending = banl->checked;
if (banl->pending)
if (supports_exempt (sess->server))
{ {
snprintf (tbuf, sizeof (tbuf), "quote mode %s +e", sess->channel); tbufp = tbuf + g_snprintf (tbuf, sizeof tbuf, "quote mode %s +", sess->channel);
for (i = 0; i < MODE_CT; i++)
if (banl->pending & 1<<i)
{
*tbufp++ = modes[i].letter;
}
*tbufp = 0;
handle_command (sess, tbuf, FALSE); handle_command (sess, tbuf, FALSE);
} }
}
} else else
{ {
fe_message (_("Not connected."), FE_MSG_ERROR); fe_message (_("Not connected."), FE_MSG_ERROR);
} }
} }
static void static void
banlist_refresh (GtkWidget * wid, struct session *sess) banlist_refresh (GtkWidget * wid, banlist_info *banl)
{ {
/* JG NOTE: Didn't see actual use of wid here, so just forwarding /* JG NOTE: Didn't see actual use of wid here, so just forwarding
* * this to chanlist_do_refresh because I use it without any widget * * this to chanlist_do_refresh because I use it without any widget
@ -159,123 +396,106 @@ banlist_refresh (GtkWidget * wid, struct session *sess)
* * or apply for the first time if the list has not yet been * * or apply for the first time if the list has not yet been
* * received. * * received.
* */ * */
banlist_do_refresh (sess); banlist_do_refresh (banl);
} }
static int static int
banlist_unban_inner (gpointer none, struct session *sess, int do_exempts) banlist_unban_inner (gpointer none, banlist_info *banl, int mode_num)
{ {
session *sess = banl->sess;
GtkTreeModel *model; GtkTreeModel *model;
GtkTreeSelection *sel; GtkTreeSelection *sel;
GtkTreeIter iter; GtkTreeIter iter;
char tbuf[2048]; char tbuf[2048];
char **masks, *tmp, *space; char **masks, *mask;
int num_sel, i; int num_sel, taglen, i;
/* grab the list of selected items */ /* grab the list of selected items */
model = GTK_TREE_MODEL (get_store (sess)); model = GTK_TREE_MODEL (get_store (sess));
sel = gtk_tree_view_get_selection (get_view (sess)); sel = gtk_tree_view_get_selection (get_view (sess));
num_sel = 0;
if (gtk_tree_model_get_iter_first (model, &iter))
{
do
{
if (gtk_tree_selection_iter_is_selected (sel, &iter))
num_sel++;
}
while (gtk_tree_model_iter_next (model, &iter));
}
if (num_sel < 1) if (!gtk_tree_model_get_iter_first (model, &iter))
return 0; return 0;
/* create an array of all the masks */ taglen = strlen (modes[mode_num].tag);
masks = calloc (1, num_sel * sizeof (char *)); masks = g_malloc (sizeof (char *) * banl->line_ct);
num_sel = 0;
i = 0;
gtk_tree_model_get_iter_first (model, &iter);
do do
{ {
if (gtk_tree_selection_iter_is_selected (sel, &iter)) if (gtk_tree_selection_iter_is_selected (sel, &iter))
{ {
gtk_tree_model_get (model, &iter, MASK_COLUMN, &masks[i], -1); /* Get the mask part of this selected line */
space = strchr (masks[i], ' '); gtk_tree_model_get (model, &iter, MASK_COLUMN, &mask, -1);
if (do_exempts) /* If it's the wrong type of mask, just continue */
{ if (strncmp (modes[mode_num].tag, mask, taglen) != 0)
if (space) continue;
{
/* remove the "(EX) " */ /* Otherwise add it to our array of mask pointers */
tmp = masks[i]; masks[num_sel++] = g_strdup (mask + taglen);
masks[i] = g_strdup (space + 1); g_free (mask);
g_free (tmp);
i++;
}
} else
{
if (!space)
i++;
}
} }
} }
while (gtk_tree_model_iter_next (model, &iter)); while (gtk_tree_model_iter_next (model, &iter));
/* and send to server */ /* and send to server */
if (do_exempts) if (num_sel)
send_channel_modes (sess, tbuf, masks, 0, i, '-', 'e', 0); send_channel_modes (sess, tbuf, masks, 0, num_sel, '-', modes[mode_num].letter, 0);
else
send_channel_modes (sess, tbuf, masks, 0, i, '-', 'b', 0);
/* now free everything, and refresh banlist */ /* now free everything */
for (i=0; i < num_sel; i++) for (i=0; i < num_sel; i++)
g_free (masks[i]); g_free (masks[i]);
free (masks); g_free (masks);
return num_sel; return num_sel;
} }
static void static void
banlist_unban (GtkWidget * wid, struct session *sess) banlist_unban (GtkWidget * wid, banlist_info *banl)
{ {
int num = 0; int i, num = 0;
num += banlist_unban_inner (wid, sess, FALSE); for (i = 0; i < MODE_CT; i++)
num += banlist_unban_inner (wid, sess, TRUE); num += banlist_unban_inner (wid, banl, i);
/* This really should not occur with the redesign */
if (num < 1) if (num < 1)
{ {
fe_message (_("You must select some bans."), FE_MSG_ERROR); fe_message (_("You must select some bans."), FE_MSG_ERROR);
return; return;
} }
banlist_do_refresh (sess); banlist_do_refresh (banl);
} }
static void static void
banlist_clear_cb (GtkDialog *dialog, gint response, gpointer sess) banlist_clear_cb (GtkDialog *dialog, gint response, gpointer data)
{ {
banlist_info *banl = data;
GtkTreeSelection *sel; GtkTreeSelection *sel;
gtk_widget_destroy (GTK_WIDGET (dialog)); gtk_widget_destroy (GTK_WIDGET (dialog));
if (response == GTK_RESPONSE_OK) if (response == GTK_RESPONSE_OK)
{ {
sel = gtk_tree_view_get_selection (get_view (sess)); sel = gtk_tree_view_get_selection (get_view (banl->sess));
gtk_tree_selection_select_all (sel); gtk_tree_selection_select_all (sel);
banlist_unban (NULL, sess); banlist_unban (NULL, banl);
} }
} }
static void static void
banlist_clear (GtkWidget * wid, struct session *sess) banlist_clear (GtkWidget * wid, banlist_info *banl)
{ {
GtkWidget *dialog; GtkWidget *dialog;
dialog = gtk_message_dialog_new (NULL, 0, dialog = gtk_message_dialog_new (NULL, 0,
GTK_MESSAGE_QUESTION, GTK_BUTTONS_OK_CANCEL, GTK_MESSAGE_QUESTION, GTK_BUTTONS_OK_CANCEL,
_("Are you sure you want to remove all bans in %s?"), sess->channel); _("Are you sure you want to remove all listed items in %s?"), banl->sess->channel);
g_signal_connect (G_OBJECT (dialog), "response", g_signal_connect (G_OBJECT (dialog), "response",
G_CALLBACK (banlist_clear_cb), sess); G_CALLBACK (banlist_clear_cb), banl);
gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_MOUSE); gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_MOUSE);
gtk_widget_show (dialog); gtk_widget_show (dialog);
} }
@ -298,8 +518,9 @@ banlist_add_selected_cb (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *it
} }
static void static void
banlist_crop (GtkWidget * wid, struct session *sess) banlist_crop (GtkWidget * wid, banlist_info *banl)
{ {
session *sess = banl->sess;
GtkTreeSelection *select; GtkTreeSelection *select;
GSList *list = NULL, *node; GSList *list = NULL, *node;
int num_sel; int num_sel;
@ -322,22 +543,112 @@ banlist_crop (GtkWidget * wid, struct session *sess)
g_slist_foreach (list, (GFunc)g_free, NULL); g_slist_foreach (list, (GFunc)g_free, NULL);
g_slist_free (list); g_slist_free (list);
banlist_unban (NULL, sess); banlist_unban (NULL, banl);
} else } else
fe_message (_("You must select some bans."), FE_MSG_ERROR); fe_message (_("You must select some bans."), FE_MSG_ERROR);
} }
static void
banlist_toggle (GtkWidget *item, gpointer data)
{
banlist_info *banl = data;
int i, bit = 0;
for (i = 0; i < MODE_CT; i++)
if (banl->checkboxes[i] == item)
{
bit = 1<<i;
break;
}
if (bit) /* Should be gassert() */
{
banl->checked &= ~bit;
banl->checked |= (GTK_TOGGLE_BUTTON (item)->active)? bit: 0;
banlist_do_refresh (banl);
}
}
/* NOTICE: The official strptime() is not available on all platforms so
* I've implemented a special version here. The official version is
* vastly more general than this: it uses locales for weekday and month
* names and its second arg is a format character-string. This special
* version depends on the format returned by ctime(3) whose manpage
* says it returns:
* "a null-terminated string of the form "Wed Jun 30 21:49:08 1993\n"
*
* If the real strpftime() comes available, use this format string:
* #define DATE_FORMAT "%a %b %d %T %Y"
*/
static void
strptime (char *ti, struct tm *tm)
{
/* Expect something like "Sat Mar 16 21:24:27 2013" */
static char *mon[] = { "Jan", "Feb", "Mar", "Apr", "May", "Jun",
"Jul", "Aug", "Sep", "Oct", "Nov", "Dec", NULL };
int M = -1, d = -1, h = -1, m = -1, s = -1, y = -1;
if (*ti == 0)
{
memset (tm, 0, sizeof *tm);
return;
}
/* No need to supply tm->tm_wday; mktime() doesn't read it */
ti += 4;
while ((mon[++M]))
if (strncmp (ti, mon[M], 3) == 0)
break;
ti += 4;
d = strtol (ti, &ti, 10);
h = strtol (++ti, &ti, 10);
m = strtol (++ti, &ti, 10);
s = strtol (++ti, &ti, 10);
y = strtol (++ti, NULL, 10) - 1900;
tm->tm_sec = s;
tm->tm_min = m;
tm->tm_hour = h;
tm->tm_mday = d;
tm->tm_mon = M;
tm->tm_year = y;
}
gint
banlist_date_sort (GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer user_data)
{
struct tm tm1, tm2;
time_t t1, t2;
char *time1, *time2;
gtk_tree_model_get(model, a, 2, &time1, -1);
gtk_tree_model_get(model, b, 2, &time2, -1);
strptime (time1, &tm1);
strptime (time2, &tm2);
t1 = mktime (&tm1);
t2 = mktime (&tm2);
if (t1 < t2) return 1;
if (t1 == t2) return 0;
return -1;
}
static GtkWidget * static GtkWidget *
banlist_treeview_new (GtkWidget *box) banlist_treeview_new (GtkWidget *box, banlist_info *banl)
{ {
GtkListStore *store; GtkListStore *store;
GtkWidget *view; GtkWidget *view;
GtkTreeSelection *select; GtkTreeSelection *select;
GtkTreeViewColumn *col; GtkTreeViewColumn *col;
GtkTreeSortable *sortable;
store = gtk_list_store_new (N_COLUMNS, G_TYPE_STRING, G_TYPE_STRING, store = gtk_list_store_new (N_COLUMNS, G_TYPE_STRING, G_TYPE_STRING,
G_TYPE_STRING); G_TYPE_STRING);
g_return_val_if_fail (store != NULL, NULL); g_return_val_if_fail (store != NULL, NULL);
sortable = GTK_TREE_SORTABLE (store);
gtk_tree_sortable_set_sort_func (sortable, 2, banlist_date_sort, GINT_TO_POINTER (2), NULL);
view = gtkutil_treeview_new (box, GTK_TREE_MODEL (store), NULL, view = gtkutil_treeview_new (box, GTK_TREE_MODEL (store), NULL,
MASK_COLUMN, _("Mask"), MASK_COLUMN, _("Mask"),
FROM_COLUMN, _("From"), FROM_COLUMN, _("From"),
@ -345,17 +656,25 @@ banlist_treeview_new (GtkWidget *box)
col = gtk_tree_view_get_column (GTK_TREE_VIEW (view), MASK_COLUMN); col = gtk_tree_view_get_column (GTK_TREE_VIEW (view), MASK_COLUMN);
gtk_tree_view_column_set_alignment (col, 0.5); gtk_tree_view_column_set_alignment (col, 0.5);
gtk_tree_view_column_set_min_width (col, 300); gtk_tree_view_column_set_min_width (col, 100);
gtk_tree_view_column_set_sort_column_id (col, MASK_COLUMN); gtk_tree_view_column_set_sort_column_id (col, MASK_COLUMN);
gtk_tree_view_column_set_sizing (col, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
gtk_tree_view_column_set_resizable (col, TRUE);
col = gtk_tree_view_get_column (GTK_TREE_VIEW (view), FROM_COLUMN); col = gtk_tree_view_get_column (GTK_TREE_VIEW (view), FROM_COLUMN);
gtk_tree_view_column_set_alignment (col, 0.5); gtk_tree_view_column_set_alignment (col, 0.5);
gtk_tree_view_column_set_sort_column_id (col, FROM_COLUMN); gtk_tree_view_column_set_sort_column_id (col, FROM_COLUMN);
gtk_tree_view_column_set_sizing (col, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
gtk_tree_view_column_set_resizable (col, TRUE);
col = gtk_tree_view_get_column (GTK_TREE_VIEW (view), DATE_COLUMN); col = gtk_tree_view_get_column (GTK_TREE_VIEW (view), DATE_COLUMN);
gtk_tree_view_column_set_alignment (col, 0.5); gtk_tree_view_column_set_alignment (col, 0.5);
gtk_tree_view_column_set_sort_column_id (col, DATE_COLUMN);
gtk_tree_view_column_set_sizing (col, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
gtk_tree_view_column_set_resizable (col, TRUE);
select = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); select = gtk_tree_view_get_selection (GTK_TREE_VIEW (view));
g_signal_connect (G_OBJECT (select), "changed", G_CALLBACK (banlist_select_changed), banl);
gtk_tree_selection_set_mode (select, GTK_SELECTION_MULTIPLE); gtk_tree_selection_set_mode (select, GTK_SELECTION_MULTIPLE);
gtk_widget_show (view); gtk_widget_show (view);
@ -363,56 +682,99 @@ banlist_treeview_new (GtkWidget *box)
} }
static void static void
banlist_closegui (GtkWidget *wid, session *sess) banlist_closegui (GtkWidget *wid, banlist_info *banl)
{ {
if (is_session (sess)) session *sess = banl->sess;
sess->res->banlist_window = 0;
if (sess->res->banlist == banl)
{
g_free (banl);
sess->res->banlist = NULL;
}
} }
void void
banlist_opengui (struct session *sess) banlist_opengui (struct session *sess)
{ {
GtkWidget *vbox1; banlist_info *banl;
GtkWidget *bbox; int i;
GtkWidget *table, *vbox, *bbox;
char tbuf[256]; char tbuf[256];
if (sess->res->banlist_window)
{
mg_bring_tofront (sess->res->banlist_window);
return;
}
if (sess->type != SESS_CHANNEL) if (sess->type != SESS_CHANNEL)
{ {
fe_message (_("You can only open the Ban List window while in a channel tab."), FE_MSG_ERROR); fe_message (_("You can only open the Ban List window while in a channel tab."), FE_MSG_ERROR);
return; return;
} }
snprintf (tbuf, sizeof tbuf, _(DISPLAY_NAME": Ban List (%s)"), if (!sess->res->banlist)
{
sess->res->banlist = g_malloc0 (sizeof (banlist_info));
if (!sess->res->banlist)
{
fe_message (_("Banlist initialization failed."), FE_MSG_ERROR);
return;
}
}
banl = sess->res->banlist;
if (banl->window)
{
mg_bring_tofront (banl->window);
return;
}
/* New banlist for this session -- Initialize it */
banl->sess = sess;
/* For each mode set its bit in capable/readable/writeable */
for (i = 0; i < MODE_CT; i++)
modes[i].tester (banl, i);
/* Force on the checkmark in the "Bans" box */
banl->checked = 1<<MODE_BAN;
g_snprintf (tbuf, sizeof tbuf, _(DISPLAY_NAME": Ban List (%s)"),
sess->server->servername); sess->server->servername);
sess->res->banlist_window = mg_create_generic_tab ("BanList", tbuf, FALSE, banl->window = mg_create_generic_tab ("BanList", tbuf, FALSE,
TRUE, banlist_closegui, sess, 550, 200, &vbox1, sess->server); TRUE, banlist_closegui, banl, 550, 200, &vbox, sess->server);
gtkutil_destroy_on_esc (banl->window);
gtk_container_set_border_width (GTK_CONTAINER (banl->window), 3);
gtk_box_set_spacing (GTK_BOX (vbox), 3);
/* create banlist view */ /* create banlist view */
sess->res->banlist_treeview = banlist_treeview_new (vbox1); banl->treeview = banlist_treeview_new (vbox, banl);
table = gtk_table_new (1, 3, FALSE);
gtk_table_set_col_spacings (GTK_TABLE (table), 16);
gtk_box_pack_start (GTK_BOX (vbox), table, 0, 0, 0);
for (i = 0; i < MODE_CT; i++)
{
if (!(banl->capable & 1<<i))
continue;
banl->checkboxes[i] = gtk_check_button_new_with_label (_(modes[i].name));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (banl->checkboxes[i]), (banl->checked & 1<<i? TRUE: FALSE));
g_signal_connect (G_OBJECT (banl->checkboxes[i]), "toggled",
G_CALLBACK (banlist_toggle), banl);
gtk_table_attach (GTK_TABLE (table), banl->checkboxes[i], i+1, i+2, 0, 1, GTK_FILL, GTK_FILL, 0, 0);
}
bbox = gtk_hbutton_box_new (); bbox = gtk_hbutton_box_new ();
gtk_button_box_set_layout (GTK_BUTTON_BOX (bbox), GTK_BUTTONBOX_SPREAD); gtk_button_box_set_layout (GTK_BUTTON_BOX (bbox), GTK_BUTTONBOX_SPREAD);
gtk_container_set_border_width (GTK_CONTAINER (bbox), 5); gtk_container_set_border_width (GTK_CONTAINER (bbox), 5);
gtk_box_pack_end (GTK_BOX (vbox1), bbox, 0, 0, 0); gtk_box_pack_end (GTK_BOX (vbox), bbox, 0, 0, 0);
gtk_widget_show (bbox); gtk_widget_show (bbox);
gtkutil_button (bbox, GTK_STOCK_REMOVE, 0, banlist_unban, sess, banl->but_remove = gtkutil_button (bbox, GTK_STOCK_REMOVE, 0, banlist_unban, banl,
_("Remove")); _("Remove"));
gtkutil_button (bbox, GTK_STOCK_REMOVE, 0, banlist_crop, sess, banl->but_crop = gtkutil_button (bbox, GTK_STOCK_REMOVE, 0, banlist_crop, banl,
_("Crop")); _("Crop"));
gtkutil_button (bbox, GTK_STOCK_CLEAR, 0, banlist_clear, sess, banl->but_clear = gtkutil_button (bbox, GTK_STOCK_CLEAR, 0, banlist_clear, banl,
_("Clear")); _("Clear"));
sess->res->banlist_butRefresh = gtkutil_button (bbox, GTK_STOCK_REFRESH, 0, banlist_refresh, sess, _("Refresh")); banl->but_refresh = gtkutil_button (bbox, GTK_STOCK_REFRESH, 0, banlist_refresh, banl, _("Refresh"));
banlist_do_refresh (sess); banlist_do_refresh (banl);
gtk_widget_show (sess->res->banlist_window); gtk_widget_show_all (banl->window);
} }

View File

@ -1 +1,60 @@
#ifndef BANLIST_H
#define BANLIST_H
#include "../common/hexchat.h"
void banlist_opengui (session *sess); void banlist_opengui (session *sess);
#ifndef RPL_BANLIST
/* Where's that darn header file, that would have all these defines ? */
#define RPL_BANLIST 367
#define RPL_ENDOFBANLIST 368
#define RPL_INVITELIST 346
#define RPL_ENDOFINVITELIST 347
#define RPL_EXCEPTLIST 348
#define RPL_ENDOFEXCEPTLIST 349
#define RPL_QUIETLIST 728
#define RPL_ENDOFQUIETLIST 729
#endif
typedef enum banlist_modes_e {
MODE_BAN,
MODE_EXEMPT,
MODE_INVITE,
MODE_QUIET,
MODE_CT
} banlist_modes;
typedef struct banlist_info_s banlist_info;
typedef struct mode_info_s {
char *name; /* Checkbox name, e.g. "Bans" */
char *tag; /* Prefix line with this, e.g. "(EX) " for MODE_EXEMPT */
char letter; /* /mode-command letter, e.g. 'b' for MODE_BAN */
int code; /* rfc RPL_foo code, e.g. 367 for RPL_BANLIST */
int endcode; /* rfc RPL_ENDOFfoo code, e.g. 368 for RPL_ENDOFBANLIST */
int bit; /* Mask bit, e.g., 1<<MODE_BAN */
void (*tester)(banlist_info *, int); /* Function returns true to set bit into checkable */
} mode_info;
typedef struct banlist_info_s {
session *sess;
int capable; /* MODE bitmask */
int readable; /* subset of capable if not op */
int writeable; /* subset of capable if op */
int checked; /* subset of (op? writeable: readable) */
int pending; /* subset of checked */
int current; /* index of currently processing mode */
int line_ct; /* count of presented lines */
int select_ct; /* count of selected lines */
/* Not really; 1 if any are selected otherwise 0 */
GtkWidget *window;
GtkWidget *treeview;
GtkWidget *radios[MODE_CT];
GtkWidget *checkboxes[MODE_CT];
GtkWidget *but_remove;
GtkWidget *but_crop;
GtkWidget *but_clear;
GtkWidget *but_refresh;
GtkWidget *checkbox[MODE_CT]; /* Checkbox widget for mode */
} banlist_info;
#endif /* BANLIST_H */

View File

@ -604,14 +604,6 @@ fe_is_chanwindow (struct server *serv)
return 1; return 1;
} }
int
fe_is_banwindow (struct session *sess)
{
if (!sess->res->banlist_window)
return 0;
return 1;
}
void void
fe_notify_update (char *name) fe_notify_update (char *name)
{ {

View File

@ -29,6 +29,7 @@
#include <gtk/gtkwidget.h> #include <gtk/gtkwidget.h>
#include <gtk/gtkcontainer.h> #include <gtk/gtkcontainer.h>
#include <gtk/gtksignal.h> #include <gtk/gtksignal.h>
#include "banlist.h"
#undef gtk_signal_connect #undef gtk_signal_connect
#define gtk_signal_connect g_signal_connect #define gtk_signal_connect g_signal_connect
@ -97,10 +98,7 @@ struct server_gui
typedef struct restore_gui typedef struct restore_gui
{ {
/* banlist stuff */ banlist_info *banlist;
GtkWidget *banlist_window;
GtkWidget *banlist_treeview;
GtkWidget *banlist_butRefresh;
void *tab; /* (chan *) */ void *tab; /* (chan *) */

View File

@ -3530,8 +3530,8 @@ fe_server_callback (server *serv)
void void
fe_session_callback (session *sess) fe_session_callback (session *sess)
{ {
if (sess->res->banlist_window) if (sess->res->banlist && sess->res->banlist->window)
mg_close_gen (NULL, sess->res->banlist_window); mg_close_gen (NULL, sess->res->banlist->window);
if (sess->res->input_text) if (sess->res->input_text)
free (sess->res->input_text); free (sess->res->input_text);

View File

@ -651,17 +651,12 @@ void
fe_chan_list_end (struct server *serv) fe_chan_list_end (struct server *serv)
{ {
} }
int gboolean
fe_is_banwindow (struct session *sess) fe_add_ban_list (struct session *sess, char *mask, char *who, char *when, int rplcode)
{
return 0;
}
void
fe_add_ban_list (struct session *sess, char *mask, char *who, char *when, int is_exemption)
{ {
} }
void gboolean
fe_ban_list_end (struct session *sess, int is_exemption) fe_ban_list_end (struct session *sess, int rplcode)
{ {
} }
void void