-/* This file is part of
- * ======================================================
- *
- * LyX, The Document Processor
- *
- * Copyright 1995 Matthias Ettrich
- * Copyright 1995-1999 The LyX Team.
+/**
+ * \file insetcommand.C
+ * This file is part of LyX, the document processor.
+ * Licence details can be found in the file COPYING.
*
- * ====================================================== */
+ * \author Angus Leeming
+ * \author Lars Gullik Bjønnes
+ *
+ * Full author contact details are available in file CREDITS
+ */
#include <config.h>
-#ifdef __GNUG__
-#pragma implementation
-#endif
-
#include "insetcommand.h"
-#include "lyxdraw.h"
+#include "BufferView.h"
#include "debug.h"
+#include "funcrequest.h"
+#include "lyxlex.h"
+#include "metricsinfo.h"
-InsetCommand::InsetCommand()
-{
-}
+#include "frontends/Painter.h"
+#include "support/lstrings.h"
-InsetCommand::InsetCommand(string const & cmd, string const & arg,
- string const & opt)
- : command(cmd), options(opt), contents(arg)
-{
-}
+#include "Lsstream.h"
+using std::ostream;
-int InsetCommand::Ascent(LyXFont const & font) const
-{
- LyXFont f = font;
- f.decSize();
- return f.maxAscent() + 3;
-}
+
+InsetCommand::InsetCommand(InsetCommandParams const & p)
+ : p_(p.getCmdName(), p.getContents(), p.getOptions()),
+ set_label_(false)
+{}
-int InsetCommand::Descent(LyXFont const & font) const
+BufferView * InsetCommand::view() const
{
- LyXFont f = font;
- f.decSize();
- return f.maxDescent() + 3;
+ return button_.view();
}
-int InsetCommand::Width(LyXFont const & font) const
+void InsetCommand::metrics(MetricsInfo & mi, Dimension & dim) const
{
- LyXFont f = font;
- f.decSize();
- string s = getScreenLabel();
- return 10 + f.stringWidth(s);
+ if (!set_label_) {
+ set_label_ = true;
+ button_.update(getScreenLabel(mi.base.bv->buffer()),
+ editable() != NOT_EDITABLE);
+ }
+ button_.metrics(mi, dim);
+ dim_ = dim;
}
-void InsetCommand::Draw(LyXFont font, LyXScreen & scr,
- int baseline, float & x)
+void InsetCommand::draw(PainterInfo & pi, int x, int y) const
{
- // Draw it as a box with the LaTeX text
- x += 3;
-
- scr.fillRectangle(gc_lighted,
- int(x), baseline - Ascent(font) + 1,
- Width(font) - 6,
- Ascent(font) + Descent(font)-2);
- // Tell whether this slows down the drawing (ale)
- // lets draw editable and non-editable insets differently
- if (Editable()) {
- int y = baseline - Ascent(font)+1, w = Width(font)-6,
- h = (Ascent(font)+Descent(font)-2);
- scr.drawFrame(FL_UP_FRAME, int(x), y, w, h, FL_BLACK, -1);
- } else {
- scr.drawRectangle(gc_note_frame,
- int(x), baseline - Ascent(font)+1,
- Width(font)-6,
- Ascent(font)+Descent(font)-2);
- }
- string s = getScreenLabel();
- LyXFont f = font;
- f.decSize();
- f.setColor(LyXFont::NONE);
- f.setLatex(LyXFont::OFF);
- scr.drawString(f, s, baseline, int(x+2));
-
- x += Width(font) - 3;
+ button_.draw(pi, x, y);
}
-// In lyxf3 this will be just LaTeX
-void InsetCommand::Write(FILE * file)
+void InsetCommand::setParams(InsetCommandParams const & p)
{
- fprintf(file, "LatexCommand %s\n", getCommand().c_str());
+ p_.setCmdName(p.getCmdName());
+ p_.setContents(p.getContents());
+ p_.setOptions(p.getOptions());
+ set_label_ = false;
}
-void InsetCommand::scanCommand(string const & cmd)
+int InsetCommand::latex(Buffer const *, ostream & os,
+ LatexRunParams const &) const
{
- string tcommand, toptions, tcontents;
-
- if (cmd.empty()) return;
-
- enum { WS, Command, Option, Content } state = WS;
-
- // Used to handle things like \command[foo[bar]]{foo{bar}}
- int nestdepth = 0;
-
- for (string::size_type i = 0; i < cmd.length(); ++i) {
- char c = cmd[i];
- if ((state == Command && c == ' ') ||
- (state == Command && c == '[') ||
- (state == Command && c == '{')) {
- state = WS;
- }
- if ((state == Option && c == ']') ||
- (state == Content && c == '}')) {
- if (nestdepth == 0) {
- state = WS;
- } else {
- nestdepth--;
- }
- }
- if ((state == Option && c == '[') ||
- (state == Content && c == '{')) {
- nestdepth++;
- }
- switch (state) {
- case Command: tcommand += c; break;
- case Option: toptions += c; break;
- case Content: tcontents += c; break;
- case WS:
- if (c == '\\') {
- state = Command;
- } else if (c == '[') {
- state = Option;
- nestdepth = 0; // Just to be sure
- } else if (c == '{') {
- state = Content;
- nestdepth = 0; // Just to be sure
- }
- break;
- }
- }
-
- // Don't mess with this.
- if (!tcommand.empty()) command = tcommand;
- if (!toptions.empty()) options = toptions;
- if (!tcontents.empty()) setContents(tcontents);
- // setContents is overloaded in InsetInclude
-
- if (lyxerr.debugging(Debug::PARSER))
- lyxerr << "Command <" << cmd
- << "> == <" << getCommand()
- << "> == <" << getCmdName()
- << '|' << getContents()
- << '|' << getOptions() << '>' << endl;
+ os << getCommand();
+ return 0;
}
-// This function will not be necessary when lyx3
-void InsetCommand::Read(LyXLex & lex)
-{
- if (lex.EatLine()) {
- string t = lex.GetString();
- scanCommand(t);
- } else
- lex.printError("InsetCommand: Parse error: `$$Token'");
+int InsetCommand::ascii(Buffer const *, ostream &, int) const
+{
+ return 0;
}
-int InsetCommand::Latex(FILE * file, signed char /*fragile*/)
+int InsetCommand::linuxdoc(Buffer const *, ostream &) const
{
- fprintf(file, "%s", getCommand().c_str());
return 0;
}
-int InsetCommand::Latex(string & file, signed char /*fragile*/)
+int InsetCommand::docbook(Buffer const *, ostream &, bool) const
{
- file += getCommand();
return 0;
}
-int InsetCommand::Linuxdoc(string &/*file*/)
+dispatch_result InsetCommand::localDispatch(FuncRequest const & cmd)
{
- return 0;
+ switch (cmd.action) {
+ case LFUN_INSET_MODIFY: {
+ InsetCommandParams p;
+ InsetCommandMailer::string2params(cmd.argument, p);
+ if (p.getCmdName().empty())
+ return UNDISPATCHED;
+
+ setParams(p);
+ cmd.view()->updateInset(this);
+ return DISPATCHED;
+ }
+
+ case LFUN_INSET_DIALOG_UPDATE:
+ InsetCommandMailer(cmd.argument, *this).updateDialog(cmd.view());
+ return DISPATCHED;
+
+ case LFUN_MOUSE_RELEASE:
+ return localDispatch(FuncRequest(cmd.view(), LFUN_INSET_EDIT));
+
+ default:
+ return InsetOld::localDispatch(cmd);
+ }
+
}
-int InsetCommand::DocBook(string &/*file*/)
+InsetCommandMailer::InsetCommandMailer(string const & name,
+ InsetCommand & inset)
+ : name_(name), inset_(inset)
+{}
+
+
+string const InsetCommandMailer::inset2string(Buffer const &) const
{
- return 0;
+ return params2string(name(), inset_.params());
}
-Inset * InsetCommand::Clone() const
+void InsetCommandMailer::string2params(string const & in,
+ InsetCommandParams & params)
{
- return new InsetCommand(command, contents, options);
+ params.setCmdName(string());
+ params.setContents(string());
+ params.setOptions(string());
+
+ if (in.empty())
+ return;
+
+ istringstream data(STRCONV(in));
+ LyXLex lex(0,0);
+ lex.setStream(data);
+
+ if (lex.isOK()) {
+ lex.next();
+ string const name = lex.getString();
+ }
+
+ // This is part of the inset proper that is usually swallowed
+ // by Buffer::readInset
+ if (lex.isOK()) {
+ lex.next();
+ string const token = lex.getString();
+ if (token != "LatexCommand")
+ return;
+ }
+ if (lex.isOK()) {
+ params.read(lex);
+ }
}
-string InsetCommand::getCommand() const
-{
- string s;
- if (!command.empty()) s += "\\"+command;
- if (!options.empty()) s += "["+options+']';
- s += "{"+contents+'}';
- return s;
+string const InsetCommandMailer::params2string(string const & name,
+ InsetCommandParams const & params)
+{
+ ostringstream data;
+ data << name << ' ';
+ params.write(data);
+ data << "\\end_inset\n";
+ return STRCONV(data.str());
}