]> git.lyx.org Git - lyx.git/blobdiff - src/mathed/math_symbolinset.C
Fix math cursor positioning bug
[lyx.git] / src / mathed / math_symbolinset.C
index 1afc2938aa6b321b0623b84a9b41d1a828c0eb9e..2d07bd6240e7cca2e9b7e97887598dbb552b7805 100644 (file)
+/**
+ * \file math_symbolinset.C
+ * This file is part of LyX, the document processor.
+ * Licence details can be found in the file COPYING.
+ *
+ * \author André Pönitz
+ *
+ * Full author contact details are available in file CREDITS.
+ */
+
+#include <config.h>
+
 #include "math_symbolinset.h"
-#include "mathed/math_parser.h"
-#include "support/LOstream.h"
+#include "dimension.h"
+#include "math_mathmlstream.h"
+#include "math_streamstr.h"
+#include "math_support.h"
+#include "math_parser.h"
+#include "math_atom.h"
+#include "LaTeXFeatures.h"
+#include "debug.h"
+
+
+using std::string;
+using std::auto_ptr;
+
+
+MathSymbolInset::MathSymbolInset(latexkeys const * l)
+       : sym_(l), h_(0)
+{}
+
 
+MathSymbolInset::MathSymbolInset(char const * name)
+       : sym_(in_word_set(name)), h_(0)
+{}
 
-using std::ostream;
 
-MathSymbolInset::MathSymbolInset(const latexkeys * l)
-       : sym_(l)
+MathSymbolInset::MathSymbolInset(string const & name)
+       : sym_(in_word_set(name.c_str())), h_(0)
 {}
 
 
-MathInset * MathSymbolInset::clone() const
+auto_ptr<InsetBase> MathSymbolInset::doClone() const
 {
-       return new MathSymbolInset(*this);
+       return auto_ptr<InsetBase>(new MathSymbolInset(*this));
 }
 
 
-void MathSymbolInset::write(ostream & os, bool /* fragile */) const
+string MathSymbolInset::name() const
 {
-       os << '\\' << sym_->name << ' ';
+       return sym_->name;
 }
 
 
-void MathSymbolInset::writeNormal(ostream & os) const
+void MathSymbolInset::metrics(MetricsInfo & mi, Dimension & dim) const
 {
-       os << "[bigop " << sym_->name << "] ";
+       //lyxerr << "metrics: symbol: '" << sym_->name
+       //      << "' in font: '" << sym_->inset
+       //      << "' drawn as: '" << sym_->draw
+       //      << "'" << std::endl;
+
+       int const em = mathed_char_width(mi.base.font, 'M');
+       FontSetChanger dummy(mi.base, sym_->inset.c_str());
+       mathed_string_dim(mi.base.font, sym_->draw, dim);
+       // correct height for broken cmex and wasy font
+#if defined(__APPLE__) && defined(__GNUC__)
+       if (sym_->inset == "cmex") {
+               h_ = 4 * dim.des / 5;
+               dim.asc += 0*h_;
+               dim.des -= h_;
+               h_ = dim.asc;
+       } else if (sym_->inset == "wasy") {
+               h_ = 4 * dim.des / 5;
+               dim.asc += h_;
+               dim.des -= h_;
+       }
+#else
+       if (sym_->inset == "cmex" || sym_->inset == "wasy") {
+               h_ = 4 * dim.des / 5;
+               dim.asc += h_;
+               dim.des -= h_;
+       }
+#endif
+       // seperate things a bit
+       if (isRelOp())
+               dim.wid += static_cast<int>(0.5 * em + 0.5);
+       else
+               dim.wid += static_cast<int>(0.1667 * em + 0.5);
+
+       scriptable_ = false;
+       if (mi.base.style == LM_ST_DISPLAY)
+               if (sym_->inset == "cmex" || sym_->extra == "funclim")
+                       scriptable_ = true;
+
+       width_ = dim.wid;
 }
 
 
-void MathSymbolInset::metrics(MathStyles st) const
+void MathSymbolInset::draw(PainterInfo & pi, int x, int y) const
 {
-       size(st);
-       
-       if (sym_->id > 0 && sym_->id < 256) {
-               ssym_ = string();
-               ssym_ += sym_->id;
-               code_ = (sym_->token == LM_TK_BIGSYM) ? LM_TC_BSYM : LM_TC_SYMB;
-       } else {
-               ssym_ = sym_->name;
-               code_ = LM_TC_TEXTRM;
-       }
+       //lyxerr << "metrics: symbol: '" << sym_->name
+       //      << "' in font: '" << sym_->inset
+       //      << "' drawn as: '" << sym_->draw
+       //      << "'" << std::endl;
+       int const em = mathed_char_width(pi.base.font, 'M');
+       if (isRelOp())
+               x += static_cast<int>(0.25*em+0.5);
+       else
+               x += static_cast<int>(0.0833*em+0.5);
+
+       FontSetChanger dummy(pi.base, sym_->inset.c_str());
+       drawStr(pi, pi.base.font, x, y - h_, sym_->draw);
+}
+
+
+bool MathSymbolInset::isRelOp() const
+{
+       return sym_->extra == "mathrel";
+}
+
+
+bool MathSymbolInset::isScriptable() const
+{
+       return scriptable_;
+}
+
+
+bool MathSymbolInset::takesLimits() const
+{
+       return
+               sym_->inset == "cmex" ||
+               sym_->inset == "lyxboldsymb" ||
+               sym_->extra == "funclim";
+}
+
+
+void MathSymbolInset::validate(LaTeXFeatures & features) const
+{
+       if (sym_->inset == "msa" || sym_->inset == "msb")
+               features.require("amssymb");
+       else if (sym_->inset == "wasy")
+               features.require("wasysym");
+}
+
+
+void MathSymbolInset::normalize(NormalStream & os) const
+{
+       os << "[symbol " << name() << ']';
+}
+
+
+void MathSymbolInset::maple(MapleStream & os) const
+{
+       if (name() == "cdot")
+               os << '*';
+       else if (name() == "infty")
+               os << "infinity";
+       else
+               os << name();
+}
 
-       mathed_string_dim(code_, size(), ssym_, ascent_, descent_, width_);
+void MathSymbolInset::maxima(MaximaStream & os) const
+{
+       if (name() == "cdot")
+               os << '*';
+       else if (name() == "infty")
+               os << "INF";
+       else
+               os << name();
 }
 
 
-void MathSymbolInset::draw(Painter & pain, int x, int y) const
-{  
-       xo(x);
-       yo(y);
+void MathSymbolInset::mathematica(MathematicaStream & os) const
+{
+       if ( name() == "pi")    { os << "Pi"; return;}
+       if ( name() == "infty") { os << "Infinity"; return;}
+       os << name();
+}
+
 
-       drawStr(pain, code_, size_, x, y, ssym_);
+char const * MathMLtype(string const & s)
+{
+       if (s == "mathop")
+               return "mo";
+       return "mi";
+}
+
+
+void MathSymbolInset::mathmlize(MathMLStream & os) const
+{
+       char const * type = MathMLtype(sym_->extra);
+       os << '<' << type << "> ";
+       if (sym_->xmlname == "x") // unknown so far
+               os << name();
+       else
+               os << sym_->xmlname;
+       os << " </" << type << '>';
+}
+
+
+void MathSymbolInset::octave(OctaveStream & os) const
+{
+       if (name() == "cdot")
+               os << '*';
+       else
+               os << name();
+}
+
+
+void MathSymbolInset::write(WriteStream & os) const
+{
+       os << '\\' << name();
+       os.pendingSpace(true);
+}
+
+
+void MathSymbolInset::infoize2(std::ostream & os) const
+{
+       os << "Symbol: " << name();
 }