geode/loader/src/hooks/GeodeNodeMetadata.cpp

253 lines
7.7 KiB
C++
Raw Normal View History

2022-10-30 15:07:49 -04:00
#include <Geode/modify/Field.hpp>
2022-10-13 05:56:23 -04:00
#include <Geode/utils/cocos.hpp>
2022-10-13 04:31:23 -04:00
#include <Geode/modify/Field.hpp>
2022-10-17 09:08:12 -04:00
#include <Geode/modify/CCNode.hpp>
2022-10-30 14:56:36 -04:00
#include <cocos2d.h>
using namespace geode::prelude;
using namespace geode::modifier;
#pragma warning(push)
2022-10-30 14:56:36 -04:00
#pragma warning(disable : 4273)
constexpr auto METADATA_TAG = 0xB324ABC;
struct ProxyCCNode;
class GeodeNodeMetadata final : public cocos2d::CCObject {
private:
FieldContainer* m_fieldContainer;
Ref<cocos2d::CCObject> m_userObject;
std::string m_id = "";
2023-03-19 09:02:49 -04:00
Ref<Layout> m_layout = nullptr;
Ref<LayoutOptions> m_layoutOptions = nullptr;
std::unordered_map<std::string, matjson::Value> m_attributes;
std::unordered_set<std::unique_ptr<EventListenerProtocol>> m_eventListeners;
std::unordered_map<std::string, std::unique_ptr<EventListenerProtocol>> m_idEventListeners;
friend class ProxyCCNode;
friend class cocos2d::CCNode;
GeodeNodeMetadata() : m_fieldContainer(new FieldContainer()) {}
2022-10-30 14:56:36 -04:00
virtual ~GeodeNodeMetadata() {
delete m_fieldContainer;
}
public:
static GeodeNodeMetadata* set(CCNode* target) {
if (!target) return nullptr;
2022-10-30 14:56:36 -04:00
auto old = target->m_pUserObject;
2022-10-30 14:56:36 -04:00
// faster than dynamic_cast, technically can
// but extremely unlikely to fail
if (old && old->getTag() == METADATA_TAG) {
return static_cast<GeodeNodeMetadata*>(old);
}
auto meta = new GeodeNodeMetadata();
meta->autorelease();
meta->setTag(METADATA_TAG);
// set user object
target->m_pUserObject = meta;
meta->retain();
if (old) {
meta->m_userObject = old;
// the old user object has been retained by CCNode
old->release();
}
return meta;
}
FieldContainer* getFieldContainer() {
return m_fieldContainer;
}
};
// proxy forwards
2022-10-30 14:56:36 -04:00
#include <Geode/modify/CCNode.hpp>
struct ProxyCCNode : Modify<ProxyCCNode, CCNode> {
virtual CCObject* getUserObject() {
if (typeinfo_cast<CCNode*>(this)) {
return GeodeNodeMetadata::set(this)->m_userObject;
}
// apparently this function is the same as
// CCDirector::getNextScene so yeah
return m_pUserObject;
}
virtual void setUserObject(CCObject* obj) {
if (typeinfo_cast<CCNode*>(this)) {
GeodeNodeMetadata::set(this)->m_userObject = obj;
}
m_pUserObject = obj;
}
};
static inline std::unordered_map<std::string, size_t> s_nextIndex;
size_t modifier::getFieldIndexForClass(char const* name) {
return s_nextIndex[name]++;
2022-12-03 08:51:46 -05:00
}
2022-10-30 14:56:36 -04:00
// not const because might modify contents
FieldContainer* CCNode::getFieldContainer() {
return GeodeNodeMetadata::set(this)->getFieldContainer();
}
std::string CCNode::getID() {
return GeodeNodeMetadata::set(this)->m_id;
}
void CCNode::setID(std::string const& id) {
GeodeNodeMetadata::set(this)->m_id = id;
}
CCNode* CCNode::getChildByID(std::string const& id) {
2024-01-09 08:54:29 -05:00
for (auto child : CCArrayExt<CCNode*>(m_pChildren)) {
if (child->getID() == id) {
return child;
}
}
return nullptr;
}
CCNode* CCNode::getChildByIDRecursive(std::string const& id) {
if (auto child = this->getChildByID(id)) {
return child;
}
2024-01-09 08:54:29 -05:00
for (auto child : CCArrayExt<CCNode*>(m_pChildren)) {
if ((child = child->getChildByIDRecursive(id))) {
return child;
}
}
return nullptr;
}
2023-03-11 03:19:26 -05:00
void CCNode::removeChildByID(std::string const& id) {
if (auto child = this->getChildByID(id)) {
this->removeChild(child);
}
}
2023-02-02 10:08:13 -05:00
void CCNode::setLayout(Layout* layout, bool apply, bool respectAnchor) {
if (respectAnchor && this->isIgnoreAnchorPointForPosition()) {
2024-01-09 08:54:29 -05:00
for (auto child : CCArrayExt<CCNode*>(m_pChildren)) {
2023-02-02 10:08:13 -05:00
child->setPosition(child->getPosition() + this->getScaledContentSize());
}
this->ignoreAnchorPointForPosition(false);
}
2023-03-19 09:02:49 -04:00
GeodeNodeMetadata::set(this)->m_layout = layout;
2022-10-10 13:58:47 -04:00
if (apply) {
this->updateLayout();
}
}
Layout* CCNode::getLayout() {
2023-03-19 09:02:49 -04:00
return GeodeNodeMetadata::set(this)->m_layout.data();
2022-10-10 13:58:47 -04:00
}
2023-02-06 14:36:08 -05:00
void CCNode::setLayoutOptions(LayoutOptions* options, bool apply) {
GeodeNodeMetadata::set(this)->m_layoutOptions = options;
2023-02-06 14:36:08 -05:00
if (apply && m_pParent) {
m_pParent->updateLayout();
}
}
LayoutOptions* CCNode::getLayoutOptions() {
return GeodeNodeMetadata::set(this)->m_layoutOptions.data();
2023-02-06 14:36:08 -05:00
}
void CCNode::updateLayout(bool updateChildOrder) {
if (updateChildOrder) {
this->sortAllChildren();
}
2023-03-19 09:02:49 -04:00
if (auto layout = GeodeNodeMetadata::set(this)->m_layout.data()) {
2023-02-02 10:08:13 -05:00
layout->apply(this);
2022-10-10 13:58:47 -04:00
}
}
AttributeSetEvent::AttributeSetEvent(CCNode* node, std::string const& id, matjson::Value& value)
2023-03-29 09:05:20 -04:00
: node(node), id(id), value(value) {}
ListenerResult AttributeSetFilter::handle(MiniFunction<Callback> fn, AttributeSetEvent* event) {
if (event->id == m_targetID) {
fn(event);
}
return ListenerResult::Propagate;
}
AttributeSetFilter::AttributeSetFilter(std::string const& id) : m_targetID(id) {}
void CCNode::setAttribute(std::string const& attr, matjson::Value const& value) {
2023-03-29 09:05:20 -04:00
auto meta = GeodeNodeMetadata::set(this);
meta->m_attributes[attr] = value;
AttributeSetEvent(this, attr, meta->m_attributes.at(attr)).post();
2022-10-25 15:57:40 -04:00
}
std::optional<matjson::Value> CCNode::getAttributeInternal(std::string const& attr) {
2022-10-25 15:57:40 -04:00
auto meta = GeodeNodeMetadata::set(this);
if (meta->m_attributes.count(attr)) {
return meta->m_attributes.at(attr);
}
return std::nullopt;
}
void CCNode::addEventListenerInternal(std::string const& id, EventListenerProtocol* listener) {
auto meta = GeodeNodeMetadata::set(this);
if (id.size()) {
if (meta->m_idEventListeners.contains(id)) {
meta->m_idEventListeners.at(id).reset(listener);
}
else {
meta->m_idEventListeners.emplace(id, listener);
}
}
else {
std::erase_if(meta->m_eventListeners, [=](auto& l) {
return l.get() == listener;
});
meta->m_eventListeners.emplace(listener);
}
2023-03-25 12:22:34 -04:00
}
void CCNode::removeEventListener(EventListenerProtocol* listener) {
auto meta = GeodeNodeMetadata::set(this);
std::erase_if(meta->m_eventListeners, [=](auto& l) {
return l.get() == listener;
});
std::erase_if(meta->m_idEventListeners, [=](auto& l) {
return l.second.get() == listener;
});
}
void CCNode::removeEventListener(std::string const& id) {
GeodeNodeMetadata::set(this)->m_idEventListeners.erase(id);
}
EventListenerProtocol* CCNode::getEventListener(std::string const& id) {
auto meta = GeodeNodeMetadata::set(this);
if (meta->m_idEventListeners.contains(id)) {
return meta->m_idEventListeners.at(id).get();
}
return nullptr;
2023-03-23 15:42:01 -04:00
}
2023-04-05 12:26:46 -04:00
size_t CCNode::getEventListenerCount() {
return GeodeNodeMetadata::set(this)->m_idEventListeners.size() +
GeodeNodeMetadata::set(this)->m_eventListeners.size();
}
void CCNode::addChildAtPosition(CCNode* child, Anchor anchor, CCPoint const& offset, bool useAnchorLayout) {
2024-01-31 16:11:43 -05:00
auto layout = this->getLayout();
if (!layout && useAnchorLayout) {
2024-01-31 16:11:43 -05:00
this->setLayout(AnchorLayout::create());
}
child->setPosition(AnchorLayout::getAnchoredPosition(this, anchor, offset));
if (useAnchorLayout) {
child->setLayoutOptions(AnchorLayoutOptions::create()->setAnchor(anchor)->setOffset(offset));
}
2024-01-31 16:11:43 -05:00
this->addChild(child);
}
#pragma warning(pop)