3 * This file is part of LyX, the document processor.
4 * Licence details can be found in the file COPYING.
6 * \author Kevin Atkinson
9 * Full author contact details are available in file CREDITS.
14 #include "support/debug.h"
18 #include "ASpell_local.h"
20 #include "WordLangTuple.h"
22 #include "support/lassert.h"
28 ASpell::ASpell(BufferParams
const &, string
const & lang
)
29 : els(0), spell_error_object(0)
37 if (spell_error_object
) {
38 delete_aspell_can_have_error(spell_error_object
);
39 spell_error_object
= 0;
43 delete_aspell_string_enumeration(els
);
45 Spellers::iterator it
= spellers_
.begin();
46 Spellers::iterator end
= spellers_
.end();
48 for (; it
!= end
; ++it
) {
49 aspell_speller_save_all_word_lists(it
->second
.speller
);
50 delete_aspell_speller(it
->second
.speller
);
51 delete_aspell_config(it
->second
.config
);
56 void ASpell::addSpeller(string
const & lang
)
58 AspellConfig
* config
= new_aspell_config();
59 // FIXME The aspell documentation says to use "lang"
60 aspell_config_replace(config
, "language-tag", lang
.c_str());
61 // Set the encoding to utf-8.
62 // aspell does also understand "ucs-4", so we would not need a
63 // conversion in theory, but if this is used it expects all
64 // char const * arguments to be a cast from uint const *, and it
65 // seems that this uint is not compatible with our char_type on some
66 // platforms (cygwin, OS X). Therefore we use utf-8, that does
68 aspell_config_replace(config
, "encoding", "utf-8");
69 if (lyxrc
.isp_accept_compound
)
70 // Consider run-together words as legal compounds
71 aspell_config_replace(config
, "run-together", "true");
73 // Report run-together words as errors
74 aspell_config_replace(config
, "run-together", "false");
75 AspellCanHaveError
* err
= new_aspell_speller(config
);
76 if (spell_error_object
)
77 delete_aspell_can_have_error(spell_error_object
);
78 spell_error_object
= 0;
80 if (aspell_error_number(err
) == 0) {
82 m
.speller
= to_aspell_speller(err
);
86 spell_error_object
= err
;
91 ASpell::Result
ASpell::check(WordLangTuple
const & word
)
93 Result res
= UNKNOWN_WORD
;
95 Spellers::iterator it
= spellers_
.find(word
.lang_code());
96 if (it
== spellers_
.end()) {
97 addSpeller(word
.lang_code());
98 it
= spellers_
.find(word
.lang_code());
100 if (it
== spellers_
.end())
104 AspellSpeller
* m
= it
->second
.speller
;
106 int const word_ok
= aspell_speller_check(m
, to_utf8(word
.word()).c_str(), -1);
107 LASSERT(word_ok
!= -1, /**/);
112 AspellWordList
const * sugs
=
113 aspell_speller_suggest(m
, to_utf8(word
.word()).c_str(), -1);
114 LASSERT(sugs
!= 0, /**/);
115 els
= aspell_word_list_elements(sugs
);
116 if (aspell_word_list_empty(sugs
))
119 res
= SUGGESTED_WORDS
;
125 void ASpell::insert(WordLangTuple
const & word
)
127 Spellers::iterator it
= spellers_
.find(word
.lang_code());
128 if (it
!= spellers_
.end())
129 aspell_speller_add_to_personal(it
->second
.speller
, to_utf8(word
.word()).c_str(), -1);
133 void ASpell::accept(WordLangTuple
const & word
)
135 Spellers::iterator it
= spellers_
.find(word
.lang_code());
136 if (it
!= spellers_
.end())
137 aspell_speller_add_to_session(it
->second
.speller
, to_utf8(word
.word()).c_str(), -1);
141 docstring
const ASpell::nextMiss()
143 char const * str
= 0;
146 str
= aspell_string_enumeration_next(els
);
148 return (str
? from_utf8(str
) : docstring());
152 docstring
const ASpell::error()
154 char const * err
= 0;
156 if (spell_error_object
&& aspell_error_number(spell_error_object
) != 0)
157 err
= aspell_error_message(spell_error_object
);
159 // FIXME UNICODE: err is not in UTF8, but probably the locale encoding
160 return (err
? from_utf8(err
) : docstring());