419 lines
10 KiB
C
419 lines
10 KiB
C
/* X-Chat
|
|
* Copyright (C) 1998 Peter Zelezny.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <fcntl.h>
|
|
#include <time.h>
|
|
|
|
#ifndef WIN32
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#include "fe-gtk.h"
|
|
|
|
#include <gtk/gtkhbox.h>
|
|
#include <gtk/gtkvbox.h>
|
|
#include <gtk/gtkhbbox.h>
|
|
#include <gtk/gtkstock.h>
|
|
#include <gtk/gtkliststore.h>
|
|
#include <gtk/gtkmessagedialog.h>
|
|
#include <gtk/gtktreeview.h>
|
|
#include <gtk/gtktreeselection.h>
|
|
|
|
#include "../common/hexchat.h"
|
|
#include "../common/fe.h"
|
|
#include "../common/modes.h"
|
|
#include "../common/outbound.h"
|
|
#include "../common/hexchatc.h"
|
|
#include "gtkutil.h"
|
|
#include "maingui.h"
|
|
#include "banlist.h"
|
|
|
|
/* model for the banlist tree */
|
|
enum
|
|
{
|
|
MASK_COLUMN,
|
|
FROM_COLUMN,
|
|
DATE_COLUMN,
|
|
N_COLUMNS
|
|
};
|
|
|
|
static GtkTreeView *
|
|
get_view (struct session *sess)
|
|
{
|
|
return GTK_TREE_VIEW (sess->res->banlist_treeview);
|
|
}
|
|
|
|
static GtkListStore *
|
|
get_store (struct session *sess)
|
|
{
|
|
return GTK_LIST_STORE (gtk_tree_view_get_model (get_view (sess)));
|
|
}
|
|
|
|
static gboolean
|
|
supports_exempt (server *serv)
|
|
{
|
|
char *cm = serv->chanmodes;
|
|
|
|
if (serv->have_except)
|
|
return TRUE;
|
|
|
|
if (!cm)
|
|
return FALSE;
|
|
|
|
while (*cm)
|
|
{
|
|
if (*cm == ',')
|
|
break;
|
|
if (*cm == 'e')
|
|
return TRUE;
|
|
cm++;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
fe_add_ban_list (struct session *sess, char *mask, char *who, char *when, int is_exempt)
|
|
{
|
|
GtkListStore *store;
|
|
GtkTreeIter iter;
|
|
char buf[512];
|
|
|
|
store = get_store (sess);
|
|
gtk_list_store_append (store, &iter);
|
|
|
|
if (is_exempt)
|
|
{
|
|
snprintf (buf, sizeof (buf), "(EX) %s", mask);
|
|
gtk_list_store_set (store, &iter, 0, buf, 1, who, 2, when, -1);
|
|
} else
|
|
{
|
|
gtk_list_store_set (store, &iter, 0, mask, 1, who, 2, when, -1);
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_ban_list_end (struct session *sess, int is_exemption)
|
|
{
|
|
gtk_widget_set_sensitive (sess->res->banlist_butRefresh, TRUE);
|
|
}
|
|
|
|
/**
|
|
* * Performs the actual refresh operations.
|
|
* */
|
|
static void
|
|
banlist_do_refresh (struct session *sess)
|
|
{
|
|
char tbuf[256];
|
|
if (sess->server->connected)
|
|
{
|
|
GtkListStore *store;
|
|
|
|
gtk_widget_set_sensitive (sess->res->banlist_butRefresh, FALSE);
|
|
|
|
snprintf (tbuf, sizeof tbuf, DISPLAY_NAME": Ban List (%s, %s)",
|
|
sess->channel, sess->server->servername);
|
|
mg_set_title (sess->res->banlist_window, tbuf);
|
|
|
|
store = get_store (sess);
|
|
gtk_list_store_clear (store);
|
|
|
|
handle_command (sess, "ban", FALSE);
|
|
|
|
if (supports_exempt (sess->server))
|
|
{
|
|
snprintf (tbuf, sizeof (tbuf), "quote mode %s +e", sess->channel);
|
|
handle_command (sess, tbuf, FALSE);
|
|
}
|
|
|
|
} else
|
|
{
|
|
fe_message (_("Not connected."), FE_MSG_ERROR);
|
|
}
|
|
}
|
|
|
|
static void
|
|
banlist_refresh (GtkWidget * wid, struct session *sess)
|
|
{
|
|
/* 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
|
|
* * param in chanlist_build_gui_list when the user presses enter
|
|
* * or apply for the first time if the list has not yet been
|
|
* * received.
|
|
* */
|
|
banlist_do_refresh (sess);
|
|
}
|
|
|
|
static int
|
|
banlist_unban_inner (gpointer none, struct session *sess, int do_exempts)
|
|
{
|
|
GtkTreeModel *model;
|
|
GtkTreeSelection *sel;
|
|
GtkTreeIter iter;
|
|
char tbuf[2048];
|
|
char **masks, *tmp, *space;
|
|
int num_sel, i;
|
|
|
|
/* grab the list of selected items */
|
|
model = GTK_TREE_MODEL (get_store (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)
|
|
return 0;
|
|
|
|
/* create an array of all the masks */
|
|
masks = calloc (1, num_sel * sizeof (char *));
|
|
|
|
i = 0;
|
|
gtk_tree_model_get_iter_first (model, &iter);
|
|
do
|
|
{
|
|
if (gtk_tree_selection_iter_is_selected (sel, &iter))
|
|
{
|
|
gtk_tree_model_get (model, &iter, MASK_COLUMN, &masks[i], -1);
|
|
space = strchr (masks[i], ' ');
|
|
|
|
if (do_exempts)
|
|
{
|
|
if (space)
|
|
{
|
|
/* remove the "(EX) " */
|
|
tmp = masks[i];
|
|
masks[i] = g_strdup (space + 1);
|
|
g_free (tmp);
|
|
i++;
|
|
}
|
|
} else
|
|
{
|
|
if (!space)
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
while (gtk_tree_model_iter_next (model, &iter));
|
|
|
|
/* and send to server */
|
|
if (do_exempts)
|
|
send_channel_modes (sess, tbuf, masks, 0, i, '-', 'e', 0);
|
|
else
|
|
send_channel_modes (sess, tbuf, masks, 0, i, '-', 'b', 0);
|
|
|
|
/* now free everything, and refresh banlist */
|
|
for (i=0; i < num_sel; i++)
|
|
g_free (masks[i]);
|
|
free (masks);
|
|
|
|
return num_sel;
|
|
}
|
|
|
|
static void
|
|
banlist_unban (GtkWidget * wid, struct session *sess)
|
|
{
|
|
int num = 0;
|
|
|
|
num += banlist_unban_inner (wid, sess, FALSE);
|
|
num += banlist_unban_inner (wid, sess, TRUE);
|
|
|
|
if (num < 1)
|
|
{
|
|
fe_message (_("You must select some bans."), FE_MSG_ERROR);
|
|
return;
|
|
}
|
|
|
|
banlist_do_refresh (sess);
|
|
}
|
|
|
|
static void
|
|
banlist_clear_cb (GtkDialog *dialog, gint response, gpointer sess)
|
|
{
|
|
GtkTreeSelection *sel;
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (dialog));
|
|
|
|
if (response == GTK_RESPONSE_OK)
|
|
{
|
|
sel = gtk_tree_view_get_selection (get_view (sess));
|
|
gtk_tree_selection_select_all (sel);
|
|
banlist_unban (NULL, sess);
|
|
}
|
|
}
|
|
|
|
static void
|
|
banlist_clear (GtkWidget * wid, struct session *sess)
|
|
{
|
|
GtkWidget *dialog;
|
|
|
|
dialog = gtk_message_dialog_new (NULL, 0,
|
|
GTK_MESSAGE_QUESTION, GTK_BUTTONS_OK_CANCEL,
|
|
_("Are you sure you want to remove all bans in %s?"), sess->channel);
|
|
g_signal_connect (G_OBJECT (dialog), "response",
|
|
G_CALLBACK (banlist_clear_cb), sess);
|
|
gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_MOUSE);
|
|
gtk_widget_show (dialog);
|
|
}
|
|
|
|
static void
|
|
banlist_add_selected_cb (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
|
|
{
|
|
GSList **lp = data;
|
|
GSList *list = NULL;
|
|
GtkTreeIter *copy;
|
|
|
|
if (!lp) return;
|
|
list = *lp;
|
|
copy = g_malloc (sizeof (GtkTreeIter));
|
|
g_return_if_fail (copy != NULL);
|
|
*copy = *iter;
|
|
|
|
list = g_slist_append (list, copy);
|
|
*(GSList **)data = list;
|
|
}
|
|
|
|
static void
|
|
banlist_crop (GtkWidget * wid, struct session *sess)
|
|
{
|
|
GtkTreeSelection *select;
|
|
GSList *list = NULL, *node;
|
|
int num_sel;
|
|
|
|
/* remember which bans are selected */
|
|
select = gtk_tree_view_get_selection (get_view (sess));
|
|
/* gtk_tree_selected_get_selected_rows() isn't present in gtk 2.0.x */
|
|
gtk_tree_selection_selected_foreach (select, banlist_add_selected_cb,
|
|
&list);
|
|
|
|
num_sel = g_slist_length (list);
|
|
/* select all, then unselect those that we remembered */
|
|
if (num_sel)
|
|
{
|
|
gtk_tree_selection_select_all (select);
|
|
|
|
for (node = list; node; node = node->next)
|
|
gtk_tree_selection_unselect_iter (select, node->data);
|
|
|
|
g_slist_foreach (list, (GFunc)g_free, NULL);
|
|
g_slist_free (list);
|
|
|
|
banlist_unban (NULL, sess);
|
|
} else
|
|
fe_message (_("You must select some bans."), FE_MSG_ERROR);
|
|
}
|
|
|
|
static GtkWidget *
|
|
banlist_treeview_new (GtkWidget *box)
|
|
{
|
|
GtkListStore *store;
|
|
GtkWidget *view;
|
|
GtkTreeSelection *select;
|
|
GtkTreeViewColumn *col;
|
|
|
|
store = gtk_list_store_new (N_COLUMNS, G_TYPE_STRING, G_TYPE_STRING,
|
|
G_TYPE_STRING);
|
|
g_return_val_if_fail (store != NULL, NULL);
|
|
view = gtkutil_treeview_new (box, GTK_TREE_MODEL (store), NULL,
|
|
MASK_COLUMN, _("Mask"),
|
|
FROM_COLUMN, _("From"),
|
|
DATE_COLUMN, _("Date"), -1);
|
|
|
|
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_min_width (col, 300);
|
|
gtk_tree_view_column_set_sort_column_id (col, MASK_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_sort_column_id (col, FROM_COLUMN);
|
|
|
|
col = gtk_tree_view_get_column (GTK_TREE_VIEW (view), DATE_COLUMN);
|
|
gtk_tree_view_column_set_alignment (col, 0.5);
|
|
|
|
select = gtk_tree_view_get_selection (GTK_TREE_VIEW (view));
|
|
gtk_tree_selection_set_mode (select, GTK_SELECTION_MULTIPLE);
|
|
|
|
gtk_widget_show (view);
|
|
return view;
|
|
}
|
|
|
|
static void
|
|
banlist_closegui (GtkWidget *wid, session *sess)
|
|
{
|
|
if (is_session (sess))
|
|
sess->res->banlist_window = 0;
|
|
}
|
|
|
|
void
|
|
banlist_opengui (struct session *sess)
|
|
{
|
|
GtkWidget *vbox1;
|
|
GtkWidget *bbox;
|
|
char tbuf[256];
|
|
|
|
if (sess->res->banlist_window)
|
|
{
|
|
mg_bring_tofront (sess->res->banlist_window);
|
|
return;
|
|
}
|
|
|
|
if (sess->type != SESS_CHANNEL)
|
|
{
|
|
fe_message (_("You can only open the Ban List window while in a channel tab."), FE_MSG_ERROR);
|
|
return;
|
|
}
|
|
|
|
snprintf (tbuf, sizeof tbuf, _(DISPLAY_NAME": Ban List (%s)"),
|
|
sess->server->servername);
|
|
|
|
sess->res->banlist_window = mg_create_generic_tab ("BanList", tbuf, FALSE,
|
|
TRUE, banlist_closegui, sess, 550, 200, &vbox1, sess->server);
|
|
|
|
/* create banlist view */
|
|
sess->res->banlist_treeview = banlist_treeview_new (vbox1);
|
|
|
|
bbox = gtk_hbutton_box_new ();
|
|
gtk_button_box_set_layout (GTK_BUTTON_BOX (bbox), GTK_BUTTONBOX_SPREAD);
|
|
gtk_container_set_border_width (GTK_CONTAINER (bbox), 5);
|
|
gtk_box_pack_end (GTK_BOX (vbox1), bbox, 0, 0, 0);
|
|
gtk_widget_show (bbox);
|
|
|
|
gtkutil_button (bbox, GTK_STOCK_REMOVE, 0, banlist_unban, sess,
|
|
_("Remove"));
|
|
gtkutil_button (bbox, GTK_STOCK_REMOVE, 0, banlist_crop, sess,
|
|
_("Crop"));
|
|
gtkutil_button (bbox, GTK_STOCK_CLEAR, 0, banlist_clear, sess,
|
|
_("Clear"));
|
|
|
|
sess->res->banlist_butRefresh = gtkutil_button (bbox, GTK_STOCK_REFRESH, 0, banlist_refresh, sess, _("Refresh"));
|
|
|
|
banlist_do_refresh (sess);
|
|
|
|
gtk_widget_show (sess->res->banlist_window);
|
|
}
|