QML in Android Studio Projects
Overview
This example contains a QML project that you can import into Android Studio with the Qt Tools for Android Studio plugin and Java and Kotlin projects that utilize the QtQuickView API.
For more information on how QML works, see the Qt Qml. This documentation will focus on how a QML component is embedded into Java- and Kotlin-based Android applications.
First, we look at the MainActivity
's onCreate() method of the Java and Kotlin projects.
For a Java-based project:
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); m_mainLinear = findViewById(R.id.mainLinear); m_getPropertyValueText = findViewById(R.id.getPropertyValueText); m_qmlStatus = findViewById(R.id.qmlStatus); m_androidControlsLayout = findViewById(R.id.javaLinear); m_box = findViewById(R.id.box); m_switch = findViewById(R.id.switch1); m_switch.setOnClickListener(view -> switchListener()); m_qmlView = new QtQuickView(this, "qrc:/qt/qml/qml_in_android_view/main.qml", "qml_in_android_view"); // Set status change listener for m_qmlView // listener implemented below in OnStatusChanged m_qmlView.setStatusChangeListener(this); ViewGroup.LayoutParams params = new FrameLayout.LayoutParams( ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT); m_qmlFrameLayout = findViewById(R.id.qmlFrame); m_qmlFrameLayout.addView(m_qmlView, params); Button button = findViewById(R.id.button); button.setOnClickListener(view -> onClickListener()); // Check target device orientation on launch handleOrientationChanges(); }
For a Kotlin-based project:
override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) m_binding = ActivityMainBinding.inflate(layoutInflater) val view = m_binding.root setContentView(view) m_binding.signalSwitch.setOnClickListener { switchListener() } m_qmlView = QtQuickView( this, "qrc:/qt/qml/qml_in_android_view/main.qml", "qml_in_android_view" ) // Set status change listener for m_qmlView // listener implemented below in OnStatusChanged m_qmlView!!.setStatusChangeListener(this) val params: ViewGroup.LayoutParams = FrameLayout.LayoutParams( ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT ) m_binding.qmlFrame.addView(m_qmlView, params) m_binding.changeColorButton.setOnClickListener { onClickListener() } // Check target device orientation on launch handleOrientationChanges() }
Note: in the Kotlin project we use View binding to access the UI components of the application:
m_binding = ActivityMainBinding.inflate(layoutInflater) val view = m_binding.root setContentView(view)
Inside the onCreate()
method, an instance of QtQuickView named m_qmlView
is created by giving it the Java/Kotlin application Context, URI of the QML project's main.qml
file and the name of the QML project's main library as parameters.
For a Java-based project:
m_qmlView = new QtQuickView(this, "qrc:/qt/qml/qml_in_android_view/main.qml", "qml_in_android_view");
For a Kotlin-based project:
m_qmlView = QtQuickView( this, "qrc:/qt/qml/qml_in_android_view/main.qml", "qml_in_android_view" )
m_qmlView
is then added to Android FrameLayout ViewGroup with appropriate layout parameters.
For a Java-based project:
ViewGroup.LayoutParams params = new FrameLayout.LayoutParams( ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT); m_qmlFrameLayout = findViewById(R.id.qmlFrame); m_qmlFrameLayout.addView(m_qmlView, params);
For a Kotlin-based project:
val params: ViewGroup.LayoutParams = FrameLayout.LayoutParams( ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT ) m_binding.qmlFrame.addView(m_qmlView, params)
Interacting with the QML component
To interact with the embedded QML component we first need to implement the QtQuickView public interface StatusChangeListener.
For a Java-based project:
public class MainActivity extends AppCompatActivity implements QtQuickView.StatusChangeListener{ ... }
IFor a Kotlin-based project:
class MainActivity : AppCompatActivity(), QtQuickView.StatusChangeListener{ ... }
Then, define an override for the StatusChangeListener callback function onStatusChanged()
.
For a Java-based project:
@Override public void onStatusChanged(int status) { Log.i(TAG, "Status of QtQuickView: " + status); final String qmlStatus = getResources().getString(R.string.qml_view_status) + m_statusNames.get(status); // Show current QML View status in a textview m_qmlStatus.setText(qmlStatus); // Connect signal listener to "onClicked" signal from main.qml // addSignalListener returns int which can be used later to identify the listener if (status == QtQuickView.STATUS_READY && !m_switch.isChecked()) { m_qmlButtonSignalListenerId = m_qmlView.connectSignalListener("onClicked", Object.class, (String signal, Object o) -> { Log.i(TAG, "QML button clicked"); m_androidControlsLayout.setBackgroundColor(Color.parseColor(m_colors.getColor())); }); } }
For a Kotlin-based project:
override fun onStatusChanged(status: Int) { Log.v(TAG, "Status of QtQuickView: $status") val qmlStatus = (resources.getString(R.string.qml_view_status) + m_statusNames[status]) // Show current QML View status in a textview m_binding.qmlStatus.text = qmlStatus // Connect signal listener to "onClicked" signal from main.qml // addSignalListener returns int which can be used later to identify the listener if (status == QtQuickView.STATUS_READY && !m_binding.signalSwitch.isChecked) { m_qmlButtonSignalListenerId = m_qmlView!!.connectSignalListener( "onClicked", Any::class.java ) { _: String?, _: Any? -> Log.v(TAG, "QML button clicked") m_binding.kotlinLinear.setBackgroundColor(Color.parseColor(m_colors.getColor())) } } }
Then, set that listener to listen for status changes of m_qmlView
with the setStatusChangeListener().
For a Java-based project:
m_qmlView.setStatusChangeListener(this);
For a Kotlin-based project:
m_qmlView!!.setStatusChangeListener(this)
The overridden callback function onStatusChanged()
receives StatusChanged()
signal containing the current Status value of the m_qmlView
. If this Status value is confirmed to be STATUS_READY, we can start interacting with the QML view.
Getting and setting QML view property values
Getting and setting QML view property values happens through the QtQuickView.getProperty() and QtQuickView.setProperty() methods.
The root object of the QML component's background color is set when a click event of an Android button occurs.
For a Java-based project:
public void onClickListener() { // Set the QML view root object property "colorStringFormat" value to // color from Colors.getColor() m_qmlView.setProperty("colorStringFormat", m_colors.getColor()); String qmlBackgroundColor = m_qmlView.getProperty("colorStringFormat"); // Display the QML View background color code m_getPropertyValueText.setText(qmlBackgroundColor); // Display the QML View background color in a view m_box.setBackgroundColor(Color.parseColor(qmlBackgroundColor)); }
For a Kotlin-based project:
private fun onClickListener() { // Set the QML view root object property "colorStringFormat" value to // color from Colors.getColor() m_qmlView!!.setProperty("colorStringFormat", m_colors.getColor()) val qmlBackgroundColor = m_qmlView!!.getProperty<String>("colorStringFormat") // Display the QML View background color code m_binding.getPropertyValueText.text = qmlBackgroundColor // Display the QML View background color in a view m_binding.colorBox.setBackgroundColor(Color.parseColor(qmlBackgroundColor)) }
With the QtQuickView.setProperty() method we set the "colorStringFormat" property value to a random color value that is fetched from the project's Colors.java
class.
The QtQuickView.getProperty(){QtQuickView.getProperty()} method is used here to fetch the current background color of the root object of the QML component and then show it to the user on the Android side of the application.
Signal listeners
QtQuickView class offers a connectSignalListener() and disconnectSignalListener() methods which are used to connect and disconnect a signal listener to a signal that is declared in the QML component root object.
Here we connect a signal listener to the onClicked()
signal of the QML component.
For a Java-based project:
if (status == QtQuickView.STATUS_READY && !m_switch.isChecked()) { m_qmlButtonSignalListenerId = m_qmlView.connectSignalListener("onClicked", Object.class, (String signal, Object o) -> { Log.i(TAG, "QML button clicked"); m_androidControlsLayout.setBackgroundColor(Color.parseColor(m_colors.getColor())); }); }
For a Kotlin-based project:
if (status == QtQuickView.STATUS_READY && !m_binding.signalSwitch.isChecked) { m_qmlButtonSignalListenerId = m_qmlView!!.connectSignalListener( "onClicked", Any::class.java ) { _: String?, _: Any? -> Log.v(TAG, "QML button clicked") m_binding.kotlinLinear.setBackgroundColor(Color.parseColor(m_colors.getColor())) } }
The onClicked()
signal is emitted every time the button on the QML UI is clicked. That signal is then received by this listener and the background color of the layout holding the Android side of the application is set to a random color value fetched from the project's Colors.java
class.
The QtQuickView.connectSignalListener() returns a unique signal listener id which we store and use later to identify and disconnect the listener.
For a Java-based project:
m_qmlView.disconnectSignalListener(m_qmlButtonSignalListenerId);
For a Kotlin-based project:
m_qmlView!!.disconnectSignalListener(m_qmlButtonSignalListenerId)
Here, the previously connected signal listener is disconnected using the QtQuickView.disconnectSignalListener() method by giving it the unique signal listener id.