目的:
在qml的设计中,处于安全原因,其被设计成不能直接读写本地文件,不能读取环境变量,等等与本地数据直接交互的功能都不能直接使用,想要使用本地数据和资源就只能通过c++/python-qt来进行,这样就要求要么安装qt的编译环境,要么安装python-qt的python库,对于有些情景的linux环境,这样做并不划算,因此这里尝试了一种hack的qml使用方式:
- 使用bash设置qml的启动参数,
- 并通过localstorage传递数据给python脚本,并从python脚本中读取处理结果
使用bash设置qml的启动参数,
简单来说就是将命令行参数写入临时文件,并在qml中读取。但是因为纯qml不支持文件读写,所以要用hack的方式,以加载组件的方式将数据加载到组件的属性中,并从组件的属性中读取该数据
首先创建start.bash
#!/bin/bash
# start.bash
CUR_DIR="$(realpath $(dirname $0))"
VALUE="$@"
cat > /tmp/target_args_reader.qml << EOF
import QtQuick 2.0
Item {
id: argsReader
property string value: "${VALUE}"
}
EOF
qmlscene "${CUR_DIR}/target.qml"
接下来在target.qml中:
import QtQuick 2.0
Rectangle {
id: main
Component.onCompleted: {
var data = null;
var argsReaderComponent = Qt.createComponent ("/tmp/target_args_reader.qml");
if (argsReaderComponent.status == Component.Ready) {
var reader = argsReaderComponent.createObject(main);
console.log(reader.value);
data = reader.value;
reader.destroy();
}
argsReaderComponent.destroy();
// use data
}
在纯qml中调用python程序
纯qml没有直接调用可执行程序的方法,所以我们仍然要使用hack的方式来在纯qml中启动可执行程序。qml虽然不能运行程序,但是可以通过系统绑定程序打开文件,那么只要系统关联文件是以可运行程序的方式打开某中文件,这就使我们在qml中启动可执行文件成为可能。而一般的linux系统中恰恰有这样一种文件会被默认作为可执行文件打开,它就是*.desktop,所以我们启动外部可执行文件的方法就是将可执行文件路径写入一个desktop文件,并在qml中以外部应用打开该文件。当然这种打开方式无法传递额外参数,所以为了传递参数,我们要使用qml的localstorage将数据写入数据库,并指定desktop中的可执行程序是用于解析参数的python脚本。
同样,我们创建desktop文件的方式还是在start.bash中
# ...
CUR_DIR="$(realpath $(dirname $0))"
# ...
cat > ${CUR_DIR}/execute_command.desktop << EOF
[Desktop Entry]
Exec=/usr/bin/python "${CUR_DIR}/execute_command.py"
Name=execute_command
NoDisplay=false
StartupNotify=false
Type=Application
EOF
chmod +x ${CUR_DIR}/execute_command.desktop
# ...
并提前创建execute_command.py
#!/usr/bin/python
# execut_command.py
# 内容待续,主要为连接数据库,读取参数
# 以qmlscene启动的qml应用,localstorage的目默认目录:
# "${HOME}/.local/share/QtProject/QtQmlViewer/QML/OfflineStorage/Databases/"
# 如果不在这里,可以在${HOME}/.local/share下搜索qt相关目录
# 遍历该目录下所有<database_id>.ini文件,获得<database_id>与数据库名称的绑定,找到目标数据库文件<target_database_id>.sqlite,以sqlite3模块连接该数据库文件,并读取相应数据
# xxx.ini文件的内容如下,可以依照该格式解析:
"""
[General]
Description=<数据库介绍的文本,无需引号>
Driver=QSQLITE
EstimatedSize=1000000
Name=<数据库名称>
Version=<数据库版本,格式类似1.0>
"""
接下来在qml中添加启动该python脚本的代码:
// ...
var db = LocalStorage.openDatabaseSync("arguments_transferer", "1.0", "transfer arguments to python script", 1000000);
db.transaction(
function(tx) {
// Create the database if it doesn't already exist
tx.executeSql('CREATE TABLE IF NOT EXISTS Arguments(name TEXT, value TEXT)');
// Add a row as one argument
tx.executeSql('INSERT INTO Arguments VALUES(?, ?)', [ 'a', '1' ]);
}
);
Qt.openUrlExternally("execute_command.desktop");
// ...
这样,当该部分代码被调用时,execute_command.py就会被调用,如果想在其执行结束后回传数据,则同样使用数据库即可。