I worked under the erroneous assumption, that Doxygen will use its internal entity-IDs as the link-IDs when generating mardown-links. Yes, this seemed logical and this would be the way I'd implement it.... But seemingly, Doxygen is not so consistent when it comes to questions of syntax. The same holds true for markdown, which lacking a coherent definition anyway. Another problem is that Doxygen's auto-link generation frequently fails, for reasons not yet clear to me. Sometimes it seems to be necessary to give it a nudge by including the \ref command. While I'm not willing to go into focussed invstigation of Doxygen syntax right now, at least I've done a search-and-replace to remove the malformed links I've written the last days
288 lines
7.3 KiB
C++
288 lines
7.3 KiB
C++
/*
|
|
IterCursor(Test) - verify operation of a iterator based navigation cursor
|
|
|
|
Copyright (C) Lumiera.org
|
|
2015, 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.
|
|
|
|
* *****************************************************/
|
|
|
|
|
|
|
|
#include "lib/test/run.hpp"
|
|
#include "lib/test/test-helper.hpp"
|
|
#include "lib/format-util.hpp"
|
|
#include "lib/iter-cursor.hpp"
|
|
#include "lib/util.hpp"
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace lib {
|
|
namespace test{
|
|
|
|
using ::Test;
|
|
using util::join;
|
|
using util::isnil;
|
|
using std::vector;
|
|
|
|
using lumiera::error::LUMIERA_ERROR_ITER_EXHAUST;
|
|
|
|
|
|
namespace { // test fixture
|
|
|
|
const uint NUM_ELMS = 10;
|
|
|
|
using Numz = vector<uint>;
|
|
using Iter = IterCursor<Numz::iterator>;
|
|
using CIter = IterCursor<Numz::const_iterator>;
|
|
|
|
inline Numz
|
|
makeNumz()
|
|
{
|
|
Numz numz;
|
|
for (uint i=0; i<NUM_ELMS; ++i)
|
|
numz.push_back(i);
|
|
return numz;
|
|
}
|
|
|
|
} // (END)fixture
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************//**
|
|
* @test demonstrate and cover the properties of IterCursor.
|
|
* This wrapper allows to change between iterating forward and backwards.
|
|
*
|
|
* @see iter-cursor.hpp
|
|
* @see iter-adapter.hpp
|
|
* @see [usage example](event-log.hpp)
|
|
*/
|
|
class IterCursor_test : public Test
|
|
{
|
|
|
|
virtual void
|
|
run (Arg)
|
|
{
|
|
simpleIteration();
|
|
switchDirection();
|
|
iterTypeVariations();
|
|
}
|
|
|
|
|
|
/** @test just iterate in various ways. */
|
|
void
|
|
simpleIteration ()
|
|
{
|
|
Numz numz{makeNumz()};
|
|
Iter i1{numz.begin(), numz.end()};
|
|
|
|
CHECK (!isnil(i1));
|
|
CHECK (0 == *i1);
|
|
++++++i1;
|
|
CHECK (3 == *i1);
|
|
for (uint i=*i1 ; i1; ++i1, ++i)
|
|
CHECK (i == *i1);
|
|
|
|
CHECK (isnil(i1));
|
|
|
|
Iter i2{numz};
|
|
uint sum =0;
|
|
while (++i2)
|
|
sum += *i2;
|
|
uint n = numz.size() - 1;
|
|
CHECK (sum == n*(n+1) / 2);
|
|
|
|
Iter i3{numz};
|
|
for (auto & i : i3)
|
|
++i; // note: manipulate the contents...
|
|
|
|
Iter i4{numz};
|
|
CHECK (join(i4,"-+-") == "1-+-2-+-3-+-4-+-5-+-6-+-7-+-8-+-9-+-10");
|
|
verifyComparisons (i4);
|
|
}
|
|
|
|
|
|
|
|
/** @test verify the ability of IterCursor to switch
|
|
* the direction of the iteration. This "gear switch" can be done
|
|
* any time, while in the middle of iteration, and even after
|
|
* iteration end. That means, even an exhausted iterator can be
|
|
* "turned back". This does not work on a default constructed
|
|
* IterCursor, though.
|
|
*/
|
|
void
|
|
switchDirection ()
|
|
{
|
|
Numz numz{makeNumz()};
|
|
Iter iter{numz};
|
|
|
|
CHECK (0 == *iter);
|
|
++++++++iter;
|
|
CHECK (4 == *iter);
|
|
CHECK (!isnil(iter));
|
|
CHECK (join(iter) == "4, 5, 6, 7, 8, 9");
|
|
verifyComparisons (iter);
|
|
|
|
iter.switchDir();
|
|
CHECK (4 == *iter);
|
|
CHECK (!isnil(iter));
|
|
++iter;
|
|
CHECK (3 == *iter);
|
|
CHECK (!isnil(iter));
|
|
CHECK (join(iter) == "3, 2, 1, 0");
|
|
verifyComparisons (iter);
|
|
|
|
++iter;
|
|
CHECK (2 == *iter);
|
|
++++iter;
|
|
CHECK (0 == *iter);
|
|
CHECK (!isnil(iter));
|
|
|
|
iter.switchDir();
|
|
CHECK (0 == *iter);
|
|
CHECK (!isnil(iter));
|
|
++iter;
|
|
CHECK (1 == *iter);
|
|
|
|
iter.switchDir();
|
|
++iter;
|
|
CHECK (0 == *iter);
|
|
CHECK (!isnil(iter));
|
|
|
|
++iter;
|
|
CHECK (isnil(iter));
|
|
VERIFY_ERROR (ITER_EXHAUST, *iter);
|
|
VERIFY_ERROR (ITER_EXHAUST, ++iter);
|
|
|
|
iter.switchDir();
|
|
CHECK (!isnil(iter));
|
|
CHECK (0 == *iter);
|
|
|
|
while (++iter);
|
|
CHECK (isnil(iter));
|
|
VERIFY_ERROR (ITER_EXHAUST, *iter);
|
|
VERIFY_ERROR (ITER_EXHAUST, ++iter);
|
|
|
|
iter.switchDir();
|
|
CHECK (!isnil(iter));
|
|
CHECK (9 == *iter);
|
|
|
|
Iter nil;
|
|
CHECK (isnil (nil));
|
|
iter.switchDir();
|
|
CHECK (isnil (nil));
|
|
VERIFY_ERROR (ITER_EXHAUST, *nil);
|
|
VERIFY_ERROR (ITER_EXHAUST, ++nil);
|
|
}
|
|
|
|
|
|
|
|
/** @test verify the const and dereferencing variants,
|
|
* based on the const-ness of the underlying STL iterator
|
|
*/
|
|
void
|
|
iterTypeVariations ()
|
|
{
|
|
Numz numz{makeNumz()};
|
|
Numz const& const_numz{numz};
|
|
|
|
uint i = 0;
|
|
for (Iter iter{numz};
|
|
iter; ++iter, ++i
|
|
)
|
|
{
|
|
CHECK (iter);
|
|
CHECK (iter != Iter());
|
|
CHECK (*iter == i);
|
|
--(*iter);
|
|
CHECK (*iter == i-1);
|
|
}
|
|
|
|
i = 0;
|
|
for (CIter iter{const_numz};
|
|
iter; ++iter, ++i
|
|
)
|
|
{
|
|
CHECK (iter);
|
|
CHECK (iter != CIter());
|
|
CHECK (*iter == i-1);
|
|
|
|
// note: the previous run indeed modified
|
|
// the elements within the container.
|
|
|
|
// ++(*iter); // doesn't compile, because it yields a "* const"
|
|
}
|
|
|
|
verifyComparisons (CIter{numz});
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** @test verify equality handling and NIL detection
|
|
* for the given iterator/wrapper handed in.
|
|
* @note the argument is not altered; rather we create
|
|
* several copies, to iterate and compare those
|
|
*/
|
|
template<class IT>
|
|
void
|
|
verifyComparisons (IT const& ii)
|
|
{
|
|
IT i1(ii);
|
|
IT i2(ii);
|
|
IT iN;
|
|
CHECK ( isnil (iN));
|
|
CHECK (!isnil (i1));
|
|
CHECK (!isnil (i2));
|
|
|
|
CHECK (i1 == i2); CHECK (i2 == i1);
|
|
CHECK (i1 != iN); CHECK (iN != i1);
|
|
CHECK (i2 != iN); CHECK (iN != i2);
|
|
|
|
++i1;
|
|
CHECK (i1 != i2);
|
|
CHECK (i1 != iN);
|
|
|
|
++i2;
|
|
CHECK (i1 == i2);
|
|
CHECK (i1 != iN);
|
|
CHECK (i2 != iN);
|
|
|
|
while (++i1) { }
|
|
CHECK (isnil(i1));
|
|
CHECK (i1 != i2);
|
|
CHECK (i1 == iN);
|
|
|
|
while (++i2) { }
|
|
CHECK (isnil(i2));
|
|
CHECK (i2 == i1);
|
|
CHECK (i2 == iN);
|
|
}
|
|
};
|
|
|
|
LAUNCHER (IterCursor_test, "unit common");
|
|
|
|
|
|
}} // namespace lib::test
|
|
|