/* -*- 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 "vbaborders.hxx" #include #include #include #include #include #include #include #include "vbapalette.hxx" using namespace ::com::sun::star; using namespace ::ooo::vba; typedef ::cppu::WeakImplHelper RangeBorders_Base; typedef InheritedHelperInterfaceWeakImpl SwVbaBorder_Base; // #TODO sort these indexes to match the order in which Word iterates over the // borders, the enumeration will match the order in this list const sal_Int16 supportedIndexTable[] = { word::WdBorderType::wdBorderBottom, word::WdBorderType::wdBorderDiagonalDown, word::WdBorderType::wdBorderDiagonalUp, word::WdBorderType::wdBorderHorizontal, word::WdBorderType::wdBorderLeft, word::WdBorderType::wdBorderRight, word::WdBorderType::wdBorderTop, word::WdBorderType::wdBorderVertical }; // Equiv widths in 1/100 mm const sal_Int32 OOLineHairline = 2; namespace { class SwVbaBorder : public SwVbaBorder_Base { private: uno::Reference< beans::XPropertySet > m_xProps; sal_Int32 m_LineType; void setBorderLine( table::BorderLine const & rBorderLine ) { table::TableBorder aTableBorder; m_xProps->getPropertyValue( "TableBorder" ) >>= aTableBorder; switch ( m_LineType ) { case word::WdBorderType::wdBorderLeft: aTableBorder.IsLeftLineValid = true; aTableBorder.LeftLine= rBorderLine; break; case word::WdBorderType::wdBorderTop: aTableBorder.IsTopLineValid = true; aTableBorder.TopLine = rBorderLine; break; case word::WdBorderType::wdBorderBottom: aTableBorder.IsBottomLineValid = true; aTableBorder.BottomLine = rBorderLine; break; case word::WdBorderType::wdBorderRight: aTableBorder.IsRightLineValid = true; aTableBorder.RightLine = rBorderLine; break; case word::WdBorderType::wdBorderVertical: aTableBorder.IsVerticalLineValid = true; aTableBorder.VerticalLine = rBorderLine; break; case word::WdBorderType::wdBorderHorizontal: aTableBorder.IsHorizontalLineValid = true; aTableBorder.HorizontalLine = rBorderLine; break; case word::WdBorderType::wdBorderDiagonalDown: case word::WdBorderType::wdBorderDiagonalUp: // #TODO have to ignore at the moment, would be // nice to investigate what we can do here break; default: return; } m_xProps->setPropertyValue( "TableBorder", uno::Any(aTableBorder) ); } bool getBorderLine( table::BorderLine& rBorderLine ) { table::TableBorder aTableBorder; m_xProps->getPropertyValue( "TableBorder" ) >>= aTableBorder; switch ( m_LineType ) { case word::WdBorderType::wdBorderLeft: if ( aTableBorder.IsLeftLineValid ) rBorderLine = aTableBorder.LeftLine; break; case word::WdBorderType::wdBorderTop: if ( aTableBorder.IsTopLineValid ) rBorderLine = aTableBorder.TopLine; break; case word::WdBorderType::wdBorderBottom: if ( aTableBorder.IsBottomLineValid ) rBorderLine = aTableBorder.BottomLine; break; case word::WdBorderType::wdBorderRight: if ( aTableBorder.IsRightLineValid ) rBorderLine = aTableBorder.RightLine; break; case word::WdBorderType::wdBorderVertical: if ( aTableBorder.IsVerticalLineValid ) rBorderLine = aTableBorder.VerticalLine; break; case word::WdBorderType::wdBorderHorizontal: if ( aTableBorder.IsHorizontalLineValid ) rBorderLine = aTableBorder.HorizontalLine; break; case word::WdBorderType::wdBorderDiagonalDown: case word::WdBorderType::wdBorderDiagonalUp: // #TODO have to ignore at the moment, would be // nice to investigate what we can do here break; default: return false; } return true; } protected: virtual OUString getServiceImplName() override { return "SwVbaBorder"; } virtual css::uno::Sequence getServiceNames() override { static uno::Sequence< OUString > const aServiceNames { "ooo.vba.word.Border" }; return aServiceNames; } public: SwVbaBorder( const uno::Reference< beans::XPropertySet > & xProps, const uno::Reference< uno::XComponentContext >& xContext, sal_Int32 lineType ) : SwVbaBorder_Base( uno::Reference< XHelperInterface >( xProps, uno::UNO_QUERY ), xContext ), m_xProps( xProps ), m_LineType( lineType ) {} uno::Any SAL_CALL getLineStyle() override { sal_Int32 nLineStyle = word::WdLineStyle::wdLineStyleNone; table::BorderLine aBorderLine; if ( getBorderLine( aBorderLine ) ) { if( aBorderLine.InnerLineWidth !=0 && aBorderLine.OuterLineWidth !=0 ) { nLineStyle = word::WdLineStyle::wdLineStyleDouble; } else if( aBorderLine.InnerLineWidth !=0 || aBorderLine.OuterLineWidth !=0 ) { nLineStyle = word::WdLineStyle::wdLineStyleSingle; } else { nLineStyle = word::WdLineStyle::wdLineStyleNone; } } return uno::Any( nLineStyle ); } void SAL_CALL setLineStyle( const uno::Any& _linestyle ) override { // Urk no choice but to silently ignore we don't support this attribute // #TODO would be nice to support the word line styles sal_Int32 nLineStyle = 0; _linestyle >>= nLineStyle; table::BorderLine aBorderLine; if ( !getBorderLine( aBorderLine ) ) throw uno::RuntimeException("Method failed" ); switch ( nLineStyle ) { case word::WdLineStyle::wdLineStyleNone: { aBorderLine.InnerLineWidth = 0; aBorderLine.OuterLineWidth = 0; break; } case word::WdLineStyle::wdLineStyleDashDot: case word::WdLineStyle::wdLineStyleDashDotDot: case word::WdLineStyle::wdLineStyleDashDotStroked: case word::WdLineStyle::wdLineStyleDashLargeGap: case word::WdLineStyle::wdLineStyleDashSmallGap: case word::WdLineStyle::wdLineStyleDot: case word::WdLineStyle::wdLineStyleDouble: case word::WdLineStyle::wdLineStyleDoubleWavy: case word::WdLineStyle::wdLineStyleEmboss3D: case word::WdLineStyle::wdLineStyleEngrave3D: case word::WdLineStyle::wdLineStyleInset: case word::WdLineStyle::wdLineStyleOutset: case word::WdLineStyle::wdLineStyleSingle: case word::WdLineStyle::wdLineStyleSingleWavy: case word::WdLineStyle::wdLineStyleThickThinLargeGap: case word::WdLineStyle::wdLineStyleThickThinMedGap: case word::WdLineStyle::wdLineStyleThickThinSmallGap: case word::WdLineStyle::wdLineStyleThinThickLargeGap: case word::WdLineStyle::wdLineStyleThinThickMedGap: case word::WdLineStyle::wdLineStyleThinThickSmallGap: case word::WdLineStyle::wdLineStyleThinThickThinLargeGap: case word::WdLineStyle::wdLineStyleThinThickThinMedGap: case word::WdLineStyle::wdLineStyleThinThickThinSmallGap: case word::WdLineStyle::wdLineStyleTriple: { aBorderLine.InnerLineWidth = 0; aBorderLine.OuterLineWidth = OOLineHairline; break; } default: throw uno::RuntimeException("Bad param" ); } setBorderLine( aBorderLine ); } }; class RangeBorders : public RangeBorders_Base { private: uno::Reference< table::XCellRange > m_xRange; uno::Reference< uno::XComponentContext > m_xContext; VbaPalette m_Palette; sal_Int32 getTableIndex( sal_Int32 nConst ) { // okay return position of the index in the table sal_Int32 nIndexes = getCount(); sal_Int32 realIndex = 0; const sal_Int16* pTableEntry = supportedIndexTable; for ( ; realIndex < nIndexes; ++realIndex, ++pTableEntry ) { if ( *pTableEntry == nConst ) return realIndex; } return getCount(); // error condition } public: RangeBorders( const uno::Reference< table::XCellRange >& xRange, const uno::Reference< uno::XComponentContext > & xContext, VbaPalette const & rPalette ) : m_xRange( xRange ), m_xContext( xContext ), m_Palette( rPalette ) { } // XIndexAccess virtual ::sal_Int32 SAL_CALL getCount( ) override { return SAL_N_ELEMENTS( supportedIndexTable ); } virtual uno::Any SAL_CALL getByIndex( ::sal_Int32 Index ) override { sal_Int32 nIndex = getTableIndex( Index ); if ( nIndex >= 0 && nIndex < getCount() ) { uno::Reference< beans::XPropertySet > xProps( m_xRange, uno::UNO_QUERY_THROW ); return uno::Any( uno::Reference< word::XBorder >( new SwVbaBorder( xProps, m_xContext, supportedIndexTable[ nIndex ] )) ); } throw lang::IndexOutOfBoundsException(); } virtual uno::Type SAL_CALL getElementType( ) override { return cppu::UnoType::get(); } virtual sal_Bool SAL_CALL hasElements( ) override { return true; } }; } static uno::Reference< container::XIndexAccess > rangeToBorderIndexAccess( const uno::Reference< table::XCellRange >& xRange, const uno::Reference< uno::XComponentContext > & xContext, VbaPalette const & rPalette ) { return new RangeBorders( xRange, xContext, rPalette ); } namespace { class RangeBorderEnumWrapper : public EnumerationHelper_BASE { uno::Reference m_xIndexAccess; sal_Int32 nIndex; public: explicit RangeBorderEnumWrapper( const uno::Reference< container::XIndexAccess >& xIndexAccess ) : m_xIndexAccess( xIndexAccess ), nIndex( 0 ) {} virtual sal_Bool SAL_CALL hasMoreElements( ) override { return ( nIndex < m_xIndexAccess->getCount() ); } virtual uno::Any SAL_CALL nextElement( ) override { if ( nIndex < m_xIndexAccess->getCount() ) return m_xIndexAccess->getByIndex( nIndex++ ); throw container::NoSuchElementException(); } }; } // for Table borders SwVbaBorders::SwVbaBorders( const uno::Reference< XHelperInterface >& xParent, const uno::Reference< uno::XComponentContext > & xContext, const uno::Reference< table::XCellRange >& xRange, VbaPalette const & rPalette ): SwVbaBorders_BASE( xParent, xContext, rangeToBorderIndexAccess( xRange ,xContext, rPalette ) ) { m_xProps.set( xRange, uno::UNO_QUERY_THROW ); } uno::Reference< container::XEnumeration > SwVbaBorders::createEnumeration() { return new RangeBorderEnumWrapper( m_xIndexAccess ); } uno::Any SwVbaBorders::createCollectionObject( const css::uno::Any& aSource ) { return aSource; // it's already a Border object } uno::Type SwVbaBorders::getElementType() { return cppu::UnoType::get(); } uno::Any SwVbaBorders::getItemByIntIndex( const sal_Int32 nIndex ) { return createCollectionObject( m_xIndexAccess->getByIndex( nIndex ) ); } sal_Bool SAL_CALL SwVbaBorders::getShadow() { // always return False for table border in MS Word return false; } void SAL_CALL SwVbaBorders::setShadow( sal_Bool /*_shadow*/ ) { // not support in Table border in Word // TODO: } OUString SwVbaBorders::getServiceImplName() { return "SwVbaBorders"; } uno::Sequence< OUString > SwVbaBorders::getServiceNames() { static uno::Sequence< OUString > const aServiceNames { "ooo.vba.word.Borders" }; return aServiceNames; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */