source: trunk/gtk/util.c @ 10537

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

(trunk gtk) #3174 "Use IEC standard units (KiB, MiB, GiB) instead of (KB, MB, GB)" -- don't use g_format_size_for_display(), since we can't control what units it uses

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