ewk_js.cpp 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752
  1. /*
  2. Copyright (C) 2011 ProFUSION embedded systems
  3. This library is free software; you can redistribute it and/or
  4. modify it under the terms of the GNU Library General Public
  5. License as published by the Free Software Foundation; either
  6. version 2 of the License, or (at your option) any later version.
  7. This library is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  10. Library General Public License for more details.
  11. You should have received a copy of the GNU Library General Public License
  12. along with this library; see the file COPYING.LIB. If not, write to
  13. the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
  14. Boston, MA 02110-1301, USA.
  15. */
  16. #include "config.h"
  17. #include "ewk_js.h"
  18. #if ENABLE(NETSCAPE_PLUGIN_API)
  19. #include "NP_jsobject.h"
  20. #include "Operations.h"
  21. #include "ewk_js_private.h"
  22. #include "ewk_private.h"
  23. #include "npruntime.h"
  24. #include "npruntime_impl.h"
  25. #include <string.h>
  26. #define EINA_MAGIC_CHECK_OR_RETURN(o, ...) \
  27. if (!EINA_MAGIC_CHECK(o, EWK_JS_OBJECT_MAGIC)) { \
  28. EINA_MAGIC_FAIL(o, EWK_JS_OBJECT_MAGIC); \
  29. return __VA_ARGS__; \
  30. }
  31. struct _Ewk_JS_Class {
  32. const Ewk_JS_Class_Meta* meta;
  33. Eina_Hash* methods; // Key=NPIdentifier(name), value=pointer to meta->methods.
  34. Eina_Hash* properties; // Key=NPIdentifier(name), value=pointer to meta->properties.
  35. Ewk_JS_Default default_prop;
  36. };
  37. static Eina_Bool ewk_js_npvariant_to_variant(Ewk_JS_Variant* data, const NPVariant* result);
  38. static Eina_Bool ewk_js_variant_to_npvariant(const Ewk_JS_Variant* data, NPVariant* result)
  39. {
  40. EINA_SAFETY_ON_NULL_RETURN_VAL(data, false);
  41. EINA_SAFETY_ON_NULL_RETURN_VAL(result, false);
  42. const char* string_value;
  43. switch (data->type) {
  44. case EWK_JS_VARIANT_VOID:
  45. VOID_TO_NPVARIANT(*result);
  46. break;
  47. case EWK_JS_VARIANT_NULL:
  48. NULL_TO_NPVARIANT(*result);
  49. break;
  50. case EWK_JS_VARIANT_INT32:
  51. INT32_TO_NPVARIANT(data->value.i, *result);
  52. break;
  53. case EWK_JS_VARIANT_DOUBLE:
  54. DOUBLE_TO_NPVARIANT(data->value.d, *result);
  55. break;
  56. case EWK_JS_VARIANT_STRING:
  57. string_value = eina_stringshare_add(data->value.s);
  58. if (string_value)
  59. STRINGZ_TO_NPVARIANT(string_value, *result);
  60. else
  61. return false;
  62. break;
  63. case EWK_JS_VARIANT_BOOL:
  64. BOOLEAN_TO_NPVARIANT(data->value.b, *result);
  65. break;
  66. case EWK_JS_VARIANT_OBJECT:
  67. OBJECT_TO_NPVARIANT(reinterpret_cast<NPObject*>(data->value.o), *result);
  68. break;
  69. default:
  70. return false;
  71. }
  72. return true;
  73. }
  74. // These methods are used by NPAI, thats the reason to use bool instead of Eina_Bool.
  75. static bool ewk_js_property_has(NPObject* npObject, NPIdentifier name)
  76. {
  77. Ewk_JS_Object* object = reinterpret_cast<Ewk_JS_Object*>(npObject);
  78. EINA_SAFETY_ON_NULL_RETURN_VAL(npObject, false);
  79. EINA_MAGIC_CHECK_OR_RETURN(object, false);
  80. if (!_NPN_IdentifierIsString(name)) {
  81. ERR("int NPIdentifier is not supported.");
  82. return false;
  83. }
  84. char* prop_name = _NPN_UTF8FromIdentifier(name);
  85. bool fail = eina_hash_find(object->properties, prop_name); // FIXME: should search methods too?
  86. free(prop_name);
  87. return fail;
  88. }
  89. static bool ewk_js_property_get(NPObject* npObject, NPIdentifier name, NPVariant* result)
  90. {
  91. Ewk_JS_Object* object = reinterpret_cast<Ewk_JS_Object*>(npObject);
  92. Ewk_JS_Variant* value;
  93. Ewk_JS_Property* prop;
  94. bool fail = false;
  95. EINA_SAFETY_ON_NULL_RETURN_VAL(npObject, false);
  96. EINA_MAGIC_CHECK_OR_RETURN(object, false);
  97. if (!_NPN_IdentifierIsString(name)) {
  98. ERR("int NPIdentifier is not supported.");
  99. return false;
  100. }
  101. value = static_cast<Ewk_JS_Variant*>(malloc(sizeof(Ewk_JS_Variant)));
  102. if (!value) {
  103. ERR("Could not allocate memory for ewk_js_variant");
  104. return false;
  105. }
  106. prop = static_cast<Ewk_JS_Property*>(eina_hash_find(object->cls->properties, name));
  107. if (prop && prop->get) { // Class has property and property has getter.
  108. fail = prop->get(object, prop->name, value);
  109. if (!fail)
  110. fail = ewk_js_variant_to_npvariant(value, result);
  111. } else if (object->cls->default_prop.get) { // Default getter exists.
  112. fail = object->cls->default_prop.get(object, prop->name, value);
  113. if (!fail)
  114. fail = ewk_js_variant_to_npvariant(value, result);
  115. } else { // Fallback to objects hash map.
  116. char* prop_name = _NPN_UTF8FromIdentifier(name);
  117. free(value);
  118. value = static_cast<Ewk_JS_Variant*>(eina_hash_find(object->properties, prop_name));
  119. free(prop_name);
  120. if (value)
  121. return ewk_js_variant_to_npvariant(value, result);
  122. }
  123. free(value);
  124. return fail;
  125. }
  126. static bool ewk_js_property_set(NPObject* npObject, NPIdentifier name, const NPVariant* npValue)
  127. {
  128. Ewk_JS_Object* object = reinterpret_cast<Ewk_JS_Object*>(npObject);
  129. Ewk_JS_Variant* value;
  130. Ewk_JS_Property* prop;
  131. bool fail = false;
  132. EINA_SAFETY_ON_NULL_RETURN_VAL(npObject, false);
  133. EINA_MAGIC_CHECK_OR_RETURN(object, false);
  134. if (!_NPN_IdentifierIsString(name)) {
  135. ERR("int NPIdentifier is not supported.");
  136. return fail;
  137. }
  138. value = static_cast<Ewk_JS_Variant*>(malloc(sizeof(Ewk_JS_Variant)));
  139. if (!value) {
  140. ERR("Could not allocate memory for ewk_js_variant");
  141. return false;
  142. }
  143. ewk_js_npvariant_to_variant(value, npValue);
  144. char* prop_name = _NPN_UTF8FromIdentifier(name);
  145. prop = static_cast<Ewk_JS_Property*>(eina_hash_find(object->cls->properties, prop_name));
  146. if (prop && prop->set)
  147. fail = prop->set(object, prop->name, value); // Class has property and property has setter.
  148. else if (object->cls->default_prop.set)
  149. fail = object->cls->default_prop.set(object, prop_name, value); // Default getter exists.
  150. else { // Fallback to objects hash map.
  151. void* old = eina_hash_set(object->properties, prop_name, value);
  152. free(old);
  153. fail = true;
  154. }
  155. free(prop_name);
  156. return fail;
  157. }
  158. static bool ewk_js_property_remove(NPObject* npObject, NPIdentifier name)
  159. {
  160. Ewk_JS_Object* object = reinterpret_cast<Ewk_JS_Object*>(npObject);
  161. Ewk_JS_Property* prop;
  162. bool fail = false;
  163. EINA_SAFETY_ON_NULL_RETURN_VAL(npObject, false);
  164. EINA_MAGIC_CHECK_OR_RETURN(object, false);
  165. if (!_NPN_IdentifierIsString(name)) {
  166. ERR("int NPIdentifier is not supported.");
  167. return fail;
  168. }
  169. char* prop_name = _NPN_UTF8FromIdentifier(name);
  170. prop = static_cast<Ewk_JS_Property*>(eina_hash_find(object->cls->properties, prop_name));
  171. if (prop && prop->del)
  172. fail = prop->del(object, prop->name); // Class has property and property has getter.
  173. else if (object->cls->default_prop.del)
  174. fail = object->cls->default_prop.del(object, prop_name);
  175. else
  176. fail = eina_hash_del(object->properties, prop_name, 0);
  177. free(prop_name);
  178. return fail;
  179. }
  180. static bool ewk_js_properties_enumerate(NPObject* npObject, NPIdentifier** value, uint32_t* count)
  181. {
  182. Ewk_JS_Object* object = reinterpret_cast<Ewk_JS_Object*>(npObject);
  183. Eina_Iterator* it;
  184. char* key;
  185. int i = 0;
  186. EINA_SAFETY_ON_NULL_RETURN_VAL(npObject, false);
  187. EINA_MAGIC_CHECK_OR_RETURN(object, false);
  188. *count = eina_hash_population(object->properties);
  189. *value = static_cast<NPIdentifier*>(malloc(sizeof(NPIdentifier) * *count));
  190. if (!*value) {
  191. ERR("Could not allocate memory for NPIdentifier");
  192. return false;
  193. }
  194. it = eina_hash_iterator_key_new(object->properties);
  195. EINA_ITERATOR_FOREACH(it, key)
  196. (*value)[i++] = _NPN_GetStringIdentifier(key);
  197. eina_iterator_free(it);
  198. return true;
  199. }
  200. static bool ewk_js_method_has(NPObject* npObject, NPIdentifier name)
  201. {
  202. Ewk_JS_Object* object = reinterpret_cast<Ewk_JS_Object*>(npObject);
  203. EINA_SAFETY_ON_NULL_RETURN_VAL(npObject, false);
  204. EINA_MAGIC_CHECK_OR_RETURN(object, false);
  205. if (!_NPN_IdentifierIsString(name)) {
  206. ERR("int NPIdentifier is not supported.");
  207. return false;
  208. }
  209. return eina_hash_find(object->cls->methods, name); // Returns pointer if found(true), 0(false) otherwise.
  210. }
  211. static bool ewk_js_method_invoke(NPObject* npObject, NPIdentifier name, const NPVariant* npArgs, uint32_t npArgCount, NPVariant* result)
  212. {
  213. Ewk_JS_Object* object = reinterpret_cast<Ewk_JS_Object*>(npObject);
  214. Ewk_JS_Method* method;
  215. Ewk_JS_Variant* args;
  216. Ewk_JS_Variant* ret_val;
  217. EINA_SAFETY_ON_NULL_RETURN_VAL(npObject, false);
  218. EINA_MAGIC_CHECK_OR_RETURN(object, false);
  219. if (!_NPN_IdentifierIsString(name)) {
  220. ERR("int NPIdentifier is not supported.");
  221. return false;
  222. }
  223. method = static_cast<Ewk_JS_Method*>(eina_hash_find(object->cls->methods, name));
  224. if (!method)
  225. return false;
  226. args = static_cast<Ewk_JS_Variant*>(malloc(sizeof(Ewk_JS_Variant) *npArgCount));
  227. if (!args) {
  228. ERR("Could not allocate memory for ewk_js_variant");
  229. return false;
  230. }
  231. for (uint32_t i = 0; i < npArgCount; i++)
  232. ewk_js_npvariant_to_variant(&args[i], &npArgs[i]);
  233. ret_val = method->invoke(object, args, npArgCount);
  234. ewk_js_variant_to_npvariant(ret_val, result);
  235. ewk_js_variant_free(ret_val);
  236. return true;
  237. }
  238. static Eina_Bool ewk_js_npobject_property_get(Ewk_JS_Object* jsObject, const char* name, Ewk_JS_Variant* value)
  239. {
  240. NPIdentifier id = _NPN_GetStringIdentifier(name);
  241. NPVariant var;
  242. bool fail = _NPN_GetProperty(0, reinterpret_cast<NPObject*>(jsObject), id, &var);
  243. if (!fail)
  244. fail = ewk_js_npvariant_to_variant(value, &var);
  245. return fail;
  246. }
  247. static Eina_Bool ewk_js_npobject_property_set(Ewk_JS_Object* jsObject, const char* name, const Ewk_JS_Variant* value)
  248. {
  249. NPIdentifier id = _NPN_GetStringIdentifier(name);
  250. NPVariant var;
  251. bool fail = ewk_js_variant_to_npvariant(value, &var);
  252. if (fail)
  253. fail = _NPN_SetProperty(0, reinterpret_cast<NPObject*>(jsObject), id, &var);
  254. return fail;
  255. }
  256. static Eina_Bool ewk_js_npobject_property_del(Ewk_JS_Object* jsObject, const char* name)
  257. {
  258. NPIdentifier id = _NPN_GetStringIdentifier(name);
  259. return _NPN_RemoveProperty(0, reinterpret_cast<NPObject*>(jsObject), id);
  260. }
  261. static void ewk_js_property_free(Ewk_JS_Property* prop)
  262. {
  263. free(const_cast<char*>(prop->name));
  264. if (prop->value.type == EWK_JS_VARIANT_STRING)
  265. eina_stringshare_del(prop->value.value.s);
  266. else if (prop->value.type == EWK_JS_VARIANT_OBJECT)
  267. ewk_js_object_free(prop->value.value.o);
  268. free(prop);
  269. }
  270. /**
  271. * Create a Ewk_JS_Class to be used in @a ewk_js_object_new.
  272. *
  273. * @param meta @a Ewk_JS_Class_Meta that describes the class to be created.
  274. *
  275. * @return The Ewk_JS_Class created.
  276. */
  277. Ewk_JS_Class* ewk_js_class_new(const Ewk_JS_Class_Meta* jsMetaClass)
  278. {
  279. Ewk_JS_Class* cls;
  280. EINA_SAFETY_ON_NULL_RETURN_VAL(jsMetaClass, 0);
  281. cls = static_cast<Ewk_JS_Class*>(malloc(sizeof(Ewk_JS_Class)));
  282. if (!cls) {
  283. ERR("Could not allocate memory for ewk_js_class");
  284. return 0;
  285. }
  286. cls->meta = jsMetaClass;
  287. cls->default_prop = cls->meta->default_prop;
  288. // Don't free methods since they point to meta class methods(will be freed when meta class is freed).
  289. cls->methods = eina_hash_pointer_new(0);
  290. for (int i = 0; cls->meta->methods && cls->meta->methods[i].name; i++) {
  291. NPIdentifier id = _NPN_GetStringIdentifier(cls->meta->methods[i].name);
  292. eina_hash_add(cls->methods, id, &cls->meta->methods[i]);
  293. }
  294. // Don't free properties since they point to cls->meta class properties(will be freed when cls->meta class is freed).
  295. cls->properties = eina_hash_pointer_new(0);
  296. for (int i = 0; cls->meta->properties && cls->meta->properties[i].name; i++) {
  297. NPIdentifier id = _NPN_GetStringIdentifier(cls->meta->properties[i].name);
  298. eina_hash_add(cls->properties, id, &cls->meta->properties[i]);
  299. }
  300. return cls;
  301. }
  302. /**
  303. * Release resources allocated by @a cls.
  304. *
  305. * @param cls @a Ewk_JS_Class to be released.
  306. */
  307. void ewk_js_class_free(Ewk_JS_Class* jsClass)
  308. {
  309. EINA_SAFETY_ON_NULL_RETURN(jsClass);
  310. eina_hash_free(jsClass->methods);
  311. eina_hash_free(jsClass->properties);
  312. free(jsClass);
  313. }
  314. static NPClass EWK_NPCLASS = {
  315. NP_CLASS_STRUCT_VERSION,
  316. 0, // NPAllocateFunctionPtr
  317. 0, // NPDeallocateFunctionPtr
  318. 0, // NPInvalidateFunctionPtr
  319. ewk_js_method_has, // NPHasMethodFunctionPtr
  320. ewk_js_method_invoke, // NPInvokeFunctionPtr
  321. 0, // NPInvokeDefaultFunctionPtr
  322. ewk_js_property_has, // NPHasPropertyFunctionPtr
  323. ewk_js_property_get, // NPGetPropertyFunctionPtr
  324. ewk_js_property_set, // NPSetPropertyFunctionPtr
  325. ewk_js_property_remove, // NPRemovePropertyFunctionPtr
  326. ewk_js_properties_enumerate, // NPEnumerationFunctionPtr
  327. 0 // NPConstructFunction
  328. };
  329. static Ewk_JS_Object* ewk_js_npobject_to_object(NPObject* npObject)
  330. {
  331. NPIdentifier* values;
  332. uint32_t np_props_count;
  333. Ewk_JS_Class* cls;
  334. Ewk_JS_Object* object;
  335. Eina_Iterator* it;
  336. Ewk_JS_Property* prop;
  337. JavaScriptObject* jso;
  338. if (EINA_MAGIC_CHECK(reinterpret_cast<Ewk_JS_Object*>(npObject), EWK_JS_OBJECT_MAGIC))
  339. return reinterpret_cast<Ewk_JS_Object*>(npObject);
  340. if (!_NPN_Enumerate(0, npObject, &values, &np_props_count))
  341. return 0;
  342. cls = static_cast<Ewk_JS_Class*>(malloc(sizeof(Ewk_JS_Class)));
  343. if (!cls) {
  344. ERR("Could not allocate memory for ewk_js_class");
  345. return 0;
  346. }
  347. cls->meta = 0;
  348. Ewk_JS_Default def = {
  349. ewk_js_npobject_property_set,
  350. ewk_js_npobject_property_get,
  351. ewk_js_npobject_property_del
  352. };
  353. cls->default_prop = def;
  354. cls->methods = eina_hash_pointer_new(0);
  355. cls->properties = eina_hash_pointer_new(reinterpret_cast<Eina_Free_Cb>(ewk_js_property_free));
  356. for (uint32_t i = 0; i < np_props_count; i++) {
  357. if (_NPN_HasProperty(0, npObject, values[i])) {
  358. NPVariant var;
  359. Ewk_JS_Property* prop = static_cast<Ewk_JS_Property*>(calloc(sizeof(Ewk_JS_Property), 1));
  360. if (!prop) {
  361. ERR("Could not allocate memory for ewk_js_property");
  362. goto error;
  363. }
  364. _NPN_GetProperty(0, npObject, values[i], &var);
  365. ewk_js_npvariant_to_variant(&(prop->value), &var);
  366. prop->name = _NPN_UTF8FromIdentifier(values[i]);
  367. eina_hash_add(cls->properties, values[i], prop);
  368. }
  369. }
  370. // Can't use ewk_js_object_new(cls) because it expects cls->meta to exist.
  371. object = static_cast<Ewk_JS_Object*>(malloc(sizeof(Ewk_JS_Object)));
  372. if (!object) {
  373. ERR("Could not allocate memory for ewk_js_object");
  374. goto error;
  375. }
  376. free(values);
  377. EINA_MAGIC_SET(object, EWK_JS_OBJECT_MAGIC);
  378. object->name = 0;
  379. object->cls = cls;
  380. object->view = 0;
  381. jso = reinterpret_cast<JavaScriptObject*>(npObject);
  382. if (!strcmp("Array", jso->imp->methodTable()->className(jso->imp).ascii().data()))
  383. object->type = EWK_JS_OBJECT_ARRAY;
  384. else if (!strcmp("Function", jso->imp->methodTable()->className(jso->imp).ascii().data()))
  385. object->type = EWK_JS_OBJECT_FUNCTION;
  386. else
  387. object->type = EWK_JS_OBJECT_OBJECT;
  388. if (eina_hash_population(cls->properties) < 25)
  389. object->properties = eina_hash_string_small_new(0);
  390. else
  391. object->properties = eina_hash_string_superfast_new(0);
  392. it = eina_hash_iterator_data_new(cls->properties);
  393. EINA_ITERATOR_FOREACH(it, prop) {
  394. const char* key = prop->name;
  395. Ewk_JS_Variant* value = &prop->value;
  396. eina_hash_add(object->properties, key, value);
  397. }
  398. eina_iterator_free(it);
  399. object->base = *reinterpret_cast<JavaScriptObject*>(npObject);
  400. return object;
  401. error:
  402. ewk_js_class_free(cls);
  403. free(values);
  404. return 0;
  405. }
  406. static Eina_Bool ewk_js_npvariant_to_variant(Ewk_JS_Variant* data, const NPVariant* result)
  407. {
  408. EINA_SAFETY_ON_NULL_RETURN_VAL(data, false);
  409. EINA_SAFETY_ON_NULL_RETURN_VAL(result, false);
  410. switch (result->type) {
  411. case NPVariantType_Void:
  412. data->type = EWK_JS_VARIANT_VOID;
  413. data->value.o = 0;
  414. break;
  415. case NPVariantType_Null:
  416. data->type = EWK_JS_VARIANT_NULL;
  417. data->value.o = 0;
  418. break;
  419. case NPVariantType_Int32:
  420. data->type = EWK_JS_VARIANT_INT32;
  421. data->value.i = NPVARIANT_TO_INT32(*result);
  422. break;
  423. case NPVariantType_Double:
  424. data->type = EWK_JS_VARIANT_DOUBLE;
  425. data->value.d = NPVARIANT_TO_DOUBLE(*result);
  426. break;
  427. case NPVariantType_String:
  428. data->value.s = eina_stringshare_add_length(NPVARIANT_TO_STRING(*result).UTF8Characters, NPVARIANT_TO_STRING(*result).UTF8Length);
  429. data->type = EWK_JS_VARIANT_STRING;
  430. break;
  431. case NPVariantType_Bool:
  432. data->type = EWK_JS_VARIANT_BOOL;
  433. data->value.b = NPVARIANT_TO_BOOLEAN(*result);
  434. break;
  435. case NPVariantType_Object:
  436. data->type = EWK_JS_VARIANT_OBJECT;
  437. data->value.o = ewk_js_npobject_to_object(NPVARIANT_TO_OBJECT(*result));
  438. break;
  439. default:
  440. return false;
  441. }
  442. return true;
  443. }
  444. #endif // ENABLE(NETSCAPE_PLUGIN_API)
  445. Ewk_JS_Object* ewk_js_object_new(const Ewk_JS_Class_Meta* jsMetaClass)
  446. {
  447. #if ENABLE(NETSCAPE_PLUGIN_API)
  448. Ewk_JS_Object* object;
  449. EINA_SAFETY_ON_NULL_RETURN_VAL(jsMetaClass, 0);
  450. object = static_cast<Ewk_JS_Object*>(malloc(sizeof(Ewk_JS_Object)));
  451. if (!object) {
  452. ERR("Could not allocate memory for ewk_js_object");
  453. return 0;
  454. }
  455. EINA_MAGIC_SET(object, EWK_JS_OBJECT_MAGIC);
  456. object->cls = ewk_js_class_new(jsMetaClass);
  457. object->view = 0;
  458. object->name = 0;
  459. object->type = EWK_JS_OBJECT_OBJECT;
  460. if (eina_hash_population(object->cls->properties) < 25)
  461. object->properties = eina_hash_string_small_new(reinterpret_cast<Eina_Free_Cb>(ewk_js_variant_free));
  462. else
  463. object->properties = eina_hash_string_superfast_new(reinterpret_cast<Eina_Free_Cb>(ewk_js_variant_free));
  464. for (int i = 0; object->cls->meta->properties && object->cls->meta->properties[i].name; i++) {
  465. Ewk_JS_Property prop = object->cls->meta->properties[i];
  466. const char* key = object->cls->meta->properties[i].name;
  467. Ewk_JS_Variant* value = static_cast<Ewk_JS_Variant*>(malloc(sizeof(Ewk_JS_Variant)));
  468. if (!value) {
  469. ERR("Could not allocate memory for ewk_js_variant");
  470. goto error;
  471. }
  472. if (prop.get)
  473. prop.get(object, key, value);
  474. else {
  475. value->type = prop.value.type;
  476. switch (value->type) {
  477. case EWK_JS_VARIANT_VOID:
  478. case EWK_JS_VARIANT_NULL:
  479. value->value.o = 0;
  480. break;
  481. case EWK_JS_VARIANT_STRING:
  482. value->value.s = eina_stringshare_add(prop.value.value.s);
  483. break;
  484. case EWK_JS_VARIANT_BOOL:
  485. value->value.b = prop.value.value.b;
  486. break;
  487. case EWK_JS_VARIANT_INT32:
  488. value->value.i = prop.value.value.i;
  489. break;
  490. case EWK_JS_VARIANT_DOUBLE:
  491. value->value.d = prop.value.value.d;
  492. break;
  493. case EWK_JS_VARIANT_OBJECT:
  494. value->value.o = prop.value.value.o;
  495. break;
  496. }
  497. }
  498. eina_hash_add(object->properties, key, value);
  499. }
  500. object->base.object.referenceCount = 1;
  501. object->base.object._class = &EWK_NPCLASS;
  502. return object;
  503. error:
  504. ewk_js_object_free(object);
  505. return 0;
  506. #else
  507. UNUSED_PARAM(jsMetaClass);
  508. return 0;
  509. #endif
  510. }
  511. void ewk_js_object_free(Ewk_JS_Object* jsObject)
  512. {
  513. #if ENABLE(NETSCAPE_PLUGIN_API)
  514. EINA_SAFETY_ON_NULL_RETURN(jsObject);
  515. EINA_MAGIC_CHECK_OR_RETURN(jsObject);
  516. Eina_Bool script_obj = !jsObject->cls->meta;
  517. eina_hash_free(jsObject->properties);
  518. eina_stringshare_del(jsObject->name);
  519. ewk_js_class_free(const_cast<Ewk_JS_Class*>(jsObject->cls));
  520. EINA_MAGIC_SET(jsObject, EINA_MAGIC_NONE);
  521. if (script_obj)
  522. free(jsObject);
  523. #else
  524. UNUSED_PARAM(jsObject);
  525. #endif
  526. }
  527. Evas_Object* ewk_js_object_view_get(const Ewk_JS_Object* jsObject)
  528. {
  529. #if ENABLE(NETSCAPE_PLUGIN_API)
  530. EINA_SAFETY_ON_NULL_RETURN_VAL(jsObject, 0);
  531. EINA_MAGIC_CHECK_OR_RETURN(jsObject, 0);
  532. return jsObject->view;
  533. #else
  534. UNUSED_PARAM(jsObject);
  535. return 0;
  536. #endif
  537. }
  538. Eina_Hash* ewk_js_object_properties_get(const Ewk_JS_Object* jsObject)
  539. {
  540. #if ENABLE(NETSCAPE_PLUGIN_API)
  541. EINA_SAFETY_ON_NULL_RETURN_VAL(jsObject, 0);
  542. EINA_MAGIC_CHECK_OR_RETURN(jsObject, 0);
  543. return jsObject->properties;
  544. #else
  545. UNUSED_PARAM(jsObject);
  546. return 0;
  547. #endif
  548. }
  549. const char* ewk_js_object_name_get(const Ewk_JS_Object* jsObject)
  550. {
  551. #if ENABLE(NETSCAPE_PLUGIN_API)
  552. EINA_SAFETY_ON_NULL_RETURN_VAL(jsObject, 0);
  553. EINA_MAGIC_CHECK_OR_RETURN(jsObject, 0);
  554. return jsObject->name;
  555. #else
  556. UNUSED_PARAM(jsObject);
  557. return 0;
  558. #endif
  559. }
  560. Eina_Bool ewk_js_object_invoke(Ewk_JS_Object* jsObject, Ewk_JS_Variant* args, int argCount, Ewk_JS_Variant* result)
  561. {
  562. #if ENABLE(NETSCAPE_PLUGIN_API)
  563. NPVariant* np_args;
  564. NPVariant np_result;
  565. bool fail = false;
  566. EINA_MAGIC_CHECK_OR_RETURN(jsObject, false);
  567. if (ewk_js_object_type_get(jsObject) != EWK_JS_OBJECT_FUNCTION)
  568. return false;
  569. if (argCount)
  570. EINA_SAFETY_ON_NULL_RETURN_VAL(args, false);
  571. np_args = static_cast<NPVariant*>(malloc(sizeof(NPVariant) *argCount));
  572. if (!np_args) {
  573. ERR("Could not allocate memory to method arguments");
  574. return false;
  575. }
  576. for (int i = 0; i < argCount; i++)
  577. if (!ewk_js_variant_to_npvariant(&args[i], &np_args[i]))
  578. goto end;
  579. if (!(fail = _NPN_InvokeDefault(0, reinterpret_cast<NPObject*>(jsObject), np_args, argCount, &np_result)))
  580. goto end;
  581. if (result)
  582. fail = ewk_js_npvariant_to_variant(result, &np_result);
  583. end:
  584. free(np_args);
  585. return fail;
  586. #else
  587. UNUSED_PARAM(jsObject);
  588. UNUSED_PARAM(args);
  589. UNUSED_PARAM(argCount);
  590. UNUSED_PARAM(result);
  591. return false;
  592. #endif
  593. }
  594. Ewk_JS_Object_Type ewk_js_object_type_get(Ewk_JS_Object* jsObject)
  595. {
  596. #if ENABLE(NETSCAPE_PLUGIN_API)
  597. EINA_SAFETY_ON_NULL_RETURN_VAL(jsObject, EWK_JS_OBJECT_OBJECT);
  598. EINA_MAGIC_CHECK_OR_RETURN(jsObject, EWK_JS_OBJECT_OBJECT);
  599. return jsObject->type;
  600. #else
  601. UNUSED_PARAM(jsObject);
  602. return EWK_JS_OBJECT_INVALID;
  603. #endif
  604. }
  605. void ewk_js_object_type_set(Ewk_JS_Object* jsObject, Ewk_JS_Object_Type type)
  606. {
  607. #if ENABLE(NETSCAPE_PLUGIN_API)
  608. EINA_SAFETY_ON_NULL_RETURN(jsObject);
  609. EINA_MAGIC_CHECK_OR_RETURN(jsObject);
  610. jsObject->type = type;
  611. #else
  612. UNUSED_PARAM(jsObject);
  613. UNUSED_PARAM(type);
  614. #endif
  615. }
  616. void ewk_js_variant_free(Ewk_JS_Variant* jsVariant)
  617. {
  618. #if ENABLE(NETSCAPE_PLUGIN_API)
  619. EINA_SAFETY_ON_NULL_RETURN(jsVariant);
  620. if (jsVariant->type == EWK_JS_VARIANT_STRING)
  621. eina_stringshare_del(jsVariant->value.s);
  622. else if (jsVariant->type == EWK_JS_VARIANT_OBJECT)
  623. ewk_js_object_free(jsVariant->value.o);
  624. free(jsVariant);
  625. #else
  626. UNUSED_PARAM(jsVariant);
  627. #endif
  628. }
  629. void ewk_js_variant_array_free(Ewk_JS_Variant* jsVariant, int count)
  630. {
  631. #if ENABLE(NETSCAPE_PLUGIN_API)
  632. EINA_SAFETY_ON_NULL_RETURN(jsVariant);
  633. for (int i = 0; i < count; i++) {
  634. if (jsVariant[i].type == EWK_JS_VARIANT_STRING)
  635. eina_stringshare_del(jsVariant[i].value.s);
  636. else if (jsVariant[i].type == EWK_JS_VARIANT_OBJECT)
  637. ewk_js_object_free(jsVariant[i].value.o);
  638. }
  639. free(jsVariant);
  640. #else
  641. UNUSED_PARAM(jsVariant);
  642. UNUSED_PARAM(count);
  643. #endif
  644. }