mirror of
https://github.com/smallmain/cocos-enhance-kit.git
synced 2025-01-15 07:21:07 +00:00
489 lines
14 KiB
C++
489 lines
14 KiB
C++
/****************************************************************************
|
|
Copyright (c) 2018 Xiamen Yaji Software Co., Ltd.
|
|
|
|
http://www.cocos2d-x.org
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
THE SOFTWARE.
|
|
****************************************************************************/
|
|
|
|
#include "EventDispatcher.h"
|
|
|
|
#include "cocos/scripting/js-bindings/jswrapper/SeApi.h"
|
|
#include "cocos/scripting/js-bindings/manual/jsb_global.h"
|
|
#include "cocos/scripting/js-bindings/event/CustomEventTypes.h"
|
|
|
|
namespace {
|
|
se::Value _tickVal;
|
|
std::vector<se::Object*> _jsTouchObjPool;
|
|
se::Object* _jsTouchObjArray = nullptr;
|
|
se::Object* _jsMouseEventObj = nullptr;
|
|
se::Object* _jsKeyboardEventObj = nullptr;
|
|
se::Object* _jsResizeEventObj = nullptr;
|
|
se::Object* _jsOrientationEventObj = nullptr;
|
|
bool _inited = false;
|
|
}
|
|
|
|
namespace cocos2d
|
|
{
|
|
std::unordered_map<std::string, EventDispatcher::Node*> EventDispatcher::_listeners;
|
|
|
|
void EventDispatcher::init()
|
|
{
|
|
_inited = true;
|
|
se::ScriptEngine::getInstance()->addBeforeCleanupHook([](){
|
|
EventDispatcher::destroy();
|
|
});
|
|
}
|
|
|
|
void EventDispatcher::destroy()
|
|
{
|
|
for (auto touchObj : _jsTouchObjPool)
|
|
{
|
|
touchObj->unroot();
|
|
touchObj->decRef();
|
|
}
|
|
_jsTouchObjPool.clear();
|
|
|
|
if (_jsTouchObjArray != nullptr)
|
|
{
|
|
_jsTouchObjArray->unroot();
|
|
_jsTouchObjArray->decRef();
|
|
_jsTouchObjArray = nullptr;
|
|
}
|
|
|
|
if (_jsMouseEventObj != nullptr)
|
|
{
|
|
_jsMouseEventObj->unroot();
|
|
_jsMouseEventObj->decRef();
|
|
_jsMouseEventObj = nullptr;
|
|
}
|
|
|
|
if (_jsKeyboardEventObj != nullptr)
|
|
{
|
|
_jsKeyboardEventObj->unroot();
|
|
_jsKeyboardEventObj->decRef();
|
|
_jsKeyboardEventObj = nullptr;
|
|
}
|
|
|
|
if (_jsResizeEventObj != nullptr)
|
|
{
|
|
_jsResizeEventObj->unroot();
|
|
_jsResizeEventObj->decRef();
|
|
_jsResizeEventObj = nullptr;
|
|
}
|
|
|
|
if (_jsOrientationEventObj != nullptr)
|
|
{
|
|
_jsOrientationEventObj->unroot();
|
|
_jsOrientationEventObj->decRef();
|
|
_jsOrientationEventObj = nullptr;
|
|
}
|
|
|
|
_inited = false;
|
|
_tickVal.setUndefined();
|
|
}
|
|
|
|
void EventDispatcher::dispatchTouchEvent(const struct TouchEvent& touchEvent)
|
|
{
|
|
if (!se::ScriptEngine::getInstance()->isValid())
|
|
return;
|
|
|
|
se::AutoHandleScope scope;
|
|
assert(_inited);
|
|
|
|
if (_jsTouchObjArray == nullptr)
|
|
{
|
|
_jsTouchObjArray = se::Object::createArrayObject(0);
|
|
_jsTouchObjArray->root();
|
|
}
|
|
|
|
_jsTouchObjArray->setProperty("length", se::Value(touchEvent.touches.size()));
|
|
|
|
while (_jsTouchObjPool.size() < touchEvent.touches.size())
|
|
{
|
|
se::Object* touchObj = se::Object::createPlainObject();
|
|
touchObj->root();
|
|
_jsTouchObjPool.push_back(touchObj);
|
|
}
|
|
|
|
uint32_t touchIndex = 0;
|
|
int poolIndex = 0;
|
|
for (const auto& touch : touchEvent.touches)
|
|
{
|
|
se::Object* jsTouch = _jsTouchObjPool.at(poolIndex++);
|
|
jsTouch->setProperty("identifier", se::Value(touch.index));
|
|
jsTouch->setProperty("clientX", se::Value(touch.x));
|
|
jsTouch->setProperty("clientY", se::Value(touch.y));
|
|
jsTouch->setProperty("pageX", se::Value(touch.x));
|
|
jsTouch->setProperty("pageY", se::Value(touch.y));
|
|
|
|
_jsTouchObjArray->setArrayElement(touchIndex, se::Value(jsTouch));
|
|
++touchIndex;
|
|
}
|
|
|
|
const char* eventName = nullptr;
|
|
switch (touchEvent.type) {
|
|
case TouchEvent::Type::BEGAN:
|
|
eventName = "onTouchStart";
|
|
break;
|
|
case TouchEvent::Type::MOVED:
|
|
eventName = "onTouchMove";
|
|
break;
|
|
case TouchEvent::Type::ENDED:
|
|
eventName = "onTouchEnd";
|
|
break;
|
|
case TouchEvent::Type::CANCELLED:
|
|
eventName = "onTouchCancel";
|
|
break;
|
|
default:
|
|
assert(false);
|
|
break;
|
|
}
|
|
|
|
se::Value callbackVal;
|
|
if (__jsbObj->getProperty(eventName, &callbackVal) && !callbackVal.isNullOrUndefined())
|
|
{
|
|
se::ValueArray args;
|
|
args.push_back(se::Value(_jsTouchObjArray));
|
|
callbackVal.toObject()->call(args, nullptr);
|
|
}
|
|
}
|
|
|
|
void EventDispatcher::dispatchMouseEvent(const struct MouseEvent& mouseEvent)
|
|
{
|
|
if (!se::ScriptEngine::getInstance()->isValid())
|
|
return;
|
|
|
|
se::AutoHandleScope scope;
|
|
assert(_inited);
|
|
|
|
if (_jsMouseEventObj == nullptr)
|
|
{
|
|
_jsMouseEventObj = se::Object::createPlainObject();
|
|
_jsMouseEventObj->root();
|
|
}
|
|
|
|
const auto& xVal = se::Value(mouseEvent.x);
|
|
const auto& yVal = se::Value(mouseEvent.y);
|
|
const MouseEvent::Type type = mouseEvent.type;
|
|
|
|
if (type == MouseEvent::Type::WHEEL)
|
|
{
|
|
_jsMouseEventObj->setProperty("wheelDeltaX", xVal);
|
|
_jsMouseEventObj->setProperty("wheelDeltaY", yVal);
|
|
}
|
|
else
|
|
{
|
|
if (type == MouseEvent::Type::DOWN || type == MouseEvent::Type::UP)
|
|
{
|
|
_jsMouseEventObj->setProperty("button", se::Value(mouseEvent.button));
|
|
}
|
|
_jsMouseEventObj->setProperty("x", xVal);
|
|
_jsMouseEventObj->setProperty("y", yVal);
|
|
}
|
|
|
|
const char* eventName = nullptr;
|
|
switch (type) {
|
|
case MouseEvent::Type::DOWN:
|
|
eventName = "onMouseDown";
|
|
break;
|
|
case MouseEvent::Type::MOVE:
|
|
eventName = "onMouseMove";
|
|
break;
|
|
case MouseEvent::Type::UP:
|
|
eventName = "onMouseUp";
|
|
break;
|
|
case MouseEvent::Type::WHEEL:
|
|
eventName = "onMouseWheel";
|
|
break;
|
|
default:
|
|
assert(false);
|
|
break;
|
|
}
|
|
|
|
se::Value callbackVal;
|
|
if (__jsbObj->getProperty(eventName, &callbackVal) && !callbackVal.isNullOrUndefined())
|
|
{
|
|
se::ValueArray args;
|
|
args.push_back(se::Value(_jsMouseEventObj));
|
|
callbackVal.toObject()->call(args, nullptr);
|
|
}
|
|
}
|
|
|
|
void EventDispatcher::dispatchKeyboardEvent(const struct KeyboardEvent& keyboardEvent)
|
|
{
|
|
if (!se::ScriptEngine::getInstance()->isValid())
|
|
return;
|
|
|
|
|
|
se::AutoHandleScope scope;
|
|
assert(_inited);
|
|
|
|
if (_jsKeyboardEventObj == nullptr)
|
|
{
|
|
_jsKeyboardEventObj = se::Object::createPlainObject();
|
|
_jsKeyboardEventObj->root();
|
|
}
|
|
|
|
const char* eventName = nullptr;
|
|
switch (keyboardEvent.action) {
|
|
case KeyboardEvent::Action::PRESS:
|
|
case KeyboardEvent::Action::REPEAT:
|
|
eventName = "onKeyDown";
|
|
break;
|
|
case KeyboardEvent::Action::RELEASE:
|
|
eventName = "onKeyUp";
|
|
break;
|
|
default:
|
|
assert(false);
|
|
break;
|
|
}
|
|
|
|
se::Value callbackVal;
|
|
if (__jsbObj->getProperty(eventName, &callbackVal) && !callbackVal.isNullOrUndefined())
|
|
{
|
|
_jsKeyboardEventObj->setProperty("altKey", se::Value(keyboardEvent.altKeyActive));
|
|
_jsKeyboardEventObj->setProperty("ctrlKey", se::Value(keyboardEvent.ctrlKeyActive));
|
|
_jsKeyboardEventObj->setProperty("metaKey", se::Value(keyboardEvent.metaKeyActive));
|
|
_jsKeyboardEventObj->setProperty("shiftKey", se::Value(keyboardEvent.shiftKeyActive));
|
|
_jsKeyboardEventObj->setProperty("repeat", se::Value(keyboardEvent.action == KeyboardEvent::Action::REPEAT));
|
|
_jsKeyboardEventObj->setProperty("keyCode", se::Value(keyboardEvent.key));
|
|
|
|
se::ValueArray args;
|
|
args.push_back(se::Value(_jsKeyboardEventObj));
|
|
callbackVal.toObject()->call(args, nullptr);
|
|
}
|
|
}
|
|
|
|
void EventDispatcher::dispatchTickEvent(float dt)
|
|
{
|
|
if (!se::ScriptEngine::getInstance()->isValid())
|
|
return;
|
|
|
|
se::AutoHandleScope scope;
|
|
if (_tickVal.isUndefined())
|
|
{
|
|
se::ScriptEngine::getInstance()->getGlobalObject()->getProperty("gameTick", &_tickVal);
|
|
}
|
|
|
|
static std::chrono::steady_clock::time_point prevTime;
|
|
prevTime = std::chrono::steady_clock::now();
|
|
|
|
se::ValueArray args;
|
|
long long microSeconds = std::chrono::duration_cast<std::chrono::microseconds>(prevTime - se::ScriptEngine::getInstance()->getStartTime()).count();
|
|
args.push_back(se::Value((double)(microSeconds * 0.001)));
|
|
_tickVal.toObject()->call(args, nullptr);
|
|
}
|
|
|
|
void EventDispatcher::dispatchResizeEvent(int width, int height)
|
|
{
|
|
if (!se::ScriptEngine::getInstance()->isValid())
|
|
return;
|
|
|
|
se::AutoHandleScope scope;
|
|
assert(_inited);
|
|
|
|
if (_jsResizeEventObj == nullptr)
|
|
{
|
|
_jsResizeEventObj = se::Object::createPlainObject();
|
|
_jsResizeEventObj->root();
|
|
}
|
|
|
|
se::Value func;
|
|
__jsbObj->getProperty("onResize", &func);
|
|
if (func.isObject() && func.toObject()->isFunction())
|
|
{
|
|
_jsResizeEventObj->setProperty("width", se::Value(width));
|
|
_jsResizeEventObj->setProperty("height", se::Value(height));
|
|
|
|
se::ValueArray args;
|
|
args.push_back(se::Value(_jsResizeEventObj));
|
|
func.toObject()->call(args, nullptr);
|
|
}
|
|
}
|
|
|
|
void EventDispatcher::dispatchOrientationChangeEvent(int rotation)
|
|
{
|
|
if (!se::ScriptEngine::getInstance()->isValid())
|
|
return;
|
|
|
|
se::AutoHandleScope scope;
|
|
assert(_inited);
|
|
|
|
if (_jsOrientationEventObj == nullptr)
|
|
{
|
|
_jsOrientationEventObj = se::Object::createPlainObject();
|
|
_jsOrientationEventObj->root();
|
|
}
|
|
|
|
se::Value func;
|
|
__jsbObj->getProperty("onOrientationChanged", &func);
|
|
if (func.isObject() && func.toObject()->isFunction())
|
|
{
|
|
_jsOrientationEventObj->setProperty("rotation", se::Value(rotation));
|
|
|
|
se::ValueArray args;
|
|
args.push_back(se::Value(_jsOrientationEventObj));
|
|
func.toObject()->call(args, nullptr);
|
|
}
|
|
}
|
|
|
|
static void dispatchEnterBackgroundOrForegroundEvent(const char* funcName)
|
|
{
|
|
if (!se::ScriptEngine::getInstance()->isValid())
|
|
return;
|
|
|
|
se::AutoHandleScope scope;
|
|
assert(_inited);
|
|
|
|
se::Value func;
|
|
__jsbObj->getProperty(funcName, &func);
|
|
if (func.isObject() && func.toObject()->isFunction())
|
|
{
|
|
func.toObject()->call(se::EmptyValueArray, nullptr);
|
|
}
|
|
}
|
|
|
|
void EventDispatcher::dispatchOnPauseEvent()
|
|
{
|
|
// dispatch to Native
|
|
CustomEvent event;
|
|
event.name = EVENT_ON_PAUSE;
|
|
EventDispatcher::dispatchCustomEvent(event);
|
|
|
|
// dispatch to JavaScript
|
|
dispatchEnterBackgroundOrForegroundEvent("onPause");
|
|
}
|
|
|
|
void EventDispatcher::dispatchOnResumeEvent()
|
|
{
|
|
// dispatch to Native
|
|
CustomEvent event;
|
|
event.name = EVENT_ON_RESUME;
|
|
EventDispatcher::dispatchCustomEvent(event);
|
|
|
|
// dispatch to JavaScript
|
|
dispatchEnterBackgroundOrForegroundEvent("onResume");
|
|
}
|
|
|
|
uint32_t EventDispatcher::addCustomEventListener(const std::string& eventName, const CustomEventListener& listener)
|
|
{
|
|
static uint32_t __listenerIDCounter = 0;
|
|
uint32_t listenerID = ++__listenerIDCounter;
|
|
listenerID = listenerID == 0 ? 1 : listenerID;
|
|
|
|
Node* newNode = new Node();
|
|
newNode->listener = listener;
|
|
newNode->listenerID = listenerID;
|
|
newNode->next = nullptr;
|
|
|
|
auto iter = _listeners.find(eventName);
|
|
if (iter == _listeners.end())
|
|
{
|
|
_listeners.emplace(eventName, newNode);
|
|
}
|
|
else
|
|
{
|
|
Node* node = iter->second;
|
|
assert(node != nullptr);
|
|
Node* prev = nullptr;
|
|
while (node != nullptr)
|
|
{
|
|
prev = node;
|
|
node = node->next;
|
|
}
|
|
prev->next = newNode;
|
|
}
|
|
return listenerID;
|
|
}
|
|
|
|
void EventDispatcher::removeCustomEventListener(const std::string& eventName, uint32_t listenerID)
|
|
{
|
|
if (eventName.empty())
|
|
return;
|
|
|
|
if (listenerID == 0)
|
|
return;
|
|
|
|
auto iter = _listeners.find(eventName);
|
|
if (iter != _listeners.end())
|
|
{
|
|
Node* prev = nullptr;
|
|
Node* node = iter->second;
|
|
while (node != nullptr)
|
|
{
|
|
if (node->listenerID == listenerID)
|
|
{
|
|
if (prev != nullptr)
|
|
{
|
|
prev->next = node->next;
|
|
}
|
|
else if (node->next)
|
|
{
|
|
_listeners[eventName] = node->next;
|
|
}
|
|
else
|
|
{
|
|
_listeners.erase(iter);
|
|
}
|
|
|
|
delete node;
|
|
return;
|
|
}
|
|
|
|
prev = node;
|
|
node = node->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
void EventDispatcher::removeAllCustomEventListeners(const std::string& eventName)
|
|
{
|
|
auto iter = _listeners.find(eventName);
|
|
if (iter != _listeners.end())
|
|
{
|
|
Node* node = iter->second;
|
|
while (node != nullptr)
|
|
{
|
|
Node* next = node->next;
|
|
delete node;
|
|
node = next;
|
|
}
|
|
_listeners.erase(iter);
|
|
}
|
|
}
|
|
|
|
void EventDispatcher::dispatchCustomEvent(const CustomEvent& event)
|
|
{
|
|
auto iter = _listeners.find(event.name);
|
|
if (iter != _listeners.end())
|
|
{
|
|
Node* next = nullptr;
|
|
Node* node = iter->second;
|
|
while (node != nullptr)
|
|
{
|
|
next = node->next;
|
|
node->listener(event);
|
|
node = next;
|
|
}
|
|
}
|
|
}
|
|
|
|
} // end of namespace cocos2d
|