1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include <refupdat.hxx>
21 #include <document.hxx>
22 #include <bigrange.hxx>
23 #include <refdata.hxx>
25 #include <osl/diagnose.h>
27 template< typename R
, typename S
, typename U
>
28 static bool lcl_MoveStart( R
& rRef
, U nStart
, S nDelta
, U nMask
, bool bShrink
= true )
32 rRef
= sal::static_int_cast
<R
>( rRef
+ nDelta
);
33 else if ( nDelta
< 0 && bShrink
&& rRef
>= nStart
+ nDelta
)
34 rRef
= nStart
+ nDelta
; //TODO: limit ???
40 else if ( rRef
> nMask
)
48 template< typename R
, typename S
, typename U
>
49 static bool lcl_MoveEnd( R
& rRef
, U nStart
, S nDelta
, U nMask
, bool bShrink
= true )
53 rRef
= sal::static_int_cast
<R
>( rRef
+ nDelta
);
54 else if ( nDelta
< 0 && bShrink
&& rRef
>= nStart
+ nDelta
)
55 rRef
= nStart
+ nDelta
- 1; //TODO: limit ???
69 template< typename R
, typename S
, typename U
>
70 static bool lcl_MoveReorder( R
& rRef
, U nStart
, U nEnd
, S nDelta
)
72 if ( rRef
>= nStart
&& rRef
<= nEnd
)
74 rRef
= sal::static_int_cast
<R
>( rRef
+ nDelta
);
78 if ( nDelta
> 0 ) // move backward
80 if ( rRef
>= nStart
&& rRef
<= nEnd
+ nDelta
)
83 rRef
= sal::static_int_cast
<R
>( rRef
+ nDelta
); // in the moved range
85 rRef
-= nEnd
- nStart
+ 1; // move up
91 if ( rRef
>= nStart
+ nDelta
&& rRef
<= nEnd
)
94 rRef
= sal::static_int_cast
<R
>( rRef
+ nDelta
); // in the moved range
96 rRef
+= nEnd
- nStart
+ 1; // move up
104 template< typename R
, typename S
, typename U
>
105 static bool lcl_MoveItCut( R
& rRef
, S nDelta
, U nMask
)
108 rRef
= sal::static_int_cast
<R
>( rRef
+ nDelta
);
114 else if ( rRef
> nMask
)
122 template< typename R
, typename U
>
123 static void lcl_MoveItWrap( R
& rRef
, U nMask
)
125 rRef
= sal::static_int_cast
<R
>( rRef
);
128 else if ( rRef
> nMask
)
132 template< typename R
, typename S
, typename U
>
133 static bool IsExpand( R n1
, R n2
, U nStart
, S nD
)
134 { // before normal Move...
137 && n1
< n2
// at least two Cols/Rows/Tabs in Ref
139 (nStart
<= n1
&& n1
< nStart
+ nD
) // n1 within the Insert
140 || (n2
+ 1 == nStart
) // n2 directly before Insert
141 ); // n1 < nStart <= n2 is expanded anyway!
144 template< typename R
, typename S
, typename U
>
145 static void Expand( R
& n1
, R
& n2
, U nStart
, S nD
)
146 { // after normal Move..., only if IsExpand was true before!
148 if ( n2
+ 1 == nStart
)
150 n2
= sal::static_int_cast
<R
>( n2
+ nD
);
154 n1
= sal::static_int_cast
<R
>( n1
- nD
);
157 static bool lcl_IsWrapBig( sal_Int64 nRef
, sal_Int32 nDelta
)
159 if ( nRef
> 0 && nDelta
> 0 )
160 return nRef
+ nDelta
<= 0;
161 else if ( nRef
< 0 && nDelta
< 0 )
162 return nRef
+ nDelta
>= 0;
166 static bool lcl_MoveBig( sal_Int64
& rRef
, sal_Int64 nStart
, sal_Int32 nDelta
)
169 if ( rRef
>= nStart
)
172 bCut
= lcl_IsWrapBig( rRef
, nDelta
);
174 rRef
= ScBigRange::nRangeMax
;
181 static bool lcl_MoveItCutBig( sal_Int64
& rRef
, sal_Int32 nDelta
)
183 bool bCut
= lcl_IsWrapBig( rRef
, nDelta
);
188 ScRefUpdateRes
ScRefUpdate::Update( const ScDocument
* pDoc
, UpdateRefMode eUpdateRefMode
,
189 SCCOL nCol1
, SCROW nRow1
, SCTAB nTab1
,
190 SCCOL nCol2
, SCROW nRow2
, SCTAB nTab2
,
191 SCCOL nDx
, SCROW nDy
, SCTAB nDz
,
192 SCCOL
& theCol1
, SCROW
& theRow1
, SCTAB
& theTab1
,
193 SCCOL
& theCol2
, SCROW
& theRow2
, SCTAB
& theTab2
)
195 ScRefUpdateRes eRet
= UR_NOTHING
;
197 SCCOL oldCol1
= theCol1
;
198 SCROW oldRow1
= theRow1
;
199 SCTAB oldTab1
= theTab1
;
200 SCCOL oldCol2
= theCol2
;
201 SCROW oldRow2
= theRow2
;
202 SCTAB oldTab2
= theTab2
;
206 if (eUpdateRefMode
== URM_INSDEL
)
208 bool bExpand
= pDoc
->IsExpandRefs();
209 if ( nDx
&& (theRow1
>= nRow1
) && (theRow2
<= nRow2
) &&
210 (theTab1
>= nTab1
) && (theTab2
<= nTab2
))
212 bool bExp
= (bExpand
&& IsExpand( theCol1
, theCol2
, nCol1
, nDx
));
213 bCut1
= lcl_MoveStart( theCol1
, nCol1
, nDx
, pDoc
->MaxCol() );
214 bCut2
= lcl_MoveEnd( theCol2
, nCol1
, nDx
, pDoc
->MaxCol() );
215 if ( theCol2
< theCol1
)
220 else if (bCut2
&& theCol2
== 0)
222 else if ( bCut1
|| bCut2
)
226 Expand( theCol1
, theCol2
, nCol1
, nDx
);
229 if (eRet
!= UR_NOTHING
&& oldCol1
== 0 && oldCol2
== pDoc
->MaxCol())
235 else if (oldCol2
== pDoc
->MaxCol() && oldCol1
< pDoc
->MaxCol())
237 // End was sticky, but start may have been moved. Only on range.
239 if (eRet
== UR_NOTHING
)
242 // Else, if (bCut2 && theCol2 == pDoc->MaxCol()) then end becomes sticky,
243 // but currently there's nothing to do.
245 if ( nDy
&& (theCol1
>= nCol1
) && (theCol2
<= nCol2
) &&
246 (theTab1
>= nTab1
) && (theTab2
<= nTab2
))
248 bool bExp
= (bExpand
&& IsExpand( theRow1
, theRow2
, nRow1
, nDy
));
249 bCut1
= lcl_MoveStart( theRow1
, nRow1
, nDy
, pDoc
->MaxRow() );
250 bCut2
= lcl_MoveEnd( theRow2
, nRow1
, nDy
, pDoc
->MaxRow() );
251 if ( theRow2
< theRow1
)
256 else if (bCut2
&& theRow2
== 0)
258 else if ( bCut1
|| bCut2
)
262 Expand( theRow1
, theRow2
, nRow1
, nDy
);
265 if (eRet
!= UR_NOTHING
&& oldRow1
== 0 && oldRow2
== pDoc
->MaxRow())
271 else if (oldRow2
== pDoc
->MaxRow() && oldRow1
< pDoc
->MaxRow())
273 // End was sticky, but start may have been moved. Only on range.
275 if (eRet
== UR_NOTHING
)
278 // Else, if (bCut2 && theRow2 == pDoc->MaxRow()) then end becomes sticky,
279 // but currently there's nothing to do.
281 if ( nDz
&& (theCol1
>= nCol1
) && (theCol2
<= nCol2
) &&
282 (theRow1
>= nRow1
) && (theRow2
<= nRow2
) )
284 SCTAB nMaxTab
= pDoc
->GetTableCount() - 1;
285 nMaxTab
= sal::static_int_cast
<SCTAB
>(nMaxTab
+ nDz
); // adjust to new count
286 bool bExp
= (bExpand
&& IsExpand( theTab1
, theTab2
, nTab1
, nDz
));
287 bCut1
= lcl_MoveStart( theTab1
, nTab1
, nDz
, nMaxTab
, false /*bShrink*/);
288 bCut2
= lcl_MoveEnd( theTab2
, nTab1
, nDz
, nMaxTab
, false /*bShrink*/);
289 if ( theTab2
< theTab1
)
294 else if ( bCut1
|| bCut2
)
298 Expand( theTab1
, theTab2
, nTab1
, nDz
);
303 else if (eUpdateRefMode
== URM_MOVE
)
305 if ((theCol1
>= nCol1
-nDx
) && (theRow1
>= nRow1
-nDy
) && (theTab1
>= nTab1
-nDz
) &&
306 (theCol2
<= nCol2
-nDx
) && (theRow2
<= nRow2
-nDy
) && (theTab2
<= nTab2
-nDz
))
310 bCut1
= lcl_MoveItCut( theCol1
, nDx
, pDoc
->MaxCol() );
311 bCut2
= lcl_MoveItCut( theCol2
, nDx
, pDoc
->MaxCol() );
312 if ( bCut1
|| bCut2
)
314 if (eRet
!= UR_NOTHING
&& oldCol1
== 0 && oldCol2
== pDoc
->MaxCol())
323 bCut1
= lcl_MoveItCut( theRow1
, nDy
, pDoc
->MaxRow() );
324 bCut2
= lcl_MoveItCut( theRow2
, nDy
, pDoc
->MaxRow() );
325 if ( bCut1
|| bCut2
)
327 if (eRet
!= UR_NOTHING
&& oldRow1
== 0 && oldRow2
== pDoc
->MaxRow())
336 SCTAB nMaxTab
= pDoc
->GetTableCount() - 1;
337 bCut1
= lcl_MoveItCut( theTab1
, nDz
, nMaxTab
);
338 bCut2
= lcl_MoveItCut( theTab2
, nDz
, nMaxTab
);
339 if ( bCut1
|| bCut2
)
344 else if (eUpdateRefMode
== URM_REORDER
)
346 // so far only for nDz (MoveTab)
347 OSL_ENSURE ( !nDx
&& !nDy
, "URM_REORDER for x and y not yet implemented" );
349 if ( nDz
&& (theCol1
>= nCol1
) && (theCol2
<= nCol2
) &&
350 (theRow1
>= nRow1
) && (theRow2
<= nRow2
) )
352 bCut1
= lcl_MoveReorder( theTab1
, nTab1
, nTab2
, nDz
);
353 bCut2
= lcl_MoveReorder( theTab2
, nTab1
, nTab2
, nDz
);
354 if ( bCut1
|| bCut2
)
359 if ( eRet
== UR_NOTHING
)
361 if (oldCol1
!= theCol1
362 || oldRow1
!= theRow1
363 || oldTab1
!= theTab1
364 || oldCol2
!= theCol2
365 || oldRow2
!= theRow2
366 || oldTab2
!= theTab2
373 // simple UpdateReference for ScBigRange (ScChangeAction/ScChangeTrack)
374 // References can also be located outside of the document!
375 // Whole columns/rows (ScBigRange::nRangeMin..ScBigRange::nRangeMax) stay as such!
376 ScRefUpdateRes
ScRefUpdate::Update( UpdateRefMode eUpdateRefMode
,
377 const ScBigRange
& rWhere
, sal_Int32 nDx
, sal_Int32 nDy
, sal_Int32 nDz
,
380 ScRefUpdateRes eRet
= UR_NOTHING
;
381 const ScBigRange
aOldRange( rWhat
);
383 sal_Int64 nCol1
, nRow1
, nTab1
, nCol2
, nRow2
, nTab2
;
384 sal_Int64 theCol1
, theRow1
, theTab1
, theCol2
, theRow2
, theTab2
;
385 rWhere
.GetVars( nCol1
, nRow1
, nTab1
, nCol2
, nRow2
, nTab2
);
386 rWhat
.GetVars( theCol1
, theRow1
, theTab1
, theCol2
, theRow2
, theTab2
);
390 if (eUpdateRefMode
== URM_INSDEL
)
392 if ( nDx
&& (theRow1
>= nRow1
) && (theRow2
<= nRow2
) &&
393 (theTab1
>= nTab1
) && (theTab2
<= nTab2
) &&
394 (theCol1
!= ScBigRange::nRangeMin
|| theCol2
!= ScBigRange::nRangeMax
) )
396 bCut1
= lcl_MoveBig( theCol1
, nCol1
, nDx
);
397 bCut2
= lcl_MoveBig( theCol2
, nCol1
, nDx
);
398 if ( bCut1
|| bCut2
)
400 rWhat
.aStart
.SetCol( theCol1
);
401 rWhat
.aEnd
.SetCol( theCol2
);
403 if ( nDy
&& (theCol1
>= nCol1
) && (theCol2
<= nCol2
) &&
404 (theTab1
>= nTab1
) && (theTab2
<= nTab2
) &&
405 (theRow1
!= ScBigRange::nRangeMin
|| theRow2
!= ScBigRange::nRangeMax
) )
407 bCut1
= lcl_MoveBig( theRow1
, nRow1
, nDy
);
408 bCut2
= lcl_MoveBig( theRow2
, nRow1
, nDy
);
409 if ( bCut1
|| bCut2
)
411 rWhat
.aStart
.SetRow( theRow1
);
412 rWhat
.aEnd
.SetRow( theRow2
);
414 if ( nDz
&& (theCol1
>= nCol1
) && (theCol2
<= nCol2
) &&
415 (theRow1
>= nRow1
) && (theRow2
<= nRow2
) &&
416 (theTab1
!= ScBigRange::nRangeMin
|| theTab2
!= ScBigRange::nRangeMax
) )
418 bCut1
= lcl_MoveBig( theTab1
, nTab1
, nDz
);
419 bCut2
= lcl_MoveBig( theTab2
, nTab1
, nDz
);
420 if ( bCut1
|| bCut2
)
422 rWhat
.aStart
.SetTab( theTab1
);
423 rWhat
.aEnd
.SetTab( theTab2
);
426 else if (eUpdateRefMode
== URM_MOVE
)
428 if ( rWhere
.Contains( rWhat
) )
430 if ( nDx
&& (theCol1
!= ScBigRange::nRangeMin
|| theCol2
!= ScBigRange::nRangeMax
) )
432 bCut1
= lcl_MoveItCutBig( theCol1
, nDx
);
433 bCut2
= lcl_MoveItCutBig( theCol2
, nDx
);
434 if ( bCut1
|| bCut2
)
436 rWhat
.aStart
.SetCol( theCol1
);
437 rWhat
.aEnd
.SetCol( theCol2
);
439 if ( nDy
&& (theRow1
!= ScBigRange::nRangeMin
|| theRow2
!= ScBigRange::nRangeMax
) )
441 bCut1
= lcl_MoveItCutBig( theRow1
, nDy
);
442 bCut2
= lcl_MoveItCutBig( theRow2
, nDy
);
443 if ( bCut1
|| bCut2
)
445 rWhat
.aStart
.SetRow( theRow1
);
446 rWhat
.aEnd
.SetRow( theRow2
);
448 if ( nDz
&& (theTab1
!= ScBigRange::nRangeMin
|| theTab2
!= ScBigRange::nRangeMax
) )
450 bCut1
= lcl_MoveItCutBig( theTab1
, nDz
);
451 bCut2
= lcl_MoveItCutBig( theTab2
, nDz
);
452 if ( bCut1
|| bCut2
)
454 rWhat
.aStart
.SetTab( theTab1
);
455 rWhat
.aEnd
.SetTab( theTab2
);
460 if ( eRet
== UR_NOTHING
&& rWhat
!= aOldRange
)
466 void ScRefUpdate::MoveRelWrap( const ScDocument
& rDoc
, const ScAddress
& rPos
,
467 SCCOL nMaxCol
, SCROW nMaxRow
, ScComplexRefData
& rRef
)
469 ScRange aAbsRange
= rRef
.toAbs(rDoc
, rPos
);
470 if( rRef
.Ref1
.IsColRel() )
472 SCCOL nCol
= aAbsRange
.aStart
.Col();
473 lcl_MoveItWrap(nCol
, nMaxCol
);
474 aAbsRange
.aStart
.SetCol(nCol
);
476 if( rRef
.Ref2
.IsColRel() )
478 SCCOL nCol
= aAbsRange
.aEnd
.Col();
479 lcl_MoveItWrap(nCol
, nMaxCol
);
480 aAbsRange
.aEnd
.SetCol(nCol
);
482 if( rRef
.Ref1
.IsRowRel() )
484 SCROW nRow
= aAbsRange
.aStart
.Row();
485 lcl_MoveItWrap(nRow
, nMaxRow
);
486 aAbsRange
.aStart
.SetRow(nRow
);
488 if( rRef
.Ref2
.IsRowRel() )
490 SCROW nRow
= aAbsRange
.aEnd
.Row();
491 lcl_MoveItWrap(nRow
, nMaxRow
);
492 aAbsRange
.aEnd
.SetRow(nRow
);
494 SCTAB nMaxTab
= rDoc
.GetTableCount() - 1;
495 if( rRef
.Ref1
.IsTabRel() )
497 SCTAB nTab
= aAbsRange
.aStart
.Tab();
498 lcl_MoveItWrap(nTab
, nMaxTab
);
499 aAbsRange
.aStart
.SetTab(nTab
);
501 if( rRef
.Ref2
.IsTabRel() )
503 SCTAB nTab
= aAbsRange
.aEnd
.Tab();
504 lcl_MoveItWrap(nTab
, nMaxTab
);
505 aAbsRange
.aEnd
.SetTab(nTab
);
508 aAbsRange
.PutInOrder();
509 rRef
.SetRange(rDoc
.GetSheetLimits(), aAbsRange
, rPos
);
512 void ScRefUpdate::DoTranspose( SCCOL
& rCol
, SCROW
& rRow
, SCTAB
& rTab
,
513 const ScDocument
& rDoc
, const ScRange
& rSource
, const ScAddress
& rDest
)
515 SCTAB nDz
= rDest
.Tab() - rSource
.aStart
.Tab();
518 SCTAB nNewTab
= rTab
+nDz
;
519 SCTAB nCount
= rDoc
.GetTableCount();
520 while (nNewTab
<0) nNewTab
= sal::static_int_cast
<SCTAB
>( nNewTab
+ nCount
);
521 while (nNewTab
>=nCount
) nNewTab
= sal::static_int_cast
<SCTAB
>( nNewTab
- nCount
);
524 OSL_ENSURE( rCol
>=rSource
.aStart
.Col() && rRow
>=rSource
.aStart
.Row(),
525 "UpdateTranspose: pos. wrong" );
527 SCCOL nRelX
= rCol
- rSource
.aStart
.Col();
528 SCROW nRelY
= rRow
- rSource
.aStart
.Row();
530 rCol
= static_cast<SCCOL
>(static_cast<SCCOLROW
>(rDest
.Col()) +
531 static_cast<SCCOLROW
>(nRelY
));
532 rRow
= static_cast<SCROW
>(static_cast<SCCOLROW
>(rDest
.Row()) +
533 static_cast<SCCOLROW
>(nRelX
));
536 ScRefUpdateRes
ScRefUpdate::UpdateTranspose(
537 const ScDocument
& rDoc
, const ScRange
& rSource
, const ScAddress
& rDest
, ScRange
& rRef
)
539 ScRefUpdateRes eRet
= UR_NOTHING
;
540 // Only references in source range must be updated, i.e. no references in destination area.
541 // Otherwise existing references pointing to destination area will be wrongly transposed.
542 if (rSource
.Contains(rRef
))
544 // Source range contains the reference range.
545 SCCOL nCol1
= rRef
.aStart
.Col(), nCol2
= rRef
.aEnd
.Col();
546 SCROW nRow1
= rRef
.aStart
.Row(), nRow2
= rRef
.aEnd
.Row();
547 SCTAB nTab1
= rRef
.aStart
.Tab(), nTab2
= rRef
.aEnd
.Tab();
548 DoTranspose(nCol1
, nRow1
, nTab1
, rDoc
, rSource
, rDest
);
549 DoTranspose(nCol2
, nRow2
, nTab2
, rDoc
, rSource
, rDest
);
550 rRef
.aStart
= ScAddress(nCol1
, nRow1
, nTab1
);
551 rRef
.aEnd
= ScAddress(nCol2
, nRow2
, nTab2
);
557 // UpdateGrow - expands references which point exactly to the area
558 // gets by without document
560 ScRefUpdateRes
ScRefUpdate::UpdateGrow(
561 const ScRange
& rArea
, SCCOL nGrowX
, SCROW nGrowY
, ScRange
& rRef
)
563 ScRefUpdateRes eRet
= UR_NOTHING
;
565 // in y-direction the Ref may also start one row further below,
566 // if an area contains column heads
568 bool bUpdateX
= ( nGrowX
&&
569 rRef
.aStart
.Col() == rArea
.aStart
.Col() && rRef
.aEnd
.Col() == rArea
.aEnd
.Col() &&
570 rRef
.aStart
.Row() >= rArea
.aStart
.Row() && rRef
.aEnd
.Row() <= rArea
.aEnd
.Row() &&
571 rRef
.aStart
.Tab() >= rArea
.aStart
.Tab() && rRef
.aEnd
.Tab() <= rArea
.aEnd
.Tab() );
572 bool bUpdateY
= ( nGrowY
&&
573 rRef
.aStart
.Col() >= rArea
.aStart
.Col() && rRef
.aEnd
.Col() <= rArea
.aEnd
.Col() &&
574 (rRef
.aStart
.Row() == rArea
.aStart
.Row() || rRef
.aStart
.Row() == rArea
.aStart
.Row()+1) &&
575 rRef
.aEnd
.Row() == rArea
.aEnd
.Row() &&
576 rRef
.aStart
.Tab() >= rArea
.aStart
.Tab() && rRef
.aEnd
.Tab() <= rArea
.aEnd
.Tab() );
580 rRef
.aEnd
.SetCol(sal::static_int_cast
<SCCOL
>(rRef
.aEnd
.Col() + nGrowX
));
585 rRef
.aEnd
.SetRow(sal::static_int_cast
<SCROW
>(rRef
.aEnd
.Row() + nGrowY
));
592 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */