]> git.lyx.org Git - lyx.git/blobdiff - src/AppleSpellChecker.cpp
Avoid row breaking at inconvenient places.
[lyx.git] / src / AppleSpellChecker.cpp
index c5ba8088d02cf2d9029c2e51fa79683cadf5fa65..0a82d38b3228f04ee27ad3e45bbd889e6947690d 100644 (file)
@@ -13,7 +13,6 @@
 #include "AppleSpellChecker.h"
 #include "WordLangTuple.h"
 
-#include "support/lassert.h"
 #include "support/debug.h"
 #include "support/docstring_list.h"
 #include "support/AppleSpeller.h"
@@ -29,8 +28,15 @@ struct AppleSpellChecker::Private
 
        ~Private();
 
+       SpellChecker::Result toResult(SpellCheckResult status);
+       string toString(SpellCheckResult status);
+       int numDictionaries() const;
+
        /// the speller
        AppleSpeller speller;
+
+       /// language map
+       map<string, string> languageMap;
 };
 
 
@@ -47,9 +53,9 @@ AppleSpellChecker::Private::~Private()
 }
 
 
-AppleSpellChecker::AppleSpellChecker(): d(new Private)
-{
-}
+AppleSpellChecker::AppleSpellChecker()
+       : d(new Private)
+{}
 
 
 AppleSpellChecker::~AppleSpellChecker()
@@ -58,11 +64,50 @@ AppleSpellChecker::~AppleSpellChecker()
 }
 
 
-SpellChecker::Result AppleSpellChecker::check(WordLangTuple const & word)
+SpellChecker::Result AppleSpellChecker::Private::toResult(SpellCheckResult status)
+{
+       return status == SPELL_CHECK_FAILED ? UNKNOWN_WORD :
+               status == SPELL_CHECK_LEARNED ? LEARNED_WORD : WORD_OK ;
+}
+
+
+string AppleSpellChecker::Private::toString(SpellCheckResult status)
+{
+       return status == SPELL_CHECK_FAILED ? "FAILED" :
+                status == SPELL_CHECK_LEARNED ? "LEARNED" : "OK";
+}
+
+
+SpellChecker::Result AppleSpellChecker::check(WordLangTuple const & word,
+        std::vector<WordLangTuple> const & docdict)
 {
+       if (!hasDictionary(word.lang()))
+               return NO_DICTIONARY;
+
        string const word_str = to_utf8(word.word());
-       int const word_ok = checkAppleSpeller(d->speller, word_str.c_str(), word.lang()->code().c_str());
-       return (word_ok) ? OK : UNKNOWN_WORD;
+       string const lang = d->languageMap[word.lang()->lang()];
+
+       vector<WordLangTuple>::const_iterator it = docdict.begin();
+       for (; it != docdict.end(); ++it) {
+               if (it->lang()->code() != word.lang()->code())
+                       continue;
+               if (it->word() == word.word())
+                       return DOCUMENT_LEARNED_WORD;
+       }
+
+       SpellCheckResult result =
+               AppleSpeller_check(d->speller,
+                       word_str.c_str(), lang.c_str());
+       LYXERR(Debug::GUI, "spellCheck: \"" <<
+                  word.word() << "\" = " << d->toString(result) <<
+                  ", lang = " << lang) ;
+       return d->toResult(result);
+}
+
+
+void AppleSpellChecker::advanceChangeNumber()
+{
+       nextChangeNumber();
 }
 
 
@@ -70,7 +115,19 @@ SpellChecker::Result AppleSpellChecker::check(WordLangTuple const & word)
 void AppleSpellChecker::insert(WordLangTuple const & word)
 {
        string const word_str = to_utf8(word.word());
-       learnAppleSpeller(d->speller, word_str.c_str(), word.lang()->code().c_str());
+       AppleSpeller_learn(d->speller, word_str.c_str());
+       LYXERR(Debug::GUI, "learn word: \"" << word.word() << "\"");
+       advanceChangeNumber();
+}
+
+
+// remove from personal dictionary
+void AppleSpellChecker::remove(WordLangTuple const & word)
+{
+       string const word_str = to_utf8(word.word());
+       AppleSpeller_unlearn(d->speller, word_str.c_str());
+       LYXERR(Debug::GUI, "unlearn word: \"" << word.word() << "\"");
+       advanceChangeNumber();
 }
 
 
@@ -78,7 +135,9 @@ void AppleSpellChecker::insert(WordLangTuple const & word)
 void AppleSpellChecker::accept(WordLangTuple const & word)
 {
        string const word_str = to_utf8(word.word());
-       ignoreAppleSpeller(d->speller, word_str.c_str(), word.lang()->code().c_str());
+       AppleSpeller_ignore(d->speller, word_str.c_str());
+       LYXERR(Debug::GUI, "ignore word: \"" << word.word() << "\"");
+       advanceChangeNumber();
 }
 
 
@@ -87,9 +146,10 @@ void AppleSpellChecker::suggest(WordLangTuple const & wl,
 {
        suggestions.clear();
        string const word_str = to_utf8(wl.word());
-       size_t num = makeSuggestionAppleSpeller(d->speller, word_str.c_str(), wl.lang()->code().c_str());
+       size_t num = AppleSpeller_makeSuggestion(d->speller,
+                                       word_str.c_str(), wl.lang()->code().c_str());
        for (size_t i = 0; i < num; i++) {
-               char const * next = getSuggestionAppleSpeller(d->speller, i);
+               char const * next = AppleSpeller_getSuggestion(d->speller, i);
                if (!next) break;
                suggestions.push_back(from_utf8(next));
        }
@@ -98,7 +158,48 @@ void AppleSpellChecker::suggest(WordLangTuple const & wl,
 
 bool AppleSpellChecker::hasDictionary(Language const * lang) const
 {
-       return hasLanguageAppleSpeller(d->speller,lang->code().c_str());
+       string const langmap = d->languageMap[lang->lang()];
+       bool result = !langmap.empty();
+
+       if (result)
+               return result;
+
+       result = AppleSpeller_hasLanguage(d->speller, lang->code().c_str());
+       if (result) {
+               d->languageMap[lang->lang()] = lang->code();
+       } else {
+               result = AppleSpeller_hasLanguage(d->speller, lang->lang().c_str());
+               if (result)
+                       d->languageMap[lang->lang()] = lang->lang();
+       }
+       LYXERR(Debug::GUI, "has dictionary: " << lang->lang() << " = " << result);
+       return result;
+}
+
+
+int AppleSpellChecker::numDictionaries() const
+{
+       int result = 0;
+       map<string, string>::const_iterator it = d->languageMap.begin();
+       map<string, string>::const_iterator et = d->languageMap.end();
+
+       for (; it != et; ++it) {
+               string const langmap = it->second;
+               result += langmap.empty() ? 0 : 1;
+       }
+       return result;
+}
+
+
+int AppleSpellChecker::numMisspelledWords() const
+{
+       return AppleSpeller_numMisspelledWords(d->speller);
+}
+
+
+void AppleSpellChecker::misspelledWord(int index, int & start, int & length) const
+{
+       AppleSpeller_misspelledWord(d->speller, index, &start, &length);
 }