Files
libreoffice/canvas/source/tools/pagemanager.cxx

225 lines
8.2 KiB
C++
Raw Normal View History

/*************************************************************************
*
* OpenOffice.org - a multi-platform office productivity suite
*
* $RCSfile: pagemanager.cxx,v $
*
* $Revision: 1.4 $
*
* last change: $Author: obo $ $Date: 2006-09-17 03:26:32 $
*
* The Contents of this file are made available subject to
* the terms of GNU Lesser General Public License Version 2.1.
*
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2005 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
*
************************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_canvas.hxx"
#include <boost/bind.hpp>
#include "pagemanager.hxx"
namespace canvas
{
//////////////////////////////////////////////////////////////////////////////////
// PageManager
//////////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////////
// PageManager::allocateSpace
//////////////////////////////////////////////////////////////////////////////////
FragmentSharedPtr PageManager::allocateSpace( const ::basegfx::B2ISize& rSize )
{
// we are asked to find a location for the requested size.
// first we try to satisfy the request from the
// remaining space in the existing pages.
const PageContainer_t::iterator aEnd(maPages.end());
PageContainer_t::iterator it(maPages.begin());
while(it != aEnd)
{
FragmentSharedPtr pFragment((*it)->allocateSpace(rSize));
if(pFragment)
{
// the page created a new fragment, since we maybe want
// to consolidate sparse pages we keep a reference to
// the fragment.
maFragments.push_back(pFragment);
return pFragment;
}
++it;
}
// otherwise try to create a new page and allocate space there...
PageSharedPtr pPage(new Page(mpRenderModule));
if(pPage->isValid())
{
maPages.push_back(pPage);
FragmentSharedPtr pFragment(pPage->allocateSpace(rSize));
maFragments.push_back(pFragment);
return pFragment;
}
// the rendermodule failed to create a new page [maybe out
// of videomemory], and all other pages could not take
// the new request. we decide to create a 'naked' fragment
// which will receive its location later.
FragmentSharedPtr pFragment(new PageFragment(rSize));
maFragments.push_back(pFragment);
return pFragment;
}
//////////////////////////////////////////////////////////////////////////////////
// PageManager::free
//////////////////////////////////////////////////////////////////////////////////
void PageManager::free( const FragmentSharedPtr& pFragment )
{
// erase the reference to the given fragment from our
// internal container.
FragmentContainer_t::iterator it(
std::remove(
maFragments.begin(),maFragments.end(),pFragment));
maFragments.erase(it,maFragments.end());
// let the fragment itself know about it...
// we need to pass 'this' as argument since the fragment
// needs to pass this to the page and can't create
// shared_ptr from itself...
pFragment->free(pFragment);
}
//////////////////////////////////////////////////////////////////////////////////
// PageManager::nakedFragment
//////////////////////////////////////////////////////////////////////////////////
void PageManager::nakedFragment( const FragmentSharedPtr& pFragment )
{
if(maPages.empty())
return;
// okay, one last chance is left, we try all available
// pages again. maybe some other fragment was deleted
// and we can exploit the space.
while(!(relocate(pFragment)))
{
// no way, we need to free up some space...
// TODO(F1): this is a heuristic, could
// be designed as a policy.
const FragmentContainer_t::const_iterator aEnd(maFragments.end());
FragmentContainer_t::const_iterator candidate(maFragments.begin());
while(candidate != aEnd)
{
if(!((*candidate)->isNaked()))
break;
++candidate;
}
const ::basegfx::B2ISize& rSize((*candidate)->getSize());
sal_uInt32 nMaxArea(rSize.getX()*rSize.getY());
FragmentContainer_t::const_iterator it(candidate);
while(it != aEnd)
{
if(!((*it)->isNaked()))
{
const ::basegfx::B2ISize& rCandidateSize((*it)->getSize());
const sal_uInt32 nArea(rCandidateSize.getX()*rCandidateSize.getY());
if(nArea > nMaxArea)
{
candidate=it;
nMaxArea=nArea;
}
}
++it;
}
// this does not erase the candidate,
// but makes it 'naked'...
(*candidate)->free(*candidate);
}
}
//////////////////////////////////////////////////////////////////////////////////
// PageManager::relocate
//////////////////////////////////////////////////////////////////////////////////
bool PageManager::relocate( const FragmentSharedPtr& pFragment )
{
// the fragment passed as argument is assumed to
// be naked, that is it is not located on any page.
// we try all available pages again, maybe some
// other fragment was deleted and we can exploit the space.
const PageContainer_t::iterator aEnd(maPages.end());
PageContainer_t::iterator it(maPages.begin());
while(it != aEnd)
{
// if the page at hand takes the fragment, we immediatelly
// call select() to pull the information from the associated
// image to the hardware surface.
if((*it)->nakedFragment(pFragment))
{
// dirty, since newly allocated.
pFragment->select(true);
return true;
}
++it;
}
return false;
}
//////////////////////////////////////////////////////////////////////////////////
// PageManager::validatePages
//////////////////////////////////////////////////////////////////////////////////
void PageManager::validatePages()
{
::std::for_each( maPages.begin(),
maPages.end(),
::boost::mem_fn(&Page::validate));
}
//////////////////////////////////////////////////////////////////////////////////
// PageManager::getPageSize
//////////////////////////////////////////////////////////////////////////////////
::basegfx::B2ISize PageManager::getPageSize() const
{
return mpRenderModule->getPageSize();
}
//////////////////////////////////////////////////////////////////////////////////
// PageManager::getRenderModule
//////////////////////////////////////////////////////////////////////////////////
canvas::IRenderModuleSharedPtr PageManager::getRenderModule() const
{
return mpRenderModule;
}
}