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