/* -*- 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 "AccessiblePageShape.hxx" #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::accessibility; using ::com::sun::star::uno::Reference; namespace accessibility { //===== internal ============================================================ AccessiblePageShape::AccessiblePageShape ( const uno::Reference& rxPage, const uno::Reference& rxParent, const AccessibleShapeTreeInfo& rShapeTreeInfo, long nIndex) : AccessibleShape (AccessibleShapeInfo (NULL, rxParent, nIndex), rShapeTreeInfo), mxPage (rxPage) { // The main part of the initialization is done in the init method which // has to be called from this constructor's caller. } AccessiblePageShape::~AccessiblePageShape() { OSL_TRACE ("~AccessiblePageShape"); } void AccessiblePageShape::Init() { AccessibleShape::Init (); } //===== XAccessibleContext ================================================== sal_Int32 SAL_CALL AccessiblePageShape::getAccessibleChildCount() throw (std::exception) { return 0; } /** Forward the request to the shape. Return the requested shape or throw an exception for a wrong index. */ uno::Reference SAL_CALL AccessiblePageShape::getAccessibleChild( sal_Int32 ) throw (lang::IndexOutOfBoundsException, uno::RuntimeException, std::exception) { throw lang::IndexOutOfBoundsException ("page shape has no children", static_cast(this)); } //===== XAccessibleComponent ================================================ awt::Rectangle SAL_CALL AccessiblePageShape::getBounds() throw (::com::sun::star::uno::RuntimeException, std::exception) { ThrowIfDisposed (); awt::Rectangle aBoundingBox; if (maShapeTreeInfo.GetViewForwarder() != NULL) { uno::Reference xSet (mxPage, uno::UNO_QUERY); if (xSet.is()) { uno::Any aValue; awt::Point aPosition; awt::Size aSize; aValue = xSet->getPropertyValue ("BorderLeft"); aValue >>= aBoundingBox.X; aValue = xSet->getPropertyValue ("BorderTop"); aValue >>= aBoundingBox.Y; aValue = xSet->getPropertyValue ("Width"); aValue >>= aBoundingBox.Width; aValue = xSet->getPropertyValue ("Height"); aValue >>= aBoundingBox.Height; } // Transform coordinates from internal to pixel. ::Size aPixelSize = maShapeTreeInfo.GetViewForwarder()->LogicToPixel ( ::Size (aBoundingBox.Width, aBoundingBox.Height)); ::Point aPixelPosition = maShapeTreeInfo.GetViewForwarder()->LogicToPixel ( ::Point (aBoundingBox.X, aBoundingBox.Y)); // Clip the shape's bounding box with the bounding box of its parent. Reference xParentComponent ( getAccessibleParent(), uno::UNO_QUERY); if (xParentComponent.is()) { // Make the coordinates relative to the parent. awt::Point aParentLocation (xParentComponent->getLocationOnScreen()); int x = aPixelPosition.getX() - aParentLocation.X; int y = aPixelPosition.getY() - aParentLocation.Y; // Clip with parent (with coordinates relative to itself). ::Rectangle aBBox ( x, y, x + aPixelSize.getWidth(), y + aPixelSize.getHeight()); awt::Size aParentSize (xParentComponent->getSize()); ::Rectangle aParentBBox (0,0, aParentSize.Width, aParentSize.Height); aBBox = aBBox.GetIntersection (aParentBBox); aBoundingBox = awt::Rectangle ( aBBox.getX(), aBBox.getY(), aBBox.getWidth(), aBBox.getHeight()); } else aBoundingBox = awt::Rectangle ( aPixelPosition.getX(), aPixelPosition.getY(), aPixelSize.getWidth(), aPixelSize.getHeight()); } return aBoundingBox; } sal_Int32 SAL_CALL AccessiblePageShape::getForeground() throw (::com::sun::star::uno::RuntimeException, std::exception) { ThrowIfDisposed (); sal_Int32 nColor (0x0ffffffL); try { uno::Reference aSet (mxPage, uno::UNO_QUERY); if (aSet.is()) { uno::Any aColor; aColor = aSet->getPropertyValue ("LineColor"); aColor >>= nColor; } } catch (const ::com::sun::star::beans::UnknownPropertyException&) { // Ignore exception and return default color. } return nColor; } /** Extract the background color from the Background property of the draw page or its master page. */ sal_Int32 SAL_CALL AccessiblePageShape::getBackground() throw (::com::sun::star::uno::RuntimeException, std::exception) { ThrowIfDisposed (); sal_Int32 nColor (0x01020ffL); try { uno::Reference xSet (mxPage, uno::UNO_QUERY); if (xSet.is()) { uno::Any aBGSet; aBGSet = xSet->getPropertyValue ("Background"); Reference xBGSet (aBGSet, uno::UNO_QUERY); if ( ! xBGSet.is()) { // Draw page has no Background property. Try the master // page instead. Reference xTarget (mxPage, uno::UNO_QUERY); if (xTarget.is()) { xSet = Reference (xTarget->getMasterPage(), uno::UNO_QUERY); aBGSet = xSet->getPropertyValue ("Background"); xBGSet = Reference (aBGSet, uno::UNO_QUERY); } } // Fetch the fill color. Has to be extended to cope with // gradients, hashes, and bitmaps. if (xBGSet.is()) { uno::Any aColor; aColor = xBGSet->getPropertyValue ("FillColor"); aColor >>= nColor; } else OSL_TRACE ("no Background property in page"); } } catch (const ::com::sun::star::beans::UnknownPropertyException&) { OSL_TRACE ("caught exception due to unknown property"); // Ignore exception and return default color. } return nColor; } // XServiceInfo OUString SAL_CALL AccessiblePageShape::getImplementationName() throw (::com::sun::star::uno::RuntimeException, std::exception) { ThrowIfDisposed (); return OUString("AccessiblePageShape"); } ::com::sun::star::uno::Sequence< OUString> SAL_CALL AccessiblePageShape::getSupportedServiceNames() throw (::com::sun::star::uno::RuntimeException, std::exception) { ThrowIfDisposed (); return AccessibleShape::getSupportedServiceNames(); } //===== lang::XEventListener ================================================ void SAL_CALL AccessiblePageShape::disposing (const ::com::sun::star::lang::EventObject& aEvent) throw (::com::sun::star::uno::RuntimeException, std::exception) { ThrowIfDisposed (); AccessibleShape::disposing (aEvent); } //===== XComponent ========================================================== void AccessiblePageShape::dispose() throw (::com::sun::star::uno::RuntimeException, std::exception) { OSL_TRACE ("AccessiblePageShape::dispose"); // Unregister listeners. Reference xComponent (mxShape, uno::UNO_QUERY); if (xComponent.is()) xComponent->removeEventListener (this); // Cleanup. mxShape = NULL; // Call base classes. AccessibleContextBase::dispose (); } //===== protected internal ================================================== OUString AccessiblePageShape::CreateAccessibleBaseName() throw (::com::sun::star::uno::RuntimeException) { return OUString ("PageShape"); } OUString AccessiblePageShape::CreateAccessibleName() throw (::com::sun::star::uno::RuntimeException) { Reference xPageProperties (mxPage, UNO_QUERY); // Get name of the current slide. OUString sCurrentSlideName; try { if (xPageProperties.is()) { xPageProperties->getPropertyValue( "LinkDisplayName" ) >>= sCurrentSlideName; } } catch (const beans::UnknownPropertyException&) { } return CreateAccessibleBaseName()+": "+sCurrentSlideName; } OUString AccessiblePageShape::CreateAccessibleDescription() throw (::com::sun::star::uno::RuntimeException) { return OUString ("Page Shape"); } } // end of namespace accessibility /* vim:set shiftwidth=4 softtabstop=4 expandtab: */