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