/** * \file AspellChecker.cpp * This file is part of LyX, the document processor. * Licence details can be found in the file COPYING. * * \author Kevin Atkinson * \author John Levon * * Full author contact details are available in file CREDITS. */ #include #include "AspellChecker.h" #include "LyXRC.h" #include "WordLangTuple.h" #include "support/lassert.h" #include "support/debug.h" #include "support/docstring_list.h" #include #include #include using namespace std; namespace lyx { namespace { struct Speller { AspellSpeller * speller; AspellConfig * config; }; typedef std::map Spellers; } // anon namespace struct AspellChecker::Private { Private(): spell_error_object(0) {} ~Private(); /// add a speller of the given language AspellSpeller * addSpeller(string const & lang); /// AspellSpeller * speller(string const & lang); /// the spellers Spellers spellers_; /// FIXME AspellCanHaveError * spell_error_object; }; AspellChecker::Private::~Private() { if (spell_error_object) { delete_aspell_can_have_error(spell_error_object); spell_error_object = 0; } Spellers::iterator it = spellers_.begin(); Spellers::iterator end = spellers_.end(); for (; it != end; ++it) { aspell_speller_save_all_word_lists(it->second.speller); delete_aspell_speller(it->second.speller); delete_aspell_config(it->second.config); } } AspellSpeller * AspellChecker::Private::addSpeller(string const & lang) { AspellConfig * config = new_aspell_config(); // FIXME The aspell documentation says to use "lang" aspell_config_replace(config, "language-tag", lang.c_str()); // Set the encoding to utf-8. // aspell does also understand "ucs-4", so we would not need a // conversion in theory, but if this is used it expects all // char const * arguments to be a cast from uint const *, and it // seems that this uint is not compatible with our char_type on some // platforms (cygwin, OS X). Therefore we use utf-8, that does // always work. aspell_config_replace(config, "encoding", "utf-8"); if (lyxrc.spellchecker_accept_compound) // Consider run-together words as legal compounds aspell_config_replace(config, "run-together", "true"); else // Report run-together words as errors aspell_config_replace(config, "run-together", "false"); AspellCanHaveError * err = new_aspell_speller(config); if (spell_error_object) delete_aspell_can_have_error(spell_error_object); spell_error_object = 0; if (aspell_error_number(err) != 0) { // FIXME: We should we indicate somehow that this language is not // supported. spell_error_object = err; return 0; } Speller m; m.speller = to_aspell_speller(err); m.config = config; spellers_[lang] = m; return m.speller; } AspellSpeller * AspellChecker::Private::speller(string const & lang) { Spellers::iterator it = spellers_.find(lang); if (it != spellers_.end()) return it->second.speller; return addSpeller(lang); } AspellChecker::AspellChecker(): d(new Private) { } AspellChecker::~AspellChecker() { delete d; } SpellChecker::Result AspellChecker::check(WordLangTuple const & word) { AspellSpeller * m = d->speller(word.lang_code()); if (!m) return OK; if (word.word().empty()) // MSVC compiled Aspell doesn't like it. return OK; int const word_ok = aspell_speller_check(m, to_utf8(word.word()).c_str(), -1); LASSERT(word_ok != -1, /**/); return (word_ok) ? OK : UNKNOWN_WORD; } void AspellChecker::insert(WordLangTuple const & word) { Spellers::iterator it = d->spellers_.find(word.lang_code()); if (it != d->spellers_.end()) aspell_speller_add_to_personal(it->second.speller, to_utf8(word.word()).c_str(), -1); } void AspellChecker::accept(WordLangTuple const & word) { Spellers::iterator it = d->spellers_.find(word.lang_code()); if (it != d->spellers_.end()) aspell_speller_add_to_session(it->second.speller, to_utf8(word.word()).c_str(), -1); } void AspellChecker::suggest(WordLangTuple const & wl, docstring_list & suggestions) { suggestions.clear(); AspellSpeller * m = d->speller(wl.lang_code()); if (!m) return; AspellWordList const * sugs = aspell_speller_suggest(m, to_utf8(wl.word()).c_str(), -1); LASSERT(sugs != 0, /**/); AspellStringEnumeration * els = aspell_word_list_elements(sugs); if (!els || aspell_word_list_empty(sugs)) return; for (;;) { char const * str = aspell_string_enumeration_next(els); if (!str) break; suggestions.push_back(from_utf8(str)); } delete_aspell_string_enumeration(els); } docstring const AspellChecker::error() { char const * err = 0; if (d->spell_error_object && aspell_error_number(d->spell_error_object) != 0) err = aspell_error_message(d->spell_error_object); // FIXME UNICODE: err is not in UTF8, but probably the locale encoding return (err ? from_utf8(err) : docstring()); } } // namespace lyx