/************************************************************** * * 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_sd.hxx" #include "AccessibleDrawDocumentView.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "Window.hxx" #include #include "ViewShell.hxx" #include "View.hxx" #include #include "accessibility.hrc" #include "sdresid.hxx" #include using ::rtl::OUString; using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::accessibility; class SfxViewFrame; #define A2S(pString) (::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(pString))) namespace accessibility { //===== internal ============================================================ AccessibleDrawDocumentView::AccessibleDrawDocumentView ( ::sd::Window* pSdWindow, ::sd::ViewShell* pViewShell, const uno::Reference& rxController, const uno::Reference& rxParent) : AccessibleDocumentViewBase (pSdWindow, pViewShell, rxController, rxParent), mpChildrenManager (NULL) { OSL_TRACE ("AccessibleDrawDocumentView"); UpdateAccessibleName(); } AccessibleDrawDocumentView::~AccessibleDrawDocumentView (void) { OSL_TRACE ("~AccessibleDrawDocumentView"); DBG_ASSERT (rBHelper.bDisposed || rBHelper.bInDispose, "~AccessibleDrawDocumentView: object has not been disposed"); } void AccessibleDrawDocumentView::Init (void) { AccessibleDocumentViewBase::Init (); // Determine the list of shapes on the current page. uno::Reference xShapeList; uno::Reference xView (mxController, uno::UNO_QUERY); if (xView.is()) xShapeList = uno::Reference ( xView->getCurrentPage(), uno::UNO_QUERY); // Create the children manager. mpChildrenManager = new ChildrenManager(this, xShapeList, maShapeTreeInfo, *this); if (mpChildrenManager != NULL) { // Create the page shape and initialize it. The shape is acquired // before initialization and released after transferring ownership // to the children manager to prevent premature disposing of the // shape. AccessiblePageShape* pPage = CreateDrawPageShape(); if (pPage != NULL) { pPage->acquire(); pPage->Init(); mpChildrenManager->AddAccessibleShape ( std::auto_ptr(pPage)); pPage->release(); mpChildrenManager->Update (); } mpChildrenManager->UpdateSelection (); } } void AccessibleDrawDocumentView::ViewForwarderChanged (ChangeType aChangeType, const IAccessibleViewForwarder* pViewForwarder) { AccessibleDocumentViewBase::ViewForwarderChanged (aChangeType, pViewForwarder); if (mpChildrenManager != NULL) mpChildrenManager->ViewForwarderChanged (aChangeType, pViewForwarder); } /** The page shape is created on every call at the moment (provided that every thing goes well). */ AccessiblePageShape* AccessibleDrawDocumentView::CreateDrawPageShape (void) { AccessiblePageShape* pShape = NULL; // Create a shape that represents the actual draw page. uno::Reference xView (mxController, uno::UNO_QUERY); if (xView.is()) { uno::Reference xSet ( uno::Reference (xView->getCurrentPage(), uno::UNO_QUERY)); if (xSet.is()) { // Create a rectangle shape that will represent the draw page. uno::Reference xFactory (mxModel, uno::UNO_QUERY); uno::Reference xRectangle; if (xFactory.is()) xRectangle = uno::Reference(xFactory->createInstance ( OUString (RTL_CONSTASCII_USTRINGPARAM("com.sun.star.drawing.RectangleShape"))), uno::UNO_QUERY); // Set the shape's size and position. if (xRectangle.is()) { uno::Any aValue; awt::Point aPosition; awt::Size aSize; // Set size and position of the shape to those of the draw // page. aValue = xSet->getPropertyValue ( OUString (RTL_CONSTASCII_USTRINGPARAM("BorderLeft"))); aValue >>= aPosition.X; aValue = xSet->getPropertyValue ( OUString (RTL_CONSTASCII_USTRINGPARAM("BorderTop"))); aValue >>= aPosition.Y; xRectangle->setPosition (aPosition); aValue = xSet->getPropertyValue ( OUString (RTL_CONSTASCII_USTRINGPARAM("Width"))); aValue >>= aSize.Width; aValue = xSet->getPropertyValue ( OUString (RTL_CONSTASCII_USTRINGPARAM("Height"))); aValue >>= aSize.Height; xRectangle->setSize (aSize); // Create the accessible object for the shape and // initialize it. pShape = new AccessiblePageShape ( xView->getCurrentPage(), this, maShapeTreeInfo); } } } return pShape; } //===== XAccessibleContext ================================================== sal_Int32 SAL_CALL AccessibleDrawDocumentView::getAccessibleChildCount (void) throw (uno::RuntimeException) { ThrowIfDisposed (); long mpChildCount = AccessibleDocumentViewBase::getAccessibleChildCount(); // Forward request to children manager. if (mpChildrenManager != NULL) mpChildCount += mpChildrenManager->GetChildCount (); return mpChildCount; } uno::Reference SAL_CALL AccessibleDrawDocumentView::getAccessibleChild (sal_Int32 nIndex) throw (uno::RuntimeException, lang::IndexOutOfBoundsException) { ThrowIfDisposed (); ::osl::ClearableMutexGuard aGuard (maMutex); // Take care of children of the base class. sal_Int32 nCount = AccessibleDocumentViewBase::getAccessibleChildCount(); if (nCount > 0) { if (nIndex < nCount) return AccessibleDocumentViewBase::getAccessibleChild(nIndex); else nIndex -= nCount; } // Create a copy of the pointer to the children manager and release the // mutex before calling any of its methods. ChildrenManager* pChildrenManager = mpChildrenManager; aGuard.clear(); // Forward request to children manager. if (pChildrenManager != NULL) { return pChildrenManager->GetChild (nIndex); } else throw lang::IndexOutOfBoundsException ( ::rtl::OUString::createFromAscii ("no accessible child with index ") + rtl::OUString::valueOf(nIndex), static_cast(this)); } //===== XEventListener ====================================================== void SAL_CALL AccessibleDrawDocumentView::disposing (const lang::EventObject& rEventObject) throw (::com::sun::star::uno::RuntimeException) { ThrowIfDisposed (); AccessibleDocumentViewBase::disposing (rEventObject); if (rEventObject.Source == mxModel) { ::osl::Guard< ::osl::Mutex> aGuard (::osl::Mutex::getGlobalMutex()); // maShapeTreeInfo has been modified in base class. if (mpChildrenManager != NULL) mpChildrenManager->SetInfo (maShapeTreeInfo); } } //===== XPropertyChangeListener ============================================= void SAL_CALL AccessibleDrawDocumentView::propertyChange (const beans::PropertyChangeEvent& rEventObject) throw (::com::sun::star::uno::RuntimeException) { ThrowIfDisposed (); AccessibleDocumentViewBase::propertyChange (rEventObject); OSL_TRACE ("AccessibleDrawDocumentView::propertyChange"); if (rEventObject.PropertyName == OUString (RTL_CONSTASCII_USTRINGPARAM("CurrentPage"))) { OSL_TRACE (" current page changed"); // Update the accessible name to reflect the current slide. UpdateAccessibleName(); // The current page changed. Update the children manager accordingly. uno::Reference xView (mxController, uno::UNO_QUERY); if (xView.is() && mpChildrenManager!=NULL) { // Inform the children manager to forget all children and give // him the new ones. mpChildrenManager->ClearAccessibleShapeList (); mpChildrenManager->SetShapeList (uno::Reference ( xView->getCurrentPage(), uno::UNO_QUERY)); // Create the page shape and initialize it. The shape is // acquired before initialization and released after // transferring ownership to the children manager to prevent // premature disposing of the shape. AccessiblePageShape* pPage = CreateDrawPageShape (); if (pPage != NULL) { pPage->acquire(); pPage->Init(); mpChildrenManager->AddAccessibleShape ( std::auto_ptr(pPage)); mpChildrenManager->Update (false); pPage->release(); } } else OSL_TRACE ("View invalid"); } else if (rEventObject.PropertyName == OUString (RTL_CONSTASCII_USTRINGPARAM("VisibleArea"))) { OSL_TRACE (" visible area changed"); if (mpChildrenManager != NULL) mpChildrenManager->ViewForwarderChanged ( IAccessibleViewForwarderListener::VISIBLE_AREA, &maViewForwarder); } else { OSL_TRACE (" unhandled"); } OSL_TRACE (" done"); } //===== XServiceInfo ======================================================== ::rtl::OUString SAL_CALL AccessibleDrawDocumentView::getImplementationName (void) throw (::com::sun::star::uno::RuntimeException) { return ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM( "AccessibleDrawDocumentView")); } ::com::sun::star::uno::Sequence< ::rtl::OUString> SAL_CALL AccessibleDrawDocumentView::getSupportedServiceNames (void) throw (::com::sun::star::uno::RuntimeException) { ThrowIfDisposed(); // Get list of supported service names from base class... uno::Sequence aServiceNames = AccessibleDocumentViewBase::getSupportedServiceNames(); sal_Int32 nCount (aServiceNames.getLength()); // ...and add additional names. aServiceNames.realloc (nCount + 1); static const OUString sAdditionalServiceName (RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.drawing.AccessibleDrawDocumentView")); aServiceNames[nCount] = sAdditionalServiceName; return aServiceNames; } /// Create a name for this view. ::rtl::OUString AccessibleDrawDocumentView::CreateAccessibleName (void) throw (::com::sun::star::uno::RuntimeException) { rtl::OUString sName; uno::Reference xInfo (mxController, uno::UNO_QUERY); if (xInfo.is()) { uno::Sequence< ::rtl::OUString > aServices( xInfo->getSupportedServiceNames() ); OUString sFirstService = aServices[0]; if (sFirstService == OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.drawing.DrawingDocumentDrawView"))) { if( aServices.getLength() >= 2 && aServices[1] == OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.presentation.PresentationView"))) { ::vos::OGuard aGuard( Application::GetSolarMutex() ); sName = String( SdResId(SID_SD_A11Y_I_DRAWVIEW_N) ); } else { ::vos::OGuard aGuard( Application::GetSolarMutex() ); sName = String( SdResId(SID_SD_A11Y_D_DRAWVIEW_N) ); } } else if (sFirstService == OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.presentation.NotesView"))) { ::vos::OGuard aGuard( Application::GetSolarMutex() ); sName = String( SdResId(SID_SD_A11Y_I_NOTESVIEW_N) ); } else if (sFirstService == OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.presentation.HandoutView"))) { ::vos::OGuard aGuard( Application::GetSolarMutex() ); sName = String( SdResId(SID_SD_A11Y_I_HANDOUTVIEW_N) ); } else { sName = sFirstService; } } else { sName = OUString(RTL_CONSTASCII_USTRINGPARAM("AccessibleDrawDocumentView")); } return sName; } /** Create a description for this view. Use the model's description or URL if a description is not available. */ ::rtl::OUString AccessibleDrawDocumentView::CreateAccessibleDescription (void) throw (::com::sun::star::uno::RuntimeException) { rtl::OUString sDescription; uno::Reference xInfo (mxController, uno::UNO_QUERY); if (xInfo.is()) { uno::Sequence< ::rtl::OUString > aServices( xInfo->getSupportedServiceNames() ); OUString sFirstService = aServices[0]; if (sFirstService == OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.drawing.DrawingDocumentDrawView"))) { if( aServices.getLength() >= 2 && aServices[1] == OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.presentation.PresentationView"))) { ::vos::OGuard aGuard( Application::GetSolarMutex() ); sDescription = String( SdResId(SID_SD_A11Y_I_DRAWVIEW_D) ); } else { ::vos::OGuard aGuard( Application::GetSolarMutex() ); sDescription = String( SdResId(SID_SD_A11Y_D_DRAWVIEW_D) ); } } else if (sFirstService == OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.presentation.NotesView"))) { ::vos::OGuard aGuard( Application::GetSolarMutex() ); sDescription = String( SdResId(SID_SD_A11Y_I_NOTESVIEW_D) ); } else if (sFirstService == OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.presentation.HandoutView"))) { ::vos::OGuard aGuard( Application::GetSolarMutex() ); sDescription = String( SdResId(SID_SD_A11Y_I_HANDOUTVIEW_D) ); } else { sDescription = sFirstService; } } else { sDescription = OUString(RTL_CONSTASCII_USTRINGPARAM("Accessible Draw Document")); } return sDescription; } /** Return selection state of specified child */ sal_Bool AccessibleDrawDocumentView::implIsSelected( sal_Int32 nAccessibleChildIndex ) throw (uno::RuntimeException) { const vos::OGuard aSolarGuard( Application::GetSolarMutex() ); uno::Reference< view::XSelectionSupplier > xSel( mxController, uno::UNO_QUERY ); sal_Bool bRet = sal_False; OSL_ENSURE( 0 <= nAccessibleChildIndex, "AccessibleDrawDocumentView::implIsSelected: invalid index!" ); if( xSel.is() && ( 0 <= nAccessibleChildIndex ) ) { uno::Any aAny( xSel->getSelection() ); uno::Reference< drawing::XShapes > xShapes; aAny >>= xShapes; if( xShapes.is() ) { AccessibleShape* pAcc = AccessibleShape::getImplementation( getAccessibleChild( nAccessibleChildIndex ) ); if( pAcc ) { uno::Reference< drawing::XShape > xShape( pAcc->GetXShape() ); if( xShape.is() ) { for( sal_Int32 i = 0, nCount = xShapes->getCount(); ( i < nCount ) && !bRet; ++i ) if( xShapes->getByIndex( i ) == xShape ) bRet = sal_True; } } } } return( bRet ); } /** Select or delselect the specified shapes. The corresponding accessible shapes are notified over the selection change listeners registered with the XSelectionSupplier of the controller. */ void AccessibleDrawDocumentView::implSelect( sal_Int32 nAccessibleChildIndex, sal_Bool bSelect ) throw (lang::IndexOutOfBoundsException, uno::RuntimeException) { const vos::OGuard aSolarGuard( Application::GetSolarMutex() ); uno::Reference< view::XSelectionSupplier > xSel( mxController, uno::UNO_QUERY ); AccessibleShape* pAccessibleChild; if( xSel.is() ) { uno::Any aAny; if( ACCESSIBLE_SELECTION_CHILD_ALL == nAccessibleChildIndex ) { // Select or deselect all children. if( !bSelect ) xSel->select( aAny ); else { uno::Reference< drawing::XShapes > xShapes( new SvxShapeCollection() ); for(sal_Int32 i = 0, nCount = getAccessibleChildCount(); i < nCount; ++i ) { AccessibleShape* pAcc = AccessibleShape::getImplementation( getAccessibleChild( i ) ); if( pAcc && pAcc->GetXShape().is() ) { xShapes->add( pAcc->GetXShape() ); pAccessibleChild = pAcc; } } if( xShapes->getCount() ) { aAny <<= xShapes; xSel->select( aAny ); } } } else if( nAccessibleChildIndex >= 0 ) { // Select or deselect only the child with index // nAccessibleChildIndex. AccessibleShape* pAcc = AccessibleShape::getImplementation( getAccessibleChild( nAccessibleChildIndex )); pAccessibleChild = pAcc; // Add or remove the shape that is made accessible from the // selection of the controller. if( pAcc ) { uno::Reference< drawing::XShape > xShape( pAcc->GetXShape() ); if( xShape.is() ) { uno::Reference< drawing::XShapes > xShapes; sal_Bool bFound = sal_False; aAny = xSel->getSelection(); aAny >>= xShapes; // Search shape to be selected in current selection. if (xShapes.is()) { sal_Int32 nCount = xShapes->getCount(); for (sal_Int32 i=0; ( i < nCount ) && !bFound; ++i ) if( xShapes->getByIndex( i ) == xShape ) bFound = sal_True; } else // Create an empty selection to add the shape to. xShapes = new SvxShapeCollection(); // Update the selection. if( !bFound && bSelect ) xShapes->add( xShape ); else if( bFound && !bSelect ) xShapes->remove( xShape ); aAny <<= xShapes; xSel->select( aAny ); } } } } } void AccessibleDrawDocumentView::Activated (void) { if (mpChildrenManager != NULL) { mpChildrenManager->UpdateSelection(); // When none of the children has the focus then claim it for the // view. if ( ! mpChildrenManager->HasFocus()) SetState (AccessibleStateType::FOCUSED); else ResetState (AccessibleStateType::FOCUSED); } } void AccessibleDrawDocumentView::Deactivated (void) { if (mpChildrenManager != NULL) mpChildrenManager->RemoveFocus(); ResetState (AccessibleStateType::FOCUSED); } void AccessibleDrawDocumentView::impl_dispose (void) { if (mpChildrenManager != NULL) { delete mpChildrenManager; mpChildrenManager = NULL; } AccessibleDocumentViewBase::impl_dispose(); } /** This method is called from the component helper base class while disposing. */ void SAL_CALL AccessibleDrawDocumentView::disposing (void) { // Release resources. if (mpChildrenManager != NULL) { delete mpChildrenManager; mpChildrenManager = NULL; } // Forward call to base classes. AccessibleDocumentViewBase::disposing (); } void AccessibleDrawDocumentView::UpdateAccessibleName (void) { OUString sNewName (CreateAccessibleName()); sNewName += A2S(": "); // Add the number of the current slide. uno::Reference xView (mxController, uno::UNO_QUERY); if (xView.is()) { uno::Reference xProperties (xView->getCurrentPage(), UNO_QUERY); if (xProperties.is()) try { sal_Int16 nPageNumber (0); if (xProperties->getPropertyValue(A2S("Number")) >>= nPageNumber) { sNewName += OUString::valueOf(sal_Int32(nPageNumber)); } } catch (beans::UnknownPropertyException&) { } } // Add the number of pages/slides. Reference xPagesSupplier (mxModel, UNO_QUERY); if (xPagesSupplier.is()) { Reference xPages (xPagesSupplier->getDrawPages(), UNO_QUERY); if (xPages.is()) { sNewName += A2S(" / "); sNewName += OUString::valueOf(xPages->getCount()); } } SetAccessibleName (sNewName, AutomaticallyCreated); } } // end of namespace accessibility