| /************************************************************** |
| * |
| * 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 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| * |
| *************************************************************/ |
| |
| |
| |
| // MARKER(update_precomp.py): autogen include statement, do not remove |
| #include "precompiled_forms.hxx" |
| #include "richtextcontrol.hxx" |
| #include "frm_module.hxx" |
| #ifndef _FRM_PROPERTY_HRC_ |
| #include "property.hrc" |
| #endif |
| #include "services.hxx" |
| |
| #include "richtextmodel.hxx" |
| #include "richtextvclcontrol.hxx" |
| #include "clipboarddispatcher.hxx" |
| #include "parametrizedattributedispatcher.hxx" |
| #include "specialdispatchers.hxx" |
| |
| /** === begin UNO includes === **/ |
| #include <com/sun/star/awt/PosSize.hpp> |
| /** === end UNO includes === **/ |
| |
| #include <toolkit/helper/vclunohelper.hxx> |
| #include <tools/diagnose_ex.h> |
| #include <vcl/svapp.hxx> |
| |
| #include <svx/svxids.hrc> |
| #include <editeng/editview.hxx> |
| #include <svl/itemset.hxx> |
| #include <svl/itempool.hxx> |
| #include <sfx2/msgpool.hxx> |
| |
| //-------------------------------------------------------------------------- |
| extern "C" void SAL_CALL createRegistryInfo_ORichTextControl() |
| { |
| static ::frm::OMultiInstanceAutoRegistration< ::frm::ORichTextControl > aAutoRegistration; |
| } |
| |
| //......................................................................... |
| namespace frm |
| { |
| //......................................................................... |
| |
| using namespace ::com::sun::star::uno; |
| using namespace ::com::sun::star::beans; |
| using namespace ::com::sun::star::awt; |
| using namespace ::com::sun::star::lang; |
| using namespace ::com::sun::star::frame; |
| |
| #define FORWARD_TO_PEER_1( unoInterface, method, param1 ) \ |
| Reference< unoInterface > xTypedPeer( getPeer(), UNO_QUERY ); \ |
| if ( xTypedPeer.is() ) \ |
| { \ |
| xTypedPeer->method( param1 ); \ |
| } |
| |
| #define FORWARD_TO_PEER_1_RET( returnType, unoInterface, method, param1 ) \ |
| returnType aReturn; \ |
| Reference< unoInterface > xTypedPeer( getPeer(), UNO_QUERY ); \ |
| if ( xTypedPeer.is() ) \ |
| { \ |
| aReturn = xTypedPeer->method( param1 ); \ |
| } \ |
| return aReturn; |
| |
| #define FORWARD_TO_PEER_3( unoInterface, method, param1, param2, param3 ) \ |
| Reference< unoInterface > xTypedPeer( getPeer(), UNO_QUERY ); \ |
| if ( xTypedPeer.is() ) \ |
| { \ |
| xTypedPeer->method( param1, param2, param3 ); \ |
| } |
| |
| #define FORWARD_TO_PEER_3_RET( returnType, unoInterface, method, param1, param2, param3 ) \ |
| returnType aReturn; \ |
| Reference< unoInterface > xTypedPeer( getPeer(), UNO_QUERY ); \ |
| if ( xTypedPeer.is() ) \ |
| { \ |
| aReturn = xTypedPeer->method( param1, param2, param3 ); \ |
| } \ |
| return aReturn; |
| |
| //================================================================== |
| // ORichTextControl |
| //================================================================== |
| DBG_NAME( ORichTextControl ) |
| //------------------------------------------------------------------ |
| ORichTextControl::ORichTextControl( const Reference< XMultiServiceFactory >& _rxORB ) |
| :UnoEditControl( _rxORB ) |
| { |
| DBG_CTOR( ORichTextControl, NULL ); |
| } |
| |
| //------------------------------------------------------------------ |
| ORichTextControl::~ORichTextControl() |
| { |
| DBG_DTOR( ORichTextControl, NULL ); |
| } |
| |
| //------------------------------------------------------------------ |
| IMPLEMENT_FORWARD_XTYPEPROVIDER2( ORichTextControl, UnoEditControl, ORichTextControl_Base ) |
| |
| //------------------------------------------------------------------ |
| Any SAL_CALL ORichTextControl::queryAggregation( const Type& _rType ) throw ( RuntimeException ) |
| { |
| Any aReturn = UnoEditControl::queryAggregation( _rType ); |
| |
| if ( !aReturn.hasValue() ) |
| aReturn = ORichTextControl_Base::queryInterface( _rType ); |
| |
| return aReturn; |
| } |
| |
| //------------------------------------------------------------------ |
| namespace |
| { |
| //.............................................................. |
| static void implAdjustTriStateFlag( const Reference< XPropertySet >& _rxProps, const ::rtl::OUString& _rPropertyName, |
| WinBits& _rAllBits, WinBits _nPositiveFlag, WinBits nNegativeFlag ) |
| { |
| sal_Bool bFlagValue = sal_False; |
| if ( _rxProps->getPropertyValue( _rPropertyName ) >>= bFlagValue ) |
| _rAllBits |= ( bFlagValue ? _nPositiveFlag : nNegativeFlag ); |
| } |
| |
| //.............................................................. |
| static void implAdjustTwoStateFlag( const Any& _rValue, WinBits& _rAllBits, WinBits _nFlag, bool _bInvert = false ) |
| { |
| sal_Bool bFlagValue = sal_False; |
| if ( _rValue >>= bFlagValue ) |
| { |
| if ( _bInvert ) |
| bFlagValue = !bFlagValue; |
| if ( bFlagValue ) |
| _rAllBits |= _nFlag; |
| else |
| _rAllBits &= ~_nFlag; |
| } |
| } |
| |
| //.............................................................. |
| static void implAdjustTwoStateFlag( const Reference< XPropertySet >& _rxProps, const ::rtl::OUString& _rPropertyName, |
| WinBits& _rAllBits, WinBits _nFlag, bool _bInvert = false ) |
| { |
| implAdjustTwoStateFlag( _rxProps->getPropertyValue( _rPropertyName ), _rAllBits, _nFlag, _bInvert ); |
| } |
| |
| //.............................................................. |
| static void adjustTwoStateWinBit( Window* _pWindow, const Any& _rValue, WinBits _nFlag, bool _bInvert = false ) |
| { |
| WinBits nBits = _pWindow->GetStyle(); |
| implAdjustTwoStateFlag( _rValue, nBits, _nFlag, _bInvert ); |
| _pWindow->SetStyle( nBits ); |
| } |
| |
| //.............................................................. |
| static WinBits getWinBits( const Reference< XControlModel >& _rxModel, WinBits nBaseBits = 0 ) |
| { |
| WinBits nBits = nBaseBits; |
| try |
| { |
| Reference< XPropertySet > xProps( _rxModel, UNO_QUERY ); |
| if ( xProps.is() ) |
| { |
| sal_Int16 nBorder = 0; |
| xProps->getPropertyValue( PROPERTY_BORDER ) >>= nBorder; |
| if ( nBorder ) |
| nBits |= WB_BORDER; |
| |
| implAdjustTriStateFlag( xProps, PROPERTY_TABSTOP, nBits, WB_TABSTOP, WB_NOTABSTOP ); |
| implAdjustTwoStateFlag( xProps, PROPERTY_HSCROLL, nBits, WB_HSCROLL ); |
| implAdjustTwoStateFlag( xProps, PROPERTY_VSCROLL, nBits, WB_VSCROLL ); |
| implAdjustTwoStateFlag( xProps, PROPERTY_HARDLINEBREAKS, nBits, WB_WORDBREAK, true ); |
| } |
| } |
| catch( const Exception& ) |
| { |
| DBG_UNHANDLED_EXCEPTION(); |
| } |
| return nBits; |
| } |
| } |
| |
| //------------------------------------------------------------------ |
| void SAL_CALL ORichTextControl::createPeer( const Reference< XToolkit >& _rToolkit, const Reference< XWindowPeer >& _rParentPeer ) throw( RuntimeException ) |
| { |
| sal_Bool bReallyActAsRichText = sal_False; |
| try |
| { |
| Reference< XPropertySet > xModelProps( getModel(), UNO_QUERY_THROW ); |
| xModelProps->getPropertyValue( PROPERTY_RICH_TEXT ) >>= bReallyActAsRichText; |
| } |
| catch( const Exception& ) |
| { |
| DBG_UNHANDLED_EXCEPTION(); |
| } |
| |
| if ( !bReallyActAsRichText ) |
| { |
| UnoEditControl::createPeer( _rToolkit, _rParentPeer ); |
| OControl::initFormControlPeer( getPeer() ); |
| return; |
| } |
| |
| ::vos::OGuard aGuard( Application::GetSolarMutex() ); |
| |
| if (!getPeer().is()) |
| { |
| mbCreatingPeer = sal_True; |
| |
| // determine the VLC window for the parent |
| Window* pParentWin = NULL; |
| if ( _rParentPeer.is() ) |
| { |
| VCLXWindow* pParentXWin = VCLXWindow::GetImplementation( _rParentPeer ); |
| if ( pParentXWin ) |
| pParentWin = pParentXWin->GetWindow(); |
| DBG_ASSERT( pParentWin, "ORichTextControl::createPeer: could not obtain the VCL-level parent window!" ); |
| } |
| |
| // create the peer |
| Reference< XControlModel > xModel( getModel() ); |
| ORichTextPeer* pPeer = ORichTextPeer::Create( xModel, pParentWin, getWinBits( xModel ) ); |
| DBG_ASSERT( pPeer, "ORichTextControl::createPeer: invalid peer returned!" ); |
| if ( pPeer ) |
| { |
| // by definition, the returned component is aquired once |
| pPeer->release(); |
| |
| // announce the peer to the base class |
| setPeer( pPeer ); |
| |
| // initialize ourself (and thus the peer) with the model properties |
| updateFromModel(); |
| |
| Reference< XView > xPeerView( getPeer(), UNO_QUERY ); |
| if ( xPeerView.is() ) |
| { |
| xPeerView->setZoom( maComponentInfos.nZoomX, maComponentInfos.nZoomY ); |
| xPeerView->setGraphics( mxGraphics ); |
| } |
| |
| // a lot of initial settings from our component infos |
| setPosSize( maComponentInfos.nX, maComponentInfos.nY, maComponentInfos.nWidth, maComponentInfos.nHeight, PosSize::POSSIZE ); |
| |
| pPeer->setVisible ( maComponentInfos.bVisible && !mbDesignMode ); |
| pPeer->setEnable ( maComponentInfos.bEnable ); |
| pPeer->setDesignMode( mbDesignMode ); |
| |
| peerCreated(); |
| } |
| |
| mbCreatingPeer = sal_False; |
| |
| OControl::initFormControlPeer( getPeer() ); |
| } |
| } |
| |
| //------------------------------------------------------------------ |
| ::rtl::OUString SAL_CALL ORichTextControl::getImplementationName() throw( RuntimeException ) |
| { |
| return getImplementationName_Static(); |
| } |
| |
| //------------------------------------------------------------------ |
| Sequence< ::rtl::OUString > SAL_CALL ORichTextControl::getSupportedServiceNames() throw( RuntimeException ) |
| { |
| return getSupportedServiceNames_Static(); |
| } |
| |
| //------------------------------------------------------------------ |
| ::rtl::OUString SAL_CALL ORichTextControl::getImplementationName_Static() |
| { |
| return ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.form.ORichTextControl" ) ); |
| } |
| |
| //------------------------------------------------------------------ |
| Sequence< ::rtl::OUString > SAL_CALL ORichTextControl::getSupportedServiceNames_Static() |
| { |
| Sequence< ::rtl::OUString > aServices( 3 ); |
| aServices[ 0 ] = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.awt.UnoControl" ) ); |
| aServices[ 1 ] = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.awt.UnoControlEdit" ) ); |
| aServices[ 2 ] = FRM_SUN_CONTROL_RICHTEXTCONTROL; |
| return aServices; |
| } |
| |
| //------------------------------------------------------------------ |
| Reference< XInterface > SAL_CALL ORichTextControl::Create( const Reference< XMultiServiceFactory >& _rxFactory ) |
| { |
| return *( new ORichTextControl( _rxFactory ) ); |
| } |
| |
| //-------------------------------------------------------------------- |
| Reference< XDispatch > SAL_CALL ORichTextControl::queryDispatch( const ::com::sun::star::util::URL& _rURL, const ::rtl::OUString& _rTargetFrameName, sal_Int32 _nSearchFlags ) throw (RuntimeException) |
| { |
| FORWARD_TO_PEER_3_RET( Reference< XDispatch >, XDispatchProvider, queryDispatch, _rURL, _rTargetFrameName, _nSearchFlags ); |
| } |
| |
| //-------------------------------------------------------------------- |
| Sequence< Reference< XDispatch > > SAL_CALL ORichTextControl::queryDispatches( const Sequence< DispatchDescriptor >& _rRequests ) throw (RuntimeException) |
| { |
| FORWARD_TO_PEER_1_RET( Sequence< Reference< XDispatch > >, XDispatchProvider, queryDispatches, _rRequests ); |
| } |
| |
| //-------------------------------------------------------------------- |
| sal_Bool ORichTextControl::requiresNewPeer( const ::rtl::OUString& _rPropertyName ) const |
| { |
| return UnoControl::requiresNewPeer( _rPropertyName ) || _rPropertyName.equals( PROPERTY_RICH_TEXT ); |
| } |
| |
| //================================================================== |
| // ORichTextPeer |
| //================================================================== |
| DBG_NAME( ORichTextPeer ) |
| //------------------------------------------------------------------ |
| ORichTextPeer* ORichTextPeer::Create( const Reference< XControlModel >& _rxModel, Window* _pParentWindow, WinBits _nStyle ) |
| { |
| DBG_TESTSOLARMUTEX(); |
| |
| // the EditEngine of the model |
| RichTextEngine* pEngine = ORichTextModel::getEditEngine( _rxModel ); |
| OSL_ENSURE( pEngine, "ORichTextPeer::Create: could not obtaine the edit engine from the model!" ); |
| if ( !pEngine ) |
| return NULL; |
| |
| // the peer itself |
| ORichTextPeer* pPeer = new ORichTextPeer; |
| pPeer->acquire(); // by definition, the returned object is aquired once |
| |
| // the VCL control for the peer |
| RichTextControl* pRichTextControl = new RichTextControl( pEngine, _pParentWindow, _nStyle, NULL, pPeer ); |
| |
| // some knittings |
| pRichTextControl->SetComponentInterface( pPeer ); |
| |
| // outta here |
| return pPeer; |
| } |
| |
| //------------------------------------------------------------------ |
| ORichTextPeer::ORichTextPeer() |
| { |
| DBG_CTOR( ORichTextPeer, NULL ); |
| } |
| |
| //------------------------------------------------------------------ |
| ORichTextPeer::~ORichTextPeer() |
| { |
| DBG_DTOR( ORichTextPeer, NULL ); |
| } |
| |
| //------------------------------------------------------------------ |
| void ORichTextPeer::dispose( ) throw(RuntimeException) |
| { |
| { |
| ::vos::OGuard aGuard( GetMutex() ); |
| RichTextControl* pRichTextControl = static_cast< RichTextControl* >( GetWindow() ); |
| |
| if ( pRichTextControl ) |
| { |
| for ( AttributeDispatchers::iterator aDisposeLoop = m_aDispatchers.begin(); |
| aDisposeLoop != m_aDispatchers.end(); |
| ++aDisposeLoop |
| ) |
| { |
| pRichTextControl->disableAttributeNotification( aDisposeLoop->first ); |
| aDisposeLoop->second->dispose(); |
| } |
| } |
| |
| AttributeDispatchers aEmpty; |
| m_aDispatchers.swap( aEmpty ); |
| } |
| |
| VCLXWindow::dispose(); |
| } |
| |
| //-------------------------------------------------------------------- |
| void SAL_CALL ORichTextPeer::draw( sal_Int32 _nX, sal_Int32 _nY ) throw(::com::sun::star::uno::RuntimeException) |
| { |
| ::vos::OGuard aGuard( Application::GetSolarMutex() ); |
| |
| RichTextControl* pControl = static_cast< RichTextControl* >( GetWindow() ); |
| if ( !pControl ) |
| return; |
| |
| OutputDevice* pTargetDevice = VCLUnoHelper::GetOutputDevice( getGraphics() ); |
| OSL_ENSURE( pTargetDevice != NULL, "ORichTextPeer::draw: no graphics -> no drawing!" ); |
| if ( !pTargetDevice ) |
| return; |
| |
| ::Size aSize = pControl->GetSizePixel(); |
| const MapUnit eTargetUnit = pTargetDevice->GetMapMode().GetMapUnit(); |
| if ( eTargetUnit != MAP_PIXEL ) |
| aSize = pControl->PixelToLogic( aSize, eTargetUnit ); |
| |
| ::Point aPos( _nX, _nY ); |
| // the XView::draw API talks about pixels, always ... |
| if ( eTargetUnit != MAP_PIXEL ) |
| aPos = pTargetDevice->PixelToLogic( aPos ); |
| |
| pControl->Draw( pTargetDevice, aPos, aSize, WINDOW_DRAW_NOCONTROLS ); |
| } |
| |
| //-------------------------------------------------------------------- |
| void SAL_CALL ORichTextPeer::setProperty( const ::rtl::OUString& _rPropertyName, const Any& _rValue ) throw (RuntimeException) |
| { |
| if ( !GetWindow() ) |
| { |
| VCLXWindow::setProperty( _rPropertyName, _rValue ); |
| return; |
| } |
| |
| if ( _rPropertyName.equals( PROPERTY_BACKGROUNDCOLOR ) ) |
| { |
| RichTextControl* pControl = static_cast< RichTextControl* >( GetWindow() ); |
| if ( !_rValue.hasValue() ) |
| { |
| pControl->SetBackgroundColor( ); |
| } |
| else |
| { |
| sal_Int32 nColor = COL_TRANSPARENT; |
| _rValue >>= nColor; |
| pControl->SetBackgroundColor( Color( nColor ) ); |
| } |
| } |
| else if ( _rPropertyName.equals( PROPERTY_HSCROLL ) ) |
| { |
| adjustTwoStateWinBit( GetWindow(), _rValue, WB_HSCROLL ); |
| } |
| else if ( _rPropertyName.equals( PROPERTY_VSCROLL ) ) |
| { |
| adjustTwoStateWinBit( GetWindow(), _rValue, WB_VSCROLL ); |
| } |
| else if ( _rPropertyName.equals( PROPERTY_HARDLINEBREAKS ) ) |
| { |
| adjustTwoStateWinBit( GetWindow(), _rValue, WB_WORDBREAK, true ); |
| } |
| else if ( _rPropertyName.equals( PROPERTY_READONLY ) ) |
| { |
| RichTextControl* pControl = static_cast< RichTextControl* >( GetWindow() ); |
| sal_Bool bReadOnly( pControl->IsReadOnly() ); |
| OSL_VERIFY( _rValue >>= bReadOnly ); |
| pControl->SetReadOnly( bReadOnly ); |
| |
| // update the dispatchers |
| for ( AttributeDispatchers::iterator aDispatcherLoop = m_aDispatchers.begin(); |
| aDispatcherLoop != m_aDispatchers.end(); |
| ++aDispatcherLoop |
| ) |
| { |
| aDispatcherLoop->second->invalidate(); |
| } |
| } |
| else if ( _rPropertyName.equals( PROPERTY_HIDEINACTIVESELECTION ) ) |
| { |
| RichTextControl* pRichTextControl = static_cast< RichTextControl* >( GetWindow() ); |
| sal_Bool bHide = pRichTextControl->GetHideInactiveSelection(); |
| OSL_VERIFY( _rValue >>= bHide ); |
| pRichTextControl->SetHideInactiveSelection( bHide ); |
| } |
| else |
| VCLXWindow::setProperty( _rPropertyName, _rValue ); |
| } |
| |
| //------------------------------------------------------------------ |
| IMPLEMENT_FORWARD_XINTERFACE2( ORichTextPeer, VCLXWindow, ORichTextPeer_Base ) |
| |
| //------------------------------------------------------------------ |
| IMPLEMENT_FORWARD_XTYPEPROVIDER2( ORichTextPeer, VCLXWindow, ORichTextPeer_Base ) |
| |
| //-------------------------------------------------------------------- |
| namespace |
| { |
| static SfxSlotId lcl_translateConflictingSlot( SfxSlotId _nIDFromPool ) |
| { |
| // HACK HACK HACK |
| // unfortunately, some of our applications have some conflicting slots, |
| // i.e. slots which have the same UNO name as an existing other (common) |
| // slot. |
| // For instance, both the slots SID_SET_SUPER_SCRIPT (from SVX) and FN_SET_SUPER_SCRIPT |
| // (from SW) have the UNO name "SuperScript". |
| // Now, if the controls lives in a text document, and asks the SfxSlotPool for |
| // the id belonging to "SuperScript", it gets the FN_SET_SUPER_SCRIPT - which |
| // is completely unknown to the EditEngine. |
| // So, we need to translate such conflicting ids. |
| // |
| // Note that the real solution would be to fix the applications to |
| // *not* define conflicting slots. Alternatively, if SFX would provide a slot pool |
| // which is *static* (i.e. independent on the active application), then we |
| // would also never encounter such a conflict. |
| SfxSlotId nReturn( _nIDFromPool ); |
| switch ( _nIDFromPool ) |
| { |
| case 20411: /* FM_SET_SUPER_SCRIPT, originating in SW */ |
| nReturn = SID_SET_SUPER_SCRIPT; |
| break; |
| case 20412: /* FN_SET_SUB_SCRIPT, originating in SW */ |
| nReturn = SID_SET_SUB_SCRIPT; |
| break; |
| } |
| return nReturn; |
| } |
| } |
| |
| //-------------------------------------------------------------------- |
| ORichTextPeer::SingleAttributeDispatcher ORichTextPeer::implCreateDispatcher( SfxSlotId _nSlotId, const ::com::sun::star::util::URL& _rURL ) |
| { |
| RichTextControl* pRichTextControl = static_cast< RichTextControl* >( GetWindow() ); |
| OSL_PRECOND( pRichTextControl, "ORichTextPeer::implCreateDispatcher: invalid window!" ); |
| if ( !pRichTextControl ) |
| return SingleAttributeDispatcher( NULL ); |
| |
| ORichTextFeatureDispatcher* pDispatcher = NULL; |
| OAttributeDispatcher* pAttributeDispatcher = NULL; |
| switch ( _nSlotId ) |
| { |
| case SID_CUT: |
| pDispatcher = new OClipboardDispatcher( pRichTextControl->getView(), OClipboardDispatcher::eCut ); |
| break; |
| |
| case SID_COPY: |
| pDispatcher = new OClipboardDispatcher( pRichTextControl->getView(), OClipboardDispatcher::eCopy ); |
| break; |
| |
| case SID_PASTE: |
| pDispatcher = new OPasteClipboardDispatcher( pRichTextControl->getView() ); |
| break; |
| |
| case SID_SELECTALL: |
| pDispatcher = new OSelectAllDispatcher( pRichTextControl->getView(), _rURL ); |
| break; |
| |
| case SID_ATTR_PARA_LEFT_TO_RIGHT: |
| case SID_ATTR_PARA_RIGHT_TO_LEFT: |
| pAttributeDispatcher = new OParagraphDirectionDispatcher( pRichTextControl->getView(), _nSlotId, _rURL, pRichTextControl ); |
| break; |
| |
| case SID_TEXTDIRECTION_TOP_TO_BOTTOM: |
| case SID_TEXTDIRECTION_LEFT_TO_RIGHT: |
| pDispatcher = new OTextDirectionDispatcher( pRichTextControl->getView(), _rURL ); |
| break; |
| |
| case SID_ATTR_PARA_HANGPUNCTUATION: |
| case SID_ATTR_PARA_FORBIDDEN_RULES: |
| case SID_ATTR_PARA_SCRIPTSPACE: |
| pAttributeDispatcher = new OAsianFontLayoutDispatcher( pRichTextControl->getView(), _nSlotId, _rURL, pRichTextControl ); |
| break; |
| |
| default: |
| { |
| // is it a supported slot? |
| bool bSupportedSlot = false; |
| if ( !bSupportedSlot ) |
| { |
| const SfxItemPool& rPool = *pRichTextControl->getView().GetEmptyItemSet().GetPool(); |
| bSupportedSlot = rPool.IsInRange( rPool.GetWhich( _nSlotId ) ); |
| } |
| if ( !bSupportedSlot ) |
| bSupportedSlot = RichTextControl::isMappableSlot( _nSlotId ); |
| |
| if ( bSupportedSlot ) |
| { // it's really a slot which is supported by the EditEngine |
| |
| bool bNeedParametrizedDispatcher = true; |
| if ( ( _nSlotId == SID_ATTR_CHAR_POSTURE ) |
| || ( _nSlotId == SID_ATTR_CHAR_CJK_POSTURE ) |
| || ( _nSlotId == SID_ATTR_CHAR_CTL_POSTURE ) |
| || ( _nSlotId == SID_ATTR_CHAR_LATIN_POSTURE ) |
| || ( _nSlotId == SID_ATTR_CHAR_WEIGHT ) |
| || ( _nSlotId == SID_ATTR_CHAR_CJK_WEIGHT ) |
| || ( _nSlotId == SID_ATTR_CHAR_CTL_WEIGHT ) |
| || ( _nSlotId == SID_ATTR_CHAR_LATIN_WEIGHT ) |
| || ( _nSlotId == SID_ATTR_CHAR_LANGUAGE ) |
| || ( _nSlotId == SID_ATTR_CHAR_CJK_LANGUAGE ) |
| || ( _nSlotId == SID_ATTR_CHAR_CTL_LANGUAGE ) |
| || ( _nSlotId == SID_ATTR_CHAR_LATIN_LANGUAGE ) |
| || ( _nSlotId == SID_ATTR_CHAR_CONTOUR ) |
| || ( _nSlotId == SID_ATTR_CHAR_SHADOWED ) |
| || ( _nSlotId == SID_ATTR_CHAR_WORDLINEMODE ) |
| || ( _nSlotId == SID_ATTR_CHAR_COLOR ) |
| || ( _nSlotId == SID_ATTR_CHAR_RELIEF ) |
| || ( _nSlotId == SID_ATTR_CHAR_KERNING ) |
| || ( _nSlotId == SID_ATTR_CHAR_AUTOKERN ) |
| || ( _nSlotId == SID_ATTR_CHAR_SCALEWIDTH ) |
| ) |
| { |
| bNeedParametrizedDispatcher = true; |
| } |
| else if ( ( _nSlotId == SID_ATTR_PARA_HANGPUNCTUATION ) |
| || ( _nSlotId == SID_ATTR_PARA_FORBIDDEN_RULES ) |
| || ( _nSlotId == SID_ATTR_PARA_SCRIPTSPACE ) |
| ) |
| { |
| bNeedParametrizedDispatcher = false; |
| } |
| else |
| { |
| SfxSlotPool& rSlotPool = SfxSlotPool::GetSlotPool( NULL ); |
| const SfxSlot* pSlot = rSlotPool.GetSlot( _nSlotId ); |
| const SfxType* pType = pSlot ? pSlot->GetType() : NULL; |
| if ( pType ) |
| { |
| bNeedParametrizedDispatcher = ( pType->nAttribs > 0 ); |
| } |
| } |
| |
| if ( bNeedParametrizedDispatcher ) |
| { |
| #if OSL_DEBUG_LEVEL > 0 |
| ::rtl::OString sTrace( "ORichTextPeer::implCreateDispatcher: creating *parametrized* dispatcher for " ); |
| sTrace += ::rtl::OString( _rURL.Complete.getStr(), _rURL.Complete.getLength(), RTL_TEXTENCODING_ASCII_US ); |
| DBG_TRACE( sTrace.getStr() ); |
| #endif |
| pAttributeDispatcher = new OParametrizedAttributeDispatcher( pRichTextControl->getView(), _nSlotId, _rURL, pRichTextControl ); |
| } |
| else |
| { |
| #if OSL_DEBUG_LEVEL > 0 |
| ::rtl::OString sTrace( "ORichTextPeer::implCreateDispatcher: creating *normal* dispatcher for " ); |
| sTrace += ::rtl::OString( _rURL.Complete.getStr(), _rURL.Complete.getLength(), RTL_TEXTENCODING_ASCII_US ); |
| DBG_TRACE( sTrace.getStr() ); |
| #endif |
| pAttributeDispatcher = new OAttributeDispatcher( pRichTextControl->getView(), _nSlotId, _rURL, pRichTextControl ); |
| } |
| } |
| #if OSL_DEBUG_LEVEL > 0 |
| else |
| { |
| ::rtl::OString sTrace( "ORichTextPeer::implCreateDispatcher: not creating dispatcher (unsupported slot) for " ); |
| sTrace += ::rtl::OString( _rURL.Complete.getStr(), _rURL.Complete.getLength(), RTL_TEXTENCODING_ASCII_US ); |
| DBG_TRACE( sTrace.getStr() ); |
| } |
| #endif |
| } |
| break; |
| } |
| |
| SingleAttributeDispatcher xDispatcher( pDispatcher ); |
| if ( pAttributeDispatcher ) |
| { |
| xDispatcher = SingleAttributeDispatcher( pAttributeDispatcher ); |
| pRichTextControl->enableAttributeNotification( _nSlotId, pAttributeDispatcher ); |
| } |
| |
| return xDispatcher; |
| } |
| |
| //-------------------------------------------------------------------- |
| namespace |
| { |
| SfxSlotId lcl_getSlotFromUnoName( SfxSlotPool& _rSlotPool, const ::rtl::OUString& _rUnoSlotName ) |
| { |
| const SfxSlot* pSlot = _rSlotPool.GetUnoSlot( _rUnoSlotName ); |
| if ( pSlot ) |
| { |
| // okay, there's a slot with the given UNO name |
| return lcl_translateConflictingSlot( pSlot->GetSlotId() ); |
| } |
| |
| // some hard-coded slots, which do not have a UNO name at SFX level, but which |
| // we nevertheless need to transport via UNO mechanisms, so we need a name |
| if ( _rUnoSlotName.equalsAscii( "AllowHangingPunctuation" ) ) |
| return SID_ATTR_PARA_HANGPUNCTUATION; |
| if ( _rUnoSlotName.equalsAscii( "ApplyForbiddenCharacterRules" ) ) |
| return SID_ATTR_PARA_FORBIDDEN_RULES; |
| if ( _rUnoSlotName.equalsAscii( "UseScriptSpacing" ) ) |
| return SID_ATTR_PARA_SCRIPTSPACE; |
| |
| OSL_ENSURE( pSlot, "lcl_getSlotFromUnoName: unknown UNO slot name!" ); |
| return 0; |
| } |
| } |
| |
| //-------------------------------------------------------------------- |
| Reference< XDispatch > SAL_CALL ORichTextPeer::queryDispatch( const ::com::sun::star::util::URL& _rURL, const ::rtl::OUString& /*_rTargetFrameName*/, sal_Int32 /*_nSearchFlags*/ ) throw (RuntimeException) |
| { |
| Reference< XDispatch > xReturn; |
| if ( !GetWindow() ) |
| { |
| OSL_ENSURE( sal_False, "ORichTextPeer::queryDispatch: already disposed?" ); |
| return xReturn; |
| } |
| |
| // is it an UNO slot? |
| ::rtl::OUString sUnoProtocolPrefix( RTL_CONSTASCII_USTRINGPARAM( ".uno:" ) ); |
| if ( 0 == _rURL.Complete.compareTo( sUnoProtocolPrefix, sUnoProtocolPrefix.getLength() ) ) |
| { |
| ::rtl::OUString sUnoSlotName = _rURL.Complete.copy( sUnoProtocolPrefix.getLength() ); |
| SfxSlotId nSlotId = lcl_getSlotFromUnoName( SfxSlotPool::GetSlotPool( NULL ), sUnoSlotName ); |
| if ( nSlotId > 0 ) |
| { |
| // do we already have a dispatcher for this? |
| AttributeDispatchers::const_iterator aDispatcherPos = m_aDispatchers.find( nSlotId ); |
| if ( aDispatcherPos == m_aDispatchers.end() ) |
| { |
| SingleAttributeDispatcher pDispatcher = implCreateDispatcher( nSlotId, _rURL ); |
| if ( pDispatcher.is() ) |
| { |
| aDispatcherPos = m_aDispatchers.insert( AttributeDispatchers::value_type( nSlotId, pDispatcher ) ).first; |
| } |
| } |
| |
| if ( aDispatcherPos != m_aDispatchers.end() ) |
| xReturn = aDispatcherPos->second.getRef(); |
| } |
| } |
| |
| return xReturn; |
| } |
| |
| //-------------------------------------------------------------------- |
| Sequence< Reference< XDispatch > > SAL_CALL ORichTextPeer::queryDispatches( const Sequence< DispatchDescriptor >& _rRequests ) throw (RuntimeException) |
| { |
| Sequence< Reference< XDispatch > > aReturn( _rRequests.getLength() ); |
| Reference< XDispatch >* pReturn = aReturn.getArray(); |
| |
| const DispatchDescriptor* pRequest = _rRequests.getConstArray(); |
| const DispatchDescriptor* pRequestEnd = pRequest + _rRequests.getLength(); |
| for ( ; pRequest != pRequestEnd; ++pRequest, ++pReturn ) |
| { |
| *pReturn = queryDispatch( pRequest->FeatureURL, pRequest->FrameName, pRequest->SearchFlags ); |
| } |
| return aReturn; |
| } |
| |
| //-------------------------------------------------------------------- |
| void ORichTextPeer::onSelectionChanged( const ESelection& /*_rSelection*/ ) |
| { |
| AttributeDispatchers::iterator aDispatcherPos = m_aDispatchers.find( SID_COPY ); |
| if ( aDispatcherPos != m_aDispatchers.end() ) |
| aDispatcherPos->second.get()->invalidate(); |
| |
| aDispatcherPos = m_aDispatchers.find( SID_CUT ); |
| if ( aDispatcherPos != m_aDispatchers.end() ) |
| aDispatcherPos->second.get()->invalidate(); |
| } |
| |
| //........................................................................ |
| } // namespace frm |
| //........................................................................ |
| |