xref: /trunk/main/lingucomponent/source/spellcheck/spell/sspellimp.cxx (revision a893be29343ee97512d484e6e8fefa91df2b44cb)
1b0844812SAndrew Rist /**************************************************************
2cdf0e10cSrcweir  *
3b0844812SAndrew Rist  * Licensed to the Apache Software Foundation (ASF) under one
4b0844812SAndrew Rist  * or more contributor license agreements.  See the NOTICE file
5b0844812SAndrew Rist  * distributed with this work for additional information
6b0844812SAndrew Rist  * regarding copyright ownership.  The ASF licenses this file
7b0844812SAndrew Rist  * to you under the Apache License, Version 2.0 (the
8b0844812SAndrew Rist  * "License"); you may not use this file except in compliance
9b0844812SAndrew Rist  * with the License.  You may obtain a copy of the License at
10cdf0e10cSrcweir  *
11b0844812SAndrew Rist  *   http://www.apache.org/licenses/LICENSE-2.0
12cdf0e10cSrcweir  *
13b0844812SAndrew Rist  * Unless required by applicable law or agreed to in writing,
14b0844812SAndrew Rist  * software distributed under the License is distributed on an
15b0844812SAndrew Rist  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16b0844812SAndrew Rist  * KIND, either express or implied.  See the License for the
17b0844812SAndrew Rist  * specific language governing permissions and limitations
18b0844812SAndrew Rist  * under the License.
19cdf0e10cSrcweir  *
20b0844812SAndrew Rist  *************************************************************/
21b0844812SAndrew Rist 
22b0844812SAndrew Rist 
23cdf0e10cSrcweir 
24cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
25cdf0e10cSrcweir #include "precompiled_lingucomponent.hxx"
26cdf0e10cSrcweir 
27cdf0e10cSrcweir #include <com/sun/star/uno/Reference.h>
28cdf0e10cSrcweir #include <com/sun/star/linguistic2/XSearchableDictionaryList.hpp>
29cdf0e10cSrcweir 
30cdf0e10cSrcweir #include <com/sun/star/linguistic2/SpellFailure.hpp>
31cdf0e10cSrcweir #include <cppuhelper/factory.hxx>   // helper for factories
32cdf0e10cSrcweir #include <com/sun/star/registry/XRegistryKey.hpp>
33cdf0e10cSrcweir #include <tools/debug.hxx>
34cdf0e10cSrcweir #include <unotools/processfactory.hxx>
35cdf0e10cSrcweir #include <osl/mutex.hxx>
36cdf0e10cSrcweir 
37cdf0e10cSrcweir #include <lingutil.hxx>
38cdf0e10cSrcweir #include <hunspell.hxx>
39cdf0e10cSrcweir #include <dictmgr.hxx>
40cdf0e10cSrcweir #include <sspellimp.hxx>
41cdf0e10cSrcweir 
42cdf0e10cSrcweir #include <linguistic/lngprops.hxx>
43cdf0e10cSrcweir #include <linguistic/spelldta.hxx>
44cdf0e10cSrcweir #include <i18npool/mslangid.hxx>
45cdf0e10cSrcweir #include <unotools/pathoptions.hxx>
46cdf0e10cSrcweir #include <unotools/lingucfg.hxx>
47cdf0e10cSrcweir #include <unotools/useroptions.hxx>
48cdf0e10cSrcweir #include <osl/file.hxx>
49cdf0e10cSrcweir #include <rtl/ustrbuf.hxx>
50cdf0e10cSrcweir #include <rtl/textenc.h>
51cdf0e10cSrcweir 
52cdf0e10cSrcweir #include <list>
53cdf0e10cSrcweir #include <set>
54cdf0e10cSrcweir #include <string.h>
55cdf0e10cSrcweir 
56cdf0e10cSrcweir using namespace utl;
57cdf0e10cSrcweir using namespace osl;
58cdf0e10cSrcweir using namespace rtl;
59cdf0e10cSrcweir using namespace com::sun::star;
60cdf0e10cSrcweir using namespace com::sun::star::beans;
61cdf0e10cSrcweir using namespace com::sun::star::lang;
62cdf0e10cSrcweir using namespace com::sun::star::uno;
63cdf0e10cSrcweir using namespace com::sun::star::linguistic2;
64cdf0e10cSrcweir using namespace linguistic;
65cdf0e10cSrcweir 
66cdf0e10cSrcweir // XML-header of SPELLML queries
67cdf0e10cSrcweir #define SPELLML_HEADER "<?xml?>"
68cdf0e10cSrcweir 
69cdf0e10cSrcweir ///////////////////////////////////////////////////////////////////////////
70cdf0e10cSrcweir 
71cdf0e10cSrcweir SpellChecker::SpellChecker() :
72cdf0e10cSrcweir     aEvtListeners   ( GetLinguMutex() )
73cdf0e10cSrcweir {
74cdf0e10cSrcweir     aDicts = NULL;
75cdf0e10cSrcweir     aDEncs = NULL;
76cdf0e10cSrcweir     aDLocs = NULL;
77cdf0e10cSrcweir     aDNames = NULL;
78cdf0e10cSrcweir     bDisposing = sal_False;
79cdf0e10cSrcweir     pPropHelper = NULL;
80cdf0e10cSrcweir     numdict = 0;
81cdf0e10cSrcweir }
82cdf0e10cSrcweir 
83cdf0e10cSrcweir 
84cdf0e10cSrcweir SpellChecker::~SpellChecker()
85cdf0e10cSrcweir {
86cdf0e10cSrcweir     if (aDicts)
87cdf0e10cSrcweir     {
88cdf0e10cSrcweir        for (int i = 0; i < numdict; i++)
89cdf0e10cSrcweir        {
90cdf0e10cSrcweir             if (aDicts[i]) delete aDicts[i];
91cdf0e10cSrcweir             aDicts[i] = NULL;
92cdf0e10cSrcweir        }
93cdf0e10cSrcweir        delete[] aDicts;
94cdf0e10cSrcweir     }
95cdf0e10cSrcweir     aDicts = NULL;
96cdf0e10cSrcweir     numdict = 0;
97cdf0e10cSrcweir     if (aDEncs) delete[] aDEncs;
98cdf0e10cSrcweir     aDEncs = NULL;
99cdf0e10cSrcweir     if (aDLocs) delete[] aDLocs;
100cdf0e10cSrcweir     aDLocs = NULL;
101cdf0e10cSrcweir     if (aDNames) delete[] aDNames;
102cdf0e10cSrcweir     aDNames = NULL;
103cdf0e10cSrcweir     if (pPropHelper)
104cdf0e10cSrcweir         pPropHelper->RemoveAsPropListener();
105cdf0e10cSrcweir }
106cdf0e10cSrcweir 
107cdf0e10cSrcweir 
108cdf0e10cSrcweir PropertyHelper_Spell & SpellChecker::GetPropHelper_Impl()
109cdf0e10cSrcweir {
110cdf0e10cSrcweir     if (!pPropHelper)
111cdf0e10cSrcweir     {
112cdf0e10cSrcweir         Reference< XPropertySet >   xPropSet( GetLinguProperties(), UNO_QUERY );
113cdf0e10cSrcweir 
114cdf0e10cSrcweir         pPropHelper = new PropertyHelper_Spell( (XSpellChecker *) this, xPropSet );
115cdf0e10cSrcweir         xPropHelper = pPropHelper;
116cdf0e10cSrcweir         pPropHelper->AddAsPropListener();   //! after a reference is established
117cdf0e10cSrcweir     }
118cdf0e10cSrcweir     return *pPropHelper;
119cdf0e10cSrcweir }
120cdf0e10cSrcweir 
121cdf0e10cSrcweir 
122cdf0e10cSrcweir Sequence< Locale > SAL_CALL SpellChecker::getLocales()
123cdf0e10cSrcweir         throw(RuntimeException)
124cdf0e10cSrcweir {
125cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
126cdf0e10cSrcweir 
127cdf0e10cSrcweir     // this routine should return the locales supported by the installed
128cdf0e10cSrcweir     // dictionaries.
129cdf0e10cSrcweir 
130cdf0e10cSrcweir     if (!numdict)
131cdf0e10cSrcweir     {
132cdf0e10cSrcweir         SvtLinguConfig aLinguCfg;
133cdf0e10cSrcweir 
134cdf0e10cSrcweir         // get list of extension dictionaries-to-use
135cdf0e10cSrcweir         // (or better speaking: the list of dictionaries using the
136cdf0e10cSrcweir         // new configuration entries).
137cdf0e10cSrcweir         std::list< SvtLinguConfigDictionaryEntry > aDics;
138cdf0e10cSrcweir         uno::Sequence< rtl::OUString > aFormatList;
139cdf0e10cSrcweir         aLinguCfg.GetSupportedDictionaryFormatsFor( A2OU("SpellCheckers"),
140cdf0e10cSrcweir                 A2OU("org.openoffice.lingu.MySpellSpellChecker"), aFormatList );
141cdf0e10cSrcweir         sal_Int32 nLen = aFormatList.getLength();
142cdf0e10cSrcweir         for (sal_Int32 i = 0;  i < nLen;  ++i)
143cdf0e10cSrcweir         {
144cdf0e10cSrcweir             std::vector< SvtLinguConfigDictionaryEntry > aTmpDic(
145cdf0e10cSrcweir                     aLinguCfg.GetActiveDictionariesByFormat( aFormatList[i] ) );
146cdf0e10cSrcweir             aDics.insert( aDics.end(), aTmpDic.begin(), aTmpDic.end() );
147cdf0e10cSrcweir         }
148cdf0e10cSrcweir 
149cdf0e10cSrcweir         //!! for compatibility with old dictionaries (the ones not using extensions
150cdf0e10cSrcweir         //!! or new configuration entries, but still using the dictionary.lst file)
151cdf0e10cSrcweir         //!! Get the list of old style spell checking dictionaries to use...
152cdf0e10cSrcweir         std::vector< SvtLinguConfigDictionaryEntry > aOldStyleDics(
153cdf0e10cSrcweir                 GetOldStyleDics( "DICT" ) );
154cdf0e10cSrcweir 
155cdf0e10cSrcweir         // to prefer dictionaries with configuration entries we will only
156cdf0e10cSrcweir         // use those old style dictionaries that add a language that
157cdf0e10cSrcweir         // is not yet supported by the list od new style dictionaries
158cdf0e10cSrcweir         MergeNewStyleDicsAndOldStyleDics( aDics, aOldStyleDics );
159cdf0e10cSrcweir 
160cdf0e10cSrcweir         numdict = aDics.size();
161cdf0e10cSrcweir         if (numdict)
162cdf0e10cSrcweir         {
163cdf0e10cSrcweir             // get supported locales from the dictionaries-to-use...
164cdf0e10cSrcweir             sal_Int32 k = 0;
165cdf0e10cSrcweir             std::set< rtl::OUString, lt_rtl_OUString > aLocaleNamesSet;
166cdf0e10cSrcweir             std::list< SvtLinguConfigDictionaryEntry >::const_iterator aDictIt;
167cdf0e10cSrcweir             for (aDictIt = aDics.begin();  aDictIt != aDics.end();  ++aDictIt)
168cdf0e10cSrcweir             {
169cdf0e10cSrcweir                 uno::Sequence< rtl::OUString > aLocaleNames( aDictIt->aLocaleNames );
170cdf0e10cSrcweir                 sal_Int32 nLen2 = aLocaleNames.getLength();
171cdf0e10cSrcweir                 for (k = 0;  k < nLen2;  ++k)
172cdf0e10cSrcweir                 {
173cdf0e10cSrcweir                     aLocaleNamesSet.insert( aLocaleNames[k] );
174cdf0e10cSrcweir                 }
175cdf0e10cSrcweir             }
176cdf0e10cSrcweir             // ... and add them to the resulting sequence
177cdf0e10cSrcweir             aSuppLocales.realloc( aLocaleNamesSet.size() );
178cdf0e10cSrcweir             std::set< rtl::OUString, lt_rtl_OUString >::const_iterator aItB;
179cdf0e10cSrcweir             k = 0;
180cdf0e10cSrcweir             for (aItB = aLocaleNamesSet.begin();  aItB != aLocaleNamesSet.end();  ++aItB)
181cdf0e10cSrcweir             {
182cdf0e10cSrcweir                 Locale aTmp( MsLangId::convertLanguageToLocale(
183cdf0e10cSrcweir                         MsLangId::convertIsoStringToLanguage( *aItB )));
184cdf0e10cSrcweir                 aSuppLocales[k++] = aTmp;
185cdf0e10cSrcweir             }
186cdf0e10cSrcweir 
187*a893be29SPedro Giffuni             //! For each dictionary and each locale we need a separate entry.
188cdf0e10cSrcweir             //! If this results in more than one dictionary per locale than (for now)
189cdf0e10cSrcweir             //! it is undefined which dictionary gets used.
190cdf0e10cSrcweir             //! In the future the implementation should support using several dictionaries
191cdf0e10cSrcweir             //! for one locale.
192cdf0e10cSrcweir             numdict = 0;
193cdf0e10cSrcweir             for (aDictIt = aDics.begin();  aDictIt != aDics.end();  ++aDictIt)
194cdf0e10cSrcweir                 numdict = numdict + aDictIt->aLocaleNames.getLength();
195cdf0e10cSrcweir 
196cdf0e10cSrcweir             // add dictionary information
197cdf0e10cSrcweir             aDicts  = new Hunspell* [numdict];
198cdf0e10cSrcweir             aDEncs  = new rtl_TextEncoding [numdict];
199cdf0e10cSrcweir             aDLocs  = new Locale [numdict];
200cdf0e10cSrcweir             aDNames = new OUString [numdict];
201cdf0e10cSrcweir             k = 0;
202cdf0e10cSrcweir             for (aDictIt = aDics.begin();  aDictIt != aDics.end();  ++aDictIt)
203cdf0e10cSrcweir             {
204cdf0e10cSrcweir                 if (aDictIt->aLocaleNames.getLength() > 0 &&
205cdf0e10cSrcweir                     aDictIt->aLocations.getLength() > 0)
206cdf0e10cSrcweir                 {
207cdf0e10cSrcweir                     uno::Sequence< rtl::OUString > aLocaleNames( aDictIt->aLocaleNames );
208cdf0e10cSrcweir                     sal_Int32 nLocales = aLocaleNames.getLength();
209cdf0e10cSrcweir 
210cdf0e10cSrcweir                     // currently only one language per dictionary is supported in the actual implementation...
211cdf0e10cSrcweir                     // Thus here we work-around this by adding the same dictionary several times.
212cdf0e10cSrcweir                     // Once for each of it's supported locales.
213cdf0e10cSrcweir                     for (sal_Int32 i = 0;  i < nLocales;  ++i)
214cdf0e10cSrcweir                     {
215cdf0e10cSrcweir                         aDicts[k]  = NULL;
216cdf0e10cSrcweir                         aDEncs[k]  = RTL_TEXTENCODING_DONTKNOW;
217cdf0e10cSrcweir                         aDLocs[k]  = MsLangId::convertLanguageToLocale(
218cdf0e10cSrcweir                                         MsLangId::convertIsoStringToLanguage( aLocaleNames[i] ));
219cdf0e10cSrcweir                         // also both files have to be in the same directory and the
220cdf0e10cSrcweir                         // file names must only differ in the extension (.aff/.dic).
221cdf0e10cSrcweir                         // Thus we use the first location only and strip the extension part.
222cdf0e10cSrcweir                         rtl::OUString aLocation = aDictIt->aLocations[0];
223cdf0e10cSrcweir                         sal_Int32 nPos = aLocation.lastIndexOf( '.' );
224cdf0e10cSrcweir                         aLocation = aLocation.copy( 0, nPos );
225cdf0e10cSrcweir                         aDNames[k] = aLocation;
226cdf0e10cSrcweir 
227cdf0e10cSrcweir                         ++k;
228cdf0e10cSrcweir                     }
229cdf0e10cSrcweir                 }
230cdf0e10cSrcweir             }
231cdf0e10cSrcweir             DBG_ASSERT( k == numdict, "index mismatch?" );
232cdf0e10cSrcweir         }
233cdf0e10cSrcweir         else
234cdf0e10cSrcweir         {
235cdf0e10cSrcweir             /* no dictionary found so register no dictionaries */
236cdf0e10cSrcweir             numdict = 0;
237cdf0e10cSrcweir             aDicts  = NULL;
238cdf0e10cSrcweir             aDEncs  = NULL;
239cdf0e10cSrcweir             aDLocs  = NULL;
240cdf0e10cSrcweir             aDNames = NULL;
241cdf0e10cSrcweir             aSuppLocales.realloc(0);
242cdf0e10cSrcweir         }
243cdf0e10cSrcweir     }
244cdf0e10cSrcweir 
245cdf0e10cSrcweir     return aSuppLocales;
246cdf0e10cSrcweir }
247cdf0e10cSrcweir 
248cdf0e10cSrcweir 
249cdf0e10cSrcweir sal_Bool SAL_CALL SpellChecker::hasLocale(const Locale& rLocale)
250cdf0e10cSrcweir         throw(RuntimeException)
251cdf0e10cSrcweir {
252cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
253cdf0e10cSrcweir 
254cdf0e10cSrcweir     sal_Bool bRes = sal_False;
255cdf0e10cSrcweir     if (!aSuppLocales.getLength())
256cdf0e10cSrcweir         getLocales();
257cdf0e10cSrcweir 
258cdf0e10cSrcweir     sal_Int32 nLen = aSuppLocales.getLength();
259cdf0e10cSrcweir     for (sal_Int32 i = 0;  i < nLen;  ++i)
260cdf0e10cSrcweir     {
261cdf0e10cSrcweir         const Locale *pLocale = aSuppLocales.getConstArray();
262cdf0e10cSrcweir         if (rLocale == pLocale[i])
263cdf0e10cSrcweir         {
264cdf0e10cSrcweir             bRes = sal_True;
265cdf0e10cSrcweir             break;
266cdf0e10cSrcweir         }
267cdf0e10cSrcweir     }
268cdf0e10cSrcweir     return bRes;
269cdf0e10cSrcweir }
270cdf0e10cSrcweir 
271cdf0e10cSrcweir 
272cdf0e10cSrcweir sal_Int16 SpellChecker::GetSpellFailure( const OUString &rWord, const Locale &rLocale )
273cdf0e10cSrcweir {
274cdf0e10cSrcweir     Hunspell * pMS = NULL;
275cdf0e10cSrcweir     rtl_TextEncoding eEnc = RTL_TEXTENCODING_DONTKNOW;
276cdf0e10cSrcweir 
277cdf0e10cSrcweir     // initialize a myspell object for each dictionary once
278cdf0e10cSrcweir     // (note: mutex is held higher up in isValid)
279cdf0e10cSrcweir 
280cdf0e10cSrcweir     sal_Int16 nRes = -1;
281cdf0e10cSrcweir 
282cdf0e10cSrcweir     // first handle smart quotes both single and double
283cdf0e10cSrcweir     OUStringBuffer rBuf(rWord);
284cdf0e10cSrcweir     sal_Int32 n = rBuf.getLength();
285cdf0e10cSrcweir     sal_Unicode c;
286cdf0e10cSrcweir     for (sal_Int32 ix=0; ix < n; ix++)
287cdf0e10cSrcweir     {
288cdf0e10cSrcweir         c = rBuf.charAt(ix);
289cdf0e10cSrcweir         if ((c == 0x201C) || (c == 0x201D))
290cdf0e10cSrcweir             rBuf.setCharAt(ix,(sal_Unicode)0x0022);
291cdf0e10cSrcweir         if ((c == 0x2018) || (c == 0x2019))
292cdf0e10cSrcweir             rBuf.setCharAt(ix,(sal_Unicode)0x0027);
293cdf0e10cSrcweir     }
294cdf0e10cSrcweir     OUString nWord(rBuf.makeStringAndClear());
295cdf0e10cSrcweir 
296cdf0e10cSrcweir     if (n)
297cdf0e10cSrcweir     {
298cdf0e10cSrcweir         for (sal_Int32 i = 0; i < numdict; ++i)
299cdf0e10cSrcweir         {
300cdf0e10cSrcweir             pMS = NULL;
301cdf0e10cSrcweir             eEnc = RTL_TEXTENCODING_DONTKNOW;
302cdf0e10cSrcweir 
303cdf0e10cSrcweir             if (rLocale == aDLocs[i])
304cdf0e10cSrcweir             {
305cdf0e10cSrcweir                 if (!aDicts[i])
306cdf0e10cSrcweir                 {
307cdf0e10cSrcweir                     OUString dicpath = aDNames[i] + A2OU(".dic");
308cdf0e10cSrcweir                     OUString affpath = aDNames[i] + A2OU(".aff");
309cdf0e10cSrcweir                     OUString dict;
310cdf0e10cSrcweir                     OUString aff;
311cdf0e10cSrcweir                     osl::FileBase::getSystemPathFromFileURL(dicpath,dict);
312cdf0e10cSrcweir                     osl::FileBase::getSystemPathFromFileURL(affpath,aff);
313cdf0e10cSrcweir                     OString aTmpaff(OU2ENC(aff,osl_getThreadTextEncoding()));
314cdf0e10cSrcweir                     OString aTmpdict(OU2ENC(dict,osl_getThreadTextEncoding()));
315cdf0e10cSrcweir 
316cdf0e10cSrcweir #if defined(WNT)
317cdf0e10cSrcweir                     // workaround for Windows specifc problem that the
318cdf0e10cSrcweir                     // path length in calls to 'fopen' is limted to somewhat
319cdf0e10cSrcweir                     // about 120+ characters which will usually be exceed when
320cdf0e10cSrcweir                     // using dictionaries as extensions.
321cdf0e10cSrcweir                     aTmpaff = Win_GetShortPathName( aff );
322cdf0e10cSrcweir                     aTmpdict = Win_GetShortPathName( dict );
323cdf0e10cSrcweir #endif
324cdf0e10cSrcweir 
325cdf0e10cSrcweir                     aDicts[i] = new Hunspell(aTmpaff.getStr(),aTmpdict.getStr());
326cdf0e10cSrcweir                     aDEncs[i] = RTL_TEXTENCODING_DONTKNOW;
327cdf0e10cSrcweir                     if (aDicts[i])
328cdf0e10cSrcweir                         aDEncs[i] = getTextEncodingFromCharset(aDicts[i]->get_dic_encoding());
329cdf0e10cSrcweir                 }
330cdf0e10cSrcweir                 pMS = aDicts[i];
331cdf0e10cSrcweir                 eEnc = aDEncs[i];
332cdf0e10cSrcweir             }
333cdf0e10cSrcweir 
334cdf0e10cSrcweir             if (pMS)
335cdf0e10cSrcweir             {
336cdf0e10cSrcweir                 // we don't want to work with a default text encoding since following incorrect
337cdf0e10cSrcweir                 // results may occur only for specific text and thus may be hard to notice.
338cdf0e10cSrcweir                 // Thus better always make a clean exit here if the text encoding is in question.
339cdf0e10cSrcweir                 // Hopefully something not working at all will raise proper attention quickly. ;-)
340cdf0e10cSrcweir                 DBG_ASSERT( eEnc != RTL_TEXTENCODING_DONTKNOW, "failed to get text encoding! (maybe incorrect encoding string in file)" );
341cdf0e10cSrcweir                 if (eEnc == RTL_TEXTENCODING_DONTKNOW)
342cdf0e10cSrcweir                     return -1;
343cdf0e10cSrcweir 
344cdf0e10cSrcweir                 OString aWrd(OU2ENC(nWord,eEnc));
345cdf0e10cSrcweir                 int rVal = pMS->spell((char*)aWrd.getStr());
346cdf0e10cSrcweir                 if (rVal != 1)
347cdf0e10cSrcweir                     nRes = SpellFailure::SPELLING_ERROR;
348cdf0e10cSrcweir                 else
349cdf0e10cSrcweir                     return -1;
350cdf0e10cSrcweir                 pMS = NULL;
351cdf0e10cSrcweir             }
352cdf0e10cSrcweir         }
353cdf0e10cSrcweir     }
354cdf0e10cSrcweir 
355cdf0e10cSrcweir     return nRes;
356cdf0e10cSrcweir }
357cdf0e10cSrcweir 
358cdf0e10cSrcweir 
359cdf0e10cSrcweir sal_Bool SAL_CALL SpellChecker::isValid( const OUString& rWord, const Locale& rLocale,
360cdf0e10cSrcweir             const PropertyValues& rProperties )
361cdf0e10cSrcweir         throw(IllegalArgumentException, RuntimeException)
362cdf0e10cSrcweir {
363cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
364cdf0e10cSrcweir 
365cdf0e10cSrcweir     if (rLocale == Locale()  ||  !rWord.getLength())
366cdf0e10cSrcweir         return sal_True;
367cdf0e10cSrcweir 
368cdf0e10cSrcweir     if (!hasLocale( rLocale ))
369cdf0e10cSrcweir #ifdef LINGU_EXCEPTIONS
370cdf0e10cSrcweir         throw( IllegalArgumentException() );
371cdf0e10cSrcweir #else
372cdf0e10cSrcweir         return sal_True;
373cdf0e10cSrcweir #endif
374cdf0e10cSrcweir 
375cdf0e10cSrcweir     // return sal_False to process SPELLML requests (they are longer than the header)
376cdf0e10cSrcweir     if (rWord.match(A2OU(SPELLML_HEADER), 0) && (rWord.getLength() > 10)) return sal_False;
377cdf0e10cSrcweir 
378cdf0e10cSrcweir     // Get property values to be used.
379cdf0e10cSrcweir     // These are be the default values set in the SN_LINGU_PROPERTIES
380cdf0e10cSrcweir     // PropertySet which are overridden by the supplied ones from the
381cdf0e10cSrcweir     // last argument.
382*a893be29SPedro Giffuni     // You'll probably like to use a simpler solution than the provided
383cdf0e10cSrcweir     // one using the PropertyHelper_Spell.
384cdf0e10cSrcweir 
385cdf0e10cSrcweir     PropertyHelper_Spell &rHelper = GetPropHelper();
386cdf0e10cSrcweir     rHelper.SetTmpPropVals( rProperties );
387cdf0e10cSrcweir 
388cdf0e10cSrcweir     sal_Int16 nFailure = GetSpellFailure( rWord, rLocale );
389cdf0e10cSrcweir     if (nFailure != -1 && !rWord.match(A2OU(SPELLML_HEADER), 0))
390cdf0e10cSrcweir     {
391cdf0e10cSrcweir         sal_Int16 nLang = LocaleToLanguage( rLocale );
392cdf0e10cSrcweir         // postprocess result for errors that should be ignored
393cdf0e10cSrcweir         const bool bIgnoreError =
394cdf0e10cSrcweir                 (!rHelper.IsSpellUpperCase()  && IsUpper( rWord, nLang )) ||
395cdf0e10cSrcweir                 (!rHelper.IsSpellWithDigits() && HasDigits( rWord )) ||
396cdf0e10cSrcweir                 (!rHelper.IsSpellCapitalization()  &&  nFailure == SpellFailure::CAPTION_ERROR);
397cdf0e10cSrcweir         if (bIgnoreError)
398cdf0e10cSrcweir             nFailure = -1;
399cdf0e10cSrcweir     }
400cdf0e10cSrcweir 
401cdf0e10cSrcweir     return (nFailure == -1);
402cdf0e10cSrcweir }
403cdf0e10cSrcweir 
404cdf0e10cSrcweir 
405cdf0e10cSrcweir Reference< XSpellAlternatives >
406cdf0e10cSrcweir     SpellChecker::GetProposals( const OUString &rWord, const Locale &rLocale )
407cdf0e10cSrcweir {
408cdf0e10cSrcweir     // Retrieves the return values for the 'spell' function call in case
409cdf0e10cSrcweir     // of a misspelled word.
410cdf0e10cSrcweir     // Especially it may give a list of suggested (correct) words:
411cdf0e10cSrcweir 
412cdf0e10cSrcweir     Reference< XSpellAlternatives > xRes;
413cdf0e10cSrcweir     // note: mutex is held by higher up by spell which covers both
414cdf0e10cSrcweir 
415cdf0e10cSrcweir     Hunspell* pMS = NULL;
416cdf0e10cSrcweir     rtl_TextEncoding eEnc = RTL_TEXTENCODING_DONTKNOW;
417cdf0e10cSrcweir     int count = 0;
418cdf0e10cSrcweir     int numsug = 0;
419cdf0e10cSrcweir 
420cdf0e10cSrcweir     // first handle smart quotes (single and double)
421cdf0e10cSrcweir     OUStringBuffer rBuf(rWord);
422cdf0e10cSrcweir     sal_Int32 n = rBuf.getLength();
423cdf0e10cSrcweir     sal_Unicode c;
424cdf0e10cSrcweir     for (sal_Int32 ix=0; ix < n; ix++)
425cdf0e10cSrcweir     {
426cdf0e10cSrcweir         c = rBuf.charAt(ix);
427cdf0e10cSrcweir         if ((c == 0x201C) || (c == 0x201D))
428cdf0e10cSrcweir             rBuf.setCharAt(ix,(sal_Unicode)0x0022);
429cdf0e10cSrcweir         if ((c == 0x2018) || (c == 0x2019))
430cdf0e10cSrcweir             rBuf.setCharAt(ix,(sal_Unicode)0x0027);
431cdf0e10cSrcweir     }
432cdf0e10cSrcweir     OUString nWord(rBuf.makeStringAndClear());
433cdf0e10cSrcweir 
434cdf0e10cSrcweir     if (n)
435cdf0e10cSrcweir     {
436cdf0e10cSrcweir         sal_Int16 nLang = LocaleToLanguage( rLocale );
437cdf0e10cSrcweir 
438cdf0e10cSrcweir         Sequence< OUString > aStr( 0 );
439cdf0e10cSrcweir 
440cdf0e10cSrcweir         for (int i =0; i < numdict; i++)
441cdf0e10cSrcweir         {
442cdf0e10cSrcweir             pMS = NULL;
443cdf0e10cSrcweir             eEnc = RTL_TEXTENCODING_DONTKNOW;
444cdf0e10cSrcweir             count = 0;
445cdf0e10cSrcweir 
446cdf0e10cSrcweir             if (rLocale == aDLocs[i])
447cdf0e10cSrcweir             {
448cdf0e10cSrcweir                 pMS = aDicts[i];
449cdf0e10cSrcweir                 eEnc = aDEncs[i];
450cdf0e10cSrcweir             }
451cdf0e10cSrcweir 
452cdf0e10cSrcweir             if (pMS)
453cdf0e10cSrcweir             {
454cdf0e10cSrcweir                 char ** suglst = NULL;
455cdf0e10cSrcweir                 OString aWrd(OU2ENC(nWord,eEnc));
456cdf0e10cSrcweir                 count = pMS->suggest(&suglst, (const char *) aWrd.getStr());
457cdf0e10cSrcweir 
458cdf0e10cSrcweir                 if (count)
459cdf0e10cSrcweir                 {
460cdf0e10cSrcweir                     aStr.realloc( numsug + count );
461cdf0e10cSrcweir                     OUString *pStr = aStr.getArray();
462cdf0e10cSrcweir                     for (int ii=0; ii < count; ++ii)
463cdf0e10cSrcweir                     {
464cdf0e10cSrcweir                         OUString cvtwrd(suglst[ii],strlen(suglst[ii]),eEnc);
465cdf0e10cSrcweir                         pStr[numsug + ii] = cvtwrd;
466cdf0e10cSrcweir                     }
467cdf0e10cSrcweir                     pMS->free_list(&suglst, count);
468cdf0e10cSrcweir                     numsug += count;
469cdf0e10cSrcweir                 }
470cdf0e10cSrcweir             }
471cdf0e10cSrcweir         }
472cdf0e10cSrcweir 
473cdf0e10cSrcweir         // now return an empty alternative for no suggestions or the list of alternatives if some found
474cdf0e10cSrcweir         SpellAlternatives *pAlt = new SpellAlternatives;
475cdf0e10cSrcweir         String aTmp(rWord);
476cdf0e10cSrcweir         pAlt->SetWordLanguage( aTmp, nLang );
477cdf0e10cSrcweir         pAlt->SetFailureType( SpellFailure::SPELLING_ERROR );
478cdf0e10cSrcweir         pAlt->SetAlternatives( aStr );
479cdf0e10cSrcweir         xRes = pAlt;
480cdf0e10cSrcweir         return xRes;
481cdf0e10cSrcweir     }
482cdf0e10cSrcweir     return xRes;
483cdf0e10cSrcweir }
484cdf0e10cSrcweir 
485cdf0e10cSrcweir 
486cdf0e10cSrcweir Reference< XSpellAlternatives > SAL_CALL SpellChecker::spell(
487cdf0e10cSrcweir         const OUString& rWord, const Locale& rLocale,
488cdf0e10cSrcweir         const PropertyValues& rProperties )
489cdf0e10cSrcweir         throw(IllegalArgumentException, RuntimeException)
490cdf0e10cSrcweir {
491cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
492cdf0e10cSrcweir 
493cdf0e10cSrcweir     if (rLocale == Locale()  ||  !rWord.getLength())
494cdf0e10cSrcweir         return NULL;
495cdf0e10cSrcweir 
496cdf0e10cSrcweir     if (!hasLocale( rLocale ))
497cdf0e10cSrcweir #ifdef LINGU_EXCEPTIONS
498cdf0e10cSrcweir         throw( IllegalArgumentException() );
499cdf0e10cSrcweir #else
500cdf0e10cSrcweir         return NULL;
501cdf0e10cSrcweir #endif
502cdf0e10cSrcweir 
503cdf0e10cSrcweir     Reference< XSpellAlternatives > xAlt;
504cdf0e10cSrcweir     if (!isValid( rWord, rLocale, rProperties ))
505cdf0e10cSrcweir     {
506cdf0e10cSrcweir         xAlt =  GetProposals( rWord, rLocale );
507cdf0e10cSrcweir     }
508cdf0e10cSrcweir     return xAlt;
509cdf0e10cSrcweir }
510cdf0e10cSrcweir 
511cdf0e10cSrcweir 
512cdf0e10cSrcweir Reference< XInterface > SAL_CALL SpellChecker_CreateInstance(
513cdf0e10cSrcweir         const Reference< XMultiServiceFactory > & /*rSMgr*/ )
514cdf0e10cSrcweir         throw(Exception)
515cdf0e10cSrcweir {
516cdf0e10cSrcweir 
517cdf0e10cSrcweir     Reference< XInterface > xService = (cppu::OWeakObject*) new SpellChecker;
518cdf0e10cSrcweir     return xService;
519cdf0e10cSrcweir }
520cdf0e10cSrcweir 
521cdf0e10cSrcweir 
522cdf0e10cSrcweir sal_Bool SAL_CALL SpellChecker::addLinguServiceEventListener(
523cdf0e10cSrcweir         const Reference< XLinguServiceEventListener >& rxLstnr )
524cdf0e10cSrcweir         throw(RuntimeException)
525cdf0e10cSrcweir {
526cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
527cdf0e10cSrcweir 
528cdf0e10cSrcweir     sal_Bool bRes = sal_False;
529cdf0e10cSrcweir     if (!bDisposing && rxLstnr.is())
530cdf0e10cSrcweir     {
531cdf0e10cSrcweir         bRes = GetPropHelper().addLinguServiceEventListener( rxLstnr );
532cdf0e10cSrcweir     }
533cdf0e10cSrcweir     return bRes;
534cdf0e10cSrcweir }
535cdf0e10cSrcweir 
536cdf0e10cSrcweir 
537cdf0e10cSrcweir sal_Bool SAL_CALL SpellChecker::removeLinguServiceEventListener(
538cdf0e10cSrcweir         const Reference< XLinguServiceEventListener >& rxLstnr )
539cdf0e10cSrcweir         throw(RuntimeException)
540cdf0e10cSrcweir {
541cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
542cdf0e10cSrcweir 
543cdf0e10cSrcweir     sal_Bool bRes = sal_False;
544cdf0e10cSrcweir     if (!bDisposing && rxLstnr.is())
545cdf0e10cSrcweir     {
546cdf0e10cSrcweir         DBG_ASSERT( xPropHelper.is(), "xPropHelper non existent" );
547cdf0e10cSrcweir         bRes = GetPropHelper().removeLinguServiceEventListener( rxLstnr );
548cdf0e10cSrcweir     }
549cdf0e10cSrcweir     return bRes;
550cdf0e10cSrcweir }
551cdf0e10cSrcweir 
552cdf0e10cSrcweir 
553cdf0e10cSrcweir OUString SAL_CALL SpellChecker::getServiceDisplayName( const Locale& /*rLocale*/ )
554cdf0e10cSrcweir         throw(RuntimeException)
555cdf0e10cSrcweir {
556cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
557cdf0e10cSrcweir     return A2OU( "Hunspell SpellChecker" );
558cdf0e10cSrcweir }
559cdf0e10cSrcweir 
560cdf0e10cSrcweir 
561cdf0e10cSrcweir void SAL_CALL SpellChecker::initialize( const Sequence< Any >& rArguments )
562cdf0e10cSrcweir         throw(Exception, RuntimeException)
563cdf0e10cSrcweir {
564cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
565cdf0e10cSrcweir 
566cdf0e10cSrcweir     if (!pPropHelper)
567cdf0e10cSrcweir     {
568cdf0e10cSrcweir         sal_Int32 nLen = rArguments.getLength();
569cdf0e10cSrcweir         if (2 == nLen)
570cdf0e10cSrcweir         {
571cdf0e10cSrcweir             Reference< XPropertySet >   xPropSet;
572cdf0e10cSrcweir             rArguments.getConstArray()[0] >>= xPropSet;
573cdf0e10cSrcweir             //rArguments.getConstArray()[1] >>= xDicList;
574cdf0e10cSrcweir 
575cdf0e10cSrcweir             //! Pointer allows for access of the non-UNO functions.
576cdf0e10cSrcweir             //! And the reference to the UNO-functions while increasing
577cdf0e10cSrcweir             //! the ref-count and will implicitly free the memory
578cdf0e10cSrcweir             //! when the object is not longer used.
579cdf0e10cSrcweir             pPropHelper = new PropertyHelper_Spell( (XSpellChecker *) this, xPropSet );
580cdf0e10cSrcweir             xPropHelper = pPropHelper;
581cdf0e10cSrcweir             pPropHelper->AddAsPropListener();   //! after a reference is established
582cdf0e10cSrcweir         }
583cdf0e10cSrcweir         else
584cdf0e10cSrcweir         {
585cdf0e10cSrcweir             DBG_ERROR( "wrong number of arguments in sequence" );
586cdf0e10cSrcweir         }
587cdf0e10cSrcweir     }
588cdf0e10cSrcweir }
589cdf0e10cSrcweir 
590cdf0e10cSrcweir 
591cdf0e10cSrcweir void SAL_CALL SpellChecker::dispose()
592cdf0e10cSrcweir         throw(RuntimeException)
593cdf0e10cSrcweir {
594cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
595cdf0e10cSrcweir 
596cdf0e10cSrcweir     if (!bDisposing)
597cdf0e10cSrcweir     {
598cdf0e10cSrcweir         bDisposing = sal_True;
599cdf0e10cSrcweir         EventObject aEvtObj( (XSpellChecker *) this );
600cdf0e10cSrcweir         aEvtListeners.disposeAndClear( aEvtObj );
601cdf0e10cSrcweir     }
602cdf0e10cSrcweir }
603cdf0e10cSrcweir 
604cdf0e10cSrcweir 
605cdf0e10cSrcweir void SAL_CALL SpellChecker::addEventListener( const Reference< XEventListener >& rxListener )
606cdf0e10cSrcweir         throw(RuntimeException)
607cdf0e10cSrcweir {
608cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
609cdf0e10cSrcweir 
610cdf0e10cSrcweir     if (!bDisposing && rxListener.is())
611cdf0e10cSrcweir         aEvtListeners.addInterface( rxListener );
612cdf0e10cSrcweir }
613cdf0e10cSrcweir 
614cdf0e10cSrcweir 
615cdf0e10cSrcweir void SAL_CALL SpellChecker::removeEventListener( const Reference< XEventListener >& rxListener )
616cdf0e10cSrcweir         throw(RuntimeException)
617cdf0e10cSrcweir {
618cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
619cdf0e10cSrcweir 
620cdf0e10cSrcweir     if (!bDisposing && rxListener.is())
621cdf0e10cSrcweir         aEvtListeners.removeInterface( rxListener );
622cdf0e10cSrcweir }
623cdf0e10cSrcweir 
624cdf0e10cSrcweir 
625cdf0e10cSrcweir ///////////////////////////////////////////////////////////////////////////
626cdf0e10cSrcweir // Service specific part
627cdf0e10cSrcweir //
628cdf0e10cSrcweir 
629cdf0e10cSrcweir OUString SAL_CALL SpellChecker::getImplementationName()
630cdf0e10cSrcweir         throw(RuntimeException)
631cdf0e10cSrcweir {
632cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
633cdf0e10cSrcweir 
634cdf0e10cSrcweir     return getImplementationName_Static();
635cdf0e10cSrcweir }
636cdf0e10cSrcweir 
637cdf0e10cSrcweir 
638cdf0e10cSrcweir sal_Bool SAL_CALL SpellChecker::supportsService( const OUString& ServiceName )
639cdf0e10cSrcweir         throw(RuntimeException)
640cdf0e10cSrcweir {
641cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
642cdf0e10cSrcweir 
643cdf0e10cSrcweir     Sequence< OUString > aSNL = getSupportedServiceNames();
644cdf0e10cSrcweir     const OUString * pArray = aSNL.getConstArray();
645cdf0e10cSrcweir     for( sal_Int32 i = 0; i < aSNL.getLength(); i++ )
646cdf0e10cSrcweir         if( pArray[i] == ServiceName )
647cdf0e10cSrcweir             return sal_True;
648cdf0e10cSrcweir     return sal_False;
649cdf0e10cSrcweir }
650cdf0e10cSrcweir 
651cdf0e10cSrcweir 
652cdf0e10cSrcweir Sequence< OUString > SAL_CALL SpellChecker::getSupportedServiceNames()
653cdf0e10cSrcweir         throw(RuntimeException)
654cdf0e10cSrcweir {
655cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
656cdf0e10cSrcweir 
657cdf0e10cSrcweir     return getSupportedServiceNames_Static();
658cdf0e10cSrcweir }
659cdf0e10cSrcweir 
660cdf0e10cSrcweir 
661cdf0e10cSrcweir Sequence< OUString > SpellChecker::getSupportedServiceNames_Static()
662cdf0e10cSrcweir         throw()
663cdf0e10cSrcweir {
664cdf0e10cSrcweir     MutexGuard  aGuard( GetLinguMutex() );
665cdf0e10cSrcweir 
666cdf0e10cSrcweir     Sequence< OUString > aSNS( 1 ); // auch mehr als 1 Service moeglich
667cdf0e10cSrcweir     aSNS.getArray()[0] = A2OU( SN_SPELLCHECKER );
668cdf0e10cSrcweir     return aSNS;
669cdf0e10cSrcweir }
670cdf0e10cSrcweir 
671cdf0e10cSrcweir void * SAL_CALL SpellChecker_getFactory( const sal_Char * pImplName,
672cdf0e10cSrcweir             XMultiServiceFactory * pServiceManager, void *  )
673cdf0e10cSrcweir {
674cdf0e10cSrcweir     void * pRet = 0;
675cdf0e10cSrcweir     if ( !SpellChecker::getImplementationName_Static().compareToAscii( pImplName ) )
676cdf0e10cSrcweir     {
677cdf0e10cSrcweir         Reference< XSingleServiceFactory > xFactory =
678cdf0e10cSrcweir             cppu::createOneInstanceFactory(
679cdf0e10cSrcweir                 pServiceManager,
680cdf0e10cSrcweir                 SpellChecker::getImplementationName_Static(),
681cdf0e10cSrcweir                 SpellChecker_CreateInstance,
682cdf0e10cSrcweir                 SpellChecker::getSupportedServiceNames_Static());
683cdf0e10cSrcweir         // acquire, because we return an interface pointer instead of a reference
684cdf0e10cSrcweir         xFactory->acquire();
685cdf0e10cSrcweir         pRet = xFactory.get();
686cdf0e10cSrcweir     }
687cdf0e10cSrcweir     return pRet;
688cdf0e10cSrcweir }
689cdf0e10cSrcweir 
690cdf0e10cSrcweir 
691cdf0e10cSrcweir ///////////////////////////////////////////////////////////////////////////
692