// // wclibase.h Defines for the WATCOM Container List Iterator Base Classes // // Copyright by WATCOM International Corp. 1988-1996. All rights reserved. // #ifndef _WCLIBASE_H_INCLUDED #define _WCLIBASE_H_INCLUDED #if !defined(_ENABLE_AUTODEPEND) #pragma read_only_file; #endif #ifndef __cplusplus #error wclibase.h is for use with C++ #endif #ifndef _WCDEFS_H_INCLUDED #include #endif #ifndef _WCEXCEPT_H_INCLUDED #include #endif // // The WCListIterBase class is used as a basis for the iterator // classes for the various list containers. // // It is an abstract base class to prevent objects of this class // from being created. // class WCListIterBase : public WCIterExcept { protected: enum { // flags for outside_list before_first = 1, after_last }; WCIsvListBase * curr_list; WCSLink * curr_item; // if outside_list is non-zero, curr_item must be 0. int outside_list; _WPRTLINK WCSLink * base_advance( int ); _WPRTLINK WCDLink * base_retreat( int ); inline WCSLink * base_get_tail() { return( curr_list->tail ); }; // helpers for insert and append _WPRTLINK WCSLink * base_tail_hit( WCSLink * ); _WPRTLINK void base_tail_unhit( WCSLink * ); inline void base_reset(); inline void base_reset_list( WCIsvListBase * list ); public: inline WCListIterBase() : curr_item( 0 ), curr_list( 0 ) , outside_list( 0 ) {}; inline WCListIterBase( WCIsvListBase * list ) : curr_item( 0 ), curr_list( list ) , outside_list( before_first ) {}; inline virtual ~WCListIterBase() = 0; inline WCSLink * operator()(); inline WCSLink * operator++(); inline WCSLink * operator+=( int ); inline WCDLink * operator--(); inline WCDLink * operator-=( int ); }; inline WCListIterBase::~WCListIterBase() {}; inline void WCListIterBase::base_reset() { curr_item = 0; outside_list = before_first; } inline void WCListIterBase::base_reset_list( WCIsvListBase * list ) { curr_list = list; base_reset(); } inline WCSLink * WCListIterBase::operator()() { return( base_advance( 1 ) ); } inline WCSLink * WCListIterBase::operator++() { return( base_advance( 1 ) ); } inline WCDLink * WCListIterBase::operator--() { return( base_retreat( 1 ) ); } inline WCSLink * WCListIterBase::operator+=( int incr_amount ) { return( base_advance( incr_amount ) ); } inline WCDLink * WCListIterBase::operator-=( int incr_amount ) { return( base_retreat( incr_amount ) ); } // // The WCIsvListIterBase defines the base iterator class for intrusive // value type containers. // template class WCIsvListIterBase : public WCListIterBase { public: inline WCIsvListIterBase() {}; inline virtual ~WCIsvListIterBase() = 0; inline WCIsvListIterBase( FType * slist ) : WCListIterBase( slist ) {}; inline void reset( FType & slist ) { WCListIterBase::base_reset_list( &slist ); }; inline void reset() { WCListIterBase::base_reset(); }; inline Type * operator()() { return( (Type *)WCListIterBase::operator()() ); }; inline Type * operator++() { return( (Type *)WCListIterBase::operator++() ); }; inline Type * operator--() { return( (Type *)WCListIterBase::operator--() ); }; inline Type * operator+=( int adv_amount ) { return( (Type *)WCListIterBase::operator+=( adv_amount ) ); }; inline Type * operator-=( int adv_amount ) { return( (Type *)WCListIterBase::operator-=( adv_amount ) ); }; inline Type * current() const { if( ( curr_list == 0 )||( curr_item == 0 ) ) { base_throw_undef_item(); return( 0 ); } return( (Type *)curr_item ); }; inline FType * container() const { if( curr_list == 0 ) { base_throw_undef_iter(); } return( (FType *)curr_list ); }; WCbool insert( Type * ); WCbool append( Type * ); }; template inline WCIsvListIterBase::~WCIsvListIterBase() {}; template WCbool WCIsvListIterBase::insert( Type * datum ) { if( ( curr_list == 0 )||( curr_item == 0 ) ) { base_throw_undef_iter(); return( FALSE ); } WCSLink * prev_item = curr_item; WCbool ret_val; if( prev_item != 0 ) { prev_item = ((WCDLink *)prev_item)->prev.link; } // make the tail the element before the element being inserted WCSLink * curr_tail = base_tail_hit( prev_item ); ret_val = ((FType *)curr_list)->insert( datum ); // restore the tail base_tail_unhit( curr_tail ); return( ret_val ); }; template WCbool WCIsvListIterBase::append( Type * datum ) { if( ( curr_list == 0 )||( curr_item == 0 ) ) { base_throw_undef_iter(); return( FALSE ); } WCbool ret_val; if( curr_item == base_get_tail() ) { // append the new element to the end of the list ret_val = ((FType *)curr_list)->append( datum ); } else { // the new element is not the last element in the list, so make // the tail the element before the new element (curr_item) WCSLink * curr_tail = base_tail_hit( curr_item ); ret_val = ((FType *)curr_list)->insert( datum ); // restore the tail base_tail_unhit( curr_tail ); } return( ret_val ); }; // // The WCValListIterBase defines the base iterator class for value type // containers. // template class WCValListIterBase : public WCListIterBase { public: inline WCValListIterBase() {}; inline virtual ~WCValListIterBase() = 0; inline WCValListIterBase( FType * slist ) : WCListIterBase( slist ) {}; inline void reset( FType & dlist ) { WCListIterBase::base_reset_list( &dlist ); }; inline void reset() { WCListIterBase::base_reset(); }; inline int operator()() { return( WCListIterBase::operator()() != 0 ); }; inline int operator++() { return( WCListIterBase::operator++() != 0 ); }; inline int operator--() { return( WCListIterBase::operator--() != 0 ); }; inline int operator+=( int adv_amount ) { return( WCListIterBase::operator+=( adv_amount ) != 0 ); }; inline int operator-=( int adv_amount ) { return( WCListIterBase::operator-=( adv_amount ) != 0 ); }; inline FType * container() const { if( curr_list == 0 ) { base_throw_undef_iter(); } return( (FType *)curr_list ); }; Type current() const; WCbool insert( Type& ); WCbool append( Type& ); }; template inline WCValListIterBase::~WCValListIterBase() {}; template Type WCValListIterBase::current() const { if( ( curr_list == 0 )||( curr_item == 0 ) ) { base_throw_undef_item(); Type ret_val; return( ret_val ); } else { return( ((LType *)curr_item)->data ); } }; template WCbool WCValListIterBase::insert( Type& datum ) { if( ( curr_list == 0 )||( curr_item == 0 ) ) { base_throw_undef_iter(); return( FALSE ); } WCSLink * prev_item = curr_item; WCbool ret_val; if( prev_item != 0 ) { prev_item = ((WCDLink *)prev_item)->prev.link; } // make the tail the element before the element being inserted WCSLink * curr_tail = base_tail_hit( prev_item ); ret_val = ((FType *)curr_list)->insert( datum ); // restore the tail base_tail_unhit( curr_tail ); return( ret_val ); }; template WCbool WCValListIterBase::append( Type& datum ) { if( ( curr_list == 0 )||( curr_item == 0 ) ) { base_throw_undef_iter(); return( FALSE ); } WCbool ret_val; if( curr_item == base_get_tail() ) { // append the new element to the end of the list ret_val = ((FType *)curr_list)->append( datum ); } else { // the new element is not the last element in the list, so make // the tail the element before the new element (curr_item) WCSLink * curr_tail = base_tail_hit( curr_item ); ret_val = ((FType *)curr_list)->insert( datum ); // restore the tail base_tail_unhit( curr_tail ); } return( ret_val ); }; #endif