IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)

Bluetooth QML Ping Pong example

A QML example showing Bluetooth communication.

Article lu   fois.

L'auteur

Liens sociaux

Viadeo Twitter Facebook Share on Google+   

Bluetooth QML Ping Pong example

The Bluetooth QML Ping Pong example presents the socket communication between two Bluetooth devices. The basic concept is the ping pong game where two players communicate via sockets.

Image non disponible

Running the Example

To run the example from Qt Creator, open the Welcome mode and select the example from Examples. For more information, visit Building and Running an Example.

At the beginning, the user selects the role. One device acts as a server and the second one as a client. The server side starts a service named "PingPong server".

 
Sélectionnez
m_serverInfo = new QBluetoothServer(QBluetoothServiceInfo::RfcommProtocol, this);
connect(m_serverInfo, &QBluetoothServer::newConnection,
        this, &PingPong::clientConnected);
connect(m_serverInfo, &QBluetoothServer::errorOccurred, this, &PingPong::serverError);
const QBluetoothUuid uuid(serviceUuid);

m_serviceInfo = m_serverInfo->listen(uuid, QStringLiteral("PingPong server"));

On the client side, the full service discovery on the nearby Bluetooth devices is done.

 
Sélectionnez
discoveryAgent = new QBluetoothServiceDiscoveryAgent(QBluetoothAddress());

connect(discoveryAgent, &QBluetoothServiceDiscoveryAgent::serviceDiscovered,
        this, &PingPong::addService);
connect(discoveryAgent, &QBluetoothServiceDiscoveryAgent::finished,
        this, &PingPong::done);
connect(discoveryAgent, &QBluetoothServiceDiscoveryAgent::errorOccurred, this,
        &PingPong::serviceScanError);
#ifdef Q_OS_ANDROID
// QTBUG-61392
discoveryAgent->setUuidFilter(QBluetoothUuid(androidUuid));
#else
discoveryAgent->setUuidFilter(QBluetoothUuid(serviceUuid));
#endif
discoveryAgent->start(QBluetoothServiceDiscoveryAgent::FullDiscovery);

When the ping pong service is discovered, the client connects to the server using the socket.

 
Sélectionnez
socket = new QBluetoothSocket(QBluetoothServiceInfo::RfcommProtocol);
socket->connectToService(service);

connect(socket, &QBluetoothSocket::readyRead, this, &PingPong::readSocket);
connect(socket, &QBluetoothSocket::connected, this, &PingPong::serverConnected);
connect(socket, &QBluetoothSocket::disconnected, this, &PingPong::serverDisconnected);

On the server side, the connected signal is emitted initiating that the client is connected. The necessary signals and slots on the server side are connected.

 
Sélectionnez
if (!m_serverInfo->hasPendingConnections()) {
    setMessage("FAIL: expected pending server connection");
    return;
}
socket = m_serverInfo->nextPendingConnection();
if (!socket)
    return;
socket->setParent(this);
connect(socket, &QBluetoothSocket::readyRead,
        this, &PingPong::readSocket);
connect(socket, &QBluetoothSocket::disconnected,
        this, &PingPong::clientDisconnected);
connect(socket, &QBluetoothSocket::errorOccurred, this, &PingPong::socketError);

The game starts after the devices are connected and the screen is adjusted.

 
Sélectionnez
m_timer->start(10);

The server updates the ball direction and coordinates. The coordinates of pedals are sent to each other every 10ms.

 
Sélectionnez
if (m_role == 1) {
    checkBoundaries();
    m_ballY += m_speedy;
    m_ballX += m_speedx;

    size.setNum(m_ballX);
    size.append(' ');
    QByteArray size1;
    size1.setNum(m_ballY);
    size.append(size1);
    size.append(' ');
    size1.setNum(m_leftBlockY);
    size.append(size1);
    size.append(" \n");
    socket->write(size.constData());
    emit ballChanged();
}
else if (m_role == 2) {
    size.setNum(m_rightBlockY);
    size.append(" \n");
    socket->write(size.constData());
}

The coordinates are updated and exchanged via sockets. As presented, the server sends its pedal's y coordinate and the ball coordinates whereas, the client sends only its pedal y coordinate.

 
Sélectionnez
if (((m_ballX + ballWidth) > (1. - blockSize)) && (ballCenterY < (m_rightBlockY + blockHeight))
    && (ballCenterY > m_rightBlockY)) {
    // right paddle collision
    // simulating paddle surface to be a quarter of a circle
    float paddlecenter = m_rightBlockY + blockHeight / 2.;
    float relpos = (ballCenterY - paddlecenter) / (blockHeight / 2.); // [-1 : 1]
    float surfaceangle = M_PI_4 * relpos;

    // calculation of surface normal
    float normalx = -cos(surfaceangle);
    float normaly = sin(surfaceangle);

    // calculation of surface tangent
    float tangentx = sin(surfaceangle);
    float tangenty = cos(surfaceangle);

    // calculation of tangentialspeed
    float tangentialspeed = tangentx * m_speedx + tangenty * m_speedy;
    // calculation of normal speed
    float normalspeed = normalx * m_speedx + normaly * m_speedy;

    // speed increase of 10%
    normalspeed *= 1.1f;

    if (normalspeed < 0) { // if we are coming from the left. To avoid double reflections
        m_speedx = tangentialspeed * tangentx - normalspeed * normalx;
        m_speedy = tangentialspeed * tangenty - normalspeed * normaly;
    }
} else if ((m_ballX < blockSize) && (ballCenterY < (m_leftBlockY + blockHeight))
           && (ballCenterY > m_leftBlockY)) {
    // left paddle collision
    // simulating paddle surface to be a quarter of a circle
    float paddlecenter = m_leftBlockY + blockHeight / 2.;
    float relpos = (ballCenterY - paddlecenter) / (blockHeight / 2.); // [-1 : 1]
    float surfaceangle = M_PI_4 * relpos;

    // calculation of surface normal
    float normalx = cos(surfaceangle);
    float normaly = sin(surfaceangle);

    // calculation of surface tangent
    float tangentx = -sin(surfaceangle);
    float tangenty = cos(surfaceangle);

    // calculation of tangentialspeed
    float tangentialspeed = tangentx * m_speedx + tangenty * m_speedy;
    // calculation of normal speed
    float normalspeed = normalx * m_speedx + normaly * m_speedy;

    // speed increase of 10%
    normalspeed *= 1.1f;

    if (normalspeed < 0) { // if we are coming from the left. To avoid double reflections
        m_speedx = tangentialspeed * tangentx - normalspeed * normalx;
        m_speedy = tangentialspeed * tangenty - normalspeed * normaly;
    }
} else if (m_ballY < 0) {
    m_speedy = std::abs(m_speedy);
} else if (m_ballY + ballWidth > 1.f) {
    m_speedy = -std::abs(m_speedy);
}

In the code above, it was shown how the server checks whether the ball has reached the boundaries of the board. In case of a collision with a pedal, a convex surface is simulated. The part of the velocity normal to the simulated surface is inverted and increased by 10% to speed up the game. In the case of the goal, the server updates the results via its socket.

Image non disponible

Example project

Vous avez aimé ce tutoriel ? Alors partagez-le en cliquant sur les boutons suivants : Viadeo Twitter Facebook Share on Google+