aboutsummaryrefslogblamecommitdiffstats
path: root/shell/e-shell-taskbar.c
blob: 0910ea98d696c1a6d3ba728e24c06cadac4dea18 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11

                    
  


                                                                           
  



                                                                             
  

                                                                           


                                                        
  

   





                                                    



                    

                            
                                    
 
                         
 



                                                            



                                                



                                                        

                         

                                

                          



                                                                          



               
                     


                       
                         




                                         
 

















                                                                   
                                                       




























                                                                                          
           

                                                              
 


                                
                                                       

                                                                          

                                         



                                                                            
 


                                                                        
 


                                                                     

                                                                 







                                                         










                                                            
 

                                                      




                                                       
 



                                                                        
 


                                                             
 



                                                           














                                                                                   
                                                     









                                                                        


                                                           




                                                                 








                                                             


           


















                                                                   





                                                            
















                                                                       





                                                                    














                                                                       
                                                    
 



                                                  





                                                                        



                                                                 

         


                                              

                                                    
                                                                        


           



                                        
                                                    



                                                     
                                                                         





                                           
                                     
                                     


                                                                    
                                                                    
 



                                                                          
                                  
                                                
                                                                             
 
                                                             
 

                                                                            


           


































                                                                    













                                                                                  
                                                      

                                   
                                     
 





                                                                        


                                                              


                                                                                
                                                                              
 




                                                 








                                           

                                                 
 




                                                               








                                           

                                                 


           
                                                   

                          
 
                                                                          
                                                                         
                                                            









                                                                            
                                                             


                                                                            

 







                                                                  








                                                                      







                                                                    







                                                                        









                                                                         











                                                                        








                                                                            








                                                              

                                                        
                                                


                                        

                                                              

 






                                                   






                                                              















                                                                     
/*
 * e-shell-taskbar.c
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
 * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program; if not, see <http://www.gnu.org/licenses/>.
 *
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 *
 */

/**
 * SECTION: e-shell-taskbar
 * @short_description: the bottom of the main window
 * @include: shell/e-shell-taskbar.h
 **/

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include "e-shell-taskbar.h"

#include <libebackend/libebackend.h>

#include <e-shell-view.h>

#define E_SHELL_TASKBAR_GET_PRIVATE(obj) \
    (G_TYPE_INSTANCE_GET_PRIVATE \
    ((obj), E_TYPE_SHELL_TASKBAR, EShellTaskbarPrivate))

struct _EShellTaskbarPrivate {

    gpointer shell_view;  /* weak pointer */

    /* Keep a reference to the shell backend since
     * we connect to its "activity-added" signal. */
    EShellBackend *shell_backend;

    GtkWidget *label;
    GtkWidget *hbox;

    GHashTable *proxy_table;

    gint fixed_height;

    /* Basically the main() thread, aka UI thread, where the gtk calls
     * can be done. */
    GThread *main_thread;
};

enum {
    PROP_0,
    PROP_MESSAGE,
    PROP_SHELL_VIEW
};

G_DEFINE_TYPE_WITH_CODE (
    EShellTaskbar,
    e_shell_taskbar,
    GTK_TYPE_HBOX,
    G_IMPLEMENT_INTERFACE (
        E_TYPE_EXTENSIBLE, NULL))

typedef struct {
    EShellTaskbar *shell_taskbar;
    EActivity *activity;
    GtkWidget *proxy; /* not referenced */
} EShellTaskbarIdleData;

static void
shell_taskbar_free_idle_data (gpointer data)
{
    EShellTaskbarIdleData *idle_data = data;

    g_clear_object (&idle_data->shell_taskbar);
    g_clear_object (&idle_data->activity);
    g_free (idle_data);
}

static void
shell_taskbar_remove_proxy_container (EShellTaskbar *shell_taskbar,
                                      GtkWidget *proxy)
{
    GList *children;
    GtkContainer *container;

    container = GTK_CONTAINER (shell_taskbar->priv->hbox);
    gtk_container_remove (container, proxy);

    children = gtk_container_get_children (container);

    if (children == NULL)
        gtk_widget_hide (GTK_WIDGET (container));

    g_list_free (children);
}

static gboolean
shell_taskbar_remove_proxy_container_idle_cb (gpointer user_data)
{
    EShellTaskbarIdleData *idle_data = user_data;

    g_return_val_if_fail (idle_data != NULL, FALSE);
    g_return_val_if_fail (idle_data->shell_taskbar != NULL, FALSE);
    g_return_val_if_fail (idle_data->proxy != NULL, FALSE);

    shell_taskbar_remove_proxy_container (idle_data->shell_taskbar, idle_data->proxy);

    return FALSE;
}

static void
shell_taskbar_weak_notify_cb (EShellTaskbar *shell_taskbar,
                              GObject *where_the_activity_was)
{
    GtkWidget *proxy;
    GHashTable *proxy_table;

    proxy_table = shell_taskbar->priv->proxy_table;
    proxy = g_hash_table_lookup (proxy_table, where_the_activity_was);
    g_hash_table_remove (proxy_table, where_the_activity_was);
    g_return_if_fail (proxy != NULL);

    if (shell_taskbar->priv->main_thread == g_thread_self ()) {
        shell_taskbar_remove_proxy_container (shell_taskbar, proxy);
    } else {
        EShellTaskbarIdleData *idle_data;

        idle_data = g_new0 (EShellTaskbarIdleData, 1);
        idle_data->shell_taskbar = g_object_ref (shell_taskbar);
        idle_data->proxy = proxy;

        g_idle_add_full (
            G_PRIORITY_DEFAULT,
            shell_taskbar_remove_proxy_container_idle_cb,
            idle_data, shell_taskbar_free_idle_data);
    }
}

static void
shell_taskbar_activity_add (EShellTaskbar *shell_taskbar,
                            EActivity *activity)
{
    GtkBox *box;
    GtkWidget *proxy;
    EActivityState state;
    GHashTable *proxy_table;

    /* Sanity check the activity state. */
    state = e_activity_get_state (activity);
    g_return_if_fail (state == E_ACTIVITY_RUNNING);

    /* Make sure it hasn't already been added. */
    proxy_table = shell_taskbar->priv->proxy_table;
    proxy = g_hash_table_lookup (proxy_table, activity);
    g_return_if_fail (proxy == NULL);

    /* Proxy widgets manage their own visibility.
     * Don't call gtk_widget_show() on it here. */
    proxy = e_activity_proxy_new (activity);
    box = GTK_BOX (shell_taskbar->priv->hbox);
    gtk_box_pack_start (box, proxy, TRUE, TRUE, 0);
    gtk_box_reorder_child (box, proxy, 0);
    gtk_widget_show (GTK_WIDGET (box));

    /* The proxy widget also holds a weak reference to the activity,
     * so the activity should get finalized in the normal course of
     * operation.  When that happens we remove the corresponding
     * proxy widget from the taskbar. */

    g_object_weak_ref (
        G_OBJECT (activity), (GWeakNotify)
        shell_taskbar_weak_notify_cb, shell_taskbar);

    g_hash_table_insert (proxy_table, activity, proxy);
}

static gboolean
shell_taskbar_add_activity_idle_cb (gpointer user_data)
{
    EShellTaskbarIdleData *idle_data = user_data;

    g_return_val_if_fail (idle_data != NULL, FALSE);
    g_return_val_if_fail (idle_data->shell_taskbar != NULL, FALSE);
    g_return_val_if_fail (idle_data->activity != NULL, FALSE);

    shell_taskbar_activity_add (idle_data->shell_taskbar, idle_data->activity);

    return FALSE;
}

static void
shell_taskbar_activity_added_cb (EShellTaskbar *shell_taskbar,
                                 EActivity *activity)
{
    if (shell_taskbar->priv->main_thread == g_thread_self ()) {
        shell_taskbar_activity_add (shell_taskbar, activity);
    } else {
        EShellTaskbarIdleData *idle_data;

        idle_data = g_new0 (EShellTaskbarIdleData, 1);
        idle_data->shell_taskbar = g_object_ref (shell_taskbar);
        idle_data->activity = g_object_ref (activity);

        g_idle_add_full (
            G_PRIORITY_DEFAULT,
            shell_taskbar_add_activity_idle_cb,
            idle_data, shell_taskbar_free_idle_data);
    }
}

static gboolean
shell_taskbar_weak_unref (EActivity *activity,
                          EActivityProxy *proxy,
                          EShellTaskbar *shell_taskbar)
{
    g_object_weak_unref (
        G_OBJECT (activity), (GWeakNotify)
        shell_taskbar_weak_notify_cb, shell_taskbar);

    return TRUE;
}

static void
shell_taskbar_set_shell_view (EShellTaskbar *shell_taskbar,
                              EShellView *shell_view)
{
    g_return_if_fail (shell_taskbar->priv->shell_view == NULL);

    shell_taskbar->priv->shell_view = shell_view;

    g_object_add_weak_pointer (
        G_OBJECT (shell_view),
        &shell_taskbar->priv->shell_view);
}

static void
shell_taskbar_set_property (GObject *object,
                            guint property_id,
                            const GValue *value,
                            GParamSpec *pspec)
{
    switch (property_id) {
        case PROP_MESSAGE:
            e_shell_taskbar_set_message (
                E_SHELL_TASKBAR (object),
                g_value_get_string (value));
            return;

        case PROP_SHELL_VIEW:
            shell_taskbar_set_shell_view (
                E_SHELL_TASKBAR (object),
                g_value_get_object (value));
            return;
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
shell_taskbar_get_property (GObject *object,
                            guint property_id,
                            GValue *value,
                            GParamSpec *pspec)
{
    switch (property_id) {
        case PROP_MESSAGE:
            g_value_set_string (
                value, e_shell_taskbar_get_message (
                E_SHELL_TASKBAR (object)));
            return;

        case PROP_SHELL_VIEW:
            g_value_set_object (
                value, e_shell_taskbar_get_shell_view (
                E_SHELL_TASKBAR (object)));
            return;
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
shell_taskbar_dispose (GObject *object)
{
    EShellTaskbarPrivate *priv;

    priv = E_SHELL_TASKBAR_GET_PRIVATE (object);

    g_hash_table_foreach_remove (
        priv->proxy_table, (GHRFunc)
        shell_taskbar_weak_unref, object);

    if (priv->shell_view != NULL) {
        g_object_remove_weak_pointer (
            G_OBJECT (priv->shell_view), &priv->shell_view);
        priv->shell_view = NULL;
    }

    if (priv->shell_backend != NULL) {
        g_signal_handlers_disconnect_matched (
            priv->shell_backend, G_SIGNAL_MATCH_DATA,
            0, 0, NULL, NULL, object);
    }

    g_clear_object (&priv->shell_backend);
    g_clear_object (&priv->label);
    g_clear_object (&priv->hbox);

    /* Chain up to parent's dispose() method. */
    G_OBJECT_CLASS (e_shell_taskbar_parent_class)->dispose (object);
}

static void
shell_taskbar_finalize (GObject *object)
{
    EShellTaskbarPrivate *priv;

    priv = E_SHELL_TASKBAR_GET_PRIVATE (object);

    g_hash_table_destroy (priv->proxy_table);

    /* Chain up to parent's finalize() method. */
    G_OBJECT_CLASS (e_shell_taskbar_parent_class)->finalize (object);
}

static void
shell_taskbar_constructed (GObject *object)
{
    EShellView *shell_view;
    EShellBackend *shell_backend;
    EShellTaskbar *shell_taskbar;

    shell_taskbar = E_SHELL_TASKBAR (object);
    shell_view = e_shell_taskbar_get_shell_view (shell_taskbar);
    shell_backend = e_shell_view_get_shell_backend (shell_view);

    /* Keep a reference to the shell backend so we can
     * disconnect the signal handler during dispose(). */
    shell_taskbar->priv->shell_backend = g_object_ref (shell_backend);

    g_signal_connect_swapped (
        shell_backend, "activity-added",
        G_CALLBACK (shell_taskbar_activity_added_cb), shell_taskbar);

    e_extensible_load_extensions (E_EXTENSIBLE (object));

    /* Chain up to parent's constructed() method. */
    G_OBJECT_CLASS (e_shell_taskbar_parent_class)->constructed (object);
}

static void
shell_taskbar_size_allocate (GtkWidget *widget,
                             GtkAllocation *allocation)
{
    EShellTaskbar *shell_taskbar;
    gint fixed_height;

    shell_taskbar = E_SHELL_TASKBAR (widget);

    /* Maximum height allocation sticks. */
    fixed_height = shell_taskbar->priv->fixed_height;
    fixed_height = MAX (fixed_height, allocation->height);
    shell_taskbar->priv->fixed_height = fixed_height;

    /* Chain up to parent's size_allocate() method. */
    GTK_WIDGET_CLASS (e_shell_taskbar_parent_class)->
        size_allocate (widget, allocation);
}

static void
shell_taskbar_get_preferred_height (GtkWidget *widget,
                                    gint *minimum_height,
                                    gint *natural_height)
{
    EShellTaskbar *shell_taskbar;

    shell_taskbar = E_SHELL_TASKBAR (widget);

    if (minimum_height != NULL)
        *minimum_height = shell_taskbar->priv->fixed_height;

    if (natural_height != NULL)
        *natural_height = shell_taskbar->priv->fixed_height;
}

static void
shell_taskbar_get_preferred_width (GtkWidget *widget,
                                   gint *minimum_width,
                                   gint *natural_width)
{
    /* to never get larger than allocated size (which changes window width) */

    if (minimum_width != NULL)
        *minimum_width = 1;

    if (natural_width != NULL)
        *natural_width = 1;
}

static void
e_shell_taskbar_class_init (EShellTaskbarClass *class)
{
    GObjectClass *object_class;
    GtkWidgetClass *widget_class;

    g_type_class_add_private (class, sizeof (EShellTaskbarPrivate));

    object_class = G_OBJECT_CLASS (class);
    object_class->set_property = shell_taskbar_set_property;
    object_class->get_property = shell_taskbar_get_property;
    object_class->dispose = shell_taskbar_dispose;
    object_class->finalize = shell_taskbar_finalize;
    object_class->constructed = shell_taskbar_constructed;

    widget_class = GTK_WIDGET_CLASS (class);
    widget_class->size_allocate = shell_taskbar_size_allocate;
    widget_class->get_preferred_height = shell_taskbar_get_preferred_height;
    widget_class->get_preferred_width = shell_taskbar_get_preferred_width;

    /**
     * EShellTaskbar:message
     *
     * The message to display in the taskbar.
     **/
    g_object_class_install_property (
        object_class,
        PROP_MESSAGE,
        g_param_spec_string (
            "message",
            NULL,
            NULL,
            NULL,
            G_PARAM_READWRITE |
            G_PARAM_CONSTRUCT |
            G_PARAM_STATIC_STRINGS));

    /**
     * EShellTaskbar:shell-view
     *
     * The #EShellView to which the taskbar widget belongs.
     **/
    g_object_class_install_property (
        object_class,
        PROP_SHELL_VIEW,
        g_param_spec_object (
            "shell-view",
            NULL,
            NULL,
            E_TYPE_SHELL_VIEW,
            G_PARAM_READWRITE |
            G_PARAM_CONSTRUCT_ONLY |
            G_PARAM_STATIC_STRINGS));
}

static void
e_shell_taskbar_init (EShellTaskbar *shell_taskbar)
{
    GtkWidget *widget;

    shell_taskbar->priv = E_SHELL_TASKBAR_GET_PRIVATE (shell_taskbar);
    shell_taskbar->priv->proxy_table = g_hash_table_new (NULL, NULL);
    shell_taskbar->priv->main_thread = g_thread_self ();

    gtk_box_set_spacing (GTK_BOX (shell_taskbar), 12);

    widget = gtk_label_new (NULL);
    gtk_label_set_ellipsize (GTK_LABEL (widget), PANGO_ELLIPSIZE_END);
    gtk_box_pack_start (GTK_BOX (shell_taskbar), widget, TRUE, TRUE, 0);
    gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.5);
    shell_taskbar->priv->label = g_object_ref (widget);
    gtk_widget_hide (widget);

    widget = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 3);
    gtk_box_pack_start (GTK_BOX (shell_taskbar), widget, TRUE, TRUE, 0);
    shell_taskbar->priv->hbox = g_object_ref (widget);
    gtk_widget_hide (widget);
}

/**
 * e_shell_taskbar_new:
 * @shell_view: an #EShellView
 *
 * Creates a new #EShellTaskbar instance belonging to @shell_view.
 *
 * Returns: a new #EShellTaskbar instance
 **/
GtkWidget *
e_shell_taskbar_new (EShellView *shell_view)
{
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    return g_object_new (
        E_TYPE_SHELL_TASKBAR, "shell-view", shell_view, NULL);
}

/**
 * e_shell_taskbar_get_shell_view:
 * @shell_taskbar: an #EShellTaskbar
 *
 * Returns the #EShellView that was passed to e_shell_taskbar_new().
 *
 * Returns: the #EShellView to which @shell_taskbar belongs
 **/
EShellView *
e_shell_taskbar_get_shell_view (EShellTaskbar *shell_taskbar)
{
    g_return_val_if_fail (E_IS_SHELL_TASKBAR (shell_taskbar), NULL);

    return shell_taskbar->priv->shell_view;
}

/**
 * e_shell_taskbar_get_message:
 * @shell_taskbar: an #EShellTaskbar
 *
 * Returns the message currently shown in the taskbar, or an empty string
 * if no message is shown.  Taskbar messages are used primarily for menu
 * tooltips.
 *
 * Returns: the current taskbar message
 **/
const gchar *
e_shell_taskbar_get_message (EShellTaskbar *shell_taskbar)
{
    GtkWidget *label;

    g_return_val_if_fail (E_IS_SHELL_TASKBAR (shell_taskbar), NULL);

    label = shell_taskbar->priv->label;

    return gtk_label_get_text (GTK_LABEL (label));
}

/**
 * e_shell_taskbar_set_message:
 * @shell_taskbar: an #EShellTaskbar
 * @message: the message to show
 *
 * Shows a message in the taskbar.  If @message is %NULL or an empty string,
 * the taskbar message is cleared.  Taskbar messages are used primarily for
 * menu tooltips.
 **/
void
e_shell_taskbar_set_message (EShellTaskbar *shell_taskbar,
                             const gchar *message)
{
    GtkWidget *label;

    g_return_if_fail (E_IS_SHELL_TASKBAR (shell_taskbar));

    label = shell_taskbar->priv->label;
    gtk_label_set_text (GTK_LABEL (label), message);

    if (message != NULL && *message != '\0')
        gtk_widget_show (label);
    else
        gtk_widget_hide (label);

    g_object_notify (G_OBJECT (shell_taskbar), "message");
}

/**
 * e_shell_taskbar_unset_message:
 * @shell_taskbar: an #EShellTaskbar
 *
 * This is equivalent to passing a %NULL message to
 * e_shell_taskbar_set_message().
 **/
void
e_shell_taskbar_unset_message (EShellTaskbar *shell_taskbar)
{
    g_return_if_fail (E_IS_SHELL_TASKBAR (shell_taskbar));

    e_shell_taskbar_set_message (shell_taskbar, NULL);
}

/**
 * e_shell_taskbar_get_activity_count:
 * @shell_taskbar: an #EShellTaskbar
 *
 * Returns the number of active #EActivity instances being tracked.
 *
 * Returns: the number of #EActivity instances
 **/
guint
e_shell_taskbar_get_activity_count (EShellTaskbar *shell_taskbar)
{
    g_return_val_if_fail (E_IS_SHELL_TASKBAR (shell_taskbar), 0);

    return g_hash_table_size (shell_taskbar->priv->proxy_table);
}