1*3b8558fdSAndrew Rist /************************************************************** 2cdf0e10cSrcweir * 3*3b8558fdSAndrew Rist * Licensed to the Apache Software Foundation (ASF) under one 4*3b8558fdSAndrew Rist * or more contributor license agreements. See the NOTICE file 5*3b8558fdSAndrew Rist * distributed with this work for additional information 6*3b8558fdSAndrew Rist * regarding copyright ownership. The ASF licenses this file 7*3b8558fdSAndrew Rist * to you under the Apache License, Version 2.0 (the 8*3b8558fdSAndrew Rist * "License"); you may not use this file except in compliance 9*3b8558fdSAndrew Rist * with the License. You may obtain a copy of the License at 10cdf0e10cSrcweir * 11*3b8558fdSAndrew Rist * http://www.apache.org/licenses/LICENSE-2.0 12cdf0e10cSrcweir * 13*3b8558fdSAndrew Rist * Unless required by applicable law or agreed to in writing, 14*3b8558fdSAndrew Rist * software distributed under the License is distributed on an 15*3b8558fdSAndrew Rist * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16*3b8558fdSAndrew Rist * KIND, either express or implied. See the License for the 17*3b8558fdSAndrew Rist * specific language governing permissions and limitations 18*3b8558fdSAndrew Rist * under the License. 19cdf0e10cSrcweir * 20*3b8558fdSAndrew Rist *************************************************************/ 21*3b8558fdSAndrew Rist 22*3b8558fdSAndrew Rist 23cdf0e10cSrcweir 24cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove 25cdf0e10cSrcweir #include "precompiled_linguistic.hxx" 26cdf0e10cSrcweir #include <tools/string.hxx> 27cdf0e10cSrcweir #include <tools/fsys.hxx> 28cdf0e10cSrcweir #include <tools/debug.hxx> 29cdf0e10cSrcweir #include <unotools/pathoptions.hxx> 30cdf0e10cSrcweir #include <svl/lngmisc.hxx> 31cdf0e10cSrcweir #include <ucbhelper/content.hxx> 32cdf0e10cSrcweir #include <i18npool/mslangid.hxx> 33cdf0e10cSrcweir #include <com/sun/star/ucb/XCommandEnvironment.hpp> 34cdf0e10cSrcweir #include <com/sun/star/beans/XPropertySet.hpp> 35cdf0e10cSrcweir #include <com/sun/star/beans/XFastPropertySet.hpp> 36cdf0e10cSrcweir #include <com/sun/star/beans/XPropertyChangeListener.hpp> 37cdf0e10cSrcweir #include <com/sun/star/frame/XTerminateListener.hpp> 38cdf0e10cSrcweir #include <com/sun/star/frame/XDesktop.hpp> 39cdf0e10cSrcweir #include <com/sun/star/frame/XStorable.hpp> 40cdf0e10cSrcweir 41cdf0e10cSrcweir #include <com/sun/star/beans/PropertyValues.hpp> 42cdf0e10cSrcweir #include <com/sun/star/uno/Sequence.hxx> 43cdf0e10cSrcweir #include <com/sun/star/uno/Reference.h> 44cdf0e10cSrcweir #include <com/sun/star/linguistic2/DictionaryType.hpp> 45cdf0e10cSrcweir #include <com/sun/star/linguistic2/XSearchableDictionaryList.hpp> 46cdf0e10cSrcweir #include <unotools/processfactory.hxx> 47cdf0e10cSrcweir #include <unotools/localedatawrapper.hxx> 48cdf0e10cSrcweir #include <unotools/syslocale.hxx> 49cdf0e10cSrcweir 50cdf0e10cSrcweir #include <rtl/instance.hxx> 51cdf0e10cSrcweir 52cdf0e10cSrcweir #include "linguistic/misc.hxx" 53cdf0e10cSrcweir #include "defs.hxx" 54cdf0e10cSrcweir #include "linguistic/lngprops.hxx" 55cdf0e10cSrcweir #include "linguistic/hyphdta.hxx" 56cdf0e10cSrcweir #include <i18npool/mslangid.hxx> 57cdf0e10cSrcweir 58cdf0e10cSrcweir using namespace utl; 59cdf0e10cSrcweir using namespace osl; 60cdf0e10cSrcweir using namespace rtl; 61cdf0e10cSrcweir using namespace com::sun::star; 62cdf0e10cSrcweir using namespace com::sun::star::beans; 63cdf0e10cSrcweir using namespace com::sun::star::lang; 64cdf0e10cSrcweir using namespace com::sun::star::uno; 65cdf0e10cSrcweir using namespace com::sun::star::i18n; 66cdf0e10cSrcweir using namespace com::sun::star::linguistic2; 67cdf0e10cSrcweir 68cdf0e10cSrcweir namespace linguistic 69cdf0e10cSrcweir { 70cdf0e10cSrcweir 71cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 72cdf0e10cSrcweir 73cdf0e10cSrcweir //!! multi-thread safe mutex for all platforms !! 74cdf0e10cSrcweir struct LinguMutex : public rtl::Static< osl::Mutex, LinguMutex > 75cdf0e10cSrcweir { 76cdf0e10cSrcweir }; 77cdf0e10cSrcweir 78cdf0e10cSrcweir osl::Mutex & GetLinguMutex() 79cdf0e10cSrcweir { 80cdf0e10cSrcweir return LinguMutex::get(); 81cdf0e10cSrcweir } 82cdf0e10cSrcweir 83cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 84cdf0e10cSrcweir 85cdf0e10cSrcweir LocaleDataWrapper & GetLocaleDataWrapper( sal_Int16 nLang ) 86cdf0e10cSrcweir { 87cdf0e10cSrcweir static LocaleDataWrapper aLclDtaWrp( 88cdf0e10cSrcweir getProcessServiceFactory(), 89cdf0e10cSrcweir CreateLocale( SvtSysLocale().GetUILanguage() ) ); 90cdf0e10cSrcweir 91cdf0e10cSrcweir const Locale &rLcl = aLclDtaWrp.getLoadedLocale(); 92cdf0e10cSrcweir Locale aLcl( CreateLocale( nLang ) ); 93cdf0e10cSrcweir if (aLcl.Language != rLcl.Language || 94cdf0e10cSrcweir aLcl.Country != rLcl.Country || 95cdf0e10cSrcweir aLcl.Variant != rLcl.Variant) 96cdf0e10cSrcweir aLclDtaWrp.setLocale( aLcl ); 97cdf0e10cSrcweir return aLclDtaWrp; 98cdf0e10cSrcweir } 99cdf0e10cSrcweir 100cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 101cdf0e10cSrcweir 102cdf0e10cSrcweir /** 103cdf0e10cSrcweir returns text-encoding used for ByteString unicode String conversion 104cdf0e10cSrcweir */ 105cdf0e10cSrcweir rtl_TextEncoding GetTextEncoding( sal_Int16 nLanguage ) 106cdf0e10cSrcweir { 107cdf0e10cSrcweir DBG_ASSERT( nLanguage != LANGUAGE_NONE, "invalid language argument" ); 108cdf0e10cSrcweir static sal_Int16 nLastLanguage = LANGUAGE_NONE; 109cdf0e10cSrcweir 110cdf0e10cSrcweir // set default value for unknown languages 111cdf0e10cSrcweir static rtl_TextEncoding nEncoding = RTL_TEXTENCODING_DONTKNOW; 112cdf0e10cSrcweir 113cdf0e10cSrcweir if (nLastLanguage != nLanguage) 114cdf0e10cSrcweir { 115cdf0e10cSrcweir //!! IPR uses textencodings Latin-1, Latin-2, Latin-5 and Latin-7 !! 116cdf0e10cSrcweir 117cdf0e10cSrcweir nLastLanguage = nLanguage; 118cdf0e10cSrcweir switch (nLanguage) 119cdf0e10cSrcweir { 120cdf0e10cSrcweir case LANGUAGE_GERMAN : 121cdf0e10cSrcweir case LANGUAGE_GERMAN_SWISS : 122cdf0e10cSrcweir case LANGUAGE_ENGLISH_US : 123cdf0e10cSrcweir case LANGUAGE_ENGLISH_UK : 124cdf0e10cSrcweir case LANGUAGE_FRENCH : 125cdf0e10cSrcweir case LANGUAGE_ITALIAN : 126cdf0e10cSrcweir case LANGUAGE_SPANISH : 127cdf0e10cSrcweir case LANGUAGE_CATALAN : 128cdf0e10cSrcweir case LANGUAGE_PORTUGUESE : 129cdf0e10cSrcweir case LANGUAGE_PORTUGUESE_BRAZILIAN : 130cdf0e10cSrcweir case LANGUAGE_DANISH : 131cdf0e10cSrcweir case LANGUAGE_DUTCH : 132cdf0e10cSrcweir case LANGUAGE_SWEDISH : 133cdf0e10cSrcweir case LANGUAGE_FINNISH : 134cdf0e10cSrcweir case LANGUAGE_NORWEGIAN_BOKMAL : 135cdf0e10cSrcweir case LANGUAGE_NORWEGIAN_NYNORSK : 136cdf0e10cSrcweir case LANGUAGE_AFRIKAANS : 137cdf0e10cSrcweir case LANGUAGE_ENGLISH_EIRE : 138cdf0e10cSrcweir case LANGUAGE_ENGLISH_AUS : 139cdf0e10cSrcweir #ifdef WNT 140cdf0e10cSrcweir nEncoding = RTL_TEXTENCODING_MS_1252; break; 141cdf0e10cSrcweir #else 142cdf0e10cSrcweir nEncoding = RTL_TEXTENCODING_ISO_8859_1; break; 143cdf0e10cSrcweir #endif 144cdf0e10cSrcweir case LANGUAGE_CZECH : 145cdf0e10cSrcweir case LANGUAGE_HUNGARIAN : 146cdf0e10cSrcweir case LANGUAGE_POLISH : 147cdf0e10cSrcweir #ifdef WNT 148cdf0e10cSrcweir nEncoding = RTL_TEXTENCODING_MS_1250; break; 149cdf0e10cSrcweir #else 150cdf0e10cSrcweir nEncoding = RTL_TEXTENCODING_ISO_8859_2; break; 151cdf0e10cSrcweir #endif 152cdf0e10cSrcweir case LANGUAGE_RUSSIAN : 153cdf0e10cSrcweir #ifdef WNT 154cdf0e10cSrcweir nEncoding = RTL_TEXTENCODING_MS_1251; break; 155cdf0e10cSrcweir #else 156cdf0e10cSrcweir nEncoding = RTL_TEXTENCODING_ISO_8859_5; break; 157cdf0e10cSrcweir #endif 158cdf0e10cSrcweir case LANGUAGE_GREEK : 159cdf0e10cSrcweir #ifdef WNT 160cdf0e10cSrcweir nEncoding = RTL_TEXTENCODING_MS_1253; break; 161cdf0e10cSrcweir #else 162cdf0e10cSrcweir nEncoding = RTL_TEXTENCODING_ISO_8859_7; break; 163cdf0e10cSrcweir #endif 164cdf0e10cSrcweir default: 165cdf0e10cSrcweir DBG_ASSERT( 0, "unexpected language" ); 166cdf0e10cSrcweir } 167cdf0e10cSrcweir } 168cdf0e10cSrcweir 169cdf0e10cSrcweir return nEncoding; 170cdf0e10cSrcweir } 171cdf0e10cSrcweir 172cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 173cdf0e10cSrcweir 174cdf0e10cSrcweir static inline sal_Int32 Minimum( sal_Int32 n1, sal_Int32 n2, sal_Int32 n3 ) 175cdf0e10cSrcweir { 176cdf0e10cSrcweir sal_Int32 nMin = n1 < n2 ? n1 : n2; 177cdf0e10cSrcweir return nMin < n3 ? nMin : n3; 178cdf0e10cSrcweir } 179cdf0e10cSrcweir 180cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 181cdf0e10cSrcweir 182cdf0e10cSrcweir class IntArray2D 183cdf0e10cSrcweir { 184cdf0e10cSrcweir private: 185cdf0e10cSrcweir sal_Int32 *pData; 186cdf0e10cSrcweir int n1, n2; 187cdf0e10cSrcweir 188cdf0e10cSrcweir public: 189cdf0e10cSrcweir IntArray2D( int nDim1, int nDim2 ); 190cdf0e10cSrcweir ~IntArray2D(); 191cdf0e10cSrcweir 192cdf0e10cSrcweir sal_Int32 & Value( int i, int k ); 193cdf0e10cSrcweir }; 194cdf0e10cSrcweir 195cdf0e10cSrcweir IntArray2D::IntArray2D( int nDim1, int nDim2 ) 196cdf0e10cSrcweir { 197cdf0e10cSrcweir n1 = nDim1; 198cdf0e10cSrcweir n2 = nDim2; 199cdf0e10cSrcweir pData = new sal_Int32[n1 * n2]; 200cdf0e10cSrcweir } 201cdf0e10cSrcweir 202cdf0e10cSrcweir IntArray2D::~IntArray2D() 203cdf0e10cSrcweir { 204cdf0e10cSrcweir delete[] pData; 205cdf0e10cSrcweir } 206cdf0e10cSrcweir 207cdf0e10cSrcweir sal_Int32 & IntArray2D::Value( int i, int k ) 208cdf0e10cSrcweir { 209cdf0e10cSrcweir DBG_ASSERT( 0 <= i && i < n1, "first index out of range" ); 210cdf0e10cSrcweir DBG_ASSERT( 0 <= k && k < n2, "first index out of range" ); 211cdf0e10cSrcweir DBG_ASSERT( i * n2 + k < n1 * n2, "index out of range" ); 212cdf0e10cSrcweir return pData[ i * n2 + k ]; 213cdf0e10cSrcweir } 214cdf0e10cSrcweir 215cdf0e10cSrcweir 216cdf0e10cSrcweir sal_Int32 LevDistance( const OUString &rTxt1, const OUString &rTxt2 ) 217cdf0e10cSrcweir { 218cdf0e10cSrcweir sal_Int32 nLen1 = rTxt1.getLength(); 219cdf0e10cSrcweir sal_Int32 nLen2 = rTxt2.getLength(); 220cdf0e10cSrcweir 221cdf0e10cSrcweir if (nLen1 == 0) 222cdf0e10cSrcweir return nLen2; 223cdf0e10cSrcweir if (nLen2 == 0) 224cdf0e10cSrcweir return nLen1; 225cdf0e10cSrcweir 226cdf0e10cSrcweir IntArray2D aData( nLen1 + 1, nLen2 + 1 ); 227cdf0e10cSrcweir 228cdf0e10cSrcweir sal_Int32 i, k; 229cdf0e10cSrcweir for (i = 0; i <= nLen1; ++i) 230cdf0e10cSrcweir aData.Value(i, 0) = i; 231cdf0e10cSrcweir for (k = 0; k <= nLen2; ++k) 232cdf0e10cSrcweir aData.Value(0, k) = k; 233cdf0e10cSrcweir for (i = 1; i <= nLen1; ++i) 234cdf0e10cSrcweir { 235cdf0e10cSrcweir for (k = 1; k <= nLen2; ++k) 236cdf0e10cSrcweir { 237cdf0e10cSrcweir sal_Unicode c1i = rTxt1.getStr()[i - 1]; 238cdf0e10cSrcweir sal_Unicode c2k = rTxt2.getStr()[k - 1]; 239cdf0e10cSrcweir sal_Int32 nCost = c1i == c2k ? 0 : 1; 240cdf0e10cSrcweir sal_Int32 nNew = Minimum( aData.Value(i-1, k ) + 1, 241cdf0e10cSrcweir aData.Value(i , k-1) + 1, 242cdf0e10cSrcweir aData.Value(i-1, k-1) + nCost ); 243cdf0e10cSrcweir // take transposition (exchange with left or right char) in account 244cdf0e10cSrcweir if (2 < i && 2 < k) 245cdf0e10cSrcweir { 246cdf0e10cSrcweir int nT = aData.Value(i-2, k-2) + 1; 247cdf0e10cSrcweir if (rTxt1.getStr()[i - 2] != c1i) 248cdf0e10cSrcweir ++nT; 249cdf0e10cSrcweir if (rTxt2.getStr()[k - 2] != c2k) 250cdf0e10cSrcweir ++nT; 251cdf0e10cSrcweir if (nT < nNew) 252cdf0e10cSrcweir nNew = nT; 253cdf0e10cSrcweir } 254cdf0e10cSrcweir 255cdf0e10cSrcweir aData.Value(i, k) = nNew; 256cdf0e10cSrcweir } 257cdf0e10cSrcweir } 258cdf0e10cSrcweir sal_Int32 nDist = aData.Value(nLen1, nLen2); 259cdf0e10cSrcweir return nDist; 260cdf0e10cSrcweir } 261cdf0e10cSrcweir 262cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 263cdf0e10cSrcweir 264cdf0e10cSrcweir sal_Bool IsUseDicList( const PropertyValues &rProperties, 265cdf0e10cSrcweir const uno::Reference< XPropertySet > &rxProp ) 266cdf0e10cSrcweir { 267cdf0e10cSrcweir sal_Bool bRes = sal_True; 268cdf0e10cSrcweir 269cdf0e10cSrcweir sal_Int32 nLen = rProperties.getLength(); 270cdf0e10cSrcweir const PropertyValue *pVal = rProperties.getConstArray(); 271cdf0e10cSrcweir sal_Int32 i; 272cdf0e10cSrcweir 273cdf0e10cSrcweir for ( i = 0; i < nLen; ++i) 274cdf0e10cSrcweir { 275cdf0e10cSrcweir if (UPH_IS_USE_DICTIONARY_LIST == pVal[i].Handle) 276cdf0e10cSrcweir { 277cdf0e10cSrcweir pVal[i].Value >>= bRes; 278cdf0e10cSrcweir break; 279cdf0e10cSrcweir } 280cdf0e10cSrcweir } 281cdf0e10cSrcweir if (i >= nLen) // no temporary value found in 'rProperties' 282cdf0e10cSrcweir { 283cdf0e10cSrcweir uno::Reference< XFastPropertySet > xFast( rxProp, UNO_QUERY ); 284cdf0e10cSrcweir if (xFast.is()) 285cdf0e10cSrcweir xFast->getFastPropertyValue( UPH_IS_USE_DICTIONARY_LIST ) >>= bRes; 286cdf0e10cSrcweir } 287cdf0e10cSrcweir 288cdf0e10cSrcweir return bRes; 289cdf0e10cSrcweir } 290cdf0e10cSrcweir 291cdf0e10cSrcweir 292cdf0e10cSrcweir sal_Bool IsIgnoreControlChars( const PropertyValues &rProperties, 293cdf0e10cSrcweir const uno::Reference< XPropertySet > &rxProp ) 294cdf0e10cSrcweir { 295cdf0e10cSrcweir sal_Bool bRes = sal_True; 296cdf0e10cSrcweir 297cdf0e10cSrcweir sal_Int32 nLen = rProperties.getLength(); 298cdf0e10cSrcweir const PropertyValue *pVal = rProperties.getConstArray(); 299cdf0e10cSrcweir sal_Int32 i; 300cdf0e10cSrcweir 301cdf0e10cSrcweir for ( i = 0; i < nLen; ++i) 302cdf0e10cSrcweir { 303cdf0e10cSrcweir if (UPH_IS_IGNORE_CONTROL_CHARACTERS == pVal[i].Handle) 304cdf0e10cSrcweir { 305cdf0e10cSrcweir pVal[i].Value >>= bRes; 306cdf0e10cSrcweir break; 307cdf0e10cSrcweir } 308cdf0e10cSrcweir } 309cdf0e10cSrcweir if (i >= nLen) // no temporary value found in 'rProperties' 310cdf0e10cSrcweir { 311cdf0e10cSrcweir uno::Reference< XFastPropertySet > xFast( rxProp, UNO_QUERY ); 312cdf0e10cSrcweir if (xFast.is()) 313cdf0e10cSrcweir xFast->getFastPropertyValue( UPH_IS_IGNORE_CONTROL_CHARACTERS ) >>= bRes; 314cdf0e10cSrcweir } 315cdf0e10cSrcweir 316cdf0e10cSrcweir return bRes; 317cdf0e10cSrcweir } 318cdf0e10cSrcweir 319cdf0e10cSrcweir 320cdf0e10cSrcweir static sal_Bool lcl_HasHyphInfo( const uno::Reference<XDictionaryEntry> &xEntry ) 321cdf0e10cSrcweir { 322cdf0e10cSrcweir sal_Bool bRes = sal_False; 323cdf0e10cSrcweir if (xEntry.is()) 324cdf0e10cSrcweir { 325cdf0e10cSrcweir // there has to be (at least one) '=' denoting a hyphenation position 326cdf0e10cSrcweir // and it must not be before any character of the word 327cdf0e10cSrcweir sal_Int32 nIdx = xEntry->getDictionaryWord().indexOf( '=' ); 328cdf0e10cSrcweir bRes = nIdx != -1 && nIdx != 0; 329cdf0e10cSrcweir } 330cdf0e10cSrcweir return bRes; 331cdf0e10cSrcweir } 332cdf0e10cSrcweir 333cdf0e10cSrcweir 334cdf0e10cSrcweir uno::Reference< XDictionaryEntry > SearchDicList( 335cdf0e10cSrcweir const uno::Reference< XDictionaryList > &xDicList, 336cdf0e10cSrcweir const OUString &rWord, sal_Int16 nLanguage, 337cdf0e10cSrcweir sal_Bool bSearchPosDics, sal_Bool bSearchSpellEntry ) 338cdf0e10cSrcweir { 339cdf0e10cSrcweir MutexGuard aGuard( GetLinguMutex() ); 340cdf0e10cSrcweir 341cdf0e10cSrcweir uno::Reference< XDictionaryEntry > xEntry; 342cdf0e10cSrcweir 343cdf0e10cSrcweir if (!xDicList.is()) 344cdf0e10cSrcweir return xEntry; 345cdf0e10cSrcweir 346cdf0e10cSrcweir const uno::Sequence< uno::Reference< XDictionary > > 347cdf0e10cSrcweir aDics( xDicList->getDictionaries() ); 348cdf0e10cSrcweir const uno::Reference< XDictionary > 349cdf0e10cSrcweir *pDic = aDics.getConstArray(); 350cdf0e10cSrcweir sal_Int32 nDics = xDicList->getCount(); 351cdf0e10cSrcweir 352cdf0e10cSrcweir sal_Int32 i; 353cdf0e10cSrcweir for (i = 0; i < nDics; i++) 354cdf0e10cSrcweir { 355cdf0e10cSrcweir uno::Reference< XDictionary > axDic( pDic[i], UNO_QUERY ); 356cdf0e10cSrcweir 357cdf0e10cSrcweir DictionaryType eType = axDic->getDictionaryType(); 358cdf0e10cSrcweir sal_Int16 nLang = LocaleToLanguage( axDic->getLocale() ); 359cdf0e10cSrcweir 360cdf0e10cSrcweir if ( axDic.is() && axDic->isActive() 361cdf0e10cSrcweir && (nLang == nLanguage || nLang == LANGUAGE_NONE) ) 362cdf0e10cSrcweir { 363cdf0e10cSrcweir DBG_ASSERT( eType != DictionaryType_MIXED, 364cdf0e10cSrcweir "lng : unexpected dictionary type" ); 365cdf0e10cSrcweir 366cdf0e10cSrcweir if ( (!bSearchPosDics && eType == DictionaryType_NEGATIVE) 367cdf0e10cSrcweir || ( bSearchPosDics && eType == DictionaryType_POSITIVE)) 368cdf0e10cSrcweir { 369cdf0e10cSrcweir if ( (xEntry = axDic->getEntry( rWord )).is() ) 370cdf0e10cSrcweir { 371cdf0e10cSrcweir if (bSearchSpellEntry || lcl_HasHyphInfo( xEntry )) 372cdf0e10cSrcweir break; 373cdf0e10cSrcweir } 374cdf0e10cSrcweir xEntry = 0; 375cdf0e10cSrcweir } 376cdf0e10cSrcweir } 377cdf0e10cSrcweir } 378cdf0e10cSrcweir 379cdf0e10cSrcweir return xEntry; 380cdf0e10cSrcweir } 381cdf0e10cSrcweir 382cdf0e10cSrcweir 383cdf0e10cSrcweir sal_Bool SaveDictionaries( const uno::Reference< XDictionaryList > &xDicList ) 384cdf0e10cSrcweir { 385cdf0e10cSrcweir if (!xDicList.is()) 386cdf0e10cSrcweir return sal_True; 387cdf0e10cSrcweir 388cdf0e10cSrcweir sal_Bool bRet = sal_True; 389cdf0e10cSrcweir 390cdf0e10cSrcweir Sequence< uno::Reference< XDictionary > > aDics( xDicList->getDictionaries() ); 391cdf0e10cSrcweir const uno::Reference< XDictionary > *pDic = aDics.getConstArray(); 392cdf0e10cSrcweir sal_Int32 nCount = aDics.getLength(); 393cdf0e10cSrcweir for (sal_Int32 i = 0; i < nCount; i++) 394cdf0e10cSrcweir { 395cdf0e10cSrcweir try 396cdf0e10cSrcweir { 397cdf0e10cSrcweir uno::Reference< frame::XStorable > xStor( pDic[i], UNO_QUERY ); 398cdf0e10cSrcweir if (xStor.is()) 399cdf0e10cSrcweir { 400cdf0e10cSrcweir if (!xStor->isReadonly() && xStor->hasLocation()) 401cdf0e10cSrcweir xStor->store(); 402cdf0e10cSrcweir } 403cdf0e10cSrcweir } 404cdf0e10cSrcweir catch(uno::Exception &) 405cdf0e10cSrcweir { 406cdf0e10cSrcweir bRet = sal_False; 407cdf0e10cSrcweir } 408cdf0e10cSrcweir } 409cdf0e10cSrcweir 410cdf0e10cSrcweir return bRet; 411cdf0e10cSrcweir } 412cdf0e10cSrcweir 413cdf0e10cSrcweir 414cdf0e10cSrcweir sal_uInt8 AddEntryToDic( 415cdf0e10cSrcweir uno::Reference< XDictionary > &rxDic, 416cdf0e10cSrcweir const OUString &rWord, sal_Bool bIsNeg, 417cdf0e10cSrcweir const OUString &rRplcTxt, sal_Int16 /* nRplcLang */, 418cdf0e10cSrcweir sal_Bool bStripDot ) 419cdf0e10cSrcweir { 420cdf0e10cSrcweir if (!rxDic.is()) 421cdf0e10cSrcweir return DIC_ERR_NOT_EXISTS; 422cdf0e10cSrcweir 423cdf0e10cSrcweir OUString aTmp( rWord ); 424cdf0e10cSrcweir if (bStripDot) 425cdf0e10cSrcweir { 426cdf0e10cSrcweir sal_Int32 nLen = rWord.getLength(); 427cdf0e10cSrcweir if (nLen > 0 && '.' == rWord[ nLen - 1]) 428cdf0e10cSrcweir { 429cdf0e10cSrcweir // remove trailing '.' 430cdf0e10cSrcweir // (this is the official way to do this :-( ) 431cdf0e10cSrcweir aTmp = aTmp.copy( 0, nLen - 1 ); 432cdf0e10cSrcweir } 433cdf0e10cSrcweir } 434cdf0e10cSrcweir sal_Bool bAddOk = rxDic->add( aTmp, bIsNeg, rRplcTxt ); 435cdf0e10cSrcweir 436cdf0e10cSrcweir sal_uInt8 nRes = DIC_ERR_NONE; 437cdf0e10cSrcweir if (!bAddOk) 438cdf0e10cSrcweir { 439cdf0e10cSrcweir if (rxDic->isFull()) 440cdf0e10cSrcweir nRes = DIC_ERR_FULL; 441cdf0e10cSrcweir else 442cdf0e10cSrcweir { 443cdf0e10cSrcweir uno::Reference< frame::XStorable > xStor( rxDic, UNO_QUERY ); 444cdf0e10cSrcweir if (xStor.is() && xStor->isReadonly()) 445cdf0e10cSrcweir nRes = DIC_ERR_READONLY; 446cdf0e10cSrcweir else 447cdf0e10cSrcweir nRes = DIC_ERR_UNKNOWN; 448cdf0e10cSrcweir } 449cdf0e10cSrcweir } 450cdf0e10cSrcweir 451cdf0e10cSrcweir return nRes; 452cdf0e10cSrcweir } 453cdf0e10cSrcweir 454cdf0e10cSrcweir 455cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 456cdf0e10cSrcweir 457cdf0e10cSrcweir LanguageType LocaleToLanguage( const Locale& rLocale ) 458cdf0e10cSrcweir { 459cdf0e10cSrcweir // empty Locale -> LANGUAGE_NONE 460cdf0e10cSrcweir if ( rLocale.Language.getLength() == 0 ) 461cdf0e10cSrcweir return LANGUAGE_NONE; 462cdf0e10cSrcweir 463cdf0e10cSrcweir return MsLangId::convertLocaleToLanguage( rLocale ); 464cdf0e10cSrcweir } 465cdf0e10cSrcweir 466cdf0e10cSrcweir 467cdf0e10cSrcweir Locale& LanguageToLocale( Locale& rLocale, LanguageType eLang ) 468cdf0e10cSrcweir { 469cdf0e10cSrcweir if ( eLang != LANGUAGE_NONE /* && eLang != LANGUAGE_SYSTEM */) 470cdf0e10cSrcweir MsLangId::convertLanguageToLocale( eLang, rLocale ); 471cdf0e10cSrcweir 472cdf0e10cSrcweir return rLocale; 473cdf0e10cSrcweir } 474cdf0e10cSrcweir 475cdf0e10cSrcweir Locale CreateLocale( LanguageType eLang ) 476cdf0e10cSrcweir { 477cdf0e10cSrcweir Locale aLocale; 478cdf0e10cSrcweir if ( eLang != LANGUAGE_NONE /* && eLang != LANGUAGE_SYSTEM */) 479cdf0e10cSrcweir return MsLangId::convertLanguageToLocale( eLang ); 480cdf0e10cSrcweir 481cdf0e10cSrcweir return aLocale; 482cdf0e10cSrcweir } 483cdf0e10cSrcweir 484cdf0e10cSrcweir uno::Sequence< Locale > LangSeqToLocaleSeq( const uno::Sequence< sal_Int16 > &rLangSeq ) 485cdf0e10cSrcweir { 486cdf0e10cSrcweir const sal_Int16 *pLang = rLangSeq.getConstArray(); 487cdf0e10cSrcweir sal_Int32 nCount = rLangSeq.getLength(); 488cdf0e10cSrcweir 489cdf0e10cSrcweir uno::Sequence< Locale > aLocales( nCount ); 490cdf0e10cSrcweir Locale *pLocale = aLocales.getArray(); 491cdf0e10cSrcweir for (sal_Int32 i = 0; i < nCount; ++i) 492cdf0e10cSrcweir { 493cdf0e10cSrcweir LanguageToLocale( pLocale[i], pLang[ i ] ); 494cdf0e10cSrcweir } 495cdf0e10cSrcweir 496cdf0e10cSrcweir return aLocales; 497cdf0e10cSrcweir } 498cdf0e10cSrcweir 499cdf0e10cSrcweir uno::Sequence< sal_Int16 > 500cdf0e10cSrcweir LocaleSeqToLangSeq( uno::Sequence< Locale > &rLocaleSeq ) 501cdf0e10cSrcweir { 502cdf0e10cSrcweir const Locale *pLocale = rLocaleSeq.getConstArray(); 503cdf0e10cSrcweir sal_Int32 nCount = rLocaleSeq.getLength(); 504cdf0e10cSrcweir 505cdf0e10cSrcweir uno::Sequence< sal_Int16 > aLangs( nCount ); 506cdf0e10cSrcweir sal_Int16 *pLang = aLangs.getArray(); 507cdf0e10cSrcweir for (sal_Int32 i = 0; i < nCount; ++i) 508cdf0e10cSrcweir { 509cdf0e10cSrcweir pLang[i] = LocaleToLanguage( pLocale[i] ); 510cdf0e10cSrcweir } 511cdf0e10cSrcweir 512cdf0e10cSrcweir return aLangs; 513cdf0e10cSrcweir } 514cdf0e10cSrcweir 515cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 516cdf0e10cSrcweir 517cdf0e10cSrcweir sal_Bool IsReadOnly( const String &rURL, sal_Bool *pbExist ) 518cdf0e10cSrcweir { 519cdf0e10cSrcweir sal_Bool bRes = sal_False; 520cdf0e10cSrcweir sal_Bool bExists = sal_False; 521cdf0e10cSrcweir 522cdf0e10cSrcweir if (rURL.Len() > 0) 523cdf0e10cSrcweir { 524cdf0e10cSrcweir try 525cdf0e10cSrcweir { 526cdf0e10cSrcweir uno::Reference< ::com::sun::star::ucb::XCommandEnvironment > xCmdEnv; 527cdf0e10cSrcweir ::ucbhelper::Content aContent( rURL, xCmdEnv ); 528cdf0e10cSrcweir 529cdf0e10cSrcweir bExists = aContent.isDocument(); 530cdf0e10cSrcweir if (bExists) 531cdf0e10cSrcweir { 532cdf0e10cSrcweir Any aAny( aContent.getPropertyValue( A2OU( "IsReadOnly" ) ) ); 533cdf0e10cSrcweir aAny >>= bRes; 534cdf0e10cSrcweir } 535cdf0e10cSrcweir } 536cdf0e10cSrcweir catch (Exception &) 537cdf0e10cSrcweir { 538cdf0e10cSrcweir bRes = sal_True; 539cdf0e10cSrcweir } 540cdf0e10cSrcweir } 541cdf0e10cSrcweir 542cdf0e10cSrcweir if (pbExist) 543cdf0e10cSrcweir *pbExist = bExists; 544cdf0e10cSrcweir return bRes; 545cdf0e10cSrcweir } 546cdf0e10cSrcweir 547cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 548cdf0e10cSrcweir 549cdf0e10cSrcweir 550cdf0e10cSrcweir static sal_Bool GetAltSpelling( sal_Int16 &rnChgPos, sal_Int16 &rnChgLen, OUString &rRplc, 551cdf0e10cSrcweir uno::Reference< XHyphenatedWord > &rxHyphWord ) 552cdf0e10cSrcweir { 553cdf0e10cSrcweir sal_Bool bRes = rxHyphWord->isAlternativeSpelling(); 554cdf0e10cSrcweir if (bRes) 555cdf0e10cSrcweir { 556cdf0e10cSrcweir OUString aWord( rxHyphWord->getWord() ), 557cdf0e10cSrcweir aHyphenatedWord( rxHyphWord->getHyphenatedWord() ); 558cdf0e10cSrcweir sal_Int16 nHyphenationPos = rxHyphWord->getHyphenationPos(); 559cdf0e10cSrcweir /*sal_Int16 nHyphenPos = rxHyphWord->getHyphenPos()*/; 560cdf0e10cSrcweir const sal_Unicode *pWord = aWord.getStr(), 561cdf0e10cSrcweir *pAltWord = aHyphenatedWord.getStr(); 562cdf0e10cSrcweir 563cdf0e10cSrcweir // at least char changes directly left or right to the hyphen 564cdf0e10cSrcweir // should(!) be handled properly... 565cdf0e10cSrcweir //! nHyphenationPos and nHyphenPos differ at most by 1 (see above) 566cdf0e10cSrcweir //! Beware: eg "Schiffahrt" in German (pre spelling reform) 567cdf0e10cSrcweir //! proves to be a bit nasty (nChgPosLeft and nChgPosRight overlap 568cdf0e10cSrcweir //! to an extend.) 569cdf0e10cSrcweir 570cdf0e10cSrcweir // find first different char from left 571cdf0e10cSrcweir sal_Int32 nPosL = 0, 572cdf0e10cSrcweir nAltPosL = 0; 573cdf0e10cSrcweir for (sal_Int16 i = 0 ; pWord[ nPosL ] == pAltWord[ nAltPosL ]; nPosL++, nAltPosL++, i++) 574cdf0e10cSrcweir { 575cdf0e10cSrcweir // restrict changes area beginning to the right to 576cdf0e10cSrcweir // the char immediately following the hyphen. 577cdf0e10cSrcweir //! serves to insert the additional "f" in "Schiffahrt" at 578cdf0e10cSrcweir //! position 5 rather than position 6. 579cdf0e10cSrcweir if (i >= nHyphenationPos + 1) 580cdf0e10cSrcweir break; 581cdf0e10cSrcweir } 582cdf0e10cSrcweir 583cdf0e10cSrcweir // find first different char from right 584cdf0e10cSrcweir sal_Int32 nPosR = aWord.getLength() - 1, 585cdf0e10cSrcweir nAltPosR = aHyphenatedWord.getLength() - 1; 586cdf0e10cSrcweir for ( ; nPosR >= nPosL && nAltPosR >= nAltPosL 587cdf0e10cSrcweir && pWord[ nPosR ] == pAltWord[ nAltPosR ]; 588cdf0e10cSrcweir nPosR--, nAltPosR--) 589cdf0e10cSrcweir ; 590cdf0e10cSrcweir 591cdf0e10cSrcweir rnChgPos = sal::static_int_cast< sal_Int16 >(nPosL); 592cdf0e10cSrcweir rnChgLen = sal::static_int_cast< sal_Int16 >(nPosR - nPosL + 1); 593cdf0e10cSrcweir DBG_ASSERT( rnChgLen >= 0, "nChgLen < 0"); 594cdf0e10cSrcweir 595cdf0e10cSrcweir sal_Int32 nTxtStart = nPosL; 596cdf0e10cSrcweir sal_Int32 nTxtLen = nAltPosL - nPosL + 1; 597cdf0e10cSrcweir rRplc = aHyphenatedWord.copy( nTxtStart, nTxtLen ); 598cdf0e10cSrcweir } 599cdf0e10cSrcweir return bRes; 600cdf0e10cSrcweir } 601cdf0e10cSrcweir 602cdf0e10cSrcweir 603cdf0e10cSrcweir static sal_Int16 GetOrigWordPos( const OUString &rOrigWord, sal_Int16 nPos ) 604cdf0e10cSrcweir { 605cdf0e10cSrcweir sal_Int32 nLen = rOrigWord.getLength(); 606cdf0e10cSrcweir sal_Int32 i = -1; 607cdf0e10cSrcweir while (nPos >= 0 && i++ < nLen) 608cdf0e10cSrcweir { 609cdf0e10cSrcweir sal_Unicode cChar = rOrigWord[i]; 610cdf0e10cSrcweir sal_Bool bSkip = IsHyphen( cChar ) || IsControlChar( cChar ); 611cdf0e10cSrcweir if (!bSkip) 612cdf0e10cSrcweir --nPos; 613cdf0e10cSrcweir } 614cdf0e10cSrcweir return sal::static_int_cast< sal_Int16 >((0 <= i && i < nLen) ? i : -1); 615cdf0e10cSrcweir } 616cdf0e10cSrcweir 617cdf0e10cSrcweir 618cdf0e10cSrcweir sal_Int32 GetPosInWordToCheck( const OUString &rTxt, sal_Int32 nPos ) 619cdf0e10cSrcweir { 620cdf0e10cSrcweir sal_Int32 nRes = -1; 621cdf0e10cSrcweir sal_Int32 nLen = rTxt.getLength(); 622cdf0e10cSrcweir if (0 <= nPos && nPos < nLen) 623cdf0e10cSrcweir { 624cdf0e10cSrcweir nRes = 0; 625cdf0e10cSrcweir for (sal_Int32 i = 0; i < nPos; ++i) 626cdf0e10cSrcweir { 627cdf0e10cSrcweir sal_Unicode cChar = rTxt[i]; 628cdf0e10cSrcweir sal_Bool bSkip = IsHyphen( cChar ) || IsControlChar( cChar ); 629cdf0e10cSrcweir if (!bSkip) 630cdf0e10cSrcweir ++nRes; 631cdf0e10cSrcweir } 632cdf0e10cSrcweir } 633cdf0e10cSrcweir return nRes; 634cdf0e10cSrcweir } 635cdf0e10cSrcweir 636cdf0e10cSrcweir 637cdf0e10cSrcweir uno::Reference< XHyphenatedWord > RebuildHyphensAndControlChars( 638cdf0e10cSrcweir const OUString &rOrigWord, 639cdf0e10cSrcweir uno::Reference< XHyphenatedWord > &rxHyphWord ) 640cdf0e10cSrcweir { 641cdf0e10cSrcweir uno::Reference< XHyphenatedWord > xRes; 642cdf0e10cSrcweir if (rOrigWord.getLength() && rxHyphWord.is()) 643cdf0e10cSrcweir { 644cdf0e10cSrcweir sal_Int16 nChgPos = 0, 645cdf0e10cSrcweir nChgLen = 0; 646cdf0e10cSrcweir OUString aRplc; 647cdf0e10cSrcweir sal_Bool bAltSpelling = GetAltSpelling( nChgPos, nChgLen, aRplc, rxHyphWord ); 648cdf0e10cSrcweir #if OSL_DEBUG_LEVEL > 1 649cdf0e10cSrcweir OUString aWord( rxHyphWord->getWord() ); 650cdf0e10cSrcweir #endif 651cdf0e10cSrcweir 652cdf0e10cSrcweir OUString aOrigHyphenatedWord; 653cdf0e10cSrcweir sal_Int16 nOrigHyphenPos = -1; 654cdf0e10cSrcweir sal_Int16 nOrigHyphenationPos = -1; 655cdf0e10cSrcweir if (!bAltSpelling) 656cdf0e10cSrcweir { 657cdf0e10cSrcweir aOrigHyphenatedWord = rOrigWord; 658cdf0e10cSrcweir nOrigHyphenPos = GetOrigWordPos( rOrigWord, rxHyphWord->getHyphenPos() ); 659cdf0e10cSrcweir nOrigHyphenationPos = GetOrigWordPos( rOrigWord, rxHyphWord->getHyphenationPos() ); 660cdf0e10cSrcweir } 661cdf0e10cSrcweir else 662cdf0e10cSrcweir { 663cdf0e10cSrcweir //! should at least work with the German words 664cdf0e10cSrcweir //! B�-c-k-er and Sc-hif-fah-rt 665cdf0e10cSrcweir 666cdf0e10cSrcweir OUString aLeft, aRight; 667cdf0e10cSrcweir sal_Int16 nPos = GetOrigWordPos( rOrigWord, nChgPos ); 668cdf0e10cSrcweir 669cdf0e10cSrcweir // get words like Sc-hif-fah-rt to work correct 670cdf0e10cSrcweir sal_Int16 nHyphenationPos = rxHyphWord->getHyphenationPos(); 671cdf0e10cSrcweir if (nChgPos > nHyphenationPos) 672cdf0e10cSrcweir --nPos; 673cdf0e10cSrcweir 674cdf0e10cSrcweir aLeft = rOrigWord.copy( 0, nPos ); 675cdf0e10cSrcweir aRight = rOrigWord.copy( nPos + nChgLen ); 676cdf0e10cSrcweir 677cdf0e10cSrcweir aOrigHyphenatedWord = aLeft; 678cdf0e10cSrcweir aOrigHyphenatedWord += aRplc; 679cdf0e10cSrcweir aOrigHyphenatedWord += aRight; 680cdf0e10cSrcweir 681cdf0e10cSrcweir nOrigHyphenPos = sal::static_int_cast< sal_Int16 >(aLeft.getLength() + 682cdf0e10cSrcweir rxHyphWord->getHyphenPos() - nChgPos); 683cdf0e10cSrcweir nOrigHyphenationPos = GetOrigWordPos( rOrigWord, nHyphenationPos ); 684cdf0e10cSrcweir } 685cdf0e10cSrcweir 686cdf0e10cSrcweir if (nOrigHyphenPos == -1 || nOrigHyphenationPos == -1) 687cdf0e10cSrcweir { 688cdf0e10cSrcweir DBG_ASSERT( 0, "failed to get nOrigHyphenPos or nOrigHyphenationPos" ); 689cdf0e10cSrcweir } 690cdf0e10cSrcweir else 691cdf0e10cSrcweir { 692cdf0e10cSrcweir sal_Int16 nLang = LocaleToLanguage( rxHyphWord->getLocale() ); 693cdf0e10cSrcweir xRes = new HyphenatedWord( 694cdf0e10cSrcweir rOrigWord, nLang, nOrigHyphenationPos, 695cdf0e10cSrcweir aOrigHyphenatedWord, nOrigHyphenPos ); 696cdf0e10cSrcweir } 697cdf0e10cSrcweir 698cdf0e10cSrcweir } 699cdf0e10cSrcweir return xRes; 700cdf0e10cSrcweir } 701cdf0e10cSrcweir 702cdf0e10cSrcweir 703cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 704cdf0e10cSrcweir 705cdf0e10cSrcweir 706cdf0e10cSrcweir static CharClass & lcl_GetCharClass() 707cdf0e10cSrcweir { 708cdf0e10cSrcweir static CharClass aCC( CreateLocale( LANGUAGE_ENGLISH_US ) ); 709cdf0e10cSrcweir return aCC; 710cdf0e10cSrcweir } 711cdf0e10cSrcweir 712cdf0e10cSrcweir 713cdf0e10cSrcweir osl::Mutex & lcl_GetCharClassMutex() 714cdf0e10cSrcweir { 715cdf0e10cSrcweir static osl::Mutex aMutex; 716cdf0e10cSrcweir return aMutex; 717cdf0e10cSrcweir } 718cdf0e10cSrcweir 719cdf0e10cSrcweir 720cdf0e10cSrcweir sal_Bool IsUpper( const String &rText, xub_StrLen nPos, xub_StrLen nLen, sal_Int16 nLanguage ) 721cdf0e10cSrcweir { 722cdf0e10cSrcweir MutexGuard aGuard( lcl_GetCharClassMutex() ); 723cdf0e10cSrcweir 724cdf0e10cSrcweir CharClass &rCC = lcl_GetCharClass(); 725cdf0e10cSrcweir rCC.setLocale( CreateLocale( nLanguage ) ); 726cdf0e10cSrcweir sal_Int32 nFlags = rCC.getStringType( rText, nPos, nLen ); 727cdf0e10cSrcweir return (nFlags & KCharacterType::UPPER) 728cdf0e10cSrcweir && !(nFlags & KCharacterType::LOWER); 729cdf0e10cSrcweir } 730cdf0e10cSrcweir 731cdf0e10cSrcweir 732cdf0e10cSrcweir sal_Bool IsLower( const String &rText, xub_StrLen nPos, xub_StrLen nLen, sal_Int16 nLanguage ) 733cdf0e10cSrcweir { 734cdf0e10cSrcweir MutexGuard aGuard( lcl_GetCharClassMutex() ); 735cdf0e10cSrcweir 736cdf0e10cSrcweir CharClass &rCC = lcl_GetCharClass(); 737cdf0e10cSrcweir rCC.setLocale( CreateLocale( nLanguage ) ); 738cdf0e10cSrcweir sal_Int32 nFlags = rCC.getStringType( rText, nPos, nLen ); 739cdf0e10cSrcweir return (nFlags & KCharacterType::LOWER) 740cdf0e10cSrcweir && !(nFlags & KCharacterType::UPPER); 741cdf0e10cSrcweir } 742cdf0e10cSrcweir 743cdf0e10cSrcweir 744cdf0e10cSrcweir String ToLower( const String &rText, sal_Int16 nLanguage ) 745cdf0e10cSrcweir { 746cdf0e10cSrcweir MutexGuard aGuard( lcl_GetCharClassMutex() ); 747cdf0e10cSrcweir 748cdf0e10cSrcweir CharClass &rCC = lcl_GetCharClass(); 749cdf0e10cSrcweir rCC.setLocale( CreateLocale( nLanguage ) ); 750cdf0e10cSrcweir return rCC.lower( rText ); 751cdf0e10cSrcweir } 752cdf0e10cSrcweir 753cdf0e10cSrcweir 754cdf0e10cSrcweir String ToUpper( const String &rText, sal_Int16 nLanguage ) 755cdf0e10cSrcweir { 756cdf0e10cSrcweir MutexGuard aGuard( lcl_GetCharClassMutex() ); 757cdf0e10cSrcweir 758cdf0e10cSrcweir CharClass &rCC = lcl_GetCharClass(); 759cdf0e10cSrcweir rCC.setLocale( CreateLocale( nLanguage ) ); 760cdf0e10cSrcweir return rCC.upper( rText ); 761cdf0e10cSrcweir } 762cdf0e10cSrcweir 763cdf0e10cSrcweir 764cdf0e10cSrcweir String ToTitle( const String &rText, sal_Int16 nLanguage ) 765cdf0e10cSrcweir { 766cdf0e10cSrcweir MutexGuard aGuard( lcl_GetCharClassMutex() ); 767cdf0e10cSrcweir 768cdf0e10cSrcweir CharClass &rCC = lcl_GetCharClass(); 769cdf0e10cSrcweir rCC.setLocale( CreateLocale( nLanguage ) ); 770cdf0e10cSrcweir return rCC.toTitle( rText, 0, rText.Len() ); 771cdf0e10cSrcweir } 772cdf0e10cSrcweir 773cdf0e10cSrcweir 774cdf0e10cSrcweir sal_Unicode ToLower( const sal_Unicode cChar, sal_Int16 nLanguage ) 775cdf0e10cSrcweir { 776cdf0e10cSrcweir MutexGuard aGuard( lcl_GetCharClassMutex() ); 777cdf0e10cSrcweir 778cdf0e10cSrcweir CharClass &rCC = lcl_GetCharClass(); 779cdf0e10cSrcweir rCC.setLocale( CreateLocale( nLanguage ) ); 780cdf0e10cSrcweir return rCC.lower( cChar ).GetChar(0); 781cdf0e10cSrcweir } 782cdf0e10cSrcweir 783cdf0e10cSrcweir 784cdf0e10cSrcweir sal_Unicode ToUpper( const sal_Unicode cChar, sal_Int16 nLanguage ) 785cdf0e10cSrcweir { 786cdf0e10cSrcweir MutexGuard aGuard( lcl_GetCharClassMutex() ); 787cdf0e10cSrcweir 788cdf0e10cSrcweir CharClass &rCC = lcl_GetCharClass(); 789cdf0e10cSrcweir rCC.setLocale( CreateLocale( nLanguage ) ); 790cdf0e10cSrcweir return rCC.upper( cChar ).GetChar(0); 791cdf0e10cSrcweir } 792cdf0e10cSrcweir 793cdf0e10cSrcweir // sorted(!) array of unicode ranges for code points that are exclusively(!) used as numbers 794cdf0e10cSrcweir // and thus may NOT not be part of names or words like the Chinese/Japanese number characters 795cdf0e10cSrcweir static const sal_uInt32 the_aDigitZeroes [] = 796cdf0e10cSrcweir { 797cdf0e10cSrcweir 0x00000030, //0039 ; Decimal # Nd [10] DIGIT ZERO..DIGIT NINE 798cdf0e10cSrcweir 0x00000660, //0669 ; Decimal # Nd [10] ARABIC-INDIC DIGIT ZERO..ARABIC-INDIC DIGIT NINE 799cdf0e10cSrcweir 0x000006F0, //06F9 ; Decimal # Nd [10] EXTENDED ARABIC-INDIC DIGIT ZERO..EXTENDED ARABIC-INDIC DIGIT NINE 800cdf0e10cSrcweir 0x000007C0, //07C9 ; Decimal # Nd [10] NKO DIGIT ZERO..NKO DIGIT NINE 801cdf0e10cSrcweir 0x00000966, //096F ; Decimal # Nd [10] DEVANAGARI DIGIT ZERO..DEVANAGARI DIGIT NINE 802cdf0e10cSrcweir 0x000009E6, //09EF ; Decimal # Nd [10] BENGALI DIGIT ZERO..BENGALI DIGIT NINE 803cdf0e10cSrcweir 0x00000A66, //0A6F ; Decimal # Nd [10] GURMUKHI DIGIT ZERO..GURMUKHI DIGIT NINE 804cdf0e10cSrcweir 0x00000AE6, //0AEF ; Decimal # Nd [10] GUJARATI DIGIT ZERO..GUJARATI DIGIT NINE 805cdf0e10cSrcweir 0x00000B66, //0B6F ; Decimal # Nd [10] ORIYA DIGIT ZERO..ORIYA DIGIT NINE 806cdf0e10cSrcweir 0x00000BE6, //0BEF ; Decimal # Nd [10] TAMIL DIGIT ZERO..TAMIL DIGIT NINE 807cdf0e10cSrcweir 0x00000C66, //0C6F ; Decimal # Nd [10] TELUGU DIGIT ZERO..TELUGU DIGIT NINE 808cdf0e10cSrcweir 0x00000CE6, //0CEF ; Decimal # Nd [10] KANNADA DIGIT ZERO..KANNADA DIGIT NINE 809cdf0e10cSrcweir 0x00000D66, //0D6F ; Decimal # Nd [10] MALAYALAM DIGIT ZERO..MALAYALAM DIGIT NINE 810cdf0e10cSrcweir 0x00000E50, //0E59 ; Decimal # Nd [10] THAI DIGIT ZERO..THAI DIGIT NINE 811cdf0e10cSrcweir 0x00000ED0, //0ED9 ; Decimal # Nd [10] LAO DIGIT ZERO..LAO DIGIT NINE 812cdf0e10cSrcweir 0x00000F20, //0F29 ; Decimal # Nd [10] TIBETAN DIGIT ZERO..TIBETAN DIGIT NINE 813cdf0e10cSrcweir 0x00001040, //1049 ; Decimal # Nd [10] MYANMAR DIGIT ZERO..MYANMAR DIGIT NINE 814cdf0e10cSrcweir 0x00001090, //1099 ; Decimal # Nd [10] MYANMAR SHAN DIGIT ZERO..MYANMAR SHAN DIGIT NINE 815cdf0e10cSrcweir 0x000017E0, //17E9 ; Decimal # Nd [10] KHMER DIGIT ZERO..KHMER DIGIT NINE 816cdf0e10cSrcweir 0x00001810, //1819 ; Decimal # Nd [10] MONGOLIAN DIGIT ZERO..MONGOLIAN DIGIT NINE 817cdf0e10cSrcweir 0x00001946, //194F ; Decimal # Nd [10] LIMBU DIGIT ZERO..LIMBU DIGIT NINE 818cdf0e10cSrcweir 0x000019D0, //19D9 ; Decimal # Nd [10] NEW TAI LUE DIGIT ZERO..NEW TAI LUE DIGIT NINE 819cdf0e10cSrcweir 0x00001B50, //1B59 ; Decimal # Nd [10] BALINESE DIGIT ZERO..BALINESE DIGIT NINE 820cdf0e10cSrcweir 0x00001BB0, //1BB9 ; Decimal # Nd [10] SUNDANESE DIGIT ZERO..SUNDANESE DIGIT NINE 821cdf0e10cSrcweir 0x00001C40, //1C49 ; Decimal # Nd [10] LEPCHA DIGIT ZERO..LEPCHA DIGIT NINE 822cdf0e10cSrcweir 0x00001C50, //1C59 ; Decimal # Nd [10] OL CHIKI DIGIT ZERO..OL CHIKI DIGIT NINE 823cdf0e10cSrcweir 0x0000A620, //A629 ; Decimal # Nd [10] VAI DIGIT ZERO..VAI DIGIT NINE 824cdf0e10cSrcweir 0x0000A8D0, //A8D9 ; Decimal # Nd [10] SAURASHTRA DIGIT ZERO..SAURASHTRA DIGIT NINE 825cdf0e10cSrcweir 0x0000A900, //A909 ; Decimal # Nd [10] KAYAH LI DIGIT ZERO..KAYAH LI DIGIT NINE 826cdf0e10cSrcweir 0x0000AA50, //AA59 ; Decimal # Nd [10] CHAM DIGIT ZERO..CHAM DIGIT NINE 827cdf0e10cSrcweir 0x0000FF10, //FF19 ; Decimal # Nd [10] FULLWIDTH DIGIT ZERO..FULLWIDTH DIGIT NINE 828cdf0e10cSrcweir 0x000104A0, //104A9 ; Decimal # Nd [10] OSMANYA DIGIT ZERO..OSMANYA DIGIT NINE 829cdf0e10cSrcweir 0x0001D7CE //1D7FF ; Decimal # Nd [50] MATHEMATICAL BOLD DIGIT ZERO..MATHEMATICAL MONOSPACE DIGIT NINE 830cdf0e10cSrcweir }; 831cdf0e10cSrcweir 832cdf0e10cSrcweir sal_Bool HasDigits( const OUString &rText ) 833cdf0e10cSrcweir { 834cdf0e10cSrcweir static const int nNumDigitZeroes = sizeof(the_aDigitZeroes) / sizeof(the_aDigitZeroes[0]); 835cdf0e10cSrcweir const sal_Int32 nLen = rText.getLength(); 836cdf0e10cSrcweir 837cdf0e10cSrcweir sal_Int32 i = 0; 838cdf0e10cSrcweir while (i < nLen) // for all characters ... 839cdf0e10cSrcweir { 840cdf0e10cSrcweir const sal_uInt32 nCodePoint = rText.iterateCodePoints( &i ); // handle unicode surrogates correctly... 841cdf0e10cSrcweir for (int j = 0; j < nNumDigitZeroes; ++j) // ... check in all 0..9 ranges 842cdf0e10cSrcweir { 843cdf0e10cSrcweir sal_uInt32 nDigitZero = the_aDigitZeroes[ j ]; 844cdf0e10cSrcweir if (nDigitZero > nCodePoint) 845cdf0e10cSrcweir break; 846cdf0e10cSrcweir if (/*nDigitZero <= nCodePoint &&*/ nCodePoint <= nDigitZero + 9) 847cdf0e10cSrcweir return sal_True; 848cdf0e10cSrcweir } 849cdf0e10cSrcweir } 850cdf0e10cSrcweir return sal_False; 851cdf0e10cSrcweir } 852cdf0e10cSrcweir 853cdf0e10cSrcweir 854cdf0e10cSrcweir sal_Bool IsNumeric( const String &rText ) 855cdf0e10cSrcweir { 856cdf0e10cSrcweir sal_Bool bRes = sal_False; 857cdf0e10cSrcweir xub_StrLen nLen = rText.Len(); 858cdf0e10cSrcweir if (nLen) 859cdf0e10cSrcweir { 860cdf0e10cSrcweir bRes = sal_True; 861cdf0e10cSrcweir xub_StrLen i = 0; 862cdf0e10cSrcweir while (i < nLen) 863cdf0e10cSrcweir { 864cdf0e10cSrcweir sal_Unicode cChar = rText.GetChar( i++ ); 865cdf0e10cSrcweir if ( !((sal_Unicode)'0' <= cChar && cChar <= (sal_Unicode)'9') ) 866cdf0e10cSrcweir { 867cdf0e10cSrcweir bRes = sal_False; 868cdf0e10cSrcweir break; 869cdf0e10cSrcweir } 870cdf0e10cSrcweir } 871cdf0e10cSrcweir } 872cdf0e10cSrcweir return bRes; 873cdf0e10cSrcweir } 874cdf0e10cSrcweir 875cdf0e10cSrcweir 876cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 877cdf0e10cSrcweir 878cdf0e10cSrcweir uno::Reference< XInterface > GetOneInstanceService( const char *pServiceName ) 879cdf0e10cSrcweir { 880cdf0e10cSrcweir uno::Reference< XInterface > xRef; 881cdf0e10cSrcweir 882cdf0e10cSrcweir if (pServiceName) 883cdf0e10cSrcweir { 884cdf0e10cSrcweir uno::Reference< XMultiServiceFactory > xMgr( getProcessServiceFactory() ); 885cdf0e10cSrcweir if (xMgr.is()) 886cdf0e10cSrcweir { 887cdf0e10cSrcweir try 888cdf0e10cSrcweir { 889cdf0e10cSrcweir xRef = xMgr->createInstance( A2OU( pServiceName ) ); 890cdf0e10cSrcweir } 891cdf0e10cSrcweir catch (uno::Exception &) 892cdf0e10cSrcweir { 893cdf0e10cSrcweir DBG_ASSERT( 0, "createInstance failed" ); 894cdf0e10cSrcweir } 895cdf0e10cSrcweir } 896cdf0e10cSrcweir } 897cdf0e10cSrcweir 898cdf0e10cSrcweir return xRef; 899cdf0e10cSrcweir } 900cdf0e10cSrcweir 901cdf0e10cSrcweir uno::Reference< XPropertySet > GetLinguProperties() 902cdf0e10cSrcweir { 903cdf0e10cSrcweir return uno::Reference< XPropertySet > ( 904cdf0e10cSrcweir GetOneInstanceService( SN_LINGU_PROPERTIES ), UNO_QUERY ); 905cdf0e10cSrcweir } 906cdf0e10cSrcweir 907cdf0e10cSrcweir uno::Reference< XSearchableDictionaryList > GetSearchableDictionaryList() 908cdf0e10cSrcweir { 909cdf0e10cSrcweir return uno::Reference< XSearchableDictionaryList > ( 910cdf0e10cSrcweir GetOneInstanceService( SN_DICTIONARY_LIST ), UNO_QUERY ); 911cdf0e10cSrcweir } 912cdf0e10cSrcweir 913cdf0e10cSrcweir uno::Reference< XDictionaryList > GetDictionaryList() 914cdf0e10cSrcweir { 915cdf0e10cSrcweir return uno::Reference< XDictionaryList > ( 916cdf0e10cSrcweir GetOneInstanceService( SN_DICTIONARY_LIST ), UNO_QUERY ); 917cdf0e10cSrcweir } 918cdf0e10cSrcweir 919cdf0e10cSrcweir uno::Reference< XDictionary > GetIgnoreAllList() 920cdf0e10cSrcweir { 921cdf0e10cSrcweir uno::Reference< XDictionary > xRes; 922cdf0e10cSrcweir uno::Reference< XDictionaryList > xDL( GetDictionaryList() ); 923cdf0e10cSrcweir if (xDL.is()) 924cdf0e10cSrcweir xRes = xDL->getDictionaryByName( A2OU("IgnoreAllList") ); 925cdf0e10cSrcweir return xRes; 926cdf0e10cSrcweir } 927cdf0e10cSrcweir 928cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 929cdf0e10cSrcweir 930cdf0e10cSrcweir AppExitListener::AppExitListener() 931cdf0e10cSrcweir { 932cdf0e10cSrcweir // add object to Desktop EventListeners in order to properly call 933cdf0e10cSrcweir // the AtExit function at appliction exit. 934cdf0e10cSrcweir uno::Reference< XMultiServiceFactory > xMgr = getProcessServiceFactory(); 935cdf0e10cSrcweir 936cdf0e10cSrcweir if (xMgr.is()) 937cdf0e10cSrcweir { 938cdf0e10cSrcweir try 939cdf0e10cSrcweir { 940cdf0e10cSrcweir xDesktop = uno::Reference< frame::XDesktop >( 941cdf0e10cSrcweir xMgr->createInstance( A2OU( SN_DESKTOP ) ), UNO_QUERY ); 942cdf0e10cSrcweir } 943cdf0e10cSrcweir catch (uno::Exception &) 944cdf0e10cSrcweir { 945cdf0e10cSrcweir DBG_ASSERT( 0, "createInstance failed" ); 946cdf0e10cSrcweir } 947cdf0e10cSrcweir } 948cdf0e10cSrcweir } 949cdf0e10cSrcweir 950cdf0e10cSrcweir AppExitListener::~AppExitListener() 951cdf0e10cSrcweir { 952cdf0e10cSrcweir } 953cdf0e10cSrcweir 954cdf0e10cSrcweir 955cdf0e10cSrcweir void AppExitListener::Activate() 956cdf0e10cSrcweir { 957cdf0e10cSrcweir if (xDesktop.is()) 958cdf0e10cSrcweir xDesktop->addTerminateListener( this ); 959cdf0e10cSrcweir } 960cdf0e10cSrcweir 961cdf0e10cSrcweir 962cdf0e10cSrcweir void AppExitListener::Deactivate() 963cdf0e10cSrcweir { 964cdf0e10cSrcweir if (xDesktop.is()) 965cdf0e10cSrcweir xDesktop->removeTerminateListener( this ); 966cdf0e10cSrcweir } 967cdf0e10cSrcweir 968cdf0e10cSrcweir 969cdf0e10cSrcweir void SAL_CALL 970cdf0e10cSrcweir AppExitListener::disposing( const EventObject& rEvtSource ) 971cdf0e10cSrcweir throw(RuntimeException) 972cdf0e10cSrcweir { 973cdf0e10cSrcweir MutexGuard aGuard( GetLinguMutex() ); 974cdf0e10cSrcweir 975cdf0e10cSrcweir if (xDesktop.is() && rEvtSource.Source == xDesktop) 976cdf0e10cSrcweir { 977cdf0e10cSrcweir xDesktop = NULL; //! release reference to desktop 978cdf0e10cSrcweir } 979cdf0e10cSrcweir } 980cdf0e10cSrcweir 981cdf0e10cSrcweir 982cdf0e10cSrcweir void SAL_CALL 983cdf0e10cSrcweir AppExitListener::queryTermination( const EventObject& /*rEvtSource*/ ) 984cdf0e10cSrcweir throw(frame::TerminationVetoException, RuntimeException) 985cdf0e10cSrcweir { 986cdf0e10cSrcweir //MutexGuard aGuard( GetLinguMutex() ); 987cdf0e10cSrcweir } 988cdf0e10cSrcweir 989cdf0e10cSrcweir 990cdf0e10cSrcweir void SAL_CALL 991cdf0e10cSrcweir AppExitListener::notifyTermination( const EventObject& rEvtSource ) 992cdf0e10cSrcweir throw(RuntimeException) 993cdf0e10cSrcweir { 994cdf0e10cSrcweir MutexGuard aGuard( GetLinguMutex() ); 995cdf0e10cSrcweir 996cdf0e10cSrcweir if (xDesktop.is() && rEvtSource.Source == xDesktop) 997cdf0e10cSrcweir { 998cdf0e10cSrcweir AtExit(); 999cdf0e10cSrcweir } 1000cdf0e10cSrcweir } 1001cdf0e10cSrcweir 1002cdf0e10cSrcweir /////////////////////////////////////////////////////////////////////////// 1003cdf0e10cSrcweir 1004cdf0e10cSrcweir } // namespace linguistic 1005cdf0e10cSrcweir 1006