// // Copyright (c) 2008-2017 the Urho3D project. // // 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 "../Precompiled.h" #include "../AngelScript/APITemplates.h" #include "../AngelScript/ScriptAPI.h" #include "../AngelScript/ScriptFile.h" #include "../Resource/ResourceCache.h" namespace Urho3D { static bool ScriptFileExecute(const String& declaration, CScriptArray* srcParams, ScriptFile* ptr) { VariantVector destParams(srcParams ? srcParams->GetSize() : 0); if (srcParams) { unsigned numParams = srcParams->GetSize(); for (unsigned i = 0; i < numParams; ++i) destParams[i] = *(static_cast(srcParams->At(i))); } return ptr->Execute(declaration, destParams); } static void ScriptFileDelayedExecute(float delay, bool repeat, const String& declaration, CScriptArray* srcParams, ScriptFile* ptr) { VariantVector destParams(srcParams ? srcParams->GetSize() : 0); if (srcParams) { unsigned numParams = srcParams->GetSize(); for (unsigned i = 0; i < numParams; ++i) destParams[i] = *(static_cast(srcParams->At(i))); } ptr->DelayedExecute(delay, repeat, declaration, destParams); } static asIScriptObject* NodeCreateScriptObjectWithFile(ScriptFile* file, const String& className, CreateMode mode, Node* ptr) { if (!file) return nullptr; // Try first to reuse an existing, empty ScriptInstance const Vector >& components = ptr->GetComponents(); for (Vector >::ConstIterator i = components.Begin(); i != components.End(); ++i) { if ((*i)->GetType() == ScriptInstance::GetTypeStatic()) { auto* instance = static_cast(i->Get()); asIScriptObject* object = instance->GetScriptObject(); if (!object) { instance->CreateObject(file, className); return instance->GetScriptObject(); } } } // Then create a new component if not found auto* instance = ptr->CreateComponent(mode); instance->CreateObject(file, className); return instance->GetScriptObject(); } static void RegisterScriptFile(asIScriptEngine* engine) { RegisterResource(engine, "ScriptFile"); engine->RegisterObjectMethod("ScriptFile", "bool Execute(const String&in, const Array@+ params = null)", asFUNCTION(ScriptFileExecute), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("ScriptFile", "void DelayedExecute(float, bool, const String&in, const Array@+ params = null)", asFUNCTION(ScriptFileDelayedExecute), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("ScriptFile", "void ClearDelayedExecute(const String&in declaration = String())", asMETHOD(ScriptFile, ClearDelayedExecute), asCALL_THISCALL); engine->RegisterObjectMethod("ScriptFile", "bool get_compiled() const", asMETHOD(ScriptFile, IsCompiled), asCALL_THISCALL); engine->RegisterGlobalFunction("ScriptFile@+ get_scriptFile()", asFUNCTION(GetScriptContextFile), asCALL_CDECL); } static asIScriptObject* NodeCreateScriptObject(const String& scriptFileName, const String& className, CreateMode mode, Node* ptr) { auto* cache = GetScriptContext()->GetSubsystem(); return NodeCreateScriptObjectWithFile(cache->GetResource(scriptFileName), className, mode, ptr); } asIScriptObject* NodeGetScriptObject(Node* ptr) { // Get the first available ScriptInstance with an object const Vector >& components = ptr->GetComponents(); for (Vector >::ConstIterator i = components.Begin(); i != components.End(); ++i) { if ((*i)->GetType() == ScriptInstance::GetTypeStatic()) { auto* instance = static_cast(i->Get()); asIScriptObject* object = instance->GetScriptObject(); if (object) return object; } } return nullptr; } asIScriptObject* NodeGetNamedScriptObject(const String& className, Node* ptr) { const Vector >& components = ptr->GetComponents(); for (Vector >::ConstIterator i = components.Begin(); i != components.End(); ++i) { if ((*i)->GetType() == ScriptInstance::GetTypeStatic()) { auto* instance = static_cast(i->Get()); if (instance->IsA(className)) { asIScriptObject* object = instance->GetScriptObject(); if (object) return object; } } } return nullptr; } static bool ScriptInstanceExecute(const String& declaration, CScriptArray* srcParams, ScriptInstance* ptr) { VariantVector destParams(srcParams ? srcParams->GetSize() : 0); if (srcParams) { unsigned numParams = srcParams->GetSize(); for (unsigned i = 0; i < numParams; ++i) destParams[i] = *(static_cast(srcParams->At(i))); } return ptr->Execute(declaration, destParams); } static void ScriptInstanceDelayedExecute(float delay, bool repeat, const String& declaration, CScriptArray* srcParams, ScriptInstance* ptr) { VariantVector destParams(srcParams ? srcParams->GetSize() : 0); if (srcParams) { unsigned numParams = srcParams->GetSize(); for (unsigned i = 0; i < numParams; ++i) destParams[i] = *(static_cast(srcParams->At(i))); } ptr->DelayedExecute(delay, repeat, declaration, destParams); } static ScriptInstance* GetSelf() { return GetScriptContextInstance(); } static void SelfDelayedExecute(float delay, bool repeat, const String& declaration, CScriptArray* srcParams) { VariantVector destParams(srcParams ? srcParams->GetSize() : 0); if (srcParams) { unsigned numParams = srcParams->GetSize(); for (unsigned i = 0; i < numParams; ++i) destParams[i] = *(static_cast(srcParams->At(i))); } ScriptInstance* ptr = GetScriptContextInstance(); if (ptr) ptr->DelayedExecute(delay, repeat, declaration, destParams); else { ScriptFile* file = GetScriptContextFile(); if (file) file->DelayedExecute(delay, repeat, declaration, destParams); } } static void SelfClearDelayedExecute(const String& declaration) { ScriptInstance* ptr = GetScriptContextInstance(); if (ptr) ptr->ClearDelayedExecute(declaration); else { ScriptFile* file = GetScriptContextFile(); if (file) file->ClearDelayedExecute(declaration); } } static void SelfMarkNetworkUpdate() { ScriptInstance* ptr = GetScriptContextInstance(); if (ptr) ptr->MarkNetworkUpdate(); } static void SelfRemove() { ScriptInstance* ptr = GetScriptContextInstance(); if (ptr) ptr->Remove(); } static void RegisterScriptInstance(asIScriptEngine* engine) { engine->RegisterObjectMethod("Node", "ScriptObject@+ CreateScriptObject(ScriptFile@+, const String&in, CreateMode mode = REPLICATED)", asFUNCTION(NodeCreateScriptObjectWithFile), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("Node", "ScriptObject@+ CreateScriptObject(const String&in, const String&in, CreateMode mode = REPLICATED)", asFUNCTION(NodeCreateScriptObject), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("Node", "ScriptObject@+ GetScriptObject() const", asFUNCTION(NodeGetScriptObject), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("Node", "ScriptObject@+ GetScriptObject(const String&in) const", asFUNCTION(NodeGetNamedScriptObject), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("Node", "ScriptObject@+ get_scriptObject() const", asFUNCTION(NodeGetScriptObject), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("Scene", "ScriptObject@+ CreateScriptObject(ScriptFile@+, const String&in, CreateMode mode = REPLICATED)", asFUNCTION(NodeCreateScriptObjectWithFile), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("Scene", "ScriptObject@+ CreateScriptObject(const String&in, const String&in, CreateMode mode = REPLICATED)", asFUNCTION(NodeCreateScriptObject), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("Scene", "ScriptObject@+ GetScriptObject() const", asFUNCTION(NodeGetScriptObject), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("Scene", "ScriptObject@+ GetScriptObject(const String&in) const", asFUNCTION(NodeGetNamedScriptObject), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("Scene", "ScriptObject@+ get_scriptObject() const", asFUNCTION(NodeGetScriptObject), asCALL_CDECL_OBJLAST); RegisterComponent(engine, "ScriptInstance"); engine->RegisterObjectMethod("ScriptInstance", "bool CreateObject(ScriptFile@+, const String&in)", asMETHODPR(ScriptInstance, CreateObject, (ScriptFile*, const String&), bool), asCALL_THISCALL); engine->RegisterObjectMethod("ScriptInstance", "bool Execute(const String&in, const Array@+ params = null)", asFUNCTION(ScriptInstanceExecute), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("ScriptInstance", "void DelayedExecute(float, bool, const String&in, const Array@+ params = null)", asFUNCTION(ScriptInstanceDelayedExecute), asCALL_CDECL_OBJLAST); engine->RegisterObjectMethod("ScriptInstance", "void ClearDelayedExecute(const String&in declaration = String())", asMETHOD(ScriptInstance, ClearDelayedExecute), asCALL_THISCALL); engine->RegisterObjectMethod("ScriptInstance", "bool IsA(const String&in declaration) const", asMETHOD(ScriptInstance, IsA), asCALL_THISCALL); engine->RegisterObjectMethod("ScriptInstance", "bool HasMethod(const String&in declaration) const", asMETHOD(ScriptInstance, HasMethod), asCALL_THISCALL); engine->RegisterObjectMethod("ScriptInstance", "void set_scriptFile(ScriptFile@+)", asMETHOD(ScriptInstance, SetScriptFile), asCALL_THISCALL); engine->RegisterObjectMethod("ScriptInstance", "ScriptFile@+ get_scriptFile() const", asMETHOD(ScriptInstance, GetScriptFile), asCALL_THISCALL); engine->RegisterObjectMethod("ScriptInstance", "ScriptObject@+ get_scriptObject() const", asMETHOD(ScriptInstance, GetScriptObject), asCALL_THISCALL); engine->RegisterObjectMethod("ScriptInstance", "void set_className(const String&in)", asMETHOD(ScriptInstance, SetClassName), asCALL_THISCALL); engine->RegisterObjectMethod("ScriptInstance", "const String& get_className() const", asMETHOD(ScriptInstance, GetClassName), asCALL_THISCALL); engine->RegisterGlobalFunction("ScriptInstance@+ get_self()", asFUNCTION(GetSelf), asCALL_CDECL); // Register convenience functions for controlling self, similar to event sending engine->RegisterGlobalFunction("void MarkNetworkUpdate()", asFUNCTION(SelfMarkNetworkUpdate), asCALL_CDECL); engine->RegisterGlobalFunction("void DelayedExecute(float, bool, const String&in, const Array@+ params = null)", asFUNCTION(SelfDelayedExecute), asCALL_CDECL); engine->RegisterGlobalFunction("void ClearDelayedExecute(const String&in declaration = String())", asFUNCTION(SelfClearDelayedExecute), asCALL_CDECL); engine->RegisterGlobalFunction("void Remove()", asFUNCTION(SelfRemove), asCALL_CDECL); } static Script* GetScript() { return GetScriptContext()->GetSubsystem