QML can easily be extended with functionality defined in C++ code. Due to the tight integration of the QML engine with the Qt meta-object system, any functionality that is appropriately exposed by a QObject-derived class is accessible from QML code. This enables C++ data and functions to be accessible directly from QML, often with little or no modification.
The QML engine has the ability to introspect QObject instances through the meta-object system. This means any QML code can access the following members of an instance of a QObject-derived class:
- Properties
- Methods (providing they are public slots or flagged with Q_INVOKABLE)
- Signals
(Additionally, enums are available if they have been declared with Q_ENUMS. See Data Type Conversion Between QML and C++ for more details.)
Regular C functions are also accessible from QML. The file loading and saving functions are implemented in C and declared using the QINVOKABLE macro. Alternatively, we can declare the functions as a slot and the functions will be accessible from QML. H: QINVOKABLE void saveFile; QINVOKABLE void loadFile. A method that is a public Qt slot For example, the MessageBoard class below has a postMessage method that has been flagged with the QINVOKABLE macro, as well as a refresh method that is a public slot.
In general, these are accessible from QML regardless of whether a QObject-derived class has been registered with the QML type system. However, if a class is to be used in a way that requires the engine to access additional type information — for example, if the class itself is to be used as a method parameter or property, or if one of its enum types is to be used in this way — then the class may need to be registered.
Also note that a number of the important concepts covered in this document are demonstrated in the Writing QML Extensions with C++ tutorial.
Q_invokable Vs Public Slot
Data Type Handling and Ownership
Any data that is transferred from C++ to QML, whether as a property value, a method parameter or return value, or a signal parameter value, must be of a type that is supported by the QML engine.
By default, the engine supports a number of Qt C++ types and can automatically convert them as appropriately when used from QML. Additionally, C++ classes that are registered with the QML type system can be used as data types, as can their enums if appropriately registered. See Data Type Conversion Between QML and C++ for further information.
Additionally, data ownership rules are taken into consideration when data is transferred from C++ to QML. See Data Ownership for more details.
Exposing Properties
A property can be specified for any QObject-derived class using the Q_PROPERTY() macro. A property is a class data member with an associated read function and optional write function.
All properties of a QObject-derived class are accessible from QML.
For example, below is a Message
class with an author
property. As specified by the Q_PROPERTY macro call, this property is readable through the author()
method, and writable through the setAuthor()
method:
If an instance of this class was set as a context property when loading a file named MyItem.qml
from C++:
Then, the author
property could be read from MyItem.qml
:
For maximum interoperability with QML, any property that is writable should have an associated NOTIFY signal that is emitted whenever the property value has changed. This allows the property to be used with property binding, which is an essential feature of QML that enforces relationships between properties by automatically updating a property whenever any of its dependencies change in value.
In the above example, the associated NOTIFY signal for the author
property is authorChanged
, as specified in the Q_PROPERTY() macro call. This means that whenever the signal is emitted — as it is when the author changes in Message::setAuthor() — this notifies the QML engine that any bindings involving the author
property must be updated, and in turn, the engine will update the text
property by calling Message::author()
again.
If the author
property was writable but did not have an associated NOTIFY signal, the text
value would be initialized with the initial value returned by Message::author()
but would not be updated with any later changes to this property. In addition, any attempts to bind to the property from QML will produce a runtime warning from the engine.
Note: It is recommended that the NOTIFY signal be named Changed where is the name of the property. The associated property change signal handler generated by the QML engine will always take the form
onChanged
, regardless of the name of the related C++ signal, so it is recommended that the signal name follows this convention to avoid any confusion.
Notes on Use of Notify Signals
To prevent loops or excessive evaluation, developers should ensure that the property change signal is only emitted when the property value has actually changed. Also, if a property or group of properties is infrequently used, it is permitted to use the same NOTIFY signal for several properties. This should be done with care to ensure that performance doesn't suffer.
The presence of a NOTIFY signal does incur a small overhead. There are cases where a property's value is set at object construction time, and does not subsequently change. The most common case of this is when a type uses Grouped Properties, and the grouped property object is allocated once, and only freed when the object is deleted. In these cases, the CONSTANT attribute may be added to the property declaration instead of a NOTIFY signal.
The CONSTANT attribute should only be used for properties whose value is set, and finalized, only in the class constructor. All other properties that want to be used in bindings should have a NOTIFY signal instead.
Properties with Object Types
Object-type properties are accessible from QML providing that the object type has been appropriately registered with the QML type system.
For example, the Message
type might have a body
property of type MessageBody*
:
Suppose the Message
type was registered with the QML type system, allowing it to be used as an object type from QML code:
If the MessageBody
type was also registered with the type system, it would be possible to assign MessageBody
to the body
property of a Message
, all from within QML code:
Properties with Object-List Types
Properties containing lists of QObject-derived types can also be exposed to QML. For this purpose, however, one should use QQmlListProperty rather than QList as the property type. This is because QList is not a QObject-derived type, and so cannot provide the necessary QML property characteristics through the Qt meta object system, such as signal notifications when a list is modified.
For example, the MessageBoard
class below has a messages
property of type QQmlListProperty that stores a list of Message
instances:
The MessageBoard::messages() function simply creates and returns a QQmlListProperty from its QList m_messages
member, passing the appropriate list modification functions as required by the QQmlListProperty constructor:
Note that the template class type for the QQmlListProperty — in this case, Message
— must be registered with the QML type system.
Grouped Properties
Any read-only object-type property is accessible from QML code as a grouped property. This can be used to expose a group of related properties that describe a set of attributes for a type.
Q Invocable Public Slot Wins
For example, suppose the Message::author
property was of type MessageAuthor
rather than a simple string, with sub-properties of name
and email
:
The author
property could be written to using the grouped property syntax in QML, like this:
A type that is exposed as a grouped property differs from an object-type property in that the grouped property is read-only, and is initialized to a valid value by the parent object at construction. The grouped property's sub-properties may be modified from QML but the grouped property object itself will never change, whereas an object-type property may be assigned a new object value from QML at any time. Thus, the lifetime of a grouped property object is controlled strictly by the C++ parent implementation, whereas an object-type property can be freely created and destroyed through QML code.
Exposing Methods (Including Qt Slots)
Any method of a QObject-derived type is accessible from QML code if it is:
- A public method flagged with the Q_INVOKABLE() macro
- A method that is a public Qt slot
For example, the MessageBoard
class below has a postMessage()
method that has been flagged with the Q_INVOKABLE macro, as well as a refresh()
method that is a public slot:
If an instance of MessageBoard
was set as the context data for a file MyItem.qml
, then MyItem.qml
could invoke the two methods as shown in the examples below:
C++ |
QML |
If a C++ method has a parameter with a QObject*
type, the parameter value can be passed from QML using an object id
or a JavaScript var value that references the object.
QML supports the calling of overloaded C++ functions. If there are multiple C++ functions with the same name but different arguments, the correct function will be called according to the number and the types of arguments that are provided.
Values returned from C++ methods are converted to JavaScript values when accessed from JavaScript expressions in QML.
Exposing Signals
Any public signal of a QObject-derived type is accessible from QML code.
The QML engine automatically creates a signal handler for any signal of a QObject-derived type that is used from QML. Signal handlers are always named on where is the name of the signal, with the first letter capitalized. All parameters passed by the signal are available in the signal handler through the parameter names.
Q Invocable Public Slot Machine
For example, suppose the MessageBoard
class has a newMessagePosted()
signal with a single parameter, subject
:
If the MessageBoard
type was registered with the QML type system, then a MessageBoard
object declared in QML could receive the newMessagePosted()
signal using a signal handler named onNewMessagePosted
, and examine the subject
parameter value:
As with property values and method parameters, a signal parameter must have a type that is supported by the QML engine; see Data Type Conversion Between QML and C++. (Using an unregistered type will not generate an error, but the parameter value will not be accessible from the handler.)
Classes may have multiple signals with the same name, but only the final signal is accessible as a QML signal. Note that signals with the same name but different parameters cannot be distinguished from one another.
© 2020 The Qt Company Ltd. Documentation contributions included herein are the copyrights of their respective owners. The documentation provided herein is licensed under the terms of the GNU Free Documentation License version 1.3 as published by the Free Software Foundation. Qt and respective logos are trademarks of The Qt Company Ltd. in Finland and/or other countries worldwide. All other trademarks are property of their respective owners.