xref: /AOO41X/main/sw/source/core/layout/flylay.cxx (revision 43e393b30cac70c7ef8d89b6bedc332e16c86a29)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_sw.hxx"
26 #include "doc.hxx"
27 #include "pagefrm.hxx"
28 #include "rootfrm.hxx"
29 #include "cntfrm.hxx"
30 #include "dview.hxx"
31 #include "dflyobj.hxx"
32 #include "dcontact.hxx"
33 #include "flyfrm.hxx"
34 #include "ftnfrm.hxx"
35 #include "frmtool.hxx"
36 #include "frmfmt.hxx"
37 #include "errhdl.hxx"
38 #include "hints.hxx"
39 #include "pam.hxx"
40 #include "sectfrm.hxx"
41 
42 
43 #include <svx/svdpage.hxx>
44 #include <editeng/ulspitem.hxx>
45 #include <fmtanchr.hxx>
46 #include <fmtornt.hxx>
47 #include <fmtfsize.hxx>
48 #include "ndole.hxx"
49 #include "tabfrm.hxx"
50 #include "flyfrms.hxx"
51 // OD 22.09.2003 #i18732#
52 #include <fmtfollowtextflow.hxx>
53 // OD 29.10.2003 #113049#
54 #include <environmentofanchoredobject.hxx>
55 // OD 2004-05-24 #i28701#
56 #include <sortedobjs.hxx>
57 #include <viewsh.hxx>
58 #include <viewimp.hxx>
59 
60 
61 using namespace ::com::sun::star;
62 
63 
64 /*************************************************************************
65 |*
66 |*  SwFlyFreeFrm::SwFlyFreeFrm(), ~SwFlyFreeFrm()
67 |*
68 |*  Ersterstellung      MA 03. Dec. 92
69 |*  Letzte Aenderung    MA 09. Apr. 99
70 |*
71 |*************************************************************************/
72 
SwFlyFreeFrm(SwFlyFrmFmt * pFmt,SwFrm * pSib,SwFrm * pAnch)73 SwFlyFreeFrm::SwFlyFreeFrm( SwFlyFrmFmt *pFmt, SwFrm* pSib, SwFrm *pAnch ) :
74     SwFlyFrm( pFmt, pSib, pAnch ),
75     pPage( 0 ),
76     // --> OD 2004-11-15 #i34753#
77     mbNoMakePos( false ),
78     // <--
79     // --> OD 2004-11-12 #i37068#
80     mbNoMoveOnCheckClip( false )
81     // <--
82 {
83 }
84 
~SwFlyFreeFrm()85 SwFlyFreeFrm::~SwFlyFreeFrm()
86 {
87     //und Tschuess.
88     // --> OD 2004-06-30 #i28701# - use new method <GetPageFrm()>
89     if( GetPageFrm() )
90     {
91         if( GetFmt()->GetDoc()->IsInDtor() )
92         {
93             // --> OD 2004-06-04 #i29879# - remove also to-frame anchored Writer
94             // fly frame from page.
95             const bool bRemoveFromPage =
96                     GetPageFrm()->GetSortedObjs() &&
97                     ( IsFlyAtCntFrm() ||
98                       ( GetAnchorFrm() && GetAnchorFrm()->IsFlyFrm() ) );
99             if ( bRemoveFromPage )
100             {
101                 GetPageFrm()->GetSortedObjs()->Remove( *this );
102             }
103         }
104         else
105         {
106             SwRect aTmp( GetObjRectWithSpaces() );
107             SwFlyFreeFrm::NotifyBackground( GetPageFrm(), aTmp, PREP_FLY_LEAVE );
108         }
109     }
110 }
111 
112 // --> OD 2004-06-29 #i28701#
113 TYPEINIT1(SwFlyFreeFrm,SwFlyFrm);
114 // <--
115 /*************************************************************************
116 |*
117 |*  SwFlyFreeFrm::NotifyBackground()
118 |*
119 |*  Beschreibung        Benachrichtigt den Hintergrund (alle CntntFrms die
120 |*      gerade ueberlappt werden. Ausserdem wird das Window in einigen
121 |*      Faellen direkt invalidiert (vor allem dort, wo keine CntntFrms
122 |*      ueberlappt werden.
123 |*      Es werden auch die CntntFrms innerhalb von anderen Flys
124 |*      beruecksichtigt.
125 |*  Ersterstellung      MA 03. Dec. 92
126 |*  Letzte Aenderung    MA 26. Aug. 93
127 |*
128 |*************************************************************************/
129 
NotifyBackground(SwPageFrm * pPageFrm,const SwRect & rRect,PrepareHint eHint)130 void SwFlyFreeFrm::NotifyBackground( SwPageFrm *pPageFrm,
131                                      const SwRect& rRect, PrepareHint eHint )
132 {
133     ::Notify_Background( GetVirtDrawObj(), pPageFrm, rRect, eHint, sal_True );
134 }
135 
136 /*************************************************************************
137 |*
138 |*  SwFlyFreeFrm::MakeAll()
139 |*
140 |*  Ersterstellung      MA 18. Feb. 94
141 |*  Letzte Aenderung    MA 03. Mar. 97
142 |*
143 |*************************************************************************/
144 
MakeAll()145 void SwFlyFreeFrm::MakeAll()
146 {
147     // OD 2004-01-19 #110582#
148     if ( !GetFmt()->GetDoc()->IsVisibleLayerId( GetVirtDrawObj()->GetLayer() ) )
149     {
150         return;
151     }
152 
153     if ( !GetAnchorFrm() || IsLocked() || IsColLocked() )
154         return;
155     // --> OD 2004-06-30 #i28701# - use new method <GetPageFrm()>
156     if( !GetPageFrm() && GetAnchorFrm() && GetAnchorFrm()->IsInFly() )
157     {
158         SwFlyFrm* pFly = AnchorFrm()->FindFlyFrm();
159         SwPageFrm *pPageFrm = pFly ? pFly->FindPageFrm() : NULL;
160         if( pPageFrm )
161             pPageFrm->AppendFlyToPage( this );
162     }
163     if( !GetPageFrm() )
164         return;
165 
166     Lock(); //Der Vorhang faellt
167 
168     //uebernimmt im DTor die Benachrichtigung
169     const SwFlyNotify aNotify( this );
170 
171     if ( IsClipped() )
172     {
173         bValidSize = bHeightClipped = bWidthClipped = sal_False;
174         // --> OD 2004-11-03 #114798# - no invalidation of position,
175         // if anchored object is anchored inside a Writer fly frame,
176         // its position is already locked, and it follows the text flow.
177         // --> OD 2004-11-15 #i34753# - add condition:
178         // no invalidation of position, if no direct move is requested in <CheckClip(..)>
179         if ( !IsNoMoveOnCheckClip() &&
180              !( PositionLocked() &&
181                 GetAnchorFrm()->IsInFly() &&
182                 GetFrmFmt().GetFollowTextFlow().GetValue() ) )
183         // <--
184         {
185             bValidPos = sal_False;
186         }
187         // <--
188     }
189 
190     // FME 2007-08-30 #i81146# new loop control
191     sal_uInt16 nLoopControlRuns = 0;
192     const sal_uInt16 nLoopControlMax = 10;
193 
194     while ( !bValidPos || !bValidSize || !bValidPrtArea || bFormatHeightOnly )
195     {
196         SWRECTFN( this )
197         const SwFmtFrmSize *pSz;
198         {   //Zusaetzlicher Scope, damit aAccess vor dem Check zerstoert wird!
199 
200             SwBorderAttrAccess aAccess( SwFrm::GetCache(), this );
201             const SwBorderAttrs &rAttrs = *aAccess.Get();
202             pSz = &rAttrs.GetAttrSet().GetFrmSize();
203 
204             //Nur einstellen wenn das Flag gesetzt ist!!
205             if ( !bValidSize )
206             {
207                 bValidPrtArea = sal_False;
208 /*
209                 // This is also done in the Format function, so I think
210                 // this code is not necessary anymore:
211                 const Size aRelSize( CalcRel( *pSz ) );
212                 const SwTwips nMin = MINFLY + rAttrs.CalcLeftLine()+rAttrs.CalcRightLine();
213                 long nDiff = bVert ? aRelSize.Height() : aRelSize.Width();
214                 if( nDiff < nMin )
215                     nDiff = nMin;
216                 nDiff -= (aFrm.*fnRect->fnGetWidth)();
217                 if( nDiff )
218                 {
219                     (aFrm.*fnRect->fnAddRight)( nDiff );
220                     bValidPos = sal_False;
221                 }
222 */
223             }
224 
225             if ( !bValidPrtArea )
226                 MakePrtArea( rAttrs );
227 
228             if ( !bValidSize || bFormatHeightOnly )
229             {
230                 bValidSize = sal_False;
231                 Format( &rAttrs );
232                 bFormatHeightOnly = sal_False;
233             }
234 
235             if ( !bValidPos )
236             {
237                 const Point aOldPos( (Frm().*fnRect->fnGetPos)() );
238                 // OD 2004-03-23 #i26791# - use new method <MakeObjPos()>
239                 // --> OD 2004-11-15 #i34753# - no positioning, if requested.
240                 if ( IsNoMakePos() )
241                     bValidPos = sal_True;
242                 else
243                     // OD 2004-03-23 #i26791# - use new method <MakeObjPos()>
244                     MakeObjPos();
245                 // <--
246                 if( aOldPos == (Frm().*fnRect->fnGetPos)() )
247                 {
248                     if( !bValidPos && GetAnchorFrm()->IsInSct() &&
249                         !GetAnchorFrm()->FindSctFrm()->IsValid() )
250                         bValidPos = sal_True;
251                 }
252                 else
253                     bValidSize = sal_False;
254             }
255         }
256 
257         if ( bValidPos && bValidSize )
258         {
259             ++nLoopControlRuns;
260 
261 #if OSL_DEBUG_LEVEL > 1
262             ASSERT( nLoopControlRuns < nLoopControlMax, "LoopControl in SwFlyFreeFrm::MakeAll" )
263 #endif
264 
265             if ( nLoopControlRuns < nLoopControlMax )
266                 CheckClip( *pSz );
267         }
268         else
269             nLoopControlRuns = 0;
270     }
271     Unlock();
272 
273 #ifdef DBG_UTIL
274     SWRECTFN( this )
275     ASSERT( bHeightClipped || ( (Frm().*fnRect->fnGetHeight)() > 0 &&
276             (Prt().*fnRect->fnGetHeight)() > 0),
277             "SwFlyFreeFrm::Format(), flipping Fly." );
278 
279 #endif
280 }
281 
282 /** determines, if direct environment of fly frame has 'auto' size
283 
284     OD 07.08.2003 #i17297#, #111066#, #111070#
285     start with anchor frame and search via <GetUpper()> for a header, footer,
286     row or fly frame stopping at page frame.
287     return <true>, if such a frame is found and it has 'auto' size.
288     otherwise <false> is returned.
289 
290     @author OD
291 
292     @return boolean indicating, that direct environment has 'auto' size
293 */
HasEnvironmentAutoSize() const294 bool SwFlyFreeFrm::HasEnvironmentAutoSize() const
295 {
296     bool bRetVal = false;
297 
298     const SwFrm* pToBeCheckedFrm = GetAnchorFrm();
299     while ( pToBeCheckedFrm &&
300             !pToBeCheckedFrm->IsPageFrm() )
301     {
302         if ( pToBeCheckedFrm->IsHeaderFrm() ||
303              pToBeCheckedFrm->IsFooterFrm() ||
304              pToBeCheckedFrm->IsRowFrm() ||
305              pToBeCheckedFrm->IsFlyFrm() )
306         {
307             bRetVal = ATT_FIX_SIZE !=
308                       pToBeCheckedFrm->GetAttrSet()->GetFrmSize().GetHeightSizeType();
309             break;
310         }
311         else
312         {
313             pToBeCheckedFrm = pToBeCheckedFrm->GetUpper();
314         }
315     }
316 
317     return bRetVal;
318 }
319 
320 /*************************************************************************
321 |*
322 |*  SwFlyFreeFrm::CheckClip()
323 |*
324 |*  Ersterstellung      MA 21. Feb. 94
325 |*  Letzte Aenderung    MA 03. Mar. 97
326 |*
327 |*************************************************************************/
328 
CheckClip(const SwFmtFrmSize & rSz)329 void SwFlyFreeFrm::CheckClip( const SwFmtFrmSize &rSz )
330 {
331     //Jetzt ist es ggf. an der Zeit geignete Massnahmen zu ergreifen wenn
332     //der Fly nicht in seine Umgebung passt.
333     //Zuerst gibt der Fly seine Position auf. Danach wird er zunaechst
334     //formatiert. Erst wenn er auch durch die Aufgabe der Position nicht
335     //passt wird die Breite oder Hoehe aufgegeben - der Rahmen wird soweit
336     //wie notwendig zusammengequetscht.
337 
338     const SwVirtFlyDrawObj *pObj = GetVirtDrawObj();
339     SwRect aClip, aTmpStretch;
340     ::CalcClipRect( pObj, aClip, sal_True );
341     ::CalcClipRect( pObj, aTmpStretch, sal_False );
342     aClip._Intersection( aTmpStretch );
343 
344     const long nBot = Frm().Top() + Frm().Height();
345     const long nRig = Frm().Left() + Frm().Width();
346     const long nClipBot = aClip.Top() + aClip.Height();
347     const long nClipRig = aClip.Left() + aClip.Width();
348 
349     const sal_Bool bBot = nBot > nClipBot;
350     const sal_Bool bRig = nRig > nClipRig;
351     if ( bBot || bRig )
352     {
353         sal_Bool bAgain = sal_False;
354         // --> OD 2004-11-12 #i37068# - no move, if it's requested
355         if ( bBot && !IsNoMoveOnCheckClip() &&
356              !GetDrawObjs() && !GetAnchorFrm()->IsInTab() )
357         // <--
358         {
359             SwFrm* pHeader = FindFooterOrHeader();
360             // In a header, correction of the position is no good idea.
361             // If the fly moves, some paragraphs has to be formatted, this
362             // could cause a change of the height of the headerframe,
363             // now the flyframe can change its position and so on ...
364             if ( !pHeader || !pHeader->IsHeaderFrm() )
365             {
366                 const long nOld = Frm().Top();
367                 Frm().Pos().Y() = Max( aClip.Top(), nClipBot - Frm().Height() );
368                 if ( Frm().Top() != nOld )
369                     bAgain = sal_True;
370                 bHeightClipped = sal_True;
371             }
372         }
373         if ( bRig )
374         {
375             const long nOld = Frm().Left();
376             Frm().Pos().X() = Max( aClip.Left(), nClipRig - Frm().Width() );
377             if ( Frm().Left() != nOld )
378             {
379                 const SwFmtHoriOrient &rH = GetFmt()->GetHoriOrient();
380                 // Links ausgerichtete duerfen nicht nach links verschoben werden,
381                 // wenn sie einem anderen ausweichen.
382                 if( rH.GetHoriOrient() == text::HoriOrientation::LEFT )
383                     Frm().Pos().X() = nOld;
384                 else
385                     bAgain = sal_True;
386             }
387             bWidthClipped = sal_True;
388         }
389         if ( bAgain )
390             bValidSize = sal_False;
391         else
392         {
393             //Wenn wir hier ankommen ragt der Frm in unerlaubte Bereiche
394             //hinein, und eine Positionskorrektur ist nicht erlaubt bzw.
395             //moeglich oder noetig.
396 
397             //Fuer Flys mit OLE-Objekten als Lower sorgen wir dafuer, dass
398             //immer proportional Resized wird.
399             Size aOldSize( Frm().SSize() );
400 
401             //Zuerst wird das FrmRect eingestellt, und dann auf den Frm
402             //uebertragen.
403             SwRect aFrmRect( Frm() );
404 
405             if ( bBot )
406             {
407                 long nDiff = nClipBot;
408                 nDiff -= aFrmRect.Top(); //nDiff ist die verfuegbare Strecke.
409                 nDiff = aFrmRect.Height() - nDiff;
410                 aFrmRect.Height( aFrmRect.Height() - nDiff );
411                 bHeightClipped = sal_True;
412             }
413             if ( bRig )
414             {
415                 long nDiff = nClipRig;
416                 nDiff -= aFrmRect.Left();//nDiff ist die verfuegbare Strecke.
417                 nDiff = aFrmRect.Width() - nDiff;
418                 aFrmRect.Width( aFrmRect.Width() - nDiff );
419                 bWidthClipped = sal_True;
420             }
421 
422             // OD 06.08.2003 #i17297#, #111066#, #111070# - no proportional
423             // scaling of graphics in environments, which determines its size
424             // by its content ('auto' size). Otherwise layout loops can occur and
425             // layout sizes of the environment can be incorrect.
426             // Such environment are:
427             // (1) header and footer frames with 'auto' size
428             // (2) table row frames with 'auto' size
429             // (3) fly frames with 'auto' size
430             // Note: section frames seems to be not critical - didn't found
431             //       any critical layout situation so far.
432             if ( Lower() && Lower()->IsNoTxtFrm() &&
433                  ( static_cast<SwCntntFrm*>(Lower())->GetNode()->GetOLENode() ||
434                    !HasEnvironmentAutoSize() ) )
435             {
436                 //Wenn Breite und Hoehe angepasst wurden, so ist die
437                 //groessere Veraenderung massgeblich.
438                 if ( aFrmRect.Width() != aOldSize.Width() &&
439                      aFrmRect.Height()!= aOldSize.Height() )
440                 {
441                     if ( (aOldSize.Width() - aFrmRect.Width()) >
442                          (aOldSize.Height()- aFrmRect.Height()) )
443                         aFrmRect.Height( aOldSize.Height() );
444                     else
445                         aFrmRect.Width( aOldSize.Width() );
446                 }
447 
448                 //Breite angepasst? - Hoehe dann proportional verkleinern
449                 if( aFrmRect.Width() != aOldSize.Width() )
450                 {
451                     aFrmRect.Height( aFrmRect.Width() * aOldSize.Height() /
452                                      aOldSize.Width() );
453                     bHeightClipped = sal_True;
454                 }
455                 //Hoehe angepasst? - Breite dann proportional verkleinern
456                 else if( aFrmRect.Height() != aOldSize.Height() )
457                 {
458                     aFrmRect.Width( aFrmRect.Height() * aOldSize.Width() /
459                                     aOldSize.Height() );
460                     bWidthClipped = sal_True;
461                 }
462 
463                 // OD 07.08.2003 #i17297#, #111066#, #111070# - reactivate change
464                 // of size attribute for fly frames containing an ole object.
465                 // FME: 2004-05-19 Added the aFrmRect.HasArea() hack, because
466                 // the environment of the ole object does not have to be valid
467                 // at this moment, or even worse, it does not have to have a
468                 // resonable size. In this case we do not want to change to
469                 // attributes permanentely. Maybe one day somebody dares to remove
470                 // this code.
471                 if ( aFrmRect.HasArea() &&
472                      static_cast<SwCntntFrm*>(Lower())->GetNode()->GetOLENode() &&
473                      ( bWidthClipped || bHeightClipped ) )
474                 {
475                     SwFlyFrmFmt *pFmt = (SwFlyFrmFmt*)GetFmt();
476                     pFmt->LockModify();
477                     SwFmtFrmSize aFrmSize( rSz );
478                     aFrmSize.SetWidth( aFrmRect.Width() );
479                     aFrmSize.SetHeight( aFrmRect.Height() );
480                     pFmt->SetFmtAttr( aFrmSize );
481                     pFmt->UnlockModify();
482                 }
483             }
484 
485             //Jetzt die Einstellungen am Frm vornehmen, bei Spalten werden
486             //die neuen Werte in die Attribute eingetragen, weil es sonst
487             //ziemlich fiese Oszillationen gibt.
488             const long nPrtHeightDiff = Frm().Height() - Prt().Height();
489             const long nPrtWidthDiff  = Frm().Width()  - Prt().Width();
490             Frm().Height( aFrmRect.Height() );
491             Frm().Width ( Max( long(MINLAY), aFrmRect.Width() ) );
492             if ( Lower() && Lower()->IsColumnFrm() )
493             {
494                 ColLock();  //Grow/Shrink locken.
495                 const Size aTmpOldSize( Prt().SSize() );
496                 Prt().Height( Frm().Height() - nPrtHeightDiff );
497                 Prt().Width ( Frm().Width()  - nPrtWidthDiff );
498                 ChgLowersProp( aTmpOldSize );
499                 SwFrm *pLow = Lower();
500                 do
501                 {   pLow->Calc();
502                     // auch den (Column)BodyFrm mitkalkulieren
503                     ((SwLayoutFrm*)pLow)->Lower()->Calc();
504                     pLow = pLow->GetNext();
505                 } while ( pLow );
506                 ::CalcCntnt( this );
507                 ColUnlock();
508                 if ( !bValidSize && !bWidthClipped )
509                     bFormatHeightOnly = bValidSize = sal_True;
510             }
511             else
512             {
513                 Prt().Height( Frm().Height() - nPrtHeightDiff );
514                 Prt().Width ( Frm().Width()  - nPrtWidthDiff );
515             }
516         }
517     }
518 
519     // --> OD 2004-10-14 #i26945#
520     ASSERT( Frm().Height() >= 0,
521             "<SwFlyFreeFrm::CheckClip(..)> - fly frame has negative height now." );
522     // <--
523 }
524 
525 /** method to determine, if a <MakeAll()> on the Writer fly frame is possible
526 
527     OD 2005-03-03 #i43771#
528 
529     @author OD
530 */
IsFormatPossible() const531 bool SwFlyFreeFrm::IsFormatPossible() const
532 {
533     return SwFlyFrm::IsFormatPossible() &&
534            ( GetPageFrm() ||
535              ( GetAnchorFrm() && GetAnchorFrm()->IsInFly() ) );
536 }
537 
538 /*************************************************************************
539 |*
540 |*  SwFlyLayFrm::SwFlyLayFrm()
541 |*
542 |*  Ersterstellung      MA 25. Aug. 92
543 |*  Letzte Aenderung    MA 09. Apr. 99
544 |*
545 |*************************************************************************/
546 
SwFlyLayFrm(SwFlyFrmFmt * pFmt,SwFrm * pSib,SwFrm * pAnch)547 SwFlyLayFrm::SwFlyLayFrm( SwFlyFrmFmt *pFmt, SwFrm* pSib, SwFrm *pAnch ) :
548     SwFlyFreeFrm( pFmt, pSib, pAnch )
549 {
550     bLayout = sal_True;
551 }
552 
553 // --> OD 2004-06-29 #i28701#
554 TYPEINIT1(SwFlyLayFrm,SwFlyFreeFrm);
555 // <--
556 /*************************************************************************
557 |*
558 |*  SwFlyLayFrm::Modify()
559 |*
560 |*  Ersterstellung      MA 08. Feb. 93
561 |*  Letzte Aenderung    MA 28. Aug. 93
562 |*
563 |*************************************************************************/
564 
Modify(const SfxPoolItem * pOld,const SfxPoolItem * pNew)565 void SwFlyLayFrm::Modify( const SfxPoolItem* pOld, const SfxPoolItem *pNew )
566 {
567     sal_uInt16 nWhich = pNew ? pNew->Which() : 0;
568 
569     SwFmtAnchor *pAnch = 0;
570     if( RES_ATTRSET_CHG == nWhich && SFX_ITEM_SET ==
571         ((SwAttrSetChg*)pNew)->GetChgSet()->GetItemState( RES_ANCHOR, sal_False,
572             (const SfxPoolItem**)&pAnch ))
573         ;       // Beim GetItemState wird der AnkerPointer gesetzt !
574 
575     else if( RES_ANCHOR == nWhich )
576     {
577         //Ankerwechsel, ich haenge mich selbst um.
578         //Es darf sich nicht um einen Wechsel des Ankertyps handeln,
579         //dies ist nur ueber die SwFEShell moeglich.
580         pAnch = (SwFmtAnchor*)pNew;
581     }
582 
583     if( pAnch )
584     {
585         ASSERT( pAnch->GetAnchorId() ==
586                 GetFmt()->GetAnchor().GetAnchorId(),
587                 "8-) Unzulaessiger Wechsel des Ankertyps." );
588 
589         //Abmelden, Seite besorgen, an den entsprechenden LayoutFrm
590         //haengen.
591         SwRect aOld( GetObjRectWithSpaces() );
592         // --> OD 2004-06-30 #i28701# - use new method <GetPageFrm()>
593         SwPageFrm *pOldPage = GetPageFrm();
594         AnchorFrm()->RemoveFly( this );
595 
596         if ( FLY_AT_PAGE == pAnch->GetAnchorId() )
597         {
598             sal_uInt16 nPgNum = pAnch->GetPageNum();
599             SwRootFrm *pRoot = getRootFrm();
600             SwPageFrm *pTmpPage = (SwPageFrm*)pRoot->Lower();
601             for ( sal_uInt16 i = 1; (i <= nPgNum) && pTmpPage; ++i,
602                                 pTmpPage = (SwPageFrm*)pTmpPage->GetNext() )
603             {
604                 if ( i == nPgNum )
605                 {
606                     // --> OD 2005-06-09 #i50432# - adjust synopsis of <PlaceFly(..)>
607                     pTmpPage->PlaceFly( this, 0 );
608                     // <--
609                 }
610             }
611             if( !pTmpPage )
612             {
613                 pRoot->SetAssertFlyPages();
614                 pRoot->AssertFlyPages();
615             }
616         }
617         else
618         {
619             SwNodeIndex aIdx( pAnch->GetCntntAnchor()->nNode );
620             SwCntntFrm *pCntnt = GetFmt()->GetDoc()->GetNodes().GoNext( &aIdx )->
621                          GetCntntNode()->getLayoutFrm( getRootFrm(), 0, 0, sal_False );
622             if( pCntnt )
623             {
624                 SwFlyFrm *pTmp = pCntnt->FindFlyFrm();
625                 if( pTmp )
626                     pTmp->AppendFly( this );
627             }
628         }
629         // --> OD 2004-06-30 #i28701# - use new method <GetPageFrm()>
630         if ( pOldPage && pOldPage != GetPageFrm() )
631             NotifyBackground( pOldPage, aOld, PREP_FLY_LEAVE );
632         SetCompletePaint();
633         InvalidateAll();
634         SetNotifyBack();
635     }
636     else
637         SwFlyFrm::Modify( pOld, pNew );
638 }
639 
640 /*************************************************************************
641 |*
642 |*  SwPageFrm::AppendFly()
643 |*
644 |*  Ersterstellung      MA 10. Oct. 92
645 |*  Letzte Aenderung    MA 08. Jun. 96
646 |*
647 |*************************************************************************/
648 
AppendFlyToPage(SwFlyFrm * pNew)649 void SwPageFrm::AppendFlyToPage( SwFlyFrm *pNew )
650 {
651     if ( !pNew->GetVirtDrawObj()->IsInserted() )
652         getRootFrm()->GetDrawPage()->InsertObject(
653                 (SdrObject*)pNew->GetVirtDrawObj(),
654                 pNew->GetVirtDrawObj()->GetReferencedObj().GetOrdNumDirect() );
655 
656     InvalidateSpelling();
657     InvalidateSmartTags();  // SMARTTAGS
658     InvalidateAutoCompleteWords();
659     InvalidateWordCount();
660 
661     if ( GetUpper() )
662     {
663         ((SwRootFrm*)GetUpper())->SetIdleFlags();
664         ((SwRootFrm*)GetUpper())->InvalidateBrowseWidth();
665     }
666 
667     SdrObject* pObj = pNew->GetVirtDrawObj();
668     ASSERT( pNew->GetAnchorFrm(), "Fly without Anchor" );
669     SwFlyFrm* pFly = (SwFlyFrm*)pNew->GetAnchorFrm()->FindFlyFrm();
670     if ( pFly && pObj->GetOrdNum() < pFly->GetVirtDrawObj()->GetOrdNum() )
671     {
672         //#i119945# set pFly's OrdNum to _rNewObj's. So when pFly is removed by Undo, the original OrdNum will not be changed.
673         sal_uInt32 nNewNum = pObj->GetOrdNumDirect();
674         if ( pObj->GetPage() )
675             pObj->GetPage()->SetObjectOrdNum( pFly->GetVirtDrawObj()->GetOrdNumDirect(), nNewNum );
676         else
677             pFly->GetVirtDrawObj()->SetOrdNum( nNewNum );
678     }
679 
680     //Flys die im Cntnt sitzen beachten wir nicht weiter.
681     if ( pNew->IsFlyInCntFrm() )
682         InvalidateFlyInCnt();
683     else
684     {
685         InvalidateFlyCntnt();
686 
687         if ( !pSortedObjs )
688             pSortedObjs = new SwSortedObjs();
689 
690 #if OSL_DEBUG_LEVEL > 1
691         const bool bSucessInserted =
692 #endif
693         pSortedObjs->Insert( *pNew );
694 #if OSL_DEBUG_LEVEL > 1
695         ASSERT( bSucessInserted, "Fly nicht in Sorted eingetragen." )
696         (void) bSucessInserted;
697 #endif
698 
699         // --> OD 2008-04-22 #i87493#
700         ASSERT( pNew->GetPageFrm() == 0 || pNew->GetPageFrm() == this,
701                 "<SwPageFrm::AppendFlyToPage(..)> - anchored fly frame seems to be registered at another page frame. Serious defect -> please inform OD." );
702         // <--
703         // --> OD 2004-06-30 #i28701# - use new method <SetPageFrm(..)>
704         pNew->SetPageFrm( this );
705         pNew->InvalidatePage( this );
706         // OD 2004-05-17 #i28701#
707         pNew->UnlockPosition();
708 
709         // Notify accessible layout. That's required at this place for
710         // frames only where the anchor is moved. Creation of new frames
711         // is additionally handled by the SwFrmNotify class.
712         if( GetUpper() &&
713             static_cast< SwRootFrm * >( GetUpper() )->IsAnyShellAccessible() &&
714             static_cast< SwRootFrm * >( GetUpper() )->GetCurrShell() )
715         {
716             static_cast< SwRootFrm * >( GetUpper() )->GetCurrShell()->Imp()
717                                       ->AddAccessibleFrm( pNew );
718         }
719     }
720 
721     // --> OD 2004-06-09 #i28701# - correction: consider also drawing objects
722     if ( pNew->GetDrawObjs() )
723     {
724         SwSortedObjs &rObjs = *pNew->GetDrawObjs();
725         for ( sal_uInt16 i = 0; i < rObjs.Count(); ++i )
726         {
727             SwAnchoredObject* pTmpObj = rObjs[i];
728             if ( pTmpObj->ISA(SwFlyFrm) )
729             {
730                 SwFlyFrm* pTmpFly = static_cast<SwFlyFrm*>(pTmpObj);
731                 // --> OD 2004-06-30 #i28701# - use new method <GetPageFrm()>
732                 if ( pTmpFly->IsFlyFreeFrm() && !pTmpFly->GetPageFrm() )
733                     AppendFlyToPage( pTmpFly );
734             }
735             else if ( pTmpObj->ISA(SwAnchoredDrawObject) )
736             {
737                 // --> OD 2008-04-22 #i87493#
738 //                AppendDrawObjToPage( *pTmpObj );
739                 if ( pTmpObj->GetPageFrm() != this )
740                 {
741                     if ( pTmpObj->GetPageFrm() != 0 )
742                     {
743                         pTmpObj->GetPageFrm()->RemoveDrawObjFromPage( *pTmpObj );
744                     }
745                     AppendDrawObjToPage( *pTmpObj );
746                 }
747                 // <--
748             }
749         }
750     }
751 }
752 
753 /*************************************************************************
754 |*
755 |*  SwPageFrm::RemoveFly()
756 |*
757 |*  Ersterstellung      MA 10. Oct. 92
758 |*  Letzte Aenderung    MA 26. Aug. 96
759 |*
760 |*************************************************************************/
761 
RemoveFlyFromPage(SwFlyFrm * pToRemove)762 void SwPageFrm::RemoveFlyFromPage( SwFlyFrm *pToRemove )
763 {
764     const sal_uInt32 nOrdNum = pToRemove->GetVirtDrawObj()->GetOrdNum();
765     getRootFrm()->GetDrawPage()->RemoveObject( nOrdNum );
766     pToRemove->GetVirtDrawObj()->ReferencedObj().SetOrdNum( nOrdNum );
767 
768     if ( GetUpper() )
769     {
770         if ( !pToRemove->IsFlyInCntFrm() )
771             ((SwRootFrm*)GetUpper())->SetSuperfluous();
772         ((SwRootFrm*)GetUpper())->InvalidateBrowseWidth();
773     }
774 
775     //Flys die im Cntnt sitzen beachten wir nicht weiter.
776     if ( pToRemove->IsFlyInCntFrm() )
777         return;
778 
779     // Notify accessible layout. That's required at this place for
780     // frames only where the anchor is moved. Creation of new frames
781     // is additionally handled by the SwFrmNotify class.
782     if( GetUpper() &&
783         static_cast< SwRootFrm * >( GetUpper() )->IsAnyShellAccessible() &&
784         static_cast< SwRootFrm * >( GetUpper() )->GetCurrShell() )
785     {
786         static_cast< SwRootFrm * >( GetUpper() )->GetCurrShell()->Imp()
787                                   ->DisposeAccessibleFrm( pToRemove, sal_True );
788     }
789 
790     //Collections noch nicht loeschen. Das passiert am Ende
791     //der Action im RemoveSuperfluous der Seite - angestossen von gleich-
792     //namiger Methode der Root.
793     //Die FlyColl kann bereits weg sein, weil der DTor der Seite
794     //gerade 'laeuft'
795     if ( pSortedObjs )
796     {
797         pSortedObjs->Remove( *pToRemove );
798         if ( !pSortedObjs->Count() )
799         {   DELETEZ( pSortedObjs );
800         }
801     }
802     // --> OD 2004-06-30 #i28701# - use new method <SetPageFrm(..)>
803     pToRemove->SetPageFrm( 0L );
804 }
805 
806 /*************************************************************************
807 |*
808 |*  SwPageFrm::MoveFly
809 |*
810 |*  Ersterstellung      MA 25. Jan. 97
811 |*  Letzte Aenderung    MA 25. Jan. 97
812 |*
813 |*************************************************************************/
814 
MoveFly(SwFlyFrm * pToMove,SwPageFrm * pDest)815 void SwPageFrm::MoveFly( SwFlyFrm *pToMove, SwPageFrm *pDest )
816 {
817     //Invalidierungen
818     if ( GetUpper() )
819     {
820         ((SwRootFrm*)GetUpper())->SetIdleFlags();
821         if ( !pToMove->IsFlyInCntFrm() && pDest->GetPhyPageNum() < GetPhyPageNum() )
822             ((SwRootFrm*)GetUpper())->SetSuperfluous();
823     }
824 
825     pDest->InvalidateSpelling();
826     pDest->InvalidateSmartTags();   // SMARTTAGS
827     pDest->InvalidateAutoCompleteWords();
828     pDest->InvalidateWordCount();
829 
830     if ( pToMove->IsFlyInCntFrm() )
831     {
832         pDest->InvalidateFlyInCnt();
833         return;
834     }
835 
836     // Notify accessible layout. That's required at this place for
837     // frames only where the anchor is moved. Creation of new frames
838     // is additionally handled by the SwFrmNotify class.
839     if( GetUpper() &&
840         static_cast< SwRootFrm * >( GetUpper() )->IsAnyShellAccessible() &&
841         static_cast< SwRootFrm * >( GetUpper() )->GetCurrShell() )
842     {
843         static_cast< SwRootFrm * >( GetUpper() )->GetCurrShell()->Imp()
844                                   ->DisposeAccessibleFrm( pToMove, sal_True );
845     }
846 
847     //Die FlyColl kann bereits weg sein, weil der DTor der Seite
848     //gerade 'laeuft'
849     if ( pSortedObjs )
850     {
851         pSortedObjs->Remove( *pToMove );
852         if ( !pSortedObjs->Count() )
853         {   DELETEZ( pSortedObjs );
854         }
855     }
856 
857     //Anmelden
858     if ( !pDest->GetSortedObjs() )
859         pDest->pSortedObjs = new SwSortedObjs();
860 
861 #if OSL_DEBUG_LEVEL > 1
862     const bool bSucessInserted =
863 #endif
864     pDest->GetSortedObjs()->Insert( *pToMove );
865 #if OSL_DEBUG_LEVEL > 1
866     ASSERT( bSucessInserted, "Fly nicht in Sorted eingetragen." )
867     (void) bSucessInserted;
868 #endif
869 
870     // --> OD 2004-06-30 #i28701# - use new method <SetPageFrm(..)>
871     pToMove->SetPageFrm( pDest );
872     pToMove->InvalidatePage( pDest );
873     pToMove->SetNotifyBack();
874     pDest->InvalidateFlyCntnt();
875     // OD 2004-05-17 #i28701#
876     pToMove->UnlockPosition();
877 
878     // Notify accessible layout. That's required at this place for
879     // frames only where the anchor is moved. Creation of new frames
880     // is additionally handled by the SwFrmNotify class.
881     if( GetUpper() &&
882         static_cast< SwRootFrm * >( GetUpper() )->IsAnyShellAccessible() &&
883         static_cast< SwRootFrm * >( GetUpper() )->GetCurrShell() )
884     {
885         static_cast< SwRootFrm * >( GetUpper() )->GetCurrShell()->Imp()
886                                   ->AddAccessibleFrm( pToMove );
887     }
888 
889     // --> OD 2004-06-09 #i28701# - correction: move lowers of Writer fly frame
890     if ( pToMove->GetDrawObjs() )
891     {
892         SwSortedObjs &rObjs = *pToMove->GetDrawObjs();
893         for ( sal_uInt32 i = 0; i < rObjs.Count(); ++i )
894         {
895             SwAnchoredObject* pObj = rObjs[i];
896             if ( pObj->ISA(SwFlyFrm) )
897             {
898                 SwFlyFrm* pFly = static_cast<SwFlyFrm*>(pObj);
899                 if ( pFly->IsFlyFreeFrm() )
900                 {
901                     // --> OD 2004-06-30 #i28701# - use new method <GetPageFrm()>
902                     SwPageFrm* pPageFrm = pFly->GetPageFrm();
903                     if ( pPageFrm )
904                         pPageFrm->MoveFly( pFly, pDest );
905                     else
906                         pDest->AppendFlyToPage( pFly );
907                 }
908             }
909             else if ( pObj->ISA(SwAnchoredDrawObject) )
910             {
911                 RemoveDrawObjFromPage( *pObj );
912                 pDest->AppendDrawObjToPage( *pObj );
913             }
914         }
915     }
916 }
917 
918 /*************************************************************************
919 |*
920 |*  SwPageFrm::AppendDrawObjToPage(), RemoveDrawObjFromPage()
921 |*
922 |*  --> OD 2004-07-02 #i28701# - new methods
923 |*
924 |*************************************************************************/
AppendDrawObjToPage(SwAnchoredObject & _rNewObj)925 void SwPageFrm::AppendDrawObjToPage( SwAnchoredObject& _rNewObj )
926 {
927     if ( !_rNewObj.ISA(SwAnchoredDrawObject) )
928     {
929         ASSERT( false,
930                 "SwPageFrm::AppendDrawObjToPage(..) - anchored object of unexcepted type -> object not appended" );
931         return;
932     }
933 
934     if ( GetUpper() )
935     {
936         ((SwRootFrm*)GetUpper())->InvalidateBrowseWidth();
937     }
938 
939     ASSERT( _rNewObj.GetAnchorFrm(), "anchored draw object without anchor" );
940     SwFlyFrm* pFlyFrm = (SwFlyFrm*)_rNewObj.GetAnchorFrm()->FindFlyFrm();
941     if ( pFlyFrm &&
942          _rNewObj.GetDrawObj()->GetOrdNum() < pFlyFrm->GetVirtDrawObj()->GetOrdNum() )
943     {
944         //#i119945# set pFly's OrdNum to _rNewObj's. So when pFly is removed by Undo, the original OrdNum will not be changed.
945         sal_uInt32 nNewNum = _rNewObj.GetDrawObj()->GetOrdNumDirect();
946         if ( _rNewObj.GetDrawObj()->GetPage() )
947             _rNewObj.DrawObj()->GetPage()->SetObjectOrdNum( pFlyFrm->GetVirtDrawObj()->GetOrdNumDirect(), nNewNum );
948         else
949             pFlyFrm->GetVirtDrawObj()->SetOrdNum( nNewNum );
950     }
951 
952     if ( FLY_AS_CHAR == _rNewObj.GetFrmFmt().GetAnchor().GetAnchorId() )
953     {
954         return;
955     }
956 
957     if ( !pSortedObjs )
958     {
959         pSortedObjs = new SwSortedObjs();
960     }
961     if ( !pSortedObjs->Insert( _rNewObj ) )
962     {
963 #ifdef DBG_UTIL
964         ASSERT( pSortedObjs->Contains( _rNewObj ),
965                 "Drawing object not appended into list <pSortedObjs>." );
966 #endif
967     }
968     // --> OD 2008-04-22 #i87493#
969     ASSERT( _rNewObj.GetPageFrm() == 0 || _rNewObj.GetPageFrm() == this,
970             "<SwPageFrm::AppendDrawObjToPage(..)> - anchored draw object seems to be registered at another page frame. Serious defect -> please inform OD." );
971     // <--
972     _rNewObj.SetPageFrm( this );
973 
974     // invalidate page in order to force a reformat of object layout of the page.
975     InvalidateFlyLayout();
976 }
977 
RemoveDrawObjFromPage(SwAnchoredObject & _rToRemoveObj)978 void SwPageFrm::RemoveDrawObjFromPage( SwAnchoredObject& _rToRemoveObj )
979 {
980     if ( !_rToRemoveObj.ISA(SwAnchoredDrawObject) )
981     {
982         ASSERT( false,
983                 "SwPageFrm::RemoveDrawObjFromPage(..) - anchored object of unexcepted type -> object not removed" );
984         return;
985     }
986 
987     if ( pSortedObjs )
988     {
989         pSortedObjs->Remove( _rToRemoveObj );
990         if ( !pSortedObjs->Count() )
991         {
992             DELETEZ( pSortedObjs );
993         }
994         if ( GetUpper() )
995         {
996             if (FLY_AS_CHAR !=
997                     _rToRemoveObj.GetFrmFmt().GetAnchor().GetAnchorId())
998             {
999                 ((SwRootFrm*)GetUpper())->SetSuperfluous();
1000                 InvalidatePage();
1001             }
1002             ((SwRootFrm*)GetUpper())->InvalidateBrowseWidth();
1003         }
1004     }
1005     _rToRemoveObj.SetPageFrm( 0 );
1006 }
1007 
1008 /*************************************************************************
1009 |*
1010 |*  SwPageFrm::PlaceFly
1011 |*
1012 |*  Ersterstellung      MA 08. Feb. 93
1013 |*  Letzte Aenderung    MA 27. Feb. 93
1014 |*
1015 |*************************************************************************/
1016 
1017 // --> OD 2005-06-09 #i50432# - adjust method description and synopsis.
PlaceFly(SwFlyFrm * pFly,SwFlyFrmFmt * pFmt)1018 void SwPageFrm::PlaceFly( SwFlyFrm* pFly, SwFlyFrmFmt* pFmt )
1019 {
1020     // --> OD 2005-06-09 #i50432# - consider the case that page is an empty page:
1021     // In this case append the fly frame at the next page
1022     ASSERT( !IsEmptyPage() || GetNext(),
1023             "<SwPageFrm::PlaceFly(..)> - empty page with no next page! -> fly frame appended at empty page" );
1024     if ( IsEmptyPage() && GetNext() )
1025     {
1026         static_cast<SwPageFrm*>(GetNext())->PlaceFly( pFly, pFmt );
1027     }
1028     else
1029     {
1030         //Wenn ein Fly uebergeben wurde, so benutzen wir diesen, ansonsten wird
1031         //mit dem Format einer erzeugt.
1032         if ( pFly )
1033             AppendFly( pFly );
1034         else
1035         {   ASSERT( pFmt, ":-( kein Format fuer Fly uebergeben." );
1036             pFly = new SwFlyLayFrm( (SwFlyFrmFmt*)pFmt, this, this );
1037             AppendFly( pFly );
1038             ::RegistFlys( this, pFly );
1039         }
1040     }
1041     // <--
1042 }
1043 
1044 /*************************************************************************
1045 |*
1046 |*  ::CalcClipRect
1047 |*
1048 |*  Ersterstellung      AMA 24. Sep. 96
1049 |*  Letzte Aenderung    MA  18. Dec. 96
1050 |*
1051 |*************************************************************************/
1052 // OD 22.09.2003 #i18732# - adjustments for following text flow or not
1053 // AND alignment at 'page areas' for to paragraph/to character anchored objects
1054 // OD 06.11.2003 #i22305# - adjustment for following text flow
1055 // for to frame anchored objects
1056 // OD 2004-06-02 #i29778# - Because the calculation of the position of the
1057 // floating screen object (Writer fly frame or drawing object) doesn't perform
1058 // a calculation on its upper frames and its anchor frame, a calculation of
1059 // the upper frames in this method no longer sensible.
1060 // --> OD 2004-07-06 #i28701# - if document compatibility option 'Consider
1061 // wrapping style influence on object positioning' is ON, the clip area
1062 // corresponds to the one as the object doesn't follows the text flow.
CalcClipRect(const SdrObject * pSdrObj,SwRect & rRect,sal_Bool bMove)1063 sal_Bool CalcClipRect( const SdrObject *pSdrObj, SwRect &rRect, sal_Bool bMove )
1064 {
1065     sal_Bool bRet = sal_True;
1066     if ( pSdrObj->ISA(SwVirtFlyDrawObj) )
1067     {
1068         const SwFlyFrm* pFly = ((const SwVirtFlyDrawObj*)pSdrObj)->GetFlyFrm();
1069         const bool bFollowTextFlow = pFly->GetFmt()->GetFollowTextFlow().GetValue();
1070         // --> OD 2004-07-06 #i28701#
1071         const bool bConsiderWrapOnObjPos =
1072                                 pFly->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::CONSIDER_WRAP_ON_OBJECT_POSITION);
1073         // <--
1074         const SwFmtVertOrient &rV = pFly->GetFmt()->GetVertOrient();
1075         if( pFly->IsFlyLayFrm() )
1076         {
1077             const SwFrm* pClip;
1078             // OD 06.11.2003 #i22305#
1079             // --> OD 2004-07-06 #i28701#
1080             if ( !bFollowTextFlow || bConsiderWrapOnObjPos )
1081             {
1082                 pClip = pFly->GetAnchorFrm()->FindPageFrm();
1083             }
1084             else
1085             {
1086                 pClip = pFly->GetAnchorFrm();
1087             }
1088 
1089             rRect = pClip->Frm();
1090             SWRECTFN( pClip )
1091 
1092             //Vertikales clipping: Top und Bottom, ggf. an PrtArea
1093             if( rV.GetVertOrient() != text::VertOrientation::NONE &&
1094                 rV.GetRelationOrient() == text::RelOrientation::PRINT_AREA )
1095             {
1096                 (rRect.*fnRect->fnSetTop)( (pClip->*fnRect->fnGetPrtTop)() );
1097                 (rRect.*fnRect->fnSetBottom)( (pClip->*fnRect->fnGetPrtBottom)() );
1098             }
1099             //Horizontales clipping: Left und Right, ggf. an PrtArea
1100             const SwFmtHoriOrient &rH = pFly->GetFmt()->GetHoriOrient();
1101             if( rH.GetHoriOrient() != text::HoriOrientation::NONE &&
1102                 rH.GetRelationOrient() == text::RelOrientation::PRINT_AREA )
1103             {
1104                 (rRect.*fnRect->fnSetLeft)( (pClip->*fnRect->fnGetPrtLeft)() );
1105                 (rRect.*fnRect->fnSetRight)((pClip->*fnRect->fnGetPrtRight)());
1106             }
1107         }
1108         else if( pFly->IsFlyAtCntFrm() )
1109         {
1110             // OD 22.09.2003 #i18732# - consider following text flow or not
1111             // AND alignment at 'page areas'
1112             const SwFrm* pVertPosOrientFrm = pFly->GetVertPosOrientFrm();
1113             if ( !pVertPosOrientFrm )
1114             {
1115                 ASSERT( false,
1116                         "::CalcClipRect(..) - frame, vertical position is oriented at, is missing .");
1117                 pVertPosOrientFrm = pFly->GetAnchorFrm();
1118             }
1119 
1120             if ( !bFollowTextFlow || bConsiderWrapOnObjPos )
1121             {
1122                 const SwLayoutFrm* pClipFrm = pVertPosOrientFrm->FindPageFrm();
1123                 rRect = bMove ? pClipFrm->GetUpper()->Frm()
1124                               : pClipFrm->Frm();
1125                 // --> OD 2004-10-14 #i26945# - consider that a table, during
1126                 // its format, can exceed its upper printing area bottom.
1127                 // Thus, enlarge the clip rectangle, if such a case occured
1128                 if ( pFly->GetAnchorFrm()->IsInTab() )
1129                 {
1130                     const SwTabFrm* pTabFrm = const_cast<SwFlyFrm*>(pFly)
1131                                 ->GetAnchorFrmContainingAnchPos()->FindTabFrm();
1132                     SwRect aTmp( pTabFrm->Prt() );
1133                     aTmp += pTabFrm->Frm().Pos();
1134                     rRect.Union( aTmp );
1135                     // --> OD 2005-03-30 #i43913# - consider also the cell frame
1136                     const SwFrm* pCellFrm = const_cast<SwFlyFrm*>(pFly)
1137                                 ->GetAnchorFrmContainingAnchPos()->GetUpper();
1138                     while ( pCellFrm && !pCellFrm->IsCellFrm() )
1139                     {
1140                         pCellFrm = pCellFrm->GetUpper();
1141                     }
1142                     if ( pCellFrm )
1143                     {
1144                         aTmp = pCellFrm->Prt();
1145                         aTmp += pCellFrm->Frm().Pos();
1146                         rRect.Union( aTmp );
1147                     }
1148                     // <--
1149                 }
1150             }
1151             else if ( rV.GetRelationOrient() == text::RelOrientation::PAGE_FRAME ||
1152                       rV.GetRelationOrient() == text::RelOrientation::PAGE_PRINT_AREA )
1153             {
1154                 // OD 29.10.2003 #113049# - new class <SwEnvironmentOfAnchoredObject>
1155                 objectpositioning::SwEnvironmentOfAnchoredObject
1156                                                 aEnvOfObj( bFollowTextFlow );
1157                 const SwLayoutFrm& rVertClipFrm =
1158                     aEnvOfObj.GetVertEnvironmentLayoutFrm( *pVertPosOrientFrm );
1159                 if ( rV.GetRelationOrient() == text::RelOrientation::PAGE_FRAME )
1160                 {
1161                     rRect = rVertClipFrm.Frm();
1162                 }
1163                 else if ( rV.GetRelationOrient() == text::RelOrientation::PAGE_PRINT_AREA )
1164                 {
1165                     if ( rVertClipFrm.IsPageFrm() )
1166                     {
1167                         rRect = static_cast<const SwPageFrm&>(rVertClipFrm).PrtWithoutHeaderAndFooter();
1168                     }
1169                     else
1170                     {
1171                         rRect = rVertClipFrm.Frm();
1172                     }
1173                 }
1174                 const SwLayoutFrm* pHoriClipFrm =
1175                         pFly->GetAnchorFrm()->FindPageFrm()->GetUpper();
1176                 SWRECTFN( pFly->GetAnchorFrm() )
1177                 (rRect.*fnRect->fnSetLeft)( (pHoriClipFrm->Frm().*fnRect->fnGetLeft)() );
1178                 (rRect.*fnRect->fnSetRight)((pHoriClipFrm->Frm().*fnRect->fnGetRight)());
1179             }
1180             else
1181             {
1182                 // --> OD 2004-10-11 #i26945#
1183                 const SwFrm *pClip =
1184                         const_cast<SwFlyFrm*>(pFly)->GetAnchorFrmContainingAnchPos();
1185                 // <--
1186                 SWRECTFN( pClip )
1187                 const SwLayoutFrm *pUp = pClip->GetUpper();
1188                 const SwFrm *pCell = pUp->IsCellFrm() ? pUp : 0;
1189                 sal_uInt16 nType = bMove ? FRM_ROOT   | FRM_FLY | FRM_HEADER |
1190                                        FRM_FOOTER | FRM_FTN
1191                                      : FRM_BODY   | FRM_FLY | FRM_HEADER |
1192                                        FRM_FOOTER | FRM_CELL| FRM_FTN;
1193 
1194                 while ( !(pUp->GetType() & nType) || pUp->IsColBodyFrm() )
1195                 {
1196                     pUp = pUp->GetUpper();
1197                     if ( !pCell && pUp->IsCellFrm() )
1198                         pCell = pUp;
1199                 }
1200                 if ( bMove )
1201                 {
1202                     if ( pUp->IsRootFrm() )
1203                     {
1204                         rRect  = pUp->Prt();
1205                         rRect += pUp->Frm().Pos();
1206                         pUp = 0;
1207                     }
1208                 }
1209                 if ( pUp )
1210                 {
1211                     if ( pUp->GetType() & FRM_BODY )
1212                     {
1213                         const SwPageFrm *pPg;
1214                         if ( pUp->GetUpper() != (pPg = pFly->FindPageFrm()) )
1215                             pUp = pPg->FindBodyCont();
1216                         rRect = pUp->GetUpper()->Frm();
1217                         (rRect.*fnRect->fnSetTop)( (pUp->*fnRect->fnGetPrtTop)() );
1218                         (rRect.*fnRect->fnSetBottom)((pUp->*fnRect->fnGetPrtBottom)());
1219                     }
1220                     else
1221                     {
1222                         if( ( pUp->GetType() & (FRM_FLY | FRM_FTN ) ) &&
1223                             !pUp->Frm().IsInside( pFly->Frm().Pos() ) )
1224                         {
1225                             if( pUp->IsFlyFrm() )
1226                             {
1227                                 SwFlyFrm *pTmpFly = (SwFlyFrm*)pUp;
1228                                 while( pTmpFly->GetNextLink() )
1229                                 {
1230                                     pTmpFly = pTmpFly->GetNextLink();
1231                                     if( pTmpFly->Frm().IsInside( pFly->Frm().Pos() ) )
1232                                         break;
1233                                 }
1234                                 pUp = pTmpFly;
1235                             }
1236                             else if( pUp->IsInFtn() )
1237                             {
1238                                 const SwFtnFrm *pTmp = pUp->FindFtnFrm();
1239                                 while( pTmp->GetFollow() )
1240                                 {
1241                                     pTmp = pTmp->GetFollow();
1242                                     if( pTmp->Frm().IsInside( pFly->Frm().Pos() ) )
1243                                         break;
1244                                 }
1245                                 pUp = pTmp;
1246                             }
1247                         }
1248                         rRect = pUp->Prt();
1249                         rRect.Pos() += pUp->Frm().Pos();
1250                         if ( pUp->GetType() & (FRM_HEADER | FRM_FOOTER) )
1251                         {
1252                             rRect.Left ( pUp->GetUpper()->Frm().Left() );
1253                             rRect.Width( pUp->GetUpper()->Frm().Width());
1254                         }
1255                         else if ( pUp->IsCellFrm() )                //MA_FLY_HEIGHT
1256                         {
1257                             const SwFrm *pTab = pUp->FindTabFrm();
1258                             (rRect.*fnRect->fnSetBottom)(
1259                                         (pTab->GetUpper()->*fnRect->fnGetPrtBottom)() );
1260                             // OD 08.08.2003 #110978# - expand to left and right
1261                             // cell border
1262                             rRect.Left ( pUp->Frm().Left() );
1263                             rRect.Width( pUp->Frm().Width() );
1264                         }
1265                     }
1266                 }
1267                 if ( pCell )
1268                 {
1269                     //CellFrms koennen auch in 'unerlaubten' Bereichen stehen, dann
1270                     //darf der Fly das auch.
1271                     SwRect aTmp( pCell->Prt() );
1272                     aTmp += pCell->Frm().Pos();
1273                     rRect.Union( aTmp );
1274                 }
1275             }
1276         }
1277         else
1278         {
1279             const SwFrm *pUp = pFly->GetAnchorFrm()->GetUpper();
1280             SWRECTFN( pFly->GetAnchorFrm() )
1281             while( pUp->IsColumnFrm() || pUp->IsSctFrm() || pUp->IsColBodyFrm())
1282                 pUp = pUp->GetUpper();
1283             rRect = pUp->Frm();
1284             if( !pUp->IsBodyFrm() )
1285             {
1286                 rRect += pUp->Prt().Pos();
1287                 rRect.SSize( pUp->Prt().SSize() );
1288                 if ( pUp->IsCellFrm() )
1289                 {
1290                     const SwFrm *pTab = pUp->FindTabFrm();
1291                     (rRect.*fnRect->fnSetBottom)(
1292                                     (pTab->GetUpper()->*fnRect->fnGetPrtBottom)() );
1293                 }
1294             }
1295             else if ( pUp->GetUpper()->IsPageFrm() )
1296             {
1297                 // #111909# Objects anchored as character may exceed right margin
1298                 // of body frame:
1299                 (rRect.*fnRect->fnSetRight)( (pUp->GetUpper()->Frm().*fnRect->fnGetRight)() );
1300             }
1301             long nHeight = (9*(rRect.*fnRect->fnGetHeight)())/10;
1302             long nTop;
1303             const SwFmt *pFmt = ((SwContact*)GetUserCall(pSdrObj))->GetFmt();
1304             const SvxULSpaceItem &rUL = pFmt->GetULSpace();
1305             if( bMove )
1306             {
1307                 nTop = bVert ? ((SwFlyInCntFrm*)pFly)->GetRefPoint().X() :
1308                                ((SwFlyInCntFrm*)pFly)->GetRefPoint().Y();
1309                 nTop = (*fnRect->fnYInc)( nTop, -nHeight );
1310                 long nWidth = (pFly->Frm().*fnRect->fnGetWidth)();
1311                 (rRect.*fnRect->fnSetLeftAndWidth)( bVert ?
1312                             ((SwFlyInCntFrm*)pFly)->GetRefPoint().Y() :
1313                             ((SwFlyInCntFrm*)pFly)->GetRefPoint().X(), nWidth );
1314                 nHeight = 2*nHeight - rUL.GetLower() - rUL.GetUpper();
1315             }
1316             else
1317             {
1318                 nTop = (*fnRect->fnYInc)( (pFly->Frm().*fnRect->fnGetBottom)(),
1319                                            rUL.GetLower() - nHeight );
1320                 nHeight = 2*nHeight - (pFly->Frm().*fnRect->fnGetHeight)()
1321                           - rUL.GetLower() - rUL.GetUpper();
1322             }
1323             (rRect.*fnRect->fnSetTopAndHeight)( nTop, nHeight );
1324         }
1325     }
1326     else
1327     {
1328         const SwDrawContact *pC = (const SwDrawContact*)GetUserCall(pSdrObj);
1329         const SwFrmFmt  *pFmt = (const SwFrmFmt*)pC->GetFmt();
1330         const SwFmtAnchor &rAnch = pFmt->GetAnchor();
1331         if ( FLY_AS_CHAR == rAnch.GetAnchorId() )
1332         {
1333             const SwFrm* pAnchorFrm = pC->GetAnchorFrm( pSdrObj );
1334             if( !pAnchorFrm )
1335             {
1336                 ASSERT( false, "<::CalcClipRect(..)> - missing anchor frame." );
1337                 ((SwDrawContact*)pC)->ConnectToLayout();
1338                 pAnchorFrm = pC->GetAnchorFrm();
1339             }
1340             const SwFrm* pUp = pAnchorFrm->GetUpper();
1341             rRect = pUp->Prt();
1342             rRect += pUp->Frm().Pos();
1343             SWRECTFN( pAnchorFrm )
1344             long nHeight = (9*(rRect.*fnRect->fnGetHeight)())/10;
1345             long nTop;
1346             const SvxULSpaceItem &rUL = pFmt->GetULSpace();
1347             SwRect aSnapRect( pSdrObj->GetSnapRect() );
1348             long nTmpH = 0;
1349             if( bMove )
1350             {
1351                 nTop = (*fnRect->fnYInc)( bVert ? pSdrObj->GetAnchorPos().X() :
1352                                        pSdrObj->GetAnchorPos().Y(), -nHeight );
1353                 long nWidth = (aSnapRect.*fnRect->fnGetWidth)();
1354                 (rRect.*fnRect->fnSetLeftAndWidth)( bVert ?
1355                             pSdrObj->GetAnchorPos().Y() :
1356                             pSdrObj->GetAnchorPos().X(), nWidth );
1357             }
1358             else
1359             {
1360                 // OD 2004-04-13 #i26791# - value of <nTmpH> is needed to
1361                 // calculate value of <nTop>.
1362                 nTmpH = bVert ? pSdrObj->GetCurrentBoundRect().GetWidth() :
1363                                 pSdrObj->GetCurrentBoundRect().GetHeight();
1364                 nTop = (*fnRect->fnYInc)( (aSnapRect.*fnRect->fnGetTop)(),
1365                                           rUL.GetLower() + nTmpH - nHeight );
1366             }
1367             nHeight = 2*nHeight - nTmpH - rUL.GetLower() - rUL.GetUpper();
1368             (rRect.*fnRect->fnSetTopAndHeight)( nTop, nHeight );
1369         }
1370         else
1371         {
1372             // OD 23.06.2003 #108784# - restrict clip rectangle for drawing
1373             // objects in header/footer to the page frame.
1374             // OD 2004-03-29 #i26791#
1375             const SwFrm* pAnchorFrm = pC->GetAnchorFrm( pSdrObj );
1376             if ( pAnchorFrm && pAnchorFrm->FindFooterOrHeader() )
1377             {
1378                 // clip frame is the page frame the header/footer is on.
1379                 const SwFrm* pClipFrm = pAnchorFrm->FindPageFrm();
1380                 rRect = pClipFrm->Frm();
1381             }
1382             else
1383             {
1384                 bRet = sal_False;
1385             }
1386         }
1387     }
1388     return bRet;
1389 }
1390