/************************************************************************* * * $RCSfile: out_node.cxx,v $ * * $Revision: 1.1 $ * * last change: $Author: np $ $Date: 2002-11-01 17:15:27 $ * * The Contents of this file are made available subject to the terms of * either of the following licenses * * - GNU Lesser General Public License Version 2.1 * - Sun Industry Standards Source License Version 1.1 * * Sun Microsystems Inc., October, 2000 * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2000 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 * * * Sun Industry Standards Source License Version 1.1 * ================================================= * The contents of this file are subject to the Sun Industry Standards * Source License Version 1.1 (the "License"); You may not use this file * except in compliance with the License. You may obtain a copy of the * License at http://www.openoffice.org/license.html. * * Software provided under this License is provided on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, * WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS, * MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING. * See the License for the specific provisions governing your rights and * obligations concerning the Software. * * The Initial Developer of the Original Code is: Sun Microsystems, Inc. * * Copyright: 2000 by Sun Microsystems, Inc. * * All Rights Reserved. * * Contributor(s): _______________________________________ * * ************************************************************************/ #include #include // NOT FULLY DEFINED SERVICES #include namespace output { namespace { struct Less_NodePtr { bool operator()( Node * p1, Node * p2 ) const { return p1->Name() < p2->Name(); } }; struct Less_NodePtr C_Less_NodePtr; Node C_aNullNode(Node::null_object); } // namepace anonymous //********************** Node ***************************// Node::Node() : sName(), pParent(0), aChildren(), nDepth(0), nNameRoomId(0) { } Node::Node( E_NullObject ) : sName(), pParent(0), aChildren(), nDepth(-1), nNameRoomId(0) { } Node::Node( const String & i_name, Node & i_parent ) : sName(i_name), pParent(&i_parent), aChildren(), nDepth(i_parent.Depth()+1), nNameRoomId(0) { } Node::~Node() { for ( List::iterator it = aChildren.begin(); it != aChildren.end(); ++it ) { delete *it; } } Node & Node::Provide_Child( const String & i_name ) { Node * ret = find_Child(i_name); if (ret != 0) return *ret; return add_Child(i_name); } void Node::Get_Path( StreamStr & o_result, intt i_maxDepth ) const { // Intentionally 'i_maxDepth != 0', so max_Depth == -1 sets no limit: if (i_maxDepth != 0) { if (pParent != 0) pParent->Get_Path(o_result, i_maxDepth-1); o_result << sName << '/'; } } void Node::Get_Chain( StringVector & o_result, intt i_maxDepth ) const { if (i_maxDepth != 0) { // This is called also for the toplevel Node, // but there happens nothing: if (pParent != 0) { pParent->Get_Chain(o_result, i_maxDepth-1); o_result.push_back(sName); } } } Node * Node::find_Child( const String & i_name ) { // Trying to optimise search in a vector: uintt nSize = aChildren.size(); if ( nSize > 0 AND nSize < 20 ) { List::const_iterator it = aChildren.begin(); List::const_iterator itMiddle = it + nSize/2; int comp = 0; if ( i_name < (*itMiddle)->Name() ) { for ( ; it != itMiddle ? (comp = csv::compare(i_name, (*it)->Name())) <= 0 : false; ++it ) { if ( comp == 0 ) return *it; } // end for } else { List::const_iterator itEnd = aChildren.end(); for ( it = itMiddle; it != itEnd ? (comp = csv::compare(i_name, (*it)->Name())) <= 0 : false; ++it ) { if ( comp == 0 ) return *it; } // end for } } else if (nSize > 0) { Node aSearch; aSearch.sName = i_name; List::const_iterator ret = std::lower_bound( aChildren.begin(), aChildren.end(), &aSearch, C_Less_NodePtr ); if ( ret != aChildren.end() ? (*ret)->Name() == i_name : false ) return *ret; } return 0; } Node & Node::add_Child( const String & i_name ) { Dyn pNew( new Node(i_name,*this) ); Node * pSearch = pNew.Ptr(); // Necessary, because Release() invalidates pNew. List::iterator itNew = aChildren.insert( std::upper_bound( aChildren.begin(), aChildren.end(), pSearch, C_Less_NodePtr ), pSearch ); return *pNew.Release(); // Release, because it is hold by aChildren now. } Node & Node::provide_Child( StringVector::const_iterator i_next, StringVector::const_iterator i_end ) { if (i_next == i_end) return *this; return Provide_Child(*i_next).provide_Child(i_next+1,i_end); } Node & Node::Null_() { return C_aNullNode; } } // namespace output