source: cpp/frams/vm/classes/3dobject.cpp @ 151

Last change on this file since 151 was 127, checked in by sz, 11 years ago

bug fixed: all param items having GETONLY accessor must also have PARAM_READONLY flag

  • Property svn:eol-style set to native
File size: 16.3 KB
Line 
1// This file is a part of the Framsticks GDK.
2// Copyright (C) 2002-2014  Maciej Komosinski and Szymon Ulatowski.  See LICENSE.txt for details.
3// Refer to http://www.framsticks.com/ for further information.
4
5#include <frams/util/3d.h>
6#include "3dobject.h"
7#include <frams/param/param.h>
8#include "collectionobj.h"
9
10ParamEntry* Pt3D_Ext::getStaticParamtab()
11{
12#define FIELDSTRUCT Pt3D_Ext
13static ParamEntry paramtab[]=
14{
15{"XYZ",1,18,"XYZ","3D vector"},
16
17{"x",0,PARAM_NOSTATIC,"x","f",FIELD(p.x),},
18{"y",0,PARAM_NOSTATIC,"y","f",FIELD(p.y),},
19{"z",0,PARAM_NOSTATIC,"z","f",FIELD(p.z),},
20{"new",0,0,"create new XYZ object","p oXYZ(f x,f y,f z)",PROCEDURE(p_new),"3D vectors objects can be also created using the (x,y,z) notation, i.e. var v=(1,2,3) is the same as var v=XYZ.new(1,2,3);",},
21{"newFromVector",0,0,"create new XYZ object","p oXYZ(oVector)",PROCEDURE(p_newFromVector),"used for deserialization"},
22{"clone",0,PARAM_NOSTATIC,"create new XYZ object copying the coordinates","p oXYZ()",PROCEDURE(p_clone),"Note: copying object references does not create new objects. Use clone() if a new object is needed.\n\nExample:\nvar o1=(1,2,3), o2=o1, o3=o1.clone();\no1.y=9999;\n//o2 is now (1,9999,3) but o3 is still (1,2,3)",},
23{"set",0,PARAM_NOSTATIC,"set (copy) coordinates from another XYZ object","p(oXYZ)",PROCEDURE(p_set),},
24{"set3",0,PARAM_NOSTATIC,"set individual 3 coordinates","p(f x,f y,f z)",PROCEDURE(p_set3),},
25{"add",0,PARAM_NOSTATIC,"add","p(oXYZ)",PROCEDURE(p_addvec),"Note: it does not return a new object, just modifies the existing one"},
26{"sub",0,PARAM_NOSTATIC,"subtract","p(oXYZ)",PROCEDURE(p_subvec),"Note: it does not return a new object, just modifies the existing one"},
27{"scale",0,PARAM_NOSTATIC,"multiply by scalar","p(f)",PROCEDURE(p_scale),},
28{"length",0,PARAM_READONLY+PARAM_NOSTATIC,"length","f",GETONLY(length),},
29{"normalize",0,PARAM_NOSTATIC,"normalize","p()",PROCEDURE(p_normalize),"scales the vector length to 1.0"},
30{"toString",0,PARAM_READONLY+PARAM_NOSTATIC,"textual form","s",GETONLY(toString),},
31{"toVector",0,PARAM_READONLY+PARAM_NOSTATIC,"vector of [x,y,z]","oVector",GETONLY(toVector),},
32{"rotate",0,PARAM_NOSTATIC,"rotate using Orient object","p(oOrient)",PROCEDURE(p_rotate),},
33{"revRotate",0,PARAM_NOSTATIC,"reverse rotate using Orient object","p(oOrient)",PROCEDURE(p_revrotate),},
34{"get",0,PARAM_NOSTATIC,"get one of coordinates","p f(d index)",PROCEDURE(p_get),"this function makes the XYZ objects \"indexable\" (so you can use [] for accessing subsequent fields, like in Vector)",},
35{0,0,0,},
36};
37#undef FIELDSTRUCT
38return paramtab;
39}
40
41void Pt3D_Ext::p_new(ExtValue *args,ExtValue *ret)
42{
43*ret=makeDynamicObject(new Pt3D_Ext(args[2].getDouble(),args[1].getDouble(),args[0].getDouble()));
44}
45
46static double doubleFromVec(VectorObject *vec,int i)
47{
48if (i>=vec->data.size()) return 0;
49ExtValue *v=(ExtValue*)vec->data.get(i);
50if (v)
51        return v->getDouble();
52return 0;
53}
54
55static Pt3D pt3DFromVec(VectorObject* v,int offset=0)
56{
57return Pt3D(doubleFromVec(v,offset),doubleFromVec(v,offset+1),doubleFromVec(v,offset+2));
58}
59
60void Pt3D_Ext::p_newFromVector(ExtValue *args,ExtValue *ret)
61{
62VectorObject *vec=VectorObject::fromObject(args->getObject());
63if (vec)
64        *ret=makeDynamicObject(new Pt3D_Ext(pt3DFromVec(vec)));
65else
66        ret->setEmpty();
67}
68
69void Pt3D_Ext::p_clone(ExtValue *args,ExtValue *ret)
70{
71*ret=makeDynamicObject(new Pt3D_Ext(p.x,p.y,p.z));
72}
73
74void Pt3D_Ext::p_set3(ExtValue *args,ExtValue *ret)
75{
76p.x=args[2].getDouble();
77p.y=args[1].getDouble();
78p.z=args[0].getDouble();
79ret->setEmpty();
80}
81
82void Pt3D_Ext::p_set(ExtValue *args,ExtValue *ret)
83{
84Pt3D_Ext *other=fromObject(args[0]);
85if (other)
86        p=other->p;
87ret->setEmpty();
88}
89
90void Pt3D_Ext::get_length(ExtValue *ret)
91{
92ret->setDouble(p.length());
93}
94
95void Pt3D_Ext::get_toString(ExtValue *ret)
96{
97SString s="(";
98ExtValue v;
99v.setDouble(p.x); s+=v.getString();
100s+=",";
101v.setDouble(p.y); s+=v.getString();
102s+=",";
103v.setDouble(p.z); s+=v.getString();
104s+=")";
105ret->setString(s);
106}
107
108static void add3Coords(VectorObject* vec,const Pt3D& p)
109{
110vec->data+=new ExtValue(p.x);
111vec->data+=new ExtValue(p.y);
112vec->data+=new ExtValue(p.z);
113}
114
115void Pt3D_Ext::get_toVector(ExtValue *ret)
116{
117VectorObject *vec=new VectorObject;
118add3Coords(vec,p);
119ret->setObject(ExtObject(&VectorObject::par,vec));
120}
121
122void Pt3D_Ext::p_addvec(ExtValue *args,ExtValue *ret)
123{
124Pt3D_Ext *other=fromObject(args[0]);
125if (other)
126        p+=other->p;
127ret->setEmpty();
128}
129
130void Pt3D_Ext::p_subvec(ExtValue *args,ExtValue *ret)
131{
132Pt3D_Ext *other=fromObject(args[0]);
133if (other)
134        p-=other->p;
135ret->setEmpty();
136}
137
138void Pt3D_Ext::p_scale(ExtValue *args,ExtValue *ret)
139{
140double d=args[0].getDouble();
141p.x*=d; p.y*=d; p.z*=d;
142ret->setEmpty();
143}
144
145void Pt3D_Ext::p_normalize(ExtValue *args,ExtValue *ret)
146{
147p.normalize();
148ret->setEmpty();
149}
150
151void Pt3D_Ext::p_rotate(ExtValue *args,ExtValue *ret)
152{
153Orient_Ext *o=Orient_Ext::fromObject(args[0]);
154if (o)
155        {
156        Pt3D tmp=p;
157        o->o.transform(p,tmp);
158        }
159ret->setEmpty();
160}
161
162void Pt3D_Ext::p_revrotate(ExtValue *args,ExtValue *ret)
163{
164Orient_Ext *o=Orient_Ext::fromObject(args[0]);
165if (o)
166        {
167        Pt3D tmp=p;
168        o->o.revTransform(p,tmp);
169        }
170ret->setEmpty();
171}
172
173void Pt3D_Ext::p_get(ExtValue *args,ExtValue *ret)
174{
175int index=args->getInt();
176if ((index<0)||(index>2))
177        ret->setEmpty();
178else
179        ret->setDouble((&p.x)[index]);
180}
181
182Param& Pt3D_Ext::getStaticParam()
183{
184#ifdef __CODEGUARD__
185static Pt3D_Ext static_pt3dobj;
186static Param static_pt3dparam(getStaticParamtab(),&static_pt3dobj);
187#else
188static Param static_pt3dparam(getStaticParamtab());
189#endif
190return static_pt3dparam;
191}
192
193Pt3D_Ext* Pt3D_Ext::fromObject(const ExtValue& v)
194{
195if (v.type!=TObj) return 0;
196const ExtObject& o=v.getObject();
197return (Pt3D_Ext*)o.getTarget(getStaticParam().getName());
198}
199
200ParamInterface* Pt3D_Ext::getInterface() {return &getStaticParam();}
201
202ExtObject Pt3D_Ext::makeStaticObject(Pt3D* p)
203{return ExtObject(&getStaticParam(),((char*)p)+(((char*)&p->x)-((char*)&((Pt3D_Ext*)p)->p.x)));}
204
205ExtObject Pt3D_Ext::makeDynamicObject(Pt3D_Ext* p)
206{return ExtObject(&getStaticParam(),p);}
207
208//////////////////////////////////////
209
210ParamEntry* Orient_Ext::getStaticParamtab()
211{
212#define FIELDSTRUCT Orient_Ext
213static ParamEntry paramtab[]=
214{
215{"Orient",1,27,"Orient","3D orientation, stored as 3x3 matrix."},
216
217{"xx",1,PARAM_NOSTATIC,"orientation.x.x","f",FIELD(o.x.x),},
218{"xy",1,PARAM_NOSTATIC,"orientation.x.y","f",FIELD(o.x.y),},
219{"xz",1,PARAM_NOSTATIC,"orientation.x.z","f",FIELD(o.x.z),},
220{"yx",1,PARAM_NOSTATIC,"orientation.y.x","f",FIELD(o.y.x),},
221{"yy",1,PARAM_NOSTATIC,"orientation.y.y","f",FIELD(o.y.y),},
222{"yz",1,PARAM_NOSTATIC,"orientation.y.z","f",FIELD(o.y.z),},
223{"zx",1,PARAM_NOSTATIC,"orientation.z.x","f",FIELD(o.z.x),},
224{"zy",1,PARAM_NOSTATIC,"orientation.z.y","f",FIELD(o.z.y),},
225{"zz",1,PARAM_NOSTATIC,"orientation.z.z","f",FIELD(o.z.z),},
226
227{"x",0,PARAM_NOSTATIC+PARAM_READONLY,"x vector","oXYZ",GETONLY(x),},
228{"y",0,PARAM_NOSTATIC+PARAM_READONLY,"y vector","oXYZ",GETONLY(y),},
229{"z",0,PARAM_NOSTATIC+PARAM_READONLY,"z vector","oXYZ",GETONLY(z),},
230
231{"new",0,0,"create new Orient object","p oOrient()",PROCEDURE(p_new),},
232{"newFromVector",0,0,"create new Orient object","p oOrient(oVector)",PROCEDURE(p_newFromVector),},
233{"toVector",0,PARAM_READONLY+PARAM_NOSTATIC,"vector representation","oVector",GETONLY(toVector),"for serialization"},
234{"clone",0,PARAM_NOSTATIC,"create new Orient object","p oOrient()",PROCEDURE(p_clone),},
235{"set",0,PARAM_NOSTATIC,"copy from another Orient object","p(oOrient)",PROCEDURE(p_set),},
236{"reset",0,PARAM_NOSTATIC,"set identity matrix","p()",PROCEDURE(p_reset),},
237{"rotate3",0,PARAM_NOSTATIC,"rotate around 3 axes","p(f x,f y,f z)",PROCEDURE(p_rotate3),},
238{"rotate",0,PARAM_NOSTATIC,"rotate using Orient object","p(oOrient)",PROCEDURE(p_rotate),},
239{"revRotate",0,PARAM_NOSTATIC,"reverse rotate using Orient object","p(oOrient)",PROCEDURE(p_revrotate),},
240{"lookAt",0,PARAM_NOSTATIC,"calculate rotation from 2 vectors","p(oXYZ direction,oXYZ up)",PROCEDURE(p_lookat),},
241{"normalize",0,PARAM_NOSTATIC,"normalize","p()",PROCEDURE(p_normalize),},
242{"between2",0,PARAM_NOSTATIC,"interpolate orientation","p(oOrient,oOrient,f amount)",PROCEDURE(p_between2),"The calling Orient receives the orientation interpolated from 2 input orientations.\nExample:\n"
243        "var o1=Orient.new(), o2=Orient.new(), o3=Orient.new();\n"
244        "o2.rotate3(0,Math.pi/2,0);\n"
245        "o3.between2(o1,o2,0); // o3 equals o2\n"
246        "o3.between2(o1,o2,1); // o3 equals o1\n"
247        "o3.between2(o1,o2,0.5); // o3 is halfway between o1 and o2\n"},
248{"betweenOV",0,PARAM_NOSTATIC,"interpolate orientation","p(oOrient,oXYZ,f amount)",PROCEDURE(p_betweenOV),"Like between2(), but the second Orient is composed of the supplied XYZ vector (X component) and Y Z vectors from the calling object.\n"
249        "Example:\n"
250        "var o=Orient.new();\n"
251        "o.betweenOV(o,(0,1,0),1); //no change, o remains 100 010 001\n"
252        "o.betweenOV(o,(0,1,0),0.9); //o is slightly rotated towards (0,1,0)\n"
253        "o.betweenOV(o,(0,1,0),0); //o is completely transformed, o.x=(0,1,0)\n"
254        },
255{"localToWorld",0,PARAM_NOSTATIC,"transform coordinates","p XYZ(oXYZ point,oXYZ center)",PROCEDURE(p_localToWorld),},
256{"worldToLocal",0,PARAM_NOSTATIC,"transform coordinates","p XYZ(oXYZ point,oXYZ center)",PROCEDURE(p_worldToLocal),},
257{0,0,0,},
258};
259#undef FIELDSTRUCT
260return paramtab;
261}
262
263void Orient_Ext::p_new(ExtValue *args,ExtValue *ret)
264{
265*ret=makeDynamicObject(new Orient_Ext());
266}
267
268void Orient_Ext::p_newFromVector(ExtValue *args,ExtValue *ret)
269{
270VectorObject *vec=VectorObject::fromObject(args->getObject());
271if (vec)
272        *ret=makeDynamicObject(new Orient_Ext(Orient(pt3DFromVec(vec,0),pt3DFromVec(vec,3),pt3DFromVec(vec,6))));
273else
274        ret->setEmpty();
275}
276
277void Orient_Ext::get_toVector(ExtValue *ret)
278{
279VectorObject *vec=new VectorObject;
280add3Coords(vec,o.x);
281add3Coords(vec,o.y);
282add3Coords(vec,o.z);
283ret->setObject(ExtObject(&VectorObject::par,vec));
284}
285
286void Orient_Ext::p_clone(ExtValue *args,ExtValue *ret)
287{
288*ret=makeDynamicObject(new Orient_Ext(o));
289}
290
291void Orient_Ext::p_set(ExtValue *args,ExtValue *ret)
292{
293Orient_Ext *other=fromObject(args[0]);
294if (other)
295        o=other->o;
296ret->setEmpty();
297}
298
299void Orient_Ext::p_reset(ExtValue *args,ExtValue *ret)
300{
301o=Orient_1;
302ret->setEmpty();
303}
304
305void Orient_Ext::get_x(PARAMGETARGS)
306{
307*ret=Pt3D_Ext::makeStaticObject(&o.x);
308}
309
310void Orient_Ext::get_y(PARAMGETARGS)
311{
312*ret=Pt3D_Ext::makeStaticObject(&o.y);
313}
314
315void Orient_Ext::get_z(PARAMGETARGS)
316{
317*ret=Pt3D_Ext::makeStaticObject(&o.z);
318}
319
320void Orient_Ext::p_lookat(ExtValue *args,ExtValue *ret)
321{
322Pt3D_Ext *dir=Pt3D_Ext::fromObject(args[1]),*up=Pt3D_Ext::fromObject(args[0]);
323if (dir&&up)
324        o.lookAt(dir->p,up->p);
325ret->setEmpty();
326}
327
328void Orient_Ext::p_rotate3(ExtValue *args,ExtValue *ret)
329{
330Pt3D p(args[2].getDouble(),args[1].getDouble(),args[0].getDouble());
331o.rotate(p);
332ret->setEmpty();
333}
334
335void Orient_Ext::p_rotate(ExtValue *args,ExtValue *ret)
336{
337Orient_Ext *obj=Orient_Ext::fromObject(args[0]);
338if (!obj)
339        {
340        Orient tmp=o;
341        obj->o.transform(o,tmp);
342        }
343ret->setEmpty();
344}
345
346void Orient_Ext::p_revrotate(ExtValue *args,ExtValue *ret)
347{
348Orient_Ext *obj=Orient_Ext::fromObject(args[0]);
349if (obj)
350        {
351        Orient tmp=o;
352        obj->o.revTransform(o,tmp);
353        }
354ret->setEmpty();
355}
356
357void Orient_Ext::p_normalize(ExtValue *args,ExtValue *ret)
358{
359o.normalize();
360ret->setEmpty();
361}
362
363void Orient_Ext::p_between2(ExtValue *args,ExtValue *ret)
364{
365Orient_Ext *o1=Orient_Ext::fromObject(args[2]);
366Orient_Ext *o2=Orient_Ext::fromObject(args[1]);
367if (o1&&o2)
368        {
369        double q1=args[0].getDouble(),q2=1.0-q1;
370        o.x.x=q1*o1->o.x.x+q2*o2->o.x.x;
371        o.x.y=q1*o1->o.x.y+q2*o2->o.x.y;
372        o.x.z=q1*o1->o.x.z+q2*o2->o.x.z;
373        o.y.x=q1*o1->o.y.x+q2*o2->o.y.x;
374        o.y.y=q1*o1->o.y.y+q2*o2->o.y.y;
375        o.y.z=q1*o1->o.y.z+q2*o2->o.y.z;
376        o.z.x=q1*o1->o.z.x+q2*o2->o.z.x;
377        o.z.y=q1*o1->o.z.y+q2*o2->o.z.y;
378        o.z.z=q1*o1->o.z.z+q2*o2->o.z.z;
379        o.normalize();
380        }
381ret->setEmpty();
382}
383
384void Orient_Ext::p_betweenOV(ExtValue *args,ExtValue *ret)
385{
386Orient_Ext *o1=Orient_Ext::fromObject(args[2]);
387Pt3D_Ext *p2=Pt3D_Ext::fromObject(args[1]);
388if (o1&&p2)
389        {
390        double q1=args[0].getDouble(),q2=1.0-q1;
391        o.x.x=q1*o1->o.x.x+q2*p2->p.x;
392        o.x.y=q1*o1->o.x.y+q2*p2->p.y;
393        o.x.z=q1*o1->o.x.z+q2*p2->p.z;
394        o.normalize();
395        }
396ret->setEmpty();
397}
398
399void Orient_Ext::p_localToWorld(ExtValue *args,ExtValue *ret)
400{
401Pt3D_Ext *center,*point;
402point=Pt3D_Ext::fromObject(args[1]);
403center=Pt3D_Ext::fromObject(args[0]);
404if (center && point)
405        {
406        Pt3D d;
407        Pt3D src=point->p;
408        o.transform(d,src);
409        d+=center->p;
410        *ret=Pt3D_Ext::makeDynamicObject(new Pt3D_Ext(d));
411        }
412else
413        ret->setEmpty();
414}
415
416void Orient_Ext::p_worldToLocal(ExtValue *args,ExtValue *ret)
417{
418Pt3D_Ext *center,*point;
419point=Pt3D_Ext::fromObject(args[1]);
420center=Pt3D_Ext::fromObject(args[0]);
421if (center && point)
422        {
423        Pt3D d;
424        Pt3D src=point->p;
425        d-=center->p;
426        o.revTransform(d,src);
427        *ret=Pt3D_Ext::makeDynamicObject(new Pt3D_Ext(d));
428        }
429else
430        ret->setEmpty();
431}
432
433Param& Orient_Ext::getStaticParam()
434{
435#ifdef __CODEGUARD__
436static Orient_Ext static_orientobj;
437static Param static_orientparam(getStaticParamtab(),&static_orientobj);
438#else
439static Param static_orientparam(getStaticParamtab());
440#endif
441return static_orientparam;
442}
443
444Orient_Ext* Orient_Ext::fromObject(const ExtValue& v)
445{
446if (v.type!=TObj) return 0;
447const ExtObject& o=v.getObject();
448return (Orient_Ext*)o.getTarget(getStaticParam().getName());
449}
450
451ParamInterface* Orient_Ext::getInterface() {return &getStaticParam();}
452
453ExtObject Orient_Ext::makeStaticObject(Orient* o)
454{return ExtObject(&getStaticParam(),((char*)o)+(((char*)&o->x)-((char*)&((Orient_Ext*)o)->o.x)));}
455
456ExtObject Orient_Ext::makeDynamicObject(Orient_Ext* p)
457{return ExtObject(&getStaticParam(),p);}
458
459ParamEntry* ReferenceObj::getStaticParamtab()
460{
461#define FIELDSTRUCT ReferenceObj
462static ParamEntry paramtab[]=
463{
464{"Ref",1,5,"Ref","Reference objects. Useful for returning things from functions.\n\nExample:\nvar x=111;\nsquare(&x);// '&' creates the Reference object\nSimulator.print(x);//x is now 12321\n\nfunction square(r)\n{r.value=r.value*r.value;}\n//square receives the Reference objects and changes its 'value' field"},
465
466{"value",0,PARAM_NOSTATIC,"value","x",GETSET(value),},
467{"newS",0,0,"create new reference","p",PROCEDURE(p_newS),"(for internal use only) use &variablename to create Ref objects.",},
468{"newO",0,0,"create new reference","p",PROCEDURE(p_newO),"(for internal use only) use &variablename to create Ref objects.",},
469{"copyFrom",0,0,"copy the reference","p(oRef)",PROCEDURE(p_copyFrom),"make the reference point to the same target,"},
470{"toString",0,PARAM_READONLY+PARAM_NOSTATIC,"textual form","s",GETONLY(toString),},
471{0,0,0,},
472};
473#undef FIELDSTRUCT
474return paramtab;
475}
476
477Param& ReferenceObj::getStaticParam()
478{
479#ifdef __CODEGUARD__
480static ReferenceObj static_referenceobj;
481static Param static_refobjectparam(getStaticParamtab(),&static_referenceobj);
482#else
483static Param static_refobjectparam(getStaticParamtab());
484#endif
485return static_refobjectparam;
486}
487
488void ReferenceObj::p_newS(ExtValue *args,ExtValue *ret)
489{
490*ret=makeDynamicObject(new ReferenceObj((ExtValue*)args->getInt()));
491}
492
493void ReferenceObj::p_newO(ExtValue *args,ExtValue *ret)
494{
495if (args[0].type==TInt)
496        *ret=makeDynamicObject(new ReferenceObj(args[1].getObject(),args[0].getInt()));
497else
498        *ret=makeDynamicObject(new ReferenceObj(args[1].getObject(),args[0].getString()));
499}
500
501void ReferenceObj::p_copyFrom(ExtValue *args,ExtValue *ret)
502{
503ReferenceObj* other=fromObject(args[0]);
504if (other)
505        {
506        value=other->value;
507        obj=other->obj;
508        prop=other->prop;
509        }
510}
511
512void ReferenceObj::get_toString(ExtValue *ret)
513{
514SString s="(";
515static SListTempl<ReferenceObj*> trace;
516if (trace.find(this)>=0)
517        s+="...";
518else
519        {
520        trace+=this;
521        if (value)
522                s+=value->getString();
523        else
524                {
525                ExtValue v;
526                Param tmp_param;
527                ParamInterface *pi=obj.getParamInterface(tmp_param);
528                pi->get(prop,v);
529                s+=v.getString();
530                }
531        trace-=this;
532        }
533s+=")";
534ret->setString(s);
535}
536
537void ReferenceObj::get_value(ExtValue *ret)
538{
539if (value)
540        *ret=*value;
541else
542        {
543        Param tmp_param;
544        ParamInterface *pi=obj.getParamInterface(tmp_param);
545        pi->get(prop,*ret);
546        }
547}
548
549int ReferenceObj::set_value(const ExtValue *val)
550{
551if (value)
552        *value=*val;
553else
554        {
555        Param tmp_param;
556        ParamInterface *pi=obj.getParamInterface(tmp_param);
557        pi->set(prop,*val);
558        }
559return PSET_CHANGED;
560}
561
562ReferenceObj::ReferenceObj(const ExtObject &o,const SString &p)
563        :value(0),obj(o)
564{
565Param tmp_param;
566ParamInterface *pi=obj.getParamInterface(tmp_param);
567prop=pi->findId(p);
568}
569
570ExtObject ReferenceObj::makeDynamicObject(ReferenceObj* r)
571{return ExtObject(&getStaticParam(),r);}
572
573ReferenceObj* ReferenceObj::fromObject(const ExtValue& v)
574{
575if (v.type!=TObj) return 0;
576const ExtObject& o=v.getObject();
577return (ReferenceObj*)o.getTarget(getStaticParam().getName());
578}
Note: See TracBrowser for help on using the repository browser.