mirror of
https://git.lyx.org/repos/lyx.git
synced 2024-11-22 10:00:33 +00:00
#6917 correct spell check for deleted text in change tracking mode
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@35542 a592a061-630c-0410-9148-cb99ea01b6c8
This commit is contained in:
parent
029ac601a8
commit
cc7db294e2
@ -333,14 +333,16 @@ public:
|
|||||||
/// match a string against a particular point in the paragraph
|
/// match a string against a particular point in the paragraph
|
||||||
bool isTextAt(string const & str, pos_type pos) const;
|
bool isTextAt(string const & str, pos_type pos) const;
|
||||||
|
|
||||||
/// a vector of inset positions
|
/// a vector of speller skip positions
|
||||||
typedef vector<pos_type> Positions;
|
typedef vector<FontSpan> SkipPositions;
|
||||||
typedef Positions::const_iterator PositionsIterator;
|
typedef SkipPositions::const_iterator SkipPositionsIterator;
|
||||||
|
|
||||||
|
void appendSkipPosition(SkipPositions & skips, pos_type const pos) const;
|
||||||
|
|
||||||
Language * getSpellLanguage(pos_type const from) const;
|
Language * getSpellLanguage(pos_type const from) const;
|
||||||
|
|
||||||
Language * locateSpellRange(pos_type & from, pos_type & to,
|
Language * locateSpellRange(pos_type & from, pos_type & to,
|
||||||
Positions & softbreaks) const;
|
SkipPositions & skips) const;
|
||||||
|
|
||||||
bool hasSpellerChange() const {
|
bool hasSpellerChange() const {
|
||||||
SpellChecker::ChangeNumber speller_change_number = 0;
|
SpellChecker::ChangeNumber speller_change_number = 0;
|
||||||
@ -391,22 +393,23 @@ public:
|
|||||||
last = endpos;
|
last = endpos;
|
||||||
}
|
}
|
||||||
|
|
||||||
int countSoftbreaks(PositionsIterator & it, PositionsIterator const et,
|
int countSkips(SkipPositionsIterator & it, SkipPositionsIterator const et,
|
||||||
int & start) const
|
int & start) const
|
||||||
{
|
{
|
||||||
int numbreaks = 0;
|
int numskips = 0;
|
||||||
while (it != et && *it < start) {
|
while (it != et && it->first < start) {
|
||||||
++start;
|
int skip = it->last - it->first + 1;
|
||||||
++numbreaks;
|
start += skip;
|
||||||
|
numskips += skip;
|
||||||
++it;
|
++it;
|
||||||
}
|
}
|
||||||
return numbreaks;
|
return numskips;
|
||||||
}
|
}
|
||||||
|
|
||||||
void markMisspelledWords(pos_type const & first, pos_type const & last,
|
void markMisspelledWords(pos_type const & first, pos_type const & last,
|
||||||
SpellChecker::Result result,
|
SpellChecker::Result result,
|
||||||
docstring const & word,
|
docstring const & word,
|
||||||
Positions const & softbreaks);
|
SkipPositions const & skips);
|
||||||
|
|
||||||
InsetCode ownerCode() const
|
InsetCode ownerCode() const
|
||||||
{
|
{
|
||||||
@ -2891,6 +2894,8 @@ docstring Paragraph::asString(pos_type beg, pos_type end, int options) const
|
|||||||
os << d->params_.labelString() << ' ';
|
os << d->params_.labelString() << ' ';
|
||||||
|
|
||||||
for (pos_type i = beg; i < end; ++i) {
|
for (pos_type i = beg; i < end; ++i) {
|
||||||
|
if ((options & AS_STR_SKIPDELETE) && isDeleted(i))
|
||||||
|
continue;
|
||||||
char_type const c = d->text_[i];
|
char_type const c = d->text_[i];
|
||||||
if (isPrintable(c) || c == '\t'
|
if (isPrintable(c) || c == '\t'
|
||||||
|| (c == '\n' && (options & AS_STR_NEWLINES)))
|
|| (c == '\n' && (options & AS_STR_NEWLINES)))
|
||||||
@ -3410,9 +3415,24 @@ void Paragraph::updateWords()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Paragraph::Private::appendSkipPosition(SkipPositions & skips, pos_type const pos) const
|
||||||
|
{
|
||||||
|
SkipPositionsIterator begin = skips.begin();
|
||||||
|
SkipPositions::iterator end = skips.end();
|
||||||
|
if (pos > 0 && begin < end) {
|
||||||
|
--end;
|
||||||
|
if (end->last == pos - 1) {
|
||||||
|
end->last = pos;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
skips.insert(end, FontSpan(pos, pos));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
Language * Paragraph::Private::locateSpellRange(
|
Language * Paragraph::Private::locateSpellRange(
|
||||||
pos_type & from, pos_type & to,
|
pos_type & from, pos_type & to,
|
||||||
Positions & softbreaks) const
|
SkipPositions & skips) const
|
||||||
{
|
{
|
||||||
// skip leading white space
|
// skip leading white space
|
||||||
while (from < to && owner_->isWordSeparator(from))
|
while (from < to && owner_->isWordSeparator(from))
|
||||||
@ -3429,7 +3449,9 @@ Language * Paragraph::Private::locateSpellRange(
|
|||||||
// hop to end of word
|
// hop to end of word
|
||||||
while (last < to && !owner_->isWordSeparator(last)) {
|
while (last < to && !owner_->isWordSeparator(last)) {
|
||||||
if (owner_->getInset(last)) {
|
if (owner_->getInset(last)) {
|
||||||
softbreaks.insert(softbreaks.end(), last);
|
appendSkipPosition(skips, last);
|
||||||
|
} else if (owner_->isDeleted(last)) {
|
||||||
|
appendSkipPosition(skips, last);
|
||||||
}
|
}
|
||||||
++last;
|
++last;
|
||||||
}
|
}
|
||||||
@ -3437,6 +3459,9 @@ Language * Paragraph::Private::locateSpellRange(
|
|||||||
while (sameinset && last < to && owner_->isWordSeparator(last)) {
|
while (sameinset && last < to && owner_->isWordSeparator(last)) {
|
||||||
if (Inset const * inset = owner_->getInset(last))
|
if (Inset const * inset = owner_->getInset(last))
|
||||||
sameinset = inset->isChar() && inset->isLetter();
|
sameinset = inset->isChar() && inset->isLetter();
|
||||||
|
if (sameinset && owner_->isDeleted(last)) {
|
||||||
|
appendSkipPosition(skips, last);
|
||||||
|
}
|
||||||
if (sameinset)
|
if (sameinset)
|
||||||
last++;
|
last++;
|
||||||
}
|
}
|
||||||
@ -3504,7 +3529,7 @@ SpellChecker::Result Paragraph::spellCheck(pos_type & from, pos_type & to,
|
|||||||
if (from == to || from >= size())
|
if (from == to || from >= size())
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
docstring word = asString(from, to, AS_STR_INSETS);
|
docstring word = asString(from, to, AS_STR_INSETS + AS_STR_SKIPDELETE);
|
||||||
Language * lang = d->getSpellLanguage(from);
|
Language * lang = d->getSpellLanguage(from);
|
||||||
|
|
||||||
wl = WordLangTuple(word, lang);
|
wl = WordLangTuple(word, lang);
|
||||||
@ -3552,7 +3577,7 @@ void Paragraph::Private::markMisspelledWords(
|
|||||||
pos_type const & first, pos_type const & last,
|
pos_type const & first, pos_type const & last,
|
||||||
SpellChecker::Result result,
|
SpellChecker::Result result,
|
||||||
docstring const & word,
|
docstring const & word,
|
||||||
Positions const & softbreaks)
|
SkipPositions const & skips)
|
||||||
{
|
{
|
||||||
if (!SpellChecker::misspelled(result)) {
|
if (!SpellChecker::misspelled(result)) {
|
||||||
setMisspelled(first, last, SpellChecker::WORD_OK);
|
setMisspelled(first, last, SpellChecker::WORD_OK);
|
||||||
@ -3562,9 +3587,9 @@ void Paragraph::Private::markMisspelledWords(
|
|||||||
SpellChecker * speller = theSpellChecker();
|
SpellChecker * speller = theSpellChecker();
|
||||||
// locate and enumerate the error positions
|
// locate and enumerate the error positions
|
||||||
int nerrors = speller->numMisspelledWords();
|
int nerrors = speller->numMisspelledWords();
|
||||||
int numbreaks = 0;
|
int numskipped = 0;
|
||||||
PositionsIterator it = softbreaks.begin();
|
SkipPositionsIterator it = skips.begin();
|
||||||
PositionsIterator et = softbreaks.end();
|
SkipPositionsIterator et = skips.end();
|
||||||
for (int index = 0; index < nerrors; ++index) {
|
for (int index = 0; index < nerrors; ++index) {
|
||||||
int wstart;
|
int wstart;
|
||||||
int wlen = 0;
|
int wlen = 0;
|
||||||
@ -3572,15 +3597,15 @@ void Paragraph::Private::markMisspelledWords(
|
|||||||
/// should not happen if speller supports range checks
|
/// should not happen if speller supports range checks
|
||||||
if (!wlen) continue;
|
if (!wlen) continue;
|
||||||
docstring const misspelled = word.substr(wstart, wlen);
|
docstring const misspelled = word.substr(wstart, wlen);
|
||||||
wstart += first + numbreaks;
|
wstart += first + numskipped;
|
||||||
if (snext < wstart) {
|
if (snext < wstart) {
|
||||||
/// mark the range of correct spelling
|
/// mark the range of correct spelling
|
||||||
numbreaks += countSoftbreaks(it, et, wstart);
|
numskipped += countSkips(it, et, wstart);
|
||||||
setMisspelled(snext,
|
setMisspelled(snext,
|
||||||
wstart - 1, SpellChecker::WORD_OK);
|
wstart - 1, SpellChecker::WORD_OK);
|
||||||
}
|
}
|
||||||
snext = wstart + wlen;
|
snext = wstart + wlen;
|
||||||
numbreaks += countSoftbreaks(it, et, snext);
|
numskipped += countSkips(it, et, snext);
|
||||||
/// mark the range of misspelling
|
/// mark the range of misspelling
|
||||||
setMisspelled(wstart, snext, result);
|
setMisspelled(wstart, snext, result);
|
||||||
LYXERR(Debug::GUI, "misspelled word: \"" <<
|
LYXERR(Debug::GUI, "misspelled word: \"" <<
|
||||||
@ -3607,16 +3632,16 @@ void Paragraph::spellCheck() const
|
|||||||
// loop until we leave the range
|
// loop until we leave the range
|
||||||
for (pos_type first = start; first < endpos; ) {
|
for (pos_type first = start; first < endpos; ) {
|
||||||
pos_type last = endpos;
|
pos_type last = endpos;
|
||||||
Private::Positions softbreaks;
|
Private::SkipPositions skips;
|
||||||
Language * lang = d->locateSpellRange(first, last, softbreaks);
|
Language * lang = d->locateSpellRange(first, last, skips);
|
||||||
if (first >= endpos)
|
if (first >= endpos)
|
||||||
break;
|
break;
|
||||||
// start the spell checker on the unit of meaning
|
// start the spell checker on the unit of meaning
|
||||||
docstring word = asString(first, last, AS_STR_INSETS);
|
docstring word = asString(first, last, AS_STR_INSETS + AS_STR_SKIPDELETE);
|
||||||
WordLangTuple wl = WordLangTuple(word, lang);
|
WordLangTuple wl = WordLangTuple(word, lang);
|
||||||
SpellChecker::Result result = word.size() ?
|
SpellChecker::Result result = word.size() ?
|
||||||
speller->check(wl) : SpellChecker::WORD_OK;
|
speller->check(wl) : SpellChecker::WORD_OK;
|
||||||
d->markMisspelledWords(first, last, result, word, softbreaks);
|
d->markMisspelledWords(first, last, result, word, skips);
|
||||||
first = ++last;
|
first = ++last;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -101,7 +101,8 @@ enum AsStringParameter
|
|||||||
AS_STR_NONE = 0, ///< No option, only printable characters.
|
AS_STR_NONE = 0, ///< No option, only printable characters.
|
||||||
AS_STR_LABEL = 1, ///< Prefix with paragraph label.
|
AS_STR_LABEL = 1, ///< Prefix with paragraph label.
|
||||||
AS_STR_INSETS = 2, ///< Go into insets.
|
AS_STR_INSETS = 2, ///< Go into insets.
|
||||||
AS_STR_NEWLINES = 4 ///< Get also newline characters.
|
AS_STR_NEWLINES = 4, ///< Get also newline characters.
|
||||||
|
AS_STR_SKIPDELETE = 8 ///< Skip deleted text in change tracking.
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user