/************************************************************** * * 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_sw.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include "fesh.hxx" #include #include "accmap.hxx" #include "accframebase.hxx" #ifndef _CRSRSH_HXX #include #endif #ifndef _FESH_HXX #include "fesh.hxx" #endif #ifndef _TXTFRM_HXX #include #endif #ifndef _NDTXT_HXX #include #endif #ifndef _DCONTACT_HXX #include #endif #ifndef _FMTANCHR_HXX #include #endif using namespace ::com::sun::star; using namespace ::com::sun::star::accessibility; using ::rtl::OUString; sal_Bool SwAccessibleFrameBase::IsSelected() { sal_Bool bRet = sal_False; DBG_ASSERT( GetMap(), "no map?" ); const ViewShell *pVSh = GetMap()->GetShell(); DBG_ASSERT( pVSh, "no shell?" ); if( pVSh->ISA( SwFEShell ) ) { const SwFEShell *pFESh = static_cast< const SwFEShell * >( pVSh ); const SwFrm *pFlyFrm = pFESh->GetCurrFlyFrm(); if( pFlyFrm == GetFrm() ) bRet = sal_True; } return bRet; } void SwAccessibleFrameBase::GetStates( ::utl::AccessibleStateSetHelper& rStateSet ) { SwAccessibleContext::GetStates( rStateSet ); const ViewShell *pVSh = GetMap()->GetShell(); DBG_ASSERT( pVSh, "no shell?" ); sal_Bool bSelectable = pVSh->ISA( SwFEShell ); // SELECTABLE if( bSelectable ) rStateSet.AddState( AccessibleStateType::SELECTABLE ); // FOCUSABLE if( bSelectable ) rStateSet.AddState( AccessibleStateType::FOCUSABLE ); // SELECTED and FOCUSED if( IsSelected() ) { rStateSet.AddState( AccessibleStateType::SELECTED ); ASSERT( bIsSelected, "bSelected out of sync" ); ::vos::ORef < SwAccessibleContext > xThis( this ); GetMap()->SetCursorContext( xThis ); Window *pWin = GetWindow(); if( pWin && pWin->HasFocus() ) rStateSet.AddState( AccessibleStateType::FOCUSED ); } if( GetSelectedState() ) rStateSet.AddState( AccessibleStateType::SELECTED ); } sal_uInt8 SwAccessibleFrameBase::GetNodeType( const SwFlyFrm *pFlyFrm ) { sal_uInt8 nType = ND_TEXTNODE; if( pFlyFrm->Lower() ) { if( pFlyFrm->Lower()->IsNoTxtFrm() ) { const SwCntntFrm *pCntFrm = static_cast( pFlyFrm->Lower() ); nType = pCntFrm->GetNode()->GetNodeType(); } } else { const SwFrmFmt *pFrmFmt = pFlyFrm->GetFmt(); const SwFmtCntnt& rCntnt = pFrmFmt->GetCntnt(); const SwNodeIndex *pNdIdx = rCntnt.GetCntntIdx(); if( pNdIdx ) { const SwCntntNode *pCNd = (pNdIdx->GetNodes())[pNdIdx->GetIndex()+1]->GetCntntNode(); if( pCNd ) nType = pCNd->GetNodeType(); } } return nType; } SwAccessibleFrameBase::SwAccessibleFrameBase( SwAccessibleMap* pInitMap, sal_Int16 nInitRole, const SwFlyFrm* pFlyFrm ) : SwAccessibleContext( pInitMap, nInitRole, pFlyFrm ), bIsSelected( sal_False ) { vos::OGuard aGuard(Application::GetSolarMutex()); const SwFrmFmt *pFrmFmt = pFlyFrm->GetFmt(); const_cast< SwFrmFmt * >( pFrmFmt )->Add( this ); SetName( pFrmFmt->GetName() ); bIsSelected = IsSelected(); } void SwAccessibleFrameBase::_InvalidateCursorPos() { sal_Bool bNewSelected = IsSelected(); sal_Bool bOldSelected; { vos::OGuard aGuard( aMutex ); bOldSelected = bIsSelected; bIsSelected = bNewSelected; } if( bNewSelected ) { // remember that object as the one that has the caret. This is // neccessary to notify that object if the cursor leaves it. ::vos::ORef < SwAccessibleContext > xThis( this ); GetMap()->SetCursorContext( xThis ); } if( bOldSelected != bNewSelected ) { Window *pWin = GetWindow(); if( pWin && pWin->HasFocus() && bNewSelected ) FireStateChangedEvent( AccessibleStateType::FOCUSED, bNewSelected ); //FireStateChangedEvent( AccessibleStateType::SELECTED, bNewSelected ); if( pWin && pWin->HasFocus() && !bNewSelected ) FireStateChangedEvent( AccessibleStateType::FOCUSED, bNewSelected ); if(bNewSelected) { uno::Reference< XAccessible > xParent( GetWeakParent() ); if( xParent.is() ) { SwAccessibleContext *pAcc = static_cast ( xParent.get() ); AccessibleEventObject aEvent; aEvent.EventId = AccessibleEventId::SELECTION_CHANGED; uno::Reference< XAccessible > xChild(this); aEvent.NewValue <<= xChild; pAcc->FireAccessibleEvent( aEvent ); } } } } void SwAccessibleFrameBase::_InvalidateFocus() { Window *pWin = GetWindow(); if( pWin ) { sal_Bool bSelected; { vos::OGuard aGuard( aMutex ); bSelected = bIsSelected; } ASSERT( bSelected, "focus object should be selected" ); FireStateChangedEvent( AccessibleStateType::FOCUSED, pWin->HasFocus() && bSelected ); } } sal_Bool SwAccessibleFrameBase::HasCursor() { vos::OGuard aGuard( aMutex ); return bIsSelected; } SwAccessibleFrameBase::~SwAccessibleFrameBase() { } void SwAccessibleFrameBase::Modify( const SfxPoolItem* pOld, const SfxPoolItem *pNew) { sal_uInt16 nWhich = pOld ? pOld->Which() : pNew ? pNew->Which() : 0 ; const SwFlyFrm *pFlyFrm = static_cast< const SwFlyFrm * >( GetFrm() ); switch( nWhich ) { case RES_NAME_CHANGED: if( pFlyFrm ) { const SwFrmFmt *pFrmFmt = pFlyFrm->GetFmt(); ASSERT( pFrmFmt == GetRegisteredIn(), "invalid frame" ); OUString sOldName( GetName() ); ASSERT( !pOld || static_cast < const SwStringMsgPoolItem * >( pOld )->GetString() == String( sOldName ), "invalid old name" ); const String& rNewName = pFrmFmt->GetName(); SetName( rNewName ); ASSERT( !pNew || static_cast < const SwStringMsgPoolItem * >( pNew )->GetString() == rNewName, "invalid new name" ); if( sOldName != GetName() ) { AccessibleEventObject aEvent; aEvent.EventId = AccessibleEventId::NAME_CHANGED; aEvent.OldValue <<= sOldName; aEvent.NewValue <<= GetName(); FireAccessibleEvent( aEvent ); } } break; case RES_OBJECTDYING: // mba: it seems that this class intentionally does not call code in base class SwClient if( pOld && ( GetRegisteredIn() == static_cast< SwModify *>( static_cast< const SwPtrMsgPoolItem * >( pOld )->pObject ) ) ) GetRegisteredInNonConst()->Remove( this ); break; case RES_FMT_CHG: if( pOld && static_cast< const SwFmtChg * >(pNew)->pChangedFmt == GetRegisteredIn() && static_cast< const SwFmtChg * >(pOld)->pChangedFmt->IsFmtInDTOR() ) GetRegisteredInNonConst()->Remove( this ); break; default: // mba: former call to base class method removed as it is meant to handle only RES_OBJECTDYING break; } } void SwAccessibleFrameBase::Dispose( sal_Bool bRecursive ) { vos::OGuard aGuard(Application::GetSolarMutex()); if( GetRegisteredIn() ) GetRegisteredInNonConst()->Remove( this ); SwAccessibleContext::Dispose( bRecursive ); } //Get the selection cursor of the document. SwPaM* SwAccessibleFrameBase::GetCrsr() { // get the cursor shell; if we don't have any, we don't have a // cursor/selection either SwPaM* pCrsr = NULL; SwCrsrShell* pCrsrShell = GetCrsrShell(); if( pCrsrShell != NULL && !pCrsrShell->IsTableMode() ) { SwFEShell *pFESh = pCrsrShell->ISA( SwFEShell ) ? static_cast< SwFEShell * >( pCrsrShell ) : 0; if( !pFESh || !(pFESh->IsFrmSelected() || pFESh->IsObjSelected() > 0) ) { // get the selection, and test whether it affects our text node pCrsr = pCrsrShell->GetCrsr( sal_False /* ??? */ ); } } return pCrsr; } //Return the selected state of the object. //when the object's anchor are in the selection cursor, we should return true. sal_Bool SwAccessibleFrameBase::GetSelectedState( ) { vos::OGuard aGuard(Application::GetSolarMutex()); if(GetMap()->IsDocumentSelAll()) { return sal_True; } // SELETED. SwFlyFrm* pFlyFrm = getFlyFrm(); const SwFrmFmt *pFrmFmt = pFlyFrm->GetFmt(); const SwFmtAnchor& pAnchor = pFrmFmt->GetAnchor(); const SwPosition *pPos = pAnchor.GetCntntAnchor(); if( !pPos ) return sal_False; int pIndex = pPos->nContent.GetIndex(); if( pPos->nNode.GetNode().GetTxtNode() ) { SwPaM* pCrsr = GetCrsr(); if( pCrsr != NULL ) { const SwTxtNode* pNode = pPos->nNode.GetNode().GetTxtNode(); sal_uLong nHere = pNode->GetIndex(); // iterate over ring SwPaM* pRingStart = pCrsr; do { // ignore, if no mark if( pCrsr->HasMark() ) { // check whether nHere is 'inside' pCrsr SwPosition* pStart = pCrsr->Start(); sal_uLong nStartIndex = pStart->nNode.GetIndex(); SwPosition* pEnd = pCrsr->End(); sal_uLong nEndIndex = pEnd->nNode.GetIndex(); if( ( nHere >= nStartIndex ) && (nHere <= nEndIndex) ) { if( pAnchor.GetAnchorId() == FLY_AS_CHAR ) { if( (nHere == nStartIndex) && (pIndex >= pStart->nContent.GetIndex()) || (nHere > nStartIndex) ) if( (nHere == nEndIndex) && (pIndex < pEnd->nContent.GetIndex()) || (nHere < nEndIndex) ) return sal_True; } else if( pAnchor.GetAnchorId() == FLY_AT_PARA ) { if( ((nHere > nStartIndex) || pStart->nContent.GetIndex() ==0 ) && (nHere < nEndIndex ) ) return sal_True; } break; } // else: this PaM doesn't point to this paragraph } // else: this PaM is collapsed and doesn't select anything // next PaM in ring pCrsr = static_cast( pCrsr->GetNext() ); } while( pCrsr != pRingStart ); } } return sal_False; } SwFlyFrm* SwAccessibleFrameBase::getFlyFrm() const { SwFlyFrm* pFlyFrm = NULL; const SwFrm* pFrm = GetFrm(); DBG_ASSERT( pFrm != NULL, "frame expected" ); if( pFrm->IsFlyFrm() ) { pFlyFrm = static_cast( const_cast( pFrm ) ); } return pFlyFrm; } sal_Bool SwAccessibleFrameBase::SetSelectedState( sal_Bool ) { sal_Bool bParaSeleted = GetSelectedState() || IsSelected(); if(bIsSeletedInDoc != bParaSeleted) { bIsSeletedInDoc = bParaSeleted; FireStateChangedEvent( AccessibleStateType::SELECTED, bParaSeleted ); return sal_True; } return sal_False; }