/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_sc.hxx" #include "scitems.hxx" #include #include #include #include #include "cursuno.hxx" #include "cellsuno.hxx" #include "docsh.hxx" #include "hints.hxx" #include "markdata.hxx" #include "dociter.hxx" #include "miscuno.hxx" using namespace com::sun::star; //------------------------------------------------------------------------ #define SCSHEETCELLCURSOR_SERVICE "com.sun.star.sheet.SheetCellCursor" #define SCCELLCURSOR_SERVICE "com.sun.star.table.CellCursor" //------------------------------------------------------------------------ ScCellCursorObj::ScCellCursorObj(ScDocShell* pDocSh, const ScRange& rR) : ScCellRangeObj( pDocSh, rR ) { } ScCellCursorObj::~ScCellCursorObj() { } uno::Any SAL_CALL ScCellCursorObj::queryInterface( const uno::Type& rType ) throw(uno::RuntimeException) { SC_QUERYINTERFACE( sheet::XSheetCellCursor ) SC_QUERYINTERFACE( sheet::XUsedAreaCursor ) SC_QUERYINTERFACE( table::XCellCursor ) return ScCellRangeObj::queryInterface( rType ); } void SAL_CALL ScCellCursorObj::acquire() throw() { ScCellRangeObj::acquire(); } void SAL_CALL ScCellCursorObj::release() throw() { ScCellRangeObj::release(); } uno::Sequence SAL_CALL ScCellCursorObj::getTypes() throw(uno::RuntimeException) { static uno::Sequence aTypes; if ( aTypes.getLength() == 0 ) { uno::Sequence aParentTypes(ScCellRangeObj::getTypes()); long nParentLen = aParentTypes.getLength(); const uno::Type* pParentPtr = aParentTypes.getConstArray(); aTypes.realloc( nParentLen + 3 ); uno::Type* pPtr = aTypes.getArray(); pPtr[nParentLen + 0] = getCppuType((const uno::Reference*)0); pPtr[nParentLen + 1] = getCppuType((const uno::Reference*)0); pPtr[nParentLen + 2] = getCppuType((const uno::Reference*)0); for (long i=0; i SAL_CALL ScCellCursorObj::getImplementationId() throw(uno::RuntimeException) { static uno::Sequence< sal_Int8 > aId; if( aId.getLength() == 0 ) { aId.realloc( 16 ); rtl_createUuid( (sal_uInt8 *)aId.getArray(), 0, sal_True ); } return aId; } // XSheetCellCursor void SAL_CALL ScCellCursorObj::collapseToCurrentRegion() throw(uno::RuntimeException) { SolarMutexGuard aGuard; const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aOneRange( *rRanges[ 0 ] ); aOneRange.Justify(); ScDocShell* pDocSh = GetDocShell(); if ( pDocSh ) { SCCOL nStartCol = aOneRange.aStart.Col(); SCROW nStartRow = aOneRange.aStart.Row(); SCCOL nEndCol = aOneRange.aEnd.Col(); SCROW nEndRow = aOneRange.aEnd.Row(); SCTAB nTab = aOneRange.aStart.Tab(); pDocSh->GetDocument()->GetDataArea( nTab, nStartCol, nStartRow, nEndCol, nEndRow, TRUE, false ); ScRange aNew( nStartCol, nStartRow, nTab, nEndCol, nEndRow, nTab ); SetNewRange( aNew ); } } void SAL_CALL ScCellCursorObj::collapseToCurrentArray() throw(uno::RuntimeException) { SolarMutexGuard aGuard; const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aOneRange( *rRanges[ 0 ] ); aOneRange.Justify(); ScAddress aCursor(aOneRange.aStart); // use the start address of the range ScDocShell* pDocSh = GetDocShell(); if ( pDocSh ) { ScDocument* pDoc = pDocSh->GetDocument(); ScRange aMatrix; // finding the matrix range is now in GetMatrixFormulaRange in the document if ( pDoc->GetMatrixFormulaRange( aCursor, aMatrix ) ) { SetNewRange( aMatrix ); } } // thats a Bug, that this assertion comes; the API Reference says, that // if there is no Matrix, the Range is left unchanged; they says nothing // about a exception /*if (!bFound) { DBG_ERROR("keine Matrix"); //! Exception, oder was? }*/ } void SAL_CALL ScCellCursorObj::collapseToMergedArea() throw(uno::RuntimeException) { SolarMutexGuard aGuard; ScDocShell* pDocSh = GetDocShell(); if ( pDocSh ) { const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aNewRange( *rRanges[ 0 ] ); ScDocument* pDoc = pDocSh->GetDocument(); pDoc->ExtendOverlapped( aNewRange ); pDoc->ExtendMerge( aNewRange ); // after ExtendOverlapped! SetNewRange( aNewRange ); } } void SAL_CALL ScCellCursorObj::expandToEntireColumns() throw(uno::RuntimeException) { SolarMutexGuard aGuard; const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aNewRange( *rRanges[ 0 ] ); aNewRange.aStart.SetRow( 0 ); aNewRange.aEnd.SetRow( MAXROW ); SetNewRange( aNewRange ); } void SAL_CALL ScCellCursorObj::expandToEntireRows() throw(uno::RuntimeException) { SolarMutexGuard aGuard; const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aNewRange( *rRanges[ 0 ] ); aNewRange.aStart.SetCol( 0 ); aNewRange.aEnd.SetCol( MAXCOL ); SetNewRange( aNewRange ); } void SAL_CALL ScCellCursorObj::collapseToSize( sal_Int32 nColumns, sal_Int32 nRows ) throw(uno::RuntimeException) { SolarMutexGuard aGuard; if ( nColumns <= 0 || nRows <= 0 ) { DBG_ERROR("leerer Range geht nicht"); //! und dann? } else { const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aNewRange( *rRanges[ 0 ] ); aNewRange.Justify(); //! wirklich? long nEndX = aNewRange.aStart.Col() + nColumns - 1; long nEndY = aNewRange.aStart.Row() + nRows - 1; if ( nEndX < 0 ) nEndX = 0; if ( nEndX > MAXCOL ) nEndX = MAXCOL; if ( nEndY < 0 ) nEndY = 0; if ( nEndY > MAXROW ) nEndY = MAXROW; //! Fehler/Exception oder so, wenn zu gross/zu klein? aNewRange.aEnd.SetCol((SCCOL)nEndX); aNewRange.aEnd.SetRow((SCROW)nEndY); aNewRange.Justify(); //! wirklich? SetNewRange( aNewRange ); } } // XUsedAreaCursor void SAL_CALL ScCellCursorObj::gotoStartOfUsedArea( sal_Bool bExpand ) throw(uno::RuntimeException) { SolarMutexGuard aGuard; ScDocShell* pDocSh = GetDocShell(); if ( pDocSh ) { const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aNewRange( *rRanges[0] ); SCTAB nTab = aNewRange.aStart.Tab(); SCCOL nUsedX = 0; // Anfang holen SCROW nUsedY = 0; if (!pDocSh->GetDocument()->GetDataStart( nTab, nUsedX, nUsedY )) { nUsedX = 0; nUsedY = 0; } aNewRange.aStart.SetCol( nUsedX ); aNewRange.aStart.SetRow( nUsedY ); if (!bExpand) aNewRange.aEnd = aNewRange.aStart; SetNewRange( aNewRange ); } } void SAL_CALL ScCellCursorObj::gotoEndOfUsedArea( sal_Bool bExpand ) throw(uno::RuntimeException) { SolarMutexGuard aGuard; ScDocShell* pDocSh = GetDocShell(); if ( pDocSh ) { const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aNewRange( *rRanges[ 0 ]); SCTAB nTab = aNewRange.aStart.Tab(); SCCOL nUsedX = 0; // Ende holen SCROW nUsedY = 0; if (!pDocSh->GetDocument()->GetTableArea( nTab, nUsedX, nUsedY )) { nUsedX = 0; nUsedY = 0; } aNewRange.aEnd.SetCol( nUsedX ); aNewRange.aEnd.SetRow( nUsedY ); if (!bExpand) aNewRange.aStart = aNewRange.aEnd; SetNewRange( aNewRange ); } } // XCellCursor void SAL_CALL ScCellCursorObj::gotoStart() throw(uno::RuntimeException) { // this is similar to collapseToCurrentRegion //! something like gotoEdge with 4 possible directions is needed SolarMutexGuard aGuard; const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aOneRange( *rRanges[ 0 ]); aOneRange.Justify(); ScDocShell* pDocSh = GetDocShell(); if ( pDocSh ) { SCCOL nStartCol = aOneRange.aStart.Col(); SCROW nStartRow = aOneRange.aStart.Row(); SCCOL nEndCol = aOneRange.aEnd.Col(); SCROW nEndRow = aOneRange.aEnd.Row(); SCTAB nTab = aOneRange.aStart.Tab(); pDocSh->GetDocument()->GetDataArea( nTab, nStartCol, nStartRow, nEndCol, nEndRow, FALSE, false ); ScRange aNew( nStartCol, nStartRow, nTab ); SetNewRange( aNew ); } } void SAL_CALL ScCellCursorObj::gotoEnd() throw(uno::RuntimeException) { // this is similar to collapseToCurrentRegion //! something like gotoEdge with 4 possible directions is needed SolarMutexGuard aGuard; const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aOneRange( *rRanges[ 0 ] ); aOneRange.Justify(); ScDocShell* pDocSh = GetDocShell(); if ( pDocSh ) { SCCOL nStartCol = aOneRange.aStart.Col(); SCROW nStartRow = aOneRange.aStart.Row(); SCCOL nEndCol = aOneRange.aEnd.Col(); SCROW nEndRow = aOneRange.aEnd.Row(); SCTAB nTab = aOneRange.aStart.Tab(); pDocSh->GetDocument()->GetDataArea( nTab, nStartCol, nStartRow, nEndCol, nEndRow, FALSE, false ); ScRange aNew( nEndCol, nEndRow, nTab ); SetNewRange( aNew ); } } void SAL_CALL ScCellCursorObj::gotoNext() throw(uno::RuntimeException) { SolarMutexGuard aGuard; const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aOneRange( *rRanges[ 0 ] ); aOneRange.Justify(); ScAddress aCursor(aOneRange.aStart); // bei Block immer den Start nehmen ScMarkData aMark; // not used with bMarked=FALSE SCCOL nNewX = aCursor.Col(); SCROW nNewY = aCursor.Row(); SCTAB nTab = aCursor.Tab(); ScDocShell* pDocSh = GetDocShell(); if ( pDocSh ) pDocSh->GetDocument()->GetNextPos( nNewX,nNewY, nTab, 1,0, FALSE,TRUE, aMark ); //! sonst Exception oder so SetNewRange( ScRange( nNewX, nNewY, nTab ) ); } void SAL_CALL ScCellCursorObj::gotoPrevious() throw(uno::RuntimeException) { SolarMutexGuard aGuard; const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aOneRange( *rRanges[ 0 ] ); aOneRange.Justify(); ScAddress aCursor(aOneRange.aStart); // bei Block immer den Start nehmen ScMarkData aMark; // not used with bMarked=FALSE SCCOL nNewX = aCursor.Col(); SCROW nNewY = aCursor.Row(); SCTAB nTab = aCursor.Tab(); ScDocShell* pDocSh = GetDocShell(); if ( pDocSh ) pDocSh->GetDocument()->GetNextPos( nNewX,nNewY, nTab, -1,0, FALSE,TRUE, aMark ); //! sonst Exception oder so SetNewRange( ScRange( nNewX, nNewY, nTab ) ); } void SAL_CALL ScCellCursorObj::gotoOffset( sal_Int32 nColumnOffset, sal_Int32 nRowOffset ) throw(uno::RuntimeException) { SolarMutexGuard aGuard; const ScRangeList& rRanges = GetRangeList(); DBG_ASSERT( rRanges.size() == 1, "Range? Ranges?" ); ScRange aOneRange( *rRanges[ 0 ] ); aOneRange.Justify(); if ( aOneRange.aStart.Col() + nColumnOffset >= 0 && aOneRange.aEnd.Col() + nColumnOffset <= MAXCOL && aOneRange.aStart.Row() + nRowOffset >= 0 && aOneRange.aEnd.Row() + nRowOffset <= MAXROW ) { ScRange aNew( (SCCOL)(aOneRange.aStart.Col() + nColumnOffset), (SCROW)(aOneRange.aStart.Row() + nRowOffset), aOneRange.aStart.Tab(), (SCCOL)(aOneRange.aEnd.Col() + nColumnOffset), (SCROW)(aOneRange.aEnd.Row() + nRowOffset), aOneRange.aEnd.Tab() ); SetNewRange( aNew ); } } // XSheetCellRange uno::Reference SAL_CALL ScCellCursorObj::getSpreadsheet() throw(uno::RuntimeException) { SolarMutexGuard aGuard; return ScCellRangeObj::getSpreadsheet(); } // XCellRange uno::Reference SAL_CALL ScCellCursorObj::getCellByPosition( sal_Int32 nColumn, sal_Int32 nRow ) throw(lang::IndexOutOfBoundsException, uno::RuntimeException) { SolarMutexGuard aGuard; return ScCellRangeObj::getCellByPosition(nColumn,nRow); } uno::Reference SAL_CALL ScCellCursorObj::getCellRangeByPosition( sal_Int32 nLeft, sal_Int32 nTop, sal_Int32 nRight, sal_Int32 nBottom ) throw(lang::IndexOutOfBoundsException, uno::RuntimeException) { SolarMutexGuard aGuard; return ScCellRangeObj::getCellRangeByPosition(nLeft,nTop,nRight,nBottom); } uno::Reference SAL_CALL ScCellCursorObj::getCellRangeByName( const rtl::OUString& rRange ) throw(uno::RuntimeException) { SolarMutexGuard aGuard; return ScCellRangeObj::getCellRangeByName(rRange); } // XServiceInfo rtl::OUString SAL_CALL ScCellCursorObj::getImplementationName() throw(uno::RuntimeException) { return rtl::OUString(RTL_CONSTASCII_USTRINGPARAM( "ScCellCursorObj" )); } sal_Bool SAL_CALL ScCellCursorObj::supportsService( const rtl::OUString& rServiceName ) throw(uno::RuntimeException) { String aServiceStr( rServiceName ); return aServiceStr.EqualsAscii( SCSHEETCELLCURSOR_SERVICE ) || aServiceStr.EqualsAscii( SCCELLCURSOR_SERVICE ) || ScCellRangeObj::supportsService(rServiceName); } uno::Sequence SAL_CALL ScCellCursorObj::getSupportedServiceNames() throw(uno::RuntimeException) { // get all service names from cell range uno::Sequence aParentSeq(ScCellRangeObj::getSupportedServiceNames()); sal_Int32 nParentLen = aParentSeq.getLength(); const rtl::OUString* pParentArr = aParentSeq.getConstArray(); // SheetCellCursor should be first (?) uno::Sequence aTotalSeq( nParentLen + 2 ); rtl::OUString* pTotalArr = aTotalSeq.getArray(); pTotalArr[0] = rtl::OUString(RTL_CONSTASCII_USTRINGPARAM( SCSHEETCELLCURSOR_SERVICE )); pTotalArr[1] = rtl::OUString(RTL_CONSTASCII_USTRINGPARAM( SCCELLCURSOR_SERVICE )); // append cell range services for (long i=0; i