From 246d2e904f47314460f02b6cacf15d018910767a Mon Sep 17 00:00:00 2001 From: altalk23 <45172705+altalk23@users.noreply.github.com> Date: Sun, 5 Jan 2025 23:40:04 +0300 Subject: [PATCH 1/4] Removed CCScene create hook --- mod.json | 2 +- src/nodes/CCScene.h | 65 +++++++++++++++++++++++++++------------------ 2 files changed, 40 insertions(+), 27 deletions(-) diff --git a/mod.json b/mod.json index ec822d5..6de2669 100644 --- a/mod.json +++ b/mod.json @@ -1,5 +1,5 @@ { - "geode": "4.0.1", + "geode": "4.1.1", "gd": { "win": "2.2074", "android": "2.2074", diff --git a/src/nodes/CCScene.h b/src/nodes/CCScene.h index 6f6b383..73ebaad 100644 --- a/src/nodes/CCScene.h +++ b/src/nodes/CCScene.h @@ -8,37 +8,50 @@ using namespace geode::prelude; -class $modify(MyCCScene, CCScene) { - - static void onModify(auto& self) { - HOOK_LATEST("cocos2d::CCScene::create"); +class SceneHandler : public CCObject { +public: + static SceneHandler* create() { + auto ret = new SceneHandler(); + ret->autorelease(); + return ret; } - struct Fields { - int m_currentCount = 0; - bool m_isMenuLayer = false; - }; + CCScene* m_currentScene = nullptr; + + // IMPORTANT: might contain dangling pointers, + // do not use for anything other than checking + std::unordered_set m_handledNodes; - static CCScene* create() { - auto ret = CCScene::create(); - if (UIModding::get()->doModify) { - ret->schedule(schedule_selector(MyCCScene::checkForUpdates)); + void checkForUpdates(CCScene* scene) { + if (Callbacks::get()->m_ignoreUICheck) return; + + if (scene != m_currentScene) { + // we're in a new scene, clear the handled nodes + m_handledNodes.clear(); + m_currentScene = scene; + } + + for (auto node : CCArrayExt(scene->getChildren())) { + if (m_handledNodes.find(node) != m_handledNodes.end()) continue; + m_handledNodes.insert(node); + + if (node->getID() == "MenuLayer") continue; // hardcoded for now + + UIModding::get()->doUICheck(node); } - return ret; } - void checkForUpdates(float dt) { - if (this->getChildrenCount() != m_fields->m_currentCount && (this->getChildrenCount() != 1 || m_fields->m_currentCount == 0)) { - int idx = 0; - - for (CCNode* node : CCArrayExt(this->getChildren())) { - idx++; - if (node->getID() == "MenuLayer" || Callbacks::get()->m_ignoreUICheck) continue; - if (idx > m_fields->m_currentCount) { - UIModding::get()->doUICheck(node); - } - } + void update(float dt) { + auto scene = CCDirector::sharedDirector()->getRunningScene(); + + if (scene && UIModding::get()->doModify) { + this->checkForUpdates(scene); } - m_fields->m_currentCount = this->getChildrenCount(); } -}; \ No newline at end of file +}; + +$execute { + Loader::get()->queueInMainThread([]{ + CCScheduler::get()->scheduleUpdateForTarget(SceneHandler::create(), 0, false); + }); +} \ No newline at end of file From 9ec015d513153ae0ea2aef02276aaecb74b1e0fe Mon Sep 17 00:00:00 2001 From: altalk23 <45172705+altalk23@users.noreply.github.com> Date: Mon, 6 Jan 2025 01:12:38 +0300 Subject: [PATCH 2/4] Removed CCObject::autorelease hook --- src/NodeModding.cpp | 72 +++++++++++++++++++++++++++++++++++---------- 1 file changed, 56 insertions(+), 16 deletions(-) diff --git a/src/NodeModding.cpp b/src/NodeModding.cpp index 6a38d0a..b3e5550 100644 --- a/src/NodeModding.cpp +++ b/src/NodeModding.cpp @@ -7,25 +7,65 @@ using namespace geode::prelude; -class $modify(MyCCObject, CCObject) { +class CCAutoreleasePoolHack : public CCObject { +public: + CCArray* m_pManagedObjectArray; +}; - static void onModify(auto& self) { - HOOK_LATEST("cocos2d::CCObject::autorelease"); +class CCPoolManagerHack { +public: + CCArray* m_pReleasePoolStack; + CCAutoreleasePool* m_pCurReleasePool; +}; + +class NodeModding : public CCObject { +public: + static NodeModding* create() { + auto ret = new NodeModding(); + ret->autorelease(); + return ret; + } + + static NodeModding* get() { + static NodeModding* instance = nullptr; + if (!instance) { + instance = NodeModding::create(); + } + return instance; } - CCObject* autorelease() { - if (!UIModding::get()->finishedLoad || !UIModding::get()->doModify || Callbacks::get()->m_ignoreUICheck) - return CCObject::autorelease(); - - if (CCNode* node = typeinfo_cast(this)) { - node->retain(); - std::string className = Utils::getNodeName(this); - queueInMainThread([=] { - UIModding::get()->doUICheckForType(className, node); - node->release(); - }); + void handleCurrentNode(CCNode* node) { + std::string className = Utils::getNodeName(node); + UIModding::get()->doUICheckForType(className, node); + } + + void handleArray(CCArray* array) { + auto scene = CCDirector::sharedDirector()->getRunningScene(); + + if (scene && UIModding::get()->doModify && UIModding::get()->finishedLoad && !Callbacks::get()->m_ignoreUICheck) { + + for (auto object : CCArrayExt(array)) { + if (auto node = typeinfo_cast(object)) { + handleCurrentNode(node); + } + } } - - return CCObject::autorelease(); + } +}; + +#include + +class $modify(CCPoolManager) { + void pop() { + auto poolManager = reinterpret_cast(this); + + if (!poolManager || !poolManager->m_pCurReleasePool) return; + auto pool = reinterpret_cast(poolManager->m_pCurReleasePool); + + if (!pool->m_pManagedObjectArray) return; + + NodeModding::get()->handleArray(pool->m_pManagedObjectArray); + + return CCPoolManager::pop(); } }; \ No newline at end of file From 38490c5abd1e85564165c86b1cbdb84aea23010d Mon Sep 17 00:00:00 2001 From: altalk23 <45172705+altalk23@users.noreply.github.com> Date: Mon, 6 Jan 2025 01:13:20 +0300 Subject: [PATCH 3/4] cleanup the pool manager hook --- src/NodeModding.cpp | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/src/NodeModding.cpp b/src/NodeModding.cpp index b3e5550..dc9840a 100644 --- a/src/NodeModding.cpp +++ b/src/NodeModding.cpp @@ -59,12 +59,13 @@ class $modify(CCPoolManager) { void pop() { auto poolManager = reinterpret_cast(this); - if (!poolManager || !poolManager->m_pCurReleasePool) return; - auto pool = reinterpret_cast(poolManager->m_pCurReleasePool); + if (poolManager && poolManager->m_pCurReleasePool) { + auto pool = reinterpret_cast(poolManager->m_pCurReleasePool); - if (!pool->m_pManagedObjectArray) return; - - NodeModding::get()->handleArray(pool->m_pManagedObjectArray); + if (pool->m_pManagedObjectArray) { + NodeModding::get()->handleArray(pool->m_pManagedObjectArray); + } + } return CCPoolManager::pop(); } From abe0281d7f993505044886713e62ba6df946972d Mon Sep 17 00:00:00 2001 From: altalk23 <45172705+altalk23@users.noreply.github.com> Date: Mon, 6 Jan 2025 01:22:53 +0300 Subject: [PATCH 4/4] Make the SceneHandler max priority (last) --- src/nodes/CCScene.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/nodes/CCScene.h b/src/nodes/CCScene.h index 73ebaad..5758a9f 100644 --- a/src/nodes/CCScene.h +++ b/src/nodes/CCScene.h @@ -52,6 +52,6 @@ class SceneHandler : public CCObject { $execute { Loader::get()->queueInMainThread([]{ - CCScheduler::get()->scheduleUpdateForTarget(SceneHandler::create(), 0, false); + CCScheduler::get()->scheduleUpdateForTarget(SceneHandler::create(), INT_MAX, false); }); } \ No newline at end of file