-
Notifications
You must be signed in to change notification settings - Fork 11
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #106 from NuiCpp/devel
Added delocalized element utility.
- Loading branch information
Showing
5 changed files
with
240 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,127 @@ | ||
#pragma once | ||
|
||
#include <nui/frontend/element_renderer.hpp> | ||
#include <nui/frontend/dom/element.hpp> | ||
#include <nui/frontend/event_system/observed_value.hpp> | ||
#include <nui/frontend/elements/div.hpp> | ||
#include <nui/frontend/elements/nil.hpp> | ||
#include <nui/frontend/attributes/class.hpp> | ||
#include <nui/frontend/attributes/style.hpp> | ||
|
||
#include <memory> | ||
#include <optional> | ||
#include <string> | ||
|
||
namespace Nui | ||
{ | ||
/** | ||
* @brief A delocalized element can switch positions in several slots. | ||
*/ | ||
template <typename SlotId> | ||
class Delocalized | ||
{ | ||
public: | ||
Delocalized(Nui::ElementRenderer renderer = {}) | ||
: delocalizedElement_{[&renderer]() -> decltype(delocalizedElement_) { | ||
if (renderer) | ||
{ | ||
auto element = Dom::Element::makeElement(HtmlElement{"div", &RegularHtmlElementBridge}); | ||
element->replaceElement(renderer); | ||
return element; | ||
} | ||
else | ||
{ | ||
return {}; | ||
} | ||
}()} | ||
, slotId_{std::make_unique<Nui::Observed<SlotId>>()} | ||
{} | ||
Delocalized(Delocalized const&) = delete; | ||
Delocalized(Delocalized&&) = default; | ||
Delocalized& operator=(Delocalized const&) = delete; | ||
Delocalized& operator=(Delocalized&&) = default; | ||
|
||
std::shared_ptr<Dom::Element> element() | ||
{ | ||
return delocalizedElement_; | ||
} | ||
void element(Nui::ElementRenderer renderer) | ||
{ | ||
delocalizedElement_ = Dom::Element::makeElement(HtmlElement{"div", &RegularHtmlElementBridge}); | ||
delocalizedElement_->replaceElement(renderer); | ||
} | ||
bool hasElement() const | ||
{ | ||
return delocalizedElement_ != nullptr; | ||
} | ||
void initializeIfEmpty(Nui::ElementRenderer renderer) | ||
{ | ||
if (!hasElement()) | ||
element(std::move(renderer)); | ||
} | ||
|
||
void slot(SlotId slot) | ||
{ | ||
*slotId_ = slot; | ||
} | ||
SlotId slot() const | ||
{ | ||
return slotId_->value(); | ||
} | ||
|
||
template <typename SlotIdB> | ||
friend ElementRenderer delocalizedSlot( | ||
SlotIdB slot, | ||
Delocalized<SlotIdB>& delocalizedElement, | ||
std::vector<Attribute> wrapperAttributes, | ||
Nui::ElementRenderer alternative); | ||
|
||
private: | ||
std::shared_ptr<Dom::Element> delocalizedElement_; | ||
// Wrapped in a unique_ptr for pointer stability. | ||
std::unique_ptr<Nui::Observed<SlotId>> slotId_; | ||
}; | ||
|
||
template <typename SlotId> | ||
ElementRenderer delocalizedSlot( | ||
SlotId slot, | ||
Delocalized<SlotId>& delocalizedElement, | ||
std::vector<Attribute> wrapperAttributes = {}, | ||
ElementRenderer alternative = Elements::div{Attributes::style = "display: none"}()) | ||
{ | ||
using namespace Elements; | ||
using namespace Attributes; | ||
|
||
auto element = delocalizedElement.delocalizedElement_; | ||
|
||
return Elements::div{std::move(wrapperAttributes)}( | ||
observe(*delocalizedElement.slotId_), | ||
[element, slot, &observedId = *delocalizedElement.slotId_, alternative]() mutable { | ||
return [element, slot, &observedId, alternative]( | ||
Dom::Element& wrapper, Renderer const& gen) -> std::shared_ptr<Dom::Element> { | ||
if (element && slot == observedId.value()) | ||
{ | ||
wrapper.val().call<void>("appendChild", element->val()); | ||
return nil()(wrapper, gen); | ||
} | ||
else | ||
{ | ||
if (alternative) | ||
return alternative(wrapper, gen); | ||
else | ||
return nil()(wrapper, gen); | ||
} | ||
}; | ||
}); | ||
} | ||
|
||
inline ElementRenderer delocalizedSlot( | ||
char const* slot, | ||
Delocalized<std::string>& delocalizedElement, | ||
std::vector<Attribute> wrapperAttributes = {}, | ||
ElementRenderer alternative = Elements::div{Attributes::style = "display: none"}()) | ||
{ | ||
return delocalizedSlot<std::string>( | ||
std::string{slot}, delocalizedElement, std::move(wrapperAttributes), std::move(alternative)); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,108 @@ | ||
#pragma once | ||
|
||
#include <gtest/gtest.h> | ||
|
||
#include "common_test_fixture.hpp" | ||
#include "engine/global_object.hpp" | ||
#include "engine/document.hpp" | ||
|
||
#include <nui/frontend/elements.hpp> | ||
#include <nui/frontend/attributes.hpp> | ||
#include <nui/frontend/utility/delocalized.hpp> | ||
|
||
#include <vector> | ||
#include <string> | ||
|
||
namespace Nui::Tests | ||
{ | ||
using namespace Engine; | ||
using namespace std::string_literals; | ||
|
||
class TestDelocalized : public CommonTestFixture | ||
{ | ||
protected: | ||
auto bodyChildren() | ||
{ | ||
return Nui::val::global("document")["body"]["children"]; | ||
} | ||
|
||
public: | ||
Delocalized<std::string> delocalizedElement_; | ||
}; | ||
|
||
TEST_F(TestDelocalized, SingleSlotDelocalizedRender) | ||
{ | ||
using namespace Nui::Elements; | ||
using namespace Nui::Attributes; | ||
using Nui::Elements::div; | ||
using Nui::Elements::span; | ||
|
||
delocalizedElement_.initializeIfEmpty(span{}("Hello")); | ||
delocalizedElement_.slot("slot1"); | ||
|
||
render(body{}(delocalizedSlot("slot1", delocalizedElement_))); | ||
|
||
ASSERT_EQ(bodyChildren()["length"].as<long long>(), 1); | ||
EXPECT_EQ(bodyChildren()[0]["tagName"].as<std::string>(), "div"); | ||
ASSERT_EQ(bodyChildren()[0]["children"]["length"].as<long long>(), 1); | ||
EXPECT_EQ(bodyChildren()[0]["children"][0]["tagName"].as<std::string>(), "span"); | ||
EXPECT_EQ(bodyChildren()[0]["children"][0]["textContent"].as<std::string>(), "Hello"); | ||
} | ||
|
||
TEST_F(TestDelocalized, SlotShowsAlternativeWhenNotActive) | ||
{ | ||
using namespace Nui::Elements; | ||
using namespace Nui::Attributes; | ||
using Nui::Elements::div; | ||
using Nui::Elements::span; | ||
|
||
delocalizedElement_.initializeIfEmpty(span{}("Hello")); | ||
delocalizedElement_.slot("slot2"); | ||
|
||
render(body{}(delocalizedSlot("slot1", delocalizedElement_))); | ||
|
||
ASSERT_EQ(bodyChildren()["length"].as<long long>(), 1); | ||
EXPECT_EQ(bodyChildren()[0]["tagName"].as<std::string>(), "div"); | ||
ASSERT_EQ(bodyChildren()[0]["children"]["length"].as<long long>(), 1); | ||
EXPECT_EQ(bodyChildren()[0]["children"][0]["tagName"].as<std::string>(), "div"); | ||
EXPECT_EQ(bodyChildren()[0]["children"][0]["attributes"]["style"].as<std::string>(), "display: none"); | ||
} | ||
|
||
TEST_F(TestDelocalized, WrapperGetsAttributesAssignedWhenSet) | ||
{ | ||
using namespace Nui::Elements; | ||
using namespace Nui::Attributes; | ||
using Nui::Elements::div; | ||
using Nui::Elements::span; | ||
|
||
delocalizedElement_.initializeIfEmpty(span{}("Hello")); | ||
delocalizedElement_.slot("slot1"); | ||
|
||
render(body{}(delocalizedSlot("slot1", delocalizedElement_, std::vector<Attribute>{class_ = "wrapper"}))); | ||
|
||
ASSERT_EQ(bodyChildren()["length"].as<long long>(), 1); | ||
EXPECT_EQ(bodyChildren()[0]["tagName"].as<std::string>(), "div"); | ||
EXPECT_EQ(bodyChildren()[0]["attributes"]["class"].as<std::string>(), "wrapper"); | ||
ASSERT_EQ(bodyChildren()[0]["children"]["length"].as<long long>(), 1); | ||
EXPECT_EQ(bodyChildren()[0]["children"][0]["tagName"].as<std::string>(), "span"); | ||
} | ||
|
||
TEST_F(TestDelocalized, CanUseDifferentReplacementElement) | ||
{ | ||
using namespace Nui::Elements; | ||
using namespace Nui::Attributes; | ||
using Nui::Elements::div; | ||
using Nui::Elements::span; | ||
|
||
delocalizedElement_.initializeIfEmpty(span{}("Hello")); | ||
delocalizedElement_.slot("slotX"); | ||
|
||
render(body{}(delocalizedSlot("slot1", delocalizedElement_, {}, div{}("Hello")))); | ||
|
||
ASSERT_EQ(bodyChildren()["length"].as<long long>(), 1); | ||
EXPECT_EQ(bodyChildren()[0]["tagName"].as<std::string>(), "div"); | ||
ASSERT_EQ(bodyChildren()[0]["children"]["length"].as<long long>(), 1); | ||
EXPECT_EQ(bodyChildren()[0]["children"][0]["tagName"].as<std::string>(), "div"); | ||
EXPECT_EQ(bodyChildren()[0]["children"][0]["textContent"].as<std::string>(), "Hello"); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters