在用 gps-share、Gypsy、geoplue2、serialnmea 和其他方法从连接到串行端口的 GPS 访问数据进行修补(即从源代码编译、安装、配置、测试等)之后(感谢 Pa_ 的所有建议) ,但是在 gpsd 对其他应用程序完美运行时都没有结果,我决定通过对 QDeclarativePositionSource 类进行非常粗略的更改来实现对 nmeaSource 属性的 URL 中 gpsd 方案的支持,从而使 Qt 支持 gpsd。通过此更改,现在可以将 gpsd 源定义为nmeaSource: "gpsd://hostname:2947"
(2947 是标准 gpsd 端口)。
更改后的代码如下所示。我建议这应该在某个时候添加到 Qt,但与此同时,我想我需要派生这个类来实现我在新 QML 组件中的更改,但是,作为 QML 的新手,我不知道它是如何完成的。我想根据active
PositionSource 项的属性从 gpsd 停止和启动 NMEA 流也可能是一个好主意......我会在某个时候得到它,但希望能在更多的优雅的方式。
void QDeclarativePositionSource::setNmeaSource(const QUrl &nmeaSource)
{
if ((nmeaSource.scheme() == QLatin1String("socket") )
|| (nmeaSource.scheme() == QLatin1String("gpsd"))) {
if (m_nmeaSocket
&& nmeaSource.host() == m_nmeaSocket->peerName()
&& nmeaSource.port() == m_nmeaSocket->peerPort()) {
return;
}
delete m_nmeaSocket;
m_nmeaSocket = new QTcpSocket();
connect(m_nmeaSocket, static_cast<void (QTcpSocket::*)(QAbstractSocket::SocketError)> (&QAbstractSocket::error),
this, &QDeclarativePositionSource::socketError);
connect(m_nmeaSocket, &QTcpSocket::connected,
this, &QDeclarativePositionSource::socketConnected);
// If scheme is gpsd, NMEA stream must be initiated by writing a command
// on the socket (gpsd WATCH_ENABLE | WATCH_NMEA flags)
// (ref.: gps_sock_stream function in gpsd source file libgps_sock.c)
if( nmeaSource.scheme() == QLatin1String("gpsd")) {
m_nmeaSocket->connectToHost(nmeaSource.host(),
nmeaSource.port(),
QTcpSocket::ReadWrite);
char const *gpsdInit = "?WATCH={\"enable\":true,\"nmea\":true}";
m_nmeaSocket->write( gpsdInit, strlen(gpsdInit);
} else {
m_nmeaSocket->connectToHost(nmeaSource.host(), nmeaSource.port(), QTcpSocket::ReadOnly);
}
} else {
...