]> git.lyx.org Git - lyx.git/blobdiff - src/HunspellChecker.cpp
Update notes.
[lyx.git] / src / HunspellChecker.cpp
index 2310f77458771202945ec60df97edf4ded79a8fc..30d799a7f2243ad9d6a26b5cd01f358202d87d6d 100644 (file)
 #include "LyXRC.h"
 #include "WordLangTuple.h"
 
-#include "support/lassert.h"
+#include "frontends/alert.h"
+
 #include "support/debug.h"
+#include "support/docstring_list.h"
+#include "support/filetools.h"
+#include "support/FileName.h"
+#include "support/gettext.h"
+#include "support/lassert.h"
+#include "support/lstrings.h"
+#include "support/os.h"
 
 #include <hunspell/hunspell.hxx>
 
 #include <string>
 
 using namespace std;
+using namespace lyx::support;
+using namespace lyx::support::os;
 
 namespace lyx {
 
 namespace {
+
 typedef map<std::string, Hunspell *> Spellers;
-}
 
-class HunspellChecker::Private
+} // anon namespace
+
+struct HunspellChecker::Private
 {
+       Private() {}
+
+       ~Private();
+
+       Hunspell * addSpeller(string const & lang);
+       Hunspell * speller(string const & lang);
+
        /// the spellers
        Spellers spellers_;
 };
 
 
+HunspellChecker::Private::~Private()
+{
+       Spellers::iterator it = spellers_.begin();
+       Spellers::iterator end = spellers_.end();
+
+       for (; it != end; ++it) {
+               delete it->second;
+       }
+}
+
+
+namespace {
+bool haveLanguageFiles(string const & hpath)
+{
+       FileName const affix(hpath + ".aff");
+       FileName const dict(hpath + ".dic");
+       if (!affix.isReadableFile()) {
+               // FIXME: We should indicate somehow that this language is not
+               // supported.
+               LYXERR(Debug::FILES, "Hunspell affix file " << affix << " does not exist");
+               return false;
+       }
+       if (!dict.isReadableFile()) {
+               LYXERR(Debug::FILES, "Hunspell dictionary file " << dict << " does not exist");
+               return false;
+       }
+       return true;
+}
+}
+
+
+Hunspell * HunspellChecker::Private::addSpeller(string const & lang)
+{
+       string hunspell_path = lyxrc.hunspelldir_path;
+       LYXERR(Debug::FILES, "hunspell path: " << external_path(hunspell_path));
+       if (hunspell_path.empty()) {
+               // FIXME We'd like to issue a better error message here, but there seems
+               // to be a problem about thread safety, or something of the sort. If
+               // we issue the message using frontend::Alert, then the code comes
+               // back through here while the box is waiting, and causes some kind
+               // of crash. 
+               static bool warned = false;
+               if (!warned) {
+                       warned = true;
+                       LYXERR0("Hunspell path not set.");
+                       //frontend::Alert::error(_("Hunspell Path Not Found"), 
+                       //              _("You must set the Hunspell dictionary path in Tools>Preferences>Paths."));
+               }
+               return 0;
+       }
+
+       hunspell_path = external_path(addName(hunspell_path, lang));
+       if (!haveLanguageFiles(hunspell_path)) {
+               // try with '_' replaced by '-'
+               hunspell_path = subst(hunspell_path, '_', '-');
+               if (!haveLanguageFiles(hunspell_path)) {
+                       // FIXME: We should indicate somehow that this language is not
+                       // supported, probably by popping a warning. But we'll need to
+                       // remember which warnings we've issued.
+                       return 0;
+               }
+       }
+       FileName const affix(hunspell_path + ".aff");
+       FileName const dict(hunspell_path + ".dic");
+       Hunspell * h = new Hunspell(affix.absFilename().c_str(), dict.absFilename().c_str());
+       spellers_[lang] = h;
+       return h;
+}
+
+
+Hunspell * HunspellChecker::Private::speller(string const & lang)
+{
+       Spellers::iterator it = spellers_.find(lang);
+       if (it != spellers_.end())
+               return it->second;
+       
+       return addSpeller(lang);
+}
+
+
 HunspellChecker::HunspellChecker(): d(new Private)
 {
 }
@@ -49,25 +148,60 @@ HunspellChecker::~HunspellChecker()
 }
 
 
-SpellChecker::Result HunspellChecker::check(WordLangTuple const & word)
+SpellChecker::Result HunspellChecker::check(WordLangTuple const & wl)
 {
-       return OK;
+       string const word_to_check = to_utf8(wl.word());
+       Hunspell * h = d->speller(wl.lang_code());
+       if (!h)
+               return OK;
+       int info;
+       if (h->spell(word_to_check.c_str(), &info))
+               return OK;
+
+       if (info & SPELL_COMPOUND) {
+               // FIXME: What to do with that?
+               LYXERR(Debug::FILES, "Hunspell compound word found " << word_to_check);
+       }
+       if (info & SPELL_FORBIDDEN) {
+               // FIXME: What to do with that?
+               LYXERR(Debug::FILES, "Hunspell explicit forbidden word found " << word_to_check);
+       }
+
+       return UNKNOWN_WORD;
 }
 
 
-void HunspellChecker::insert(WordLangTuple const & word)
+void HunspellChecker::insert(WordLangTuple const & wl)
 {
+       string const word_to_check = to_utf8(wl.word());
+       Hunspell * h = d->speller(wl.lang_code());
+       if (!h)
+               return;
+       h->add(word_to_check.c_str());
 }
 
 
-void HunspellChecker::accept(WordLangTuple const & word)
+void HunspellChecker::accept(WordLangTuple const &)
 {
+       // FIXME: not implemented!
 }
 
 
-docstring const HunspellChecker::nextMiss()
+void HunspellChecker::suggest(WordLangTuple const & wl,
+       docstring_list & suggestions)
 {
-       return docstring();
+       suggestions.clear();
+       string const word_to_check = to_utf8(wl.word());
+       Hunspell * h = d->speller(wl.lang_code());
+       if (!h)
+               return;
+       char ** suggestion_list;
+       int const suggestion_number = h->suggest(&suggestion_list, word_to_check.c_str());
+       if (suggestion_number <= 0)
+               return;
+       for (int i = 0; i != suggestion_number; ++i)
+               suggestions.push_back(from_utf8(suggestion_list[i]));
+       h->free_list(&suggestion_list, suggestion_number);
 }