You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
koffice/kword/KWFrameLayout.cpp

650 lines
29 KiB

/* 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, TQ_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;
}
}
}
}