Sensors▲
The Sensors example consists of the following components:
-
protobuf_sensors library that you generate from the protobuf schema using the qt_add_protobuf call.
-
protobuf_sensor_emulator application that emulates simple sensor behavior.
-
protobuf_sensors_client application that displays the sensor data from the UDP socket.
The client application binds on the localhost UDP port 65500 and waits for data from the emulator application.
data:image/s3,"s3://crabby-images/fb747/fb747de07a5e8058658756e4afc32adba6d43c7f" alt="Image non disponible"
Use the emulator application to change the values of sensor data and send the data to the client's UDP port.
data:image/s3,"s3://crabby-images/d8ae9/d8ae9c66d574a923c37dd66e5ff591d927058b2c" alt="Image non disponible"
The applications use the generated messages from the protobuf_sensors library to communicate. The library is generated from the protobuf schema described in .proto files.
qt_add_protobuf(protobuf_sensors
PROTO_FILES
sensors.proto
tlv.proto
)
The first file describes the Type-Length-Value message, that wraps the sensor data:
package qt.examples.sensors.tlv;
enum
MessageType {
Coordinates =
0
;
Temperature =
1
;
WarningNotification =
2
;
}
// Protobuf messages imply inline data size.
message TlvMessage
{
MessageType type =
1
;
bytes value =
2
;
}
The second .proto file contains a description of the sensor messages:
package qt.examples.sensors;
message Coordinates
{
double
longitude =
1
;
double
latitude =
2
;
double
altitude =
3
;
}
message Temperature
{
enum
Unit {
Farenheit =
0
;
Celsius =
1
;
}
sint32 value =
1
;
Unit units =
2
;
}
message WarningNotification
{
string text =
1
;
}
Messages are serialized using QProtobufSerializer that you instantiate in the client:
class
SensorClient : public
QObject
{
Q_OBJECT
...
private
:
QUdpSocket m_client;
QProtobufSerializer m_serializer;
}
;
And the emulator:
class
SensorEmulator : public
QObject
{
Q_OBJECT
...
QUdpSocket m_socket;
QProtobufSerializer m_serializer;
}
;
After you click the Send button in the emulator application, the data from QLineEdit fields is converted from string format to the message field-specific format, for example, double for the fields of the Coordinates message:
QObject::
connect(ui-&
gt;sendCoordinates, &
amp;QPushButton::
clicked, this
, [this
]() {
qt::examples::sensors::
Coordinates coord;
coord.setLatitude(ui-&
gt;latitudeValue-&
gt;text().toDouble());
coord.setLongitude(ui-&
gt;longitudeValue-&
gt;text().toDouble());
coord.setAltitude(ui-&
gt;altitudeValue-&
gt;text().toDouble());
emit coordinatesUpdated(coord);
}
);
Then the message with all the fields is serialized and wrapped with the Type-Length-Value message:
Q_ASSERT(serializer !=
nullptr
);
qt::examples::sensors::tlv::
TlvMessage msg;
msg.setType(type);
msg.setValue(data);
return
msg.serialize(serializer);
The client applies the reverse operations on the received datagrams. First the Type-Length-Value message is deserialized from the datagram data:
const
auto
datagram =
m_client.receiveDatagram();
qt::examples::sensors::tlv::
TlvMessage msg;
msg.deserialize(&
amp;m_serializer, datagram.data());
if
(m_serializer.deserializationError()
!=
QAbstractProtobufSerializer::
NoError) {
qWarning().nospace() &
lt;&
lt; "Unable to deserialize datagram ("
&
lt;&
lt; m_serializer.deserializationError() &
lt;&
lt; ")"
&
lt;&
lt; m_serializer.deserializationErrorString();
continue
;
}
Then the Type-Length-Value message is deserialized into the sensor message:
qt::examples::sensors::
Coordinates coord;
coord.deserialize(&
amp;m_serializer, msg.value());
emit coordinatesUpdated(coord);
break
;
Finally, it is converted and displayed to the user:
ui-&
gt;latitudeValue-&
gt;setText(QString::
number(coord.latitude(), 'f'
, 7
));
ui-&
gt;longitudeValue-&
gt;setText(QString::
number(coord.longitude(), 'f'
, 7
));
ui-&
gt;altitudeValue-&
gt;setText(QString::
number(coord.altitude(), 'f'
, 7
));
Before running the example, make sure that your operating system allows binding on UDP port 65500 and sending the data over UDP.