mirror of
https://git.lyx.org/repos/lyx.git
synced 2024-12-21 13:01:24 +00:00
546a6a2a31
By default, inactive math corners are invisible. In practice they are
annoying because they are visible when selecting text, and they can
also overwrite some parts of the equation.
The code in Inset::drawMarkers2, which is only used for maths, is
moved to InsetMathHull. Moreover, the inactive corners are not drawn
when they have the same color as the math background. A better way to
achieve this would be to set the color to transparent, but we do not
support this yet.
(cherry picked from commit 68614e9783
)
441 lines
11 KiB
C++
441 lines
11 KiB
C++
/**
|
|
* \file MathRow.cpp
|
|
* This file is part of LyX, the document processor.
|
|
* Licence details can be found in the file COPYING.
|
|
*
|
|
* \author Jean-Marc Lasgouttes
|
|
*
|
|
* Full author contact details are available in file CREDITS.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "MathRow.h"
|
|
|
|
#include "InsetMath.h"
|
|
#include "MathClass.h"
|
|
#include "MathData.h"
|
|
#include "MathSupport.h"
|
|
|
|
#include "BufferView.h"
|
|
#include "ColorSet.h"
|
|
#include "CoordCache.h"
|
|
#include "MetricsInfo.h"
|
|
|
|
#include "frontends/FontMetrics.h"
|
|
#include "frontends/Painter.h"
|
|
|
|
#include "support/debug.h"
|
|
#include "support/docstring.h"
|
|
#include "support/lassert.h"
|
|
|
|
#include <algorithm>
|
|
#include <ostream>
|
|
|
|
using namespace std;
|
|
|
|
namespace lyx {
|
|
|
|
|
|
MathRow::Element::Element(MetricsInfo const & mi, Type t, MathClass mc)
|
|
: type(t), mclass(mc), before(0), after(0), macro_nesting(mi.base.macro_nesting),
|
|
marker(InsetMath::NO_MARKER), inset(0), compl_unique_to(0), ar(0),
|
|
color(Color_red)
|
|
{}
|
|
|
|
|
|
namespace {
|
|
|
|
// Helper functions for markers
|
|
|
|
int markerMargin(MathRow::Element const & e)
|
|
{
|
|
switch(e.marker) {
|
|
case InsetMath::MARKER:
|
|
case InsetMath::MARKER2:
|
|
case InsetMath::BOX_MARKER:
|
|
return 2;
|
|
case InsetMath::NO_MARKER:
|
|
return 0;
|
|
}
|
|
// should not happen
|
|
return 0;
|
|
}
|
|
|
|
|
|
void afterMetricsMarkers(MetricsInfo const & , MathRow::Element & e,
|
|
Dimension & dim)
|
|
{
|
|
// handle vertical space for markers
|
|
switch(e.marker) {
|
|
case InsetMath::NO_MARKER:
|
|
break;
|
|
case InsetMath::MARKER:
|
|
++dim.des;
|
|
break;
|
|
case InsetMath::MARKER2:
|
|
++dim.asc;
|
|
++dim.des;
|
|
break;
|
|
case InsetMath::BOX_MARKER:
|
|
FontInfo font;
|
|
font.setSize(FONT_SIZE_TINY);
|
|
Dimension namedim;
|
|
mathed_string_dim(font, e.inset->name(), namedim);
|
|
int const namewid = 1 + namedim.wid + 1;
|
|
|
|
if (namewid > dim.wid)
|
|
e.after += namewid - dim.wid;
|
|
++dim.asc;
|
|
dim.des += 3 + namedim.height();
|
|
}
|
|
}
|
|
|
|
|
|
void drawMarkers(PainterInfo const & pi, MathRow::Element const & e,
|
|
int const x, int const y)
|
|
{
|
|
if (e.marker == InsetMath::NO_MARKER)
|
|
return;
|
|
|
|
CoordCache const & coords = pi.base.bv->coordCache();
|
|
Dimension const dim = coords.getInsets().dim(e.inset);
|
|
|
|
// the marker is before/after the inset. Necessary space has been reserved already.
|
|
int const l = x + e.before - (markerMargin(e) > 0 ? 1 : 0);
|
|
int const r = x + dim.width() - e.after;
|
|
|
|
// Grey lower box
|
|
if (e.marker == InsetMath::BOX_MARKER) {
|
|
// draw header and rectangle around
|
|
FontInfo font;
|
|
font.setSize(FONT_SIZE_TINY);
|
|
font.setColor(Color_mathmacrolabel);
|
|
Dimension namedim;
|
|
mathed_string_dim(font, e.inset->name(), namedim);
|
|
pi.pain.fillRectangle(l, y + dim.des - namedim.height() - 2,
|
|
dim.wid, namedim.height() + 2, Color_mathmacrobg);
|
|
pi.pain.text(l, y + dim.des - namedim.des - 1, e.inset->name(), font);
|
|
}
|
|
|
|
// Color for corners
|
|
bool const highlight = e.inset->mouseHovered(pi.base.bv)
|
|
|| e.inset->editing(pi.base.bv);
|
|
ColorCode const pen_color = highlight ? Color_mathframe : Color_mathcorners;
|
|
// If the corners have the same color as the background, do not paint them.
|
|
if (lcolor.getX11Name(Color_mathbg) == lcolor.getX11Name(pen_color))
|
|
return;
|
|
|
|
// Lower corners in all cases
|
|
int const d = y + dim.descent();
|
|
pi.pain.line(l, d - 3, l, d, pen_color);
|
|
pi.pain.line(r, d - 3, r, d, pen_color);
|
|
pi.pain.line(l, d, l + 3, d, pen_color);
|
|
pi.pain.line(r - 3, d, r, d, pen_color);
|
|
|
|
// Upper corners
|
|
if (e.marker == InsetMath::BOX_MARKER
|
|
|| e.marker == InsetMath::MARKER2) {
|
|
int const a = y - dim.ascent();
|
|
pi.pain.line(l, a + 3, l, a, pen_color);
|
|
pi.pain.line(r, a + 3, r, a, pen_color);
|
|
pi.pain.line(l, a, l + 3, a, pen_color);
|
|
pi.pain.line(r - 3, a, r, a, pen_color);
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
MathRow::MathRow(MetricsInfo & mi, MathData const * ar)
|
|
{
|
|
// First there is a dummy element of type "open"
|
|
push_back(Element(mi, DUMMY, MC_OPEN));
|
|
|
|
// Then insert the MathData argument
|
|
bool const has_contents = ar->addToMathRow(*this, mi);
|
|
|
|
// A MathRow should not be completely empty
|
|
if (!has_contents) {
|
|
Element e(mi, BOX, MC_ORD);
|
|
// empty arrays are visible when they are editable
|
|
e.color = mi.base.macro_nesting == 0 ? Color_mathline : Color_none;
|
|
push_back(e);
|
|
}
|
|
|
|
// Finally there is a dummy element of type "close"
|
|
push_back(Element(mi, DUMMY, MC_CLOSE));
|
|
|
|
/* Do spacing only in math mode. This test is a bit clumsy,
|
|
* but it is used in other places for guessing the current mode.
|
|
*/
|
|
bool const dospacing = isMathFont(mi.base.fontname);
|
|
|
|
// update classes
|
|
if (dospacing) {
|
|
for (int i = 1 ; i != static_cast<int>(elements_.size()) - 1 ; ++i) {
|
|
if (elements_[i].mclass != MC_UNKNOWN)
|
|
update_class(elements_[i].mclass, elements_[before(i)].mclass,
|
|
elements_[after(i)].mclass);
|
|
}
|
|
}
|
|
|
|
// set spacing
|
|
// We go to the end to handle spacing at the end of equation
|
|
for (int i = 1 ; i != static_cast<int>(elements_.size()) ; ++i) {
|
|
Element & e = elements_[i];
|
|
|
|
Element & bef = elements_[before(i)];
|
|
if (dospacing && e.mclass != MC_UNKNOWN) {
|
|
int spc = class_spacing(bef.mclass, e.mclass, mi.base);
|
|
bef.after += spc / 2;
|
|
// this is better than spc / 2 to avoid rounding problems
|
|
e.before += spc - spc / 2;
|
|
}
|
|
|
|
// finally reserve space for markers
|
|
bef.after = max(bef.after, markerMargin(bef));
|
|
if (e.mclass != MC_UNKNOWN)
|
|
e.before = max(e.before, markerMargin(e));
|
|
// for linearized insets (macros...) too
|
|
if (e.type == BEGIN)
|
|
bef.after = max(bef.after, markerMargin(e));
|
|
if (e.type == END && e.marker != InsetMath::NO_MARKER) {
|
|
Element & aft = elements_[after(i)];
|
|
aft.before = max(aft.before, markerMargin(e));
|
|
}
|
|
}
|
|
|
|
// Do not lose spacing allocated to extremities
|
|
if (!elements_.empty()) {
|
|
elements_[after(0)].before += elements_.front().after;
|
|
elements_[before(elements_.size() - 1)].after += elements_.back().before;
|
|
}
|
|
}
|
|
|
|
|
|
int MathRow::before(int i) const
|
|
{
|
|
do
|
|
--i;
|
|
while (elements_[i].mclass == MC_UNKNOWN);
|
|
|
|
return i;
|
|
}
|
|
|
|
|
|
int MathRow::after(int i) const
|
|
{
|
|
do
|
|
++i;
|
|
while (elements_[i].mclass == MC_UNKNOWN);
|
|
|
|
return i;
|
|
}
|
|
|
|
|
|
void MathRow::metrics(MetricsInfo & mi, Dimension & dim)
|
|
{
|
|
dim.asc = 0;
|
|
dim.wid = 0;
|
|
// In order to compute the dimension of macros and their
|
|
// arguments, it is necessary to keep track of them.
|
|
vector<pair<InsetMath const *, Dimension>> dim_insets;
|
|
vector<pair<MathData const *, Dimension>> dim_arrays;
|
|
CoordCache & coords = mi.base.bv->coordCache();
|
|
for (Element & e : elements_) {
|
|
mi.base.macro_nesting = e.macro_nesting;
|
|
Dimension d;
|
|
switch (e.type) {
|
|
case DUMMY:
|
|
break;
|
|
case INSET:
|
|
e.inset->metrics(mi, d);
|
|
d.wid += e.before + e.after;
|
|
coords.insets().add(e.inset, d);
|
|
break;
|
|
case BEGIN:
|
|
if (e.inset) {
|
|
dim_insets.push_back(make_pair(e.inset, Dimension()));
|
|
dim_insets.back().second.wid += e.before + e.after;
|
|
d.wid = e.before + e.after;
|
|
e.inset->beforeMetrics();
|
|
}
|
|
if (e.ar)
|
|
dim_arrays.push_back(make_pair(e.ar, Dimension()));
|
|
break;
|
|
case END:
|
|
if (e.inset) {
|
|
e.inset->afterMetrics();
|
|
LATTEST(dim_insets.back().first == e.inset);
|
|
d = dim_insets.back().second;
|
|
afterMetricsMarkers(mi, e, d);
|
|
d.wid += e.before + e.after;
|
|
coords.insets().add(e.inset, d);
|
|
dim_insets.pop_back();
|
|
// We do not want to count the width again, but the
|
|
// padding and the vertical dimension are meaningful.
|
|
d.wid = e.before + e.after;
|
|
}
|
|
if (e.ar) {
|
|
LATTEST(dim_arrays.back().first == e.ar);
|
|
coords.arrays().add(e.ar, dim_arrays.back().second);
|
|
dim_arrays.pop_back();
|
|
}
|
|
break;
|
|
case BOX:
|
|
d = theFontMetrics(mi.base.font).dimension('I');
|
|
if (e.color != Color_none) {
|
|
// allow for one pixel before/after the box.
|
|
d.wid += e.before + e.after + 2;
|
|
} else {
|
|
// hide the box, but keep its height
|
|
d.wid = 0;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (!d.empty()) {
|
|
dim += d;
|
|
// Now add the dimension to current macros and arguments.
|
|
for (auto & dim_macro : dim_insets)
|
|
dim_macro.second += d;
|
|
for (auto & dim_array : dim_arrays)
|
|
dim_array.second += d;
|
|
}
|
|
|
|
if (e.compl_text.empty())
|
|
continue;
|
|
FontInfo font = mi.base.font;
|
|
augmentFont(font, "mathnormal");
|
|
dim.wid += mathed_string_width(font, e.compl_text);
|
|
}
|
|
LATTEST(dim_insets.empty() && dim_arrays.empty());
|
|
}
|
|
|
|
|
|
void MathRow::draw(PainterInfo & pi, int x, int const y) const
|
|
{
|
|
CoordCache & coords = pi.base.bv->coordCache();
|
|
for (Element const & e : elements_) {
|
|
switch (e.type) {
|
|
case INSET: {
|
|
// This is hackish: the math inset does not know that space
|
|
// has been added before and after it; we alter its dimension
|
|
// while it is drawing, because it relies on this value.
|
|
Dimension const d = coords.insets().dim(e.inset);
|
|
Dimension d2 = d;
|
|
d2.wid -= e.before + e.after;
|
|
coords.insets().add(e.inset, d2);
|
|
e.inset->draw(pi, x + e.before, y);
|
|
coords.insets().add(e.inset, x, y);
|
|
coords.insets().add(e.inset, d);
|
|
drawMarkers(pi, e, x, y);
|
|
x += d.wid;
|
|
break;
|
|
}
|
|
case BEGIN:
|
|
if (e.ar) {
|
|
coords.arrays().add(e.ar, x, y);
|
|
e.ar->drawSelection(pi, x, y);
|
|
}
|
|
if (e.inset) {
|
|
coords.insets().add(e.inset, x, y);
|
|
drawMarkers(pi, e, x, y);
|
|
e.inset->beforeDraw(pi);
|
|
}
|
|
x += e.before + e.after;
|
|
break;
|
|
case END:
|
|
if (e.inset)
|
|
e.inset->afterDraw(pi);
|
|
x += e.before + e.after;
|
|
break;
|
|
case BOX: {
|
|
if (e.color == Color_none)
|
|
break;
|
|
Dimension const d = theFontMetrics(pi.base.font).dimension('I');
|
|
pi.pain.rectangle(x + e.before + 1, y - d.ascent(),
|
|
d.width() - 1, d.height() - 1, e.color);
|
|
x += d.wid + 2 + e.before + e.after;
|
|
break;
|
|
}
|
|
case DUMMY:
|
|
break;
|
|
}
|
|
|
|
if (e.compl_text.empty())
|
|
continue;
|
|
FontInfo f = pi.base.font;
|
|
augmentFont(f, "mathnormal");
|
|
|
|
// draw the unique and the non-unique completion part
|
|
// Note: this is not time-critical as it is
|
|
// only done once per screen.
|
|
docstring const s1 = e.compl_text.substr(0, e.compl_unique_to);
|
|
docstring const s2 = e.compl_text.substr(e.compl_unique_to);
|
|
|
|
if (!s1.empty()) {
|
|
f.setColor(Color_inlinecompletion);
|
|
pi.pain.text(x, y, s1, f);
|
|
x += mathed_string_width(f, s1);
|
|
}
|
|
if (!s2.empty()) {
|
|
f.setColor(Color_nonunique_inlinecompletion);
|
|
pi.pain.text(x, y, s2, f);
|
|
x += mathed_string_width(f, s2);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
int MathRow::kerning(BufferView const * bv) const
|
|
{
|
|
if (elements_.empty())
|
|
return 0;
|
|
InsetMath const * inset = elements_[before(elements_.size() - 1)].inset;
|
|
return inset ? inset->kerning(bv) : 0;
|
|
}
|
|
|
|
|
|
ostream & operator<<(ostream & os, MathRow::Element const & e)
|
|
{
|
|
switch (e.type) {
|
|
case MathRow::DUMMY:
|
|
os << (e.mclass == MC_OPEN ? "{" : "}");
|
|
break;
|
|
case MathRow::INSET:
|
|
os << "<" << e.before << "-"
|
|
<< to_utf8(class_to_string(e.mclass))
|
|
<< "-" << e.after << ">";
|
|
break;
|
|
case MathRow::BEGIN:
|
|
if (e.inset)
|
|
os << "\\" << to_utf8(e.inset->name())
|
|
<< "^" << e.macro_nesting << "[";
|
|
if (e.ar)
|
|
os << "(";
|
|
break;
|
|
case MathRow::END:
|
|
if (e.ar)
|
|
os << ")";
|
|
if (e.inset)
|
|
os << "]";
|
|
break;
|
|
case MathRow::BOX:
|
|
os << "<" << e.before << "-[]-" << e.after << ">";
|
|
break;
|
|
}
|
|
return os;
|
|
}
|
|
|
|
|
|
ostream & operator<<(ostream & os, MathRow const & mrow)
|
|
{
|
|
for (MathRow::Element const & e : mrow)
|
|
os << e << " ";
|
|
return os;
|
|
}
|
|
|
|
} // namespace lyx
|