2003-02-14 00:41:44 +00:00
|
|
|
/**
|
2007-04-26 04:41:58 +00:00
|
|
|
* \file Row.cpp
|
2003-02-14 00:41:44 +00:00
|
|
|
* This file is part of LyX, the document processor.
|
|
|
|
* Licence details can be found in the file COPYING.
|
2002-03-21 17:27:08 +00:00
|
|
|
*
|
2008-11-14 15:58:50 +00:00
|
|
|
* \author Lars Gullik Bjønnes
|
2003-08-23 00:17:00 +00:00
|
|
|
* \author John Levon
|
2008-11-14 15:58:50 +00:00
|
|
|
* \author André Pönitz
|
|
|
|
* \author Jürgen Vigna
|
2013-07-17 22:25:08 +00:00
|
|
|
* \author Jean-Marc Lasgouttes
|
2002-03-21 17:27:08 +00:00
|
|
|
*
|
2003-08-23 00:17:00 +00:00
|
|
|
* Full author contact details are available in file CREDITS.
|
2000-06-08 23:16:16 +00:00
|
|
|
*
|
2003-02-14 00:41:44 +00:00
|
|
|
* Metrics for an on-screen text row.
|
|
|
|
*/
|
2000-06-08 23:16:16 +00:00
|
|
|
|
2000-06-15 15:54:05 +00:00
|
|
|
#include <config.h>
|
|
|
|
|
2007-04-26 04:41:58 +00:00
|
|
|
#include "Row.h"
|
2008-02-18 07:14:42 +00:00
|
|
|
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
#include "DocIterator.h"
|
|
|
|
|
2013-06-25 12:57:09 +00:00
|
|
|
#include "frontends/FontMetrics.h"
|
|
|
|
|
2007-11-29 07:04:28 +00:00
|
|
|
#include "support/debug.h"
|
2013-07-17 22:25:08 +00:00
|
|
|
#include "support/lassert.h"
|
2000-06-08 23:16:16 +00:00
|
|
|
|
2013-07-17 22:25:08 +00:00
|
|
|
#include <algorithm>
|
2013-06-25 12:57:09 +00:00
|
|
|
#include <ostream>
|
|
|
|
|
2014-05-19 09:35:15 +00:00
|
|
|
#include <boost/next_prior.hpp>
|
|
|
|
|
2013-06-25 12:57:09 +00:00
|
|
|
using namespace std;
|
2006-10-21 00:16:43 +00:00
|
|
|
|
|
|
|
namespace lyx {
|
2003-02-14 00:41:44 +00:00
|
|
|
|
2013-07-17 22:25:08 +00:00
|
|
|
using frontend::FontMetrics;
|
|
|
|
|
|
|
|
double Row::Element::pos2x(pos_type const i) const
|
|
|
|
{
|
2014-05-14 15:46:43 +00:00
|
|
|
// This can happen with inline completion when clicking on the
|
|
|
|
// row after the completion.
|
|
|
|
if (i < pos || i > endpos)
|
|
|
|
return 0;
|
2013-10-18 15:55:30 +00:00
|
|
|
|
2013-07-17 22:25:08 +00:00
|
|
|
bool const rtl = font.isVisibleRightToLeft();
|
|
|
|
|
2014-07-28 07:46:13 +00:00
|
|
|
double w = 0;
|
2013-10-18 15:55:30 +00:00
|
|
|
//handle first the two bounds of the element
|
2014-08-06 00:21:18 +00:00
|
|
|
if (i == endpos && !(inset && inset->lyxCode() == SEPARATOR_CODE))
|
2014-05-14 15:46:43 +00:00
|
|
|
w = rtl ? 0 : width();
|
2014-07-28 20:03:57 +00:00
|
|
|
else if (i == pos || type != STRING)
|
|
|
|
w = rtl ? width() : 0;
|
2013-10-18 15:55:30 +00:00
|
|
|
else {
|
|
|
|
FontMetrics const & fm = theFontMetrics(font);
|
2014-05-14 15:46:43 +00:00
|
|
|
w = fm.pos2x(str, i - pos, font.isVisibleRightToLeft());
|
2013-10-18 15:55:30 +00:00
|
|
|
}
|
2013-07-17 22:25:08 +00:00
|
|
|
|
2014-05-14 15:46:43 +00:00
|
|
|
return w;
|
2013-07-17 22:25:08 +00:00
|
|
|
}
|
|
|
|
|
2000-06-08 23:16:16 +00:00
|
|
|
|
2014-05-14 15:46:43 +00:00
|
|
|
pos_type Row::Element::x2pos(double &x) const
|
2013-07-21 18:22:32 +00:00
|
|
|
{
|
|
|
|
//lyxerr << "x2pos: x=" << x << " w=" << width() << " " << *this;
|
|
|
|
bool const rtl = font.isVisibleRightToLeft();
|
2014-03-19 13:44:53 +00:00
|
|
|
size_t i = 0;
|
2014-05-14 15:46:43 +00:00
|
|
|
|
2014-03-21 10:56:42 +00:00
|
|
|
switch (type) {
|
|
|
|
case STRING: {
|
|
|
|
FontMetrics const & fm = theFontMetrics(font);
|
2014-05-14 15:46:43 +00:00
|
|
|
// FIXME: is it really necessary for x to be a double?
|
2014-07-28 07:46:13 +00:00
|
|
|
int xx = int(x);
|
2014-05-14 15:46:43 +00:00
|
|
|
i = fm.x2pos(str, xx, rtl);
|
|
|
|
x = xx;
|
2014-03-21 10:56:42 +00:00
|
|
|
break;
|
|
|
|
}
|
2014-05-14 15:46:43 +00:00
|
|
|
case VIRTUAL:
|
|
|
|
// those elements are actually empty (but they have a width)
|
|
|
|
i = 0;
|
|
|
|
x = rtl ? width() : 0;
|
|
|
|
break;
|
2014-03-21 10:56:42 +00:00
|
|
|
case SEPARATOR:
|
|
|
|
case INSET:
|
|
|
|
case SPACE:
|
2014-05-14 15:46:43 +00:00
|
|
|
// those elements contain only one position. Round to
|
|
|
|
// the closest side.
|
|
|
|
if (x > width()) {
|
|
|
|
x = width();
|
|
|
|
i = !rtl;
|
|
|
|
} else {
|
|
|
|
x = 0;
|
|
|
|
i = rtl;
|
|
|
|
}
|
2013-07-21 18:22:32 +00:00
|
|
|
|
2014-05-14 15:46:43 +00:00
|
|
|
}
|
2013-12-20 13:02:31 +00:00
|
|
|
//lyxerr << "=> p=" << pos + i << " x=" << x << endl;
|
2013-07-21 18:22:32 +00:00
|
|
|
return pos + i;
|
2014-05-14 15:46:43 +00:00
|
|
|
|
2013-07-21 18:22:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-05-19 09:35:15 +00:00
|
|
|
bool Row::Element::breakAt(double w)
|
|
|
|
{
|
|
|
|
if (type != STRING || width() <= w)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bool const rtl = font.isVisibleRightToLeft();
|
|
|
|
if (rtl)
|
|
|
|
w = width() - w;
|
|
|
|
pos_type new_pos = x2pos(w);
|
|
|
|
if (new_pos == pos)
|
|
|
|
return false;
|
|
|
|
str = str.substr(0, new_pos - pos);
|
|
|
|
if (rtl)
|
2014-07-28 07:46:13 +00:00
|
|
|
dim.wid -= int(w);
|
2014-05-19 09:35:15 +00:00
|
|
|
else
|
2014-07-28 07:46:13 +00:00
|
|
|
dim.wid = int(w);
|
2014-05-19 09:35:15 +00:00
|
|
|
endpos = new_pos;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-21 10:56:42 +00:00
|
|
|
pos_type Row::Element::left_pos() const
|
|
|
|
{
|
|
|
|
return font.isVisibleRightToLeft() ? endpos : pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pos_type Row::Element::right_pos() const
|
|
|
|
{
|
|
|
|
return font.isVisibleRightToLeft() ? pos : endpos;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-06-08 23:16:16 +00:00
|
|
|
Row::Row()
|
2013-07-21 18:22:32 +00:00
|
|
|
: separator(0), label_hfill(0), x(0), right_margin(0),
|
2013-07-23 14:24:01 +00:00
|
|
|
sel_beg(-1), sel_end(-1),
|
|
|
|
begin_margin_sel(false), end_margin_sel(false),
|
|
|
|
changed_(false), crc_(0), pos_(0), end_(0), right_boundary_(false)
|
2000-06-08 23:16:16 +00:00
|
|
|
{}
|
|
|
|
|
|
|
|
|
2007-11-17 11:27:03 +00:00
|
|
|
void Row::setCrc(size_type crc) const
|
2000-06-08 23:16:16 +00:00
|
|
|
{
|
2007-11-17 11:27:03 +00:00
|
|
|
changed_ = crc != crc_;
|
2007-08-29 21:03:41 +00:00
|
|
|
crc_ = crc;
|
2000-06-08 23:16:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-09-14 22:09:10 +00:00
|
|
|
bool Row::isMarginSelected(bool left_margin, DocIterator const & beg,
|
|
|
|
DocIterator const & end) const
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
{
|
2008-09-14 22:09:10 +00:00
|
|
|
pos_type const sel_pos = left_margin ? sel_beg : sel_end;
|
|
|
|
pos_type const margin_pos = left_margin ? pos_ : end_;
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
|
|
|
|
// Is the chosen margin selected ?
|
|
|
|
if (sel_pos == margin_pos) {
|
|
|
|
if (beg.pos() == end.pos())
|
2013-06-25 12:57:09 +00:00
|
|
|
// This is a special case in which the space between after
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
// pos i-1 and before pos i is selected, i.e. the margins
|
|
|
|
// (see DocIterator::boundary_).
|
|
|
|
return beg.boundary() && !end.boundary();
|
|
|
|
else if (end.pos() == margin_pos)
|
|
|
|
// If the selection ends around the margin, it is only
|
|
|
|
// drawn if the cursor is after the margin.
|
|
|
|
return !end.boundary();
|
|
|
|
else if (beg.pos() == margin_pos)
|
2013-06-25 12:57:09 +00:00
|
|
|
// If the selection begins around the margin, it is
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
// only drawn if the cursor is before the margin.
|
|
|
|
return beg.boundary();
|
2013-06-25 12:57:09 +00:00
|
|
|
else
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-25 12:57:09 +00:00
|
|
|
void Row::setSelectionAndMargins(DocIterator const & beg,
|
2008-09-14 22:09:10 +00:00
|
|
|
DocIterator const & end) const
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
{
|
|
|
|
setSelection(beg.pos(), end.pos());
|
2013-06-25 12:57:09 +00:00
|
|
|
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
if (selection()) {
|
2008-10-13 23:19:06 +00:00
|
|
|
end_margin_sel = isMarginSelected(false, beg, end);
|
|
|
|
begin_margin_sel = isMarginSelected(true, beg, end);
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-03-03 11:03:47 +00:00
|
|
|
void Row::setSelection(pos_type beg, pos_type end) const
|
2007-09-05 13:04:05 +00:00
|
|
|
{
|
|
|
|
if (pos_ >= beg && pos_ <= end)
|
|
|
|
sel_beg = pos_;
|
|
|
|
else if (beg > pos_ && beg <= end_)
|
|
|
|
sel_beg = beg;
|
|
|
|
else
|
|
|
|
sel_beg = -1;
|
|
|
|
|
|
|
|
if (end_ >= beg && end_ <= end)
|
|
|
|
sel_end = end_;
|
|
|
|
else if (end < end_ && end >= pos_)
|
|
|
|
sel_end = end;
|
|
|
|
else
|
|
|
|
sel_end = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
bool Row::selection() const
|
|
|
|
{
|
|
|
|
return sel_beg != -1 && sel_end != -1;
|
|
|
|
}
|
|
|
|
|
2013-07-17 22:25:08 +00:00
|
|
|
|
|
|
|
ostream & operator<<(ostream & os, Row::Element const & e)
|
|
|
|
{
|
|
|
|
if (e.font.isVisibleRightToLeft())
|
|
|
|
os << e.endpos << "<<" << e.pos << " ";
|
|
|
|
else
|
|
|
|
os << e.pos << ">>" << e.endpos << " ";
|
|
|
|
|
|
|
|
switch (e.type) {
|
2013-07-21 18:22:32 +00:00
|
|
|
case Row::STRING:
|
2014-03-21 10:56:42 +00:00
|
|
|
os << "STRING: `" << to_utf8(e.str) << "', ";
|
2013-07-17 22:25:08 +00:00
|
|
|
break;
|
2013-07-21 18:22:32 +00:00
|
|
|
case Row::VIRTUAL:
|
2014-03-21 10:56:42 +00:00
|
|
|
os << "VIRTUAL: `" << to_utf8(e.str) << "', ";
|
2013-07-17 22:25:08 +00:00
|
|
|
break;
|
2013-07-21 18:22:32 +00:00
|
|
|
case Row::INSET:
|
2014-03-21 10:56:42 +00:00
|
|
|
os << "INSET: " << to_utf8(e.inset->layoutName()) << ", ";
|
2013-07-17 22:25:08 +00:00
|
|
|
break;
|
2013-07-21 18:22:32 +00:00
|
|
|
case Row::SEPARATOR:
|
2014-03-21 10:56:42 +00:00
|
|
|
os << "SEPARATOR: extra=" << e.extra << ", ";
|
2013-07-17 22:25:08 +00:00
|
|
|
break;
|
2013-07-21 18:22:32 +00:00
|
|
|
case Row::SPACE:
|
2014-03-21 10:56:42 +00:00
|
|
|
os << "SPACE: ";
|
2013-07-17 22:25:08 +00:00
|
|
|
break;
|
|
|
|
}
|
2014-03-21 10:56:42 +00:00
|
|
|
os << "width=" << e.width();
|
2013-07-17 22:25:08 +00:00
|
|
|
return os;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-25 12:57:09 +00:00
|
|
|
ostream & operator<<(ostream & os, Row const & row)
|
|
|
|
{
|
|
|
|
os << " pos: " << row.pos_ << " end: " << row.end_
|
2013-07-21 18:22:32 +00:00
|
|
|
<< " x: " << row.x
|
2013-06-25 12:57:09 +00:00
|
|
|
<< " width: " << row.dim_.wid
|
2014-03-19 13:44:53 +00:00
|
|
|
<< " right_margin: " << row.right_margin
|
2013-06-25 12:57:09 +00:00
|
|
|
<< " ascent: " << row.dim_.asc
|
2013-07-17 22:25:08 +00:00
|
|
|
<< " descent: " << row.dim_.des
|
|
|
|
<< " separator: " << row.separator
|
2014-07-28 21:31:32 +00:00
|
|
|
<< " label_hfill: " << row.label_hfill
|
|
|
|
<< " row_boundary: " << row.right_boundary() << "\n";
|
2013-10-11 14:12:20 +00:00
|
|
|
double x = row.x;
|
2013-06-25 12:57:09 +00:00
|
|
|
Row::Elements::const_iterator it = row.elements_.begin();
|
|
|
|
for ( ; it != row.elements_.end() ; ++it) {
|
2013-10-11 14:12:20 +00:00
|
|
|
os << "x=" << x << " => " << *it << endl;
|
|
|
|
x += it->width();
|
2013-06-25 12:57:09 +00:00
|
|
|
}
|
|
|
|
return os;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Row::sameString(Font const & f, Change const & ch) const
|
|
|
|
{
|
|
|
|
if (elements_.empty())
|
|
|
|
return false;
|
|
|
|
Element const & elt = elements_.back();
|
2013-07-21 18:22:32 +00:00
|
|
|
return elt.type == STRING && !elt.final
|
2013-06-25 12:57:09 +00:00
|
|
|
&& elt.font == f && elt.change == ch;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Row::finalizeLast()
|
|
|
|
{
|
|
|
|
if (elements_.empty())
|
|
|
|
return;
|
|
|
|
Element & elt = elements_.back();
|
|
|
|
if (elt.final)
|
|
|
|
return;
|
|
|
|
elt.final = true;
|
|
|
|
|
2013-07-21 18:22:32 +00:00
|
|
|
if (elt.type == STRING) {
|
2013-06-25 12:57:09 +00:00
|
|
|
elt.dim.wid = theFontMetrics(elt.font).width(elt.str);
|
|
|
|
dim_.wid += elt.dim.wid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-16 22:59:34 +00:00
|
|
|
void Row::add(pos_type const pos, Inset const * ins, Dimension const & dim,
|
|
|
|
Font const & f, Change const & ch)
|
2013-06-25 12:57:09 +00:00
|
|
|
{
|
|
|
|
finalizeLast();
|
2013-07-21 18:22:32 +00:00
|
|
|
Element e(INSET, pos, f, ch);
|
2013-06-25 12:57:09 +00:00
|
|
|
e.inset = ins;
|
|
|
|
e.dim = dim;
|
|
|
|
elements_.push_back(e);
|
|
|
|
dim_.wid += dim.wid;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-16 22:59:34 +00:00
|
|
|
void Row::add(pos_type const pos, char_type const c,
|
2013-06-25 12:57:09 +00:00
|
|
|
Font const & f, Change const & ch)
|
|
|
|
{
|
2013-07-16 22:59:34 +00:00
|
|
|
if (!sameString(f, ch)) {
|
2013-06-25 12:57:09 +00:00
|
|
|
finalizeLast();
|
2013-07-21 18:22:32 +00:00
|
|
|
Element e(STRING, pos, f, ch);
|
2013-06-25 12:57:09 +00:00
|
|
|
elements_.push_back(e);
|
|
|
|
}
|
2013-07-16 22:59:34 +00:00
|
|
|
back().str += c;
|
|
|
|
back().endpos = pos + 1;
|
2013-06-25 12:57:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-21 18:22:32 +00:00
|
|
|
void Row::addVirtual(pos_type const pos, docstring const & s,
|
|
|
|
Font const & f, Change const & ch)
|
2013-06-25 12:57:09 +00:00
|
|
|
{
|
2013-07-17 22:25:08 +00:00
|
|
|
finalizeLast();
|
2013-07-21 18:22:32 +00:00
|
|
|
Element e(VIRTUAL, pos, f, ch);
|
2013-07-17 22:25:08 +00:00
|
|
|
e.str = s;
|
2014-03-20 10:00:14 +00:00
|
|
|
e.dim.wid = theFontMetrics(f).width(s);
|
|
|
|
dim_.wid += e.dim.wid;
|
2013-07-17 22:25:08 +00:00
|
|
|
e.endpos = pos;
|
|
|
|
elements_.push_back(e);
|
|
|
|
finalizeLast();
|
2013-06-25 12:57:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Row::addSeparator(pos_type const pos, char_type const c,
|
|
|
|
Font const & f, Change const & ch)
|
|
|
|
{
|
|
|
|
finalizeLast();
|
2013-07-21 18:22:32 +00:00
|
|
|
Element e(SEPARATOR, pos, f, ch);
|
2013-06-25 12:57:09 +00:00
|
|
|
e.str += c;
|
|
|
|
e.dim.wid = theFontMetrics(f).width(c);
|
|
|
|
elements_.push_back(e);
|
|
|
|
dim_.wid += e.dim.wid;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-16 22:59:34 +00:00
|
|
|
void Row::addSpace(pos_type const pos, int const width,
|
|
|
|
Font const & f, Change const & ch)
|
2013-06-25 12:57:09 +00:00
|
|
|
{
|
|
|
|
finalizeLast();
|
2013-07-21 18:22:32 +00:00
|
|
|
Element e(SPACE, pos, f, ch);
|
2013-06-25 12:57:09 +00:00
|
|
|
e.dim.wid = width;
|
|
|
|
elements_.push_back(e);
|
|
|
|
dim_.wid += e.dim.wid;
|
|
|
|
}
|
|
|
|
|
Patch by Vincent that solves a number of problems related to the painting of a selection:
1. When a listing is inserted in a bit of text, the line above the listing is not drawn over the full width like it is done for lines above other insets. This is because InsetListing has a AlignLeft alignment. Now, if you start selecting downwards with the mouse in this empty area, strange selection drawings appear (see attachment).
This is caused by the fact that starting your selection at such a place, causes beg.boundary() to be true in TextMetrics::drawRowSelection(..). This is correct, but this value is true for _all_ selected lines. Now, the selection acts as if it is RTL text. Therefore, just like for end.boundary, this value needs to be reset for every line.
2. Starting your selection in an end margin often causes the selection in this end margin to be painted later. This is because when starting your selection in an end margin, you may have set a (possible empty) selection before really selecting the end margin. The problem is that the checksum (computed later) is the same for this empty selection and for the end margin selection. Therfore, we need a call to cur.setSelection() before evaluating cur.selection().
3. In the following two lines, it is assumed that there is only an end margin to be painted if the selection extends to the next paragraph. This is not true for the above described case of an AlignLeft Inset. Then, the margin has also be drawn within a paragraph
4. The end and begin margins are only painted when the selection extends into the following or previous paragraph. This difference is not resembled in the checksum if you first select a row completely and then procede to the next or previous paragraph as the selection remains at the end of a row. This also holds for the AlignLeft case. Therefore I added a term to the checksum to monitor whether the end and begin margins need to be drawn.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26399 a592a061-630c-0410-9148-cb99ea01b6c8
2008-09-14 14:32:40 +00:00
|
|
|
|
2013-06-25 12:57:09 +00:00
|
|
|
void Row::pop_back()
|
2003-08-01 14:12:04 +00:00
|
|
|
{
|
2013-06-25 12:57:09 +00:00
|
|
|
dim_.wid -= elements_.back().dim.wid;
|
|
|
|
elements_.pop_back();
|
2003-08-01 14:12:04 +00:00
|
|
|
}
|
2006-10-21 00:16:43 +00:00
|
|
|
|
|
|
|
|
2014-05-19 09:35:15 +00:00
|
|
|
void Row::shortenIfNeeded(pos_type const keep, int const w)
|
2013-06-25 12:57:09 +00:00
|
|
|
{
|
2014-05-19 09:35:15 +00:00
|
|
|
if (empty() || width() <= w)
|
2013-06-25 12:57:09 +00:00
|
|
|
return;
|
2013-07-23 14:24:01 +00:00
|
|
|
|
|
|
|
/** First, we try to remove elements one by one from the end
|
2014-05-19 09:35:15 +00:00
|
|
|
* until a separator is found. cit points to the first element
|
|
|
|
* we want to remove from the row.
|
2013-07-23 14:24:01 +00:00
|
|
|
*/
|
2014-05-19 09:35:15 +00:00
|
|
|
Elements::iterator const beg = elements_.begin();
|
|
|
|
Elements::iterator const end = elements_.end();
|
|
|
|
Elements::iterator cit = end;
|
|
|
|
Elements::iterator first_below = end;
|
2013-06-25 12:57:09 +00:00
|
|
|
int new_end = end_;
|
|
|
|
int new_wid = dim_.wid;
|
2014-05-19 09:35:15 +00:00
|
|
|
// if the row ends with a separator, skip it.
|
|
|
|
if (cit != beg && boost::prior(cit)->type == SEPARATOR && new_end > keep) {
|
|
|
|
--cit;
|
|
|
|
new_end = cit->pos;
|
|
|
|
new_wid -= cit->dim.wid;
|
2013-06-25 12:57:09 +00:00
|
|
|
}
|
|
|
|
|
2014-05-19 09:35:15 +00:00
|
|
|
// Search for a separator where the row can be broken.
|
|
|
|
while (cit != beg && boost::prior(cit)->type != SEPARATOR && new_end > keep) {
|
|
|
|
--cit;
|
|
|
|
new_end = cit->pos;
|
|
|
|
new_wid -= cit->dim.wid;
|
|
|
|
if (new_wid < w && first_below == end)
|
|
|
|
first_below = cit;
|
2013-06-25 12:57:09 +00:00
|
|
|
}
|
2014-05-19 09:35:15 +00:00
|
|
|
|
|
|
|
if (cit != beg) {
|
|
|
|
// We have found a suitable separator. This is the
|
|
|
|
// common case.
|
|
|
|
end_ = new_end;
|
|
|
|
dim_.wid = new_wid;
|
|
|
|
elements_.erase(cit, end);
|
2013-07-23 14:24:01 +00:00
|
|
|
return;
|
2013-07-21 18:22:32 +00:00
|
|
|
}
|
2014-05-19 09:35:15 +00:00
|
|
|
|
|
|
|
/* If we are here, it means that we have not found a separator
|
|
|
|
* to shorten the row. There is one case where we can do
|
|
|
|
* something: when we have one big string, maybe with some
|
|
|
|
* other things after it.
|
|
|
|
*/
|
|
|
|
double max_w = w - x;
|
|
|
|
if (first_below->breakAt(max_w)) {
|
|
|
|
end_ = first_below->endpos;
|
2014-07-28 07:46:13 +00:00
|
|
|
dim_.wid = int(x + first_below->width());
|
2014-05-19 09:35:15 +00:00
|
|
|
// If there are other elements, they should be removed.
|
|
|
|
elements_.erase(boost::next(first_below), end);
|
|
|
|
} else if (first_below->pos > pos_) {
|
|
|
|
end_ = first_below->pos;
|
|
|
|
dim_.wid = new_wid;
|
|
|
|
// Remove all elements from first_below.
|
|
|
|
elements_.erase(first_below, end);
|
|
|
|
}
|
2013-06-25 12:57:09 +00:00
|
|
|
}
|
|
|
|
|
2013-07-16 22:59:34 +00:00
|
|
|
|
2014-03-19 13:44:31 +00:00
|
|
|
void Row::reverseRTL(bool const rtl_par)
|
2013-07-16 22:59:34 +00:00
|
|
|
{
|
|
|
|
pos_type i = 0;
|
|
|
|
pos_type const end = elements_.size();
|
|
|
|
while (i < end) {
|
2014-03-19 13:44:31 +00:00
|
|
|
// gather a sequence of elements with the same direction
|
|
|
|
bool const rtl = elements_[i].font.isVisibleRightToLeft();
|
2013-07-16 22:59:34 +00:00
|
|
|
pos_type j = i;
|
2014-03-19 13:44:31 +00:00
|
|
|
while (j < end && elements_[j].font.isVisibleRightToLeft() == rtl)
|
2013-07-16 22:59:34 +00:00
|
|
|
++j;
|
2014-03-19 13:44:31 +00:00
|
|
|
// if the direction is not the same as the paragraph
|
|
|
|
// direction, the sequence has to be reverted.
|
|
|
|
if (rtl != rtl_par)
|
|
|
|
reverse(elements_.begin() + i, elements_.begin() + j);
|
2013-07-16 22:59:34 +00:00
|
|
|
i = j;
|
|
|
|
}
|
2014-03-19 13:44:31 +00:00
|
|
|
// If the paragraph itself is RTL, reverse everything
|
|
|
|
if (rtl_par)
|
|
|
|
reverse(elements_.begin(), elements_.end());
|
2013-07-16 22:59:34 +00:00
|
|
|
}
|
|
|
|
|
2006-10-21 00:16:43 +00:00
|
|
|
} // namespace lyx
|