Summary: Adds `LayoutPassStart` and `LayoutPassEnd` events. The existing `NodeLayout` event in isolation is not as useful as it could be. Having events that mark start and end of a layout pass are a useful addition. Differential Revision: D15305467 fbshipit-source-id: 14af6f65e698fb1e3112eb2ffd87a74d31df4840
146 lines
3.5 KiB
C++
146 lines
3.5 KiB
C++
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the LICENSE
|
|
* file in the root directory of this source tree.
|
|
*/
|
|
#include <gtest/gtest.h>
|
|
#include <yoga/Yoga.h>
|
|
#include <yoga/event/event.h>
|
|
#include <yoga/YGNode.h>
|
|
#include <yoga/testutil/testutil.h>
|
|
|
|
#include <algorithm>
|
|
#include <functional>
|
|
#include <memory>
|
|
#include <vector>
|
|
|
|
namespace facebook {
|
|
namespace yoga {
|
|
namespace test {
|
|
|
|
struct EventArgs {
|
|
const YGNode* node;
|
|
Event::Type type;
|
|
std::unique_ptr<void, std::function<void(void*)>> dataPtr;
|
|
|
|
template <Event::Type E>
|
|
const Event::TypedData<E>& data() {
|
|
return *static_cast<Event::TypedData<E>*>(dataPtr.get());
|
|
}
|
|
};
|
|
|
|
class EventTest : public ::testing::Test {
|
|
ScopedEventSubscription subscription = {&EventTest::listen};
|
|
static void listen(const YGNode&, Event::Type, Event::Data);
|
|
|
|
public:
|
|
static std::vector<EventArgs> events;
|
|
static EventArgs& lastEvent() { return events.back(); }
|
|
void TearDown() override;
|
|
};
|
|
|
|
TEST_F(EventTest, new_node_has_event) {
|
|
auto c = YGConfigGetDefault();
|
|
auto n = YGNodeNew();
|
|
|
|
ASSERT_EQ(lastEvent().node, n);
|
|
ASSERT_EQ(lastEvent().type, Event::NodeAllocation);
|
|
ASSERT_EQ(lastEvent().data<Event::NodeAllocation>().config, c);
|
|
|
|
YGNodeFree(n);
|
|
}
|
|
|
|
TEST_F(EventTest, new_node_with_config_event) {
|
|
auto c = YGConfigNew();
|
|
auto n = YGNodeNewWithConfig(c);
|
|
|
|
ASSERT_EQ(lastEvent().node, n);
|
|
ASSERT_EQ(lastEvent().type, Event::NodeAllocation);
|
|
ASSERT_EQ(lastEvent().data<Event::NodeAllocation>().config, c);
|
|
|
|
YGNodeFree(n);
|
|
YGConfigFree(c);
|
|
}
|
|
|
|
TEST_F(EventTest, clone_node_event) {
|
|
auto c = YGConfigNew();
|
|
auto n = YGNodeNewWithConfig(c);
|
|
auto clone = YGNodeClone(n);
|
|
|
|
ASSERT_EQ(lastEvent().node, clone);
|
|
ASSERT_EQ(lastEvent().type, Event::NodeAllocation);
|
|
ASSERT_EQ(lastEvent().data<Event::NodeAllocation>().config, c);
|
|
|
|
YGNodeFree(n);
|
|
YGNodeFree(clone);
|
|
YGConfigFree(c);
|
|
}
|
|
|
|
TEST_F(EventTest, free_node_event) {
|
|
auto c = YGConfigNew();
|
|
auto n = YGNodeNewWithConfig(c);
|
|
YGNodeFree(n);
|
|
|
|
ASSERT_EQ(lastEvent().node, n);
|
|
ASSERT_EQ(lastEvent().type, Event::NodeDeallocation);
|
|
ASSERT_EQ(lastEvent().data<Event::NodeDeallocation>().config, c);
|
|
|
|
YGConfigFree(c);
|
|
}
|
|
|
|
TEST_F(EventTest, layout_events) {
|
|
auto root = YGNodeNew();
|
|
auto child = YGNodeNew();
|
|
YGNodeInsertChild(root, child, 0);
|
|
|
|
YGNodeCalculateLayout(root, 123, 456, YGDirectionLTR);
|
|
|
|
ASSERT_EQ(events[2].node, root);
|
|
ASSERT_EQ(events[2].type, Event::NodeLayout);
|
|
|
|
ASSERT_EQ(events[3].node, child);
|
|
ASSERT_EQ(events[3].type, Event::NodeLayout);
|
|
|
|
YGNodeFreeRecursive(root);
|
|
}
|
|
|
|
namespace {
|
|
|
|
template <Event::Type E>
|
|
EventArgs createArgs(const YGNode& node, const Event::Data& data) {
|
|
using Data = Event::TypedData<E>;
|
|
auto deleteData = [](void* x) { delete static_cast<Data*>(x); };
|
|
return {&node, E, {new Data{data.get<E>()}, deleteData}};
|
|
}
|
|
|
|
} // namespace
|
|
|
|
void EventTest::listen(const YGNode& node, Event::Type type, Event::Data data) {
|
|
switch (type) {
|
|
case Event::NodeAllocation:
|
|
events.push_back(createArgs<Event::NodeAllocation>(node, data));
|
|
break;
|
|
case Event::NodeDeallocation:
|
|
events.push_back(createArgs<Event::NodeDeallocation>(node, data));
|
|
break;
|
|
case Event::NodeLayout:
|
|
events.push_back(createArgs<Event::NodeLayout>(node, data));
|
|
break;
|
|
case Event::LayoutPassStart:
|
|
break;
|
|
case Event::LayoutPassEnd:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void EventTest::TearDown() {
|
|
events.clear();
|
|
}
|
|
|
|
std::vector<EventArgs> EventTest::events{};
|
|
|
|
} // namespace test
|
|
} // namespace yoga
|
|
} // namespace facebook
|