/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; #define CFG_READONLY_DEFAULT false // SvtCJKOptions_Impl ---------------------------------------------------------- class SvtCTLOptions_Impl : public utl::ConfigItem { private: bool m_bIsLoaded; bool m_bCTLFontEnabled; bool m_bCTLSequenceChecking; bool m_bCTLRestricted; bool m_bCTLTypeAndReplace; SvtCTLOptions::CursorMovement m_eCTLCursorMovement; SvtCTLOptions::TextNumerals m_eCTLTextNumerals; bool m_bROCTLFontEnabled; bool m_bROCTLSequenceChecking; bool m_bROCTLRestricted; bool m_bROCTLTypeAndReplace; bool m_bROCTLCursorMovement; bool m_bROCTLTextNumerals; public: SvtCTLOptions_Impl(); virtual ~SvtCTLOptions_Impl(); virtual void Notify( const Sequence< OUString >& _aPropertyNames ) SAL_OVERRIDE; virtual void Commit() SAL_OVERRIDE; void Load(); bool IsLoaded() { return m_bIsLoaded; } void SetCTLFontEnabled( bool _bEnabled ); bool IsCTLFontEnabled() const { return m_bCTLFontEnabled; } void SetCTLSequenceChecking( bool _bEnabled ); bool IsCTLSequenceChecking() const { return m_bCTLSequenceChecking;} void SetCTLSequenceCheckingRestricted( bool _bEnable ); bool IsCTLSequenceCheckingRestricted( void ) const { return m_bCTLRestricted; } void SetCTLSequenceCheckingTypeAndReplace( bool _bEnable ); bool IsCTLSequenceCheckingTypeAndReplace() const { return m_bCTLTypeAndReplace; } void SetCTLCursorMovement( SvtCTLOptions::CursorMovement _eMovement ); SvtCTLOptions::CursorMovement GetCTLCursorMovement() const { return m_eCTLCursorMovement; } void SetCTLTextNumerals( SvtCTLOptions::TextNumerals _eNumerals ); SvtCTLOptions::TextNumerals GetCTLTextNumerals() const { return m_eCTLTextNumerals; } bool IsReadOnly(SvtCTLOptions::EOption eOption) const; }; namespace { struct PropertyNames : public rtl::Static< Sequence< OUString >, PropertyNames > {}; } bool SvtCTLOptions_Impl::IsReadOnly(SvtCTLOptions::EOption eOption) const { bool bReadOnly = CFG_READONLY_DEFAULT; switch(eOption) { case SvtCTLOptions::E_CTLFONT : bReadOnly = m_bROCTLFontEnabled ; break; case SvtCTLOptions::E_CTLSEQUENCECHECKING : bReadOnly = m_bROCTLSequenceChecking ; break; case SvtCTLOptions::E_CTLCURSORMOVEMENT : bReadOnly = m_bROCTLCursorMovement ; break; case SvtCTLOptions::E_CTLTEXTNUMERALS : bReadOnly = m_bROCTLTextNumerals ; break; case SvtCTLOptions::E_CTLSEQUENCECHECKINGRESTRICTED: bReadOnly = m_bROCTLRestricted ; break; case SvtCTLOptions::E_CTLSEQUENCECHECKINGTYPEANDREPLACE: bReadOnly = m_bROCTLTypeAndReplace; break; default: OSL_FAIL( "SvtCTLOptions_Impl::IsReadOnly() - invalid option" ); } return bReadOnly; } SvtCTLOptions_Impl::SvtCTLOptions_Impl() : utl::ConfigItem("Office.Common/I18N/CTL"), m_bIsLoaded ( false ), m_bCTLFontEnabled ( false ), m_bCTLSequenceChecking ( false ), m_bCTLRestricted ( false ), m_bCTLTypeAndReplace ( false ), m_eCTLCursorMovement ( SvtCTLOptions::MOVEMENT_LOGICAL ), m_eCTLTextNumerals ( SvtCTLOptions::NUMERALS_ARABIC ), m_bROCTLFontEnabled ( CFG_READONLY_DEFAULT ), m_bROCTLSequenceChecking( CFG_READONLY_DEFAULT ), m_bROCTLRestricted ( CFG_READONLY_DEFAULT ), m_bROCTLTypeAndReplace ( CFG_READONLY_DEFAULT ), m_bROCTLCursorMovement ( CFG_READONLY_DEFAULT ), m_bROCTLTextNumerals ( CFG_READONLY_DEFAULT ) { } SvtCTLOptions_Impl::~SvtCTLOptions_Impl() { if ( IsModified() ) Commit(); } void SvtCTLOptions_Impl::Notify( const Sequence< OUString >& ) { Load(); NotifyListeners(SFX_HINT_CTL_SETTINGS_CHANGED); } void SvtCTLOptions_Impl::Commit() { Sequence< OUString > &rPropertyNames = PropertyNames::get(); OUString* pOrgNames = rPropertyNames.getArray(); sal_Int32 nOrgCount = rPropertyNames.getLength(); Sequence< OUString > aNames( nOrgCount ); Sequence< Any > aValues( nOrgCount ); OUString* pNames = aNames.getArray(); Any* pValues = aValues.getArray(); sal_Int32 nRealCount = 0; const uno::Type& rType = ::getBooleanCppuType(); for ( int nProp = 0; nProp < nOrgCount; nProp++ ) { switch ( nProp ) { case 0: { if (!m_bROCTLFontEnabled) { pNames[nRealCount] = pOrgNames[nProp]; pValues[nRealCount].setValue( &m_bCTLFontEnabled, rType ); ++nRealCount; } } break; case 1: { if (!m_bROCTLSequenceChecking) { pNames[nRealCount] = pOrgNames[nProp]; pValues[nRealCount].setValue( &m_bCTLSequenceChecking, rType ); ++nRealCount; } } break; case 2: { if (!m_bROCTLCursorMovement) { pNames[nRealCount] = pOrgNames[nProp]; pValues[nRealCount] <<= (sal_Int32)m_eCTLCursorMovement; ++nRealCount; } } break; case 3: { if (!m_bROCTLTextNumerals) { pNames[nRealCount] = pOrgNames[nProp]; pValues[nRealCount] <<= (sal_Int32)m_eCTLTextNumerals; ++nRealCount; } } break; case 4: { if (!m_bROCTLRestricted) { pNames[nRealCount] = pOrgNames[nProp]; pValues[nRealCount].setValue( &m_bCTLRestricted, rType ); ++nRealCount; } } break; case 5: { if(!m_bROCTLTypeAndReplace) { pNames[nRealCount] = pOrgNames[nProp]; pValues[nRealCount].setValue( &m_bCTLTypeAndReplace, rType ); ++nRealCount; } } break; } } aNames.realloc(nRealCount); aValues.realloc(nRealCount); PutProperties( aNames, aValues ); //broadcast changes NotifyListeners(SFX_HINT_CTL_SETTINGS_CHANGED); } void SvtCTLOptions_Impl::Load() { Sequence< OUString >& rPropertyNames = PropertyNames::get(); if ( !rPropertyNames.getLength() ) { rPropertyNames.realloc(6); OUString* pNames = rPropertyNames.getArray(); pNames[0] = "CTLFont"; pNames[1] = "CTLSequenceChecking"; pNames[2] = "CTLCursorMovement"; pNames[3] = "CTLTextNumerals"; pNames[4] = "CTLSequenceCheckingRestricted"; pNames[5] = "CTLSequenceCheckingTypeAndReplace"; EnableNotification( rPropertyNames ); } Sequence< Any > aValues = GetProperties( rPropertyNames ); Sequence< sal_Bool > aROStates = GetReadOnlyStates( rPropertyNames ); const Any* pValues = aValues.getConstArray(); const sal_Bool* pROStates = aROStates.getConstArray(); DBG_ASSERT( aValues.getLength() == rPropertyNames.getLength(), "GetProperties failed" ); DBG_ASSERT( aROStates.getLength() == rPropertyNames.getLength(), "GetReadOnlyStates failed" ); if ( aValues.getLength() == rPropertyNames.getLength() && aROStates.getLength() == rPropertyNames.getLength() ) { bool bValue = false; sal_Int32 nValue = 0; for ( int nProp = 0; nProp < rPropertyNames.getLength(); nProp++ ) { if ( pValues[nProp].hasValue() ) { if ( pValues[nProp] >>= bValue ) { switch ( nProp ) { case 0: { m_bCTLFontEnabled = bValue; m_bROCTLFontEnabled = pROStates[nProp]; } break; case 1: { m_bCTLSequenceChecking = bValue; m_bROCTLSequenceChecking = pROStates[nProp]; } break; case 4: { m_bCTLRestricted = bValue; m_bROCTLRestricted = pROStates[nProp]; } break; case 5: { m_bCTLTypeAndReplace = bValue; m_bROCTLTypeAndReplace = pROStates[nProp]; } break; } } else if ( pValues[nProp] >>= nValue ) { switch ( nProp ) { case 2: { m_eCTLCursorMovement = (SvtCTLOptions::CursorMovement)nValue; m_bROCTLCursorMovement = pROStates[nProp]; } break; case 3: { m_eCTLTextNumerals = (SvtCTLOptions::TextNumerals)nValue; m_bROCTLTextNumerals = pROStates[nProp]; } break; } } } } } if (!m_bCTLFontEnabled) { sal_uInt16 nScriptType = SvtLanguageOptions::GetScriptTypeOfLanguage(LANGUAGE_SYSTEM); //system locale is CTL bool bAutoEnableCTL = (nScriptType & SCRIPTTYPE_COMPLEX); LanguageType eSystemLanguage = LANGUAGE_SYSTEM; if (!bAutoEnableCTL) { SvtSystemLanguageOptions aSystemLocaleSettings; //windows secondary system locale is CTL eSystemLanguage = aSystemLocaleSettings.GetWin16SystemLanguage(); if (eSystemLanguage != LANGUAGE_SYSTEM) { sal_uInt16 nWinScript = SvtLanguageOptions::GetScriptTypeOfLanguage( eSystemLanguage ); bAutoEnableCTL = (nWinScript & SCRIPTTYPE_COMPLEX); } //CTL keyboard is installed if (!bAutoEnableCTL) bAutoEnableCTL = aSystemLocaleSettings.isCTLKeyboardLayoutInstalled(); } if (bAutoEnableCTL) { m_bCTLFontEnabled = true; sal_uInt16 nLanguage = SvtSysLocale().GetLanguageTag().getLanguageType(); //enable sequence checking for the appropriate languages m_bCTLSequenceChecking = m_bCTLRestricted = m_bCTLTypeAndReplace = (MsLangId::needsSequenceChecking( nLanguage) || MsLangId::needsSequenceChecking( eSystemLanguage)); Commit(); } } m_bIsLoaded = true; } void SvtCTLOptions_Impl::SetCTLFontEnabled( bool _bEnabled ) { if(!m_bROCTLFontEnabled && m_bCTLFontEnabled != _bEnabled) { m_bCTLFontEnabled = _bEnabled; SetModified(); NotifyListeners(0); } } void SvtCTLOptions_Impl::SetCTLSequenceChecking( bool _bEnabled ) { if(!m_bROCTLSequenceChecking && m_bCTLSequenceChecking != _bEnabled) { SetModified(); m_bCTLSequenceChecking = _bEnabled; NotifyListeners(0); } } void SvtCTLOptions_Impl::SetCTLSequenceCheckingRestricted( bool _bEnabled ) { if(!m_bROCTLRestricted && m_bCTLRestricted != _bEnabled) { SetModified(); m_bCTLRestricted = _bEnabled; NotifyListeners(0); } } void SvtCTLOptions_Impl::SetCTLSequenceCheckingTypeAndReplace( bool _bEnabled ) { if(!m_bROCTLTypeAndReplace && m_bCTLTypeAndReplace != _bEnabled) { SetModified(); m_bCTLTypeAndReplace = _bEnabled; NotifyListeners(0); } } void SvtCTLOptions_Impl::SetCTLCursorMovement( SvtCTLOptions::CursorMovement _eMovement ) { if (!m_bROCTLCursorMovement && m_eCTLCursorMovement != _eMovement ) { SetModified(); m_eCTLCursorMovement = _eMovement; NotifyListeners(0); } } void SvtCTLOptions_Impl::SetCTLTextNumerals( SvtCTLOptions::TextNumerals _eNumerals ) { if (!m_bROCTLTextNumerals && m_eCTLTextNumerals != _eNumerals ) { SetModified(); m_eCTLTextNumerals = _eNumerals; NotifyListeners(0); } } // global ---------------------------------------------------------------- static SvtCTLOptions_Impl* pCTLOptions = NULL; static sal_Int32 nCTLRefCount = 0; namespace { struct CTLMutex : public rtl::Static< osl::Mutex, CTLMutex > {}; } // class SvtCTLOptions -------------------------------------------------- SvtCTLOptions::SvtCTLOptions( bool bDontLoad ) { // Global access, must be guarded (multithreading) ::osl::MutexGuard aGuard( CTLMutex::get() ); if ( !pCTLOptions ) { pCTLOptions = new SvtCTLOptions_Impl; ItemHolder2::holdConfigItem(E_CTLOPTIONS); } if( !bDontLoad && !pCTLOptions->IsLoaded() ) pCTLOptions->Load(); ++nCTLRefCount; m_pImp = pCTLOptions; m_pImp->AddListener(this); } SvtCTLOptions::~SvtCTLOptions() { // Global access, must be guarded (multithreading) ::osl::MutexGuard aGuard( CTLMutex::get() ); m_pImp->RemoveListener(this); if ( !--nCTLRefCount ) DELETEZ( pCTLOptions ); } void SvtCTLOptions::SetCTLFontEnabled( bool _bEnabled ) { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); pCTLOptions->SetCTLFontEnabled( _bEnabled ); } bool SvtCTLOptions::IsCTLFontEnabled() const { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); return pCTLOptions->IsCTLFontEnabled(); } void SvtCTLOptions::SetCTLSequenceChecking( bool _bEnabled ) { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); pCTLOptions->SetCTLSequenceChecking(_bEnabled); } bool SvtCTLOptions::IsCTLSequenceChecking() const { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); return pCTLOptions->IsCTLSequenceChecking(); } void SvtCTLOptions::SetCTLSequenceCheckingRestricted( bool _bEnable ) { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); pCTLOptions->SetCTLSequenceCheckingRestricted(_bEnable); } bool SvtCTLOptions::IsCTLSequenceCheckingRestricted( void ) const { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); return pCTLOptions->IsCTLSequenceCheckingRestricted(); } void SvtCTLOptions::SetCTLSequenceCheckingTypeAndReplace( bool _bEnable ) { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); pCTLOptions->SetCTLSequenceCheckingTypeAndReplace(_bEnable); } bool SvtCTLOptions::IsCTLSequenceCheckingTypeAndReplace() const { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); return pCTLOptions->IsCTLSequenceCheckingTypeAndReplace(); } void SvtCTLOptions::SetCTLCursorMovement( SvtCTLOptions::CursorMovement _eMovement ) { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); pCTLOptions->SetCTLCursorMovement( _eMovement ); } SvtCTLOptions::CursorMovement SvtCTLOptions::GetCTLCursorMovement() const { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); return pCTLOptions->GetCTLCursorMovement(); } void SvtCTLOptions::SetCTLTextNumerals( SvtCTLOptions::TextNumerals _eNumerals ) { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); pCTLOptions->SetCTLTextNumerals( _eNumerals ); } SvtCTLOptions::TextNumerals SvtCTLOptions::GetCTLTextNumerals() const { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); return pCTLOptions->GetCTLTextNumerals(); } bool SvtCTLOptions::IsReadOnly(EOption eOption) const { DBG_ASSERT( pCTLOptions->IsLoaded(), "CTL options not loaded" ); return pCTLOptions->IsReadOnly(eOption); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */