Refactor GuiApplication::getStatus().

* I didn't like the fact that the heart of the dispatch/status machinery is in the default clause of a long switch statement. Now, it is clear that we enter the app in getStatus, which then asks the GuiApplication itself, the GuiView, the current BufferView, the current Buffer, the document BufferView, the document Buffer,

* Shouldn't we let BufferView call the Buffer getStatus() functions ?,

* This also makes sure that if a command is not handled, it is turned off. Before r34164 this was caused by the default clause in BufferView,

* Now it is prevented that if the document BufferView is the same as the current BufferView, that the getStatus() functions are called twice,

* A warning can be outputted if the LFUN is not handled.

PS I want to do the same for the dispatch function.

git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@34165 a592a061-630c-0410-9148-cb99ea01b6c8
This commit is contained in:
Vincent van Ravesteijn 2010-04-16 16:04:35 +00:00
parent eb9fb94214
commit e8fcc92d8f
2 changed files with 68 additions and 51 deletions

View File

@ -854,21 +854,76 @@ docstring GuiApplication::iconName(FuncRequest const & f, bool unknown)
FuncStatus GuiApplication::getStatus(FuncRequest const & cmd) const FuncStatus GuiApplication::getStatus(FuncRequest const & cmd) const
{ {
FuncStatus flag; FuncStatus status;
BufferView * bv = 0;
BufferView * doc_bv = 0;
if (cmd.action() == LFUN_NOACTION) { if (cmd.action() == LFUN_NOACTION) {
flag.message(from_utf8(N_("Nothing to do"))); status.message(from_utf8(N_("Nothing to do")));
flag.setEnabled(false); status.setEnabled(false);
return flag;
} }
if (cmd.action() == LFUN_UNKNOWN_ACTION) { else if (cmd.action() == LFUN_UNKNOWN_ACTION) {
flag.setUnknown(true); status.setUnknown(true);
flag.setEnabled(false); status.message(from_utf8(N_("Unknown action")));
flag.message(from_utf8(N_("Unknown action"))); status.setEnabled(false);
return flag;
} }
// Does the GuiApplication know something?
else if (getStatus(cmd, status)) { }
// If we do not have a GuiView, then other functions are disabled
else if (!current_view_)
status.setEnabled(false);
// Does the GuiView know something?
else if (current_view_->getStatus(cmd, status)) { }
// In LyX/Mac, when a dialog is open, the menus of the
// application can still be accessed without giving focus to
// the main window. In this case, we want to disable the menu
// entries that are buffer or view-related.
//FIXME: Abdel (09/02/10) This has very bad effect on Linux, don't know why...
/*
else if (cmd.origin() == FuncRequest::MENU && !current_view_->hasFocus())
status.setEnabled(false);
*/
// If we do not have a BufferView, then other functions are disabled
else if (!(bv = current_view_->currentBufferView()))
status.setEnabled(false);
// Does the current BufferView know something?
else if (bv->getStatus(cmd, status)) { }
// Does the current Buffer know something?
else if (bv->buffer().getStatus(cmd, status)) { }
// If we do not have a document BufferView, different from the
// current BufferView, then other functions are disabled
else if (!(doc_bv = current_view_->documentBufferView()) || doc_bv == bv)
status.setEnabled(false);
// Does the document Buffer know something?
else if (doc_bv->buffer().getStatus(cmd, status)) { }
else {
LYXERR(Debug::ACTION, "LFUN not handled in getStatus(): " << cmd);
status.message(from_utf8(N_("Command not handled")));
status.setEnabled(false);
}
// the default error message if we disable the command
if (!status.enabled() && status.message().empty())
status.message(from_utf8(N_("Command disabled")));
return status;
}
bool GuiApplication::getStatus(FuncRequest const & cmd, FuncStatus & flag) const
{
// I would really like to avoid having this switch and rather try to // I would really like to avoid having this switch and rather try to
// encode this in the function itself. // encode this in the function itself.
// -- And I'd rather let an inset decide which LFUNs it is willing // -- And I'd rather let an inset decide which LFUNs it is willing
@ -974,52 +1029,12 @@ FuncStatus GuiApplication::getStatus(FuncRequest const & cmd) const
break; break;
default: default:
// Does the view know something? return false;
if (!current_view_) {
enable = false;
break;
}
if (current_view_->getStatus(cmd, flag))
break;
// In LyX/Mac, when a dialog is open, the menus of the
// application can still be accessed without giving focus to
// the main window. In this case, we want to disable the menu
// entries that are buffer or view-related.
//FIXME: Abdel (09/02/10) This has very bad effect on Linux, don't know why...
/*
if (cmd.origin() == FuncRequest::MENU && !current_view_->hasFocus()) {
enable = false;
break;
}
*/
BufferView * bv = current_view_->currentBufferView();
BufferView * doc_bv = current_view_->documentBufferView();
// If we do not have a BufferView, then other functions are disabled
if (!bv) {
enable = false;
break;
}
// try the BufferView
bool decided = bv->getStatus(cmd, flag);
if (!decided)
// try the Buffer
decided = bv->buffer().getStatus(cmd, flag);
if (!decided && doc_bv)
// try the Document Buffer
decided = doc_bv->buffer().getStatus(cmd, flag);
} }
if (!enable) if (!enable)
flag.setEnabled(false); flag.setEnabled(false);
return true;
// the default error message if we disable the command
if (!flag.enabled() && flag.message().empty())
flag.message(from_utf8(N_("Command disabled")));
return flag;
} }
/// make a post-dispatch status message /// make a post-dispatch status message

View File

@ -78,6 +78,8 @@ public:
void handleKeyFunc(FuncCode action); void handleKeyFunc(FuncCode action);
//@} //@}
/// ///
bool getStatus(FuncRequest const & cmd, FuncStatus & status) const;
///
void hideDialogs(std::string const & name, Inset * inset) const; void hideDialogs(std::string const & name, Inset * inset) const;
/// ///
void resetGui(); void resetGui();