Merge "Retrieves character's directions." into new_text
[platform/core/uifw/dali-toolkit.git] / plugins / dali-script-v8 / src / actors / actor-wrapper.cpp
1 /*
2  * Copyright (c) 2015 Samsung Electronics Co., Ltd.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  *
16  */
17
18 // CLASS HEADER
19 #include "actor-wrapper.h"
20
21 // EXTERNAL INCLUDES
22 #include <dali/public-api/object/type-registry.h>
23
24 // INTERNAL INCLUDES
25 #include <actors/layer-api.h>
26 #include <actors/actor-api.h>
27 #include <actors/image-actor-api.h>
28 #include <actors/mesh-actor-api.h>
29 #include <actors/camera-actor-api.h>
30 #include <actors/renderable-actor-api.h>
31 #include <v8-utils.h>
32 #include <dali-wrapper.h>
33
34 namespace Dali
35 {
36
37 namespace V8Plugin
38 {
39
40 v8::Persistent<v8::ObjectTemplate> ActorWrapper::mActorTemplate;
41 v8::Persistent<v8::ObjectTemplate> ActorWrapper::mImageActorTemplate;
42 v8::Persistent<v8::ObjectTemplate> ActorWrapper::mMeshActorTemplate;
43 v8::Persistent<v8::ObjectTemplate> ActorWrapper::mCameraActorTemplate;
44 v8::Persistent<v8::ObjectTemplate> ActorWrapper::mLayerActorTemplate;
45 v8::Persistent<v8::ObjectTemplate> ActorWrapper::mTextLabelTemplate;
46
47 namespace
48 {
49
50
51 /**
52  * pointer to a persistent template handle
53  */
54 struct ActorTemplate
55 {
56   v8::Persistent<v8::ObjectTemplate>* actorTemplate;
57 };
58
59 /**
60  * array of templates for each type of actor
61  */
62 const ActorTemplate ActorTemplateLookup[]=
63 {
64     { &ActorWrapper::mActorTemplate },        // ACTOR
65     { &ActorWrapper::mImageActorTemplate },   // IMAGE_ACTOR
66     { &ActorWrapper::mMeshActorTemplate  },   // MESH_ACTOR
67     { &ActorWrapper::mLayerActorTemplate },   // LAYER_ACTOR
68     { &ActorWrapper::mCameraActorTemplate},   // CAMERA_ACTOR
69     { &ActorWrapper::mTextLabelTemplate }
70 };
71
72 /**
73  * Bitmask of API's that an actor can support
74  */
75 enum ActorApiBitMask
76 {
77   ACTOR_API              = 1 << 0,
78   RENDERABLE_ACTOR_API   = 1 << 1,
79   IMAGE_ACTOR_API        = 1 << 2,
80   MESH_ACTOR_API         = 1 << 3,
81   LAYER_API              = 1 << 4,
82   CAMERA_ACTOR_API       = 1 << 5,
83 };
84
85 /**
86  * structure used for the ActorApiLookup.
87  */
88 struct ActorApiStruct
89 {
90   const char* actorName;
91   ActorWrapper::ActorType actorType;
92   Actor (*constructor)( const v8::FunctionCallbackInfo< v8::Value >& args);
93   int supportApis;
94 };
95
96 /**
97  * Lookup table to match a actor type with a constructor and supported API's.
98  */
99 const ActorApiStruct ActorApiLookup[]=
100 {
101   {"Actor",      ActorWrapper::ACTOR,        ActorApi::New,       ACTOR_API },
102   {"ImageActor", ActorWrapper::IMAGE_ACTOR,  ImageActorApi::New,  ACTOR_API | RENDERABLE_ACTOR_API | IMAGE_ACTOR_API   },
103   {"MeshActor",  ActorWrapper::MESH_ACTOR,   MeshActorApi::New,   ACTOR_API | RENDERABLE_ACTOR_API | MESH_ACTOR_API    },
104   {"Layer",      ActorWrapper::LAYER_ACTOR,  LayerApi::New,       ACTOR_API | LAYER_API                                },
105   {"CameraActor",ActorWrapper::CAMERA_ACTOR, CameraActorApi::New, ACTOR_API | CAMERA_ACTOR_API                         },
106   {"TextLabel",  ActorWrapper::TEXT_LABEL,   TextLabelApi::New,   ACTOR_API },
107
108 };
109
110 const unsigned int ActorApiLookupCount = sizeof(ActorApiLookup)/sizeof(ActorApiLookup[0]);
111
112
113
114 /**
115  * Creates an actor given a type name
116  * Uses the type registry to create an actor of the correct type
117  */
118 Actor CreateActor( const v8::FunctionCallbackInfo< v8::Value >& args,
119                         const std::string& typeName )
120 {
121   Actor actor;
122
123   ActorWrapper::ActorType actorType = ActorWrapper::GetActorType( typeName );
124
125   // if we don't currently support the actor type, then use type registry to create it
126   if( actorType == ActorWrapper::UNKNOWN_ACTOR )
127   {
128     Dali::TypeInfo typeInfo = Dali::TypeRegistry::Get().GetTypeInfo( typeName );
129     if( typeInfo ) // handle, check if it has a value
130     {
131       Dali::BaseHandle handle = typeInfo.CreateInstance();
132       if( handle )
133       {
134         actor = Actor::DownCast( handle );
135       }
136     }
137     else
138     {
139       DALI_SCRIPT_EXCEPTION(args.GetIsolate(),"Unknown actor type");
140       return Actor();
141     }
142   }
143   else
144   {
145     // run the constructor for this type of actor so it can pull out
146     // custom parameters, e.g. new ImageActor( MyImage );
147     actor = (ActorApiLookup[actorType].constructor)( args );
148   }
149   return actor;
150 }
151
152
153
154 /**
155  * given an actor type return what api's it supports
156  */
157 int GetActorSupportedApis( ActorWrapper::ActorType type )
158 {
159   return ActorApiLookup[ type].supportApis;
160 }
161
162 /**
163  * Used for the ActorFunctionTable to map function names to functions
164  * with for a specific API
165  */
166 struct ActorFunctions
167 {
168   const char* name;               ///< function name
169   void (*function)( const v8::FunctionCallbackInfo< v8::Value >& args);
170   ActorApiBitMask api;
171 };
172
173 /**
174  * Contains a list of all functions that can be called an
175  * actor / image-actor / mesh-actor/ layer / camera-actor
176  */
177 const ActorFunctions ActorFunctionTable[]=
178 {
179     /**************************************
180     * Actor API (in order of actor.h)
181     * Any properties that have accessor functions are ignored to avoid duplication
182     **************************************/
183     // ignore. GetName()  use Actor.name
184     // ignore. SetName()  use Actor.name
185     { "GetId",             ActorApi::GetId,            ACTOR_API },
186     { "IsRoot",            ActorApi::IsRoot,           ACTOR_API },
187     { "OnStage",           ActorApi::OnStage,          ACTOR_API },
188     { "IsLayer",           ActorApi::IsLayer,          ACTOR_API },
189     { "GetLayer",          ActorApi::GetLayer,         ACTOR_API },
190     { "Add",               ActorApi::AddActor,         ACTOR_API },
191     { "Remove",            ActorApi::RemoveActor,      ACTOR_API },
192     { "IsEqualTo" ,        ActorApi::IsEqualTo,        ACTOR_API },
193     { "Unparent",          ActorApi::Unparent,         ACTOR_API },
194     { "GetChildCount",     ActorApi::GetChildCount,    ACTOR_API },
195     { "GetChildAt"   ,     ActorApi::GetChildAt,       ACTOR_API },
196     { "FindChildByName",   ActorApi::FindChildByName,  ACTOR_API },
197     { "FindChildById",     ActorApi::FindChildById,    ACTOR_API },
198     { "GetParent" ,        ActorApi::GetParent,        ACTOR_API },
199     { "GetActorType" ,     ActorApi::GetActorType,     ACTOR_API }, // custom for javascript
200
201     // ignore. SetParentOrigin() use Actor.parentOrigin
202     // ignore. GetCurrentParentOrigin()  use Actor.parentOrigin
203     // ignore. SetAnchorPoint()  use Actor.anchorPoint
204     // ignore. GetCurrentAnchorPoint()  use Actor.anchorPoint
205     // ignore. SetSize() use Actor.size
206     // ignore. GetCurrentSize() use Actor.size
207     // ignore. SetPosition(....) use Actor.position
208     // ignore. SetX, SetY, SetZ,  use Actor.position.x, Actor.position.y, Actor.position.z
209     { "TranslateBy",         ActorApi::TranslateBy,              ACTOR_API },
210     // ignore GetCurrentPosition(). use Actor.position
211     // ignore GetCurrentWorldPosition() use Actor.worldPosition
212     // ignore SetPositionInheritanceMode() use Actor.positionInheritance
213     // ignore GetPositionInheritanceMode()  use Actor.positionInheritance
214     // ignore SetOrientation() use Actor.orientation
215     { "RotateBy",         ActorApi::RotateBy,          ACTOR_API },
216     // ignore GetCurrentOrientation() use Actor.orientation
217     // ignore SetInheritOrientation() use Actor.inheritOrientation
218     // ignore IsOrientationInherited() use Actor.inheritOrientation
219     // ignore GetCurrentWorldOrientation() use Actor.worldOrientation
220     // ignore SetScale() use Actor.scale
221     { "ScaleBy",         ActorApi::ScaleBy,            ACTOR_API },
222     // ignore GetCurrentScale() use Actor.scale
223     // ignore GetCurrentWorldScale() use Actor.worldScale
224     // ignore SetInheritScale() use Actor.inheritScale
225     // ignore IsScaleInherited() use Actor.inheritScale
226     // ignore GetCurrentWorldMatrix() use Actor.worldMatrix
227     // ignore SetVisible() use Actor.visible
228     // ignore IsVisible() use Actor.visible
229     // ignore SetOpacity() use Actor.opacity
230     // ignore GetCurrentOpacity() use Actor.opacity
231     // ignore SetColor() use Actor.color
232     // ignore GetCurrentColor() use Actor.color
233     // ignore SetColorMode() use Actor.colorMode
234     // ignore GetColorMode() use Actor.colorMode
235     // ignore GetCurrentWorldColor() use Actor.worldColor
236     // ignore SetInheritShaderEffect() use Actor.inheritShaderEffect
237     // ignore GetInheritShaderEffect() use Actor.inheritShaderEffect
238     // ignore SetDrawMode() use Actor.drawMode
239     // ignore GetDrawMode() use Actor.drawMode
240     // ignore SetSensitive() use Actor.sensitve
241     // ignore IsSensitive() use Actor.sensitive
242     { "ScreenToLocal"       , ActorApi::ScreenToLocal,         ACTOR_API},
243     // ignore SetLeaveRequired() use Actor.leaveRequired
244     // ignore GetLeaveRequired() use Actor.leaveRequired
245     { "SetKeyboardFocusable", ActorApi::SetKeyboardFocusable,  ACTOR_API }, //-- should this be a property???
246     { "IsKeyboardFocusable" , ActorApi::IsKeyboardFocusable,   ACTOR_API }, //-- should this be a property???
247
248     /**************************************
249      * Renderable Actor API (in order of renderable-actor.h)
250      **************************************/
251     { "SetSortModifier",    RenderableActorApi::SetSortModifier,   RENDERABLE_ACTOR_API  },
252     { "GetSortModifier",    RenderableActorApi::GetSortModifier,   RENDERABLE_ACTOR_API  },
253     { "SetCullFace",        RenderableActorApi::SetCullFace,       RENDERABLE_ACTOR_API  },
254     { "GetCullFace",        RenderableActorApi::GetCullFace,       RENDERABLE_ACTOR_API  },
255     { "SetBlendMode",       RenderableActorApi::SetBlendMode,      RENDERABLE_ACTOR_API  },
256     { "GetBlendMode",       RenderableActorApi::GetBlendMode,      RENDERABLE_ACTOR_API  },
257     { "SetBlendFunc",       RenderableActorApi::SetBlendFunc,      RENDERABLE_ACTOR_API  },
258     { "GetBlendFunc",       RenderableActorApi::GetBlendFunc,      RENDERABLE_ACTOR_API  },
259     { "SetBlendEquation",   RenderableActorApi::SetBlendEquation,  RENDERABLE_ACTOR_API  },
260     { "GetBlendEquation",   RenderableActorApi::GetBlendEquation,  RENDERABLE_ACTOR_API  },
261     { "SetBlendColor",      RenderableActorApi::SetBlendColor,     RENDERABLE_ACTOR_API  },
262     { "GetBlendColor",      RenderableActorApi::GetBlendColor,     RENDERABLE_ACTOR_API  },
263     { "SetShaderEffect",    RenderableActorApi::SetShaderEffect,   RENDERABLE_ACTOR_API  },
264     { "GetShaderEffect",    RenderableActorApi::GetShaderEffect,   RENDERABLE_ACTOR_API  },
265     { "RemoveShaderEffect", RenderableActorApi::RemoveShaderEffect,RENDERABLE_ACTOR_API  },
266
267
268
269
270     /**************************************
271      * Layer  API (in order of layer.h)
272      **************************************/
273     { "GetDepth",           LayerApi::GetDepth,                 LAYER_API  },
274     { "Raise",              LayerApi::Raise,                    LAYER_API  },
275     { "Lower",              LayerApi::Lower,                    LAYER_API  },
276     { "RaiseAbove",         LayerApi::RaiseAbove,               LAYER_API  },
277     { "RaiseBelow",         LayerApi::LowerBelow,               LAYER_API  },
278     { "RaiseToTop",         LayerApi::RaiseToTop,               LAYER_API  },
279     { "LowerToBottom",      LayerApi::ToBottom,                 LAYER_API  },
280     { "MoveAbove",          LayerApi::MoveAbove,                LAYER_API  },
281     { "MoveBelow",          LayerApi::MoveBelow,                LAYER_API  },
282     // ignore SetClipping, use layer.clippingEnable
283     // ignore IsClipping, use layer.clippingEnable
284     // ignore SetClippingBox, use layer.clippingBox
285     { "SetDepthTestDisabled", LayerApi::SetDepthTestDisabled,   LAYER_API },
286     { "IsDepthTestDisabled",  LayerApi::IsDepthTestDisabled,    LAYER_API },
287     // @todo SetSortFunction
288
289     /**************************************
290      * Image Actor API (in order of image-actor.h)
291      **************************************/
292
293     { "SetImage",           ImageActorApi::SetImage,              IMAGE_ACTOR_API },
294     { "GetImage",           ImageActorApi::GetImage,              IMAGE_ACTOR_API },
295     { "SetToNaturalSize",   ImageActorApi::SetToNaturalSize,      IMAGE_ACTOR_API },
296     // ignore SetPixelArea, use imageActor.pixelArea
297     // ignore GetPixelArea, use imageActor.pixelArea
298     { "IsPixelAreaSet",     ImageActorApi::IsPixelAreaSet,        IMAGE_ACTOR_API },
299     { "ClearPixelArea",     ImageActorApi::ClearPixelArea,        IMAGE_ACTOR_API },
300     // ignore SetStyle, use imageActor.style
301     // ignore GetStyle, use imageActor.style
302     // ignore SetNinePatchBorder use imageActor.border
303     // ignore GetNinePatchBorder use imageActor.border
304     // ignore SetFadeIn use imageActor.fadeIn
305     // ignore GetFadeIn use imageActor.fadeIn
306     // ignore SetFadeInDuration use imageActor.fadeInDuration
307     // ignore GetFadeInDuration use imageActor.fadeInDuration
308     //{ "GetCurrentImageSize", ImageActorApi::GetCurrentImageSize,  IMAGE_ACTOR_API },
309
310     /**************************************
311      * Mesh Actor API (in order of mesh-actor.h)
312      **************************************/
313     // @todo a version of MeshActor::New( mesh )
314     // @todo a version of MeshActor::New( AnimatableMesh )
315     // @todo SetMaterial
316     // @todo GetMaterial
317     // @todo BindBonesToMesh
318
319     /**************************************
320      * Camera Actor API (in order of camera.h)
321      **************************************/
322     // ignore SetType use camera.type
323     // ignore GetType use camera.type
324     // ignore SetProjectionMode use camera.projectionMode
325     // ignore GetProjectionMode use camera.projectionMode
326     // ignore SetFieldOfView use camera.fieldOfView
327     // ignore GetFieldOfView use camera.fieldOfView
328     // ignore SetAspectRatio use camera.aspectRatio
329     // ignore GetAspectRatio use camera.aspectRatio
330     // ignore SetNearClippingPlane use camera.nearPlaneDistance
331     // ignore GetNearClippingPlane use camera.nearPlaneDistance
332     // ignore SetFarClippingPlane use camera.farPlaneDistance
333     // ignore GetFarClippingPlane use camera.farPlaneDistance
334     // ignore GetTargetPosition use camera.targetPosition
335     // ignore SetInvertYAxis use camera.invertYAxis
336     // ignore GetInvertYAxis use camera.invertYAxis
337     { "SetPerspectiveProjection",   CameraActorApi::SetPerspectiveProjection,   CAMERA_ACTOR_API },
338     { "SetOrthographicProjection",  CameraActorApi::SetOrthographicProjection,  CAMERA_ACTOR_API },
339
340 };
341
342 const unsigned int ActorFunctionTableCount = sizeof(ActorFunctionTable)/sizeof(ActorFunctionTable[0]);
343 } //un-named space
344
345
346 ActorWrapper::ActorWrapper( Actor actor,
347               GarbageCollectorInterface& gc )
348 : HandleWrapper( BaseWrappedObject::ACTOR , actor, gc ),
349   mActor( actor )
350
351 {
352 }
353
354 v8::Handle<v8::Object> ActorWrapper::WrapActor(v8::Isolate* isolate, Actor actor )
355 {
356   v8::EscapableHandleScope handleScope( isolate );
357   v8::Local<v8::Object> object = WrapActor( isolate, actor, GetActorType( actor.GetTypeName() ) );
358
359   return handleScope.Escape( object );
360 }
361
362 Actor ActorWrapper::GetActor()
363 {
364   return mActor;
365 }
366
367 v8::Handle<v8::Object> ActorWrapper::WrapActor( v8::Isolate* isolate, Actor actor, ActorType actorType )
368 {
369   v8::EscapableHandleScope handleScope( isolate );
370   v8::Local<v8::ObjectTemplate> objectTemplate;
371
372   objectTemplate = GetActorTemplate( isolate, actorType );
373
374   // create an instance of the template
375   v8::Local<v8::Object> localObject = objectTemplate->NewInstance();
376
377   // create teh actor object
378   ActorWrapper* pointer = new ActorWrapper( actor, Dali::V8Plugin::DaliWrapper::Get().GetDaliGarbageCollector() );
379
380   // assign the JavaScript object to the wrapper.
381   // This also stores Dali object, in an internal field inside the JavaScript object.
382   pointer->SetJavascriptObject( isolate, localObject );
383
384   return handleScope.Escape( localObject );
385 }
386
387 v8::Local<v8::ObjectTemplate> ActorWrapper::GetActorTemplate( v8::Isolate* isolate, ActorWrapper::ActorType type )
388 {
389   v8::EscapableHandleScope handleScope( isolate );
390   v8::Local<v8::ObjectTemplate> objectTemplate;
391
392   if( ActorTemplateLookup[type].actorTemplate->IsEmpty() )
393   {
394     objectTemplate = MakeDaliActorTemplate( isolate, type );
395     ActorTemplateLookup[type].actorTemplate->Reset( isolate, objectTemplate );
396   }
397   else
398   {
399     // get the object template
400     objectTemplate = v8::Local<v8::ObjectTemplate>::New( isolate, *ActorTemplateLookup[type].actorTemplate );
401   }
402
403   return handleScope.Escape( objectTemplate );
404 }
405
406 v8::Handle<v8::ObjectTemplate> ActorWrapper::MakeDaliActorTemplate( v8::Isolate* isolate, ActorType actorType )
407 {
408   v8::EscapableHandleScope handleScope( isolate );
409
410   v8::Local<v8::ObjectTemplate> objTemplate = v8::ObjectTemplate::New();
411
412   objTemplate->SetInternalFieldCount( BaseWrappedObject::FIELD_COUNT );
413
414   // find out what API's this actor supports
415   int supportApis = GetActorSupportedApis( actorType );
416
417   // add our function properties
418   for( unsigned int i = 0; i < ActorFunctionTableCount; ++i )
419   {
420     const ActorFunctions property =  ActorFunctionTable[i];
421
422     // check to see if the actor supports a certain type of API
423     // e.g. ImageActor will support ACTOR_API, RENDERABLE_API and IMAGE_ACTOR_API
424     if( supportApis &  property.api )
425     {
426       std::string funcName = V8Utils::GetJavaScriptFunctionName( property.name);
427
428       objTemplate->Set( v8::String::NewFromUtf8(   isolate, funcName.c_str() ),
429                       v8::FunctionTemplate::New( isolate, property.function ) );
430     }
431   }
432
433   // property handle intercepts property getters and setters and signals
434   HandleWrapper::AddInterceptsToTemplate( isolate, objTemplate );
435
436
437   return handleScope.Escape( objTemplate );
438 }
439
440 void ActorWrapper::NewActor( const v8::FunctionCallbackInfo< v8::Value >& args)
441 {
442   v8::Isolate* isolate = args.GetIsolate();
443   v8::HandleScope handleScope( isolate );
444
445   if( !args.IsConstructCall() )
446   {
447     DALI_SCRIPT_EXCEPTION( isolate, "constructor called without 'new" );
448     return;
449   }
450
451   // find out the callee function name...e.g. ImageActor, MeshActor
452   v8::Local<v8::Function> callee = args.Callee();
453   v8::Local<v8::Value> v8String = callee->GetName();
454   std::string typeName = V8Utils::v8StringToStdString( v8String );
455
456   // create a new actor based on type, using the type registry.
457   Actor actor = CreateActor( args, typeName );
458
459   v8::Local<v8::Object> localObject = WrapActor( isolate, actor );
460
461   args.GetReturnValue().Set( localObject );
462 }
463
464 /**
465  * given an actor type name, e.g. ImageActor returns the type, e.g. ActorWrapper::IMAGE_ACTOR
466  */
467 ActorWrapper::ActorType ActorWrapper::GetActorType( const std::string& name )
468 {
469   for( unsigned int i = 0 ; i < ActorApiLookupCount ; i++ )
470   {
471     if( ActorApiLookup[i].actorName == name )
472     {
473       return ActorApiLookup[i].actorType;
474     }
475   }
476   return ActorWrapper::UNKNOWN_ACTOR;
477 }
478
479
480
481 } // namespace V8Plugin
482
483 } // namespace Dali