transmission/gtk/main.c

969 lines
31 KiB
C
Raw Normal View History

2006-01-12 18:32:29 +00:00
/*
Copyright (c) 2005 Joshua Elsasser. All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
POSSIBILITY OF SUCH DAMAGE.
*/
2006-01-12 18:40:47 +00:00
#include <sys/param.h>
2006-01-12 18:32:29 +00:00
#include <assert.h>
2006-01-12 18:40:47 +00:00
#include <errno.h>
2006-01-12 18:32:29 +00:00
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <unistd.h>
#include <gtk/gtk.h>
#include "conf.h"
2006-01-12 18:47:30 +00:00
#include "gtkcellrenderertorrent.h"
2006-01-12 18:40:47 +00:00
#include "prefs.h"
2006-01-12 18:32:29 +00:00
#include "transmission.h"
#include "util.h"
2006-01-12 18:47:30 +00:00
#define TRACKER_EXIT_TIMEOUT 5
2006-01-12 18:32:29 +00:00
struct cbdata {
tr_handle_t *tr;
2006-01-12 18:40:47 +00:00
GtkWindow *wind;
2006-01-12 18:32:29 +00:00
GtkListStore *model;
GtkTreeView *view;
2006-01-12 18:47:30 +00:00
GtkStatusbar *bar;
GtkWidget **buttons;
2006-01-12 18:32:29 +00:00
guint timer;
};
struct exitdata {
struct cbdata *cbdata;
time_t started;
guint timer;
};
struct pieces {
char p[120];
};
void
maketypes(void);
gpointer
tr_pieces_copy(gpointer);
void
tr_pieces_free(gpointer);
void
makewind(GtkWidget *wind, tr_handle_t *tr, GList *saved);
gboolean
winclose(GtkWidget *widget, GdkEvent *event, gpointer gdata);
gboolean
exitcheck(gpointer gdata);
GtkWidget *
makewind_toolbar(struct cbdata *data);
GtkWidget *
makewind_list(struct cbdata *data);
2006-01-12 18:56:00 +00:00
static void
stylekludge(GObject *obj, GParamSpec *spec, gpointer gdata);
2006-01-12 18:47:30 +00:00
void
fixbuttons(GtkTreeSelection *sel, gpointer gdata);
void
dfname(GtkTreeViewColumn *col, GtkCellRenderer *rend, GtkTreeModel *model,
GtkTreeIter *iter, gpointer gdata);
void
dfprog(GtkTreeViewColumn *col, GtkCellRenderer *rend, GtkTreeModel *model,
GtkTreeIter *iter, gpointer gdata);
2006-01-12 18:32:29 +00:00
gboolean
updatemodel(gpointer gdata);
gboolean
listclick(GtkWidget *widget, GdkEventButton *event, gpointer gdata);
gboolean
2006-01-12 18:47:30 +00:00
listpopup(GtkWidget *widget, gpointer gdata);
2006-01-12 18:32:29 +00:00
void
2006-01-12 18:47:30 +00:00
dopopupmenu(GdkEventButton *event, struct cbdata *data, GtkTreeIter *iter);
2006-01-12 18:32:29 +00:00
void
actionclick(GtkWidget *widget, gpointer gdata);
gboolean
2006-01-12 18:40:47 +00:00
addtorrent(tr_handle_t *tr, GtkWindow *parentwind, const char *torrent,
2006-01-12 18:32:29 +00:00
const char *dir, gboolean paused);
void
2006-01-12 18:55:27 +00:00
addedtorrents(void *vdata);
2006-01-12 18:32:29 +00:00
const char *
statusstr(int status);
void
makeinfowind(struct cbdata *data, int index);
gboolean
2006-01-12 18:40:47 +00:00
savetorrents(tr_handle_t *tr, GtkWindow *wind, int count, tr_stat_t *stat);
2006-01-12 18:32:29 +00:00
#define TR_TYPE_PIECES_NAME "tr-type-pieces"
#define TR_TYPE_PIECES ((const GType)tr_type_pieces)
#define TR_PIECES(ptr) ((struct pieces*)ptr)
GType tr_type_pieces;
#define LIST_ACTION "torrent-list-action"
2006-01-12 18:40:47 +00:00
enum listact { ACT_OPEN, ACT_START, ACT_STOP, ACT_DELETE, ACT_INFO, ACT_PREF };
2006-01-12 18:32:29 +00:00
#define LIST_ACTION_FROM "torrent-list-action-from"
enum listfrom { FROM_BUTTON, FROM_POPUP };
#define LIST_INDEX "torrent-list-index"
2006-01-12 18:47:30 +00:00
struct { const gchar *name; const gchar *id; enum listact act; gboolean nomenu;
int avail; const char *ttext; const char *tpriv; }
2006-01-12 18:32:29 +00:00
actionitems[] = {
2006-01-12 18:47:30 +00:00
{"Add", GTK_STOCK_ADD, ACT_OPEN, FALSE, 0,
2006-01-12 18:32:29 +00:00
"Add a new torrent file", "XXX"},
2006-01-12 18:47:30 +00:00
{"Resume", GTK_STOCK_MEDIA_PLAY, ACT_START, FALSE,
(TR_STATUS_STOPPING | TR_STATUS_PAUSE),
2006-01-12 18:32:29 +00:00
"Resume a torrent that has been paused", "XXX"},
2006-01-12 18:47:30 +00:00
{"Pause", GTK_STOCK_MEDIA_PAUSE, ACT_STOP, FALSE,
~(TR_STATUS_STOPPING | TR_STATUS_PAUSE),
2006-01-12 18:32:29 +00:00
"Pause a torrent", "XXX"},
2006-01-12 18:47:30 +00:00
{"Remove", GTK_STOCK_REMOVE, ACT_DELETE, FALSE, ~0,
2006-01-12 18:32:29 +00:00
"Remove a torrent from the list", "XXX"},
2006-01-12 18:47:30 +00:00
{"Properties", GTK_STOCK_PROPERTIES, ACT_INFO, FALSE, ~0,
2006-01-12 18:32:29 +00:00
"Get additional information for a torrent", "XXX"},
2006-01-12 18:47:30 +00:00
{"Preferences", GTK_STOCK_PREFERENCES, ACT_PREF, TRUE, 0,
2006-01-12 18:40:47 +00:00
"Open preferences dialog", "XXX"},
2006-01-12 18:32:29 +00:00
};
#define CBDATA_PTR "callback-data-pointer"
int
main(int argc, char **argv) {
GtkWidget *mainwind, *preferr, *stateerr;
char *err;
tr_handle_t *tr;
GList *saved;
2006-01-12 18:40:47 +00:00
const char *pref;
long intval;
2006-01-12 18:32:29 +00:00
gtk_init(&argc, &argv);
tr = tr_init();
2006-01-12 18:58:57 +00:00
if(cf_init(tr_getPrefsDirectory(), &err)) {
2006-01-12 18:32:29 +00:00
if(cf_lock(&err)) {
/* create main window now so any error dialogs can be it's children */
mainwind = gtk_window_new(GTK_WINDOW_TOPLEVEL);
preferr = NULL;
stateerr = NULL;
if(!cf_loadprefs(&err)) {
2006-01-12 18:40:47 +00:00
preferr = errmsg(GTK_WINDOW(mainwind), "%s", err);
2006-01-12 18:32:29 +00:00
g_free(err);
}
saved = cf_loadstate(&err);
if(NULL != err) {
2006-01-12 18:40:47 +00:00
stateerr = errmsg(GTK_WINDOW(mainwind), "%s", err);
2006-01-12 18:32:29 +00:00
g_free(err);
}
2006-01-12 18:40:47 +00:00
/* set the upload limit */
setlimit(tr);
2006-01-12 18:32:29 +00:00
2006-01-12 18:40:47 +00:00
/* set the listening port */
if(NULL != (pref = cf_getpref(PREF_PORT)) &&
0 < (intval = strtol(pref, NULL, 10)) && 0xffff >= intval)
tr_setBindPort(tr, intval);
2006-01-12 18:32:29 +00:00
maketypes();
makewind(mainwind, tr, saved);
if(NULL != preferr)
gtk_widget_show_all(preferr);
if(NULL != stateerr)
gtk_widget_show_all(stateerr);
} else {
2006-01-12 18:40:47 +00:00
gtk_widget_show(errmsg_full(NULL, (errfunc_t)gtk_main_quit,
NULL, "%s", err));
2006-01-12 18:32:29 +00:00
g_free(err);
}
} else {
2006-01-12 18:40:47 +00:00
gtk_widget_show(errmsg_full(NULL, (errfunc_t)gtk_main_quit,
NULL, "%s", err));
2006-01-12 18:32:29 +00:00
g_free(err);
}
gtk_main();
return 0;
}
void
maketypes(void) {
tr_type_pieces = g_boxed_type_register_static(
TR_TYPE_PIECES_NAME, tr_pieces_copy, tr_pieces_free);
}
gpointer
tr_pieces_copy(gpointer data) {
return g_memdup(data, sizeof(struct pieces));
}
void
tr_pieces_free(gpointer data) {
g_free(data);
}
void
makewind(GtkWidget *wind, tr_handle_t *tr, GList *saved) {
GtkWidget *vbox = gtk_vbox_new(FALSE, 0);
GtkWidget *scroll = gtk_scrolled_window_new(NULL, NULL);
2006-01-12 18:47:30 +00:00
GtkWidget *status = gtk_statusbar_new();
2006-01-12 18:32:29 +00:00
struct cbdata *data = g_new0(struct cbdata, 1);
GtkWidget *list;
GtkRequisition req;
GList *ii;
struct cf_torrentstate *ts;
2006-01-12 18:48:53 +00:00
gint height;
2006-01-12 18:32:29 +00:00
data->tr = tr;
2006-01-12 18:40:47 +00:00
data->wind = GTK_WINDOW(wind);
2006-01-12 18:32:29 +00:00
data->timer = -1;
/* filled in by makewind_list */
data->model = NULL;
data->view = NULL;
2006-01-12 18:47:30 +00:00
data->bar = GTK_STATUSBAR(status);
data->buttons = NULL;
2006-01-12 18:32:29 +00:00
2006-01-12 18:48:53 +00:00
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll), GTK_POLICY_NEVER,
GTK_POLICY_AUTOMATIC);
2006-01-12 18:32:29 +00:00
gtk_box_pack_start(GTK_BOX(vbox), makewind_toolbar(data), FALSE, FALSE, 0);
list = makewind_list(data);
gtk_container_add(GTK_CONTAINER(scroll), list);
2006-01-12 18:47:30 +00:00
gtk_box_pack_start(GTK_BOX(vbox), scroll, TRUE, TRUE, 0);
gtk_statusbar_push(GTK_STATUSBAR(status), 0, "");
gtk_box_pack_start(GTK_BOX(vbox), status, FALSE, FALSE, 0);
2006-01-12 18:32:29 +00:00
gtk_container_add(GTK_CONTAINER(wind), vbox);
g_signal_connect(G_OBJECT(wind), "delete_event", G_CALLBACK(winclose), data);
for(ii = g_list_first(saved); NULL != ii; ii = ii->next) {
ts = ii->data;
2006-01-12 18:40:47 +00:00
addtorrent(tr, GTK_WINDOW(wind),
ts->ts_torrent, ts->ts_directory, ts->ts_paused);
2006-01-12 18:32:29 +00:00
cf_freestate(ts);
}
g_list_free(saved);
data->timer = g_timeout_add(500, updatemodel, data);
updatemodel(data);
2006-01-12 18:48:53 +00:00
gtk_widget_show_all(vbox);
gtk_widget_realize(wind);
gtk_widget_size_request(wind, &req);
height = req.height;
gtk_widget_size_request(scroll, &req);
height -= req.height;
gtk_widget_size_request(list, &req);
height += req.height;
gtk_window_set_default_size(GTK_WINDOW(wind), -1, MAX(height, 100));
gtk_widget_show(wind);
2006-01-12 18:32:29 +00:00
}
/* XXX is this the right thing to do? */
#define TR_TORRENT_NEEDS_STOP(t) \
((t) & TR_STATUS_CHECK || (t) & TR_STATUS_DOWNLOAD || (t) & TR_STATUS_SEED)
gboolean
winclose(GtkWidget *widget SHUTUP, GdkEvent *event SHUTUP, gpointer gdata) {
struct cbdata *data = gdata;
struct exitdata *edata;
tr_stat_t *st;
int ii;
if(0 >= data->timer)
g_source_remove(data->timer);
data->timer = -1;
for(ii = tr_torrentStat(data->tr, &st); 0 < ii; ii--) {
if(TR_TORRENT_NEEDS_STOP(st[ii-1].status)) {
2006-01-12 18:48:53 +00:00
/*fprintf(stderr, "quit: stopping %i %s\n", ii, st[ii-1].info.name);*/
2006-01-12 18:32:29 +00:00
tr_torrentStop(data->tr, ii - 1);
} else {
2006-01-12 18:48:53 +00:00
/*fprintf(stderr, "quit: closing %i %s\n", ii, st[ii-1].info.name);*/
2006-01-12 18:32:29 +00:00
tr_torrentClose(data->tr, ii - 1);
}
}
free(st);
/* XXX should disable widgets or something */
/* try to wait until torrents stop before exiting */
edata = g_new0(struct exitdata, 1);
edata->cbdata = data;
edata->started = time(NULL);
edata->timer = g_timeout_add(500, exitcheck, edata);
2006-01-12 18:48:53 +00:00
/*fprintf(stderr, "quit: starting timeout at %i\n", edata->started);*/
2006-01-12 18:32:29 +00:00
/* returning FALSE means to destroy the window */
return TRUE;
}
gboolean
exitcheck(gpointer gdata) {
struct exitdata *data = gdata;
tr_stat_t *st;
int ii;
for(ii = tr_torrentStat(data->cbdata->tr, &st); 0 < ii; ii--) {
if(TR_STATUS_PAUSE == st[ii-1].status) {
2006-01-12 18:48:53 +00:00
/*fprintf(stderr, "quit: closing %i %s\n", ii, st[ii-1].info.name);*/
2006-01-12 18:32:29 +00:00
tr_torrentClose(data->cbdata->tr, ii - 1);
}
}
free(st);
2006-01-12 18:48:53 +00:00
/*fprintf(stderr, "quit: %i torrents left at %i\n",
tr_torrentCount(data->cbdata->tr), time(NULL));*/
2006-01-12 18:32:29 +00:00
/* keep going if we still have torrents and haven't hit the exit timeout */
if(0 < tr_torrentCount(data->cbdata->tr) &&
time(NULL) - data->started < TRACKER_EXIT_TIMEOUT) {
updatemodel(data->cbdata);
return TRUE;
}
2006-01-12 18:48:53 +00:00
/*fprintf(stderr, "quit: giving up on %i torrents\n",
tr_torrentCount(data->cbdata->tr));*/
2006-01-12 18:32:29 +00:00
for(ii = tr_torrentCount(data->cbdata->tr); 0 < ii; ii--)
tr_torrentClose(data->cbdata->tr, ii - 1);
/* exit otherwise */
if(0 >= data->timer)
g_source_remove(data->timer);
data->timer = -1;
2006-01-12 18:40:47 +00:00
gtk_widget_destroy(GTK_WIDGET(data->cbdata->wind));
2006-01-12 18:32:29 +00:00
tr_close(data->cbdata->tr);
g_free(data->cbdata);
g_free(data);
gtk_main_quit();
return FALSE;
}
GtkWidget *
makewind_toolbar(struct cbdata *data) {
GtkWidget *bar = gtk_toolbar_new();
GtkToolItem *item;
unsigned int ii;
gtk_toolbar_set_tooltips(GTK_TOOLBAR(bar), TRUE);
2006-01-12 18:48:53 +00:00
gtk_toolbar_set_show_arrow(GTK_TOOLBAR(bar), FALSE);
2006-01-12 18:32:29 +00:00
gtk_toolbar_set_style(GTK_TOOLBAR(bar), GTK_TOOLBAR_BOTH);
2006-01-12 18:47:30 +00:00
data->buttons = g_new(GtkWidget*, ALEN(actionitems));
2006-01-12 18:40:47 +00:00
for(ii = 0; ii < ALEN(actionitems); ii++) {
2006-01-12 18:32:29 +00:00
item = gtk_tool_button_new_from_stock(actionitems[ii].id);
2006-01-12 18:47:30 +00:00
data->buttons[ii] = GTK_WIDGET(item);
2006-01-12 18:32:29 +00:00
gtk_tool_button_set_label(GTK_TOOL_BUTTON(item), actionitems[ii].name);
gtk_tool_item_set_tooltip(GTK_TOOL_ITEM(item), GTK_TOOLBAR(bar)->tooltips,
actionitems[ii].ttext, actionitems[ii].tpriv);
g_object_set_data(G_OBJECT(item), LIST_ACTION,
GINT_TO_POINTER(actionitems[ii].act));
g_object_set_data(G_OBJECT(item), LIST_ACTION_FROM,
GINT_TO_POINTER(FROM_BUTTON));
g_signal_connect(G_OBJECT(item), "clicked", G_CALLBACK(actionclick), data);
2006-01-12 18:47:30 +00:00
gtk_toolbar_insert(GTK_TOOLBAR(bar), item, -1);
2006-01-12 18:32:29 +00:00
}
return bar;
}
2006-01-12 18:47:30 +00:00
/* XXX check for unused data in model */
2006-01-12 18:32:29 +00:00
enum {MC_NAME, MC_SIZE, MC_STAT, MC_ERR, MC_PROG, MC_DRATE, MC_URATE,
MC_ETA, MC_PEERS, MC_UPEERS, MC_DPEERS, MC_PIECES, MC_DOWN, MC_UP,
MC_ROW_INDEX, MC_ROW_COUNT};
GtkWidget *
makewind_list(struct cbdata *data) {
GType types[] = {
/* info->name, info->totalSize, status, error, progress */
2006-01-12 18:47:30 +00:00
G_TYPE_STRING, G_TYPE_UINT64, G_TYPE_INT, G_TYPE_STRING, G_TYPE_FLOAT,
2006-01-12 18:32:29 +00:00
/* rateDownload, rateUpload, eta, peersTotal, peersUploading */
G_TYPE_FLOAT, G_TYPE_FLOAT, G_TYPE_INT, G_TYPE_INT, G_TYPE_INT,
/* peersDownloading, pieces, downloaded, uploaded */
G_TYPE_INT, TR_TYPE_PIECES, G_TYPE_UINT64, G_TYPE_UINT64,
/* index into the torrent array */
G_TYPE_INT};
GtkListStore *model;
GtkWidget *view;
2006-01-12 18:47:30 +00:00
GtkTreeViewColumn *col;
2006-01-12 18:32:29 +00:00
GtkTreeSelection *sel;
2006-01-12 18:47:30 +00:00
GtkCellRenderer *namerend, *progrend;
2006-01-12 18:32:29 +00:00
2006-01-12 18:40:47 +00:00
assert(MC_ROW_COUNT == ALEN(types));
2006-01-12 18:32:29 +00:00
model = gtk_list_store_newv(MC_ROW_COUNT, types);
view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
/* XXX do I need to worry about reference counts anywhere else? */
g_object_unref(G_OBJECT(model));
data->model = model;
data->view = GTK_TREE_VIEW(view);
2006-01-12 18:47:30 +00:00
namerend = gtk_cell_renderer_text_new();
col = gtk_tree_view_column_new_with_attributes("Name", namerend, NULL);
gtk_tree_view_column_set_cell_data_func(col, namerend, dfname, NULL, NULL);
gtk_tree_view_column_set_expand(col, TRUE);
gtk_tree_view_append_column(GTK_TREE_VIEW(view), col);
progrend = gtk_cell_renderer_torrent_new();
g_object_set(progrend, "label", "<big> fnord fnord </big>", NULL);
col = gtk_tree_view_column_new_with_attributes("Progress", progrend, NULL);
gtk_tree_view_column_set_cell_data_func(col, progrend, dfprog, NULL, NULL);
gtk_tree_view_append_column(GTK_TREE_VIEW(view), col);
2006-01-12 18:32:29 +00:00
2006-01-12 18:56:00 +00:00
/* XXX this shouldn't be necessary */
g_signal_connect(view, "notify", G_CALLBACK(stylekludge), progrend);
2006-01-12 18:32:29 +00:00
gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(view), TRUE);
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(view));
gtk_tree_selection_set_mode(GTK_TREE_SELECTION(sel), GTK_SELECTION_SINGLE);
2006-01-12 18:47:30 +00:00
g_signal_connect(G_OBJECT(sel), "changed", G_CALLBACK(fixbuttons), data);
2006-01-12 18:32:29 +00:00
g_signal_connect(G_OBJECT(view), "button-press-event",
G_CALLBACK(listclick), data);
g_signal_connect(G_OBJECT(view), "popup-menu", G_CALLBACK(listpopup), data);
gtk_widget_show_all(view);
return view;
}
2006-01-12 18:56:00 +00:00
/* kludge to have the progress bars notice theme changes */
static void
stylekludge(GObject *obj, GParamSpec *spec, gpointer gdata) {
if(0 == strcmp("style", spec->name)) {
gtk_cell_renderer_torrent_reset_style(GTK_CELL_RENDERER_TORRENT(gdata));
gtk_widget_queue_draw(GTK_WIDGET(obj));
}
}
2006-01-12 18:47:30 +00:00
/* disable buttons the user shouldn't be able to click on */
void
fixbuttons(GtkTreeSelection *sel, gpointer gdata) {
struct cbdata *data = gdata;
GtkTreeModel *model;
GtkTreeIter iter;
gboolean selected;
unsigned int ii;
int status;
if(NULL == sel)
sel = gtk_tree_view_get_selection(data->view);
if((selected = gtk_tree_selection_get_selected(sel, &model, &iter)))
gtk_tree_model_get(model, &iter, MC_STAT, &status, -1);
for(ii = 0; ii < ALEN(actionitems); ii++)
if(actionitems[ii].avail)
gtk_widget_set_sensitive(data->buttons[ii],
(selected && (actionitems[ii].avail & status)));
}
void
dfname(GtkTreeViewColumn *col SHUTUP, GtkCellRenderer *rend,
GtkTreeModel *model, GtkTreeIter *iter, gpointer gdata SHUTUP) {
char *name, *mb, *err, *str, *top, *bottom;
guint64 size;
gfloat prog;
int status, eta, tpeers, upeers, dpeers;
/* XXX should I worry about gtk_tree_model_get failing? */
gtk_tree_model_get(model, iter, MC_NAME, &name, MC_STAT, &status,
MC_SIZE, &size, MC_PROG, &prog, MC_ETA, &eta, MC_PEERS, &tpeers,
MC_UPEERS, &upeers, MC_DPEERS, &dpeers, -1);
if(0 > eta)
eta = 0;
if(0 > tpeers)
tpeers = 0;
if(0 > upeers)
upeers = 0;
if(0 > dpeers)
dpeers = 0;
mb = readablesize(size, 1);
prog *= 100;
if(status & TR_STATUS_CHECK)
top = g_strdup_printf("Checking existing files (%.1f%%)", prog);
else if(status & TR_STATUS_DOWNLOAD)
top = g_strdup_printf("Finishing in %02i:%02i:%02i (%.1f%%)",
eta / 60 / 60, eta / 60 % 60, eta % 60, prog);
else if(status & TR_STATUS_SEED)
top = g_strdup_printf("Seeding, uploading to %d of %d peer%s",
dpeers, tpeers, (1 == tpeers ? "" : "s"));
else if(status & TR_STATUS_STOPPING)
top = g_strdup("Stopping...");
else if(status & TR_STATUS_PAUSE)
top = g_strdup_printf("Paused (%.1f%%)", prog);
else {
top = g_strdup("");
assert("XXX unknown status");
}
if(status & TR_TRACKER_ERROR) {
gtk_tree_model_get(model, iter, MC_ERR, &err, -1);
bottom = g_strconcat("Error: ", err, NULL);
g_free(err);
}
else if(status & TR_STATUS_DOWNLOAD)
bottom = g_strdup_printf("Downloading from %i of %i peer%s",
upeers, tpeers, (1 == tpeers ? "" : "s"));
else
bottom = NULL;
str = g_markup_printf_escaped("<big>%s (%s)</big>\n<small>%s\n%s</small>",
name, mb, top, (NULL == bottom ? "" : bottom));
g_object_set(rend, "markup", str, NULL);
g_free(name);
g_free(mb);
g_free(str);
g_free(top);
g_free(bottom);
}
void
dfprog(GtkTreeViewColumn *col SHUTUP, GtkCellRenderer *rend,
GtkTreeModel *model, GtkTreeIter *iter, gpointer gdata SHUTUP) {
char *dlstr, *ulstr, *str;
gfloat prog, dl, ul;
/* XXX should I worry about gtk_tree_model_get failing? */
gtk_tree_model_get(model, iter, MC_PROG, &prog,
MC_DRATE, &dl, MC_URATE, &ul, -1);
if(0.0 > prog)
prog = 0.0;
else if(1.0 < prog)
prog = 1.0;
dlstr = readablesize(dl * 1024.0, 2);
ulstr = readablesize(ul * 1024.0, 2);
str = g_strdup_printf("<small>DL: %s/s\nUL: %s/s</small>", dlstr, ulstr);
g_object_set(rend, "text", str, "value", prog, NULL);
g_free(dlstr);
g_free(ulstr);
g_free(str);
}
2006-01-12 18:32:29 +00:00
gboolean
updatemodel(gpointer gdata) {
struct cbdata *data = gdata;
tr_stat_t *st;
2006-01-12 18:47:30 +00:00
int ii, max;
2006-01-12 18:32:29 +00:00
GtkTreeIter iter;
2006-01-12 18:47:30 +00:00
float up, down;
char *upstr, *downstr, *str;
2006-01-12 18:32:29 +00:00
max = tr_torrentStat(data->tr, &st);
for(ii = 0; ii < max; ii++) {
if(!(ii ? gtk_tree_model_iter_next(GTK_TREE_MODEL(data->model), &iter) :
gtk_tree_model_get_iter_first(GTK_TREE_MODEL(data->model), &iter)))
gtk_list_store_append(data->model, &iter);
/* XXX find out if setting the same data emits changed signal */
2006-01-12 18:47:30 +00:00
gtk_list_store_set(
data->model, &iter, MC_ROW_INDEX, ii,
MC_NAME, st[ii].info.name, MC_SIZE, st[ii].info.totalSize,
MC_STAT, st[ii].status, MC_ERR, st[ii].error, MC_PROG, st[ii].progress,
MC_DRATE, st[ii].rateDownload, MC_URATE, st[ii].rateUpload,
MC_ETA, st[ii].eta, MC_PEERS, st[ii].peersTotal,
2006-01-12 18:32:29 +00:00
MC_UPEERS, st[ii].peersUploading, MC_DPEERS, st[ii].peersDownloading,
2006-01-12 18:40:47 +00:00
MC_DOWN, st[ii].downloaded, MC_UP, st[ii].uploaded, -1);
2006-01-12 18:32:29 +00:00
}
free(st);
2006-01-12 18:45:41 +00:00
/* remove any excess rows */
2006-01-12 18:48:53 +00:00
if(ii ? gtk_tree_model_iter_next(GTK_TREE_MODEL(data->model), &iter) :
gtk_tree_model_get_iter_first(GTK_TREE_MODEL(data->model), &iter))
2006-01-12 18:45:41 +00:00
while(gtk_list_store_remove(data->model, &iter))
;
2006-01-12 18:47:30 +00:00
/* update the status bar */
tr_torrentRates(data->tr, &up, &down);
downstr = readablesize(down * 1024.0, 2);
upstr = readablesize(up * 1024.0, 2);
str = g_strdup_printf(" Total DL: %s/s Total UL: %s/s", upstr, downstr);
gtk_statusbar_pop(data->bar, 0);
gtk_statusbar_push(data->bar, 0, str);
g_free(str);
g_free(upstr);
g_free(downstr);
/* the status of the selected item may have changed, so update the buttons */
fixbuttons(NULL, data);
2006-01-12 18:32:29 +00:00
return TRUE;
}
gboolean
2006-01-12 18:47:30 +00:00
listclick(GtkWidget *widget SHUTUP, GdkEventButton *event, gpointer gdata) {
2006-01-12 18:32:29 +00:00
struct cbdata *data = gdata;
2006-01-12 18:47:30 +00:00
GtkTreePath *path;
GtkTreeIter iter;
2006-01-12 18:32:29 +00:00
if(GDK_BUTTON_PRESS == event->type && 3 == event->button) {
2006-01-12 18:47:30 +00:00
if(!gtk_tree_view_get_path_at_pos(data->view, event->x, event->y, &path,
NULL, NULL, NULL))
dopopupmenu(event, data, NULL);
else {
if(gtk_tree_model_get_iter(GTK_TREE_MODEL(data->model), &iter, path))
dopopupmenu(event, data, &iter);
gtk_tree_path_free(path);
}
2006-01-12 18:32:29 +00:00
return TRUE;
}
return FALSE;
}
gboolean
2006-01-12 18:47:30 +00:00
listpopup(GtkWidget *widget SHUTUP, gpointer gdata) {
struct cbdata *data = gdata;
GtkTreeSelection *sel = gtk_tree_view_get_selection(data->view);
GtkTreeModel *model;
GtkTreeIter iter;
if(!gtk_tree_selection_get_selected(sel, &model, &iter))
dopopupmenu(NULL, data, NULL);
else {
assert(model == GTK_TREE_MODEL(data->model));
dopopupmenu(NULL, data, &iter);
}
2006-01-12 18:32:29 +00:00
return TRUE;
}
void
2006-01-12 18:47:30 +00:00
dopopupmenu(GdkEventButton *event, struct cbdata *data, GtkTreeIter *iter) {
2006-01-12 18:32:29 +00:00
GtkWidget *menu = gtk_menu_new();
GtkWidget *item;
unsigned int ii;
2006-01-12 18:47:30 +00:00
int index, status;
2006-01-12 18:32:29 +00:00
index = -1;
2006-01-12 18:47:30 +00:00
if(NULL != iter)
gtk_tree_model_get(GTK_TREE_MODEL(data->model), iter,
MC_ROW_INDEX, &index, MC_STAT, &status, -1);
2006-01-12 18:32:29 +00:00
/* XXX am I leaking references here? */
2006-01-12 18:40:47 +00:00
for(ii = 0; ii < ALEN(actionitems); ii++) {
2006-01-12 18:47:30 +00:00
if(actionitems[ii].nomenu ||
(actionitems[ii].avail &&
(0 > index || !(actionitems[ii].avail & status))))
continue;
2006-01-12 18:32:29 +00:00
item = gtk_menu_item_new_with_label(actionitems[ii].name);
g_object_set_data(G_OBJECT(item), LIST_ACTION,
GINT_TO_POINTER(actionitems[ii].act));
g_object_set_data(G_OBJECT(item), LIST_ACTION_FROM,
GINT_TO_POINTER(FROM_POPUP));
g_object_set_data(G_OBJECT(item), LIST_INDEX, GINT_TO_POINTER(index));
g_signal_connect(G_OBJECT(item), "activate",
G_CALLBACK(actionclick), data);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
}
gtk_widget_show_all(menu);
gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL,
(NULL == event ? 0 : event->button),
gdk_event_get_time((GdkEvent*)event));
}
void
actionclick(GtkWidget *widget, gpointer gdata) {
struct cbdata *data = gdata;
GtkTreeSelection *sel = gtk_tree_view_get_selection(data->view);
GtkTreeModel *model;
GtkTreeIter iter;
enum listact act =
GPOINTER_TO_INT(g_object_get_data(G_OBJECT(widget), LIST_ACTION));
enum listfrom from =
GPOINTER_TO_INT(g_object_get_data(G_OBJECT(widget), LIST_ACTION_FROM));
int index;
tr_stat_t *sb;
2006-01-12 18:40:47 +00:00
switch(act) {
case ACT_OPEN:
2006-01-12 18:55:27 +00:00
makeaddwind(addtorrent, data->wind, data->tr, addedtorrents, data);
2006-01-12 18:40:47 +00:00
return;
case ACT_PREF:
makeprefwindow(data->wind, data->tr);
return;
default:
break;
2006-01-12 18:32:29 +00:00
}
index = -1;
switch(from) {
case FROM_BUTTON:
if(gtk_tree_selection_get_selected(sel, &model, &iter))
gtk_tree_model_get(GTK_TREE_MODEL(model), &iter,
MC_ROW_INDEX, &index, -1);
/* XXX should I assert(0 <= index) to insure a row was selected? */
break;
case FROM_POPUP:
index = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(widget), LIST_INDEX));
break;
default:
assert(!"unknown action source");
break;
}
if(0 <= index) {
switch(act) {
case ACT_START:
tr_torrentStart(data->tr, index);
savetorrents(data->tr, data->wind, -1, NULL);
2006-01-12 18:47:30 +00:00
updatemodel(data);
2006-01-12 18:32:29 +00:00
break;
case ACT_STOP:
tr_torrentStop(data->tr, index);
savetorrents(data->tr, data->wind, -1, NULL);
2006-01-12 18:47:30 +00:00
updatemodel(data);
2006-01-12 18:32:29 +00:00
break;
case ACT_DELETE:
/* XXX need to be able to stat just one torrent */
if(index >= tr_torrentStat(data->tr, &sb)) {
assert(!"XXX i'm tired");
}
if(TR_TORRENT_NEEDS_STOP(sb[index].status))
tr_torrentStop(data->tr, index);
free(sb);
tr_torrentClose(data->tr, index);
savetorrents(data->tr, data->wind, -1, NULL);
2006-01-12 18:45:41 +00:00
updatemodel(data);
2006-01-12 18:32:29 +00:00
break;
case ACT_INFO:
makeinfowind(data, index);
break;
default:
assert(!"unknown type");
break;
}
}
}
gboolean
2006-01-12 18:40:47 +00:00
addtorrent(tr_handle_t *tr, GtkWindow *parentwind, const char *torrent,
2006-01-12 18:32:29 +00:00
const char *dir, gboolean paused) {
char *wd;
2006-01-12 18:55:27 +00:00
if(NULL == dir && NULL != (dir = cf_getpref(PREF_DIR))) {
2006-01-12 18:40:47 +00:00
if(!mkdir_p(dir, 0777)) {
errmsg(parentwind, "Failed to create download directory %s:\n%s",
dir, strerror(errno));
return FALSE;
}
}
2006-01-12 18:32:29 +00:00
if(0 != tr_torrentInit(tr, torrent)) {
/* XXX would be nice to have errno strings, are they printed to stdout? */
errmsg(parentwind, "Failed to open torrent file %s", torrent);
return FALSE;
}
if(NULL != dir)
tr_torrentSetFolder(tr, tr_torrentCount(tr) - 1, dir);
else {
wd = g_new(char, MAXPATHLEN + 1);
if(NULL == getcwd(wd, MAXPATHLEN + 1))
tr_torrentSetFolder(tr, tr_torrentCount(tr) - 1, ".");
else {
tr_torrentSetFolder(tr, tr_torrentCount(tr) - 1, wd);
free(wd);
}
}
if(!paused)
tr_torrentStart(tr, tr_torrentCount(tr) - 1);
return TRUE;
}
void
2006-01-12 18:55:27 +00:00
addedtorrents(void *vdata) {
struct cbdata *data = vdata;
2006-01-12 18:32:29 +00:00
2006-01-12 18:55:27 +00:00
updatemodel(data);
savetorrents(data->tr, data->wind, -1, NULL);
2006-01-12 18:32:29 +00:00
}
const char *
statusstr(int status) {
switch(status) {
case TR_STATUS_CHECK: return "check";
case TR_STATUS_DOWNLOAD: return "download";
case TR_STATUS_SEED: return "seed";
case TR_STATUS_STOPPING: return "stopping";
case TR_STATUS_STOPPED: return "stopped";
case TR_STATUS_PAUSE: return "pause";
case TR_TRACKER_ERROR: return "error";
default:
assert(!"unknown status code");
return NULL;
}
}
void
makeinfowind(struct cbdata *data, int index) {
tr_stat_t *sb;
GtkWidget *wind, *table, *name, *value;
char buf[32];
if(index >= tr_torrentStat(data->tr, &sb)) {
assert(!"XXX i'm tired");
}
2006-01-12 18:40:47 +00:00
wind = gtk_dialog_new_with_buttons(sb[index].info.name, data->wind,
2006-01-12 18:32:29 +00:00
GTK_DIALOG_DESTROY_WITH_PARENT, GTK_STOCK_OK, GTK_RESPONSE_ACCEPT, NULL);
table = gtk_table_new(21, 2, FALSE);
name = gtk_label_new("Torrent File");
value = gtk_label_new(sb[index].info.torrent);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 0, 1);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 0, 1);
name = gtk_label_new("Name");
value = gtk_label_new(sb[index].info.name);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 1, 2);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 1, 2);
name = gtk_label_new("Tracker Address");
value = gtk_label_new(sb[index].info.trackerAddress);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 2, 3);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 2, 3);
name = gtk_label_new("Tracker Port");
snprintf(buf, sizeof buf, "%i", sb[index].info.trackerPort);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 3, 4);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 3, 4);
name = gtk_label_new("Tracker Announce URL");
value = gtk_label_new(sb[index].info.trackerAnnounce);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 4, 5);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 4, 5);
name = gtk_label_new("Piece Size");
snprintf(buf, sizeof buf, "%i", sb[index].info.pieceSize);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 5, 6);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 5, 6);
name = gtk_label_new("Piece Count");
snprintf(buf, sizeof buf, "%i", sb[index].info.pieceCount);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 6, 7);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 6, 7);
name = gtk_label_new("Total Size");
snprintf(buf, sizeof buf, "%llu", sb[index].info.totalSize);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 7, 8);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 7, 8);
name = gtk_label_new("File Count");
snprintf(buf, sizeof buf, "%i", sb[index].info.fileCount);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 8, 9);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 8, 9);
name = gtk_label_new("Status");
value = gtk_label_new(statusstr(sb[index].status));
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 9, 10);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 9, 10);
name = gtk_label_new("Error");
value = gtk_label_new(sb[index].error);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 10, 11);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 10, 11);
name = gtk_label_new("Progress");
snprintf(buf, sizeof buf, "%f", sb[index].progress);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 11, 12);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 11, 12);
name = gtk_label_new("Download Rate");
value = gtk_label_new(buf);
snprintf(buf, sizeof buf, "%f", sb[index].rateDownload);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 12, 13);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 12, 13);
name = gtk_label_new("Upload Rate");
snprintf(buf, sizeof buf, "%f", sb[index].rateUpload);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 13, 14);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 13, 14);
name = gtk_label_new("ETA");
snprintf(buf, sizeof buf, "%i", sb[index].eta);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 14, 15);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 14, 15);
name = gtk_label_new("Total Peers");
snprintf(buf, sizeof buf, "%i", sb[index].peersTotal);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 15, 16);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 15, 16);
name = gtk_label_new("Uploading Peers");
snprintf(buf, sizeof buf, "%i", sb[index].peersUploading);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 16, 17);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 16, 17);
name = gtk_label_new("Downloading Peers");
snprintf(buf, sizeof buf, "%i", sb[index].peersDownloading);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 17, 18);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 17, 18);
name = gtk_label_new("Downloaded");
snprintf(buf, sizeof buf, "%llu", sb[index].downloaded);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 18, 19);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 18, 19);
name = gtk_label_new("Uploaded");
snprintf(buf, sizeof buf, "%llu", sb[index].uploaded);
value = gtk_label_new(buf);
gtk_table_attach_defaults(GTK_TABLE(table), name, 0, 1, 19, 20);
gtk_table_attach_defaults(GTK_TABLE(table), value, 1, 2, 19, 20);
gtk_box_pack_start_defaults(GTK_BOX(GTK_DIALOG(wind)->vbox), table);
g_signal_connect(G_OBJECT(wind), "response",
G_CALLBACK(gtk_widget_destroy), NULL);
gtk_widget_show_all(wind);
free(sb);
}
gboolean
2006-01-12 18:40:47 +00:00
savetorrents(tr_handle_t *tr, GtkWindow *wind, int count, tr_stat_t *stat) {
2006-01-12 18:32:29 +00:00
char *errstr;
tr_stat_t *st;
gboolean ret;
assert(NULL != tr || 0 <= count);
if(0 <= count)
ret = cf_savestate(count, stat, &errstr);
else {
count = tr_torrentStat(tr, &st);
ret = cf_savestate(count, st, &errstr);
free(st);
}
if(!ret) {
errmsg(wind, "%s", errstr);
g_free(errstr);
}
return ret;
}