#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:
Stephan Witt 2010-09-29 19:31:16 +00:00
parent 029ac601a8
commit cc7db294e2
2 changed files with 52 additions and 26 deletions

View File

@ -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 {

View File

@ -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.
}; };