minetest/src/unittest/test_clientactiveobjectmgr.cpp

189 lines
5.6 KiB
C++
Raw Permalink Normal View History

/*
Minetest
Copyright (C) 2018 nerzhul, Loic Blot <loic.blot@unix-experience.fr>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include "test.h"
#include "client/activeobjectmgr.h"
#include <catch.hpp>
#include <unordered_set>
#include <utility>
class TestClientActiveObject : public ClientActiveObject
{
public:
TestClientActiveObject() : ClientActiveObject(0, nullptr, nullptr) {}
~TestClientActiveObject() = default;
ActiveObjectType getType() const { return ACTIVEOBJECT_TYPE_TEST; }
virtual void addToScene(ITextureSource *tsrc, scene::ISceneManager *smgr) {}
};
class TestSelectableClientActiveObject : public ClientActiveObject
{
public:
TestSelectableClientActiveObject(aabb3f _selection_box)
: ClientActiveObject(0, nullptr, nullptr)
, selection_box(_selection_box)
{}
~TestSelectableClientActiveObject() = default;
ActiveObjectType getType() const override { return ACTIVEOBJECT_TYPE_TEST; }
void addToScene(ITextureSource *tsrc, scene::ISceneManager *smgr) override {}
bool getSelectionBox(aabb3f *toset) const override { *toset = selection_box; return true; }
const v3f getPosition() const override { return position; }
v3f position;
aabb3f selection_box;
};
class TestClientActiveObjectMgr : public TestBase
{
public:
TestClientActiveObjectMgr() { TestManager::registerTestModule(this); }
const char *getName() { return "TestClientActiveObjectMgr"; }
void runTests(IGameDef *gamedef);
void testGetActiveSelectableObjects();
};
static TestClientActiveObjectMgr g_test_instance;
void TestClientActiveObjectMgr::runTests(IGameDef *gamedef)
{
TEST(testGetActiveSelectableObjects)
}
////////////////////////////////////////////////////////////////////////////////
static TestClientActiveObject* register_default_test_object(
client::ActiveObjectMgr &caomgr) {
auto test_obj = std::make_unique<TestClientActiveObject>();
auto result = test_obj.get();
REQUIRE(caomgr.registerObject(std::move(test_obj)));
return result;
}
TEST_CASE("test client active object manager")
{
client::ActiveObjectMgr caomgr;
auto tcao1 = register_default_test_object(caomgr);
SECTION("When we register many client objects, "
"then all the assigned IDs should be unique.")
{
// This should be enough rounds to be pretty confident
// there are no duplicates.
u16 n = 255;
std::unordered_set<u16> ids;
ids.insert(tcao1->getId());
for (u16 i = 0; i < n; ++i) {
auto other_tcao = register_default_test_object(caomgr);
ids.insert(other_tcao->getId());
}
// n added objects & tcao1
CHECK(n + 1 == static_cast<u16>(ids.size()));
}
SECTION("two registered objects")
{
auto tcao2 = register_default_test_object(caomgr);
auto tcao2_id = tcao2->getId();
auto obj1 = caomgr.getActiveObject(tcao1->getId());
REQUIRE(obj1 != nullptr);
auto obj2 = caomgr.getActiveObject(tcao2_id);
REQUIRE(obj2 != nullptr);
SECTION("When we query an object by its ID, "
"then we should get back an object with that ID.")
{
CHECK(obj1->getId() == tcao1->getId());
CHECK(obj2->getId() == tcao2->getId());
}
SECTION("When we register and query for an object, "
"its memory location should not have changed.")
{
CHECK(obj1 == tcao1);
CHECK(obj2 == tcao2);
}
}
SECTION("Given an object has been removed, "
"when we query for it, "
"then we should get nullptr.")
{
auto id = tcao1->getId();
caomgr.removeObject(tcao1->getId()); // may invalidate tcao1
CHECK(caomgr.getActiveObject(id) == nullptr);
}
caomgr.clear();
}
void TestClientActiveObjectMgr::testGetActiveSelectableObjects()
{
client::ActiveObjectMgr caomgr;
2023-10-09 17:13:04 +02:00
auto obj_u = std::make_unique<TestSelectableClientActiveObject>(
aabb3f{v3f{-1, -1, -1}, v3f{1, 1, 1}});
auto obj = obj_u.get();
UASSERT(caomgr.registerObject(std::move(obj_u)));
auto assert_obj_selected = [&] (v3f a, v3f b) {
auto actual = caomgr.getActiveSelectableObjects({a, b});
UASSERTEQ(auto, actual.size(), 1u);
UASSERTEQ(auto, actual.at(0).obj, obj);
};
auto assert_obj_missed = [&] (v3f a, v3f b) {
auto actual = caomgr.getActiveSelectableObjects({a, b});
UASSERTEQ(auto, actual.size(), 0u);
};
float x = 12, y = 3, z = 6;
obj->position = {x, y, z};
assert_obj_selected({0, 0, 0}, {x-1, y-1, z-1});
assert_obj_selected({0, 0, 0}, {2*(x-1), 2*(y-1), 2*(z-1)});
assert_obj_selected({0, 0, 0}, {2*(x+1), 2*(y-1), 2*(z+1)});
assert_obj_selected({0, 0, 0}, {20, 5, 10});
assert_obj_selected({30, -12, 17}, {x+1, y+1, z-1});
assert_obj_selected({30, -12, 17}, {x, y+1, z});
assert_obj_selected({30, -12, 17}, {-6, 20, -5});
assert_obj_selected({30, -12, 17}, {-8, 20, -7});
assert_obj_selected({-21, 6, -13}, {x+1.4f, y, z});
assert_obj_selected({-21, 6, -13}, {x-1.4f, y, z});
assert_obj_missed({-21, 6, -13}, {x-3.f, y, z});
assert_obj_selected({-21, 6, -13}, {x, y-1.4f, z});
assert_obj_selected({-21, 6, -13}, {x, y+1.4f, z});
assert_obj_missed({-21, 6, -13}, {x, y+3.f, z});
assert_obj_selected({-21, 6, -13}, {x, y, z+1.4f});
assert_obj_selected({-21, 6, -13}, {x, y, z-1.4f});
assert_obj_missed({-21, 6, -13}, {x, y, z-3.f});
caomgr.clear();
}