/************************************************************************* * * $RCSfile: misc.cxx,v $ * * $Revision: 1.21 $ * * last change: $Author: obo $ $Date: 2004-04-27 16:08:42 $ * * The Contents of this file are made available subject to the terms of * either of the following licenses * * - GNU Lesser General Public License Version 2.1 * - Sun Industry Standards Source License Version 1.1 * * Sun Microsystems Inc., October, 2000 * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2000 by Sun Microsystems, Inc. * 901 San Antonio Road, Palo Alto, CA 94303, USA * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA * * * Sun Industry Standards Source License Version 1.1 * ================================================= * The contents of this file are subject to the Sun Industry Standards * Source License Version 1.1 (the "License"); You may not use this file * except in compliance with the License. You may obtain a copy of the * License at http://www.openoffice.org/license.html. * * Software provided under this License is provided on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, * WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS, * MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING. * See the License for the specific provisions governing your rights and * obligations concerning the Software. * * The Initial Developer of the Original Code is: Sun Microsystems, Inc. * * Copyright: 2000 by Sun Microsystems, Inc. * * All Rights Reserved. * * Contributor(s): _______________________________________ * * ************************************************************************/ #ifndef _STRING_HXX #include #endif #ifndef _FSYS_HXX #include #endif #ifndef _TOOLS_DEBUG_HXX #include #endif //#ifndef _RTL_USTRBUF_HXX_ //#include //#endif #ifndef INCLUDED_SVTOOLS_PATHOPTIONS_HXX #include #endif #ifndef _SVTOOLS_LNGMISC_HXX_ #include #endif #ifndef _UCBHELPER_CONTENT_HXX #include #endif #ifndef _COM_SUN_STAR_UCB_XCOMMANDENVIRONMENT_HPP_ #include #endif #ifndef _COM_SUN_STAR_BEANS_XPROPERTYSET_HPP_ #include #endif #ifndef _COM_SUN_STAR_BEANS_XFASTPROPERTYSET_HPP_ #include #endif #ifndef _COM_SUN_STAR_BEANS_XPROPERTYCHANGELISTENER_HPP_ #include #endif #ifndef _COM_SUN_STAR_FRAME_XTERMINATELISTENER_HPP_ #include #endif #ifndef _COM_SUN_STAR_FRAME_XDESKTOP_HPP_ #include #endif #include #include #include #include #include #include #ifndef _UNOTOOLS_PROCESSFACTORY_HXX_ #include #endif #ifndef _UNOTOOLS_LOCALEDATAWRAPPER_HXX #include #endif #include "misc.hxx" #include "defs.hxx" #include "lngprops.hxx" #include using namespace utl; using namespace osl; using namespace rtl; using namespace com::sun::star; using namespace com::sun::star::beans; using namespace com::sun::star::lang; using namespace com::sun::star::uno; using namespace com::sun::star::i18n; using namespace com::sun::star::linguistic2; namespace linguistic { /////////////////////////////////////////////////////////////////////////// osl::Mutex & GetLinguMutex() { static osl::Mutex aMutex; return aMutex; } /////////////////////////////////////////////////////////////////////////// LocaleDataWrapper & GetLocaleDataWrapper( INT16 nLang ) { static LocaleDataWrapper aLclDtaWrp( getProcessServiceFactory(), CreateLocale( Application::GetSettings().GetUILanguage() ) ); const Locale &rLcl = aLclDtaWrp.getLoadedLocale(); Locale aLcl( CreateLocale( nLang ) ); if (aLcl.Language != rLcl.Language || aLcl.Country != rLcl.Country || aLcl.Variant != rLcl.Variant) aLclDtaWrp.setLocale( aLcl ); return aLclDtaWrp; } /////////////////////////////////////////////////////////////////////////// /** returns text-encoding used for ByteString unicode String conversion */ rtl_TextEncoding GetTextEncoding( INT16 nLanguage ) { DBG_ASSERT( nLanguage != LANGUAGE_NONE, "invalid language argument" ) static INT16 nLastLanguage = LANGUAGE_NONE; // set default value for unknown languages static rtl_TextEncoding nEncoding = RTL_TEXTENCODING_DONTKNOW; if (nLastLanguage != nLanguage) { //!! IPR uses textencodings Latin-1, Latin-2, Latin-5 and Latin-7 !! nLastLanguage = nLanguage; switch (nLanguage) { case LANGUAGE_GERMAN : case LANGUAGE_GERMAN_SWISS : case LANGUAGE_ENGLISH_US : case LANGUAGE_ENGLISH_UK : case LANGUAGE_FRENCH : case LANGUAGE_ITALIAN : case LANGUAGE_SPANISH : case LANGUAGE_CATALAN : case LANGUAGE_PORTUGUESE : case LANGUAGE_PORTUGUESE_BRAZILIAN : case LANGUAGE_DANISH : case LANGUAGE_DUTCH : case LANGUAGE_SWEDISH : case LANGUAGE_FINNISH : case LANGUAGE_NORWEGIAN_BOKMAL : case LANGUAGE_NORWEGIAN_NYNORSK : case LANGUAGE_AFRIKAANS : case LANGUAGE_ENGLISH_EIRE : case LANGUAGE_ENGLISH_AUS : #ifdef WNT nEncoding = RTL_TEXTENCODING_MS_1252; break; #else nEncoding = RTL_TEXTENCODING_ISO_8859_1; break; #endif case LANGUAGE_CZECH : case LANGUAGE_HUNGARIAN : case LANGUAGE_POLISH : #ifdef WNT nEncoding = RTL_TEXTENCODING_MS_1250; break; #else nEncoding = RTL_TEXTENCODING_ISO_8859_2; break; #endif case LANGUAGE_RUSSIAN : #ifdef WNT nEncoding = RTL_TEXTENCODING_MS_1251; break; #else nEncoding = RTL_TEXTENCODING_ISO_8859_5; break; #endif case LANGUAGE_GREEK : #ifdef WNT nEncoding = RTL_TEXTENCODING_MS_1253; break; #else nEncoding = RTL_TEXTENCODING_ISO_8859_7; break; #endif default: DBG_ERROR( "unexpected language" ); } } return nEncoding; } /////////////////////////////////////////////////////////////////////////// BOOL IsUseDicList( const PropertyValues &rProperties, const Reference< XPropertySet > &rxProp ) { BOOL bRes = TRUE; INT32 nLen = rProperties.getLength(); const PropertyValue *pVal = rProperties.getConstArray(); INT32 i; for ( i = 0; i < nLen; ++i) { if (UPH_IS_USE_DICTIONARY_LIST == pVal[i].Handle) { pVal[i].Value >>= bRes; break; } } if (i >= nLen) // no temporary value found in 'rProperties' { Reference< XFastPropertySet > xFast( rxProp, UNO_QUERY ); if (xFast.is()) xFast->getFastPropertyValue( UPH_IS_USE_DICTIONARY_LIST ) >>= bRes; } return bRes; } BOOL IsIgnoreControlChars( const PropertyValues &rProperties, const Reference< XPropertySet > &rxProp ) { BOOL bRes = TRUE; INT32 nLen = rProperties.getLength(); const PropertyValue *pVal = rProperties.getConstArray(); INT32 i; for ( i = 0; i < nLen; ++i) { if (UPH_IS_IGNORE_CONTROL_CHARACTERS == pVal[i].Handle) { pVal[i].Value >>= bRes; break; } } if (i >= nLen) // no temporary value found in 'rProperties' { Reference< XFastPropertySet > xFast( rxProp, UNO_QUERY ); if (xFast.is()) xFast->getFastPropertyValue( UPH_IS_IGNORE_CONTROL_CHARACTERS ) >>= bRes; } return bRes; } static BOOL lcl_HasHyphInfo( const Reference &xEntry ) { BOOL bRes = FALSE; if (xEntry.is()) { // there has to be (at least one) '=' denoting a hyphenation position // and it must not be before any character of the word sal_Int32 nIdx = xEntry->getDictionaryWord().indexOf( '=' ); bRes = nIdx != -1 && nIdx != 0; } return bRes; } Reference< XDictionaryEntry > SearchDicList( const Reference< XDictionaryList > &xDicList, const OUString &rWord, INT16 nLanguage, BOOL bSearchPosDics, BOOL bSearchSpellEntry ) { MutexGuard aGuard( GetLinguMutex() ); Reference< XDictionaryEntry > xEntry; if (!xDicList.is()) return xEntry; const uno::Sequence< Reference< XDictionary > > aDics( xDicList->getDictionaries() ); const Reference< XDictionary > *pDic = aDics.getConstArray(); INT32 nDics = xDicList->getCount(); INT32 i; for (i = 0; i < nDics; i++) { Reference< XDictionary1 > axDic( pDic[i], UNO_QUERY ); DictionaryType eType = axDic->getDictionaryType(); INT16 nLang = axDic->getLanguage(); if ( axDic.is() && axDic->isActive() && (nLang == nLanguage || nLang == LANGUAGE_NONE) ) { DBG_ASSERT( eType != DictionaryType_MIXED, "lng : unexpected dictionary type" ); if ( (!bSearchPosDics && eType == DictionaryType_NEGATIVE) || ( bSearchPosDics && eType == DictionaryType_POSITIVE)) { if ( (xEntry = axDic->getEntry( rWord )).is() ) { if (bSearchSpellEntry || lcl_HasHyphInfo( xEntry )) break; } xEntry = 0; } } } return xEntry; } /////////////////////////////////////////////////////////////////////////// LanguageType LocaleToLanguage( const Locale& rLocale ) { // empty Locale -> LANGUAGE_NONE if ( rLocale.Language.getLength() == 0 ) return LANGUAGE_NONE; // Variant of Locale is ignored return ConvertIsoNamesToLanguage( rLocale.Language, rLocale.Country ); } Locale& LanguageToLocale( Locale& rLocale, LanguageType eLang ) { String aLangStr, aCtryStr; if ( eLang != LANGUAGE_NONE /* && eLang != LANGUAGE_SYSTEM */) ConvertLanguageToIsoNames( eLang, aLangStr, aCtryStr ); rLocale.Language = aLangStr; rLocale.Country = aCtryStr; rLocale.Variant = OUString(); return rLocale; } Locale CreateLocale( LanguageType eLang ) { String aLangStr, aCtryStr; if ( eLang != LANGUAGE_NONE /* && eLang != LANGUAGE_SYSTEM */) ConvertLanguageToIsoNames( eLang, aLangStr, aCtryStr ); return Locale( aLangStr, aCtryStr, OUString() ); } uno::Sequence< Locale > LangSeqToLocaleSeq( const uno::Sequence< INT16 > &rLangSeq ) { const INT16 *pLang = rLangSeq.getConstArray(); INT32 nCount = rLangSeq.getLength(); uno::Sequence< Locale > aLocales( nCount ); Locale *pLocale = aLocales.getArray(); for (INT32 i = 0; i < nCount; ++i) { LanguageToLocale( pLocale[i], pLang[ i ] ); } return aLocales; } uno::Sequence< INT16 > LocaleSeqToLangSeq( uno::Sequence< Locale > &rLocaleSeq ) { const Locale *pLocale = rLocaleSeq.getConstArray(); INT32 nCount = rLocaleSeq.getLength(); uno::Sequence< INT16 > aLangs( nCount ); INT16 *pLang = aLangs.getArray(); for (INT32 i = 0; i < nCount; ++i) { pLang[i] = LocaleToLanguage( pLocale[i] ); } return aLangs; } /////////////////////////////////////////////////////////////////////////// BOOL IsReadOnly( const String &rURL, BOOL *pbExist ) { BOOL bRes = FALSE; BOOL bExists = FALSE; if (rURL.Len() > 0) { try { Reference< ::com::sun::star::ucb::XCommandEnvironment > xCmdEnv; ::ucb::Content aContent( rURL, xCmdEnv ); bExists = aContent.isDocument(); if (bExists) { Any aAny( aContent.getPropertyValue( A2OU( "IsReadOnly" ) ) ); aAny >>= bRes; } } catch (Exception &) { bRes = TRUE; } } if (pbExist) *pbExist = bExists; return bRes; } /////////////////////////////////////////////////////////////////////////// static BOOL GetAltSpelling( INT16 &rnChgPos, INT16 &rnChgLen, OUString &rRplc, Reference< XHyphenatedWord > &rxHyphWord ) { BOOL bRes = rxHyphWord->isAlternativeSpelling(); if (bRes) { OUString aWord( rxHyphWord->getWord() ), aHyphenatedWord( rxHyphWord->getHyphenatedWord() ); INT16 nHyphenationPos = rxHyphWord->getHyphenationPos(), nHyphenPos = rxHyphWord->getHyphenPos(); const sal_Unicode *pWord = aWord.getStr(), *pAltWord = aHyphenatedWord.getStr(); // at least char changes directly left or right to the hyphen // should(!) be handled properly... //! nHyphenationPos and nHyphenPos differ at most by 1 (see above) //! Beware: eg "Schiffahrt" in German (pre spelling reform) //! proves to be a bit nasty (nChgPosLeft and nChgPosRight overlap //! to an extend.) // find first different char from left sal_Int32 nPosL = 0, nAltPosL = 0; for (INT16 i = 0 ; pWord[ nPosL ] == pAltWord[ nAltPosL ]; nPosL++, nAltPosL++, i++) { // restrict changes area beginning to the right to // the char immediately following the hyphen. //! serves to insert the additional "f" in "Schiffahrt" at //! position 5 rather than position 6. if (i >= nHyphenationPos + 1) break; } // find first different char from right sal_Int32 nPosR = aWord.getLength() - 1, nAltPosR = aHyphenatedWord.getLength() - 1; for ( ; nPosR >= nPosL && nAltPosR >= nAltPosL && pWord[ nPosR ] == pAltWord[ nAltPosR ]; nPosR--, nAltPosR--) ; rnChgPos = (INT16) nPosL; rnChgLen = nPosR - nPosL + 1; DBG_ASSERT( rnChgLen >= 0, "nChgLen < 0"); sal_Int32 nTxtStart = nPosL; sal_Int32 nTxtLen = nAltPosL - nPosL + 1; rRplc = aHyphenatedWord.copy( nTxtStart, nTxtLen ); } return bRes; } static INT16 GetOrigWordPos( const OUString &rOrigWord, INT16 nPos ) { INT32 nLen = rOrigWord.getLength(); INT32 i = -1; while (nPos >= 0 && i++ < nLen) { sal_Unicode cChar = rOrigWord[i]; BOOL bSkip = IsHyphen( cChar ) || IsControlChar( cChar ); if (!bSkip) --nPos; } return (0 <= i && i < nLen) ? i : -1; } INT32 GetPosInWordToCheck( const OUString &rTxt, INT32 nPos ) { INT32 nRes = -1; INT32 nLen = rTxt.getLength(); if (0 <= nPos && nPos < nLen) { nRes = 0; for (INT32 i = 0; i < nPos; ++i) { sal_Unicode cChar = rTxt[i]; BOOL bSkip = IsHyphen( cChar ) || IsControlChar( cChar ); if (!bSkip) ++nRes; } } return nRes; } Reference< XHyphenatedWord > RebuildHyphensAndControlChars( const OUString &rOrigWord, Reference< XHyphenatedWord > &rxHyphWord ) { Reference< XHyphenatedWord > xRes; if (rOrigWord.getLength() && rxHyphWord.is()) { INT16 nChgPos = 0, nChgLen = 0; OUString aRplc; BOOL bAltSpelling = GetAltSpelling( nChgPos, nChgLen, aRplc, rxHyphWord ); #if OSL_DEBUG_LEVEL > 1 OUString aWord( rxHyphWord->getWord() ); #endif OUString aOrigHyphenatedWord; INT16 nOrigHyphenPos = -1; INT16 nOrigHyphenationPos = -1; if (!bAltSpelling) { aOrigHyphenatedWord = rOrigWord; nOrigHyphenPos = GetOrigWordPos( rOrigWord, rxHyphWord->getHyphenPos() ); nOrigHyphenationPos = GetOrigWordPos( rOrigWord, rxHyphWord->getHyphenationPos() ); } else { //! should at least work with the German words //! Bä-c-k-er and Sc-hif-fah-rt OUString aLeft, aRight; INT16 nPos = GetOrigWordPos( rOrigWord, nChgPos ); // get words like Sc-hif-fah-rt to work correct INT16 nHyphenationPos = rxHyphWord->getHyphenationPos(); if (nChgPos > nHyphenationPos) --nPos; aLeft = rOrigWord.copy( 0, nPos ); aRight = rOrigWord.copy( nPos + nChgLen ); aOrigHyphenatedWord = aLeft; aOrigHyphenatedWord += aRplc; aOrigHyphenatedWord += aRight; nOrigHyphenPos = aLeft.getLength() + rxHyphWord->getHyphenPos() - nChgPos; nOrigHyphenationPos = GetOrigWordPos( rOrigWord, nHyphenationPos ); } if (nOrigHyphenPos == -1 || nOrigHyphenationPos == -1) { DBG_ERROR( "failed to get nOrigHyphenPos or nOrigHyphenationPos" ); } else { INT16 nLang = LocaleToLanguage( rxHyphWord->getLocale() ); xRes = new HyphenatedWord( rOrigWord, nLang, nOrigHyphenationPos, aOrigHyphenatedWord, nOrigHyphenPos ); } } return xRes; } /////////////////////////////////////////////////////////////////////////// static CharClass & lcl_GetCharClass() { static CharClass aCC( CreateLocale( LANGUAGE_ENGLISH_US ) ); return aCC; } osl::Mutex & lcl_GetCharClassMutex() { static osl::Mutex aMutex; return aMutex; } BOOL IsUpper( const String &rText, INT16 nLanguage ) { MutexGuard aGuard( lcl_GetCharClassMutex() ); CharClass &rCC = lcl_GetCharClass(); rCC.setLocale( CreateLocale( nLanguage ) ); sal_Int32 nFlags = rCC.getStringType( rText, 0, rText.Len() ); return (nFlags & KCharacterType::UPPER) && !(nFlags & KCharacterType::LOWER); } BOOL IsLower( const String &rText, INT16 nLanguage ) { MutexGuard aGuard( lcl_GetCharClassMutex() ); CharClass &rCC = lcl_GetCharClass(); rCC.setLocale( CreateLocale( nLanguage ) ); sal_Int32 nFlags = rCC.getStringType( rText, 0, rText.Len() ); return (nFlags & KCharacterType::LOWER) && !(nFlags & KCharacterType::UPPER); } String ToLower( const String &rText, INT16 nLanguage ) { MutexGuard aGuard( lcl_GetCharClassMutex() ); CharClass &rCC = lcl_GetCharClass(); rCC.setLocale( CreateLocale( nLanguage ) ); return rCC.lower( rText ); } sal_Unicode ToLower( const sal_Unicode cChar, INT16 nLanguage ) { MutexGuard aGuard( lcl_GetCharClassMutex() ); CharClass &rCC = lcl_GetCharClass(); rCC.setLocale( CreateLocale( nLanguage ) ); return rCC.lower( cChar ).GetChar(0); } sal_Unicode ToUpper( const sal_Unicode cChar, INT16 nLanguage ) { MutexGuard aGuard( lcl_GetCharClassMutex() ); CharClass &rCC = lcl_GetCharClass(); rCC.setLocale( CreateLocale( nLanguage ) ); return rCC.upper( cChar ).GetChar(0); } BOOL HasDigits( const String &rText ) { xub_StrLen nLen = rText.Len(); xub_StrLen i = 0; while (i < nLen) { sal_Unicode cChar = rText.GetChar( i++ ); if ((sal_Unicode)'0' <= cChar && cChar <= (sal_Unicode)'9') return TRUE; } return FALSE; } BOOL IsNumeric( const String &rText ) { BOOL bRes = FALSE; xub_StrLen nLen = rText.Len(); if (nLen) { bRes = TRUE; xub_StrLen i = 0; while (i < nLen) { sal_Unicode cChar = rText.GetChar( i++ ); if ( !((sal_Unicode)'0' <= cChar && cChar <= (sal_Unicode)'9') ) { bRes = FALSE; break; } } } return bRes; } /////////////////////////////////////////////////////////////////////////// Reference< XInterface > GetOneInstanceService( const char *pServiceName ) { Reference< XInterface > xRef; if (pServiceName) { Reference< XMultiServiceFactory > xMgr( getProcessServiceFactory() ); if (xMgr.is()) { xRef = xMgr->createInstance( A2OU( pServiceName ) ); } } return xRef; } Reference< XPropertySet > GetLinguProperties() { return Reference< XPropertySet > ( GetOneInstanceService( SN_LINGU_PROPERTIES ), UNO_QUERY ); } Reference< XSearchableDictionaryList > GetSearchableDictionaryList() { return Reference< XSearchableDictionaryList > ( GetOneInstanceService( SN_DICTIONARY_LIST ), UNO_QUERY ); } Reference< XDictionaryList > GetDictionaryList() { return Reference< XDictionaryList > ( GetOneInstanceService( SN_DICTIONARY_LIST ), UNO_QUERY ); } /////////////////////////////////////////////////////////////////////////// AppExitListener::AppExitListener() { // add object to Desktop EventListeners in order to properly call // the AtExit function at appliction exit. Reference< XMultiServiceFactory > xMgr = getProcessServiceFactory(); if (xMgr.is()) { xDesktop = Reference< frame::XDesktop >( xMgr->createInstance( A2OU( SN_DESKTOP ) ), UNO_QUERY ); } } AppExitListener::~AppExitListener() { } void AppExitListener::Activate() { if (xDesktop.is()) xDesktop->addTerminateListener( this ); } void AppExitListener::Deactivate() { if (xDesktop.is()) xDesktop->removeTerminateListener( this ); } void SAL_CALL AppExitListener::disposing( const EventObject& rEvtSource ) throw(RuntimeException) { MutexGuard aGuard( GetLinguMutex() ); if (xDesktop.is() && rEvtSource.Source == xDesktop) { xDesktop = NULL; //! release reference to desktop } } void SAL_CALL AppExitListener::queryTermination( const EventObject& rEvtSource ) throw(frame::TerminationVetoException, RuntimeException) { //MutexGuard aGuard( GetLinguMutex() ); } void SAL_CALL AppExitListener::notifyTermination( const EventObject& rEvtSource ) throw(RuntimeException) { MutexGuard aGuard( GetLinguMutex() ); if (xDesktop.is() && rEvtSource.Source == xDesktop) { AtExit(); } } /////////////////////////////////////////////////////////////////////////// } // namespace linguistic