source: trunk/gtk/util.c @ 10512

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

(trunk gtk) #3166 "Add Tracker Favicons to Tracker Tab in Properties Window." -- implemented in trunk for 2.00

  • Property svn:keywords set to Date Rev Author Id
File size: 23.0 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 10512 2010-04-23 01:46:02Z 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/* pattern-matching text; ie, legaltorrents.com */
335char*
336gtr_get_host_from_url( const char * url )
337{
338    char * h = NULL;
339    char * name;
340    const char * first_dot;
341    const char * last_dot;
342
343    tr_urlParse( url, -1, NULL, &h, NULL, NULL );
344    first_dot = strchr( h, '.' );
345    last_dot = strrchr( h, '.' );
346
347    if( ( first_dot ) && ( last_dot ) && ( first_dot != last_dot ) )
348        name = g_strdup( first_dot + 1 );
349    else
350        name = g_strdup( h );
351
352    tr_free( h );
353    return name;
354}
355
356gboolean
357gtr_is_supported_url( const char * str )
358{
359    return !strncmp( str, "ftp://", 6 )
360        || !strncmp( str, "http://", 7 )
361        || !strncmp( str, "https://", 8 );
362}
363
364gboolean
365gtr_is_magnet_link( const char * str )
366{
367    return !strncmp( str, "magnet:?", 8 );
368}
369
370gboolean
371gtr_is_hex_hashcode( const char * str )
372{
373    int i;
374
375    if( !str || ( strlen( str ) != 40 ) )
376        return FALSE;
377
378    for( i=0; i<40; ++i )
379        if( !isxdigit( str[i] ) )
380            return FALSE;
381
382    return TRUE;
383}
384
385static GtkWindow *
386getWindow( GtkWidget * w )
387{
388    if( w == NULL )
389        return NULL;
390
391    if( GTK_IS_WINDOW( w ) )
392        return GTK_WINDOW( w );
393
394    return GTK_WINDOW( gtk_widget_get_ancestor( w, GTK_TYPE_WINDOW ) );
395}
396
397void
398addTorrentErrorDialog( GtkWidget *  child,
399                       int          err,
400                       const char * filename )
401{
402    char * secondary;
403    const char * fmt;
404    GtkWidget * w;
405    GtkWindow * win = getWindow( child );
406
407    switch( err )
408    {
409        case TR_PARSE_ERR: fmt = _( "The torrent file \"%s\" contains invalid data." ); break;
410        case TR_PARSE_DUPLICATE: fmt = _( "The torrent file \"%s\" is already in use." ); break;
411        default: fmt = _( "The torrent file \"%s\" encountered an unknown error." ); break;
412    }
413    secondary = g_strdup_printf( fmt, filename );
414
415    w = gtk_message_dialog_new( win,
416                                GTK_DIALOG_DESTROY_WITH_PARENT,
417                                GTK_MESSAGE_ERROR,
418                                GTK_BUTTONS_CLOSE,
419                                "%s", _( "Error opening torrent" ) );
420    gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG( w ),
421                                              "%s", secondary );
422    g_signal_connect_swapped( w, "response",
423                              G_CALLBACK( gtk_widget_destroy ), w );
424    gtk_widget_show_all( w );
425    g_free( secondary );
426}
427
428typedef void ( PopupFunc )( GtkWidget*, GdkEventButton* );
429
430/* pop up the context menu if a user right-clicks.
431   if the row they right-click on isn't selected, select it. */
432
433gboolean
434on_tree_view_button_pressed( GtkWidget *      view,
435                             GdkEventButton * event,
436                             gpointer         func )
437{
438    GtkTreeView * tv = GTK_TREE_VIEW( view );
439
440    if( event->type == GDK_BUTTON_PRESS  &&  event->button == 3 )
441    {
442        GtkTreeSelection * selection = gtk_tree_view_get_selection( tv );
443        GtkTreePath *      path;
444        if( gtk_tree_view_get_path_at_pos ( tv,
445                                            (gint) event->x,
446                                            (gint) event->y,
447                                            &path, NULL, NULL, NULL ) )
448        {
449            if( !gtk_tree_selection_path_is_selected ( selection, path ) )
450            {
451                gtk_tree_selection_unselect_all ( selection );
452                gtk_tree_selection_select_path ( selection, path );
453            }
454            gtk_tree_path_free( path );
455        }
456
457        if( func != NULL )
458            ( (PopupFunc*)func )( view, event );
459
460        return TRUE;
461    }
462
463    return FALSE;
464}
465
466/* if the user clicked in an empty area of the list,
467 * clear all the selections. */
468gboolean
469on_tree_view_button_released( GtkWidget *      view,
470                              GdkEventButton * event,
471                              gpointer         unused UNUSED )
472{
473    GtkTreeView * tv = GTK_TREE_VIEW( view );
474
475    if( !gtk_tree_view_get_path_at_pos ( tv,
476                                         (gint) event->x,
477                                         (gint) event->y,
478                                         NULL, NULL, NULL, NULL ) )
479    {
480        GtkTreeSelection * selection = gtk_tree_view_get_selection( tv );
481        gtk_tree_selection_unselect_all ( selection );
482    }
483
484    return FALSE;
485}
486
487gpointer
488gtr_object_ref_sink( gpointer object )
489{
490#if GLIB_CHECK_VERSION( 2, 10, 0 )
491    g_object_ref_sink( object );
492#else
493    g_object_ref( object );
494    gtk_object_sink( GTK_OBJECT( object ) );
495#endif
496    return object;
497}
498
499int
500gtr_file_trash_or_remove( const char * filename )
501{
502    if( filename && *filename )
503    {
504        gboolean trashed = FALSE;
505#ifdef HAVE_GIO
506        GError * err = NULL;
507        GFile *  file = g_file_new_for_path( filename );
508        trashed = g_file_trash( file, NULL, &err );
509        if( err )
510            g_message( "Unable to trash file \"%s\": %s", filename, err->message );
511        g_clear_error( &err );
512        g_object_unref( G_OBJECT( file ) );
513#endif
514
515        if( !trashed && g_remove( filename ) )
516        {
517            const int err = errno;
518            g_message( "Unable to remove file \"%s\": %s", filename, g_strerror( err ) );
519        }
520    }
521
522    return 0;
523}
524
525char*
526gtr_get_help_url( void )
527{
528    const char * fmt = "http://www.transmissionbt.com/help/gtk/%d.%dx";
529    int          major, minor;
530
531    sscanf( SHORT_VERSION_STRING, "%d.%d", &major, &minor );
532    return g_strdup_printf( fmt, major, minor / 10 );
533}
534
535void
536gtr_open_file( const char * path )
537{
538    if( path )
539    {
540        gboolean opened = FALSE;
541#ifdef HAVE_GIO
542        if( !opened )
543        {
544            GFile * file = g_file_new_for_path( path );
545            char *  uri = g_file_get_uri( file );
546            opened = g_app_info_launch_default_for_uri( uri, NULL, NULL );
547            g_free( uri );
548            g_object_unref( G_OBJECT( file ) );
549        }
550#endif
551        if( !opened )
552        {
553            char * argv[] = { (char*)"xdg-open", (char*)path, NULL };
554            opened = g_spawn_async( NULL, argv, NULL, G_SPAWN_SEARCH_PATH,
555                                    NULL, NULL, NULL, NULL );
556        }
557
558        if( !opened )
559        {
560            g_message( "Unable to open \"%s\"", path );
561        }
562    }
563}
564
565#define VALUE_SERVICE_NAME        "com.transmissionbt.Transmission"
566#define VALUE_SERVICE_OBJECT_PATH "/com/transmissionbt/Transmission"
567#define VALUE_SERVICE_INTERFACE   "com.transmissionbt.Transmission"
568
569gboolean
570gtr_dbus_add_torrent( const char * filename )
571{
572    /* FIXME: why is this static? */
573    static gboolean handled = FALSE;
574
575#ifdef HAVE_DBUS_GLIB
576    char * payload;
577    gsize file_length;
578    char * file_contents = NULL;
579
580    /* If it's a file, load its contents and send them over the wire...
581     * it might be a temporary file that's going to disappear. */
582    if( g_file_get_contents( filename, &file_contents, &file_length, NULL ) )
583        payload = tr_base64_encode( file_contents, file_length, NULL );
584    else if( gtr_is_supported_url( filename ) || gtr_is_magnet_link( filename ) )
585        payload = tr_strdup( filename );
586    else
587        payload = NULL;
588
589    if( payload != NULL )
590    {
591        GError * err = NULL;
592        DBusGConnection * conn;
593        DBusGProxy * proxy = NULL;
594
595        if(( conn = dbus_g_bus_get( DBUS_BUS_SESSION, &err )))
596            proxy = dbus_g_proxy_new_for_name (conn, VALUE_SERVICE_NAME,
597                                                     VALUE_SERVICE_OBJECT_PATH,
598                                                     VALUE_SERVICE_INTERFACE );
599        else if( err )
600           g_message( "err: %s", err->message );
601
602        if( proxy )
603            dbus_g_proxy_call( proxy, "AddMetainfo", &err,
604                               G_TYPE_STRING, payload,
605                               G_TYPE_INVALID,
606                               G_TYPE_BOOLEAN, &handled,
607                               G_TYPE_INVALID );
608        if( err )
609           g_message( "err: %s", err->message );
610
611        if( proxy )
612            g_object_unref( proxy );
613        if( conn )
614            dbus_g_connection_unref( conn );
615
616        tr_free( payload );
617    }
618
619    g_free( file_contents );
620
621#endif
622    return handled;
623}
624
625gboolean
626gtr_dbus_present_window( void )
627{
628    static gboolean   success = FALSE;
629
630#ifdef HAVE_DBUS_GLIB
631    DBusGProxy *      proxy = NULL;
632    GError *          err = NULL;
633    DBusGConnection * conn;
634    if( ( conn = dbus_g_bus_get( DBUS_BUS_SESSION, &err ) ) )
635        proxy = dbus_g_proxy_new_for_name ( conn, VALUE_SERVICE_NAME,
636                                            VALUE_SERVICE_OBJECT_PATH,
637                                            VALUE_SERVICE_INTERFACE );
638    else if( err )
639        g_message( "err: %s", err->message );
640    if( proxy )
641        dbus_g_proxy_call( proxy, "PresentWindow", &err,
642                           G_TYPE_INVALID,
643                           G_TYPE_BOOLEAN, &success,
644                           G_TYPE_INVALID );
645    if( err )
646        g_message( "err: %s", err->message );
647
648    g_object_unref( proxy );
649    dbus_g_connection_unref( conn );
650#endif
651    return success;
652}
653
654GtkWidget *
655gtr_button_new_from_stock( const char * stock,
656                           const char * mnemonic )
657{
658    GtkWidget * image = gtk_image_new_from_stock( stock,
659                                                  GTK_ICON_SIZE_BUTTON );
660    GtkWidget * button = gtk_button_new_with_mnemonic( mnemonic );
661
662    gtk_button_set_image( GTK_BUTTON( button ), image );
663    return button;
664}
665
666/***
667****
668***/
669
670void
671gtr_priority_combo_set_value( GtkWidget * w, tr_priority_t value )
672{
673    int i;
674    int currentValue;
675    const int column = 0;
676    GtkTreeIter iter;
677    GtkComboBox * combobox = GTK_COMBO_BOX( w );
678    GtkTreeModel * model = gtk_combo_box_get_model( combobox );
679
680    /* do the value and current value match? */
681    if( gtk_combo_box_get_active_iter( combobox, &iter ) ) {
682        gtk_tree_model_get( model, &iter, column, &currentValue, -1 );
683        if( currentValue == value )
684            return;
685    }
686
687    /* find the one to select */
688    i = 0;
689    while(( gtk_tree_model_iter_nth_child( model, &iter, NULL, i++ ))) {
690        gtk_tree_model_get( model, &iter, column, &currentValue, -1 );
691        if( currentValue == value ) {
692            gtk_combo_box_set_active_iter( combobox, &iter );
693            return;
694        }
695    }
696}
697
698tr_priority_t
699gtr_priority_combo_get_value( GtkWidget * w )
700{
701    int value = 0;
702    GtkTreeIter iter;
703    GtkComboBox * combo_box = GTK_COMBO_BOX( w );
704
705    if( gtk_combo_box_get_active_iter( combo_box, &iter ) )
706        gtk_tree_model_get( gtk_combo_box_get_model( combo_box ), &iter, 0, &value, -1 );
707
708    return value;
709}
710
711GtkWidget *
712gtr_priority_combo_new( void )
713{
714    int i;
715    GtkWidget * w;
716    GtkCellRenderer * r;
717    GtkListStore * store;
718    const struct {
719        int value;
720        const char * text;
721    } items[] = {
722        { TR_PRI_HIGH,   N_( "High" )  },
723        { TR_PRI_NORMAL, N_( "Normal" ) },
724        { TR_PRI_LOW,    N_( "Low" )  }
725    };
726
727    store = gtk_list_store_new( 2, G_TYPE_INT, G_TYPE_STRING );
728    for( i=0; i<(int)G_N_ELEMENTS(items); ++i ) {
729        GtkTreeIter iter;
730        gtk_list_store_append( store, &iter );
731        gtk_list_store_set( store, &iter, 0, items[i].value,
732                                          1, _( items[i].text ),
733                                         -1 );
734    }
735
736    w = gtk_combo_box_new_with_model( GTK_TREE_MODEL( store ) );
737    r = gtk_cell_renderer_text_new( );
738    gtk_cell_layout_pack_start( GTK_CELL_LAYOUT( w ), r, TRUE );
739    gtk_cell_layout_set_attributes( GTK_CELL_LAYOUT( w ), r, "text", 1, NULL );
740
741    /* cleanup */
742    g_object_unref( store );
743    return w;
744}
745
746/***
747****
748***/
749
750void
751gtr_widget_set_tooltip_text( GtkWidget * w, const char * tip )
752{
753#if GTK_CHECK_VERSION( 2,12,0 )
754    gtk_widget_set_tooltip_text( w, tip );
755#else
756    static GtkTooltips * tips = NULL;
757    if( tips == NULL )
758        tips = gtk_tooltips_new( );
759    gtk_tooltips_set_tip( tips, w, tip, NULL );
760#endif
761}
762
763void
764gtr_toolbar_set_orientation( GtkToolbar      * toolbar,
765                             GtkOrientation    orientation )
766{
767#if GTK_CHECK_VERSION( 2,16,0 )
768    gtk_orientable_set_orientation( GTK_ORIENTABLE( toolbar ), orientation );
769#else
770    gtk_toolbar_set_orientation( toolbar, orientation );
771#endif
772}
773
774/***
775****
776***/
777
778#if !GTK_CHECK_VERSION( 2,12,0 )
779struct gtr_func_data
780{
781    GSourceFunc function;
782    gpointer data;
783};
784
785static void
786gtr_func_data_free( gpointer data )
787{
788#if GTK_CHECK_VERSION( 2,10,0 )
789    g_slice_free( struct gtr_func_data, data );
790#else
791    g_free( data );
792#endif
793}
794
795static struct gtr_func_data *
796gtr_func_data_new( GSourceFunc function, gpointer data )
797{
798#if GTK_CHECK_VERSION( 2,10,0 )
799    struct gtr_func_data * d = g_slice_new( struct gtr_func_data );
800#else
801    struct gtr_func_data * d = g_new( struct gtr_func_data, 1 );
802#endif
803    d->function = function;
804    d->data = data;
805    return d;
806}
807
808static gboolean
809gtr_thread_func( gpointer data )
810{
811    gboolean more;
812    struct gtr_func_data * idle_data = data;
813
814    gdk_threads_enter( );
815    more = idle_data->function( idle_data->data );
816    gdk_threads_leave( );
817
818    return more;
819}
820#endif
821
822void
823gtr_idle_add( GSourceFunc function, gpointer data )
824{
825#if GTK_CHECK_VERSION( 2,12,0 )
826    gdk_threads_add_idle( function, data );
827#else
828    g_idle_add_full( G_PRIORITY_DEFAULT,
829                     gtr_thread_func,
830                     gtr_func_data_new( function, data ),
831                     gtr_func_data_free );
832#endif
833}
834
835guint
836gtr_timeout_add_seconds( guint seconds, GSourceFunc function, gpointer data )
837{
838#if GTK_CHECK_VERSION( 2,14,0 )
839    return gdk_threads_add_timeout_seconds( seconds, function, data );
840#elif GTK_CHECK_VERSION( 2,12,0 )
841    return gdk_threads_add_timeout( seconds*1000, function, data );
842#else
843    return g_timeout_add_full( G_PRIORITY_DEFAULT,
844                               seconds * 1000,
845                               gtr_thread_func,
846                               gtr_func_data_new( function, data ),
847                               gtr_func_data_free );
848#endif
849}
850
851void
852gtr_unrecognized_url_dialog( GtkWidget * parent, const char * url )
853{
854    const char * xt = "xt=urn:btih";
855
856    GtkWindow * window = getWindow( parent );
857
858    GString * gstr = g_string_new( NULL );
859
860    GtkWidget * w = gtk_message_dialog_new( window, 0,
861                                            GTK_MESSAGE_ERROR,
862                                            GTK_BUTTONS_CLOSE,
863                                            "%s", _( "Unrecognized URL" ) );
864
865    g_string_append_printf( gstr, _( "Transmission doesn't know how to use \"%s\"" ), url );
866
867    if( gtr_is_magnet_link( url ) && ( strstr( url, xt ) == NULL ) )
868    {
869        g_string_append_printf( gstr, "\n \n" );
870        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 );
871    }
872
873    gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG( w ), "%s", gstr->str );
874    g_signal_connect_swapped( w, "response", G_CALLBACK( gtk_widget_destroy ), w );
875    gtk_widget_show( w );
876    g_string_free( gstr, TRUE );
877}
Note: See TracBrowser for help on using the repository browser.