You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
299 lines
9.5 KiB
299 lines
9.5 KiB
#include "selfdrive/ui/qt/widgets/prime.h"
|
|
|
|
#include <QDebug>
|
|
#include <QJsonDocument>
|
|
#include <QJsonObject>
|
|
#include <QLabel>
|
|
#include <QPushButton>
|
|
#include <QStackedWidget>
|
|
#include <QTimer>
|
|
#include <QVBoxLayout>
|
|
#include <QrCode.hpp>
|
|
|
|
#include "selfdrive/ui/qt/request_repeater.h"
|
|
#include "selfdrive/ui/qt/util.h"
|
|
|
|
using qrcodegen::QrCode;
|
|
|
|
PairingQRWidget::PairingQRWidget(QWidget* parent) : QWidget(parent) {
|
|
qrCode = new QLabel;
|
|
qrCode->setScaledContents(true);
|
|
QVBoxLayout* main_layout = new QVBoxLayout(this);
|
|
main_layout->addWidget(qrCode, 0, Qt::AlignCenter);
|
|
|
|
QTimer* timer = new QTimer(this);
|
|
timer->start(30 * 1000);
|
|
connect(timer, &QTimer::timeout, this, &PairingQRWidget::refresh);
|
|
}
|
|
|
|
void PairingQRWidget::showEvent(QShowEvent *event) {
|
|
refresh();
|
|
}
|
|
|
|
void PairingQRWidget::refresh() {
|
|
QString pairToken = CommaApi::create_jwt({{"pair", true}});
|
|
QString qrString = "https://connect.comma.ai/?pair=" + pairToken;
|
|
this->updateQrCode(qrString);
|
|
}
|
|
|
|
void PairingQRWidget::updateQrCode(const QString &text) {
|
|
QrCode qr = QrCode::encodeText(text.toUtf8().data(), QrCode::Ecc::LOW);
|
|
qint32 sz = qr.getSize();
|
|
// make the image larger so we can have a white border
|
|
QImage im(sz + 2, sz + 2, QImage::Format_RGB32);
|
|
QRgb black = qRgb(0, 0, 0);
|
|
QRgb white = qRgb(255, 255, 255);
|
|
|
|
for (int y = 0; y < sz + 2; y++) {
|
|
for (int x = 0; x < sz + 2; x++) {
|
|
im.setPixel(x, y, white);
|
|
}
|
|
}
|
|
for (int y = 0; y < sz; y++) {
|
|
for (int x = 0; x < sz; x++) {
|
|
im.setPixel(x + 1, y + 1, qr.getModule(x, y) ? black : white);
|
|
}
|
|
}
|
|
// Integer division to prevent anti-aliasing
|
|
int approx500 = (500 / (sz + 2)) * (sz + 2);
|
|
qrCode->setPixmap(QPixmap::fromImage(im.scaled(approx500, approx500, Qt::KeepAspectRatio, Qt::FastTransformation), Qt::MonoOnly));
|
|
qrCode->setFixedSize(approx500, approx500);
|
|
}
|
|
|
|
PrimeUserWidget::PrimeUserWidget(QWidget* parent) : QWidget(parent) {
|
|
mainLayout = new QVBoxLayout(this);
|
|
mainLayout->setMargin(0);
|
|
mainLayout->setSpacing(30);
|
|
|
|
// subscribed prime layout
|
|
QWidget *primeWidget = new QWidget;
|
|
primeWidget->setObjectName("primeWidget");
|
|
QVBoxLayout *primeLayout = new QVBoxLayout(primeWidget);
|
|
primeLayout->setMargin(0);
|
|
primeWidget->setContentsMargins(60, 50, 60, 50);
|
|
|
|
QLabel* subscribed = new QLabel("✓ SUBSCRIBED");
|
|
subscribed->setStyleSheet("font-size: 41px; font-weight: bold; color: #86FF4E;");
|
|
primeLayout->addWidget(subscribed, 0, Qt::AlignTop);
|
|
|
|
primeLayout->addSpacing(60);
|
|
|
|
QLabel* commaPrime = new QLabel("comma prime");
|
|
commaPrime->setStyleSheet("font-size: 75px; font-weight: bold;");
|
|
primeLayout->addWidget(commaPrime, 0, Qt::AlignTop);
|
|
|
|
primeLayout->addSpacing(20);
|
|
|
|
QLabel* connectUrl = new QLabel("CONNECT.COMMA.AI");
|
|
connectUrl->setStyleSheet("font-size: 41px; font-family: Inter SemiBold; color: #A0A0A0;");
|
|
primeLayout->addWidget(connectUrl, 0, Qt::AlignTop);
|
|
|
|
mainLayout->addWidget(primeWidget);
|
|
|
|
// comma points layout
|
|
QWidget *pointsWidget = new QWidget;
|
|
pointsWidget->setObjectName("primeWidget");
|
|
QVBoxLayout *pointsLayout = new QVBoxLayout(pointsWidget);
|
|
pointsLayout->setMargin(0);
|
|
pointsWidget->setContentsMargins(60, 50, 60, 50);
|
|
|
|
QLabel* commaPoints = new QLabel("COMMA POINTS");
|
|
commaPoints->setStyleSheet("font-size: 41px; font-family: Inter SemiBold;");
|
|
pointsLayout->addWidget(commaPoints, 0, Qt::AlignTop);
|
|
|
|
points = new QLabel("210");
|
|
points->setStyleSheet("font-size: 91px; font-weight: bold;");
|
|
pointsLayout->addWidget(points, 0, Qt::AlignTop);
|
|
|
|
mainLayout->addWidget(pointsWidget);
|
|
|
|
mainLayout->addStretch();
|
|
|
|
// set up API requests
|
|
if (auto dongleId = getDongleId()) {
|
|
QString url = "https://api.commadotai.com/v1/devices/" + *dongleId + "/owner";
|
|
RequestRepeater *repeater = new RequestRepeater(this, url, "ApiCache_Owner", 6);
|
|
QObject::connect(repeater, &RequestRepeater::receivedResponse, this, &PrimeUserWidget::replyFinished);
|
|
}
|
|
}
|
|
|
|
void PrimeUserWidget::replyFinished(const QString &response) {
|
|
QJsonDocument doc = QJsonDocument::fromJson(response.toUtf8());
|
|
if (doc.isNull()) {
|
|
qDebug() << "JSON Parse failed on getting points";
|
|
return;
|
|
}
|
|
|
|
QJsonObject json = doc.object();
|
|
points->setText(QString::number(json["points"].toInt()));
|
|
}
|
|
|
|
PrimeAdWidget::PrimeAdWidget(QWidget* parent) : QFrame(parent) {
|
|
QVBoxLayout* main_layout = new QVBoxLayout(this);
|
|
main_layout->setContentsMargins(80, 90, 80, 60);
|
|
main_layout->setSpacing(0);
|
|
|
|
QLabel *upgrade = new QLabel("Upgrade Now");
|
|
upgrade->setStyleSheet("font-size: 75px; font-weight: bold;");
|
|
main_layout->addWidget(upgrade, 0, Qt::AlignTop);
|
|
main_layout->addSpacing(50);
|
|
|
|
QLabel *description = new QLabel("Become a comma prime member at connect.comma.ai");
|
|
description->setStyleSheet("font-size: 60px; font-weight: light; color: white;");
|
|
description->setWordWrap(true);
|
|
main_layout->addWidget(description, 0, Qt::AlignTop);
|
|
|
|
main_layout->addStretch();
|
|
|
|
QLabel *features = new QLabel("PRIME FEATURES:");
|
|
features->setStyleSheet("font-size: 41px; font-weight: bold; color: #E5E5E5;");
|
|
main_layout->addWidget(features, 0, Qt::AlignBottom);
|
|
main_layout->addSpacing(30);
|
|
|
|
QVector<QString> bullets = {"Remote access", "14 days of storage", "Developer perks"};
|
|
for (auto &b: bullets) {
|
|
const QString check = "<b><font color='#465BEA'>✓</font></b> ";
|
|
QLabel *l = new QLabel(check + b);
|
|
l->setAlignment(Qt::AlignLeft);
|
|
l->setStyleSheet("font-size: 50px; margin-bottom: 15px;");
|
|
main_layout->addWidget(l, 0, Qt::AlignBottom);
|
|
}
|
|
|
|
setStyleSheet(R"(
|
|
PrimeAdWidget {
|
|
border-radius: 10px;
|
|
background-color: #333333;
|
|
}
|
|
)");
|
|
}
|
|
|
|
|
|
SetupWidget::SetupWidget(QWidget* parent) : QFrame(parent) {
|
|
mainLayout = new QStackedWidget;
|
|
|
|
// Unpaired, registration prompt layout
|
|
|
|
QWidget* finishRegistration = new QWidget;
|
|
finishRegistration->setObjectName("primeWidget");
|
|
QVBoxLayout* finishRegistationLayout = new QVBoxLayout(finishRegistration);
|
|
finishRegistationLayout->setContentsMargins(30, 75, 30, 45);
|
|
finishRegistationLayout->setSpacing(0);
|
|
|
|
QLabel* registrationTitle = new QLabel("Finish Setup");
|
|
registrationTitle->setStyleSheet("font-size: 75px; font-weight: bold; margin-left: 55px;");
|
|
finishRegistationLayout->addWidget(registrationTitle);
|
|
|
|
finishRegistationLayout->addSpacing(30);
|
|
|
|
QLabel* registrationDescription = new QLabel("Pair your device with comma connect (connect.comma.ai) and claim your comma prime offer.");
|
|
registrationDescription->setWordWrap(true);
|
|
registrationDescription->setStyleSheet("font-size: 55px; font-weight: light; margin-left: 55px;");
|
|
finishRegistationLayout->addWidget(registrationDescription);
|
|
|
|
finishRegistationLayout->addStretch();
|
|
|
|
QPushButton* finishButton = new QPushButton("Pair device");
|
|
finishButton->setFixedHeight(220);
|
|
finishButton->setStyleSheet(R"(
|
|
QPushButton {
|
|
font-size: 55px;
|
|
font-weight: 400;
|
|
border-radius: 10px;
|
|
background-color: #465BEA;
|
|
}
|
|
QPushButton:pressed {
|
|
background-color: #3049F4;
|
|
}
|
|
)");
|
|
finishRegistationLayout->addWidget(finishButton);
|
|
QObject::connect(finishButton, &QPushButton::clicked, this, &SetupWidget::showQrCode);
|
|
|
|
mainLayout->addWidget(finishRegistration);
|
|
|
|
// Pairing QR code layout
|
|
|
|
QWidget* q = new QWidget;
|
|
q->setObjectName("primeWidget");
|
|
QVBoxLayout* qrLayout = new QVBoxLayout(q);
|
|
qrLayout->setContentsMargins(90, 90, 90, 90);
|
|
|
|
QLabel* qrLabel = new QLabel("Scan the QR code to pair.");
|
|
qrLabel->setAlignment(Qt::AlignHCenter);
|
|
qrLabel->setStyleSheet("font-size: 47px; font-weight: light;");
|
|
qrLayout->addWidget(qrLabel);
|
|
qrLayout->addSpacing(50);
|
|
|
|
qrLayout->addWidget(new PairingQRWidget);
|
|
qrLayout->addStretch();
|
|
|
|
// setup widget
|
|
QVBoxLayout *outer_layout = new QVBoxLayout(this);
|
|
outer_layout->setContentsMargins(0, 0, 0, 0);
|
|
outer_layout->addWidget(mainLayout);
|
|
|
|
mainLayout->addWidget(q);
|
|
|
|
primeAd = new PrimeAdWidget;
|
|
mainLayout->addWidget(primeAd);
|
|
|
|
primeUser = new PrimeUserWidget;
|
|
mainLayout->addWidget(primeUser);
|
|
|
|
mainLayout->setCurrentWidget(primeAd);
|
|
|
|
setFixedWidth(750);
|
|
setStyleSheet(R"(
|
|
#primeWidget {
|
|
border-radius: 10px;
|
|
background-color: #333333;
|
|
}
|
|
)");
|
|
|
|
// Retain size while hidden
|
|
QSizePolicy sp_retain = sizePolicy();
|
|
sp_retain.setRetainSizeWhenHidden(true);
|
|
setSizePolicy(sp_retain);
|
|
|
|
// set up API requests
|
|
if (auto dongleId = getDongleId()) {
|
|
QString url = "https://api.commadotai.com/v1.1/devices/" + *dongleId + "/";
|
|
RequestRepeater* repeater = new RequestRepeater(this, url, "ApiCache_Device", 5);
|
|
|
|
QObject::connect(repeater, &RequestRepeater::receivedResponse, this, &SetupWidget::replyFinished);
|
|
QObject::connect(repeater, &RequestRepeater::failedResponse, this, &SetupWidget::parseError);
|
|
}
|
|
hide(); // Only show when first request comes back
|
|
}
|
|
|
|
void SetupWidget::parseError(const QString &response) {
|
|
show();
|
|
if (mainLayout->currentIndex() == 1) {
|
|
showQr = false;
|
|
mainLayout->setCurrentIndex(0);
|
|
}
|
|
}
|
|
|
|
void SetupWidget::showQrCode() {
|
|
showQr = true;
|
|
mainLayout->setCurrentIndex(1);
|
|
}
|
|
|
|
void SetupWidget::replyFinished(const QString &response) {
|
|
show();
|
|
QJsonDocument doc = QJsonDocument::fromJson(response.toUtf8());
|
|
if (doc.isNull()) {
|
|
qDebug() << "JSON Parse failed on getting pairing and prime status";
|
|
return;
|
|
}
|
|
|
|
QJsonObject json = doc.object();
|
|
if (!json["is_paired"].toBool()) {
|
|
mainLayout->setCurrentIndex(showQr);
|
|
} else if (!json["prime"].toBool()) {
|
|
showQr = false;
|
|
mainLayout->setCurrentWidget(primeAd);
|
|
} else {
|
|
showQr = false;
|
|
mainLayout->setCurrentWidget(primeUser);
|
|
}
|
|
}
|
|
|