/* -*- 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 "doc.hxx" #include "frmtool.hxx" #include "hints.hxx" #include #include "rootfrm.hxx" #include "txtfrm.hxx" #include "flyfrms.hxx" #include #include #include SwFlyInContentFrame::SwFlyInContentFrame( SwFlyFrameFormat *pFormat, SwFrame* pSib, SwFrame *pAnch ) : SwFlyFrame( pFormat, pSib, pAnch ) { m_bInCnt = true; SwTwips nRel = pFormat->GetVertOrient().GetPos(); // OD 2004-05-27 #i26791# - member moved to Point aRelPos; if( pAnch && pAnch->IsVertical() ) aRelPos.setX(pAnch->IsReverse() ? nRel : -nRel); else aRelPos.setY(nRel); SetCurrRelPos( aRelPos ); } void SwFlyInContentFrame::DestroyImpl() { if ( !GetFormat()->GetDoc()->IsInDtor() && GetAnchorFrame() ) { SwRect aTmp( GetObjRectWithSpaces() ); SwFlyInContentFrame::NotifyBackground( FindPageFrame(), aTmp, PREP_FLY_LEAVE ); } SwFlyFrame::DestroyImpl(); } SwFlyInContentFrame::~SwFlyInContentFrame() { } // #i28701# void SwFlyInContentFrame::SetRefPoint( const Point& rPoint, const Point& rRelAttr, const Point& rRelPos ) { // OD 2004-05-27 #i26791# - member moved to OSL_ENSURE( rPoint != aRef || rRelAttr != GetCurrRelPos(), "SetRefPoint: no change" ); SwFlyNotify *pNotify = nullptr; // No notify at a locked fly frame, if a fly frame is locked, there's // already a SwFlyNotify object on the stack (MakeAll). if( !IsLocked() ) pNotify = new SwFlyNotify( this ); aRef = rPoint; SetCurrRelPos( rRelAttr ); SwRectFnSet aRectFnSet(GetAnchorFrame()); aRectFnSet.SetPos( Frame(), rPoint + rRelPos ); // #i68520# InvalidateObjRectWithSpaces(); if( pNotify ) { InvalidatePage(); mbValidPos = false; m_bInvalid = true; Calc(getRootFrame()->GetCurrShell()->GetOut()); delete pNotify; } } void SwFlyInContentFrame::Modify( const SfxPoolItem* pOld, const SfxPoolItem *pNew ) { bool bCallPrepare = false; const sal_uInt16 nWhich = pOld ? pOld->Which() : pNew ? pNew->Which() : 0; if (RES_ATTRSET_CHG == nWhich && pNew) { if(pOld && (SfxItemState::SET == static_cast(pNew)->GetChgSet()-> GetItemState(RES_SURROUND, false) || SfxItemState::SET == static_cast(pNew)->GetChgSet()-> GetItemState(RES_FRMMACRO, false)) ) { SwAttrSetChg aOld( *static_cast(pOld) ); SwAttrSetChg aNew( *static_cast(pNew) ); aOld.ClearItem( RES_SURROUND ); aNew.ClearItem( RES_SURROUND ); aOld.ClearItem( RES_FRMMACRO ); aNew.ClearItem( RES_FRMMACRO ); if( aNew.Count() ) { SwFlyFrame::Modify( &aOld, &aNew ); bCallPrepare = true; } } else if( static_cast(pNew)->GetChgSet()->Count()) { SwFlyFrame::Modify( pOld, pNew ); bCallPrepare = true; } } else if( nWhich != RES_SURROUND && RES_FRMMACRO != nWhich ) { SwFlyFrame::Modify( pOld, pNew ); bCallPrepare = true; } if ( bCallPrepare && GetAnchorFrame() ) AnchorFrame()->Prepare( PREP_FLY_ATTR_CHG, GetFormat() ); } /// Here the content gets formatted initially. void SwFlyInContentFrame::Format( vcl::RenderContext* pRenderContext, const SwBorderAttrs *pAttrs ) { if ( !Frame().Height() ) { Lock(); //don't format the anchor on the crook. SwContentFrame *pContent = ContainsContent(); while ( pContent ) { pContent->Calc(pRenderContext); pContent = pContent->GetNextContentFrame(); } Unlock(); } SwFlyFrame::Format( pRenderContext, pAttrs ); } /** Calculate object position * * @note: In contrast to other Frames, we only calculate the relative position * here. The absolute position is only calculated using SetAbsPos. **/ void SwFlyInContentFrame::MakeObjPos() { if ( !mbValidPos ) { mbValidPos = true; SwFlyFrameFormat *pFormat = GetFormat(); const SwFormatVertOrient &rVert = pFormat->GetVertOrient(); //Update the current values in the format if needed, during this we of //course must not send any Modify. const bool bVert = GetAnchorFrame()->IsVertical(); const bool bRev = GetAnchorFrame()->IsReverse(); SwTwips nOld = rVert.GetPos(); SwTwips nAct = bVert ? -GetCurrRelPos().X() : GetCurrRelPos().Y(); if( bRev ) nAct = -nAct; if( nAct != nOld ) { SwFormatVertOrient aVert( rVert ); aVert.SetPos( nAct ); pFormat->LockModify(); pFormat->SetFormatAttr( aVert ); pFormat->UnlockModify(); } } } void SwFlyInContentFrame::ActionOnInvalidation( const InvalidationType _nInvalid ) { if ( INVALID_POS == _nInvalid || INVALID_ALL == _nInvalid ) AnchorFrame()->Prepare( PREP_FLY_ATTR_CHG, &GetFrameFormat() ); } void SwFlyInContentFrame::NotifyBackground( SwPageFrame *, const SwRect& rRect, PrepareHint eHint) { if ( eHint == PREP_FLY_ATTR_CHG ) AnchorFrame()->Prepare( PREP_FLY_ATTR_CHG ); else AnchorFrame()->Prepare( eHint, static_cast(&rRect) ); } const Point SwFlyInContentFrame::GetRelPos() const { Calc(getRootFrame()->GetCurrShell()->GetOut()); return GetCurrRelPos(); } /// @see SwRowFrame::RegistFlys() void SwFlyInContentFrame::RegistFlys() { SwPageFrame *pPage = FindPageFrame(); OSL_ENSURE( pPage, "Register Flys without pages?" ); ::RegistFlys( pPage, this ); } void SwFlyInContentFrame::MakeAll(vcl::RenderContext* /*pRenderContext*/) { // OD 2004-01-19 #110582# if ( !GetFormat()->GetDoc()->getIDocumentDrawModelAccess().IsVisibleLayerId( GetVirtDrawObj()->GetLayer() ) ) { return; } if ( !GetAnchorFrame() || IsLocked() || IsColLocked() || !FindPageFrame() ) return; Lock(); // The curtain falls //does the notification in the DTor const SwFlyNotify aNotify( this ); SwBorderAttrAccess aAccess( SwFrame::GetCache(), this ); const SwBorderAttrs &rAttrs = *aAccess.Get(); if ( IsClipped() ) mbValidSize = m_bHeightClipped = m_bWidthClipped = false; while ( !mbValidPos || !mbValidSize || !mbValidPrtArea || !m_bValidContentPos ) { //Only stop, if the flag is set!! if ( !mbValidSize ) { mbValidPrtArea = false; } if ( !mbValidPrtArea ) { MakePrtArea( rAttrs ); m_bValidContentPos = false; } if ( !mbValidSize ) Format( getRootFrame()->GetCurrShell()->GetOut(), &rAttrs ); if ( !mbValidPos ) { MakeObjPos(); } if ( !m_bValidContentPos ) MakeContentPos( rAttrs ); // re-activate clipping of as-character anchored Writer fly frames // depending on compatibility option if ( mbValidPos && mbValidSize && GetFormat()->getIDocumentSettingAccess().get( DocumentSettingId::CLIP_AS_CHARACTER_ANCHORED_WRITER_FLY_FRAME ) ) { SwFrame* pFrame = AnchorFrame(); if ( Frame().Left() == (pFrame->Frame().Left()+pFrame->Prt().Left()) && Frame().Width() > pFrame->Prt().Width() ) { Frame().Width( pFrame->Prt().Width() ); mbValidPrtArea = false; m_bWidthClipped = true; } } } Unlock(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */