363 lines
9.9 KiB
C
363 lines
9.9 KiB
C
/******************************************************************************
|
|
* $Id$
|
|
*
|
|
* Copyright (c) 2006-2007 Transmission authors and contributors
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
* DEALINGS IN THE SOFTWARE.
|
|
*****************************************************************************/
|
|
|
|
#include <gtk/gtk.h>
|
|
#include <glib/gi18n.h>
|
|
|
|
#include "tr_icon.h"
|
|
#include "util.h"
|
|
|
|
#define ITEM_ACTION "tr-icon-item-action"
|
|
|
|
enum
|
|
{
|
|
PROP_ICON = 1,
|
|
PROP_DOCKED,
|
|
PROP_CLICK,
|
|
};
|
|
|
|
static void
|
|
tr_icon_init( GTypeInstance * instance, gpointer g_class );
|
|
static void
|
|
tr_icon_set_property( GObject * object, guint property_id,
|
|
const GValue * value, GParamSpec * pspec );
|
|
static void
|
|
tr_icon_get_property( GObject * object, guint property_id,
|
|
GValue * value, GParamSpec * pspec);
|
|
static void
|
|
tr_icon_class_init( gpointer g_class, gpointer g_class_data );
|
|
static void
|
|
tr_icon_dispose( GObject * obj );
|
|
static void
|
|
itemclick( GObject * obj, gpointer data );
|
|
static void
|
|
emitaction( TrIcon * self, int id );
|
|
#ifdef TR_ICON_SUPPORTED
|
|
static void
|
|
clicked( TrIcon * self, gpointer data );
|
|
static void
|
|
popup( TrIcon * self, guint button, guint when, gpointer data );
|
|
#endif
|
|
|
|
GType
|
|
tr_icon_get_type( void )
|
|
{
|
|
static GType type = 0;
|
|
|
|
if( 0 == type )
|
|
{
|
|
static const GTypeInfo info =
|
|
{
|
|
sizeof( TrIconClass ),
|
|
NULL, /* base_init */
|
|
NULL, /* base_finalize */
|
|
tr_icon_class_init, /* class_init */
|
|
NULL, /* class_finalize */
|
|
NULL, /* class_data */
|
|
sizeof( TrIcon ),
|
|
0, /* n_preallocs */
|
|
tr_icon_init, /* instance_init */
|
|
NULL,
|
|
};
|
|
#ifdef TR_ICON_SUPPORTED
|
|
type = GTK_TYPE_STATUS_ICON;
|
|
#else
|
|
type = G_TYPE_OBJECT;
|
|
#endif
|
|
type = g_type_register_static( type, "TrIcon", &info, 0 );
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
static void
|
|
tr_icon_class_init( gpointer g_class, gpointer g_class_data SHUTUP )
|
|
{
|
|
GObjectClass * gobject_class;
|
|
TrIconClass * tricon_class;
|
|
GParamSpec * pspec;
|
|
|
|
gobject_class = G_OBJECT_CLASS( g_class );
|
|
gobject_class->set_property = tr_icon_set_property;
|
|
gobject_class->get_property = tr_icon_get_property;
|
|
gobject_class->dispose = tr_icon_dispose;
|
|
|
|
pspec = g_param_spec_boolean( "icon", "Icon",
|
|
"Icon has been set from default window icon.",
|
|
TRUE, G_PARAM_CONSTRUCT|G_PARAM_READWRITE );
|
|
g_object_class_install_property( gobject_class, PROP_ICON, pspec );
|
|
|
|
pspec = g_param_spec_boolean( "docked", "Docked",
|
|
"Icon is docked in a system tray.",
|
|
FALSE, G_PARAM_READABLE );
|
|
g_object_class_install_property( gobject_class, PROP_DOCKED, pspec );
|
|
|
|
pspec = g_param_spec_int( "activate-action", "Activate action",
|
|
"The action id to signal when icon is activated.",
|
|
G_MININT, G_MAXINT, -1, G_PARAM_READWRITE );
|
|
g_object_class_install_property( gobject_class, PROP_CLICK, pspec );
|
|
|
|
tricon_class = TR_ICON_CLASS( g_class );
|
|
tricon_class->actionsig =
|
|
g_signal_new( "action", G_TYPE_FROM_CLASS( g_class ),
|
|
G_SIGNAL_RUN_LAST, 0, NULL, NULL,
|
|
g_cclosure_marshal_VOID__INT,
|
|
G_TYPE_NONE, 1, G_TYPE_INT );
|
|
}
|
|
|
|
static void
|
|
tr_icon_init( GTypeInstance * instance, gpointer g_class SHUTUP )
|
|
{
|
|
TrIcon * self = ( TrIcon * )instance;
|
|
|
|
self->clickact = -1;
|
|
self->menu = NULL;
|
|
self->actions = NULL;
|
|
self->disposed = FALSE;
|
|
|
|
#ifdef TR_ICON_SUPPORTED
|
|
self->menu = gtk_menu_new();
|
|
gtk_widget_show( self->menu );
|
|
g_signal_connect( self, "activate", G_CALLBACK( clicked ), NULL );
|
|
g_signal_connect( self, "popup-menu", G_CALLBACK( popup ), NULL );
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
tr_icon_set_property( GObject * object, guint property_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
TrIcon * self = ( TrIcon * )object;
|
|
|
|
if( self->disposed )
|
|
{
|
|
return;
|
|
}
|
|
|
|
switch( property_id )
|
|
{
|
|
case PROP_ICON:
|
|
#ifdef TR_ICON_SUPPORTED
|
|
if( g_value_get_boolean( value ) )
|
|
{
|
|
GList * icons = gtk_window_get_default_icon_list();
|
|
if( NULL != icons && NULL != icons->data )
|
|
{
|
|
gtk_status_icon_set_from_pixbuf( GTK_STATUS_ICON( self ),
|
|
icons->data );
|
|
}
|
|
g_list_free( icons );
|
|
}
|
|
else
|
|
{
|
|
gtk_status_icon_set_from_pixbuf( GTK_STATUS_ICON( self ),
|
|
NULL );
|
|
}
|
|
#endif
|
|
break;
|
|
case PROP_CLICK:
|
|
self->clickact = g_value_get_int( value );
|
|
break;
|
|
case PROP_DOCKED:
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID( object, property_id, pspec );
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
tr_icon_get_property( GObject * object, guint property_id,
|
|
GValue * value, GParamSpec * pspec )
|
|
{
|
|
TrIcon * self = ( TrIcon * )object;
|
|
#ifdef TR_ICON_SUPPORTED
|
|
GtkStatusIcon * icon;
|
|
#endif
|
|
|
|
if( self->disposed )
|
|
{
|
|
return;
|
|
}
|
|
|
|
switch( property_id )
|
|
{
|
|
case PROP_ICON:
|
|
#ifdef TR_ICON_SUPPORTED
|
|
icon = GTK_STATUS_ICON( self );
|
|
if( GTK_IMAGE_PIXBUF == gtk_status_icon_get_storage_type( icon ) &&
|
|
NULL != gtk_status_icon_get_pixbuf( icon ) )
|
|
{
|
|
g_value_set_boolean( value, TRUE );
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
g_value_set_boolean( value, FALSE );
|
|
}
|
|
break;
|
|
case PROP_DOCKED:
|
|
#ifdef TR_ICON_SUPPORTED
|
|
if( gtk_status_icon_is_embedded( GTK_STATUS_ICON( self ) ) )
|
|
{
|
|
g_value_set_boolean( value, TRUE );
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
g_value_set_boolean( value, FALSE );
|
|
}
|
|
break;
|
|
case PROP_CLICK:
|
|
g_value_set_int( value, self->clickact );
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID( object, property_id, pspec );
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
tr_icon_dispose( GObject * obj )
|
|
{
|
|
TrIcon * self = ( TrIcon * )obj;
|
|
GObjectClass * parent;
|
|
|
|
if( self->disposed )
|
|
{
|
|
return;
|
|
}
|
|
self->disposed = TRUE;
|
|
|
|
g_list_foreach( self->actions, ( GFunc )action_free, NULL );
|
|
g_list_free( self->actions );
|
|
|
|
/* Chain up to the parent class */
|
|
parent = g_type_class_peek( g_type_parent( TR_ICON_TYPE ) );
|
|
parent->dispose( obj );
|
|
}
|
|
|
|
TrIcon *
|
|
tr_icon_new( void )
|
|
{
|
|
return g_object_new( TR_ICON_TYPE, NULL );
|
|
}
|
|
|
|
gboolean
|
|
tr_icon_docked( TrIcon * self )
|
|
{
|
|
gboolean ret;
|
|
|
|
g_object_get( self, "docked", &ret, NULL );
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
tr_icon_action_add( TrIcon * self, int id, int flags, const char * name,
|
|
const char * icon )
|
|
{
|
|
struct action * act;
|
|
GtkWidget * item;
|
|
|
|
TR_IS_ICON( self );
|
|
if( self->disposed )
|
|
{
|
|
return;
|
|
}
|
|
|
|
act = action_new( id, flags, name, icon );
|
|
|
|
if( NULL != self->menu )
|
|
{
|
|
if( ACTF_MENU & flags && ACTF_ALWAYS & flags )
|
|
{
|
|
item = action_makemenu( act, ITEM_ACTION, NULL, NULL, 0,
|
|
G_CALLBACK( itemclick ), self );
|
|
gtk_menu_shell_append( GTK_MENU_SHELL( self->menu ), item );
|
|
act->menu = item;
|
|
}
|
|
else if( ACTF_SEPARATOR & flags )
|
|
{
|
|
item = gtk_separator_menu_item_new();
|
|
gtk_widget_show( item );
|
|
gtk_menu_shell_append( GTK_MENU_SHELL( self->menu ), item );
|
|
}
|
|
}
|
|
|
|
self->actions = g_list_append( self->actions, act );
|
|
}
|
|
|
|
static void
|
|
itemclick( GObject * obj, gpointer data )
|
|
{
|
|
TrIcon * self;
|
|
struct action * act;
|
|
|
|
TR_IS_ICON( data );
|
|
self = TR_ICON( data );
|
|
act = g_object_get_data( obj, ITEM_ACTION );
|
|
|
|
emitaction( self, act->id );
|
|
}
|
|
|
|
static void
|
|
emitaction( TrIcon * self, int id )
|
|
{
|
|
TrIconClass * class;
|
|
|
|
class = g_type_class_peek( TR_ICON_TYPE );
|
|
g_signal_emit( self, class->actionsig, 0, id );
|
|
}
|
|
|
|
#ifdef TR_ICON_SUPPORTED
|
|
|
|
static void
|
|
clicked( TrIcon * self, gpointer data SHUTUP )
|
|
{
|
|
TR_IS_ICON( self );
|
|
|
|
if( self->disposed || 0 > self->clickact )
|
|
{
|
|
return;
|
|
}
|
|
|
|
emitaction( self, self->clickact );
|
|
}
|
|
|
|
static void
|
|
popup( TrIcon * self, guint button, guint when, gpointer data SHUTUP )
|
|
{
|
|
TR_IS_ICON( self );
|
|
|
|
if( self->disposed )
|
|
{
|
|
return;
|
|
}
|
|
|
|
gtk_menu_popup( GTK_MENU( self->menu ), NULL, NULL,
|
|
gtk_status_icon_position_menu, self, button, when );
|
|
}
|
|
|
|
#endif /* TR_ICON_SUPPORTED */
|