aboutsummaryrefslogblamecommitdiffstats
path: root/calendar/gui/e-cal-menu.c
blob: 30d287ff6be81e6aaf81db48f9028cb658e6ed99 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
  



                                                                
  



                                                                    
  
                                                                   
                                                                             
  
  



                                                        















                            
                                                                   




















                                                                    
                       








































                                                                                                                 
                                             
 
                                                                  








                                                           
               
                                                         
  
                                       

                




























                                                                                                       

                                                                                                                     
                                                                                                       
                                                              
 

                                                                                                     
 

                                                                               

                                                                                


























                                                                                                                










                                              
                                                      









                                                                            

                                           



                                                      
                                            














                                              
                                    













                                                            
                                                          
                



                                                                    
                












                                                           
               

                                                            
                                                                                        
 
                                                









                                                                                              
 








                                                                                                              
 

                    
/*
 * 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; either
 * version 2 of the License, or (at your option) version 3.
 *
 * 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with the program; if not, see <http://www.gnu.org/licenses/>
 *
 *
 * Authors:
 *      Michael Zucchi <notzed@ximian.com>
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 *
 */

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

#include <string.h>
#include <stdlib.h>

#include <glib.h>

#include "e-cal-menu.h"
#include "gui/e-cal-model.h"
#include "itip-utils.h"

static void ecalm_standard_menu_factory(EMenu *emp, gpointer data);

static GObjectClass *ecalm_parent;

static void
ecalm_init(GObject *o)
{
    /*ECalMenu *emp = (ECalMenu *)o; */
}

static void
ecalm_finalise(GObject *o)
{
    ((GObjectClass *)ecalm_parent)->finalize(o);
}

static void
ecalm_target_free(EMenu *ep, EMenuTarget *t)
{
    switch (t->type) {
    case E_CAL_MENU_TARGET_SELECT: {
        ECalMenuTargetSelect *s = (ECalMenuTargetSelect *)t;
        gint i;

        for (i=0;i<s->events->len;i++)
            e_cal_model_free_component_data(s->events->pdata[i]);
        g_ptr_array_free(s->events, TRUE);
        g_object_unref(s->model);
        break; }
    }

    ((EMenuClass *)ecalm_parent)->target_free(ep, t);
}

static void
ecalm_class_init(GObjectClass *klass)
{
    klass->finalize = ecalm_finalise;
    ((EMenuClass *)klass)->target_free = ecalm_target_free;

    e_menu_class_add_factory((EMenuClass *)klass, NULL, (EMenuFactoryFunc)ecalm_standard_menu_factory, NULL);
}

GType
e_cal_menu_get_type(void)
{
    static GType type = 0;

    if (type == 0) {
        static const GTypeInfo info = {
            sizeof(ECalMenuClass),
            NULL, NULL,
            (GClassInitFunc)ecalm_class_init,
            NULL, NULL,
            sizeof(ECalMenu), 0,
            (GInstanceInitFunc)ecalm_init
        };
        ecalm_parent = g_type_class_ref(e_menu_get_type());
        type = g_type_register_static(e_menu_get_type(), "ECalMenu", &info, 0);
    }

    return type;
}

ECalMenu *e_cal_menu_new(const gchar *menuid)
{
    ECalMenu *emp = g_object_new(e_cal_menu_get_type(), NULL);

    e_menu_construct(&emp->menu, menuid);

    return emp;
}

/**
 * e_cal_menu_target_new_select:
 * @folder: The selection will ref this for the life of it.
 * @folder_uri:
 * @uids: The selection will free this when done with it.
 *
 * Create a new selection popup target.
 *
 * Return value:
 **/
ECalMenuTargetSelect *
e_cal_menu_target_new_select(ECalMenu *eabp, struct _ECalModel *model, GPtrArray *events)
{
    ECalMenuTargetSelect *t = e_menu_target_new(&eabp->menu, E_CAL_MENU_TARGET_SELECT, sizeof(*t));
    guint32 mask = ~0;
    ECal *client;
    gboolean read_only;

    /* FIXME: This is duplicated in e-cal-popup */

    t->model = model;
    g_object_ref(t->model);
    t->events = events;

    if (t->events->len == 0) {
        client = e_cal_model_get_default_client(t->model);
    } else {
        ECalModelComponent *comp_data = (ECalModelComponent *)t->events->pdata[0];

        mask &= ~E_CAL_MENU_SELECT_ANY;
        if (t->events->len == 1)
            mask &= ~E_CAL_MENU_SELECT_ONE;
        else
            mask &= ~E_CAL_MENU_SELECT_MANY;

        if (icalcomponent_get_first_property (comp_data->icalcomp, ICAL_URL_PROPERTY))
            mask &= ~E_CAL_MENU_SELECT_HASURL;

        if (!e_cal_get_static_capability (comp_data->client, CAL_STATIC_CAPABILITY_NO_TASK_ASSIGNMENT)
            && !e_cal_get_static_capability (comp_data->client, CAL_STATIC_CAPABILITY_NO_CONV_TO_ASSIGN_TASK)
            && !icalcomponent_get_first_property (comp_data->icalcomp, ICAL_ATTENDEE_PROPERTY))
            mask &= ~E_CAL_MENU_SELECT_ASSIGNABLE;

        if (!icalcomponent_get_first_property (comp_data->icalcomp, ICAL_COMPLETED_PROPERTY))
            mask &= ~ E_CAL_MENU_SELECT_NOTCOMPLETE;

        if (e_cal_util_component_has_recurrences (comp_data->icalcomp))
            mask &= ~E_CAL_MENU_SELECT_RECURRING;
        else if (e_cal_util_component_is_instance (comp_data->icalcomp))
            mask &= ~E_CAL_MENU_SELECT_RECURRING;
        else
            mask &= ~E_CAL_MENU_SELECT_NONRECURRING;

        if (e_cal_util_component_is_instance (comp_data->icalcomp))
            mask &= ~E_CAL_MENU_SELECT_INSTANCE;

        if (e_cal_util_component_has_organizer (comp_data->icalcomp)) {
            ECalComponent *comp;

            comp = e_cal_component_new ();
            e_cal_component_set_icalcomponent (comp, icalcomponent_new_clone (comp_data->icalcomp));
            if (!itip_organizer_is_user (comp, comp_data->client))
                mask &= ~E_CAL_MENU_SELECT_ORGANIZER;

            g_object_unref (comp);
        } else {
            /* organiser is synonym for owner in this case */
            mask &= ~(E_CAL_MENU_SELECT_ORGANIZER|E_CAL_MENU_SELECT_NOTMEETING);
        }

        client = comp_data->client;
    }

    if (client) {
        e_cal_is_read_only(client, &read_only, NULL);
        if (!read_only)
            mask &= ~E_CAL_MENU_SELECT_EDITABLE;
    }

    /* This bit isn't implemented ... */
    mask &= ~E_CAL_MENU_SELECT_NOTEDITING;

    t->target.mask = mask;

    return t;
}

static void
ecalm_standard_menu_factory(EMenu *emp, gpointer data)
{
    /* noop */
}

/* ********************************************************************** */

/* menu plugin handler */

/*
<e-plugin
  class="org.gnome.mail.plugin.popup:1.0"
  id="org.gnome.mail.plugin.popup.item:1.0"
  type="shlib"
  location="/opt/gnome2/lib/camel/1.0/libcamelimap.so"
  name="imap"
  description="IMAP4 and IMAP4v1 mail store">
  <hook class="org.gnome.mail.popupMenu:1.0"
        handler="HandlePopup">
  <menu id="any" target="select">
   <item
    type="item|toggle|radio|image|submenu|bar"
    active
    path="foo/bar"
    label="label"
    icon="foo"
    mask="select_one"
    activate="ecalm_view_emacs"/>
  </menu>
  </extension>

*/

static gpointer ecalph_parent_class;
#define ecalph ((ECalMenuHook *)eph)

static const EMenuHookTargetMask ecalph_select_masks[] = {
    { "one", E_CAL_MENU_SELECT_ONE },
    { "many", E_CAL_MENU_SELECT_MANY },
    { "editable", E_CAL_MENU_SELECT_EDITABLE },
    { "recurring", E_CAL_MENU_SELECT_RECURRING },
    { "non-recurring", E_CAL_MENU_SELECT_NONRECURRING },
    { "instance", E_CAL_MENU_SELECT_INSTANCE },
    { "organizer", E_CAL_MENU_SELECT_ORGANIZER },
    { "not-editing", E_CAL_MENU_SELECT_NOTEDITING },
    { "not-meeting", E_CAL_MENU_SELECT_NOTMEETING },
    { "assignable", E_CAL_MENU_SELECT_ASSIGNABLE },
    { "hasurl", E_CAL_MENU_SELECT_HASURL },
    { "not-complete", E_CAL_MENU_SELECT_NOTCOMPLETE },
    { NULL }
};

static const EMenuHookTargetMap ecalph_targets[] = {
    { "select", E_CAL_MENU_TARGET_SELECT, ecalph_select_masks },
    { NULL }
};

static void
ecalph_finalise(GObject *o)
{
    /*EPluginHook *eph = (EPluginHook *)o;*/

    ((GObjectClass *)ecalph_parent_class)->finalize(o);
}

static void
ecalph_class_init(EPluginHookClass *klass)
{
    gint i;

    ((GObjectClass *)klass)->finalize = ecalph_finalise;
    ((EPluginHookClass *)klass)->id = "org.gnome.evolution.calendar.bonobomenu:1.0";

    for (i = 0; ecalph_targets[i].type; i++)
        e_menu_hook_class_add_target_map((EMenuHookClass *)klass, &ecalph_targets[i]);

    /* FIXME: leaks parent set class? */
    ((EMenuHookClass *)klass)->menu_class = g_type_class_ref(e_cal_menu_get_type());
}

GType
e_cal_menu_hook_get_type(void)
{
    static GType type = 0;

    if (!type) {
        static const GTypeInfo info = {
            sizeof(ECalMenuHookClass), NULL, NULL, (GClassInitFunc) ecalph_class_init, NULL, NULL,
            sizeof(ECalMenuHook), 0, (GInstanceInitFunc) NULL,
        };

        ecalph_parent_class = g_type_class_ref(e_menu_hook_get_type());
        type = g_type_register_static(e_menu_hook_get_type(), "ECalMenuHook", &info, 0);
    }

    return type;
}