lyx_mirror/src/frontends/qt4/Toolbars.cpp
Jean-Marc Lasgouttes 7ac700920f Auto feature for minibuffer toolbar
Now the minibuffer toolbar is "auto" by default. It is opened by
command-execute (M-x) and closed when the command is executed without error.

* make lyx::dispatch return a DispatchResult struct

* there is a new MINIBUFFER type of toolbar, that can be used for this use.

* remove special handling of M-x in minnibuffer; Escape can be used instead. Fix focus in this case.

* when minibuffer toolbar is "auto", make the toolbar close itself after
  - a command has been executed without error
  - an empty command has been executed
  - the Escape key has been used

[this is actually commit fdcff02a, which was later reverted at dd61d8cf]
2015-07-15 17:41:09 +02:00

413 lines
9.4 KiB
C++

/**
* \file Toolbars.cpp
* This file is part of LyX, the document processor.
* Licence details can be found in the file COPYING.
*
* \author Jean-Marc Lasgouttes
* \author John Levon
*
* Full author contact details are available in file CREDITS.
*/
#include <config.h>
#include "Toolbars.h"
#include "Converter.h"
#include "Format.h"
#include "FuncRequest.h"
#include "Lexer.h"
#include "LyXAction.h"
#include "qt_helpers.h"
#include "support/debug.h"
#include "support/gettext.h"
#include "support/lstrings.h"
#include "support/bind.h"
#include <algorithm>
using namespace std;
using namespace lyx::support;
namespace lyx {
namespace frontend {
/////////////////////////////////////////////////////////////////////////
//
// ToolbarItem
//
/////////////////////////////////////////////////////////////////////////
ToolbarItem::ToolbarItem(Type type, FuncRequest const & func, docstring const & label)
: type_(type), func_(func), label_(label)
{
}
ToolbarItem::ToolbarItem(Type type, string const & name, docstring const & label)
: type_(type), label_(label), name_(name)
{
}
void ToolbarInfo::add(ToolbarItem const & item)
{
items.push_back(item);
items.back().func_.setOrigin(FuncRequest::TOOLBAR);
}
ToolbarInfo & ToolbarInfo::read(Lexer & lex)
{
enum {
TO_COMMAND = 1,
TO_ENDTOOLBAR,
TO_SEPARATOR,
TO_LAYOUTS,
TO_MINIBUFFER,
TO_TABLEINSERT,
TO_POPUPMENU,
TO_STICKYPOPUPMENU,
TO_ICONPALETTE,
TO_EXPORTFORMATS,
TO_IMPORTFORMATS,
TO_UPDATEFORMATS,
TO_VIEWFORMATS
};
struct LexerKeyword toolTags[] = {
{ "end", TO_ENDTOOLBAR },
{ "exportformats", TO_EXPORTFORMATS },
{ "iconpalette", TO_ICONPALETTE },
{ "importformats", TO_IMPORTFORMATS },
{ "item", TO_COMMAND },
{ "layouts", TO_LAYOUTS },
{ "minibuffer", TO_MINIBUFFER },
{ "popupmenu", TO_POPUPMENU },
{ "separator", TO_SEPARATOR },
{ "stickypopupmenu", TO_STICKYPOPUPMENU },
{ "tableinsert", TO_TABLEINSERT },
{ "updateformats", TO_UPDATEFORMATS },
{ "viewformats", TO_VIEWFORMATS },
};
//consistency check
if (compare_ascii_no_case(lex.getString(), "toolbar")) {
LYXERR0("ToolbarInfo::read: ERROR wrong token:`"
<< lex.getString() << '\'');
}
lex.next(true);
name = lex.getString();
lex.next(true);
gui_name = _(lex.getString());
// FIXME what to do here?
if (!lex) {
LYXERR0("ToolbarInfo::read: Malformed toolbar "
"description " << lex.getString());
return *this;
}
bool quit = false;
lex.pushTable(toolTags);
if (lyxerr.debugging(Debug::PARSER))
lex.printTable(lyxerr);
while (lex.isOK() && !quit) {
int const code = lex.lex();
switch (code) {
case TO_COMMAND:
if (lex.next(true)) {
docstring const tooltip = translateIfPossible(lex.getDocString());
lex.next(true);
string const func_arg = lex.getString();
LYXERR(Debug::PARSER, "ToolbarInfo::read TO_COMMAND func: `"
<< func_arg << '\'');
FuncRequest func =
lyxaction.lookupFunc(func_arg);
add(ToolbarItem(ToolbarItem::COMMAND, func, tooltip));
}
break;
case TO_MINIBUFFER:
add(ToolbarItem(ToolbarItem::MINIBUFFER,
FuncRequest(FuncCode(ToolbarItem::MINIBUFFER))));
break;
case TO_SEPARATOR:
add(ToolbarItem(ToolbarItem::SEPARATOR,
FuncRequest(FuncCode(ToolbarItem::SEPARATOR))));
break;
case TO_POPUPMENU:
if (lex.next(true)) {
string const name = lex.getString();
lex.next(true);
docstring const label = lex.getDocString();
add(ToolbarItem(ToolbarItem::POPUPMENU, name, label));
}
break;
case TO_STICKYPOPUPMENU:
if (lex.next(true)) {
string const name = lex.getString();
lex.next(true);
docstring const label = lex.getDocString();
add(ToolbarItem(ToolbarItem::STICKYPOPUPMENU, name, label));
}
break;
case TO_ICONPALETTE:
if (lex.next(true)) {
string const name = lex.getString();
lex.next(true);
docstring const label = lex.getDocString();
add(ToolbarItem(ToolbarItem::ICONPALETTE, name, label));
}
break;
case TO_LAYOUTS:
add(ToolbarItem(ToolbarItem::LAYOUTS,
FuncRequest(FuncCode(ToolbarItem::LAYOUTS))));
break;
case TO_TABLEINSERT:
if (lex.next(true)) {
docstring const tooltip = lex.getDocString();
add(ToolbarItem(ToolbarItem::TABLEINSERT,
FuncRequest(FuncCode(ToolbarItem::TABLEINSERT)), tooltip));
}
break;
case TO_ENDTOOLBAR:
quit = true;
break;
case TO_EXPORTFORMATS:
case TO_IMPORTFORMATS:
case TO_UPDATEFORMATS:
case TO_VIEWFORMATS: {
vector<Format const *> formats = (code == TO_IMPORTFORMATS) ?
theConverters().importableFormats() :
theConverters().exportableFormats(code != TO_EXPORTFORMATS);
sort(formats.begin(), formats.end());
vector<Format const *>::const_iterator fit = formats.begin();
vector<Format const *>::const_iterator end = formats.end();
for (; fit != end ; ++fit) {
if ((*fit)->dummy())
continue;
if (code != TO_IMPORTFORMATS &&
!(*fit)->documentFormat())
continue;
docstring const prettyname =
from_utf8((*fit)->prettyname());
docstring tooltip;
FuncCode lfun = LFUN_NOACTION;
switch (code) {
case TO_EXPORTFORMATS:
lfun = LFUN_BUFFER_EXPORT;
tooltip = _("Export %1$s");
break;
case TO_IMPORTFORMATS:
lfun = LFUN_BUFFER_IMPORT;
tooltip = _("Import %1$s");
break;
case TO_UPDATEFORMATS:
lfun = LFUN_BUFFER_UPDATE;
tooltip = _("Update %1$s");
break;
case TO_VIEWFORMATS:
lfun = LFUN_BUFFER_VIEW;
tooltip = _("View %1$s");
break;
}
FuncRequest func(lfun, (*fit)->name(),
FuncRequest::TOOLBAR);
add(ToolbarItem(ToolbarItem::COMMAND, func,
bformat(tooltip, prettyname)));
}
break;
}
default:
lex.printError("ToolbarInfo::read: "
"Unknown toolbar tag: `$$Token'");
break;
}
}
lex.popTable();
return *this;
}
/////////////////////////////////////////////////////////////////////////
//
// Toolbars
//
/////////////////////////////////////////////////////////////////////////
void Toolbars::reset()
{
toolbar_info_.clear();
toolbar_visibility_.clear();
}
void Toolbars::readToolbars(Lexer & lex)
{
enum {
TO_TOOLBAR = 1,
TO_ENDTOOLBARSET
};
struct LexerKeyword toolTags[] = {
{ "end", TO_ENDTOOLBARSET },
{ "toolbar", TO_TOOLBAR }
};
//consistency check
if (compare_ascii_no_case(lex.getString(), "toolbarset")) {
LYXERR0("Toolbars::readToolbars: ERROR wrong token:`"
<< lex.getString() << '\'');
}
lex.pushTable(toolTags);
if (lyxerr.debugging(Debug::PARSER))
lex.printTable(lyxerr);
bool quit = false;
while (lex.isOK() && !quit) {
switch (lex.lex()) {
case TO_TOOLBAR: {
ToolbarInfo tbinfo;
tbinfo.read(lex);
toolbar_info_.push_back(tbinfo);
break;
}
case TO_ENDTOOLBARSET:
quit = true;
break;
default:
lex.printError("Toolbars::readToolbars: "
"Unknown toolbar tag: `$$Token'");
break;
}
}
lex.popTable();
}
void Toolbars::readToolbarSettings(Lexer & lex)
{
//consistency check
if (compare_ascii_no_case(lex.getString(), "toolbars")) {
LYXERR0("Toolbars::readToolbarSettings: ERROR wrong token:`"
<< lex.getString() << '\'');
}
lex.next(true);
while (lex.isOK()) {
string name = lex.getString();
lex.next(true);
if (!compare_ascii_no_case(name, "end"))
return;
int visibility = 0;
string flagstr = lex.getString();
lex.next(true);
vector<string> flags = getVectorFromString(flagstr);
vector<string>::const_iterator cit = flags.begin();
vector<string>::const_iterator end = flags.end();
for (; cit != end; ++cit) {
Visibility flag = ON;
if (!compare_ascii_no_case(*cit, "off"))
flag = OFF;
else if (!compare_ascii_no_case(*cit, "on"))
flag = ON;
else if (!compare_ascii_no_case(*cit, "math"))
flag = MATH;
else if (!compare_ascii_no_case(*cit, "table"))
flag = TABLE;
else if (!compare_ascii_no_case(*cit, "mathmacrotemplate"))
flag = MATHMACROTEMPLATE;
else if (!compare_ascii_no_case(*cit, "review"))
flag = REVIEW;
else if (!compare_ascii_no_case(*cit, "minibuffer"))
flag = MINIBUFFER;
else if (!compare_ascii_no_case(*cit, "top"))
flag = TOP;
else if (!compare_ascii_no_case(*cit, "bottom"))
flag = BOTTOM;
else if (!compare_ascii_no_case(*cit, "left"))
flag = LEFT;
else if (!compare_ascii_no_case(*cit, "right"))
flag = RIGHT;
else if (!compare_ascii_no_case(*cit, "auto"))
flag = AUTO;
else if (!compare_ascii_no_case(*cit, "samerow"))
flag = SAMEROW;
else if (!compare_ascii_no_case(*cit, "ipa"))
flag = IPA;
else {
LYXERR(Debug::ANY,
"Toolbars::readToolbarSettings: unrecognised token:`"
<< *cit << '\'');
continue;
}
visibility |= flag;
}
toolbar_visibility_[name] = visibility;
if (visibility & ALLOWAUTO) {
if (ToolbarInfo const * ti = info(name))
const_cast<ToolbarInfo *>(ti)->gui_name +=
" (" + _("auto") + ")";
}
}
}
ToolbarInfo const * Toolbars::info(std::string const & name) const
{
Infos::const_iterator end = toolbar_info_.end();
for (Infos::const_iterator it = toolbar_info_.begin(); it != end; ++it)
if (it->name == name)
return &(*it);
return 0;
}
int Toolbars::defaultVisibility(std::string const & name) const
{
map<string, int>::const_iterator it = toolbar_visibility_.find(name);
if (it != toolbar_visibility_.end())
return it->second;
return OFF | BOTTOM;
}
bool Toolbars::isMainToolbar(std::string const & name) const
{
return toolbar_visibility_.find(name) != toolbar_visibility_.end();
}
} // namespace frontend
} // namespace lyx