/* This file is part of the KOffice project
* Copyright ( C ) 2002 - 2005 David Faure < faure @ kde . org >
* Copyright ( C ) 2005 Thomas Zander < zander @ kde . org >
*
* This library is free software ; you can redistribute it and / or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation ; version 2.
* 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
* Library General Public License for more details .
*
* You should have received a copy of the GNU Library General Public License
* along with this library ; see the file COPYING . LIB . If not , write to
* the Free Software Foundation , Inc . , 51 Franklin Street , Fifth Floor ,
* Boston , MA 02110 - 1301 , USA .
*/
# include "KWFrameLayout.h"
# include "KWFrameList.h"
# include "KWPageManager.h"
# include "KWPage.h"
# include "KWTextFrameSet.h"
# include "KWDocument.h"
# include <tqtimer.h>
// #define DEBUG_FRAMELAYOUT
# ifdef NDEBUG
# undef DEBUG_FRAMELAYOUT
# endif
KWFrameLayout : : HeaderFooterFrameset : : HeaderFooterFrameset ( KWTextFrameSet * fs , int start , int end ,
double spacing , OddEvenAll oea )
: m_frameset ( fs ) , m_startAtPage ( start ) , m_endAtPage ( end ) , m_oddEvenAll ( oea ) ,
m_spacing ( spacing ) , m_minY ( 0 ) , m_positioned ( false )
{
if ( fs - > frameCount ( ) > 0 )
m_height = fs - > frame ( 0 ) - > height ( ) ;
else
m_height = 20 ; // whatever. The text layout will resize it.
Q_ASSERT ( m_height > 0 ) ;
}
void KWFrameLayout : : HeaderFooterFrameset : : debug ( )
{
# ifdef DEBUG_FRAMELAYOUT
HeaderFooterFrameset * hff = this ;
kdDebug ( 32002 ) < < " * " < < hff - > m_frameset - > name ( )
< < " pages: " < < hff - > m_startAtPage < < " - " < < ( hff - > m_endAtPage = = - 1 ? TQString ( " (all) " ) : TQString : : number ( hff - > m_endAtPage ) )
< < " page-selection: " < < ( hff - > m_oddEvenAll = = HeaderFooterFrameset : : Odd ? " Odd " :
hff - > m_oddEvenAll = = HeaderFooterFrameset : : Even ? " Even " : " All " )
< < " frames: " < < hff - > m_frameset - > frameCount ( )
< < " height: " < < hff - > m_height
< < " spacing: " < < hff - > m_spacing < < endl ;
# endif
}
bool KWFrameLayout : : HeaderFooterFrameset : : deleteFramesAfterLast ( int lastPage )
{
int lastFrame = lastFrameNumber ( lastPage ) ;
# ifdef DEBUG_FRAMELAYOUT
//kdDebug(32002) << " Final cleanup: frameset " << m_frameset->name() << ": lastFrame=" << lastFrame << endl;
# endif
KWTextFrameSet * fs = m_frameset ;
// Special case for odd/even headers: keep at least one frame even if it doesn't appear,
// otherwise the frame properties are lost.
if ( fs - > isHeaderOrFooter ( ) & & lastFrame = = - 1 ) {
fs - > setVisible ( false ) ;
lastFrame = 0 ;
}
bool deleted = false ;
while ( ( int ) fs - > frameCount ( ) - 1 > lastFrame ) {
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Final cleanup: deleting frame " < < fs - > frameCount ( ) - 1 < < " of " < < fs - > name ( ) < < endl ;
# endif
fs - > deleteFrame ( fs - > frameCount ( ) - 1 ) ;
deleted = true ;
}
return deleted ;
}
int KWFrameLayout : : HeaderFooterFrameset : : lastFrameNumber ( int lastPage ) const {
if ( lastPage < m_startAtPage )
return - 1 ; // we need none
int pg = lastPage ;
if ( m_endAtPage > - 1 )
pg = TQMIN ( m_endAtPage , pg ) ;
pg - = m_startAtPage ; // always >=0
Q_ASSERT ( pg > = 0 ) ;
switch ( m_oddEvenAll ) {
case Odd :
case Even :
return pg / 2 ; // page 0 and 1 -> 0. page 2 and 3 -> 1.
case All :
return pg ; // page 0 -> 0 etc. ;)
default :
return - 1 ;
}
}
int KWFrameLayout : : HeaderFooterFrameset : : frameNumberForPage ( int page ) const
{
if ( page < m_startAtPage | | ( m_endAtPage ! = - 1 & & page > m_endAtPage ) )
return - 1 ;
int pg = page - m_startAtPage ; // always >=0
switch ( m_oddEvenAll ) {
case Odd :
// we test the absolute page number for odd/even, not pg!
if ( page % 2 )
return pg / 2 ; // page start+(0 or 1) -> frame 0, page start+(2 or 3) -> frame 1
else
return - 1 ;
case Even :
if ( page % 2 = = 0 )
return pg / 2 ; // page start+(0 or 1) -> frame 0, page start+(2 or 3) -> frame 1
else
return - 1 ;
case All :
return pg ; // page 0[+start] -> frame 0, etc.
default :
return - 1 ;
}
}
/////
void KWFrameLayout : : layout ( KWFrameSet * mainTextFrameSet , int numColumns ,
int fromPage , int toPage , uint flags )
{
//kdDebug(32002) << "KWFrameLayout::layout " << kdBacktrace() << endl;
// Just debug stuff
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " KWFrameLayout::layout " < < fromPage < < " to " < < toPage < < endl ;
Q_ASSERT ( toPage > = fromPage ) ;
TQPtrListIterator < HeaderFooterFrameset > itdbg ( m_headersFooters ) ;
for ( ; itdbg . current ( ) ; + + itdbg )
itdbg . current ( ) - > debug ( ) ;
TQPtrListIterator < HeaderFooterFrameset > itdbg2 ( m_footnotes ) ;
for ( ; itdbg2 . current ( ) ; + + itdbg2 )
itdbg2 . current ( ) - > debug ( ) ;
TQPtrListIterator < HeaderFooterFrameset > itdbg3 ( m_endnotes ) ;
for ( ; itdbg3 . current ( ) ; + + itdbg3 )
itdbg3 . current ( ) - > debug ( ) ;
# endif
#if 0 // old code
// Necessary for end notes: calculate where the text goes down to
Q_ASSERT ( mainTextFrameSet - > type ( ) = = FT_TEXT ) ;
double textBottom = 0.0 ;
if ( mainTextFrameSet - > hasFramesInPageArray ( ) )
{
KoPoint textBottomPoint ;
KoTextParag * lastParag = static_cast < KWTextFrameSet * > ( mainTextFrameSet ) - > textDocument ( ) - > lastParag ( ) ;
if ( lastParag - > isValid ( ) )
{
TQRect rect = lastParag - > rect ( ) ;
int bottom = rect . top ( ) + rect . height ( ) + 2 ; // cf kwtextframeset
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " bottom LU= " < < bottom < < endl ;
# endif
if ( static_cast < KWTextFrameSet * > ( mainTextFrameSet ) - > internalToDocument ( TQPoint ( rect . left ( ) , bottom ) , textBottomPoint ) )
textBottom = textBottomPoint . y ( ) ;
}
}
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " textBottom = " < < textBottom < < " pt " < < endl ;
# endif
# endif
m_framesetsToUpdate . clear ( ) ;
// Necessary for end notes: find out the last frame of the main textframeset
KWFrame * lastMainFrame = mainTextFrameSet - > frameIterator ( ) . getLast ( ) ;
double lastMainFrameBottom = lastMainFrame - > bottom ( ) ; // before we change it below!
double ptColumnWidth = m_doc - > ptColumnWidth ( ) ;
int mainTextFrameResized = - 1 ; // contains the page number of the first resized main textframe
// The main loop is: "for each page". We lay out each page separately.
for ( int pageNum = fromPage ; pageNum < = toPage ; + + pageNum )
{
KWPage * page = m_doc - > pageManager ( ) - > page ( pageNum ) ;
double top = page - > offsetInDocument ( ) + page - > topMargin ( ) ;
double bottom = page - > offsetInDocument ( ) + page - > height ( ) - page - > bottomMargin ( ) ;
double left = page - > leftMargin ( ) ;
double right = page - > width ( ) - page - > rightMargin ( ) ;
Q_ASSERT ( left < right ) ;
KoRect oldColumnRect = firstColumnRect ( mainTextFrameSet , pageNum , numColumns ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Page " < < pageNum < < endl ;
# endif
// For each header/footer....
for ( TQPtrListIterator < HeaderFooterFrameset > it ( m_headersFooters ) ; it . current ( ) ; + + it )
{
int frameNum = it . current ( ) - > frameNumberForPage ( pageNum ) ;
if ( frameNum ! = - 1 )
{
it . current ( ) - > m_positioned = true ;
KWTextFrameSet * fs = it . current ( ) - > m_frameset ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Page " < < pageNum < < " : adding frame " < < frameNum < < " from " < < fs - > name ( ) < < endl ;
# endif
KoRect rect ;
if ( fs - > isAHeader ( ) ) // add on top
{
rect . setRect ( left , top , right - left , it . current ( ) - > m_height ) ;
top + = it . current ( ) - > m_height + it . current ( ) - > m_spacing ;
} else // footer, add at bottom
{
double frameHeight = it . current ( ) - > m_height ;
double frameTop = bottom - frameHeight ;
rect . setRect ( left , frameTop , right - left , frameHeight ) ;
bottom - = frameHeight + it . current ( ) - > m_spacing ;
}
Q_ASSERT ( bottom > 0 ) ;
Q_ASSERT ( top < bottom ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " rect: " < < rect < < " - new_top: " < < top < < " new_bottom: " < < bottom < < endl ;
# endif
resizeOrCreateHeaderFooter ( fs , frameNum , rect ) ;
}
}
// All headers/footers for this page have been done,
// now resize the frame from the main textframeset (if any)
// the first time _before_ doing the footnotes.
resizeMainTextFrame ( mainTextFrameSet , pageNum , numColumns , ptColumnWidth , m_doc - > ptColumnSpacing ( ) , left , top , bottom , NoFootNote ) ;
// Recalc footnote pages
checkFootNotes ( ) ;
bool firstFootNote = true ;
//// Stay seated... We need to know if there are any footnotes on top of us, although we're going
//// to lay them out _AFTER_. But we need their total height for the minY stuff.
//// So we first iterate over all footnotes of the page, to get their total height.
//// Then we'll reduce this height after every footnote being positionned, so it's always
//// the "height on top of us".
double totalFootNotesHeight = 0 ;
for ( TQPtrListIterator < HeaderFooterFrameset > it ( m_footnotes ) ; it . current ( ) ; + + it )
{
int frameNum = it . current ( ) - > frameNumberForPage ( pageNum ) ;
if ( frameNum ! = - 1 )
totalFootNotesHeight + = it . current ( ) - > m_height ;
}
// For each footnote (caller sorted them from bottom to top)
for ( TQPtrListIterator < HeaderFooterFrameset > it ( m_footnotes ) ; it . current ( ) ; + + it )
{
int frameNum = it . current ( ) - > frameNumberForPage ( pageNum ) ;
if ( frameNum ! = - 1 )
{
it . current ( ) - > m_positioned = true ;
totalFootNotesHeight - = it . current ( ) - > m_height ; // as discussed above
KWTextFrameSet * fs = it . current ( ) - > m_frameset ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Page " < < pageNum < < " : adding footnote frame " < < frameNum < < " from " < < fs - > name ( ) < < endl ;
# endif
KoRect rect ;
// When two footnotes are in the same page there should be 0 spacing between them
// Yeah, write a generic frame layouter and then realize it's not flexible enough :(
if ( fs - > isFootEndNote ( ) & & ! firstFootNote )
{
// Undo "bottom -= spacing" (done below). This assumes equal spacing for all footnotes
bottom + = it . current ( ) - > m_spacing ;
bottom - = 1 ; // keep them one pixel apart though
}
double frameTop = bottom - it . current ( ) - > m_height ;
double frameHeight = it . current ( ) - > m_height ;
Q_ASSERT ( fs - > isFootNote ( ) ) ;
// This is where we add the "total height of the footnotes on top of this one".
// The footnote variable can't be behind them....
double minY = it . current ( ) - > m_minY + totalFootNotesHeight ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " footnote: frameHeight= " < < frameHeight < < " frameTop ( " < < frameTop < < " ) <? minY ( " < < minY < < " ) " < < endl ;
# endif
if ( frameTop < minY )
{
// Ok, this is the complex case of a footnote var too far down in the page,
// and its footnote text is too big, so both won't fit.
// We do like other WPs: we create a frame on the next page
it . current ( ) - > m_endAtPage + + ; // this will do so
// In the current page we stop at minY
frameTop = minY ;
frameHeight = bottom - frameTop ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " footnote: new top= " < < frameTop < < " new height= " < < frameHeight < < " remaining height= " < < it . current ( ) - > m_height - frameHeight < < endl ;
# endif
Q_ASSERT ( frameHeight < it . current ( ) - > m_height ) ;
it . current ( ) - > m_height - = frameHeight ; // calculate what remains to be done in the next frame
//fnFrameBehavior = KWFrame::Ignore;
// Make sure there'll actually be a next page
if ( pageNum = = m_doc - > pageCount ( ) - 1 ) {
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Adding a page for the footnote overflow. " < < endl ;
# endif
m_doc - > appendPage ( ) ;
m_doc - > updateAllFrames ( ) ;
toPage = m_doc - > pageCount ( ) - 1 ;
}
}
rect . setRect ( left , frameTop , right - left , frameHeight ) ;
bottom - = frameHeight + it . current ( ) - > m_spacing ;
Q_ASSERT ( bottom > 0 ) ;
Q_ASSERT ( top < bottom ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " footnote rect: " < < rect < < " - new_top: " < < top < < " new_bottom: " < < bottom < < endl ;
# endif
resizeOrCreateHeaderFooter ( fs , frameNum , rect ) ;
firstFootNote = false ;
// We added a footnote, update main text frame size
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Laid out a footnote -> call resizeMainTextFrame/checkFootNotes again " < < endl ;
# endif
resizeMainTextFrame ( mainTextFrameSet , pageNum , numColumns , ptColumnWidth , m_doc - > ptColumnSpacing ( ) , left , top , bottom , WithFootNotes ) ;
checkFootNotes ( ) ;
}
} // for all footnotes
// Check for endnotes, on the last page of main text
// and on any end-notes-only page, i.e. after the last page of main text
if ( pageNum > = m_lastMainFramePage & & m_doc - > hasEndNotes ( ) ) {
bool pageHasMainText = ( pageNum = = m_lastMainFramePage ) ;
if ( pageHasMainText )
lastMainFrame - > setDrawFootNoteLine ( true ) ;
double textBottom = pageHasMainText ? lastMainFrameBottom : top ;
// Leave some space on top of the endnotes, for the horizontal line
double endNoteTop = textBottom + m_doc - > ptFootnoteBodySpacing ( ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Endnotes: textBottom= " < < textBottom < < " pt, endNoteTop= " < < endNoteTop < < " pt, bottom= " < < bottom < < " pt " < < endl ;
# endif
bool firstEndNote = true ;
for ( TQPtrListIterator < HeaderFooterFrameset > it ( m_endnotes ) ; it . current ( ) ; + + it )
{
if ( ! it . current ( ) - > m_positioned )
{
KWTextFrameSet * fs = it . current ( ) - > m_frameset ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Page " < < pageNum < < " : adding endnote frame from " < < fs - > name ( ) < < endl ;
# endif
double frameHeight = it . current ( ) - > m_height ;
if ( it . current ( ) - > m_startAtPage < 0 ) // not set yet
it . current ( ) - > m_startAtPage = pageNum ;
// Check if the endnote is bigger than the available space
if ( endNoteTop + frameHeight > bottom )
{
// In the current page we stop at bottom
frameHeight = bottom - endNoteTop ;
if ( frameHeight > 1E-10 ) // means, if frameHeight > 0
{
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " endnote: new height= " < < frameHeight < < " remaining height= " < < it . current ( ) - > m_height - frameHeight < < endl ;
# endif
Q_ASSERT ( frameHeight < it . current ( ) - > m_height ) ;
it . current ( ) - > m_height - = frameHeight ; // calculate what remains to be done in the next frame
} else {
// No room at all on this page. Schedule for next page.
it . current ( ) - > m_startAtPage + + ;
break ;
}
// Make sure there'll actually be a next page
if ( pageNum = = m_doc - > pageCount ( ) - 1 ) {
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Adding a page for the endnote overflow. " < < endl ;
# endif
m_doc - > appendPage ( ) ;
m_doc - > updateAllFrames ( ) ;
toPage = m_doc - > pageCount ( ) - 1 ;
}
}
else // It'll all fit in this page
{
it . current ( ) - > m_positioned = true ;
}
KoRect rect ( left , endNoteTop , right - left , frameHeight ) ;
endNoteTop + = frameHeight + 1 ; // not + it.current()->m_spacing;
Q_ASSERT ( bottom > 0 ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " rect: " < < rect < < " - new_top: " < < endNoteTop < < " new_bottom: " < < bottom < < endl ;
# endif
int frameNum = pageNum - it . current ( ) - > m_startAtPage ;
resizeOrCreateHeaderFooter ( fs , frameNum , rect ) ;
#if 0 // Disabled. The main frame is resized by KWTextFrameSet::slotAfterFormatting already.
if ( pageHasMainText & & firstEndNote )
{
// We positionned the first endnote, update main text frame size
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Laid out an endnote and the page has a maintextframe too -> call resizeMainTextFrame/checkFootNotes again top= " < < top < < " textBottom= " < < textBottom < < endl ;
# endif
resizeMainTextFrame ( mainTextFrameSet , pageNum , numColumns , ptColumnWidth , m_doc - > ptColumnSpacing ( ) , left , top , textBottom , NoChange ) ;
}
# endif
} // if not positionned yet
firstEndNote = false ; // yes, out of the if
} // for all endnotes
} // if page can have endnotes
if ( mainTextFrameResized = = - 1 ) {
// Test if the main text frame for this page was really resized or not.
KoRect newColumnRect = firstColumnRect ( mainTextFrameSet , pageNum , numColumns ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Comparing old= " < < oldColumnRect < < " and new= " < < newColumnRect < < endl ;
# endif
if ( oldColumnRect ! = newColumnRect ) {
mainTextFrameResized = pageNum ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " changed -> mainTextFrameResized= " < < mainTextFrameResized < < endl ;
# endif
}
}
} // for all pages
m_lastMainFramePage = lastMainFrame - > pageNumber ( ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " m_lastMainFramePage = " < < m_lastMainFramePage < < " lastMainFrameBottom= " < < lastMainFrameBottom < < endl ;
# endif
if ( ! ( flags & DontRemovePages ) )
{
m_doc - > updateAllFrames ( KWFrameSet : : UpdateFramesInPage ) ;
// Check if the last page is now empty (e.g. this can happen when removing
// some text above an endnote, so the endnote moves up)
( void ) m_doc - > tryRemovingPages ( ) ;
}
const int lastPage = m_doc - > lastPage ( ) ;
// Final cleanup: delete all frames after lastFrameNumber in each frameset
TQPtrListIterator < HeaderFooterFrameset > it ( m_headersFooters ) ;
for ( ; it . current ( ) ; + + it )
if ( it . current ( ) - > deleteFramesAfterLast ( lastPage ) )
m_framesetsToUpdate . insert ( it . current ( ) - > m_frameset , true ) ;
TQPtrListIterator < HeaderFooterFrameset > it2 ( m_footnotes ) ;
for ( ; it2 . current ( ) ; + + it2 )
if ( it2 . current ( ) - > deleteFramesAfterLast ( lastPage ) )
m_framesetsToUpdate . insert ( it2 . current ( ) - > m_frameset , true ) ;
if ( mainTextFrameSet ) {
// For the last main text frameset, we use m_lastMainFramePage, so that
// there's no frame on the "end notes only" page(s).
int lastFrame = m_lastMainFramePage * numColumns + ( numColumns - 1 ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " lastFrame: " < < lastFrame < < " due to " < < m_lastMainFramePage < < endl ;
# endif
bool deleted = false ;
while ( ( int ) mainTextFrameSet - > frameCount ( ) - 1 > lastFrame ) {
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Final cleanup: deleting frame " < < mainTextFrameSet - > frameCount ( ) - 1 < < " of main textframeset (lastFrame= " < < lastFrame < < " ) " < < endl ;
# endif
mainTextFrameSet - > deleteFrame ( mainTextFrameSet - > frameCount ( ) - 1 , true , false /*do not updateFrames!*/ ) ;
deleted = true ;
}
if ( deleted )
m_framesetsToUpdate . insert ( mainTextFrameSet , true ) ;
// The last frame before the first endnote, is in auto-extend mode
if ( m_doc - > hasEndNotes ( ) ) {
KWFrame * lastMainFrame = mainTextFrameSet - > frameIterator ( ) . getLast ( ) ;
if ( lastMainFrame - > frameBehavior ( ) ! = KWFrame : : AutoExtendFrame )
{
lastMainFrame - > setFrameBehavior ( KWFrame : : AutoExtendFrame ) ;
// make sure it gets resized
if ( mainTextFrameResized = = - 1 )
mainTextFrameResized = lastMainFrame - > pageNumber ( ) ;
}
}
}
TQMap < KWFrameSet * , bool > : : iterator fsit = m_framesetsToUpdate . begin ( ) ;
for ( ; fsit ! = m_framesetsToUpdate . end ( ) ; + + fsit )
fsit . key ( ) - > updateFrames ( ) ;
// ## TODO: only if something changed? (resizing, new frames, or deleted frames...)
KWFrameList : : recalcFrames ( m_doc , fromPage , toPage ) ;
if ( mainTextFrameResized ! = - 1 & & mainTextFrameSet - > type ( ) = = FT_TEXT ) {
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Done. First maintextframe resized: " < < mainTextFrameResized < < endl ;
# endif
KWTextFrameSet * fs = static_cast < KWTextFrameSet * > ( mainTextFrameSet ) ;
// Not right now, this could be called during formatting...
//m_doc->invalidate();
// ### This means the layout will be done during painting. Not good.
// What about mainTextFrameSet->invalidate() or even layout()?
//TQTimer::singleShot( 0, m_doc, TQT_SLOT( invalidate() ) );
// Invalidate main textframeset only, and from top of page only.
// Otherwise loading a long document (with headers/footers) takes ages,
// if we redo it all from the beginning at each new page!
int topLU , bottomLU ;
if ( fs - > minMaxInternalOnPage ( mainTextFrameResized , topLU , bottomLU ) )
{
// Find parag at topLU
KoTextParag * parag = fs - > paragAtLUPos ( topLU ) ;
if ( parag ) {
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Invalidating from parag " < < parag - > paragId ( ) < < endl ;
# endif
fs - > textObject ( ) - > setLastFormattedParag ( parag ) ;
fs - > textObject ( ) - > formatMore ( 2 ) ;
}
}
}
}
void KWFrameLayout : : resizeOrCreateHeaderFooter ( KWTextFrameSet * headerFooter , uint frameNumber , const KoRect & rect )
{
if ( frameNumber < headerFooter - > frameCount ( ) ) {
KWFrame * frame = headerFooter - > frame ( frameNumber ) ;
if ( * frame = = rect )
return ;
frame - > setRect ( rect ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " KWFrameLayout::resizeOrCreateHeaderFooter frame " < < headerFooter - > name ( ) < < " " < < frame < < " resized to " < < rect < < " pagenum= " < < frame - > pageNumber ( ) < < endl ;
# endif
}
else
{
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " KWFrameLayout::resizeOrCreateHeaderFooter creating frame for " < < headerFooter - > name ( ) < < endl ;
# endif
KWFrame * frame = new KWFrame ( headerFooter , rect . x ( ) , rect . y ( ) , rect . width ( ) , rect . height ( ) ) ;
frame - > setFrameBehavior ( KWFrame : : AutoExtendFrame ) ;
if ( headerFooter - > isHeaderOrFooter ( ) ) // not for footnotes!
{
frame - > setNewFrameBehavior ( KWFrame : : Copy ) ;
frame - > setCopy ( true ) ;
}
else
frame - > setNewFrameBehavior ( KWFrame : : NoFollowup ) ;
headerFooter - > addFrame ( frame , false /*no recalc*/ ) ;
}
// This updates e.g. availableHeight. Very important in the case
// of the footnote frameset with 2 frames.
headerFooter - > updateFrames ( 0 /*fast one*/ ) ;
m_framesetsToUpdate . insert ( headerFooter , true ) ;
}
// Called at beginning and end of the layout for a given page,
// to determine if the main-text-frame layout really changed or not.
// Testing in resizeMainTextFrame doesn't work, we call it several times,
// once for each footnote, so it can't detect the "no change" case.
KoRect KWFrameLayout : : firstColumnRect ( KWFrameSet * mainTextFrameSet , int pageNum , int numColumns ) const
{
uint frameNum = pageNum * numColumns /*+ col 0 here*/ ;
if ( mainTextFrameSet & & frameNum < mainTextFrameSet - > frameCount ( ) )
return * mainTextFrameSet - > frame ( frameNum ) ;
else
return KoRect ( ) ;
}
bool KWFrameLayout : : resizeMainTextFrame ( KWFrameSet * mainTextFrameSet , int pageNum , int numColumns , double ptColumnWidth , double ptColumnSpacing , double left , double top , double bottom , HasFootNotes hasFootNotes )
{
if ( ! mainTextFrameSet )
return false ;
bool mainTextFrameResized = false ;
for ( int col = 0 ; col < numColumns ; col + + ) {
Q_ASSERT ( bottom > top ) ;
// Calculate wanted rect for this frame
KoRect rect ( left + col * ( ptColumnWidth + ptColumnSpacing ) ,
top , ptColumnWidth , bottom - top ) ;
uint frameNum = ( pageNum - m_doc - > startPage ( ) ) * numColumns + col ;
KWFrame * frame ;
if ( frameNum < mainTextFrameSet - > frameCount ( ) ) {
// Resize existing frame
frame = mainTextFrameSet - > frame ( frameNum ) ;
// Special case for last-frame-before-endnotes: don't resize its bottom
if ( m_doc - > hasEndNotes ( ) & & pageNum > = m_lastMainFramePage )
rect . setBottom ( frame - > bottom ( ) ) ;
bool resized = ( rect ! = * frame ) ;
if ( resized ) {
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Page " < < pageNum < < " : resizing main text frame " < < frameNum < < " ( " < < frame < < " ) to " < < rect < < endl ;
# endif
frame - > setRect ( rect ) ;
frame - > updateRulerHandles ( ) ;
mainTextFrameResized = true ;
mainTextFrameSet - > updateFrames ( 0xff - KWFrameSet : : SortFrames ) ; // Don't sort frames yet!
}
} else {
// Create new frame
frame = new KWFrame ( mainTextFrameSet , rect . x ( ) , rect . y ( ) , rect . width ( ) , rect . height ( ) ) ;
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " Page " < < pageNum < < " : creating new main text frame " < < frameNum < < " ( " < < frame < < " ) to " < < rect < < endl ;
# endif
mainTextFrameSet - > addFrame ( frame ) ;
Q_ASSERT ( frameNum = = mainTextFrameSet - > frameCount ( ) - 1 ) ;
mainTextFrameResized = true ;
mainTextFrameSet - > updateFrames ( 0xff - KWFrameSet : : SortFrames ) ; // Don't sort frames yet!
}
if ( hasFootNotes = = NoFootNote )
frame - > setDrawFootNoteLine ( false ) ;
else if ( hasFootNotes = = WithFootNotes )
frame - > setDrawFootNoteLine ( true ) ;
// unchanged in the other cases
// By default, all main-text frames are in "auto-create new frames" mode
frame - > setFrameBehavior ( KWFrame : : AutoCreateNewFrame ) ;
}
return mainTextFrameResized ;
}
void KWFrameLayout : : checkFootNotes ( )
{
// We recalculate all footnotes pages, but we return true
// if those on pageNum have changed.
TQPtrListIterator < HeaderFooterFrameset > it ( m_footnotes ) ;
for ( ; it . current ( ) ; + + it )
{
HeaderFooterFrameset * hff = it . current ( ) ;
if ( ! hff - > m_positioned )
{
Q_ASSERT ( hff - > m_frameset - > isFootEndNote ( ) ) ;
KWFootNoteFrameSet * fnfs = static_cast < KWFootNoteFrameSet * > ( hff - > m_frameset ) ;
KWFootNoteVariable * fnvar = fnfs - > footNoteVariable ( ) ;
//necessary to test paragraph because when we delete mutli
//footnote, first footnote who delete call setDelete(true)
//and force recalc, but with multi footnote deleted
//paragraph is null before we apply attribute to
//kotextcustom.
if ( ! fnvar | | ! fnvar - > paragraph ( ) )
continue ;
double varY = fnvar - > varY ( ) ;
if ( varY = = 0 ) // not able to calculate it yet
continue ;
hff - > m_minY = varY + /*2 * */ hff - > m_spacing + 2 /* some spacing */ ;
int pageNum = m_doc - > pageManager ( ) - > pageNumber ( varY ) ;
if ( pageNum ! = hff - > m_startAtPage ) {
# ifdef DEBUG_FRAMELAYOUT
kdDebug ( 32002 ) < < " checkFootNotes: found minY= " < < hff - > m_minY < < " start/end= " < < pageNum < < " for footnote " < < fnvar - > text ( ) < < endl ;
# endif
hff - > m_startAtPage = pageNum ;
hff - > m_endAtPage = pageNum ;
}
}
}
}