2010-07-07 17:40:50 +00:00
|
|
|
/**
|
|
|
|
* \file AppleSpellChecker.cpp
|
|
|
|
* This file is part of LyX, the document processor.
|
|
|
|
* Licence details can be found in the file COPYING.
|
|
|
|
*
|
|
|
|
* \author Stephan Witt
|
|
|
|
*
|
|
|
|
* Full author contact details are available in file CREDITS.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "AppleSpellChecker.h"
|
|
|
|
#include "WordLangTuple.h"
|
|
|
|
|
|
|
|
#include "support/debug.h"
|
|
|
|
#include "support/docstring_list.h"
|
|
|
|
#include "support/AppleSpeller.h"
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace lyx::support;
|
|
|
|
|
|
|
|
namespace lyx {
|
|
|
|
|
|
|
|
struct AppleSpellChecker::Private
|
|
|
|
{
|
|
|
|
Private();
|
|
|
|
|
|
|
|
~Private();
|
|
|
|
|
2010-08-05 20:10:40 +00:00
|
|
|
SpellChecker::Result toResult(SpellCheckResult status);
|
|
|
|
string toString(SpellCheckResult status);
|
2012-01-04 22:02:07 +00:00
|
|
|
int numDictionaries() const;
|
2017-07-03 13:53:14 -04:00
|
|
|
|
2010-07-07 17:40:50 +00:00
|
|
|
/// the speller
|
|
|
|
AppleSpeller speller;
|
2017-07-03 13:53:14 -04:00
|
|
|
|
2010-12-20 20:40:32 +00:00
|
|
|
/// language map
|
|
|
|
map<string, string> languageMap;
|
2010-07-07 17:40:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
AppleSpellChecker::Private::Private()
|
|
|
|
{
|
|
|
|
speller = newAppleSpeller();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
AppleSpellChecker::Private::~Private()
|
|
|
|
{
|
|
|
|
freeAppleSpeller(speller);
|
|
|
|
speller = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-27 15:45:27 +02:00
|
|
|
AppleSpellChecker::AppleSpellChecker()
|
|
|
|
: d(new Private)
|
|
|
|
{}
|
2010-07-07 17:40:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
AppleSpellChecker::~AppleSpellChecker()
|
|
|
|
{
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-08-05 20:10:40 +00:00
|
|
|
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";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-07 17:40:50 +00:00
|
|
|
SpellChecker::Result AppleSpellChecker::check(WordLangTuple const & word)
|
|
|
|
{
|
2010-12-20 20:40:32 +00:00
|
|
|
if (!hasDictionary(word.lang()))
|
2012-01-04 22:02:07 +00:00
|
|
|
return NO_DICTIONARY;
|
2010-12-20 20:40:32 +00:00
|
|
|
|
2010-07-07 17:40:50 +00:00
|
|
|
string const word_str = to_utf8(word.word());
|
2011-02-21 06:22:37 +00:00
|
|
|
string const lang = d->languageMap[word.lang()->lang()];
|
2010-09-04 07:48:15 +00:00
|
|
|
SpellCheckResult result =
|
|
|
|
AppleSpeller_check(d->speller,
|
2010-12-20 20:40:32 +00:00
|
|
|
word_str.c_str(), lang.c_str());
|
2010-09-04 07:48:15 +00:00
|
|
|
LYXERR(Debug::GUI, "spellCheck: \"" <<
|
|
|
|
word.word() << "\" = " << d->toString(result) <<
|
2010-12-20 20:40:32 +00:00
|
|
|
", lang = " << lang) ;
|
2010-08-05 20:10:40 +00:00
|
|
|
return d->toResult(result);
|
2010-07-07 17:40:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-14 05:24:04 +00:00
|
|
|
void AppleSpellChecker::advanceChangeNumber()
|
|
|
|
{
|
|
|
|
nextChangeNumber();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-07 17:40:50 +00:00
|
|
|
// add to personal dictionary
|
|
|
|
void AppleSpellChecker::insert(WordLangTuple const & word)
|
|
|
|
{
|
|
|
|
string const word_str = to_utf8(word.word());
|
2010-09-04 07:48:15 +00:00
|
|
|
AppleSpeller_learn(d->speller, word_str.c_str());
|
2017-07-30 09:30:58 +02:00
|
|
|
LYXERR(Debug::GUI, "learn word: \"" << word.word() << "\"");
|
2010-09-14 05:24:04 +00:00
|
|
|
advanceChangeNumber();
|
2010-08-05 20:10:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// remove from personal dictionary
|
|
|
|
void AppleSpellChecker::remove(WordLangTuple const & word)
|
|
|
|
{
|
|
|
|
string const word_str = to_utf8(word.word());
|
2010-09-04 07:48:15 +00:00
|
|
|
AppleSpeller_unlearn(d->speller, word_str.c_str());
|
2017-07-30 09:30:58 +02:00
|
|
|
LYXERR(Debug::GUI, "unlearn word: \"" << word.word() << "\"");
|
2010-09-14 05:24:04 +00:00
|
|
|
advanceChangeNumber();
|
2010-07-07 17:40:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ignore for session
|
|
|
|
void AppleSpellChecker::accept(WordLangTuple const & word)
|
|
|
|
{
|
|
|
|
string const word_str = to_utf8(word.word());
|
2010-09-04 07:48:15 +00:00
|
|
|
AppleSpeller_ignore(d->speller, word_str.c_str());
|
2017-07-30 09:30:58 +02:00
|
|
|
LYXERR(Debug::GUI, "ignore word: \"" << word.word() << "\"");
|
2010-09-14 05:24:04 +00:00
|
|
|
advanceChangeNumber();
|
2010-07-07 17:40:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void AppleSpellChecker::suggest(WordLangTuple const & wl,
|
|
|
|
docstring_list & suggestions)
|
|
|
|
{
|
|
|
|
suggestions.clear();
|
|
|
|
string const word_str = to_utf8(wl.word());
|
2017-07-03 13:53:14 -04:00
|
|
|
size_t num = AppleSpeller_makeSuggestion(d->speller,
|
2011-02-21 06:22:37 +00:00
|
|
|
word_str.c_str(), wl.lang()->code().c_str());
|
2010-07-07 17:40:50 +00:00
|
|
|
for (size_t i = 0; i < num; i++) {
|
2010-09-04 07:48:15 +00:00
|
|
|
char const * next = AppleSpeller_getSuggestion(d->speller, i);
|
2010-07-07 17:40:50 +00:00
|
|
|
if (!next) break;
|
|
|
|
suggestions.push_back(from_utf8(next));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool AppleSpellChecker::hasDictionary(Language const * lang) const
|
|
|
|
{
|
2011-02-21 06:22:37 +00:00
|
|
|
string const langmap = d->languageMap[lang->lang()];
|
2010-12-20 20:40:32 +00:00
|
|
|
bool result = !langmap.empty();
|
|
|
|
|
|
|
|
if (result)
|
|
|
|
return result;
|
|
|
|
|
2017-07-30 09:30:58 +02:00
|
|
|
result = AppleSpeller_hasLanguage(d->speller, lang->code().c_str());
|
2010-12-20 20:40:32 +00:00
|
|
|
if (result) {
|
2011-02-21 06:22:37 +00:00
|
|
|
d->languageMap[lang->lang()] = lang->code();
|
2010-12-20 20:40:32 +00:00
|
|
|
} else {
|
2017-07-30 09:30:58 +02:00
|
|
|
result = AppleSpeller_hasLanguage(d->speller, lang->lang().c_str());
|
2010-12-20 20:40:32 +00:00
|
|
|
if (result)
|
2011-02-21 06:22:37 +00:00
|
|
|
d->languageMap[lang->lang()] = lang->lang();
|
2010-12-20 20:40:32 +00:00
|
|
|
}
|
2017-07-30 09:30:58 +02:00
|
|
|
LYXERR(Debug::GUI, "has dictionary: " << lang->lang() << " = " << result);
|
2010-12-20 20:40:32 +00:00
|
|
|
return result;
|
2010-09-04 07:48:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-04 22:02:07 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2017-07-03 13:53:14 -04:00
|
|
|
|
2010-09-04 07:48:15 +00:00
|
|
|
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);
|
2010-07-07 17:40:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
docstring const AppleSpellChecker::error()
|
|
|
|
{
|
|
|
|
return docstring();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace lyx
|