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