13
0

hand-merge libs/gtkmm2ext part of f49d31a82 from Tracks; provides a way to disable/re-enable OS X main menu

This commit is contained in:
Paul Davis 2015-05-06 11:51:04 -04:00
parent 7e097c5efc
commit 2363fb71e3
2 changed files with 60 additions and 52 deletions

View File

@ -139,11 +139,11 @@ ActionManager::lookup_entry (const ustring accel_path, Gtk::AccelKey& key)
} }
struct SortActionsByLabel { struct SortActionsByLabel {
bool operator() (Glib::RefPtr<Gtk::Action> a, Glib::RefPtr<Gtk::Action> b) { bool operator() (Glib::RefPtr<Gtk::Action> a, Glib::RefPtr<Gtk::Action> b) {
ustring astr = a->get_accel_path(); ustring astr = a->get_accel_path();
ustring bstr = b->get_accel_path(); ustring bstr = b->get_accel_path();
return astr < bstr; return astr < bstr;
} }
}; };
void void
@ -289,71 +289,63 @@ struct ActionState {
typedef std::vector<ActionState> ActionStates; typedef std::vector<ActionState> ActionStates;
static std::stack<boost::shared_ptr<ActionStates> > state_stack; static ActionStates action_states_to_restore;
static bool actions_disabled = false;
static boost::shared_ptr<ActionStates> void
get_action_state () ActionManager::save_action_states ()
{ {
boost::shared_ptr<ActionStates> state = boost::shared_ptr<ActionStates>(new ActionStates);
/* the C++ API for functions used here appears to be broken in /* the C++ API for functions used here appears to be broken in
gtkmm2.6, so we fall back to the C level. gtkmm2.6, so we fall back to the C level.
*/ */
GList* list = gtk_ui_manager_get_action_groups (ActionManager::ui_manager->gobj()); GList* list = gtk_ui_manager_get_action_groups (ActionManager::ui_manager->gobj());
GList* node; GList* node;
GList* acts; GList* acts;
for (node = list; node; node = g_list_next (node)) { for (node = list; node; node = g_list_next (node)) {
GtkActionGroup* group = (GtkActionGroup*) node->data; GtkActionGroup* group = (GtkActionGroup*) node->data;
/* first pass: collect them all */
typedef std::list<Glib::RefPtr<Gtk::Action> > action_list;
action_list the_acts;
for (acts = gtk_action_group_list_actions (group); acts; acts = g_list_next (acts)) { for (acts = gtk_action_group_list_actions (group); acts; acts = g_list_next (acts)) {
GtkAction* action = (GtkAction*) acts->data; GtkAction* action = (GtkAction*) acts->data;
action_states_to_restore.push_back (ActionState (action, gtk_action_get_sensitive (action)));
state->push_back (ActionState (action, gtk_action_get_sensitive (action)));
} }
} }
return state;
} }
void void
ActionManager::push_action_state () ActionManager::enable_active_actions ()
{ {
state_stack.push (get_action_state()); 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 void
ActionManager::pop_action_state () ActionManager::disable_active_actions ()
{ {
if (state_stack.empty()) { if (actions_disabled == true ) {
warning << string_compose (_("programming error: %1"), X_("ActionManager::pop_action_state called with empty stack")) << endmsg; return ;
return;
} }
// save all action's states to action_states_to_restore
boost::shared_ptr<ActionStates> as = state_stack.top (); save_action_states ();
state_stack.pop ();
for (ActionStates::iterator i = as->begin(); i != as->end(); ++i) { // set all action's states disabled
gtk_action_set_sensitive ((*i).action, (*i).sensitive); 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);
}
void
ActionManager::disable_all_actions ()
{
push_action_state ();
boost::shared_ptr<ActionStates> as = state_stack.top ();
for (ActionStates::iterator i = as->begin(); i != as->end(); ++i) {
gtk_action_set_sensitive ((*i).action, false);
} }
actions_disabled = true;
} }
void void
@ -464,8 +456,24 @@ ActionManager::get_action_from_name (const char* name)
void void
ActionManager::set_sensitive (vector<RefPtr<Action> >& actions, bool state) ActionManager::set_sensitive (vector<RefPtr<Action> >& actions, bool state)
{ {
for (vector<RefPtr<Action> >::iterator i = actions.begin(); i != actions.end(); ++i) { // if actions weren't disabled
(*i)->set_sensitive (state); if (!actions_disabled) {
for (vector<RefPtr<Action> >::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<RefPtr<Action> >::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;
}
}
}
} }
} }
@ -503,10 +511,10 @@ ActionManager::set_toggleaction_state (string n, bool s)
const char* action_name = last_slash + 1; const char* action_name = last_slash + 1;
RefPtr<Action> act = get_action (group_name, action_name); RefPtr<Action> act = get_action (group_name, action_name);
if (act) { if (act) {
RefPtr<ToggleAction> tact = RefPtr<ToggleAction>::cast_dynamic(act); RefPtr<ToggleAction> tact = RefPtr<ToggleAction>::cast_dynamic(act);
tact->set_active (s); tact->set_active (s);
} else { } else {
error << string_compose (_("Unknown action name: %1"), name) << endmsg; error << string_compose (_("Unknown action name: %1"), name) << endmsg;
} }

View File

@ -94,9 +94,9 @@ namespace ActionManager {
LIBGTKMM2EXT_API extern void set_toggleaction_state (std::string, bool); LIBGTKMM2EXT_API extern void set_toggleaction_state (std::string, bool);
LIBGTKMM2EXT_API extern void push_action_state (); LIBGTKMM2EXT_API extern void save_action_states ();
LIBGTKMM2EXT_API extern void pop_action_state (); LIBGTKMM2EXT_API extern void enable_active_actions ();
LIBGTKMM2EXT_API extern void disable_all_actions (); LIBGTKMM2EXT_API extern void disable_active_actions ();
}; };
#endif /* __libgtkmm2ext_actions_h__ */ #endif /* __libgtkmm2ext_actions_h__ */