/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
* Authors: Michael Zucchi <notzed@ximian.com>
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* 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 General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#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, void *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;
int 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 char *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, void *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 void *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)
{
int 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;
}