/* Copyright (C) 2005 Paul Davis 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., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "pbd/error.h" #include "gtkmm2ext/actions.h" #include "gtkmm2ext/utils.h" #include "i18n.h" using namespace std; using namespace Gtk; using namespace Glib; using namespace sigc; using namespace PBD; using namespace Gtkmm2ext; RefPtr ActionManager::ui_manager; string ActionManager::unbound_string = "--"; RefPtr ActionManager::register_action (RefPtr group, const char * name, const char * label, slot sl) { RefPtr act; act = Action::create (name, label); group->add (act, sl); return act; } RefPtr ActionManager::register_action (RefPtr group, const char * name, const char * label) { RefPtr act; act = Action::create (name, label); group->add (act); return act; } RefPtr ActionManager::register_radio_action (RefPtr group, RadioAction::Group& rgroup, const char * name, const char * label, slot sl) { RefPtr act; act = RadioAction::create (rgroup, name, label); group->add (act, sl); return act; } RefPtr ActionManager::register_radio_action ( RefPtr group, RadioAction::Group& rgroup, string const & name, string const & label, string const & tooltip, slot sl ) { RefPtr act; act = RadioAction::create (rgroup, name, label, tooltip); group->add (act, sl); return act; } RefPtr ActionManager::register_toggle_action (RefPtr group, const char * name, const char * label, slot sl) { RefPtr act; act = ToggleAction::create (name, label); group->add (act, sl); return act; } RefPtr ActionManager::register_toggle_action (RefPtr group, string const & name, string const & label, string const & tooltip, slot sl) { RefPtr act; act = ToggleAction::create (name, label, tooltip); group->add (act, sl); return act; } bool ActionManager::lookup_entry (const ustring accel_path, Gtk::AccelKey& key) { GtkAccelKey gkey; bool known = gtk_accel_map_lookup_entry (accel_path.c_str(), &gkey); if (known) { key = AccelKey (gkey.accel_key, Gdk::ModifierType (gkey.accel_mods)); } else { key = AccelKey (GDK_VoidSymbol, Gdk::ModifierType (0)); } return known; } struct SortActionsByLabel { bool operator() (Glib::RefPtr a, Glib::RefPtr b) { ustring astr = a->get_accel_path(); ustring bstr = b->get_accel_path(); return astr < bstr; } }; void ActionManager::get_all_actions (vector& groups, vector& names, vector& tooltips, vector& bindings) { /* the C++ API for functions used here appears to be broken in gtkmm2.6, so we fall back to the C level. */ GList* list = gtk_ui_manager_get_action_groups (ui_manager->gobj()); GList* node; GList* acts; for (node = list; node; node = g_list_next (node)) { GtkActionGroup* group = (GtkActionGroup*) node->data; /* first pass: collect them all */ typedef std::list > action_list; action_list the_acts; for (acts = gtk_action_group_list_actions (group); acts; acts = g_list_next (acts)) { GtkAction* action = (GtkAction*) acts->data; the_acts.push_back (Glib::wrap (action, true)); } /* now sort by label */ SortActionsByLabel cmp; the_acts.sort (cmp); for (action_list::iterator a = the_acts.begin(); a != the_acts.end(); ++a) { string accel_path = (*a)->get_accel_path (); groups.push_back (gtk_action_group_get_name(group)); names.push_back (accel_path.substr (accel_path.find_last_of ('/') + 1)); tooltips.push_back ((*a)->get_tooltip ()); AccelKey key; lookup_entry (accel_path, key); bindings.push_back (AccelKey (key.get_key(), Gdk::ModifierType (key.get_mod()))); } } } void ActionManager::get_all_actions (vector& names, vector& paths, vector& tooltips, vector& keys, vector& bindings) { /* the C++ API for functions used here appears to be broken in gtkmm2.6, so we fall back to the C level. */ GList* list = gtk_ui_manager_get_action_groups (ui_manager->gobj()); GList* node; GList* acts; for (node = list; node; node = g_list_next (node)) { GtkActionGroup* group = (GtkActionGroup*) node->data; /* first pass: collect them all */ typedef std::list > action_list; action_list the_acts; for (acts = gtk_action_group_list_actions (group); acts; acts = g_list_next (acts)) { GtkAction* action = (GtkAction*) acts->data; the_acts.push_back (Glib::wrap (action, true)); } /* now sort by label */ SortActionsByLabel cmp; the_acts.sort (cmp); for (action_list::iterator a = the_acts.begin(); a != the_acts.end(); ++a) { ustring const label = (*a)->property_label (); string const accel_path = (*a)->get_accel_path (); names.push_back (label); paths.push_back (accel_path); tooltips.push_back ((*a)->get_tooltip ()); AccelKey key; keys.push_back (get_key_representation (accel_path, key)); bindings.push_back (AccelKey (key.get_key(), Gdk::ModifierType (key.get_mod()))); } } } void ActionManager::enable_accelerators () { /* the C++ API for functions used here appears to be broken in gtkmm2.6, so we fall back to the C level. */ GList* list = gtk_ui_manager_get_action_groups (ui_manager->gobj()); GList* node; GList* acts; string ui_string = ""; /* get all actions, build a string describing them all as */ for (node = list; node; node = g_list_next (node)) { GtkActionGroup* group = (GtkActionGroup*) node->data; for (acts = gtk_action_group_list_actions (group); acts; acts = g_list_next (acts)) { ui_string += "data); ui_string += Glib::path_get_basename (fullpath); ui_string += "\"/>"; } } ui_string += ""; /* and load it */ ui_manager->add_ui_from_string (ui_string); } struct ActionState { GtkAction* action; bool sensitive; ActionState (GtkAction* a, bool s) : action (a), sensitive (s) {} }; typedef std::vector ActionStates; static ActionStates action_states_to_restore; static bool actions_disabled = false; void ActionManager::save_action_states () { /* the C++ API for functions used here appears to be broken in gtkmm2.6, so we fall back to the C level. */ GList* list = gtk_ui_manager_get_action_groups (ActionManager::ui_manager->gobj()); GList* node; GList* acts; for (node = list; node; node = g_list_next (node)) { GtkActionGroup* group = (GtkActionGroup*) node->data; for (acts = gtk_action_group_list_actions (group); acts; acts = g_list_next (acts)) { GtkAction* action = (GtkAction*) acts->data; action_states_to_restore.push_back (ActionState (action, gtk_action_get_sensitive (action))); } } } void ActionManager::enable_active_actions () { if (!actions_disabled) { return ; } for (ActionStates::iterator i = action_states_to_restore.begin(); i != action_states_to_restore.end(); ++i) { if ((*i).action && (*i).sensitive) { gtk_action_set_sensitive ((*i).action, true); } } action_states_to_restore.clear (); actions_disabled = false; } void ActionManager::disable_active_actions () { if (actions_disabled == true ) { return ; } // save all action's states to action_states_to_restore save_action_states (); // set all action's states disabled for (ActionStates::iterator i = action_states_to_restore.begin(); i != action_states_to_restore.end(); ++i) { if ((*i).sensitive) { gtk_action_set_sensitive ((*i).action, false); } } actions_disabled = true; } void ActionManager::add_action_group (RefPtr grp) { ui_manager->insert_action_group (grp); } Widget* ActionManager::get_widget (const char * name) { return ui_manager->get_widget (name); } RefPtr ActionManager::get_action (const char* path) { if (!path) { return RefPtr(); } /* Skip / in path */ int len = strlen (path); if (len < 3) { /* shortest possible path: "a/b" */ return RefPtr(); } if (len > 10 && !strncmp (path, "/", 10 )) { path = path+10; } else if (path[0] == '/') { path++; } vector copy(len+1); strcpy (©[0], path); char* slash = strchr (©[0], '/'); if (!slash) { return RefPtr (); } *slash = '\0'; return get_action (©[0], ++slash); } RefPtr ActionManager::get_action (const char* group_name, const char* action_name) { /* the C++ API for functions used here appears to be broken in gtkmm2.6, so we fall back to the C level. */ if (ui_manager == 0) { return RefPtr (); } GList* list = gtk_ui_manager_get_action_groups (ui_manager->gobj()); GList* node; RefPtr act; for (node = list; node; node = g_list_next (node)) { GtkActionGroup* _ag = (GtkActionGroup*) node->data; if (strcmp (group_name, gtk_action_group_get_name (_ag)) == 0) { GtkAction* _act; if ((_act = gtk_action_group_get_action (_ag, action_name)) != 0) { act = Glib::wrap (_act, true); break; } } } return act; } RefPtr ActionManager::get_action_from_name (const char* name) { /* the C++ API for functions used here appears to be broken in gtkmm2.6, so we fall back to the C level. */ GList* list = gtk_ui_manager_get_action_groups (ui_manager->gobj()); GList* node; GList* acts; for (node = list; node; node = g_list_next (node)) { GtkActionGroup* group = (GtkActionGroup*) node->data; for (acts = gtk_action_group_list_actions (group); acts; acts = g_list_next (acts)) { GtkAction* action = (GtkAction*) acts->data; if (!strcmp (gtk_action_get_name (action), name)) { return Glib::wrap (action, true); } } } return RefPtr(); } void ActionManager::set_sensitive (vector >& actions, bool state) { // if actions weren't disabled if (!actions_disabled) { for (vector >::iterator i = actions.begin(); i != actions.end(); ++i) { (*i)->set_sensitive (state); } } else { // actions were disabled // so we should just set necessary action's states in action_states_to_restore for (vector >::iterator i = actions.begin(); i != actions.end(); ++i) { // go through action_states_to_restore and set state of actions for (ActionStates::iterator j = action_states_to_restore.begin(); j != action_states_to_restore.end(); ++j) { // all actions should have their individual name, so we can use it for comparison if (gtk_action_get_name ((*j).action) == (*i)->get_name ()) { (*j).sensitive = state; } } } } } void ActionManager::check_toggleaction (string n) { set_toggleaction_state (n, true); } void ActionManager::uncheck_toggleaction (string n) { set_toggleaction_state (n, false); } void ActionManager::set_toggleaction_state (string n, bool s) { char const * name = n.c_str (); const char *last_slash = strrchr (name, '/'); if (last_slash == 0) { fatal << string_compose ("programmer error: %1 %2", "illegal toggle action name", name) << endmsg; abort(); /*NOTREACHED*/ return; } /* 10 = strlen ("/") */ size_t len = last_slash - (name + 10); char* group_name = new char[len+1]; memcpy (group_name, name + 10, len); group_name[len] = '\0'; const char* action_name = last_slash + 1; RefPtr act = get_action (group_name, action_name); if (act) { RefPtr tact = RefPtr::cast_dynamic(act); tact->set_active (s); } else { error << string_compose (_("Unknown action name: %1"), name) << endmsg; } delete [] group_name; } string ActionManager::get_key_representation (const string& accel_path, AccelKey& key) { bool known = lookup_entry (accel_path, key); if (known) { uint32_t k = possibly_translate_legal_accelerator_to_real_key (key.get_key()); key = AccelKey (k, Gdk::ModifierType (key.get_mod())); return ui_manager->get_accel_group()->get_label (key.get_key(), Gdk::ModifierType (key.get_mod())); } return unbound_string; } void ActionManager::do_action (const char* group, const char*action) { Glib::RefPtr act = ActionManager::get_action (group, action); if (act) { act->activate (); } } void ActionManager::set_toggle_action (const char* group, const char*action, bool yn) { Glib::RefPtr act = ActionManager::get_action (group, action); if (act) { Glib::RefPtr tact = Glib::RefPtr::cast_dynamic (act); if (tact) { tact->set_active (yn); } } }