source: trunk/gtk/util.c @ 10404

Last change on this file since 10404 was 10404, checked in by charles, 12 years ago

(trunk gtk) tinker with compact mode a little bit

  • Property svn:keywords set to Date Rev Author Id
File size: 22.5 KB
Line 
1/*
2 * This file Copyright (C) 2008-2010 Mnemosyne LLC
3 *
4 * This file is licensed by the GPL version 2.  Works owned by the
5 * Transmission project are granted a special exemption to clause 2(b)
6 * so that the bulk of its code can remain under the MIT license.
7 * This exemption does not extend to derived works not owned by
8 * the Transmission project.
9 *
10 * $Id: util.c 10404 2010-03-20 14:53:08Z charles $
11 */
12
13#include <ctype.h> /* isxdigit() */
14#include <errno.h>
15#include <stdlib.h> /* free() */
16#include <string.h> /* strcmp() */
17
18#include <sys/types.h> /* for gtr_lockfile()'s open() */
19#include <sys/stat.h> /* for gtr_lockfile()'s open() */
20#include <fcntl.h> /* for gtr_lockfile()'s open() */
21
22#include <gtk/gtk.h>
23#include <glib/gi18n.h>
24#include <glib/gstdio.h> /* g_unlink() */
25#ifdef HAVE_GIO
26 #include <gio/gio.h> /* g_file_trash() */
27#endif
28#ifdef HAVE_DBUS_GLIB
29 #include <dbus/dbus-glib.h>
30#endif
31
32#include <evhttp.h>
33
34#include <libtransmission/transmission.h> /* TR_RATIO_NA, TR_RATIO_INF */
35#include <libtransmission/utils.h> /* tr_inf */
36#include <libtransmission/version.h> /* tr_inf */
37
38#include "conf.h"
39#include "hig.h"
40#include "tr-prefs.h"
41#include "util.h"
42
43gtr_lockfile_state_t
44gtr_lockfile( const char * filename )
45{
46    gtr_lockfile_state_t ret;
47
48#ifdef WIN32
49
50    HANDLE file = CreateFile( filename,
51                              GENERIC_READ | GENERIC_WRITE,
52                              FILE_SHARE_READ | FILE_SHARE_WRITE,
53                              NULL,
54                              OPEN_ALWAYS,
55                              FILE_ATTRIBUTE_NORMAL,
56                              NULL );
57    if( file == INVALID_HANDLE_VALUE )
58        ret = GTR_LOCKFILE_EOPEN;
59    else if( !LockFile( file, 0, 0, 1, 1 ) )
60        ret = GTR_LOCKFILE_ELOCK;
61    else
62        ret = GTR_LOCKFILE_SUCCESS;
63
64#else
65
66    int fd = open( filename, O_RDWR | O_CREAT, 0666 );
67    if( fd < 0 )
68        ret = GTR_LOCKFILE_EOPEN;
69    else {
70        struct flock lk;
71        memset( &lk, 0,  sizeof( lk ) );
72        lk.l_start = 0;
73        lk.l_len = 0;
74        lk.l_type = F_WRLCK;
75        lk.l_whence = SEEK_SET;
76        if( -1 == fcntl( fd, F_SETLK, &lk ) )
77            ret = GTR_LOCKFILE_ELOCK;
78        else
79            ret = GTR_LOCKFILE_SUCCESS;
80    }
81
82#endif
83
84    return ret;
85}
86
87/***
88****
89***/
90
91const char*
92gtr_get_unicode_string( int i )
93{
94    switch( i ) {
95        case GTR_UNICODE_UP:   return "\xE2\x86\x91";
96        case GTR_UNICODE_DOWN: return "\xE2\x86\x93";
97        case GTR_UNICODE_INF:  return "\xE2\x88\x9E";
98        default:               return "err";
99    }
100}
101
102char*
103tr_strlratio( char * buf, double ratio, size_t buflen )
104{
105    return tr_strratio( buf, buflen, ratio, gtr_get_unicode_string( GTR_UNICODE_INF ) );
106}
107
108#define KILOBYTE_FACTOR 1024.0
109#define MEGABYTE_FACTOR ( 1024.0 * 1024.0 )
110#define GIGABYTE_FACTOR ( 1024.0 * 1024.0 * 1024.0 )
111
112char*
113tr_strlsize( char *  buf,
114             guint64 size,
115             size_t  buflen )
116{
117    if( !size )
118        g_strlcpy( buf, _( "None" ), buflen );
119#if GLIB_CHECK_VERSION( 2, 16, 0 )
120    else
121    {
122        char * tmp = g_format_size_for_display( size );
123        g_strlcpy( buf, tmp, buflen );
124        g_free( tmp );
125    }
126#else
127    else if( size < (guint64)KILOBYTE_FACTOR )
128        g_snprintf( buf, buflen,
129                    ngettext( "%'u byte", "%'u bytes",
130                              (guint)size ), (guint)size );
131    else
132    {
133        gdouble displayed_size;
134        if( size < (guint64)MEGABYTE_FACTOR )
135        {
136            displayed_size = (gdouble) size / KILOBYTE_FACTOR;
137            g_snprintf( buf, buflen, _( "%'.1f KB" ), displayed_size );
138        }
139        else if( size < (guint64)GIGABYTE_FACTOR )
140        {
141            displayed_size = (gdouble) size / MEGABYTE_FACTOR;
142            g_snprintf( buf, buflen, _( "%'.1f MB" ), displayed_size );
143        }
144        else
145        {
146            displayed_size = (gdouble) size / GIGABYTE_FACTOR;
147            g_snprintf( buf, buflen, _( "%'.1f GB" ), displayed_size );
148        }
149    }
150#endif
151    return buf;
152}
153
154char*
155tr_strlspeed( char * buf,
156              double kb_sec,
157              size_t buflen )
158{
159    const double speed = kb_sec;
160
161    if( speed < 1000.0 )  /* 0.0 KB to 999.9 KB */
162        g_snprintf( buf, buflen, _( "%'.1f KB/s" ), speed );
163    else if( speed < 102400.0 ) /* 0.98 MB to 99.99 MB */
164        g_snprintf( buf, buflen, _( "%'.2f MB/s" ), ( speed / KILOBYTE_FACTOR ) );
165    else if( speed < 1024000.0 ) /* 100.0 MB to 999.9 MB */
166        g_snprintf( buf, buflen, _( "%'.1f MB/s" ), ( speed / MEGABYTE_FACTOR ) );
167    else /* insane speeds */
168        g_snprintf( buf, buflen, _( "%'.2f GB/s" ), ( speed / GIGABYTE_FACTOR ) );
169
170    return buf;
171}
172
173char*
174tr_strltime( char * buf,
175             int    seconds,
176             size_t buflen )
177{
178    int  days, hours, minutes;
179    char d[128], h[128], m[128], s[128];
180
181    if( seconds < 0 )
182        seconds = 0;
183
184    days = seconds / 86400;
185    hours = ( seconds % 86400 ) / 3600;
186    minutes = ( seconds % 3600 ) / 60;
187    seconds = ( seconds % 3600 ) % 60;
188
189    g_snprintf( d, sizeof( d ), ngettext( "%'d day", "%'d days",
190                                          days ), days );
191    g_snprintf( h, sizeof( h ), ngettext( "%'d hour", "%'d hours",
192                                          hours ), hours );
193    g_snprintf( m, sizeof( m ),
194                ngettext( "%'d minute", "%'d minutes", minutes ), minutes );
195    g_snprintf( s, sizeof( s ),
196                ngettext( "%'d second", "%'d seconds", seconds ), seconds );
197
198    if( days )
199    {
200        if( days >= 4 || !hours )
201        {
202            g_strlcpy( buf, d, buflen );
203        }
204        else
205        {
206            g_snprintf( buf, buflen, "%s, %s", d, h );
207        }
208    }
209    else if( hours )
210    {
211        if( hours >= 4 || !minutes )
212        {
213            g_strlcpy( buf, h, buflen );
214        }
215        else
216        {
217            g_snprintf( buf, buflen, "%s, %s", h, m );
218        }
219    }
220    else if( minutes )
221    {
222        if( minutes >= 4 || !seconds )
223        {
224            g_strlcpy( buf, m, buflen );
225        }
226        else
227        {
228            g_snprintf( buf, buflen, "%s, %s", m, s );
229        }
230    }
231    else
232    {
233        g_strlcpy( buf, s, buflen );
234    }
235
236    return buf;
237}
238
239char *
240gtr_localtime( time_t time )
241{
242    const struct tm tm = *localtime( &time );
243    char            buf[256], *eoln;
244
245    g_strlcpy( buf, asctime( &tm ), sizeof( buf ) );
246    if( ( eoln = strchr( buf, '\n' ) ) )
247        *eoln = '\0';
248
249    return g_locale_to_utf8( buf, -1, NULL, NULL, NULL );
250}
251
252char *
253gtr_localtime2( char * buf, time_t time, size_t buflen )
254{
255    char * tmp = gtr_localtime( time );
256    g_strlcpy( buf, tmp, buflen );
257    g_free( tmp );
258    return buf;
259}
260
261int
262gtr_mkdir_with_parents( const char * path, int mode )
263{
264#if GLIB_CHECK_VERSION( 2, 8, 0 )
265    return !g_mkdir_with_parents( path, mode );
266#else
267    return !tr_mkdirp( path, mode );
268#endif
269}
270
271GSList *
272dupstrlist( GSList * l )
273{
274    GSList * ret = NULL;
275
276    for( ; l != NULL; l = l->next )
277        ret = g_slist_prepend( ret, g_strdup( l->data ) );
278    return g_slist_reverse( ret );
279}
280
281char *
282joinstrlist( GSList *list,
283             char *  sep )
284{
285    GSList * l;
286    GString *gstr = g_string_new ( NULL );
287
288    for( l = list; l != NULL; l = l->next )
289    {
290        g_string_append ( gstr, (char*)l->data );
291        if( l->next != NULL )
292            g_string_append ( gstr, ( sep ) );
293    }
294    return g_string_free ( gstr, FALSE );
295}
296
297void
298freestrlist( GSList *list )
299{
300    g_slist_foreach ( list, (GFunc)g_free, NULL );
301    g_slist_free ( list );
302}
303
304char *
305decode_uri( const char * uri )
306{
307    gboolean in_query = FALSE;
308    char *   ret = g_new( char, strlen( uri ) + 1 );
309    char *   out = ret;
310
311    for( ; uri && *uri; )
312    {
313        char ch = *uri;
314        if( ch == '?' )
315            in_query = TRUE;
316        else if( ch == '+' && in_query )
317            ch = ' ';
318        else if( ch == '%' && isxdigit( (unsigned char)uri[1] )
319               && isxdigit( (unsigned char)uri[2] ) )
320        {
321            char buf[3] = { uri[1], uri[2], '\0' };
322            ch = (char) g_ascii_strtoull( buf, NULL, 16 );
323            uri += 2;
324        }
325
326        ++uri;
327        *out++ = ch;
328    }
329
330    *out = '\0';
331    return ret;
332}
333
334
335gboolean
336gtr_is_supported_url( const char * str )
337{
338    return !strncmp( str, "ftp://", 6 )
339        || !strncmp( str, "http://", 7 )
340        || !strncmp( str, "https://", 8 );
341}
342
343gboolean
344gtr_is_magnet_link( const char * str )
345{
346    return !strncmp( str, "magnet:?", 8 );
347}
348
349gboolean
350gtr_is_hex_hashcode( const char * str )
351{
352    int i;
353
354    if( !str || ( strlen( str ) != 40 ) )
355        return FALSE;
356
357    for( i=0; i<40; ++i )
358        if( !isxdigit( str[i] ) )
359            return FALSE;
360
361    return TRUE;
362}
363
364static GtkWindow *
365getWindow( GtkWidget * w )
366{
367    if( w == NULL )
368        return NULL;
369
370    if( GTK_IS_WINDOW( w ) )
371        return GTK_WINDOW( w );
372
373    return GTK_WINDOW( gtk_widget_get_ancestor( w, GTK_TYPE_WINDOW ) );
374}
375
376void
377addTorrentErrorDialog( GtkWidget *  child,
378                       int          err,
379                       const char * filename )
380{
381    char * secondary;
382    const char * fmt;
383    GtkWidget * w;
384    GtkWindow * win = getWindow( child );
385
386    switch( err )
387    {
388        case TR_PARSE_ERR: fmt = _( "The torrent file \"%s\" contains invalid data." ); break;
389        case TR_PARSE_DUPLICATE: fmt = _( "The torrent file \"%s\" is already in use." ); break;
390        default: fmt = _( "The torrent file \"%s\" encountered an unknown error." ); break;
391    }
392    secondary = g_strdup_printf( fmt, filename );
393
394    w = gtk_message_dialog_new( win,
395                                GTK_DIALOG_DESTROY_WITH_PARENT,
396                                GTK_MESSAGE_ERROR,
397                                GTK_BUTTONS_CLOSE,
398                                "%s", _( "Error opening torrent" ) );
399    gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG( w ),
400                                              "%s", secondary );
401    g_signal_connect_swapped( w, "response",
402                              G_CALLBACK( gtk_widget_destroy ), w );
403    gtk_widget_show_all( w );
404    g_free( secondary );
405}
406
407typedef void ( PopupFunc )( GtkWidget*, GdkEventButton* );
408
409/* pop up the context menu if a user right-clicks.
410   if the row they right-click on isn't selected, select it. */
411
412gboolean
413on_tree_view_button_pressed( GtkWidget *      view,
414                             GdkEventButton * event,
415                             gpointer         func )
416{
417    GtkTreeView * tv = GTK_TREE_VIEW( view );
418
419    if( event->type == GDK_BUTTON_PRESS  &&  event->button == 3 )
420    {
421        GtkTreeSelection * selection = gtk_tree_view_get_selection( tv );
422        GtkTreePath *      path;
423        if( gtk_tree_view_get_path_at_pos ( tv,
424                                            (gint) event->x,
425                                            (gint) event->y,
426                                            &path, NULL, NULL, NULL ) )
427        {
428            if( !gtk_tree_selection_path_is_selected ( selection, path ) )
429            {
430                gtk_tree_selection_unselect_all ( selection );
431                gtk_tree_selection_select_path ( selection, path );
432            }
433            gtk_tree_path_free( path );
434        }
435
436        if( func != NULL )
437            ( (PopupFunc*)func )( view, event );
438
439        return TRUE;
440    }
441
442    return FALSE;
443}
444
445/* if the user clicked in an empty area of the list,
446 * clear all the selections. */
447gboolean
448on_tree_view_button_released( GtkWidget *      view,
449                              GdkEventButton * event,
450                              gpointer         unused UNUSED )
451{
452    GtkTreeView * tv = GTK_TREE_VIEW( view );
453
454    if( !gtk_tree_view_get_path_at_pos ( tv,
455                                         (gint) event->x,
456                                         (gint) event->y,
457                                         NULL, NULL, NULL, NULL ) )
458    {
459        GtkTreeSelection * selection = gtk_tree_view_get_selection( tv );
460        gtk_tree_selection_unselect_all ( selection );
461    }
462
463    return FALSE;
464}
465
466gpointer
467tr_object_ref_sink( gpointer object )
468{
469#if GLIB_CHECK_VERSION( 2, 10, 0 )
470    g_object_ref_sink( object );
471#else
472    g_object_ref( object );
473    gtk_object_sink( GTK_OBJECT( object ) );
474#endif
475    return object;
476}
477
478int
479tr_file_trash_or_remove( const char * filename )
480{
481    if( filename && *filename )
482    {
483        gboolean trashed = FALSE;
484#ifdef HAVE_GIO
485        GError * err = NULL;
486        GFile *  file = g_file_new_for_path( filename );
487        trashed = g_file_trash( file, NULL, &err );
488        if( err )
489            g_message( "Unable to trash file \"%s\": %s", filename, err->message );
490        g_clear_error( &err );
491        g_object_unref( G_OBJECT( file ) );
492#endif
493
494        if( !trashed && g_remove( filename ) )
495        {
496            const int err = errno;
497            g_message( "Unable to remove file \"%s\": %s", filename, g_strerror( err ) );
498        }
499    }
500
501    return 0;
502}
503
504char*
505gtr_get_help_url( void )
506{
507    const char * fmt = "http://www.transmissionbt.com/help/gtk/%d.%dx";
508    int          major, minor;
509
510    sscanf( SHORT_VERSION_STRING, "%d.%d", &major, &minor );
511    return g_strdup_printf( fmt, major, minor / 10 );
512}
513
514void
515gtr_open_file( const char * path )
516{
517    if( path )
518    {
519        gboolean opened = FALSE;
520#ifdef HAVE_GIO
521        if( !opened )
522        {
523            GFile * file = g_file_new_for_path( path );
524            char *  uri = g_file_get_uri( file );
525            opened = g_app_info_launch_default_for_uri( uri, NULL, NULL );
526            g_free( uri );
527            g_object_unref( G_OBJECT( file ) );
528        }
529#endif
530        if( !opened )
531        {
532            char * argv[] = { (char*)"xdg-open", (char*)path, NULL };
533            opened = g_spawn_async( NULL, argv, NULL, G_SPAWN_SEARCH_PATH,
534                                    NULL, NULL, NULL, NULL );
535        }
536
537        if( !opened )
538        {
539            g_message( "Unable to open \"%s\"", path );
540        }
541    }
542}
543
544#define VALUE_SERVICE_NAME        "com.transmissionbt.Transmission"
545#define VALUE_SERVICE_OBJECT_PATH "/com/transmissionbt/Transmission"
546#define VALUE_SERVICE_INTERFACE   "com.transmissionbt.Transmission"
547
548gboolean
549gtr_dbus_add_torrent( const char * filename )
550{
551    /* FIXME: why is this static? */
552    static gboolean handled = FALSE;
553
554#ifdef HAVE_DBUS_GLIB
555    char * payload;
556    gsize file_length;
557    char * file_contents = NULL;
558
559    /* If it's a file, load its contents and send them over the wire...
560     * it might be a temporary file that's going to disappear. */
561    if( g_file_get_contents( filename, &file_contents, &file_length, NULL ) )
562        payload = tr_base64_encode( file_contents, file_length, NULL );
563    else if( gtr_is_supported_url( filename ) || gtr_is_magnet_link( filename ) )
564        payload = tr_strdup( filename );
565    else
566        payload = NULL;
567
568    if( payload != NULL )
569    {
570        GError * err = NULL;
571        DBusGConnection * conn;
572        DBusGProxy * proxy = NULL;
573
574        if(( conn = dbus_g_bus_get( DBUS_BUS_SESSION, &err )))
575            proxy = dbus_g_proxy_new_for_name (conn, VALUE_SERVICE_NAME,
576                                                     VALUE_SERVICE_OBJECT_PATH,
577                                                     VALUE_SERVICE_INTERFACE );
578        else if( err )
579           g_message( "err: %s", err->message );
580
581        if( proxy )
582            dbus_g_proxy_call( proxy, "AddMetainfo", &err,
583                               G_TYPE_STRING, payload,
584                               G_TYPE_INVALID,
585                               G_TYPE_BOOLEAN, &handled,
586                               G_TYPE_INVALID );
587        if( err )
588           g_message( "err: %s", err->message );
589
590        if( proxy )
591            g_object_unref( proxy );
592        if( conn )
593            dbus_g_connection_unref( conn );
594
595        tr_free( payload );
596    }
597
598    g_free( file_contents );
599
600#endif
601    return handled;
602}
603
604gboolean
605gtr_dbus_present_window( void )
606{
607    static gboolean   success = FALSE;
608
609#ifdef HAVE_DBUS_GLIB
610    DBusGProxy *      proxy = NULL;
611    GError *          err = NULL;
612    DBusGConnection * conn;
613    if( ( conn = dbus_g_bus_get( DBUS_BUS_SESSION, &err ) ) )
614        proxy = dbus_g_proxy_new_for_name ( conn, VALUE_SERVICE_NAME,
615                                            VALUE_SERVICE_OBJECT_PATH,
616                                            VALUE_SERVICE_INTERFACE );
617    else if( err )
618        g_message( "err: %s", err->message );
619    if( proxy )
620        dbus_g_proxy_call( proxy, "PresentWindow", &err,
621                           G_TYPE_INVALID,
622                           G_TYPE_BOOLEAN, &success,
623                           G_TYPE_INVALID );
624    if( err )
625        g_message( "err: %s", err->message );
626
627    g_object_unref( proxy );
628    dbus_g_connection_unref( conn );
629#endif
630    return success;
631}
632
633GtkWidget *
634gtr_button_new_from_stock( const char * stock,
635                           const char * mnemonic )
636{
637    GtkWidget * image = gtk_image_new_from_stock( stock,
638                                                  GTK_ICON_SIZE_BUTTON );
639    GtkWidget * button = gtk_button_new_with_mnemonic( mnemonic );
640
641    gtk_button_set_image( GTK_BUTTON( button ), image );
642    return button;
643}
644
645/***
646****
647***/
648
649void
650gtr_priority_combo_set_value( GtkWidget * w, tr_priority_t value )
651{
652    int i;
653    int currentValue;
654    const int column = 0;
655    GtkTreeIter iter;
656    GtkComboBox * combobox = GTK_COMBO_BOX( w );
657    GtkTreeModel * model = gtk_combo_box_get_model( combobox );
658
659    /* do the value and current value match? */
660    if( gtk_combo_box_get_active_iter( combobox, &iter ) ) {
661        gtk_tree_model_get( model, &iter, column, &currentValue, -1 );
662        if( currentValue == value )
663            return;
664    }
665
666    /* find the one to select */
667    i = 0;
668    while(( gtk_tree_model_iter_nth_child( model, &iter, NULL, i++ ))) {
669        gtk_tree_model_get( model, &iter, column, &currentValue, -1 );
670        if( currentValue == value ) {
671            gtk_combo_box_set_active_iter( combobox, &iter );
672            return;
673        }
674    }
675}
676
677tr_priority_t
678gtr_priority_combo_get_value( GtkWidget * w )
679{
680    int value = 0;
681    GtkTreeIter iter;
682    GtkComboBox * combo_box = GTK_COMBO_BOX( w );
683
684    if( gtk_combo_box_get_active_iter( combo_box, &iter ) )
685        gtk_tree_model_get( gtk_combo_box_get_model( combo_box ), &iter, 0, &value, -1 );
686
687    return value;
688}
689
690GtkWidget *
691gtr_priority_combo_new( void )
692{
693    int i;
694    GtkWidget * w;
695    GtkCellRenderer * r;
696    GtkListStore * store;
697    const struct {
698        int value;
699        const char * text;
700    } items[] = {
701        { TR_PRI_HIGH,   N_( "High" )  },
702        { TR_PRI_NORMAL, N_( "Normal" ) },
703        { TR_PRI_LOW,    N_( "Low" )  }
704    };
705
706    store = gtk_list_store_new( 2, G_TYPE_INT, G_TYPE_STRING );
707    for( i=0; i<(int)G_N_ELEMENTS(items); ++i ) {
708        GtkTreeIter iter;
709        gtk_list_store_append( store, &iter );
710        gtk_list_store_set( store, &iter, 0, items[i].value,
711                                          1, _( items[i].text ),
712                                         -1 );
713    }
714
715    w = gtk_combo_box_new_with_model( GTK_TREE_MODEL( store ) );
716    r = gtk_cell_renderer_text_new( );
717    gtk_cell_layout_pack_start( GTK_CELL_LAYOUT( w ), r, TRUE );
718    gtk_cell_layout_set_attributes( GTK_CELL_LAYOUT( w ), r, "text", 1, NULL );
719
720    /* cleanup */
721    g_object_unref( store );
722    return w;
723}
724
725/***
726****
727***/
728
729void
730gtr_widget_set_tooltip_text( GtkWidget * w, const char * tip )
731{
732#if GTK_CHECK_VERSION( 2,12,0 )
733    gtk_widget_set_tooltip_text( w, tip );
734#else
735    static GtkTooltips * tips = NULL;
736    if( tips == NULL )
737        tips = gtk_tooltips_new( );
738    gtk_tooltips_set_tip( tips, w, tip, NULL );
739#endif
740}
741
742void
743gtr_toolbar_set_orientation( GtkToolbar      * toolbar,
744                             GtkOrientation    orientation )
745{
746#if GTK_CHECK_VERSION( 2,16,0 )
747    gtk_orientable_set_orientation( GTK_ORIENTABLE( toolbar ), orientation );
748#else
749    gtk_toolbar_set_orientation( toolbar, orientation );
750#endif
751}
752
753/***
754****
755***/
756
757#if !GTK_CHECK_VERSION( 2,12,0 )
758struct gtr_func_data
759{
760    GSourceFunc function;
761    gpointer data;
762};
763
764static void
765gtr_func_data_free( gpointer data )
766{
767#if GTK_CHECK_VERSION( 2,10,0 )
768    g_slice_free( struct gtr_func_data, data );
769#else
770    g_free( data );
771#endif
772}
773
774static struct gtr_func_data *
775gtr_func_data_new( GSourceFunc function, gpointer data )
776{
777#if GTK_CHECK_VERSION( 2,10,0 )
778    struct gtr_func_data * d = g_slice_new( struct gtr_func_data );
779#else
780    struct gtr_func_data * d = g_new( struct gtr_func_data, 1 );
781#endif
782    d->function = function;
783    d->data = data;
784    return d;
785}
786
787static gboolean
788gtr_thread_func( gpointer data )
789{
790    gboolean more;
791    struct gtr_func_data * idle_data = data;
792
793    gdk_threads_enter( );
794    more = idle_data->function( idle_data->data );
795    gdk_threads_leave( );
796
797    return more;
798}
799#endif
800
801void
802gtr_idle_add( GSourceFunc function, gpointer data )
803{
804#if GTK_CHECK_VERSION( 2,12,0 )
805    gdk_threads_add_idle( function, data );
806#else
807    g_idle_add_full( G_PRIORITY_DEFAULT,
808                     gtr_thread_func,
809                     gtr_func_data_new( function, data ),
810                     gtr_func_data_free );
811#endif
812}
813
814guint
815gtr_timeout_add_seconds( guint seconds, GSourceFunc function, gpointer data )
816{
817#if GTK_CHECK_VERSION( 2,14,0 )
818    return gdk_threads_add_timeout_seconds( seconds, function, data );
819#elif GTK_CHECK_VERSION( 2,12,0 )
820    return gdk_threads_add_timeout( seconds*1000, function, data );
821#else
822    return g_timeout_add_full( G_PRIORITY_DEFAULT,
823                               seconds * 1000,
824                               gtr_thread_func,
825                               gtr_func_data_new( function, data ),
826                               gtr_func_data_free );
827#endif
828}
829
830void
831gtr_unrecognized_url_dialog( GtkWidget * parent, const char * url )
832{
833    const char * xt = "xt=urn:btih";
834
835    GtkWindow * window = getWindow( parent );
836
837    GString * gstr = g_string_new( NULL );
838
839    GtkWidget * w = gtk_message_dialog_new( window, 0,
840                                            GTK_MESSAGE_ERROR,
841                                            GTK_BUTTONS_CLOSE,
842                                            "%s", _( "Unrecognized URL" ) );
843
844    g_string_append_printf( gstr, _( "Transmission doesn't know how to use \"%s\"" ), url );
845
846    if( gtr_is_magnet_link( url ) && ( strstr( url, xt ) == NULL ) )
847    {
848        g_string_append_printf( gstr, "\n \n" );
849        g_string_append_printf( gstr, _( "This magnet link appears to be intended for something other than BitTorrent.  BitTorrent magnet links have a section containing \"%s\"." ), xt );
850    }
851
852    gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG( w ), "%s", gstr->str );
853    g_signal_connect_swapped( w, "response", G_CALLBACK( gtk_widget_destroy ), w );
854    gtk_widget_show( w );
855    g_string_free( gstr, TRUE );
856}
Note: See TracBrowser for help on using the repository browser.