source: trunk/src/script/qscriptclass.cpp@ 553

Last change on this file since 553 was 2, checked in by Dmitry A. Kuminov, 16 years ago

Initially imported qt-all-opensource-src-4.5.1 from Trolltech.

File size: 21.9 KB
Line 
1/****************************************************************************
2**
3** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
4** Contact: Qt Software Information (qt-info@nokia.com)
5**
6** This file is part of the QtScript module of the Qt Toolkit.
7**
8** $QT_BEGIN_LICENSE:LGPL$
9** Commercial Usage
10** Licensees holding valid Qt Commercial licenses may use this file in
11** accordance with the Qt Commercial License Agreement provided with the
12** Software or, alternatively, in accordance with the terms contained in
13** a written agreement between you and Nokia.
14**
15** GNU Lesser General Public License Usage
16** Alternatively, this file may be used under the terms of the GNU Lesser
17** General Public License version 2.1 as published by the Free Software
18** Foundation and appearing in the file LICENSE.LGPL included in the
19** packaging of this file. Please review the following information to
20** ensure the GNU Lesser General Public License version 2.1 requirements
21** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
22**
23** In addition, as a special exception, Nokia gives you certain
24** additional rights. These rights are described in the Nokia Qt LGPL
25** Exception version 1.0, included in the file LGPL_EXCEPTION.txt in this
26** package.
27**
28** GNU General Public License Usage
29** Alternatively, this file may be used under the terms of the GNU
30** General Public License version 3.0 as published by the Free Software
31** Foundation and appearing in the file LICENSE.GPL included in the
32** packaging of this file. Please review the following information to
33** ensure the GNU General Public License version 3.0 requirements will be
34** met: http://www.gnu.org/copyleft/gpl.html.
35**
36** If you are unsure which license is appropriate for your use, please
37** contact the sales department at qt-sales@nokia.com.
38** $QT_END_LICENSE$
39**
40****************************************************************************/
41
42#include "qscriptclass.h"
43
44#ifndef QT_NO_SCRIPT
45
46#include <QtCore/qstringlist.h>
47
48#include "qscriptclasspropertyiterator.h"
49#include "qscriptstring.h"
50#include "qscriptstring_p.h"
51#include "qscriptclass_p.h"
52#include "qscriptclassinfo_p.h"
53#include "qscriptengine_p.h"
54#include "qscriptcontext_p.h"
55#include "qscriptvalueimpl_p.h"
56#include "qscriptmember_p.h"
57#include "qscriptobject_p.h"
58#include "qscriptfunction_p.h"
59
60Q_DECLARE_METATYPE(QScriptContext*)
61Q_DECLARE_METATYPE(QScriptValueList)
62
63QT_BEGIN_NAMESPACE
64
65/*!
66 \since 4.4
67 \class QScriptClass
68
69 \brief The QScriptClass class provides an interface for defining custom behavior of (a class of) Qt Script objects.
70
71 \ingroup script
72 \mainclass
73
74 The QScriptClass class defines an interface for handling various
75 aspects of interaction with the Qt Script objects associated with
76 the class. Such objects are created by calling
77 QScriptEngine::newObject(), passing a pointer to the QScriptClass as
78 argument.
79
80 By subclassing QScriptClass, you can define precisely how access to
81 properties of the objects that use your class is handled. This
82 enables a fully dynamic handling of properties, e.g. it's more
83 powerful than QScriptEngine::newQObject(). For example, you can use
84 QScriptClass to implement array-type objects (i.e. objects that
85 handle the \c{length} property, and properties whose names are valid
86 array indexes, in a special way), or to implement a "live"
87 (runtime-defined) proxy to an underlying object.
88
89 If you just need to handle access to a set of properties that are
90 known at the time an object is created (i.e. "semi-statically"), you
91 might consider using QScriptValue::setProperty() to define
92 getter/setter functions for the relevant properties, rather than
93 subclassing QScriptClass.
94
95 Reimplement queryProperty() to specify which properties are handled
96 in a custom way by your script class (i.e. should be
97 \bold{delegated} to the QScriptClass), and which properties should
98 be handled just like normal Qt Script object properties.
99
100 Reimplement property() and setProperty() to perform the actual
101 access (read or write) to the properties that your class
102 handles. Additionally, you can reimplement propertyFlags() to
103 specify custom flags for your properties.
104
105 Reimplement newIterator() to provide an iterator for objects of your
106 custom class. This is only necessary if objects of your class can
107 have custom properties that you want to be reported when an object
108 is used together with the QScriptValueIterator class, or when an
109 object is used in a for-in enumeration statement in a script.
110
111 When implementing custom classes of objects, you typically use
112 QScriptValue::setData() to store instance-specific data as part of
113 object initialization; the data won't be accessible from scripts
114 directly, but you can access it in e.g. your reimplementations of
115 property() and setProperty() (by calling QScriptValue::data()) to
116 perform custom processing.
117
118 Reimplement prototype() to provide a custom prototype object for
119 your script class.
120
121 Reimplement supportsExtension() and extension() if your custom
122 script class supports one or more of the extensions specified by the
123 Extension enum.
124
125 \sa QScriptClassPropertyIterator, QScriptEngine::newObject(), {Custom Script Class Example}
126*/
127
128/*!
129 \enum QScriptClass::Extension
130
131 This enum specifies the possible extensions to a QScriptClass.
132
133 \value Callable Instances of this class can be called as functions.
134
135 \value HasInstance Instances of this class implement [[HasInstance]].
136
137 \sa extension()
138*/
139
140/*!
141 \enum QScriptClass::QueryFlag
142
143 This enum describes flags that are used to query a QScriptClass
144 regarding how access to a property should be handled.
145
146 \value HandlesReadAccess The QScriptClass handles read access to this property.
147 \value HandlesWriteAccess The QScriptClass handles write access to this property.
148
149 \sa queryProperty()
150*/
151
152class QScriptCustomClassData : public QScriptClassData
153{
154public:
155 QScriptCustomClassData(QScriptClass *klass);
156 ~QScriptCustomClassData();
157
158 virtual void mark(const QScriptValueImpl &object, int generation);
159 virtual bool resolve(const QScriptValueImpl &object, QScriptNameIdImpl *nameId,
160 QScript::Member *member, QScriptValueImpl *base,
161 QScript::AccessMode access);
162 virtual bool get(const QScriptValueImpl &obj, const QScript::Member &m,
163 QScriptValueImpl *result);
164 virtual bool put(QScriptValueImpl *object, const QScript::Member &member,
165 const QScriptValueImpl &value);
166 virtual bool removeMember(const QScriptValueImpl &object,
167 const QScript::Member &member);
168 virtual bool implementsHasInstance(const QScriptValueImpl &object);
169 virtual bool hasInstance(const QScriptValueImpl &object,
170 const QScriptValueImpl &value);
171 virtual QScriptClassDataIterator *newIterator(const QScriptValueImpl &object);
172
173 QScriptClass *scriptClass() const;
174
175private:
176 QScriptClass *m_class;
177};
178
179class QScriptCustomClassDataIterator : public QScriptClassDataIterator
180{
181public:
182 QScriptCustomClassDataIterator(const QScriptValueImpl &object,
183 QScriptClass *klass);
184 virtual ~QScriptCustomClassDataIterator();
185
186 virtual bool hasNext() const;
187 virtual void next(QScript::Member *member);
188
189 virtual bool hasPrevious() const;
190 virtual void previous(QScript::Member *member);
191
192 virtual void toFront();
193 virtual void toBack();
194
195private:
196 void iteratorToMember(QScript::Member *member);
197
198 QScriptClassPropertyIterator *m_it;
199};
200
201QScriptCustomClassData::QScriptCustomClassData(QScriptClass *klass)
202 : m_class(klass)
203{
204}
205
206QScriptCustomClassData::~QScriptCustomClassData()
207{
208}
209
210void QScriptCustomClassData::mark(const QScriptValueImpl &, int)
211{
212}
213
214bool QScriptCustomClassData::resolve(const QScriptValueImpl &object, QScriptNameIdImpl *nameId,
215 QScript::Member *member, QScriptValueImpl *base,
216 QScript::AccessMode access)
217{
218 uint id = 0;
219 QScriptClass::QueryFlags queryIn = 0;
220 if (access & QScript::Read)
221 queryIn |= QScriptClass::HandlesReadAccess;
222 if (access & QScript::Write)
223 queryIn |= QScriptClass::HandlesWriteAccess;
224 QScriptEnginePrivate *eng = object.engine();
225 QScriptString str = eng->internedString(nameId);
226 QScriptClass::QueryFlags queryOut;
227 queryOut = m_class->queryProperty(eng->toPublic(object), str, queryIn, &id);
228 if (queryOut & queryIn) {
229 if (base)
230 *base = object;
231 QScriptValue::PropertyFlags flags = m_class->propertyFlags(eng->toPublic(object), str, id);
232 member->native(nameId, id, flags);
233 return true;
234 }
235 return false;
236}
237
238bool QScriptCustomClassData::get(const QScriptValueImpl &object, const QScript::Member &member,
239 QScriptValueImpl *result)
240{
241 QScriptEnginePrivate *eng = object.engine();
242 QScriptString str = eng->internedString(member.nameId());
243 *result = eng->toImpl(m_class->property(eng->toPublic(object), str, member.id()));
244 if (!result->isValid())
245 *result = eng->undefinedValue();
246 return true;
247}
248
249bool QScriptCustomClassData::put(QScriptValueImpl *object, const QScript::Member &member,
250 const QScriptValueImpl &value)
251{
252 QScriptEnginePrivate *eng = object->engine();
253 QScriptString str = eng->internedString(member.nameId());
254 QScriptValue publicObject = eng->toPublic(*object);
255 m_class->setProperty(publicObject, str, member.id(), eng->toPublic(value));
256 return true;
257}
258
259bool QScriptCustomClassData::removeMember(const QScriptValueImpl &object,
260 const QScript::Member &member)
261{
262 QScriptEnginePrivate *eng = object.engine();
263 QScriptString str = eng->internedString(member.nameId());
264 QScriptValue publicObject = eng->toPublic(object);
265 m_class->setProperty(publicObject, str, member.id(), QScriptValue());
266 return true;
267}
268
269bool QScriptCustomClassData::implementsHasInstance(const QScriptValueImpl &object)
270{
271 if (object.classInfo() != QScriptClassPrivate::get(m_class)->classInfo())
272 return false;
273 return m_class->supportsExtension(QScriptClass::HasInstance);
274}
275
276bool QScriptCustomClassData::hasInstance(const QScriptValueImpl &object,
277 const QScriptValueImpl &value)
278{
279 QScriptEnginePrivate *eng = object.engine();
280 QScriptValueList arguments;
281 arguments << eng->toPublic(object) << eng->toPublic(value);
282 QVariant ret = m_class->extension(QScriptClass::HasInstance, qVariantFromValue(arguments));
283 return ret.toBool();
284}
285
286QScriptClassDataIterator *QScriptCustomClassData::newIterator(const QScriptValueImpl &object)
287{
288 return new QScriptCustomClassDataIterator(object, m_class);
289}
290
291QScriptClass *QScriptCustomClassData::scriptClass() const
292{
293 return m_class;
294}
295
296
297
298QScriptCustomClassDataIterator::QScriptCustomClassDataIterator(const QScriptValueImpl &object,
299 QScriptClass *klass)
300{
301 QScriptEnginePrivate *eng = object.engine();
302 m_it = klass->newIterator(eng->toPublic(object));
303}
304
305QScriptCustomClassDataIterator::~QScriptCustomClassDataIterator()
306{
307 if (m_it) {
308 delete m_it;
309 m_it = 0;
310 }
311}
312
313bool QScriptCustomClassDataIterator::hasNext() const
314{
315 return m_it && m_it->hasNext();
316}
317
318void QScriptCustomClassDataIterator::next(QScript::Member *member)
319{
320 if (m_it) {
321 m_it->next();
322 iteratorToMember(member);
323 }
324}
325
326bool QScriptCustomClassDataIterator::hasPrevious() const
327{
328 return m_it && m_it->hasPrevious();
329}
330
331void QScriptCustomClassDataIterator::previous(QScript::Member *member)
332{
333 if (m_it) {
334 m_it->previous();
335 iteratorToMember(member);
336 }
337}
338
339void QScriptCustomClassDataIterator::toFront()
340{
341 if (m_it)
342 m_it->toFront();
343}
344
345void QScriptCustomClassDataIterator::toBack()
346{
347 if (m_it)
348 m_it->toBack();
349}
350
351void QScriptCustomClassDataIterator::iteratorToMember(QScript::Member *member)
352{
353 QScriptString str = m_it->name();
354 QScriptNameIdImpl *nameId = 0;
355 if (str.isValid())
356 nameId = QScriptStringPrivate::get(str)->nameId;
357 member->native(nameId, m_it->id(), m_it->flags());
358}
359
360
361
362QScriptClassPrivate::QScriptClassPrivate(QScriptClass *q)
363 : engine(0), m_classInfo(0), q_ptr(q)
364{
365}
366
367QScriptClassPrivate::~QScriptClassPrivate()
368{
369 if (m_classInfo) {
370 // classInfo is owned by engine
371 // set the data to the normal Object class data
372 delete m_classInfo->data();
373 QScriptEnginePrivate *eng_p = QScriptEnginePrivate::get(engine);
374 m_classInfo->setData(eng_p->m_class_object->data());
375 }
376}
377
378QScriptClassPrivate *QScriptClassPrivate::get(QScriptClass *klass)
379{
380 return klass->d_func();
381}
382
383QScriptClassInfo *QScriptClassPrivate::classInfo()
384{
385 Q_Q(QScriptClass);
386 if (m_classInfo)
387 return m_classInfo;
388 QScriptEnginePrivate *eng_p = QScriptEnginePrivate::get(engine);
389 int classType = eng_p->registerCustomClassType();
390 if (q->supportsExtension(QScriptClass::Callable))
391 classType |= QScriptClassInfo::FunctionBased;
392 QString name = q->name();
393 if (name.isEmpty())
394 name = QLatin1String("Object");
395 m_classInfo = eng_p->registerClass(name, classType);
396 m_classInfo->setData(new QScriptCustomClassData(q_func()));
397 return m_classInfo;
398}
399
400QScriptClass *QScriptClassPrivate::classFromInfo(QScriptClassInfo *info)
401{
402 QScriptCustomClassData *data = static_cast<QScriptCustomClassData*>(info->data());
403 Q_ASSERT(data != 0);
404 return data->scriptClass();
405}
406
407static QScriptValueImpl callScriptClassFunction(QScriptContextPrivate *ctx,
408 QScriptEnginePrivate *eng,
409 QScriptClassInfo *classInfo)
410{
411 qMetaTypeId<QScriptContext*>();
412 if (QScriptClassData *data = classInfo->data()) {
413 QScriptCustomClassData *customData = static_cast<QScriptCustomClassData*>(data);
414 QScriptClass *klass = customData->scriptClass();
415 QVariant arg = qVariantFromValue(QScriptContextPrivate::get(ctx));
416 QVariant ret = klass->extension(QScriptClass::Callable, arg);
417 QScriptValueImpl val = eng->valueFromVariant(ret);
418 if (val.isValid())
419 return val;
420 }
421 return eng->undefinedValue();
422}
423
424QScriptFunction *QScriptClassPrivate::newFunction()
425{
426 return new QScript::C2Function(callScriptClassFunction, /*length=*/0,
427 classInfo(), /*name=*/QString());
428}
429
430/*!
431 Constructs a QScriptClass object to be used in the given \a engine.
432
433 The engine does not take ownership of the QScriptClass object.
434*/
435QScriptClass::QScriptClass(QScriptEngine *engine)
436 : d_ptr(new QScriptClassPrivate(this))
437{
438 d_ptr->engine = engine;
439}
440
441/*!
442 \internal
443*/
444QScriptClass::QScriptClass(QScriptEngine *engine, QScriptClassPrivate &dd)
445 : d_ptr(&dd)
446{
447 d_ptr->q_ptr = this;
448 d_ptr->engine = engine;
449}
450
451/*!
452 Destroys the QScriptClass object.
453
454 If a QScriptClass object is deleted before the associated engine(),
455 any Qt Script objects using the QScriptClass will be "demoted" to
456 normal Qt Script objects.
457*/
458QScriptClass::~QScriptClass()
459{
460 delete d_ptr;
461 d_ptr = 0;
462}
463
464/*!
465 Returns the engine that this QScriptClass is associated with.
466*/
467QScriptEngine *QScriptClass::engine() const
468{
469 Q_D(const QScriptClass);
470 return d->engine;
471}
472
473/*!
474 Returns the object to be used as the prototype of new instances
475 of this class (created with QScriptEngine::newObject()).
476
477 The default implementation returns an invalid QScriptValue, meaning
478 that the standard Object prototype will be used. Reimplement this
479 function to provide your own custom prototype.
480
481 Typically you initialize your prototype object in the constructor of
482 your class, then return it in this function.
483
484 See the "Making Use of Prototype-Based Inheritance" section in the
485 QtScript documentation for more information on how prototypes are
486 used.
487*/
488QScriptValue QScriptClass::prototype() const
489{
490 return QScriptValue();
491}
492
493/*!
494 Returns the name of the script class.
495
496 Qt Script uses this name to generate a default string representation
497 of objects in case you do not provide a toString function.
498
499 The default implementation returns a null string.
500*/
501QString QScriptClass::name() const
502{
503 return QString();
504}
505
506/*!
507 Queries this script class for how access to the property with the
508 given \a name of the given \a object should be handled. The given \a
509 flags specify the aspects of interest. This function should return a
510 subset of \a flags to indicate which aspects of property access
511 should be further handled by the script class.
512
513 For example, if the \a flags contain HandlesReadAccess, and you
514 would like your class to handle the reading of the property (through
515 the property() function), the returned flags should include
516 HandlesReadAccess. If the returned flags do not contain
517 HandlesReadAccess, the property will be handled as a normal script
518 object property.
519
520 You can optionally use the \a id argument to store a value that will
521 subsequently be passed on to functions such as property() and
522 setProperty().
523
524 The default implementation of this function returns 0.
525
526 Note: This function is only called if the given property isn't
527 already a normal property of the object. For example, say you
528 advertise that you want to handle read access to property \c{foo},
529 but not write access; if \c{foo} is then assigned a value, it will
530 become a normal script object property, and subsequently you will no
531 longer be queried regarding read access to \c{foo}.
532
533 \sa property()
534*/
535QScriptClass::QueryFlags QScriptClass::queryProperty(
536 const QScriptValue &object, const QScriptString &name,
537 QueryFlags flags, uint *id)
538{
539 Q_UNUSED(object);
540 Q_UNUSED(name);
541 Q_UNUSED(flags);
542 Q_UNUSED(id);
543 return 0;
544}
545
546/*!
547 Returns the value of the property with the given \a name of the given
548 \a object.
549
550 The \a id argument is only useful if you assigned a value to it in
551 queryProperty().
552
553 The default implementation does nothing and returns an invalid QScriptValue.
554
555 \sa setProperty(), propertyFlags()
556*/
557QScriptValue QScriptClass::property(const QScriptValue &object,
558 const QScriptString &name, uint id)
559{
560 Q_UNUSED(object);
561 Q_UNUSED(name);
562 Q_UNUSED(id);
563 return QScriptValue();
564}
565
566/*!
567 Returns the flags of the property with the given \a name of the given
568 \a object.
569
570 The \a id argument is only useful if you assigned a value to it in
571 queryProperty().
572
573 The default implementation returns 0.
574
575 \sa property()
576*/
577QScriptValue::PropertyFlags QScriptClass::propertyFlags(
578 const QScriptValue &object, const QScriptString &name, uint id)
579{
580 Q_UNUSED(object);
581 Q_UNUSED(name);
582 Q_UNUSED(id);
583 return 0;
584}
585
586/*!
587 Sets the property with the given \a name of the given \a object to
588 the given \a value.
589
590 The \a id argument is only useful if you assigned a value to it in
591 queryProperty().
592
593 The default implementation does nothing.
594
595 An invalid \a value represents a request to remove the property.
596
597 \sa property()
598*/
599void QScriptClass::setProperty(QScriptValue &object, const QScriptString &name,
600 uint id, const QScriptValue &value)
601{
602 Q_UNUSED(object);
603 Q_UNUSED(name);
604 Q_UNUSED(id);
605 Q_UNUSED(value);
606}
607
608/*!
609 Returns an iterator for traversing custom properties of the given \a
610 object.
611
612 The default implementation returns 0, meaning that there are no
613 custom properties to traverse.
614
615 Reimplement this function if objects of your script class can have
616 one or more custom properties (e.g. those reported to be handled by
617 queryProperty()) that you want to appear when an object's properties
618 are enumerated (e.g. by a for-in statement in a script).
619
620 Qt Script takes ownership of the new iterator object.
621
622 \sa QScriptValueIterator
623*/
624QScriptClassPropertyIterator *QScriptClass::newIterator(const QScriptValue &object)
625{
626 Q_UNUSED(object);
627 return 0;
628}
629
630/*!
631 Returns true if the QScriptClass supports the given \a extension;
632 otherwise, false is returned. By default, no extensions
633 are supported.
634
635 Reimplement this function to indicate which extensions your custom
636 class supports.
637
638 \sa extension()
639*/
640bool QScriptClass::supportsExtension(Extension extension) const
641{
642 Q_UNUSED(extension);
643 return false;
644}
645
646/*!
647 This virtual function can be reimplemented in a QScriptClass
648 subclass to provide support for extensions. The optional \a argument
649 can be provided as input to the \a extension; the result must be
650 returned in the form of a QVariant. You can call supportsExtension()
651 to check if an extension is supported by the QScriptClass. By
652 default, no extensions are supported, and this function returns an
653 invalid QVariant.
654
655 If you implement the Callable extension, Qt Script will call this
656 function when an instance of your class is called as a function
657 (e.g. from a script or using QScriptValue::call()). The \a argument
658 will contain a pointer to the QScriptContext that represents the
659 function call, and you should return a QVariant that holds the
660 result of the function call. In the following example the sum of the
661 arguments to the script function are added up and returned:
662
663 \snippet doc/src/snippets/code/src_script_qscriptclass.cpp 0
664
665 If you implement the HasInstance extension, Qt Script will call this
666 function as part of evaluating the \c{instanceof} operator, as
667 described in ECMA-262 Section 11.8.6. The \a argument is a
668 QScriptValueList containing two items: The first item is the object
669 that HasInstance is being applied to (an instance of your class),
670 and the second item can be any value. extension() should return true
671 if the value delegates behavior to the object, false otherwise.
672
673 \sa supportsExtension()
674*/
675QVariant QScriptClass::extension(Extension extension, const QVariant &argument)
676{
677 Q_UNUSED(extension);
678 Q_UNUSED(argument);
679 return QVariant();
680}
681
682QT_END_NAMESPACE
683
684#endif // QT_NO_SCRIPT
Note: See TracBrowser for help on using the repository browser.