Files
libreoffice/sw/source/core/layout/laycache.cxx

1395 lines
51 KiB
C++
Raw Normal View History

/*************************************************************************
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* Copyright 2008 by Sun Microsystems, Inc.
*
* OpenOffice.org - a multi-platform office productivity suite
*
* $RCSfile: laycache.cxx,v $
* $Revision: 1.31 $
*
* This file is part of OpenOffice.org.
*
* OpenOffice.org is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License version 3
* only, as published by the Free Software Foundation.
*
* OpenOffice.org 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 version 3 for more details
* (a copy is included in the LICENSE file that accompanied this code).
*
* You should have received a copy of the GNU Lesser General Public License
* version 3 along with OpenOffice.org. If not, see
* <http://www.openoffice.org/license.html>
* for a copy of the LGPLv3 License.
*
************************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_sw.hxx"
#include <hintids.hxx>
#include <svx/brkitem.hxx>
#include <tools/stream.hxx>
#include <doc.hxx>
#include <docstat.hxx>
#include <docary.hxx>
#include <fmtpdsc.hxx>
#include <laycache.hxx>
#include <layhelp.hxx>
#include <pagefrm.hxx>
#include <rootfrm.hxx>
#include <txtfrm.hxx>
#include <ndtxt.hxx>
#include <swtable.hxx>
#include <tabfrm.hxx>
#include <rowfrm.hxx>
#include <colfrm.hxx>
#include <bodyfrm.hxx>
#include <ndindex.hxx>
#include <sectfrm.hxx>
#include <frmfmt.hxx>
#include <fmtcntnt.hxx>
#include <pagedesc.hxx>
#include <frmtool.hxx>
2001-06-29 06:58:06 +00:00
#include <dflyobj.hxx>
#include <dcontact.hxx>
#include <flyfrm.hxx>
// OD 2004-05-24 #i28701#
#include <sortedobjs.hxx>
// --> OD 2006-03-22 #b6375613#
#include <pam.hxx>
#ifndef _DOCSH_HXX
#include <docsh.hxx>
#endif
#include <com/sun/star/document/XDocumentInfoSupplier.hpp>
#include <com/sun/star/beans/XPropertySet.hpp>
using namespace ::com::sun::star;
// <--
2001-06-29 06:58:06 +00:00
#include <set>
2001-06-29 06:58:06 +00:00
SV_IMPL_PTRARR( SwPageFlyCache, SwFlyCachePtr )
/*-----------------28.5.2001 10:06------------------
* Reading and writing of the layout cache.
* The layout cache is not necessary, but it improves
* the performance and reduces the text flow during
* the formatting.
* The layout cache contains the index of the paragraphs/tables
* at the top of every page, so it's possible to create
* the right count of pages and to distribute the document content
* to this pages before the formatting starts.
*--------------------------------------------------*/
void SwLayoutCache::Read( SvStream &rStream )
{
if( !pImpl )
{
pImpl = new SwLayCacheImpl;
if( !pImpl->Read( rStream ) )
{
delete pImpl;
pImpl = 0;
}
}
}
//-----------------------------------------------------------------------------
void SwLayCacheImpl::Insert( USHORT nType, ULONG nIndex, xub_StrLen nOffset )
{
aType.Insert( nType, aType.Count() );
SvULongs::Insert( nIndex, SvULongs::Count() );
aOffset.Insert( nOffset, aOffset.Count() );
}
BOOL SwLayCacheImpl::Read( SvStream& rStream )
{
SwLayCacheIoImpl aIo( rStream, FALSE );
if( aIo.GetMajorVersion() > SW_LAYCACHE_IO_VERSION_MAJOR )
return FALSE;
// Due to an evil bug in the layout cache (#102759#), we cannot trust the
// sizes of fly frames which have been written using the "old" layout cache.
// This flag should indicate that we do not want to trust the width and
// height of fly frames
bUseFlyCache = aIo.GetMinorVersion() >= 1;
BYTE cFlags;
UINT32 nIndex, nOffset;
aIo.OpenRec( SW_LAYCACHE_IO_REC_PAGES );
aIo.OpenFlagRec();
aIo.CloseFlagRec();
while( aIo.BytesLeft() && !aIo.HasError() )
{
switch( aIo.Peek() )
{
case SW_LAYCACHE_IO_REC_PARA:
aIo.OpenRec( SW_LAYCACHE_IO_REC_PARA );
cFlags = aIo.OpenFlagRec();
aIo.GetStream() >> nIndex;
if( (cFlags & 0x01) != 0 )
aIo.GetStream() >> nOffset;
else
nOffset = STRING_LEN;
aIo.CloseFlagRec();
Insert( SW_LAYCACHE_IO_REC_PARA, nIndex, (xub_StrLen)nOffset );
aIo.CloseRec( SW_LAYCACHE_IO_REC_PARA );
break;
case SW_LAYCACHE_IO_REC_TABLE:
aIo.OpenRec( SW_LAYCACHE_IO_REC_TABLE );
aIo.OpenFlagRec();
2001-10-17 10:38:53 +00:00
aIo.GetStream() >> nIndex
>> nOffset;
Insert( SW_LAYCACHE_IO_REC_TABLE, nIndex, (xub_StrLen)nOffset );
aIo.CloseFlagRec();
aIo.CloseRec( SW_LAYCACHE_IO_REC_TABLE );
break;
2001-06-29 06:58:06 +00:00
case SW_LAYCACHE_IO_REC_FLY:
{
aIo.OpenRec( SW_LAYCACHE_IO_REC_FLY );
aIo.OpenFlagRec();
aIo.CloseFlagRec();
long nX, nY, nW, nH;
USHORT nPgNum;
aIo.GetStream() >> nPgNum >> nIndex
>> nX >> nY >> nW >> nH;
SwFlyCache* pFly = new SwFlyCache( nPgNum, nIndex, nX, nY, nW, nH );
aFlyCache.Insert( pFly, aFlyCache.Count() );
aIo.CloseRec( SW_LAYCACHE_IO_REC_FLY );
break;
}
default:
aIo.SkipRec();
break;
}
}
aIo.CloseRec( SW_LAYCACHE_IO_REC_PAGES );
return !aIo.HasError();
}
/*-----------------28.5.2001 10:19------------------
* SwLayoutCache::Write(..)
* writes the index (more precise: the difference between
* the index and the first index of the document content)
* of the first paragraph/table at the top of every page.
* If at the top of a page is the rest of a paragraph/table
* from the bottom of the previous page, the character/row
* number is stored, too.
2001-06-29 06:58:06 +00:00
* The position, size and page number of the text frames
* are stored, too
* --------------------------------------------------*/
void SwLayoutCache::Write( SvStream &rStream, const SwDoc& rDoc )
{
if( rDoc.GetRootFrm() ) // the layout itself ..
{
SwLayCacheIoImpl aIo( rStream, TRUE );
// We want to save the relative index, so we need the index
// of the first content
ULONG nStartOfContent = rDoc.GetNodes().GetEndOfContent().
StartOfSectionNode()->GetIndex();
// The first page..
SwPageFrm* pPage = (SwPageFrm*)rDoc.GetRootFrm()->Lower();
aIo.OpenRec( SW_LAYCACHE_IO_REC_PAGES );
aIo.OpenFlagRec( 0, 0 );
aIo.CloseFlagRec();
while( pPage )
{
2001-06-29 06:58:06 +00:00
if( pPage->GetPrev() )
{
2001-06-29 06:58:06 +00:00
SwLayoutFrm* pLay = pPage->FindBodyCont();
SwFrm* pTmp = pLay ? pLay->ContainsAny() : NULL;
// We are only interested in paragraph or table frames,
// a section frames contains paragraphs/tables.
if( pTmp && pTmp->IsSctFrm() )
pTmp = ((SwSectionFrm*)pTmp)->ContainsAny();
if( pTmp ) // any content
{
2001-06-29 06:58:06 +00:00
if( pTmp->IsTxtFrm() )
{
2001-06-29 06:58:06 +00:00
ULONG nNdIdx = ((SwTxtFrm*)pTmp)->GetNode()->GetIndex();
if( nNdIdx > nStartOfContent )
{
2001-06-29 06:58:06 +00:00
/* Open Paragraph Record */
aIo.OpenRec( SW_LAYCACHE_IO_REC_PARA );
BOOL bFollow = ((SwTxtFrm*)pTmp)->IsFollow();
aIo.OpenFlagRec( bFollow ? 0x01 : 0x00,
bFollow ? 8 : 4 );
nNdIdx -= nStartOfContent;
aIo.GetStream() << static_cast<sal_uInt32>(nNdIdx);
2001-06-29 06:58:06 +00:00
if( bFollow )
aIo.GetStream() << static_cast<sal_uInt32>(((SwTxtFrm*)pTmp)->GetOfst());
aIo.CloseFlagRec();
2001-06-29 06:58:06 +00:00
/* Close Paragraph Record */
aIo.CloseRec( SW_LAYCACHE_IO_REC_PARA );
}
2001-06-29 06:58:06 +00:00
}
else if( pTmp->IsTabFrm() )
{
SwTabFrm* pTab = (SwTabFrm*)pTmp;
ULONG nOfst = STRING_LEN;
if( pTab->IsFollow() )
{
2001-06-29 06:58:06 +00:00
// If the table is a follow, we have to look for the
// master and to count all rows to get the row number
nOfst = 0;
if( pTab->IsFollow() )
pTab = pTab->FindMaster( true );
2001-06-29 06:58:06 +00:00
while( pTab != pTmp )
{
SwFrm* pSub = pTab->Lower();
while( pSub )
{
++nOfst;
pSub = pSub->GetNext();
}
pTab = pTab->GetFollow();
2001-06-29 06:58:06 +00:00
ASSERT( pTab, "Table follow without master" );
}
}
do
{
ULONG nNdIdx =
pTab->GetTable()->GetTableNode()->GetIndex();
if( nNdIdx > nStartOfContent )
{
/* Open Table Record */
aIo.OpenRec( SW_LAYCACHE_IO_REC_TABLE );
aIo.OpenFlagRec( 0, 8 );
nNdIdx -= nStartOfContent;
aIo.GetStream() << static_cast<sal_uInt32>(nNdIdx)
<< static_cast<sal_uInt32>(nOfst);
2001-06-29 06:58:06 +00:00
aIo.CloseFlagRec();
/* Close Table Record */
aIo.CloseRec( SW_LAYCACHE_IO_REC_TABLE );
}
// If the table has a follow on the next page,
// we know already the row number and store this
// immediately.
if( pTab->GetFollow() )
{
if( nOfst == STRING_LEN )
nOfst = 0;
do
{
2001-06-29 06:58:06 +00:00
SwFrm* pSub = pTab->Lower();
while( pSub )
{
++nOfst;
pSub = pSub->GetNext();
}
pTab = pTab->GetFollow();
SwPageFrm *pTabPage = pTab->FindPageFrm();
if( pTabPage != pPage )
{
ASSERT( pPage->GetPhyPageNum() <
pTabPage->GetPhyPageNum(),
"Looping Tableframes" );
pPage = pTabPage;
break;
}
} while ( pTab->GetFollow() );
}
else
break;
} while( pTab );
}
}
}
if( pPage->GetSortedObjs() )
{
SwSortedObjs &rObjs = *pPage->GetSortedObjs();
2001-06-29 06:58:06 +00:00
for ( USHORT i = 0; i < rObjs.Count(); ++i )
{
SwAnchoredObject* pAnchoredObj = rObjs[i];
if ( pAnchoredObj->ISA(SwFlyFrm) )
2001-06-29 06:58:06 +00:00
{
SwFlyFrm *pFly = static_cast<SwFlyFrm*>(pAnchoredObj);
2001-06-29 06:58:06 +00:00
if( pFly->Frm().Left() != WEIT_WECH &&
!pFly->GetAnchorFrm()->FindFooterOrHeader() )
2001-06-29 06:58:06 +00:00
{
const SwContact *pC =
::GetUserCall(pAnchoredObj->GetDrawObj());
2001-06-29 06:58:06 +00:00
if( pC )
{
sal_uInt32 nOrdNum = pAnchoredObj->GetDrawObj()->GetOrdNum();
2001-06-29 06:58:06 +00:00
USHORT nPageNum = pPage->GetPhyPageNum();
/* Open Fly Record */
aIo.OpenRec( SW_LAYCACHE_IO_REC_FLY );
aIo.OpenFlagRec( 0, 0 );
aIo.CloseFlagRec();
SwRect &rRct = pFly->Frm();
sal_Int32 nX = rRct.Left() - pPage->Frm().Left();
sal_Int32 nY = rRct.Top() - pPage->Frm().Top();
2001-06-29 06:58:06 +00:00
aIo.GetStream() << nPageNum << nOrdNum
<< nX << nY << rRct.Width()
<< rRct.Height();
/* Close Fly Record */
aIo.CloseRec( SW_LAYCACHE_IO_REC_FLY );
}
}
2001-06-29 06:58:06 +00:00
}
}
}
pPage = (SwPageFrm*)pPage->GetNext();
}
aIo.CloseRec( SW_LAYCACHE_IO_REC_PAGES );
}
}
#ifndef PRODUCT
sal_Bool SwLayoutCache::CompareLayout( const SwDoc& rDoc ) const
{
sal_Bool bRet = sal_True;
if( pImpl && rDoc.GetRootFrm() )
{
USHORT nIndex = 0;
ULONG nStartOfContent = rDoc.GetNodes().GetEndOfContent().
StartOfSectionNode()->GetIndex();
SwPageFrm* pPage = (SwPageFrm*)rDoc.GetRootFrm()->Lower();
if( pPage )
pPage = (SwPageFrm*)pPage->GetNext();
while( pPage )
{
if( nIndex >= pImpl->Count() )
{
if( bRet )
bRet = sal_False;
break;
}
SwLayoutFrm* pLay = pPage->FindBodyCont();
SwFrm* pTmp = pLay ? pLay->ContainsAny() : NULL;
if( pTmp && pTmp->IsSctFrm() )
pTmp = ((SwSectionFrm*)pTmp)->ContainsAny();
if( pTmp )
{
if( pTmp->IsTxtFrm() )
{
ULONG nNdIdx = ((SwTxtFrm*)pTmp)->GetNode()->GetIndex();
if( nNdIdx > nStartOfContent )
{
BOOL bFollow = ((SwTxtFrm*)pTmp)->IsFollow();
nNdIdx -= nStartOfContent;
if( pImpl->GetBreakIndex( nIndex ) != nNdIdx ||
SW_LAYCACHE_IO_REC_PARA !=
pImpl->GetBreakType( nIndex ) ||
( bFollow ? ((SwTxtFrm*)pTmp)->GetOfst()
: STRING_LEN ) != pImpl->GetBreakOfst( nIndex ) )
{
if( bRet )
bRet = sal_False;
}
++nIndex;
}
}
else if( pTmp->IsTabFrm() )
{
SwTabFrm* pTab = (SwTabFrm*)pTmp;
ULONG nOfst = STRING_LEN;
if( pTab->IsFollow() )
{
nOfst = 0;
if( pTab->IsFollow() )
pTab = pTab->FindMaster( true );
while( pTab != pTmp )
{
SwFrm* pSub = pTab->Lower();
while( pSub )
{
++nOfst;
pSub = pSub->GetNext();
}
pTab = pTab->GetFollow();
}
}
do
{
ULONG nNdIdx =
pTab->GetTable()->GetTableNode()->GetIndex();
if( nNdIdx > nStartOfContent )
{
nNdIdx -= nStartOfContent;
if( pImpl->GetBreakIndex( nIndex ) != nNdIdx ||
SW_LAYCACHE_IO_REC_TABLE !=
pImpl->GetBreakType( nIndex ) ||
nOfst != pImpl->GetBreakOfst( nIndex ) )
{
if( bRet )
bRet = sal_False;
}
++nIndex;
}
if( pTab->GetFollow() )
{
if( nOfst == STRING_LEN )
nOfst = 0;
do
{
SwFrm* pSub = pTab->Lower();
while( pSub )
{
++nOfst;
pSub = pSub->GetNext();
}
pTab = pTab->GetFollow();
SwPageFrm *pTabPage = pTab->FindPageFrm();
if( pTabPage != pPage )
{
pPage = pTabPage;
break;
}
} while ( pTab->GetFollow() );
}
else
break;
} while( pTab );
}
}
pPage = (SwPageFrm*)pPage->GetNext();
}
}
return bRet;
}
#endif
void SwLayoutCache::ClearImpl()
{
if( !IsLocked() )
{
delete pImpl;
pImpl = 0;
}
}
SwLayoutCache::~SwLayoutCache()
{
ASSERT( !nLockCount, "Deleting a locked SwLayoutCache!?" );
delete pImpl;
}
/*-----------------28.5.2001 10:47------------------
* SwActualSection,
* a help class to create not nested section frames
* for nested sections.
* --------------------------------------------------*/
SwActualSection::SwActualSection( SwActualSection *pUp,
SwSectionFrm *pSect,
SwSectionNode *pNd ) :
pUpper( pUp ),
pSectFrm( pSect ),
pSectNode( pNd )
{
if ( !pSectNode )
{
const SwNodeIndex *pIndex = pSect->GetFmt()->GetCntnt().GetCntntIdx();
pSectNode = pSect->GetFmt()->GetDoc()->GetNodes()[*pIndex]->
FindSectionNode();
}
}
/*-----------------28.5.2001 11:09------------------
* SwLayHelper
* is the helper class, which utilizes the layout cache information
* to distribute the document content to the rigth pages.
* It's used by the _InsertCnt(..)-function.
* If there's no layout cache, the distibution to the pages is more
* a guess, but a guess with statistical background.
* --------------------------------------------------*/
SwLayHelper::SwLayHelper( SwDoc *pD, SwFrm* &rpF, SwFrm* &rpP, SwPageFrm* &rpPg,
SwLayoutFrm* &rpL, SwActualSection* &rpA, BOOL &rB,
ULONG nNodeIndex, BOOL bCache )
: rpFrm( rpF ), rpPrv( rpP ), rpPage( rpPg ), rpLay( rpL ),
rpActualSection( rpA ), rbBreakAfter(rB), pDoc(pD), nMaxParaPerPage( 25 ),
nParagraphCnt( bCache ? 0 : USHRT_MAX ), bFirst( bCache )
{
pImpl = pDoc->GetLayoutCache() ? pDoc->GetLayoutCache()->LockImpl() : NULL;
if( pImpl )
{
nMaxParaPerPage = 1000;
nStartOfContent = pDoc->GetNodes().GetEndOfContent().StartOfSectionNode()
->GetIndex();
nNodeIndex -= nStartOfContent;
nIndex = 0;
2001-06-29 06:58:06 +00:00
nFlyIdx = 0;
while( nIndex < pImpl->Count() && (*pImpl)[ nIndex ] < nNodeIndex )
++nIndex;
if( nIndex >= pImpl->Count() )
{
pDoc->GetLayoutCache()->UnlockImpl();
pImpl = NULL;
}
}
else
{
nIndex = USHRT_MAX;
nStartOfContent = ULONG_MAX;
}
}
SwLayHelper::~SwLayHelper()
{
if( pImpl )
{
ASSERT( pDoc && pDoc->GetLayoutCache(), "Missing layoutcache" );
pDoc->GetLayoutCache()->UnlockImpl();
}
}
/*-----------------23.5.2001 16:40------------------
* SwLayHelper::CalcPageCount() does not really calculate the page count,
* it returns the page count value from the layout cache, if available,
* otherwise it estimates the page count.
* --------------------------------------------------*/
ULONG SwLayHelper::CalcPageCount()
{
ULONG nPgCount;
SwLayCacheImpl *pCache = pDoc->GetLayoutCache() ?
pDoc->GetLayoutCache()->LockImpl() : NULL;
if( pCache )
{
nPgCount = pCache->Count() + 1;
pDoc->GetLayoutCache()->UnlockImpl();
}
else
{
nPgCount = pDoc->GetDocStat().nPage;
if ( nPgCount <= 10 ) // no page insertion for less than 10 pages
nPgCount = 0;
ULONG nNdCount = pDoc->GetDocStat().nPara;
if ( nNdCount <= 1 )
{
//Estimates the number of paragraphs.
ULONG nTmp = pDoc->GetNodes().GetEndOfContent().GetIndex() -
pDoc->GetNodes().GetEndOfExtras().GetIndex();
//Tables have a little overhead..
nTmp -= pDoc->GetTblFrmFmts()->Count() * 25;
//Fly frames, too ..
nTmp -= (pDoc->GetNodes().GetEndOfAutotext().GetIndex() -
pDoc->GetNodes().GetEndOfInserts().GetIndex()) / 3 * 5;
if ( nTmp > 0 )
nNdCount = nTmp;
}
if ( nNdCount > 100 ) // no estimation below this value
{
if ( nPgCount > 0 )
nMaxParaPerPage = nNdCount / nPgCount;
else
{
nMaxParaPerPage = Max( ULONG(20),
ULONG(20 + nNdCount / 1000 * 3) );
#ifdef PM2
const ULONG nMax = 49;
#else
const ULONG nMax = 53;
#endif
nMaxParaPerPage = Min( nMaxParaPerPage, nMax );
nPgCount = nNdCount / nMaxParaPerPage;
}
if ( nNdCount < 1000 )
nPgCount = 0;// no progress bar for small documents
if ( pDoc->get(IDocumentSettingAccess::BROWSE_MODE) )
nMaxParaPerPage *= 6;
}
}
return nPgCount;
}
/*-----------------23.5.2001 16:44------------------
* SwLayHelper::CheckInsertPage()
* inserts a page and return TRUE, if
* - the break after flag is set
* - the actual content wants a break before
* - the maximum count of paragraph/rows is reached
*
* The break after flag is set, if the actual content
* wants a break after.
* --------------------------------------------------*/
BOOL SwLayHelper::CheckInsertPage()
{
BOOL bEnd = 0 == rpPage->GetNext();
const SwAttrSet* pAttr = rpFrm->GetAttrSet();
const SvxFmtBreakItem& rBrk = pAttr->GetBreak();
const SwFmtPageDesc& rDesc = pAttr->GetPageDesc();
// --> FME 2004-10-26 #118195# Do not evaluate page description if frame
// is a follow frame!
const SwPageDesc* pDesc = rpFrm->IsFlowFrm() &&
SwFlowFrm::CastFlowFrm( rpFrm )->IsFollow() ?
0 :
rDesc.GetPageDesc();
// <--
BOOL bBrk = nParagraphCnt > nMaxParaPerPage || rbBreakAfter;
rbBreakAfter = rBrk.GetBreak() == SVX_BREAK_PAGE_AFTER ||
rBrk.GetBreak() == SVX_BREAK_PAGE_BOTH;
if ( !bBrk )
bBrk = rBrk.GetBreak() == SVX_BREAK_PAGE_BEFORE ||
rBrk.GetBreak() == SVX_BREAK_PAGE_BOTH;
if ( bBrk || pDesc )
{
USHORT nPgNum = 0;
if ( !pDesc )
pDesc = rpPage->GetPageDesc()->GetFollow();
else
{
if ( 0 != (nPgNum = rDesc.GetNumOffset()) )
((SwRootFrm*)rpPage->GetUpper())->SetVirtPageNum(TRUE);
}
BOOL bNextPageOdd = !rpPage->OnRightPage();
BOOL bInsertEmpty = FALSE;
if( nPgNum && bNextPageOdd != ( ( nPgNum % 2 ) != 0 ) )
{
bNextPageOdd = !bNextPageOdd;
bInsertEmpty = TRUE;
}
::InsertNewPage( (SwPageDesc&)*pDesc, rpPage->GetUpper(),
bNextPageOdd, bInsertEmpty, FALSE, rpPage->GetNext() );
if ( bEnd )
{
ASSERT( rpPage->GetNext(), "Keine neue Seite?" );
do
{ rpPage = (SwPageFrm*)rpPage->GetNext();
} while ( rpPage->GetNext() );
}
else
{
ASSERT( rpPage->GetNext(), "Keine neue Seite?" );
rpPage = (SwPageFrm*)rpPage->GetNext();
if ( rpPage->IsEmptyPage() )
{
ASSERT( rpPage->GetNext(), "Keine neue Seite?" );
rpPage = (SwPageFrm*)rpPage->GetNext();
}
}
rpLay = rpPage->FindBodyCont();
while( rpLay->Lower() )
rpLay = (SwLayoutFrm*)rpLay->Lower();
return TRUE;
}
return FALSE;
}
// --> OD 2006-03-22 #b6375613#
bool lcl_HasTextFrmAnchoredObjs( SwTxtFrm* p_pTxtFrm )
{
bool bHasTextFrmAnchoredObjs( false );
const SwSpzFrmFmts* pSpzFrmFmts = p_pTxtFrm->GetTxtNode()->GetDoc()->GetSpzFrmFmts();
for ( USHORT i = 0; i < pSpzFrmFmts->Count(); ++i )
{
SwFrmFmt *pFmt = (SwFrmFmt*)(*pSpzFrmFmts)[i];
const SwFmtAnchor &rAnch = pFmt->GetAnchor();
if ( rAnch.GetCntntAnchor() &&
( rAnch.GetAnchorId() == FLY_AT_CNTNT ||
rAnch.GetAnchorId() == FLY_AUTO_CNTNT ) &&
rAnch.GetCntntAnchor()->nNode.GetIndex() ==
p_pTxtFrm->GetTxtNode()->GetIndex() )
{
bHasTextFrmAnchoredObjs = true;
break;
}
}
return bHasTextFrmAnchoredObjs;
}
void lcl_ApplyWorkaroundForB6375613( SwFrm* p_pFirstFrmOnNewPage )
{
SwTxtFrm* pFirstTextFrmOnNewPage = dynamic_cast<SwTxtFrm*>(p_pFirstFrmOnNewPage);
//
if ( pFirstTextFrmOnNewPage &&
!pFirstTextFrmOnNewPage->IsFollow() &&
pFirstTextFrmOnNewPage->GetTxt().Len() == 0 &&
lcl_HasTextFrmAnchoredObjs( pFirstTextFrmOnNewPage ) )
{
// apply page break before at this text frame to assure, that it doesn't flow backward.
const SvxBreak eBreak =
pFirstTextFrmOnNewPage->GetAttrSet()->GetBreak().GetBreak();
if ( eBreak == SVX_BREAK_NONE )
{
pFirstTextFrmOnNewPage->GetTxtNode()->LockModify();
SwDoc* pDoc( pFirstTextFrmOnNewPage->GetTxtNode()->GetDoc() );
IDocumentContentOperations* pIDCO = pFirstTextFrmOnNewPage->GetTxtNode()->getIDocumentContentOperations();
const SwPaM aTmpPaM( *(pFirstTextFrmOnNewPage->GetTxtNode()) );
pIDCO->Insert( aTmpPaM, SvxFmtBreakItem( SVX_BREAK_PAGE_BEFORE, RES_BREAK ), 0 );
pFirstTextFrmOnNewPage->GetTxtNode()->UnlockModify();
uno::Reference< document::XDocumentInfoSupplier > xDoc(
pDoc->GetDocShell()->GetBaseModel(),
uno::UNO_QUERY);
uno::Reference< beans::XPropertySet > xDocInfo(
xDoc->getDocumentInfo(),
uno::UNO_QUERY );
try
{
xDocInfo->setPropertyValue( rtl::OUString::createFromAscii("WorkaroundForB6375613Applied"), uno::makeAny( true ) );
}
catch( uno::Exception& )
{
}
}
}
}
// <--
/*-----------------28.5.2001 11:31------------------
* SwLayHelper::CheckInsert
* is the entry point for the _InsertCnt-function.
* The document content index is checked either it is
* in the layout cache either it's time to insert a page
* cause the maximal estimation of content per page is reached.
* A really big table or long paragraph may contains more than
* one page, in this case the needed count of pages will inserted.
* --------------------------------------------------*/
BOOL SwLayHelper::CheckInsert( ULONG nNodeIndex )
{
BOOL bRet = FALSE;
2001-10-17 10:38:53 +00:00
BOOL bLongTab = FALSE;
ULONG nMaxRowPerPage( 0 );
nNodeIndex -= nStartOfContent;
USHORT nRows( 0 );
if( rpFrm->IsTabFrm() )
{
//Inside a table counts every row as a paragraph
SwFrm *pLow = ((SwTabFrm*)rpFrm)->Lower();
nRows = 0;
do
{
++nRows;
pLow = pLow->GetNext();
} while ( pLow );
nParagraphCnt += nRows;
2001-10-17 10:38:53 +00:00
if( !pImpl && nParagraphCnt > nMaxParaPerPage + 10 )
{
// OD 09.04.2003 #108698# - improve heuristics:
// Assume that a table, which has more than three times the quantity
// of maximal paragraphs per page rows, consists of rows, which have
// the height of a normal paragraph. Thus, allow as much rows per page
// as much paragraphs are allowed.
if ( nRows > ( 3*nMaxParaPerPage ) )
2001-10-17 10:38:53 +00:00
{
nMaxRowPerPage = nMaxParaPerPage;
}
else
{
SwFrm *pTmp = ((SwTabFrm*)rpFrm)->Lower();
if( pTmp->GetNext() )
pTmp = pTmp->GetNext();
pTmp = ((SwRowFrm*)pTmp)->Lower();
USHORT nCnt = 0;
do
{
++nCnt;
pTmp = pTmp->GetNext();
} while( pTmp );
nMaxRowPerPage = Max( ULONG(2), nMaxParaPerPage / nCnt );
}
2001-10-17 10:38:53 +00:00
bLongTab = TRUE;
}
}
else
++nParagraphCnt;
if( bFirst && pImpl && nIndex < pImpl->Count() &&
pImpl->GetBreakIndex( nIndex ) == nNodeIndex &&
( pImpl->GetBreakOfst( nIndex ) < STRING_LEN ||
( ++nIndex < pImpl->Count() &&
pImpl->GetBreakIndex( nIndex ) == nNodeIndex ) ) )
bFirst = FALSE;
#if OSL_DEBUG_LEVEL > 1
2001-10-17 10:38:53 +00:00
ULONG nBreakIndex = ( pImpl && nIndex < pImpl->Count() ) ?
pImpl->GetBreakIndex(nIndex) : 0xffff;
(void)nBreakIndex;
2001-10-17 10:38:53 +00:00
#endif
// OD 09.04.2003 #108698# - always split a big tables.
if ( !bFirst ||
( rpFrm->IsTabFrm() && bLongTab )
)
{
ULONG nRowCount = 0;
do
{
2001-10-17 10:38:53 +00:00
if( pImpl || bLongTab )
{
#if OSL_DEBUG_LEVEL > 1
2001-10-17 10:38:53 +00:00
ULONG nBrkIndex = ( pImpl && nIndex < pImpl->Count() ) ?
pImpl->GetBreakIndex(nIndex) : 0xffff;
(void)nBrkIndex;
2001-10-17 10:38:53 +00:00
#endif
xub_StrLen nOfst = STRING_LEN;
USHORT nType = SW_LAYCACHE_IO_REC_PAGES;
2001-10-17 10:38:53 +00:00
if( bLongTab )
{
rbBreakAfter = sal_True;
nOfst = static_cast<xub_StrLen>(nRowCount + nMaxRowPerPage);
}
2001-10-17 10:38:53 +00:00
else
{
while( nIndex < pImpl->Count() &&
pImpl->GetBreakIndex(nIndex) < nNodeIndex)
++nIndex;
if( nIndex < pImpl->Count() &&
pImpl->GetBreakIndex(nIndex) == nNodeIndex )
{
nType = pImpl->GetBreakType( nIndex );
nOfst = pImpl->GetBreakOfst( nIndex++ );
rbBreakAfter = sal_True;
}
}
if( nOfst < STRING_LEN )
{
sal_Bool bSplit = sal_False;
USHORT nRepeat( 0 );
2001-10-17 10:38:53 +00:00
if( !bLongTab && rpFrm->IsTxtFrm() &&
SW_LAYCACHE_IO_REC_PARA == nType &&
nOfst<((SwTxtFrm*)rpFrm)->GetTxtNode()->GetTxt().Len() )
bSplit = sal_True;
else if( rpFrm->IsTabFrm() && nRowCount < nOfst &&
2001-10-17 10:38:53 +00:00
( bLongTab || SW_LAYCACHE_IO_REC_TABLE == nType ) )
{
nRepeat = ((SwTabFrm*)rpFrm)->
GetTable()->GetRowsToRepeat();
bSplit = nOfst < nRows && nRowCount + nRepeat < nOfst;
2001-10-17 10:38:53 +00:00
bLongTab = bLongTab && bSplit;
}
if( bSplit )
{
rpFrm->InsertBehind( rpLay, rpPrv );
rpFrm->Frm().Pos() = rpLay->Frm().Pos();
rpFrm->Frm().Pos().Y() += 1;
rpPrv = rpFrm;
if( rpFrm->IsTabFrm() )
{
SwTabFrm* pTab = (SwTabFrm*)rpFrm;
// --> OD 2004-09-23 #i33629#, #i29955#
::RegistFlys( pTab->FindPageFrm(), pTab );
// <--
SwFrm *pRow = pTab->Lower();
SwTabFrm *pFoll = new SwTabFrm( *pTab );
SwFrm *pPrv;
if( nRepeat > 0 )
{
bDontCreateObjects = TRUE; //frmtool
// Insert new headlines:
USHORT nRowIdx = 0;
SwRowFrm* pHeadline = 0;
while( nRowIdx < nRepeat )
{
ASSERT( pTab->GetTable()->GetTabLines()[ nRowIdx ], "Table ohne Zeilen?" );
pHeadline =
new SwRowFrm( *pTab->GetTable()->GetTabLines()[ nRowIdx ] );
pHeadline->SetRepeatedHeadline( true );
pHeadline->InsertBefore( pFoll, 0 );
pHeadline->RegistFlys();
++nRowIdx;
}
bDontCreateObjects = FALSE;
pPrv = pHeadline;
nRows = nRows + nRepeat;
}
else
pPrv = 0;
while( pRow && nRowCount < nOfst )
{
pRow = pRow->GetNext();
++nRowCount;
}
while ( pRow )
{
SwFrm* pNxt = pRow->GetNext();
pRow->Remove();
pRow->InsertBehind( pFoll, pPrv );
pPrv = pRow;
pRow = pNxt;
}
rpFrm = pFoll;
}
else
{
SwTxtFrm *pNew = new SwTxtFrm( ((SwTxtFrm*)rpFrm)->
GetTxtNode() );
pNew->_SetIsFollow( sal_True );
pNew->ManipOfst( nOfst );
pNew->SetFollow( ((SwTxtFrm*)rpFrm)->GetFollow() );
((SwTxtFrm*)rpFrm)->SetFollow( pNew );
rpFrm = pNew;
}
}
}
}
2001-06-29 06:58:06 +00:00
SwPageFrm* pLastPage = rpPage;
if( CheckInsertPage() )
{
// --> OD 2006-03-21 #b6375613#
if ( pDoc->ApplyWorkaroundForB6375613() )
{
lcl_ApplyWorkaroundForB6375613( rpFrm );
}
// <--
2001-06-29 06:58:06 +00:00
_CheckFlyCache( pLastPage );
if( rpPrv && rpPrv->IsTxtFrm() && !rpPrv->GetValidSizeFlag() )
rpPrv->Frm().Height( rpPrv->GetUpper()->Prt().Height() );
bRet = TRUE;
rpPrv = 0;
nParagraphCnt = 0;
if ( rpActualSection )
{
//Hatte der SectionFrm ueberhaupt Inhalt? Wenn
//nicht kann er gleich umgehaengt werden.
SwSectionFrm *pSct;
bool bInit = false;
if ( !rpActualSection->GetSectionFrm()->ContainsCntnt())
{
pSct = rpActualSection->GetSectionFrm();
pSct->Remove();
}
else
{
pSct = new SwSectionFrm(
*rpActualSection->GetSectionFrm(), FALSE );
rpActualSection->GetSectionFrm()->SimpleFormat();
bInit = true;
}
rpActualSection->SetSectionFrm( pSct );
pSct->InsertBehind( rpLay, 0 );
2001-11-13 14:21:55 +00:00
if( bInit )
pSct->Init();
pSct->Frm().Pos() = rpLay->Frm().Pos();
pSct->Frm().Pos().Y() += 1; //wg. Benachrichtigungen.
rpLay = pSct;
if ( rpLay->Lower() && rpLay->Lower()->IsLayoutFrm() )
rpLay = rpLay->GetNextLayoutLeaf();
}
}
2001-10-17 10:38:53 +00:00
} while( bLongTab || ( pImpl && nIndex < pImpl->Count() &&
(*pImpl)[ nIndex ] == nNodeIndex ) );
}
bFirst = FALSE;
return bRet;
}
struct SdrObjectCompare
{
bool operator()( const SdrObject* pF1, const SdrObject* pF2 ) const
{
return pF1->GetOrdNum() < pF2->GetOrdNum();
}
};
struct FlyCacheCompare
{
bool operator()( const SwFlyCache* pC1, const SwFlyCache* pC2 ) const
{
return pC1->nOrdNum < pC2->nOrdNum;
}
};
2001-06-29 06:58:06 +00:00
/*-----------------28.6.2001 14:40------------------
* SwLayHelper::_CheckFlyCache(..)
* If a new page is inserted, the last page is analysed.
* If there are text frames with default position, the fly cache
* is checked, if these frames are stored in the cache.
* --------------------------------------------------*/
void SwLayHelper::_CheckFlyCache( SwPageFrm* pPage )
{
2001-06-29 07:18:46 +00:00
if( !pImpl || !pPage )
2001-06-29 06:58:06 +00:00
return;
USHORT nFlyCount = pImpl->GetFlyCount();
// Any text frames at the page, fly cache avaiable?
if( pPage->GetSortedObjs() && nFlyIdx < nFlyCount )
{
SwSortedObjs &rObjs = *pPage->GetSortedObjs();
2001-06-29 06:58:06 +00:00
USHORT nPgNum = pPage->GetPhyPageNum();
/*
//
// NOTE: This code assumes that all objects have already been
// inserted into the drawing layout, so that the cached objects
// can be identified by their ordnum. Unfortunately this function
// is called with page n if page n+1 has been inserted. Thus
// not all the objects have been inserted and the ordnums cannot
// be used to identify the objects.
//
2001-06-29 06:58:06 +00:00
for ( USHORT i = 0; i < rObjs.Count(); ++i ) // check objects
{
SdrObject *pO = rObjs[i];
if ( pO->ISA(SwVirtFlyDrawObj) ) // a text frame?
2001-06-29 06:58:06 +00:00
{
SwFlyFrm *pFly = ((SwVirtFlyDrawObj*)pO)->GetFlyFrm();
if( pFly->Frm().Left() == WEIT_WECH && pFly->GetAnchor() &&
!pFly->GetAnchor()->FindFooterOrHeader() )
{ // Only frame with default position and not in header/footer
const SwContact *pC = (SwContact*)GetUserCall(pO);
if( pC )
{
ULONG nOrdNum = pO->GetOrdNum(); // the Id
SwFlyCache* pFlyC;
while( nFlyIdx < nFlyCount && ( pFlyC = pImpl->
GetFlyCache(nFlyIdx) )->nPageNum < nPgNum)
++nFlyIdx;
if( nFlyIdx < nFlyCount &&
pFlyC->nPageNum == nPgNum )
{
USHORT nIdx = nFlyIdx;
while( nIdx < nFlyCount && ( pFlyC = pImpl->
GetFlyCache( nIdx ) )->nPageNum == nPgNum &&
pFlyC->nOrdNum != nOrdNum )
++nIdx;
if( nIdx < nFlyCount && pFlyC->nPageNum == nPgNum &&
pFlyC->nOrdNum == nOrdNum )
{ // we get the stored information
pFly->Frm().Pos().X() = pFlyC->Left() +
pPage->Frm().Left();
pFly->Frm().Pos().Y() = pFlyC->Top() +
pPage->Frm().Top();
pFly->Frm().Width( pFlyC->Width() );
pFly->Frm().Height( pFlyC->Height() );
}
}
}
}
}
}
*/
//
// NOTE: Here we do not use the absolute ordnums but
// relative ordnums for the objects on this page.
// skip fly frames from pages before the current page
SwFlyCache* pFlyC;
while( nFlyIdx < nFlyCount && ( pFlyC = pImpl->
GetFlyCache(nFlyIdx) )->nPageNum < nPgNum)
++nFlyIdx;
// sort cached objects on this page by ordnum
std::set< const SwFlyCache*, FlyCacheCompare > aFlyCacheSet;
USHORT nIdx = nFlyIdx;
while( nIdx < nFlyCount && ( pFlyC = pImpl->
GetFlyCache( nIdx ) )->nPageNum == nPgNum )
{
aFlyCacheSet.insert( pFlyC );
++nIdx;
}
// sort objects on this page by ordnum
std::set< const SdrObject*, SdrObjectCompare > aFlySet;
for ( USHORT i = 0; i < rObjs.Count(); ++i )
{
SwAnchoredObject* pAnchoredObj = rObjs[i];
if ( pAnchoredObj->ISA(SwFlyFrm) ) // a text frame?
{
SwFlyFrm *pFly = static_cast<SwFlyFrm*>(pAnchoredObj);
if( pFly->GetAnchorFrm() &&
!pFly->GetAnchorFrm()->FindFooterOrHeader() )
{
const SwContact *pC = ::GetUserCall( pAnchoredObj->GetDrawObj() );
if( pC )
{
aFlySet.insert( pAnchoredObj->GetDrawObj() );
}
}
}
}
if ( aFlyCacheSet.size() == aFlySet.size() )
{
std::set< const SwFlyCache*, FlyCacheCompare >::iterator aFlyCacheSetIt =
aFlyCacheSet.begin();
std::set< const SdrObject*, SdrObjectCompare >::iterator aFlySetIt =
aFlySet.begin();
while ( aFlyCacheSetIt != aFlyCacheSet.end() )
{
const SwFlyCache* pFlyCache = *aFlyCacheSetIt;
SwFlyFrm* pFly = ((SwVirtFlyDrawObj*)*aFlySetIt)->GetFlyFrm();
if ( pFly->Frm().Left() == WEIT_WECH )
{
// we get the stored information
pFly->Frm().Pos().X() = pFlyCache->Left() +
pPage->Frm().Left();
pFly->Frm().Pos().Y() = pFlyCache->Top() +
pPage->Frm().Top();
if ( pImpl->IsUseFlyCache() )
{
pFly->Frm().Width( pFlyCache->Width() );
pFly->Frm().Height( pFlyCache->Height() );
}
}
++aFlyCacheSetIt;
++aFlySetIt;
}
}
2001-06-29 06:58:06 +00:00
}
}
/*-----------------28.6.2001 14:48------------------
* SwLayHelper::CheckPageFlyCache(..)
* looks for the given text frame in the fly cache and sets
* the position and size, if possible.
* The fly cache is sorted by pages and we start searching with the given page.
* If we found the page number in the fly cache, we set
* the rpPage parameter to the right page, if possible.
* --------------------------------------------------*/
BOOL SwLayHelper::CheckPageFlyCache( SwPageFrm* &rpPage, SwFlyFrm* pFly )
{
if( !pFly->GetAnchorFrm() || !pFly->GetVirtDrawObj() ||
pFly->GetAnchorFrm()->FindFooterOrHeader() )
2001-06-29 06:58:06 +00:00
return FALSE;
BOOL bRet = FALSE;
SwDoc* pDoc = rpPage->GetFmt()->GetDoc();
SwLayCacheImpl *pCache = pDoc->GetLayoutCache() ?
pDoc->GetLayoutCache()->LockImpl() : NULL;
if( pCache )
{
USHORT nPgNum = rpPage->GetPhyPageNum();
USHORT nIdx = 0;
USHORT nCnt = pCache->GetFlyCount();
ULONG nOrdNum = pFly->GetVirtDrawObj()->GetOrdNum();
SwFlyCache* pFlyC = 0;
// skip fly frames from pages before the current page
2001-06-29 06:58:06 +00:00
while( nIdx < nCnt &&
nPgNum > (pFlyC = pCache->GetFlyCache( nIdx ))->nPageNum )
++nIdx;
2001-06-29 06:58:06 +00:00
while( nIdx < nCnt &&
nOrdNum != (pFlyC = pCache->GetFlyCache( nIdx ))->nOrdNum )
++nIdx;
if( nIdx < nCnt )
{
SwPageFrm *pPage = rpPage;
while( pPage && pPage->GetPhyPageNum() < pFlyC->nPageNum )
pPage = (SwPageFrm*)pPage->GetNext();
// --> OD 2005-02-22 #i43266# - if the found page is an empty page,
// take the previous one (take next one, if previous one doesn't exists)
if ( pPage && pPage->IsEmptyPage() )
{
pPage = static_cast<SwPageFrm*>( pPage->GetPrev()
? pPage->GetPrev()
: pPage->GetNext() );
}
// <--
2001-06-29 06:58:06 +00:00
if( pPage )
{
rpPage = pPage;
pFly->Frm().Pos().X() = pFlyC->Left() + pPage->Frm().Left();
pFly->Frm().Pos().Y() = pFlyC->Top() + pPage->Frm().Top();
if ( pCache->IsUseFlyCache() )
{
pFly->Frm().Width( pFlyC->Width() );
pFly->Frm().Height( pFlyC->Height() );
}
2001-06-29 06:58:06 +00:00
bRet = TRUE;
}
}
pDoc->GetLayoutCache()->UnlockImpl();
}
return bRet;
}
// -----------------------------------------------------------------------------
SwLayCacheIoImpl::SwLayCacheIoImpl( SvStream& rStrm, BOOL bWrtMd ) :
pStream( &rStrm ),
nMajorVersion(SW_LAYCACHE_IO_VERSION_MAJOR),
nMinorVersion(SW_LAYCACHE_IO_VERSION_MINOR),
bWriteMode( bWrtMd ),
bError( FALSE )
{
if( bWriteMode )
*pStream << nMajorVersion
<< nMinorVersion;
else
*pStream >> nMajorVersion
>> nMinorVersion;
}
BOOL SwLayCacheIoImpl::OpenRec( BYTE cType )
{
BOOL bRes = TRUE;
UINT16 nLvl = aRecTypes.Count();
ASSERT( nLvl == aRecSizes.Count(), "OpenRec: Level" );
UINT32 nPos = pStream->Tell();
if( bWriteMode )
{
aRecTypes.Insert( cType, nLvl );
aRecSizes.Insert( nPos, nLvl );
*pStream << (UINT32) 0;
}
else
{
UINT32 nVal;
*pStream >> nVal;
BYTE cRecTyp = (BYTE)nVal;
aRecTypes.Insert( cRecTyp, nLvl );
sal_uInt32 nSize = nVal >> 8;
aRecSizes.Insert( nPos + nSize, nLvl );
if( !nVal || cRecTyp != cType ||
pStream->GetErrorCode() != SVSTREAM_OK || pStream->IsEof() )
{
ASSERT( nVal, "OpenRec: Record-Header is 0" );
ASSERT( cRecTyp == cType,
"OpenRec: Wrong Record Type" );
aRecTypes[nLvl] = 0;
aRecSizes[nLvl] = pStream->Tell();
bRes = sal_False;
bError = TRUE;
}
}
return bRes;
}
// Close record
BOOL SwLayCacheIoImpl::CloseRec( BYTE )
{
BOOL bRes = TRUE;
UINT16 nLvl = aRecTypes.Count();
ASSERT( nLvl == aRecSizes.Count(), "CloseRec: wrong Level" );
ASSERT( nLvl, "CloseRec: no levels" );
if( nLvl )
{
nLvl--;
UINT32 nPos = pStream->Tell();
if( bWriteMode )
{
UINT32 nBgn = aRecSizes[nLvl];
pStream->Seek( nBgn );
UINT32 nSize = nPos - nBgn;
UINT32 nVal = ( nSize << 8 ) | aRecTypes[nLvl];
*pStream << nVal;
pStream->Seek( nPos );
if( pStream->GetError() != SVSTREAM_OK )
bRes = FALSE;
}
else
{
UINT32 n = aRecSizes[nLvl];
ASSERT( n >= nPos, "CloseRec: to much data read" );
if( n != nPos )
{
pStream->Seek( n );
if( n < nPos )
bRes = FALSE;
}
if( pStream->GetErrorCode() != SVSTREAM_OK )
bRes = FALSE;
}
aRecTypes.Remove( nLvl, 1 );
aRecSizes.Remove( nLvl, 1 );
}
if( !bRes )
bError = TRUE;
return bRes;
}
UINT32 SwLayCacheIoImpl::BytesLeft()
{
UINT16 nLvl = aRecSizes.Count();
UINT32 n = 0;
if( !bError && nLvl )
{
UINT32 nEndPos = aRecSizes[ nLvl-1 ];
UINT32 nPos = pStream->Tell();
if( nEndPos > nPos )
n = nEndPos - nPos;
}
return n;
}
BYTE SwLayCacheIoImpl::Peek()
{
BYTE c = 0;
if( !bError )
{
UINT32 nPos = pStream->Tell();
*pStream >> c;
pStream->Seek( nPos );
if( pStream->GetErrorCode() != SVSTREAM_OK )
{
c = 0;
bError = TRUE;
}
}
return c;
}
void SwLayCacheIoImpl::SkipRec()
{
BYTE c = Peek();
OpenRec( c );
pStream->Seek( aRecSizes[aRecSizes.Count()-1] );
CloseRec( c );
}
BYTE SwLayCacheIoImpl::OpenFlagRec()
{
ASSERT( !bWriteMode, "OpenFlagRec illegal in write mode" );
BYTE cFlags;
*pStream >> cFlags;
nFlagRecEnd = pStream->Tell() + ( cFlags & 0x0F );
return (cFlags >> 4);
}
void SwLayCacheIoImpl::OpenFlagRec( BYTE nFlags, BYTE nLen )
{
ASSERT( bWriteMode, "OpenFlagRec illegal in read mode" );
ASSERT( (nFlags & 0xF0) == 0, "illegal flags set" );
ASSERT( nLen < 16, "wrong flag record length" );
BYTE cFlags = (nFlags << 4) + nLen;
*pStream << cFlags;
nFlagRecEnd = pStream->Tell() + nLen;
}
void SwLayCacheIoImpl::CloseFlagRec()
{
if( bWriteMode )
{
ASSERT( pStream->Tell() == nFlagRecEnd, "Wrong amount of data written" );
}
else
{
ASSERT( pStream->Tell() <= nFlagRecEnd, "To many data read" );
if( pStream->Tell() != nFlagRecEnd )
pStream->Seek( nFlagRecEnd );
}
}