/************************************************************************* * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: pagemanager.cxx,v $ * * $Revision: 1.3 $ * * last change: $Author: hr $ $Date: 2006-06-20 02:18:19 $ * * 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 * ************************************************************************/ #include #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; } }