/************************************************************** * * 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_sc.hxx" #include "AccessibleEditObject.hxx" #include "scitems.hxx" #include #include "unoguard.hxx" #include "AccessibleText.hxx" #include "editsrc.hxx" #include "scmod.hxx" #include "inputhdl.hxx" #ifndef _UTL_ACCESSIBLESTATESETHELPER_HXX #include #endif #ifndef _COM_SUN_STAR_ACCESSIBILITY_XACCESSIBLEROLE_HPP_ #include #endif #ifndef _COM_SUN_STAR_ACCESSIBILITY_XACCESSIBLESTATETYPE_HPP_ #include #endif #include #include #include #include #include #include #include #include #include #include "unonames.hxx" #include "document.hxx" #include "AccessibleDocument.hxx" #include #include #include using ::com::sun::star::lang::IndexOutOfBoundsException; using ::com::sun::star::uno::RuntimeException; using namespace ::com::sun::star; using namespace ::com::sun::star::accessibility; //===== internal ============================================================ ScAccessibleEditObject::ScAccessibleEditObject( const uno::Reference& rxParent, EditView* pEditView, Window* pWin, const rtl::OUString& rName, const rtl::OUString& rDescription, EditObjectType eObjectType) : ScAccessibleContextBase(rxParent, AccessibleRole::TEXT_FRAME), mpTextHelper(NULL), mpEditView(pEditView), mpWindow(pWin), meObjectType(eObjectType), mbHasFocus(sal_False) { CreateTextHelper(); SetName(rName); SetDescription(rDescription); if( meObjectType == CellInEditMode) { const ScAccessibleDocument *pAccDoc = const_cast(static_cast(rxParent.get())) ; if (pAccDoc) { m_pScDoc = pAccDoc->GetDocument(); m_curCellAddress =pAccDoc->GetCurCellAddress(); } else { m_pScDoc=NULL; } } else m_pScDoc=NULL; } ScAccessibleEditObject::~ScAccessibleEditObject() { if (!ScAccessibleContextBase::IsDefunc() && !rBHelper.bInDispose) { // increment refcount to prevent double call off dtor osl_incrementInterlockedCount( &m_refCount ); // call dispose to inform object wich have a weak reference to this object dispose(); } } void SAL_CALL ScAccessibleEditObject::disposing() { ScUnoGuard aGuard; if (mpTextHelper) DELETEZ(mpTextHelper); ScAccessibleContextBase::disposing(); } void ScAccessibleEditObject::LostFocus() { mbHasFocus = sal_False; if (mpTextHelper) mpTextHelper->SetFocus(sal_False); CommitFocusLost(); } void ScAccessibleEditObject::GotFocus() { mbHasFocus = sal_True; CommitFocusGained(); if (mpTextHelper) mpTextHelper->SetFocus(sal_True); } //===== XInterface ========================================================== com::sun::star::uno::Any SAL_CALL ScAccessibleEditObject::queryInterface (const com::sun::star::uno::Type & rType) throw (::com::sun::star::uno::RuntimeException) { ::com::sun::star::uno::Any aReturn = ScAccessibleContextBase::queryInterface (rType); if ( ! aReturn.hasValue()) aReturn = ::cppu::queryInterface (rType, static_cast< ::com::sun::star::accessibility::XAccessibleSelection* >(this) ); return aReturn; } void SAL_CALL ScAccessibleEditObject::acquire (void) throw () { ScAccessibleContextBase::acquire (); } void SAL_CALL ScAccessibleEditObject::release (void) throw () { ScAccessibleContextBase::release (); } //===== XAccessibleComponent ============================================ uno::Reference< XAccessible > SAL_CALL ScAccessibleEditObject::getAccessibleAtPoint( const awt::Point& rPoint ) throw (uno::RuntimeException) { uno::Reference xRet; if (containsPoint(rPoint)) { ScUnoGuard aGuard; IsObjectValid(); if(!mpTextHelper) CreateTextHelper(); xRet = mpTextHelper->GetAt(rPoint); } return xRet; } Rectangle ScAccessibleEditObject::GetBoundingBoxOnScreen(void) const throw (uno::RuntimeException) { Rectangle aScreenBounds; if ( mpWindow ) { if ( meObjectType == CellInEditMode ) { if ( mpEditView && mpEditView->GetEditEngine() ) { MapMode aMapMode( mpEditView->GetEditEngine()->GetRefMapMode() ); aScreenBounds = mpWindow->LogicToPixel( mpEditView->GetOutputArea(), aMapMode ); Point aCellLoc = aScreenBounds.TopLeft(); Rectangle aWindowRect = mpWindow->GetWindowExtentsRelative( NULL ); Point aWindowLoc = aWindowRect.TopLeft(); Point aPos( aCellLoc.getX() + aWindowLoc.getX(), aCellLoc.getY() + aWindowLoc.getY() ); aScreenBounds.SetPos( aPos ); } } else { aScreenBounds = mpWindow->GetWindowExtentsRelative( NULL ); } } return aScreenBounds; } Rectangle ScAccessibleEditObject::GetBoundingBox(void) const throw (uno::RuntimeException) { Rectangle aBounds( GetBoundingBoxOnScreen() ); if ( mpWindow ) { uno::Reference< XAccessible > xThis( mpWindow->GetAccessible() ); if ( xThis.is() ) { uno::Reference< XAccessibleContext > xContext( xThis->getAccessibleContext() ); if ( xContext.is() ) { uno::Reference< XAccessible > xParent( xContext->getAccessibleParent() ); if ( xParent.is() ) { uno::Reference< XAccessibleComponent > xParentComponent( xParent->getAccessibleContext(), uno::UNO_QUERY ); if ( xParentComponent.is() ) { Point aScreenLoc = aBounds.TopLeft(); awt::Point aParentScreenLoc = xParentComponent->getLocationOnScreen(); Point aPos( aScreenLoc.getX() - aParentScreenLoc.X, aScreenLoc.getY() - aParentScreenLoc.Y ); aBounds.SetPos( aPos ); } } } } } return aBounds; } //===== XAccessibleContext ============================================== sal_Int32 SAL_CALL ScAccessibleEditObject::getAccessibleChildCount(void) throw (uno::RuntimeException) { ScUnoGuard aGuard; IsObjectValid(); if (!mpTextHelper) CreateTextHelper(); return mpTextHelper->GetChildCount(); } uno::Reference< XAccessible > SAL_CALL ScAccessibleEditObject::getAccessibleChild(sal_Int32 nIndex) throw (uno::RuntimeException, lang::IndexOutOfBoundsException) { ScUnoGuard aGuard; IsObjectValid(); if (!mpTextHelper) CreateTextHelper(); return mpTextHelper->GetChild(nIndex); } uno::Reference SAL_CALL ScAccessibleEditObject::getAccessibleStateSet(void) throw (uno::RuntimeException) { ScUnoGuard aGuard; uno::Reference xParentStates; if (getAccessibleParent().is()) { uno::Reference xParentContext = getAccessibleParent()->getAccessibleContext(); xParentStates = xParentContext->getAccessibleStateSet(); } utl::AccessibleStateSetHelper* pStateSet = new utl::AccessibleStateSetHelper(); if (IsDefunc(xParentStates)) pStateSet->AddState(AccessibleStateType::DEFUNC); else { // all states are const, because this object exists only in one state pStateSet->AddState(AccessibleStateType::EDITABLE); pStateSet->AddState(AccessibleStateType::ENABLED); pStateSet->AddState(AccessibleStateType::SENSITIVE); pStateSet->AddState(AccessibleStateType::MULTI_LINE); pStateSet->AddState(AccessibleStateType::MULTI_SELECTABLE); pStateSet->AddState(AccessibleStateType::SHOWING); pStateSet->AddState(AccessibleStateType::VISIBLE); } return pStateSet; } ::rtl::OUString SAL_CALL ScAccessibleEditObject::createAccessibleDescription(void) throw (uno::RuntimeException) { // DBG_ERRORFILE("Should never be called, because is set in the constructor.") return rtl::OUString(); } ::rtl::OUString SAL_CALL ScAccessibleEditObject::createAccessibleName(void) throw (uno::RuntimeException) { DBG_ERRORFILE("Should never be called, because is set in the constructor."); return rtl::OUString(); } ///===== XAccessibleEventBroadcaster ===================================== void SAL_CALL ScAccessibleEditObject::addEventListener(const uno::Reference& xListener) throw (uno::RuntimeException) { if (!mpTextHelper) CreateTextHelper(); mpTextHelper->AddEventListener(xListener); ScAccessibleContextBase::addEventListener(xListener); } void SAL_CALL ScAccessibleEditObject::removeEventListener(const uno::Reference& xListener) throw (uno::RuntimeException) { if (!mpTextHelper) CreateTextHelper(); mpTextHelper->RemoveEventListener(xListener); ScAccessibleContextBase::removeEventListener(xListener); } //===== XServiceInfo ==================================================== ::rtl::OUString SAL_CALL ScAccessibleEditObject::getImplementationName(void) throw (uno::RuntimeException) { return rtl::OUString(RTL_CONSTASCII_USTRINGPARAM ("ScAccessibleEditObject")); } //===== XTypeProvider ======================================================= uno::Sequence SAL_CALL ScAccessibleEditObject::getImplementationId(void) throw (uno::RuntimeException) { ScUnoGuard aGuard; IsObjectValid(); static uno::Sequence aId; if (aId.getLength() == 0) { aId.realloc (16); rtl_createUuid (reinterpret_cast(aId.getArray()), 0, sal_True); } return aId; } //==== internal ========================================================= sal_Bool ScAccessibleEditObject::IsDefunc( const uno::Reference& rxParentStates) { return ScAccessibleContextBase::IsDefunc() || !getAccessibleParent().is() || (rxParentStates.is() && rxParentStates->contains(AccessibleStateType::DEFUNC)); } void ScAccessibleEditObject::CreateTextHelper() { if (!mpTextHelper) { ::std::auto_ptr < ScAccessibleTextData > pAccessibleTextData; if (meObjectType == CellInEditMode || meObjectType == EditControl) { pAccessibleTextData.reset (new ScAccessibleEditObjectTextData(mpEditView, mpWindow)); } else { pAccessibleTextData.reset (new ScAccessibleEditLineTextData(NULL, mpWindow)); } ::std::auto_ptr< SvxEditSource > pEditSource (new ScAccessibilityEditSource(pAccessibleTextData)); mpTextHelper = new ::accessibility::AccessibleTextHelper(pEditSource ); mpTextHelper->SetEventSource(this); mpTextHelper->SetFocus(mbHasFocus); // #i54814# activate cell in edit mode if( meObjectType == CellInEditMode ) { // do not activate cell object, if top edit line is active const ScInputHandler* pInputHdl = SC_MOD()->GetInputHdl(); if( pInputHdl && !pInputHdl->IsTopMode() ) { SdrHint aHint( HINT_BEGEDIT ); mpTextHelper->GetEditSource().GetBroadcaster().Broadcast( aHint ); } } } } sal_Int32 SAL_CALL ScAccessibleEditObject::getForeground( ) throw (::com::sun::star::uno::RuntimeException) { return GetFgBgColor(rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(SC_UNONAME_CCOLOR))); } sal_Int32 SAL_CALL ScAccessibleEditObject::getBackground( ) throw (::com::sun::star::uno::RuntimeException) { return GetFgBgColor(rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(SC_UNONAME_CELLBACK))); } sal_Int32 ScAccessibleEditObject::GetFgBgColor( const rtl::OUString &strPropColor) { ScUnoGuard aGuard; sal_Int32 nColor(0); if (m_pScDoc) { SfxObjectShell* pObjSh = m_pScDoc->GetDocumentShell(); if ( pObjSh ) { uno::Reference xSpreadDoc( pObjSh->GetModel(), uno::UNO_QUERY ); if ( xSpreadDoc.is() ) { uno::Reference xSheets = xSpreadDoc->getSheets(); uno::Reference xIndex( xSheets, uno::UNO_QUERY ); if ( xIndex.is() ) { uno::Any aTable = xIndex->getByIndex(m_curCellAddress.Tab()); uno::Reference xTable; if (aTable>>=xTable) { uno::Reference xCell = xTable->getCellByPosition(m_curCellAddress.Col(), m_curCellAddress.Row()); if (xCell.is()) { uno::Reference xCellProps(xCell, uno::UNO_QUERY); if (xCellProps.is()) { uno::Any aAny = xCellProps->getPropertyValue(strPropColor); aAny >>= nColor; } } } } } } } return nColor; } //===== XAccessibleSelection ============================================ //-------------------------------------------------------------------------------- void SAL_CALL ScAccessibleEditObject::selectAccessibleChild( sal_Int32 ) throw ( IndexOutOfBoundsException, RuntimeException ) { } //---------------------------------------------------------------------------------- sal_Bool SAL_CALL ScAccessibleEditObject::isAccessibleChildSelected( sal_Int32 nChildIndex ) throw ( IndexOutOfBoundsException, RuntimeException ) { uno::Reference xAcc = getAccessibleChild( nChildIndex ); uno::Reference xContext; if( xAcc.is() ) xContext = xAcc->getAccessibleContext(); if( xContext.is() ) { if( xContext->getAccessibleRole() == AccessibleRole::PARAGRAPH ) { uno::Reference< ::com::sun::star::accessibility::XAccessibleText > xText(xAcc, uno::UNO_QUERY); if( xText.is() ) { if( xText->getSelectionStart() >= 0 ) return sal_True; } } } return sal_False; } //--------------------------------------------------------------------- void SAL_CALL ScAccessibleEditObject::clearAccessibleSelection( ) throw ( RuntimeException ) { } //------------------------------------------------------------------------- void SAL_CALL ScAccessibleEditObject::selectAllAccessibleChildren( ) throw ( RuntimeException ) { } //---------------------------------------------------------------------------- sal_Int32 SAL_CALL ScAccessibleEditObject::getSelectedAccessibleChildCount() throw ( RuntimeException ) { sal_Int32 nCount = 0; sal_Int32 TotalCount = getAccessibleChildCount(); for( sal_Int32 i = 0; i < TotalCount; i++ ) if( isAccessibleChildSelected(i) ) nCount++; return nCount; } //-------------------------------------------------------------------------------------- uno::Reference SAL_CALL ScAccessibleEditObject::getSelectedAccessibleChild( sal_Int32 nSelectedChildIndex ) throw ( IndexOutOfBoundsException, RuntimeException) { if ( nSelectedChildIndex > getSelectedAccessibleChildCount() ) throw IndexOutOfBoundsException(); sal_Int32 i1, i2; for( i1 = 0, i2 = 0; i1 < getAccessibleChildCount(); i1++ ) if( isAccessibleChildSelected(i1) ) { if( i2 == nSelectedChildIndex ) return getAccessibleChild( i1 ); i2++; } return uno::Reference(); } //---------------------------------------------------------------------------------- void SAL_CALL ScAccessibleEditObject::deselectAccessibleChild( sal_Int32 ) throw ( IndexOutOfBoundsException, RuntimeException ) { } uno::Reference< XAccessibleRelationSet > ScAccessibleEditObject::getAccessibleRelationSet( ) throw (uno::RuntimeException) { ScUnoGuard aGuard; Window* pWindow = mpWindow; utl::AccessibleRelationSetHelper* rRelationSet = new utl::AccessibleRelationSetHelper; uno::Reference< XAccessibleRelationSet > rSet = rRelationSet; if ( pWindow ) { Window *pLabeledBy = pWindow->GetAccessibleRelationLabeledBy(); if ( pLabeledBy && pLabeledBy != pWindow ) { uno::Sequence< uno::Reference< uno::XInterface > > aSequence(1); aSequence[0] = pLabeledBy->GetAccessible(); rRelationSet->AddRelation( AccessibleRelation( AccessibleRelationType::LABELED_BY, aSequence ) ); } Window* pMemberOf = pWindow->GetAccessibleRelationMemberOf(); if ( pMemberOf && pMemberOf != pWindow ) { uno::Sequence< uno::Reference< uno::XInterface > > aSequence(1); aSequence[0] = pMemberOf->GetAccessible(); rRelationSet->AddRelation( AccessibleRelation( AccessibleRelationType::MEMBER_OF, aSequence ) ); } return rSet; } return uno::Reference< XAccessibleRelationSet >(); }