2010-01-05 02:53:20 +01:00
|
|
|
/*
|
|
|
|
|
ITER-SOURCE.hpp - an interface to build an opaque iterator-based data source
|
|
|
|
|
|
|
|
|
|
Copyright (C) Lumiera.org
|
|
|
|
|
2010, Hermann Vosseler <Ichthyostega@web.de>
|
|
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
|
|
|
modify it under the terms of the GNU General Public License as
|
|
|
|
|
published by the Free Software Foundation; either version 2 of the
|
|
|
|
|
License, or (at your option) any later version.
|
|
|
|
|
|
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/** @file iter-source.hpp
|
|
|
|
|
** Extension module to build an opaque data source, accessible as
|
|
|
|
|
** <i>Lumiera Forward Iterator</i>. It is based on combining an IterAdapter
|
|
|
|
|
** with classical polymorphism; here, the data source, which is addressed
|
|
|
|
|
** by IderAdapter through the "iteration control API", is abstracted behind
|
|
|
|
|
** an interface (with virtual functions). Together this allows to build
|
|
|
|
|
** a simple data source type, without needing to disclose details of
|
|
|
|
|
** the implementation.
|
|
|
|
|
**
|
|
|
|
|
** @see iter-adapter.hpp
|
|
|
|
|
** @see itertool.hpp
|
|
|
|
|
** @see iter-source-test.cpp
|
|
|
|
|
**
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef LIB_ITER_SOURCE_H
|
|
|
|
|
#define LIB_ITER_SOURCE_H
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "lib/iter-adapter.hpp"
|
|
|
|
|
//#include "lib/bool-checkable.hpp"
|
|
|
|
|
|
2010-01-05 04:10:23 +01:00
|
|
|
#include <tr1/memory>
|
|
|
|
|
|
2010-01-05 02:53:20 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
namespace lib {
|
2010-01-05 04:10:23 +01:00
|
|
|
|
|
|
|
|
using std::tr1::shared_ptr;
|
2010-01-05 02:53:20 +01:00
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Iteration source interface to abstract a data source,
|
|
|
|
|
* which then can be accessed through IterAdapter as a frontend,
|
|
|
|
|
* allowing to pull individual elements until exhaustion.
|
|
|
|
|
*
|
2010-01-05 04:10:23 +01:00
|
|
|
* This base class is empty and makes now assumptions regarding
|
|
|
|
|
* identity, instantiation and copying.
|
|
|
|
|
*
|
2010-01-05 02:53:20 +01:00
|
|
|
* @see PlacementIndex::Table#_eachEntry_4check usage example
|
|
|
|
|
* @see iter-source-test.cpp
|
|
|
|
|
*/
|
2010-01-05 04:10:23 +01:00
|
|
|
template<typename TY>
|
2010-01-05 02:53:20 +01:00
|
|
|
class IterSource
|
2010-01-05 04:10:23 +01:00
|
|
|
// : public lib::BoolCheckable<IterSource<TY> >
|
2010-01-05 02:53:20 +01:00
|
|
|
{
|
2010-01-05 04:10:23 +01:00
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
detach_without_destroy (IterSource * source)
|
|
|
|
|
{
|
|
|
|
|
WARN_IF (!source, library, "IterSource deleter called with NULL source pointer");
|
|
|
|
|
if (source)
|
|
|
|
|
source->disconnect();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
protected: /* == data source API for implementation == */
|
|
|
|
|
|
|
|
|
|
typedef TY* Pos;
|
|
|
|
|
typedef shared_ptr<IterSource> DataHandle;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** iteration start: prepare the first element.
|
|
|
|
|
* may return NULL in case of empty data source
|
|
|
|
|
*/
|
|
|
|
|
virtual Pos firstResult () =0;
|
|
|
|
|
|
|
|
|
|
/** iteration step: switch on to the next element.
|
|
|
|
|
* The pos pointer may be set to NULL to report
|
|
|
|
|
* iteration end
|
|
|
|
|
*/
|
|
|
|
|
virtual void nextResult(Pos& pos) =0;
|
|
|
|
|
|
|
|
|
|
/** disconnect the data source / iteration frontend.
|
|
|
|
|
* Depending on the way the IterSource got created
|
|
|
|
|
* this \em might (or might not) be followed by
|
|
|
|
|
* destroying the data source.
|
|
|
|
|
* @note must not throw
|
|
|
|
|
*/
|
|
|
|
|
virtual void disconnect () { }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
|
|
virtual ~IterSource() { }; ///< is ABC
|
|
|
|
|
|
|
|
|
|
/* == Iteration control API for IterAdapter frontend == */
|
|
|
|
|
|
|
|
|
|
friend bool
|
|
|
|
|
hasNext (DataHandle const&, Pos const& pos) ////TICKET #410
|
|
|
|
|
{
|
|
|
|
|
return bool(pos);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
friend void
|
|
|
|
|
iterNext (DataHandle& source, Pos& pos)
|
|
|
|
|
{
|
|
|
|
|
source->nextResult(pos);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* == public builder API to create instances == */
|
|
|
|
|
|
|
|
|
|
typedef IterAdapter<Pos, DataHandle> iterator;
|
|
|
|
|
|
|
|
|
|
static iterator
|
|
|
|
|
build (IterSource& sourceImpl)
|
|
|
|
|
{
|
|
|
|
|
DataHandle sourceHandle (&sourceImpl, &detach_without_destroy);
|
|
|
|
|
Pos first = sourceImpl.firstResult();
|
|
|
|
|
return iterator (sourceHandle, first);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static iterator EMPTY_SOURCE;
|
2010-01-05 02:53:20 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2010-01-05 04:10:23 +01:00
|
|
|
/** storage for the empty data-source constant */
|
|
|
|
|
template<typename TY>
|
|
|
|
|
typename IterSource<TY>::iterator IterSource<TY>::EMPTY_SOURCE = iterator();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* === pre-defined Adapters for frequently used Containers === */
|
|
|
|
|
|
|
|
|
|
namespace iter_impl {
|
|
|
|
|
|
|
|
|
|
template<class CON>
|
|
|
|
|
struct _SeqType
|
|
|
|
|
{
|
|
|
|
|
typedef typename CON::iterator::value_type Val;
|
|
|
|
|
typedef typename IterSource<Val>::iterator Iter;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template<class MAP>
|
|
|
|
|
struct _MapType
|
|
|
|
|
{
|
|
|
|
|
typedef typename MAP::key_type Key;
|
|
|
|
|
typedef typename IterSource<Key>::iterator Iter;
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<class MAP>
|
|
|
|
|
typename iter_impl::_MapType<MAP>::Iter
|
|
|
|
|
eachMapKey (MAP& map)
|
|
|
|
|
{
|
|
|
|
|
UNIMPLEMENTED ("standard iter wrapper yielding all map keys");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
template<class MAP>
|
|
|
|
|
typename iter_impl::_MapType<MAP>::Iter
|
|
|
|
|
eachDistinctKey (MAP& map)
|
|
|
|
|
{
|
|
|
|
|
UNIMPLEMENTED ("standard iter wrapper yielding all distinct keys of a multimap");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
template<class CON>
|
|
|
|
|
typename iter_impl::_SeqType<CON>::Iter
|
|
|
|
|
eachEntry (CON& container)
|
|
|
|
|
{
|
|
|
|
|
UNIMPLEMENTED ("standard iter wrapper yielding all distinct keys of a multimap");
|
|
|
|
|
}
|
|
|
|
|
|
2010-01-05 02:53:20 +01:00
|
|
|
|
|
|
|
|
} // namespace lib
|
|
|
|
|
#endif
|