Files
libreoffice/sw/source/core/text/itradj.cxx

766 lines
26 KiB
C++

/*************************************************************************
*
* $RCSfile: itradj.cxx,v $
*
* $Revision: 1.8 $
*
* last change: $Author: fme $ $Date: 2001-06-18 09:18:41 $
*
* The Contents of this file are made available subject to the terms of
* either of the following licenses
*
* - GNU Lesser General Public License Version 2.1
* - Sun Industry Standards Source License Version 1.1
*
* Sun Microsystems Inc., October, 2000
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2000 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
*
* This library 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 for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*
*
* Sun Industry Standards Source License Version 1.1
* =================================================
* The contents of this file are subject to the Sun Industry Standards
* Source License Version 1.1 (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.openoffice.org/license.html.
*
* Software provided under this License is provided on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
* WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
* MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
* See the License for the specific provisions governing your rights and
* obligations concerning the Software.
*
* The Initial Developer of the Original Code is: Sun Microsystems, Inc.
*
* Copyright: 2000 by Sun Microsystems, Inc.
*
* All Rights Reserved.
*
* Contributor(s): _______________________________________
*
*
************************************************************************/
#ifdef PRECOMPILED
#include "core_pch.hxx"
#endif
#pragma hdrstop
#include "frame.hxx" // CalcFlyAdjust()
#include "paratr.hxx"
#ifdef DEBUG
# include "ndtxt.hxx" // pSwpHints, Ausgabeoperator
#endif
#include "segmentc.hxx"
#include "txtcfg.hxx"
#include "itrtxt.hxx"
#include "porglue.hxx"
#include "porlay.hxx"
#include "porfly.hxx" // CalcFlyAdjust()
#include "pordrop.hxx" // CalcFlyAdjust()
#include "pormulti.hxx"
#ifndef _PORTAB_HXX
#include <portab.hxx>
#endif
#ifndef _DOC_HXX
#include <doc.hxx>
#endif
#define MIN_TAB_WIDTH 60
/*************************************************************************
* SwTxtAdjuster::FormatBlock()
*************************************************************************/
void SwTxtAdjuster::FormatBlock( )
{
// In der letzten Zeile gibt's keinen Blocksatz.
// Und bei Tabulatoren aus Tradition auch nicht.
// 7701: wenn Flys im Spiel sind, geht's weiter
const SwLinePortion *pFly = 0;
sal_Bool bSkip = !IsLastBlock() &&
nStart + pCurr->GetLen() >= GetInfo().GetTxt().Len();
// ????: mehrzeilige Felder sind fies: wir muessen kontrollieren,
// ob es noch andere Textportions im Absatz gibt.
if( bSkip )
{
const SwLineLayout *pLay = pCurr->GetNext();
while( pLay && !pLay->GetLen() )
{
const SwLinePortion *pPor = pCurr->GetFirstPortion();
while( pPor && bSkip )
{
if( pPor->InTxtGrp() )
bSkip = sal_False;
pPor = pPor->GetPortion();
}
pLay = bSkip ? pLay->GetNext() : 0;
}
}
if( bSkip )
{
if( !GetInfo().GetParaPortion()->HasFly() )
{
if( IsLastCenter() )
CalcFlyAdjust( pCurr );
pCurr->FinishSpaceAdd();
return;
}
else
{
const SwLinePortion *pTmpFly = NULL;
// 7701: beim letzten Fly soll Schluss sein
const SwLinePortion *pPos = pCurr->GetFirstPortion();
while( pPos )
{
// Ich suche jetzt den letzten Fly, hinter dem noch Text ist:
if( pPos->IsFlyPortion() )
pTmpFly = pPos; // Ein Fly wurde gefunden
else if ( pTmpFly && pPos->InTxtGrp() )
{
pFly = pTmpFly; // Ein Fly mit nachfolgendem Text!
pTmpFly = NULL;
}
pPos = pPos->GetPortion();
}
// 8494: Wenn keiner gefunden wurde, ist sofort Schluss!
if( !pFly )
{
if( IsLastCenter() )
CalcFlyAdjust( pCurr );
pCurr->FinishSpaceAdd();
return;
}
}
}
const nOldIdx = GetInfo().GetIdx();
GetInfo().SetIdx( nStart );
CalcNewBlock( pCurr, pFly );
GetInfo().SetIdx( nOldIdx );
GetInfo().GetParaPortion()->GetRepaint()->SetOfst(0);
}
/*************************************************************************
* SwTxtAdjuster::CalcNewBlock()
*
* CalcNewBlock() darf erst nach CalcLine() gerufen werden !
* Aufgespannt wird immer zwischen zwei RandPortions oder FixPortions
* (Tabs und Flys). Dabei werden die Glues gezaehlt und ExpandBlock gerufen.
*************************************************************************/
void SwTxtAdjuster::CalcNewBlock( SwLineLayout *pCurr,
const SwLinePortion *pStopAt, SwTwips nReal )
{
ASSERT( GetInfo().IsMulti() || SVX_ADJUST_BLOCK == GetAdjust(),
"CalcNewBlock: Why?" );
ASSERT( pCurr->Height(), "SwTxtAdjuster::CalcBlockAdjust: missing CalcLine()" );
pCurr->InitSpaceAdd();
MSHORT nNull = 0;
xub_StrLen nGluePortion = 0;
xub_StrLen nCharCnt = 0;
MSHORT nSpaceIdx = 0;
// Nicht vergessen:
// CalcRightMargin() setzt pCurr->Width() auf die Zeilenbreite !
CalcRightMargin( pCurr, nReal );
SwLinePortion *pPos = pCurr->GetPortion();
while( pPos )
{
if ( pPos->IsBreakPortion() && !IsLastBlock() )
{
pCurr->FinishSpaceAdd();
break;
}
if ( pPos->InTxtGrp() )
nGluePortion += ((SwTxtPortion*)pPos)->GetSpaceCnt( GetInfo(), nCharCnt );
else if( pPos->IsMultiPortion() )
{
SwMultiPortion* pMulti = (SwMultiPortion*)pPos;
// a multiportion with a tabulator inside breaks the text adjustment
// a ruby portion will not be stretched by text adjustment
// a double line portion takes additional space for each blank
// in the wider line
if( pMulti->HasTabulator() )
{
if ( nSpaceIdx == pCurr->GetSpaceAdd().Count() )
pCurr->GetSpaceAdd().Insert( nNull, nSpaceIdx );
nSpaceIdx++;
nGluePortion = 0;
nCharCnt = 0;
}
else if( pMulti->IsDouble() )
nGluePortion += ((SwDoubleLinePortion*)pMulti)->GetSpaceCnt();
}
if( pPos->InGlueGrp() )
{
if( pPos->InFixMargGrp() )
{
if ( nSpaceIdx == pCurr->GetSpaceAdd().Count() )
pCurr->GetSpaceAdd().Insert( nNull, nSpaceIdx );
if( nGluePortion )
{
( pCurr->GetSpaceAdd() )[nSpaceIdx] =
( (SwGluePortion*)pPos )->GetPrtGlue() / nGluePortion;
pPos->Width( ( (SwGluePortion*)pPos )->GetFixWidth() );
}
else if ( IsOneBlock() && nCharCnt > 1 )
{
( pCurr->GetSpaceAdd() )[nSpaceIdx] =
- ( (SwGluePortion*)pPos )->GetPrtGlue() / (nCharCnt-1);
pPos->Width( ( (SwGluePortion*)pPos )->GetFixWidth() );
}
nSpaceIdx++;
nGluePortion = 0;
nCharCnt = 0;
}
else
++nGluePortion;
}
GetInfo().SetIdx( GetInfo().GetIdx() + pPos->GetLen() );
if ( pPos == pStopAt )
{
if ( nSpaceIdx == pCurr->GetSpaceAdd().Count() )
pCurr->GetSpaceAdd().Insert( nNull, nSpaceIdx );
else
pCurr->GetSpaceAdd()[nSpaceIdx] = 0;
break;
}
pPos = pPos->GetPortion();
}
}
/*************************************************************************
* SwTxtAdjuster::CalcKanaAdj()
*************************************************************************/
USHORT SwTxtAdjuster::CalcKanaAdj( SwLineLayout* pCurr )
{
ASSERT( pCurr->Height(), "SwTxtAdjuster::CalcBlockAdjust: missing CalcLine()" );
ASSERT( !pCurr->GetpKanaComp(), "pKanaComp already exists!!" );
SvUShorts *pNewKana = new SvUShorts;
pCurr->SetKanaComp( pNewKana );
const USHORT nNull = 0;
MSHORT nKanaIdx = 0;
long nKanaDiffSum = 0;
USHORT nRepaintOfst = 0;
USHORT nX = 0;
sal_Bool bNoCompression = sal_False;
// Nicht vergessen:
// CalcRightMargin() setzt pCurr->Width() auf die Zeilenbreite !
CalcRightMargin( pCurr, 0 );
SwLinePortion* pPos = pCurr->GetPortion();
while( pPos )
{
if ( pPos->InTxtGrp() )
{
// get maximum portion width from info structure, calculated
// during text formatting
USHORT nMaxWidthDiff = GetInfo().GetMaxWidthDiff( (ULONG)pPos );
// check, if information is stored under other key
if ( !nMaxWidthDiff && pPos == pCurr->GetFirstPortion() )
nMaxWidthDiff = GetInfo().GetMaxWidthDiff( (ULONG)pCurr );
// calculate difference between portion width and max. width
nKanaDiffSum += nMaxWidthDiff;
// we store the beginning of the first compressable portion
// for repaint
if ( nMaxWidthDiff && !nRepaintOfst )
nRepaintOfst = nX + GetLeftMargin();
}
else if( pPos->InGlueGrp() && pPos->InFixMargGrp() )
{
if ( nKanaIdx == pCurr->GetKanaComp().Count() )
pCurr->GetKanaComp().Insert( nNull, nKanaIdx );
USHORT nRest;
if ( pPos->InTabGrp() )
{
nRest = ! bNoCompression &&
( pPos->Width() > MIN_TAB_WIDTH ) ?
pPos->Width() - MIN_TAB_WIDTH :
0;
// for simplifying the handling of left, right ... tabs,
// we do expand portions, which are lying behind
// those special tabs
bNoCompression = !pPos->IsTabLeftPortion();
}
else
{
nRest = ! bNoCompression ?
((SwGluePortion*)pPos)->GetPrtGlue() :
0;
bNoCompression = sal_False;
}
if( nKanaDiffSum )
{
ULONG nCompress = ( 10000 * nRest ) / nKanaDiffSum;
if ( nCompress >= 10000 )
// kanas can be expanded to 100%, and there is still
// some space remaining
nCompress = 0;
else
nCompress = 10000 - nCompress;
( pCurr->GetKanaComp() )[ nKanaIdx ] = (USHORT)nCompress;
nKanaDiffSum = 0;
}
nKanaIdx++;
}
nX += pPos->Width();
pPos = pPos->GetPortion();
}
// set portion width
nKanaIdx = 0;
USHORT nCompress = ( pCurr->GetKanaComp() )[ nKanaIdx ];
pPos = pCurr->GetPortion();
long nDecompress = 0;
nKanaDiffSum = 0;
while( pPos )
{
if ( pPos->InTxtGrp() )
{
const USHORT nMinWidth = pPos->Width();
// get maximum portion width from info structure, calculated
// during text formatting
USHORT nMaxWidthDiff = GetInfo().GetMaxWidthDiff( (ULONG)pPos );
// check, if information is stored under other key
if ( !nMaxWidthDiff && pPos == pCurr->GetFirstPortion() )
nMaxWidthDiff = GetInfo().GetMaxWidthDiff( (ULONG)pCurr );
nKanaDiffSum += nMaxWidthDiff;
pPos->Width( nMinWidth +
( ( 10000 - nCompress ) * nMaxWidthDiff ) / 10000 );
nDecompress += pPos->Width() - nMinWidth;
}
else if( pPos->InGlueGrp() && pPos->InFixMargGrp() )
{
if( nCompress )
{
nKanaDiffSum *= nCompress;
nKanaDiffSum /= 10000;
}
pPos->Width( pPos->Width() - nDecompress );
if ( pPos->InTabGrp() )
// set fix width to width
((SwTabPortion*)pPos)->SetFixWidth( pPos->Width() );
const SvUShorts& rKanaComp = pCurr->GetKanaComp();
if ( ++nKanaIdx < rKanaComp.Count() )
nCompress = ( pCurr->GetKanaComp() )[ nKanaIdx ];
nKanaDiffSum = 0;
nDecompress = 0;
}
pPos = pPos->GetPortion();
}
return nRepaintOfst;
}
/*************************************************************************
* SwTxtAdjuster::CalcRightMargin()
*************************************************************************/
SwMarginPortion *SwTxtAdjuster::CalcRightMargin( SwLineLayout *pCurr,
SwTwips nReal )
{
long nRealWidth;
const USHORT nRealHeight = GetLineHeight();
const USHORT nLineHeight = pCurr->Height();
KSHORT nPrtWidth = pCurr->PrtWidth();
SwLinePortion *pLast = pCurr->FindLastPortion();
if( GetInfo().IsMulti() )
nRealWidth = nReal;
else
{
nRealWidth = GetLineWidth();
// Fuer jeden FlyFrm, der in den rechten Rand hineinragt,
// wird eine FlyPortion angelegt.
const long nLeftMar = GetLeftMargin();
SwRect aCurrRect( nLeftMar + nPrtWidth, Y() + nRealHeight - nLineHeight,
nRealWidth - nPrtWidth, nLineHeight );
SwFlyPortion *pFly = CalcFlyPortion( nRealWidth, aCurrRect );
while( pFly && long( nPrtWidth )< nRealWidth )
{
pLast->Append( pFly );
pLast = pFly;
if( pFly->Fix() > nPrtWidth )
pFly->Width( ( pFly->Fix() - nPrtWidth) + pFly->Width() + 1);
nPrtWidth += pFly->Width() + 1;
aCurrRect.Left( nLeftMar + nPrtWidth );
pFly = CalcFlyPortion( nRealWidth, aCurrRect );
}
if( pFly )
delete pFly;
}
SwMarginPortion *pRight = new SwMarginPortion( 0 );
pLast->Append( pRight );
if( long( nPrtWidth )< nRealWidth )
pRight->PrtWidth( KSHORT( nRealWidth - nPrtWidth ) );
// pCurr->Width() wird auf die reale Groesse gesetzt,
// da jetzt die MarginPortions eingehaengt sind.
// Dieser Trick hat wundersame Auswirkungen.
// Wenn pCurr->Width() == nRealWidth ist, dann wird das gesamte
// Adjustment implizit ausgecontert. GetLeftMarginAdjust() und
// IsBlocksatz() sind der Meinung, sie haetten eine mit Zeichen
// gefuellte Zeile.
pCurr->PrtWidth( KSHORT( nRealWidth ) );
return pRight;
}
/*************************************************************************
* SwTxtAdjuster::CalcFlyAdjust()
*************************************************************************/
void SwTxtAdjuster::CalcFlyAdjust( SwLineLayout *pCurr )
{
// 1) Es wird ein linker Rand eingefuegt:
SwMarginPortion *pLeft = pCurr->CalcLeftMargin();
SwGluePortion *pGlue = pLeft; // die letzte GluePortion
// 2) Es wird ein rechter Rand angehaengt:
// CalcRightMargin berechnet auch eventuelle Ueberlappungen mit
// FlyFrms.
CalcRightMargin( pCurr );
SwLinePortion *pPos = pLeft->GetPortion();
xub_StrLen nLen = 0;
// Wenn wir nur eine Zeile vorliegen haben und die Textportion zusammen
// haengend ist und wenn zentriert wird, dann ...
sal_Bool bComplete = 0 == nStart;
const sal_Bool bTabCompat = GetTxtFrm()->GetNode()->GetDoc()->IsTabCompat();
sal_Bool bMultiTab = sal_False;
while( pPos )
{
if ( pPos->IsMultiPortion() && ((SwMultiPortion*)pPos)->HasTabulator() )
bMultiTab = sal_True;
else if( pPos->InFixMargGrp() &&
( bTabCompat ? ! pPos->InTabGrp() : ! bMultiTab ) )
{
// in tab compat mode we do not want to change tab portions
// in non tab compat mode we do not want to change margins if we
// found a multi portion with tabs
if( SVX_ADJUST_RIGHT == GetAdjust() )
((SwGluePortion*)pPos)->MoveAllGlue( pGlue );
else
{
// Eine schlaue Idee von MA:
// Fuer die erste Textportion wird rechtsbuendig eingestellt,
// fuer die letzte linksbuendig.
// Die erste Textportion kriegt den ganzen Glue
// Aber nur, wenn wir mehr als eine Zeile besitzen.
if( bComplete && GetInfo().GetTxt().Len() == nLen )
((SwGluePortion*)pPos)->MoveHalfGlue( pGlue );
else
{
if ( ! bTabCompat )
{
if( pLeft == pGlue )
{
// Wenn es nur einen linken und rechten Rand gibt,
// dann teilen sich die Raender den Glue.
if( nLen + pPos->GetLen() >= pCurr->GetLen() )
((SwGluePortion*)pPos)->MoveHalfGlue( pGlue );
else
((SwGluePortion*)pPos)->MoveAllGlue( pGlue );
}
else
{
// Die letzte Textportion behaelt sein Glue
if( !pPos->IsMarginPortion() )
((SwGluePortion*)pPos)->MoveHalfGlue( pGlue );
}
}
else
((SwGluePortion*)pPos)->MoveHalfGlue( pGlue );
}
}
pGlue = (SwFlyPortion*)pPos;
bComplete = sal_False;
}
nLen += pPos->GetLen();
pPos = pPos->GetPortion();
}
if( ! bTabCompat && ! bMultiTab && SVX_ADJUST_RIGHT == GetAdjust() )
// portions are moved to the right if possible
pLeft->AdjustRight( pCurr );
}
/*************************************************************************
* SwTxtAdjuster::CalcAdjLine()
*************************************************************************/
void SwTxtAdjuster::CalcAdjLine( SwLineLayout *pCurr )
{
ASSERT( pCurr->IsFormatAdj(), "CalcAdjLine: Why?" );
pCurr->SetFormatAdj(sal_False);
SwParaPortion* pPara = GetInfo().GetParaPortion();
switch( GetAdjust() )
{
case SVX_ADJUST_RIGHT:
case SVX_ADJUST_CENTER:
{
CalcFlyAdjust( pCurr );
pPara->GetRepaint()->SetOfst( 0 );
break;
}
case SVX_ADJUST_BLOCK:
{
// 8311: In Zeilen mit LineBreaks gibt es keinen Blocksatz!
if( pCurr->GetLen() &&
CH_BREAK == GetInfo().GetChar( nStart + pCurr->GetLen() - 1 ) &&
!IsLastBlock() )
{
if( IsLastCenter() )
{
CalcFlyAdjust( pCurr );
pPara->GetRepaint()->SetOfst( 0 );
break;
}
return;
}
FormatBlock();
break;
}
default : return;
}
#ifdef DEBUG
/*
if( OPTDBG( *pInf ) )
{
pCurr->DebugPortions( aDbstream, pInf->GetTxt(), nStart );
if( GetHints() )
{
const SwpHints &rHt = *GetHints();
aDbstream << rHt;
SwAttrIter::Dump( aDbstream );
}
}
*/
#endif
}
/*************************************************************************
* SwTxtAdjuster::CalcFlyPortion()
*
* Die Berechnung hat es in sich: nCurrWidth geibt die Breite _vor_ dem
* aufaddieren des Wortes das noch auf die Zeile passt! Aus diesem Grund
* stimmt die Breite der FlyPortion auch, wenn die Blockierungssituation
* bFirstWord && !WORDFITS eintritt.
*************************************************************************/
SwFlyPortion *SwTxtAdjuster::CalcFlyPortion( const long nRealWidth,
const SwRect &rCurrRect )
{
SwTxtFly aTxtFly( (SwCntntFrm*)GetTxtFrm() );
const KSHORT nCurrWidth = pCurr->PrtWidth();
SwFlyPortion *pFlyPortion = 0;
// aFlyRect ist dokumentglobal !
SwRect aFlyRect( aTxtFly.GetFrm( rCurrRect ) );
// Wenn ein Frame ueberlappt, wird eine Portion eroeffnet.
if( aFlyRect.HasArea() )
{
// aLocal ist framelokal
SwRect aLocal( aFlyRect );
aLocal.Pos( aLocal.Left() - GetLeftMargin(), aLocal.Top() );
if( nCurrWidth > aLocal.Left() )
aLocal.Left( nCurrWidth );
// Wenn das Rechteck breiter als die Zeile ist, stutzen
// wir es ebenfalls zurecht.
KSHORT nLocalWidth = KSHORT( aLocal.Left() + aLocal.Width() );
if( nRealWidth < long( nLocalWidth ) )
aLocal.Width( nRealWidth - aLocal.Left() );
GetInfo().GetParaPortion()->SetFly( sal_True );
pFlyPortion = new SwFlyPortion( aLocal );
pFlyPortion->Height( KSHORT( rCurrRect.Height() ) );
// Die Width koennte kleiner sein als die FixWidth, daher:
pFlyPortion->AdjFixWidth();
}
return pFlyPortion;
}
/*************************************************************************
* SwTxtPainter::_CalcDropAdjust()
*************************************************************************/
// 6721: Drops und Adjustment
// CalcDropAdjust wird ggf. am Ende von Format() gerufen.
void SwTxtAdjuster::CalcDropAdjust()
{
ASSERT( 1<GetDropLines() && SVX_ADJUST_LEFT!=GetAdjust() && SVX_ADJUST_BLOCK!=GetAdjust(),
"CalcDropAdjust: No reason for DropAdjustment." )
const MSHORT nLineNr = GetLineNr();
// 1) Dummies ueberspringen
Top();
if( !pCurr->IsDummy() || NextLine() )
{
// Erst adjustieren.
GetAdjusted();
SwLinePortion *pPor = pCurr->GetFirstPortion();
// 2) Sicherstellen, dass die DropPortion dabei ist.
// 3) pLeft: Die GluePor vor der DropPor
if( pPor->InGlueGrp() && pPor->GetPortion()
&& pPor->GetPortion()->IsDropPortion() )
{
const SwLinePortion *pDropPor = (SwDropPortion*) pPor->GetPortion();
SwGluePortion *pLeft = (SwGluePortion*) pPor;
// 4) pRight: Die GluePor hinter der DropPor suchen
pPor = pPor->GetPortion();
while( pPor && !pPor->InFixMargGrp() )
pPor = pPor->GetPortion();
SwGluePortion *pRight = ( pPor && pPor->InGlueGrp() ) ?
(SwGluePortion*) pPor : 0;
if( pRight && pRight != pLeft )
{
// 5) nMinLeft berechnen. Wer steht am weitesten links?
const KSHORT nDropLineStart =
KSHORT(GetLineStart()) + pLeft->Width() + pDropPor->Width();
KSHORT nMinLeft = nDropLineStart;
for( MSHORT i = 1; i < GetDropLines(); ++i )
{
if( NextLine() )
{
// Erst adjustieren.
GetAdjusted();
pPor = pCurr->GetFirstPortion();
const SwMarginPortion *pMar = pPor->IsMarginPortion() ?
(SwMarginPortion*)pPor : 0;
if( !pMar )
nMinLeft = 0;
else
{
const KSHORT nLineStart =
KSHORT(GetLineStart()) + pMar->Width();
if( nMinLeft > nLineStart )
nMinLeft = nLineStart;
}
}
}
// 6) Den Glue zwischen pLeft und pRight neu verteilen.
if( nMinLeft < nDropLineStart )
{
// Glue wird immer von pLeft nach pRight abgegeben,
// damit der Text nach links wandert.
const short nGlue = nDropLineStart - nMinLeft;
if( !nMinLeft )
pLeft->MoveAllGlue( pRight );
else
pLeft->MoveGlue( pRight, nGlue );
#ifdef DBGTXT
aDbstream << "Drop adjusted: " << nGlue << endl;
#endif
}
}
}
}
if( nLineNr != GetLineNr() )
{
Top();
while( nLineNr != GetLineNr() && Next() )
;
}
}
/*************************************************************************
* SwTxtAdjuster::CalcDropRepaint()
*************************************************************************/
void SwTxtAdjuster::CalcDropRepaint()
{
Top();
SwRepaint &rRepaint = *GetInfo().GetParaPortion()->GetRepaint();
if( rRepaint.Top() > Y() )
rRepaint.Top( Y() );
for( MSHORT i = 1; i < GetDropLines(); ++i )
NextLine();
const SwTwips nBottom = Y() + GetLineHeight() - 1;
if( rRepaint.Bottom() < nBottom )
rRepaint.Bottom( nBottom );
}