JSComponent.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537
  1. // Copyright (c) 2014-2015, THUNDERBEAST GAMES LLC All rights reserved
  2. // Please see LICENSE.md in repository root for license information
  3. // https://github.com/AtomicGameEngine/AtomicGameEngine
  4. #include "Precompiled.h"
  5. #include "../Core/Context.h"
  6. #include "../IO/Log.h"
  7. #ifdef ATOMIC_PHYSICS
  8. #include "../Physics/PhysicsEvents.h"
  9. #include "../Physics/PhysicsWorld.h"
  10. #endif
  11. #include "../Core/Profiler.h"
  12. #include "../IO/MemoryBuffer.h"
  13. #include "../Resource/ResourceCache.h"
  14. #include "../Resource/ResourceEvents.h"
  15. #include "../Scene/Scene.h"
  16. #include "../Scene/SceneEvents.h"
  17. #include "../Javascript/Javascript.h"
  18. #include "../Atomic2D/PhysicsEvents2D.h"
  19. #include "../Atomic2D/PhysicsWorld2D.h"
  20. #include "../Atomic2D/RigidBody2D.h"
  21. #include "../Javascript/JSEvents.h"
  22. #include "../Javascript/JSComponent.h"
  23. #include "../Javascript/JSAPI.h"
  24. #include "../UI/UIEvents.h"
  25. #include "../UI/UIElement.h"
  26. namespace Atomic
  27. {
  28. static const char* methodDeclarations[] = {
  29. "start",
  30. "stop",
  31. "delayedStart",
  32. "update",
  33. "postUpdate",
  34. "fixedUpdate",
  35. "fixedPostUpdate",
  36. "load",
  37. "save",
  38. "readNetworkUpdate",
  39. "writeNetworkUpdate",
  40. "applyAttributes",
  41. "transformChanged"
  42. };
  43. extern const char* LOGIC_CATEGORY;
  44. JSComponent::JSComponent(Context* context) :
  45. Component(context),
  46. script_(GetSubsystem<Javascript>()),
  47. scriptObject_(0),
  48. subscribed_(false),
  49. subscribedPostFixed_(false),
  50. started_(false),
  51. destroyed_(false)
  52. {
  53. vm_ = JSVM::GetJSVM(NULL);
  54. ClearScriptMethods();
  55. }
  56. JSComponent::~JSComponent()
  57. {
  58. }
  59. void JSComponent::OnNodeSet(Node *node)
  60. {
  61. Component::OnNodeSet(node);
  62. if (node)
  63. {
  64. assert(node->JSGetHeapPtr());
  65. duk_context* ctx = vm_->GetJSContext();
  66. duk_push_global_stash(ctx);
  67. duk_get_prop_index(ctx, -1, JS_GLOBALSTASH_INDEX_NODE_REGISTRY);
  68. // can't use instance as key, as this coerces to [Object] for
  69. // string property, pointer will be string representation of
  70. // address, so, unique key
  71. duk_push_pointer(ctx, node->JSGetHeapPtr());
  72. js_push_class_object_instance(ctx, node);
  73. duk_put_prop(ctx, -3);
  74. duk_pop_2(ctx);
  75. }
  76. }
  77. void JSComponent::RegisterObject(Context* context)
  78. {
  79. context->RegisterFactory<JSComponent>(LOGIC_CATEGORY);
  80. //ACCESSOR_ATTRIBUTE(JSComponent, VAR_BOOL, "Is Enabled", IsEnabled, SetEnabled, bool, true, AM_DEFAULT);
  81. //REF_ACCESSOR_ATTRIBUTE(JSComponent, VAR_STRING, "Class Name", GetClassName, SetClassName, String, String::EMPTY, AM_DEFAULT);
  82. //ACCESSOR_ATTRIBUTE(JSComponent, VAR_RESOURCEREF, "Script File", GetScriptFileAttr, SetScriptFileAttr, ResourceRef, ResourceRef(JSFile::GetTypeStatic()), AM_DEFAULT);
  83. //ACCESSOR_ATTRIBUTE(JSComponent, VAR_BUFFER, "Delayed Method Calls", GetDelayedCallsAttr, SetDelayedCallsAttr, PODVector<unsigned char>, Variant::emptyBuffer, AM_FILE | AM_NOEDIT);
  84. //ACCESSOR_ATTRIBUTE(JSComponent, VAR_BUFFER, "Script Data", GetScriptDataAttr, SetScriptDataAttr, PODVector<unsigned char>, Variant::emptyBuffer, AM_FILE | AM_NOEDIT);
  85. //ACCESSOR_ATTRIBUTE(JSComponent, VAR_BUFFER, "Script Network Data", GetScriptNetworkDataAttr, SetScriptNetworkDataAttr, PODVector<unsigned char>, Variant::emptyBuffer, AM_NET | AM_NOEDIT);
  86. }
  87. void JSComponent::ClearScriptMethods()
  88. {
  89. for (unsigned i = 0; i < MAX_JSSCRIPT_METHODS; ++i)
  90. methods_[i] = 0;
  91. //delayedCalls_.Clear();
  92. }
  93. void JSComponent::OnSetEnabled()
  94. {
  95. UpdateEventSubscription();
  96. }
  97. void JSComponent::ListenToEvent(Object* sender, StringHash eventType, JS_HEAP_PTR __duk_function)
  98. {
  99. duk_context* ctx = vm_->GetJSContext();
  100. duk_push_heapptr(ctx, __duk_function);
  101. assert(duk_is_function(ctx, -1));
  102. duk_pop(ctx);
  103. scriptEventFunctions_[eventType] = __duk_function;
  104. if (sender)
  105. SubscribeToEvent(sender, eventType, HANDLER(JSComponent, HandleScriptEvent));
  106. else
  107. SubscribeToEvent(eventType, HANDLER(JSComponent, HandleScriptEvent));
  108. }
  109. bool JSComponent::CreateObject(JSFile* scriptFile, const String& className)
  110. {
  111. className_ = String::EMPTY; // Do not create object during SetScriptFile()
  112. SetScriptFile(scriptFile);
  113. SetClassName(className);
  114. return scriptObject_ != 0;
  115. }
  116. void JSComponent::SetClassName(const String& className)
  117. {
  118. if (className == className_ && scriptObject_)
  119. return;
  120. ReleaseObject();
  121. className_ = className;
  122. CreateObject();
  123. MarkNetworkUpdate();
  124. }
  125. void JSComponent::ReleaseObject()
  126. {
  127. if (scriptObject_)
  128. {
  129. //if (methods_[JSMETHOD_STOP])
  130. // scriptFile_->Execute(scriptObject_, methods_[JSMETHOD_STOP]);
  131. PODVector<StringHash> exceptions;
  132. exceptions.Push(E_RELOADSTARTED);
  133. exceptions.Push(E_RELOADFINISHED);
  134. UnsubscribeFromAllEventsExcept(exceptions, false);
  135. if (node_)
  136. node_->RemoveListener(this);
  137. subscribed_ = false;
  138. subscribedPostFixed_ = false;
  139. ClearScriptMethods();
  140. scriptObject_ = 0;
  141. }
  142. }
  143. void JSComponent::SetScriptFile(JSFile* scriptFile)
  144. {
  145. ReleaseObject();
  146. CreateObject();
  147. MarkNetworkUpdate();
  148. }
  149. void JSComponent::CreateObject()
  150. {
  151. if (className_.Empty())
  152. return;
  153. PROFILE(CreateScriptObject);
  154. duk_context* ctx = vm_->GetJSContext();
  155. duk_push_global_stash(ctx);
  156. duk_get_prop_index(ctx, -1, JS_GLOBALSTASH_INDEX_COMPONENTS);
  157. duk_get_prop_string(ctx, -1, className_.CString());
  158. assert(duk_is_function(ctx, -1));
  159. js_push_class_object_instance(ctx, this);
  160. if (duk_pcall(ctx, 1) != 0)
  161. {
  162. vm_->SendJSErrorEvent();
  163. }
  164. else
  165. {
  166. scriptObject_ = this->JSGetHeapPtr();
  167. }
  168. if (scriptObject_)
  169. {
  170. GetScriptMethods();
  171. UpdateEventSubscription();
  172. }
  173. duk_pop_n(ctx, 2);
  174. }
  175. void JSComponent::HandleSceneUpdate(StringHash eventType, VariantMap& eventData)
  176. {
  177. if (!scriptObject_)
  178. return;
  179. assert(!destroyed_);
  180. assert(JSGetHeapPtr());
  181. using namespace SceneUpdate;
  182. float timeStep = eventData[P_TIMESTEP].GetFloat();
  183. duk_context* ctx = vm_->GetJSContext();
  184. if (!started_)
  185. {
  186. started_ = true;
  187. if (methods_[JSMETHOD_START])
  188. {
  189. duk_push_heapptr(ctx, methods_[JSMETHOD_START]);
  190. if (duk_pcall(ctx, 0) != 0)
  191. {
  192. vm_->SendJSErrorEvent();
  193. }
  194. duk_pop(ctx);
  195. }
  196. }
  197. if (methods_[JSMETHOD_UPDATE])
  198. {
  199. duk_push_heapptr(ctx, methods_[JSMETHOD_UPDATE]);
  200. duk_push_number(ctx, timeStep);
  201. if ( duk_pcall(ctx, 1) != DUK_EXEC_SUCCESS)
  202. {
  203. if (duk_is_object(ctx, -1))
  204. {
  205. vm_->SendJSErrorEvent();
  206. }
  207. else
  208. {
  209. assert(0);
  210. }
  211. }
  212. duk_pop(ctx);
  213. }
  214. }
  215. void JSComponent::UpdateEventSubscription()
  216. {
  217. Scene* scene = GetScene();
  218. if (!scene)
  219. {
  220. LOGWARNING("Node is detached from scene, can not subscribe script object to update events");
  221. return;
  222. }
  223. bool enabled = scriptObject_ && IsEnabledEffective();
  224. if (enabled)
  225. {
  226. if (!subscribed_ && (methods_[JSMETHOD_UPDATE] || methods_[JSMETHOD_DELAYEDSTART] ))
  227. {
  228. SubscribeToEvent(scene, E_SCENEUPDATE, HANDLER(JSComponent, HandleSceneUpdate));
  229. subscribed_ = true;
  230. }
  231. if (!subscribedPostFixed_)
  232. {
  233. if (methods_[JSMETHOD_POSTUPDATE])
  234. SubscribeToEvent(scene, E_SCENEPOSTUPDATE, HANDLER(JSComponent, HandleScenePostUpdate));
  235. #ifdef ATOMIC_PHYSICS
  236. if (methods_[JSMETHOD_FIXEDUPDATE] || methods_[JSMETHOD_FIXEDPOSTUPDATE])
  237. {
  238. PhysicsWorld* world = scene->GetOrCreateComponent<PhysicsWorld>();
  239. if (world)
  240. {
  241. if (methods_[JSMETHOD_FIXEDUPDATE])
  242. SubscribeToEvent(world, E_PHYSICSPRESTEP, HANDLER(JSComponent, HandlePhysicsPreStep));
  243. if (methods_[JSMETHOD_FIXEDPOSTUPDATE])
  244. SubscribeToEvent(world, E_PHYSICSPOSTSTEP, HANDLER(JSComponent, HandlePhysicsPostStep));
  245. }
  246. else
  247. LOGERROR("No physics world, can not subscribe script object to fixed update events");
  248. }
  249. #endif
  250. subscribedPostFixed_ = true;
  251. }
  252. if (methods_[JSMETHOD_TRANSFORMCHANGED])
  253. node_->AddListener(this);
  254. }
  255. else
  256. {
  257. if (subscribed_)
  258. {
  259. UnsubscribeFromEvent(scene, E_SCENEUPDATE);
  260. subscribed_ = false;
  261. }
  262. if (subscribedPostFixed_)
  263. {
  264. UnsubscribeFromEvent(scene, E_SCENEPOSTUPDATE);
  265. #ifdef ATOMIC_PHYSICS
  266. PhysicsWorld* world = scene->GetComponent<PhysicsWorld>();
  267. if (world)
  268. {
  269. UnsubscribeFromEvent(world, E_PHYSICSPRESTEP);
  270. UnsubscribeFromEvent(world, E_PHYSICSPOSTSTEP);
  271. }
  272. #endif
  273. subscribedPostFixed_ = false;
  274. }
  275. if (methods_[JSMETHOD_TRANSFORMCHANGED])
  276. node_->RemoveListener(this);
  277. }
  278. }
  279. void JSComponent::HandleScenePostUpdate(StringHash eventType, VariantMap& eventData)
  280. {
  281. if (!scriptObject_)
  282. return;
  283. assert(!destroyed_);
  284. using namespace ScenePostUpdate;
  285. if (methods_[JSMETHOD_POSTUPDATE])
  286. {
  287. duk_context* ctx = vm_->GetJSContext();
  288. duk_push_heapptr(ctx, methods_[JSMETHOD_POSTUPDATE]);
  289. duk_push_number(ctx, eventData[P_TIMESTEP].GetFloat());
  290. duk_pcall(ctx, 1);
  291. duk_pop(ctx);
  292. }
  293. }
  294. #ifdef ATOMIC_PHYSICS
  295. void JSComponent::HandlePhysicsPreStep(StringHash eventType, VariantMap& eventData)
  296. {
  297. if (!scriptObject_)
  298. return;
  299. assert(!destroyed_);
  300. using namespace PhysicsPreStep;
  301. float timeStep = eventData[P_TIMESTEP].GetFloat();
  302. if (methods_[JSMETHOD_FIXEDUPDATE])
  303. {
  304. duk_context* ctx = vm_->GetJSContext();
  305. duk_push_heapptr(ctx, methods_[JSMETHOD_FIXEDUPDATE]);
  306. duk_push_number(ctx, timeStep);
  307. duk_pcall(ctx, 1);
  308. duk_pop(ctx);
  309. }
  310. }
  311. void JSComponent::HandlePhysicsPostStep(StringHash eventType, VariantMap& eventData)
  312. {
  313. if (!scriptObject_)
  314. return;
  315. assert(!destroyed_);
  316. using namespace PhysicsPostStep;
  317. VariantVector parameters;
  318. parameters.Push(eventData[P_TIMESTEP]);
  319. }
  320. #endif
  321. void JSComponent::HandleScriptEvent(StringHash eventType, VariantMap& eventData)
  322. {
  323. if (!IsEnabledEffective() || !scriptObject_)
  324. return;
  325. assert(!destroyed_);
  326. if (scriptEventFunctions_.Contains(eventType))
  327. {
  328. duk_context* ctx = vm_->GetJSContext();
  329. JS_HEAP_PTR function = scriptEventFunctions_[eventType];
  330. if (eventType == E_UIMOUSECLICK)
  331. {
  332. UIElement* clicked = static_cast<UIElement*>(eventData[UIMouseClick::P_ELEMENT].GetPtr());
  333. if (clicked)
  334. {
  335. duk_push_heapptr(ctx, function);
  336. js_push_class_object_instance(ctx, clicked);
  337. if (duk_pcall(ctx, 1) != 0)
  338. {
  339. vm_->SendJSErrorEvent();
  340. }
  341. duk_pop(ctx);
  342. }
  343. }
  344. else if (eventType == E_PHYSICSBEGINCONTACT2D || E_PHYSICSENDCONTACT2D)
  345. {
  346. using namespace PhysicsBeginContact2D;
  347. PhysicsWorld2D* world = static_cast<PhysicsWorld2D*>(eventData[P_WORLD].GetPtr());
  348. RigidBody2D* bodyA = static_cast<RigidBody2D*>(eventData[P_BODYA].GetPtr());
  349. RigidBody2D* bodyB = static_cast<RigidBody2D*>(eventData[P_BODYB].GetPtr());
  350. Node* nodeA = static_cast<Node*>(eventData[P_NODEA].GetPtr());
  351. Node* nodeB = static_cast<Node*>(eventData[P_NODEB].GetPtr());
  352. duk_push_heapptr(ctx, function);
  353. js_push_class_object_instance(ctx, world);
  354. js_push_class_object_instance(ctx, bodyA);
  355. js_push_class_object_instance(ctx, bodyB);
  356. js_push_class_object_instance(ctx, nodeA);
  357. js_push_class_object_instance(ctx, nodeB);
  358. if (duk_pcall(ctx, 5) != 0)
  359. {
  360. vm_->SendJSErrorEvent();
  361. }
  362. duk_pop(ctx);
  363. }
  364. else if (eventType == E_NODECOLLISION)
  365. {
  366. // Check collision contacts and see if character is standing on ground (look for a contact that has near vertical normal)
  367. using namespace NodeCollision;
  368. MemoryBuffer contacts(eventData[P_CONTACTS].GetBuffer());
  369. while (!contacts.IsEof())
  370. {
  371. Vector3 contactPosition = contacts.ReadVector3();
  372. Vector3 contactNormal = contacts.ReadVector3();
  373. float contactDistance = contacts.ReadFloat();
  374. float contactImpulse = contacts.ReadFloat();
  375. duk_push_heapptr(ctx, function);
  376. duk_push_array(ctx);
  377. duk_push_number(ctx, contactPosition.x_);
  378. duk_put_prop_index(ctx, -2, 0);
  379. duk_push_number(ctx, contactPosition.y_);
  380. duk_put_prop_index(ctx, -2, 1);
  381. duk_push_number(ctx, contactPosition.z_);
  382. duk_put_prop_index(ctx, -2, 2);
  383. duk_push_array(ctx);
  384. duk_push_number(ctx, contactNormal.x_);
  385. duk_put_prop_index(ctx, -2, 0);
  386. duk_push_number(ctx, contactNormal.y_);
  387. duk_put_prop_index(ctx, -2, 1);
  388. duk_push_number(ctx, contactNormal.z_);
  389. duk_put_prop_index(ctx, -2, 2);
  390. duk_call(ctx, 2);
  391. duk_pop(ctx);
  392. }
  393. }
  394. else
  395. {
  396. duk_push_heapptr(ctx, function);
  397. if (duk_pcall(ctx, 0) != 0)
  398. {
  399. vm_->SendJSErrorEvent();
  400. }
  401. duk_pop(ctx);
  402. }
  403. }
  404. }
  405. void JSComponent::GetScriptMethods()
  406. {
  407. if (!scriptObject_)
  408. return;
  409. duk_context* ctx = vm_->GetJSContext();
  410. duk_push_heapptr(ctx, scriptObject_);
  411. for (unsigned i = 0; i < MAX_JSSCRIPT_METHODS; ++i)
  412. {
  413. duk_get_prop_string(ctx, -1, methodDeclarations[i]);
  414. if (duk_is_function(ctx, -1))
  415. {
  416. methods_[i] = duk_get_heapptr(ctx, -1);
  417. }
  418. duk_pop(ctx);
  419. }
  420. duk_pop(ctx);
  421. }
  422. }