mirror of
https://github.com/sunnypilot/sunnypilot.git
synced 2026-02-18 21:14:01 +08:00
* Squashed commit of the following:
commit 953bcf0ecf8d03f0fec9ce0d5442cc660ae2347d
Merge: b4198608d efd04715e
Author: Shane Smiskol <shane@smiskol.com>
Date: Fri Apr 1 19:58:06 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit b4198608dac9595bfeb9443f95cd6c5d385cd62d
Merge: 24f90b2c8 0e983562d
Author: Shane Smiskol <shane@smiskol.com>
Date: Fri Apr 1 14:32:54 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit 24f90b2c8d6ae509c0776b6e33b743467aa08dee
Author: Shane Smiskol <shane@smiskol.com>
Date: Thu Mar 31 14:07:52 2022 -0700
revert changes and get ready for override
commit aa514df6b5e8431f0faa07f6b25eb5cb6c9f2749
Author: Shane Smiskol <shane@smiskol.com>
Date: Wed Mar 30 20:52:24 2022 -0700
temporary
commit c874e10c2555968c683f629c4582ab22e51d431f
Author: Shane Smiskol <shane@smiskol.com>
Date: Wed Mar 30 00:30:13 2022 -0700
move gasPressed to controlsd
commit fe670439dff5bc6dfc5963d79ca2febe0c5920c7
Merge: 49d3b6d11 c78701e47
Author: Shane Smiskol <shane@smiskol.com>
Date: Wed Mar 30 00:15:48 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit 49d3b6d1162893fa94b2294152f0718eeb4698ed
Merge: 475b27e74 3e819bc5a
Author: Shane Smiskol <shane@smiskol.com>
Date: Mon Mar 28 13:06:08 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit 475b27e74fd6479f1d148e9ded1eca6ee476865c
Author: Shane Smiskol <shane@smiskol.com>
Date: Thu Mar 24 20:24:59 2022 -0700
update SAFETY.md
commit 0798eadb82749e12ecbaa86f4e57b21bd3dbc7e8
Author: Shane Smiskol <shane@smiskol.com>
Date: Thu Mar 24 17:52:06 2022 -0700
⛽ ➡️ 🔋
bump
commit d2b64b89cf8d7265bc90848b4e3a2276c9a4cb78
Author: Shane Smiskol <shane@smiskol.com>
Date: Thu Mar 24 16:19:34 2022 -0700
these changes will be in border PR
these changes will be border PR
commit bed31e63de2244c161729773978fef8a9246ddb5
Merge: 491417640 603e07793
Author: Shane Smiskol <shane@smiskol.com>
Date: Thu Mar 24 16:17:31 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit 49141764098e3a701ff123ead52de4656b31db4d
Merge: 9291e9f00 46ed8bb8e
Author: Shane Smiskol <shane@smiskol.com>
Date: Wed Mar 23 19:00:26 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit 9291e9f0045780669b2ffa7f08fb10373c03b070
Merge: 2aed64157 2c947f193
Author: Shane Smiskol <shane@smiskol.com>
Date: Fri Mar 18 19:03:13 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit 2aed64157d6e9dc993d103a2c6abf7e643410562
Author: Shane Smiskol <shane@smiskol.com>
Date: Fri Mar 18 19:03:03 2022 -0700
Revert "send pre-enable state when gas is pressed"
This reverts commit 8d82e697dc2efade307413e767d14b56b4d2d079.
commit a219defe5342efa731a957b2ff4fd86292c1f239
Author: Shane Smiskol <shane@smiskol.com>
Date: Thu Mar 17 12:50:46 2022 -0700
bump cereal
commit 8d82e697dc2efade307413e767d14b56b4d2d079
Author: Shane Smiskol <shane@smiskol.com>
Date: Wed Mar 16 22:40:13 2022 -0700
send pre-enable state when gas is pressed
formatting and it's out
forgot one
allow gas press
show toggle
revert changes
revert changes
commit 5eb4d1ab1b77ac65bdd901f692ebfb848d5187ca
Merge: bad1bdc74 d0ef25959
Author: Shane Smiskol <shane@smiskol.com>
Date: Wed Mar 16 22:29:43 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit bad1bdc7428c5d8a4a765b259da504f0ade84424
Author: Shane Smiskol <shane@smiskol.com>
Date: Tue Mar 15 23:03:56 2022 -0700
clean up a bit
commit 7516ed9b786536a07f00a92b9c1742ca252a01d1
Author: Shane Smiskol <shane@smiskol.com>
Date: Tue Mar 15 22:55:09 2022 -0700
these are null checks right now
commit 408f5f1d633521b29b08d966c9b387fa79dff0d6
Author: Shane Smiskol <shane@smiskol.com>
Date: Tue Mar 15 22:51:14 2022 -0700
should work
commit be1978e29c03d8843cf073cc9ddf84020fd2b378
Author: Shane Smiskol <shane@smiskol.com>
Date: Tue Mar 15 22:36:16 2022 -0700
stash
commit f0bd4c47e9d27061c6cd3b88765a7c10995f42db
Author: Shane Smiskol <shane@smiskol.com>
Date: Tue Mar 15 22:26:24 2022 -0700
this was flipped
commit d5b6e30389a727e8559331acd706422e1b246f60
Merge: 820b19894 c814c1383
Author: Shane Smiskol <shane@smiskol.com>
Date: Tue Mar 15 22:25:49 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit 820b1989496225df1f66f39b477f1e9402db1b19
Merge: 91763f9ff 6d2483ff9
Author: Shane Smiskol <shane@smiskol.com>
Date: Mon Mar 14 23:39:34 2022 -0700
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit 91763f9ff4321986babce1a7b1272fcf64c8f753
Author: Shane Smiskol <shane@smiskol.com>
Date: Thu Feb 24 20:15:26 2022 -0800
add icon from jozef 🔥
commit 1885c9e2bacd9351c7e421b1436dcbfbde238436
Merge: 1f591736a eae207a84
Author: Shane Smiskol <shane@smiskol.com>
Date: Thu Feb 24 20:08:07 2022 -0800
Merge remote-tracking branch 'upstream/master' into disengage-on-gas
commit 1f591736a2767b1ed892c29ec5000cc7348f01c7
Merge: 465811f0b ebf5b27a7
Author: Jason Wen <47793918+sunnyhaibin@users.noreply.github.com>
Date: Sat Feb 12 10:24:29 2022 -0500
Merge branch 'master' into disengage-on-gas
commit 465811f0b4776e43a289398e4d4369362b7da58c
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Fri Feb 11 11:20:32 2022 -0500
Add param to process replay
commit 0344363a533505435cff87eb33356b2364a4156f
Merge: 700efcb3f 1175355b7
Author: Jason Wen <47793918+sunnyhaibin@users.noreply.github.com>
Date: Fri Feb 11 11:07:31 2022 -0500
Merge branch 'master' into disengage-on-gas
commit 700efcb3f444092f99ef10e9908f83402429d742
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Fri Feb 11 11:04:39 2022 -0500
Add param to unsafeMode
commit 220ce272fee1f7f5ace2a04052605a6e4f95712b
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Fri Feb 11 10:54:16 2022 -0500
Add param to unsafeMode
commit d273bb78acbc23ee5255e461fa71b2f0400a5e03
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Fri Feb 11 10:52:22 2022 -0500
Resolve conflicts
commit 1a85afd60c2f0d3140dfa8e7cd574d6b88abd5e3
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Fri Feb 11 10:51:50 2022 -0500
Resolve conflicts
commit e3be32afc2bb2423d29e2b4f85b3cff3fe72aab9
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Fri Feb 11 10:51:08 2022 -0500
Resolve conflicts
commit f27203af3afe612173bf9a63e15c57118598b5b8
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Wed Jan 19 00:30:09 2022 -0500
Add param to process replay
commit 04c0ad1a54f4fea5efe74b3203d9bdbd693b64a6
Author: Jason Wen <47793918+sunnyhaibin@users.noreply.github.com>
Date: Wed Jan 19 00:28:49 2022 -0500
Update interfaces.py
Co-authored-by: Willem Melching <willem.melching@gmail.com>
commit 99cf13caeb4d9f2c95c25e1e78b6e353ea508cb6
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Sat Jan 15 00:03:38 2022 -0500
Fix missing params lib
commit 2230254ca709625d591b0c73d1a6022c450f6ef6
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Fri Jan 14 23:38:46 2022 -0500
Disengage on gas toggle
commit 87475e02cb63cc39c9b16edc67f303e02d17e6ef
Author: Jason Wen <haibin.wen3@gmail.com>
Date: Fri Jan 14 23:24:00 2022 -0500
Disengage on gas toggle
* bump
* use ALTERNATIVE_EXPERIENCE
* or it
* de-bump
* update refs
* update refs!
* you can just ctrl+click ALTERNATIVE_EXPERIENCE!
* already have a params
* update text
old-commit-hash: f1f40c117a
466 lines
16 KiB
C++
466 lines
16 KiB
C++
#include "selfdrive/ui/qt/offroad/settings.h"
|
||
|
||
#include <cassert>
|
||
#include <cmath>
|
||
#include <string>
|
||
|
||
#include <QDebug>
|
||
|
||
#ifndef QCOM
|
||
#include "selfdrive/ui/qt/offroad/networking.h"
|
||
#endif
|
||
|
||
#ifdef ENABLE_MAPS
|
||
#include "selfdrive/ui/qt/maps/map_settings.h"
|
||
#endif
|
||
|
||
#include "selfdrive/common/params.h"
|
||
#include "selfdrive/common/util.h"
|
||
#include "selfdrive/hardware/hw.h"
|
||
#include "selfdrive/ui/qt/widgets/controls.h"
|
||
#include "selfdrive/ui/qt/widgets/input.h"
|
||
#include "selfdrive/ui/qt/widgets/scrollview.h"
|
||
#include "selfdrive/ui/qt/widgets/ssh_keys.h"
|
||
#include "selfdrive/ui/qt/widgets/toggle.h"
|
||
#include "selfdrive/ui/ui.h"
|
||
#include "selfdrive/ui/qt/util.h"
|
||
#include "selfdrive/ui/qt/qt_window.h"
|
||
|
||
TogglesPanel::TogglesPanel(SettingsWindow *parent) : ListWidget(parent) {
|
||
// param, title, desc, icon
|
||
std::vector<std::tuple<QString, QString, QString, QString>> toggles{
|
||
{
|
||
"OpenpilotEnabledToggle",
|
||
"Enable openpilot",
|
||
"Use the openpilot system for adaptive cruise control and lane keep driver assistance. Your attention is required at all times to use this feature. Changing this setting takes effect when the car is powered off.",
|
||
"../assets/offroad/icon_openpilot.png",
|
||
},
|
||
{
|
||
"IsLdwEnabled",
|
||
"Enable Lane Departure Warnings",
|
||
"Receive alerts to steer back into the lane when your vehicle drifts over a detected lane line without a turn signal activated while driving over 31 mph (50 km/h).",
|
||
"../assets/offroad/icon_warning.png",
|
||
},
|
||
{
|
||
"IsRHD",
|
||
"Enable Right-Hand Drive",
|
||
"Allow openpilot to obey left-hand traffic conventions and perform driver monitoring on right driver seat.",
|
||
"../assets/offroad/icon_openpilot_mirrored.png",
|
||
},
|
||
{
|
||
"IsMetric",
|
||
"Use Metric System",
|
||
"Display speed in km/h instead of mph.",
|
||
"../assets/offroad/icon_metric.png",
|
||
},
|
||
{
|
||
"RecordFront",
|
||
"Record and Upload Driver Camera",
|
||
"Upload data from the driver facing camera and help improve the driver monitoring algorithm.",
|
||
"../assets/offroad/icon_monitoring.png",
|
||
},
|
||
{
|
||
"EndToEndToggle",
|
||
"\U0001f96c Disable use of lanelines (Alpha) \U0001f96c",
|
||
"In this mode openpilot will ignore lanelines and just drive how it thinks a human would.",
|
||
"../assets/offroad/icon_road.png",
|
||
},
|
||
{
|
||
"DisengageOnAccelerator",
|
||
"Disengage On Accelerator Pedal",
|
||
"When enabled, openpilot will disengage when the accelerator pedal is pressed.",
|
||
"../assets/offroad/icon_disengage_on_accelerator.svg",
|
||
},
|
||
#ifdef ENABLE_MAPS
|
||
{
|
||
"NavSettingTime24h",
|
||
"Show ETA in 24h format",
|
||
"Use 24h format instead of am/pm",
|
||
"../assets/offroad/icon_metric.png",
|
||
},
|
||
#endif
|
||
|
||
};
|
||
|
||
Params params;
|
||
|
||
if (params.getBool("DisableRadar_Allow")) {
|
||
toggles.push_back({
|
||
"DisableRadar",
|
||
"openpilot Longitudinal Control",
|
||
"openpilot will disable the car's radar and will take over control of gas and brakes. Warning: this disables AEB!",
|
||
"../assets/offroad/icon_speed_limit.png",
|
||
});
|
||
}
|
||
|
||
for (auto &[param, title, desc, icon] : toggles) {
|
||
auto toggle = new ParamControl(param, title, desc, icon, this);
|
||
bool locked = params.getBool((param + "Lock").toStdString());
|
||
toggle->setEnabled(!locked);
|
||
if (!locked) {
|
||
connect(uiState(), &UIState::offroadTransition, toggle, &ParamControl::setEnabled);
|
||
}
|
||
addItem(toggle);
|
||
}
|
||
}
|
||
|
||
DevicePanel::DevicePanel(SettingsWindow *parent) : ListWidget(parent) {
|
||
setSpacing(50);
|
||
addItem(new LabelControl("Dongle ID", getDongleId().value_or("N/A")));
|
||
addItem(new LabelControl("Serial", params.get("HardwareSerial").c_str()));
|
||
|
||
// offroad-only buttons
|
||
|
||
auto dcamBtn = new ButtonControl("Driver Camera", "PREVIEW",
|
||
"Preview the driver facing camera to help optimize device mounting position for best driver monitoring experience. (vehicle must be off)");
|
||
connect(dcamBtn, &ButtonControl::clicked, [=]() { emit showDriverView(); });
|
||
addItem(dcamBtn);
|
||
|
||
auto resetCalibBtn = new ButtonControl("Reset Calibration", "RESET", " ");
|
||
connect(resetCalibBtn, &ButtonControl::showDescription, this, &DevicePanel::updateCalibDescription);
|
||
connect(resetCalibBtn, &ButtonControl::clicked, [&]() {
|
||
if (ConfirmationDialog::confirm("Are you sure you want to reset calibration?", this)) {
|
||
params.remove("CalibrationParams");
|
||
}
|
||
});
|
||
addItem(resetCalibBtn);
|
||
|
||
if (!params.getBool("Passive")) {
|
||
auto retrainingBtn = new ButtonControl("Review Training Guide", "REVIEW", "Review the rules, features, and limitations of openpilot");
|
||
connect(retrainingBtn, &ButtonControl::clicked, [=]() {
|
||
if (ConfirmationDialog::confirm("Are you sure you want to review the training guide?", this)) {
|
||
emit reviewTrainingGuide();
|
||
}
|
||
});
|
||
addItem(retrainingBtn);
|
||
}
|
||
|
||
if (Hardware::TICI()) {
|
||
auto regulatoryBtn = new ButtonControl("Regulatory", "VIEW", "");
|
||
connect(regulatoryBtn, &ButtonControl::clicked, [=]() {
|
||
const std::string txt = util::read_file("../assets/offroad/fcc.html");
|
||
RichTextDialog::alert(QString::fromStdString(txt), this);
|
||
});
|
||
addItem(regulatoryBtn);
|
||
}
|
||
|
||
QObject::connect(uiState(), &UIState::offroadTransition, [=](bool offroad) {
|
||
for (auto btn : findChildren<ButtonControl *>()) {
|
||
btn->setEnabled(offroad);
|
||
}
|
||
});
|
||
|
||
// power buttons
|
||
QHBoxLayout *power_layout = new QHBoxLayout();
|
||
power_layout->setSpacing(30);
|
||
|
||
QPushButton *reboot_btn = new QPushButton("Reboot");
|
||
reboot_btn->setObjectName("reboot_btn");
|
||
power_layout->addWidget(reboot_btn);
|
||
QObject::connect(reboot_btn, &QPushButton::clicked, this, &DevicePanel::reboot);
|
||
|
||
QPushButton *poweroff_btn = new QPushButton("Power Off");
|
||
poweroff_btn->setObjectName("poweroff_btn");
|
||
power_layout->addWidget(poweroff_btn);
|
||
QObject::connect(poweroff_btn, &QPushButton::clicked, this, &DevicePanel::poweroff);
|
||
|
||
if (Hardware::TICI()) {
|
||
connect(uiState(), &UIState::offroadTransition, poweroff_btn, &QPushButton::setVisible);
|
||
}
|
||
|
||
setStyleSheet(R"(
|
||
#reboot_btn { height: 120px; border-radius: 15px; background-color: #393939; }
|
||
#reboot_btn:pressed { background-color: #4a4a4a; }
|
||
#poweroff_btn { height: 120px; border-radius: 15px; background-color: #E22C2C; }
|
||
#poweroff_btn:pressed { background-color: #FF2424; }
|
||
)");
|
||
addItem(power_layout);
|
||
}
|
||
|
||
void DevicePanel::updateCalibDescription() {
|
||
QString desc =
|
||
"openpilot requires the device to be mounted within 4° left or right and "
|
||
"within 5° up or 8° down. openpilot is continuously calibrating, resetting is rarely required.";
|
||
std::string calib_bytes = Params().get("CalibrationParams");
|
||
if (!calib_bytes.empty()) {
|
||
try {
|
||
AlignedBuffer aligned_buf;
|
||
capnp::FlatArrayMessageReader cmsg(aligned_buf.align(calib_bytes.data(), calib_bytes.size()));
|
||
auto calib = cmsg.getRoot<cereal::Event>().getLiveCalibration();
|
||
if (calib.getCalStatus() != 0) {
|
||
double pitch = calib.getRpyCalib()[1] * (180 / M_PI);
|
||
double yaw = calib.getRpyCalib()[2] * (180 / M_PI);
|
||
desc += QString(" Your device is pointed %1° %2 and %3° %4.")
|
||
.arg(QString::number(std::abs(pitch), 'g', 1), pitch > 0 ? "down" : "up",
|
||
QString::number(std::abs(yaw), 'g', 1), yaw > 0 ? "left" : "right");
|
||
}
|
||
} catch (kj::Exception) {
|
||
qInfo() << "invalid CalibrationParams";
|
||
}
|
||
}
|
||
qobject_cast<ButtonControl *>(sender())->setDescription(desc);
|
||
}
|
||
|
||
void DevicePanel::reboot() {
|
||
if (!uiState()->engaged()) {
|
||
if (ConfirmationDialog::confirm("Are you sure you want to reboot?", this)) {
|
||
// Check engaged again in case it changed while the dialog was open
|
||
if (!uiState()->engaged()) {
|
||
Params().putBool("DoReboot", true);
|
||
}
|
||
}
|
||
} else {
|
||
ConfirmationDialog::alert("Disengage to Reboot", this);
|
||
}
|
||
}
|
||
|
||
void DevicePanel::poweroff() {
|
||
if (!uiState()->engaged()) {
|
||
if (ConfirmationDialog::confirm("Are you sure you want to power off?", this)) {
|
||
// Check engaged again in case it changed while the dialog was open
|
||
if (!uiState()->engaged()) {
|
||
Params().putBool("DoShutdown", true);
|
||
}
|
||
}
|
||
} else {
|
||
ConfirmationDialog::alert("Disengage to Power Off", this);
|
||
}
|
||
}
|
||
|
||
SoftwarePanel::SoftwarePanel(QWidget* parent) : ListWidget(parent) {
|
||
gitBranchLbl = new LabelControl("Git Branch");
|
||
gitCommitLbl = new LabelControl("Git Commit");
|
||
osVersionLbl = new LabelControl("OS Version");
|
||
versionLbl = new LabelControl("Version", "", QString::fromStdString(params.get("ReleaseNotes")).trimmed());
|
||
lastUpdateLbl = new LabelControl("Last Update Check", "", "The last time openpilot successfully checked for an update. The updater only runs while the car is off.");
|
||
updateBtn = new ButtonControl("Check for Update", "");
|
||
connect(updateBtn, &ButtonControl::clicked, [=]() {
|
||
if (params.getBool("IsOffroad")) {
|
||
fs_watch->addPath(QString::fromStdString(params.getParamPath("LastUpdateTime")));
|
||
fs_watch->addPath(QString::fromStdString(params.getParamPath("UpdateFailedCount")));
|
||
updateBtn->setText("CHECKING");
|
||
updateBtn->setEnabled(false);
|
||
}
|
||
std::system("pkill -1 -f selfdrive.updated");
|
||
});
|
||
|
||
|
||
auto uninstallBtn = new ButtonControl("Uninstall " + getBrand(), "UNINSTALL");
|
||
connect(uninstallBtn, &ButtonControl::clicked, [&]() {
|
||
if (ConfirmationDialog::confirm("Are you sure you want to uninstall?", this)) {
|
||
params.putBool("DoUninstall", true);
|
||
}
|
||
});
|
||
connect(uiState(), &UIState::offroadTransition, uninstallBtn, &QPushButton::setEnabled);
|
||
|
||
QWidget *widgets[] = {versionLbl, lastUpdateLbl, updateBtn, gitBranchLbl, gitCommitLbl, osVersionLbl, uninstallBtn};
|
||
for (QWidget* w : widgets) {
|
||
addItem(w);
|
||
}
|
||
|
||
fs_watch = new QFileSystemWatcher(this);
|
||
QObject::connect(fs_watch, &QFileSystemWatcher::fileChanged, [=](const QString path) {
|
||
if (path.contains("UpdateFailedCount") && std::atoi(params.get("UpdateFailedCount").c_str()) > 0) {
|
||
lastUpdateLbl->setText("failed to fetch update");
|
||
updateBtn->setText("CHECK");
|
||
updateBtn->setEnabled(true);
|
||
} else if (path.contains("LastUpdateTime")) {
|
||
updateLabels();
|
||
}
|
||
});
|
||
}
|
||
|
||
void SoftwarePanel::showEvent(QShowEvent *event) {
|
||
updateLabels();
|
||
}
|
||
|
||
void SoftwarePanel::updateLabels() {
|
||
QString lastUpdate = "";
|
||
auto tm = params.get("LastUpdateTime");
|
||
if (!tm.empty()) {
|
||
lastUpdate = timeAgo(QDateTime::fromString(QString::fromStdString(tm + "Z"), Qt::ISODate));
|
||
}
|
||
|
||
versionLbl->setText(getBrandVersion());
|
||
lastUpdateLbl->setText(lastUpdate);
|
||
updateBtn->setText("CHECK");
|
||
updateBtn->setEnabled(true);
|
||
gitBranchLbl->setText(QString::fromStdString(params.get("GitBranch")));
|
||
gitCommitLbl->setText(QString::fromStdString(params.get("GitCommit")).left(10));
|
||
osVersionLbl->setText(QString::fromStdString(Hardware::get_os_version()).trimmed());
|
||
}
|
||
|
||
C2NetworkPanel::C2NetworkPanel(QWidget *parent) : QWidget(parent) {
|
||
QVBoxLayout *layout = new QVBoxLayout(this);
|
||
layout->setContentsMargins(50, 0, 50, 0);
|
||
|
||
ListWidget *list = new ListWidget();
|
||
list->setSpacing(30);
|
||
// wifi + tethering buttons
|
||
#ifdef QCOM
|
||
auto wifiBtn = new ButtonControl("Wi-Fi Settings", "OPEN");
|
||
QObject::connect(wifiBtn, &ButtonControl::clicked, [=]() { HardwareEon::launch_wifi(); });
|
||
list->addItem(wifiBtn);
|
||
|
||
auto tetheringBtn = new ButtonControl("Tethering Settings", "OPEN");
|
||
QObject::connect(tetheringBtn, &ButtonControl::clicked, [=]() { HardwareEon::launch_tethering(); });
|
||
list->addItem(tetheringBtn);
|
||
#endif
|
||
ipaddress = new LabelControl("IP Address", "");
|
||
list->addItem(ipaddress);
|
||
|
||
// SSH key management
|
||
list->addItem(new SshToggle());
|
||
list->addItem(new SshControl());
|
||
layout->addWidget(list);
|
||
layout->addStretch(1);
|
||
}
|
||
|
||
void C2NetworkPanel::showEvent(QShowEvent *event) {
|
||
ipaddress->setText(getIPAddress());
|
||
}
|
||
|
||
QString C2NetworkPanel::getIPAddress() {
|
||
std::string result = util::check_output("ifconfig wlan0");
|
||
if (result.empty()) return "";
|
||
|
||
const std::string inetaddrr = "inet addr:";
|
||
std::string::size_type begin = result.find(inetaddrr);
|
||
if (begin == std::string::npos) return "";
|
||
|
||
begin += inetaddrr.length();
|
||
std::string::size_type end = result.find(' ', begin);
|
||
if (end == std::string::npos) return "";
|
||
|
||
return result.substr(begin, end - begin).c_str();
|
||
}
|
||
|
||
QWidget *network_panel(QWidget *parent) {
|
||
#ifdef QCOM
|
||
return new C2NetworkPanel(parent);
|
||
#else
|
||
return new Networking(parent);
|
||
#endif
|
||
}
|
||
|
||
void SettingsWindow::showEvent(QShowEvent *event) {
|
||
panel_widget->setCurrentIndex(0);
|
||
nav_btns->buttons()[0]->setChecked(true);
|
||
}
|
||
|
||
SettingsWindow::SettingsWindow(QWidget *parent) : QFrame(parent) {
|
||
|
||
// setup two main layouts
|
||
sidebar_widget = new QWidget;
|
||
QVBoxLayout *sidebar_layout = new QVBoxLayout(sidebar_widget);
|
||
sidebar_layout->setMargin(0);
|
||
panel_widget = new QStackedWidget();
|
||
panel_widget->setStyleSheet(R"(
|
||
border-radius: 30px;
|
||
background-color: #292929;
|
||
)");
|
||
|
||
// close button
|
||
QPushButton *close_btn = new QPushButton("×");
|
||
close_btn->setStyleSheet(R"(
|
||
QPushButton {
|
||
font-size: 140px;
|
||
padding-bottom: 20px;
|
||
font-weight: bold;
|
||
border 1px grey solid;
|
||
border-radius: 100px;
|
||
background-color: #292929;
|
||
font-weight: 400;
|
||
}
|
||
QPushButton:pressed {
|
||
background-color: #3B3B3B;
|
||
}
|
||
)");
|
||
close_btn->setFixedSize(200, 200);
|
||
sidebar_layout->addSpacing(45);
|
||
sidebar_layout->addWidget(close_btn, 0, Qt::AlignCenter);
|
||
QObject::connect(close_btn, &QPushButton::clicked, this, &SettingsWindow::closeSettings);
|
||
|
||
// setup panels
|
||
DevicePanel *device = new DevicePanel(this);
|
||
QObject::connect(device, &DevicePanel::reviewTrainingGuide, this, &SettingsWindow::reviewTrainingGuide);
|
||
QObject::connect(device, &DevicePanel::showDriverView, this, &SettingsWindow::showDriverView);
|
||
|
||
QList<QPair<QString, QWidget *>> panels = {
|
||
{"Device", device},
|
||
{"Network", network_panel(this)},
|
||
{"Toggles", new TogglesPanel(this)},
|
||
{"Software", new SoftwarePanel(this)},
|
||
};
|
||
|
||
#ifdef ENABLE_MAPS
|
||
auto map_panel = new MapPanel(this);
|
||
panels.push_back({"Navigation", map_panel});
|
||
QObject::connect(map_panel, &MapPanel::closeSettings, this, &SettingsWindow::closeSettings);
|
||
#endif
|
||
|
||
const int padding = panels.size() > 3 ? 25 : 35;
|
||
|
||
nav_btns = new QButtonGroup(this);
|
||
for (auto &[name, panel] : panels) {
|
||
QPushButton *btn = new QPushButton(name);
|
||
btn->setCheckable(true);
|
||
btn->setChecked(nav_btns->buttons().size() == 0);
|
||
btn->setStyleSheet(QString(R"(
|
||
QPushButton {
|
||
color: grey;
|
||
border: none;
|
||
background: none;
|
||
font-size: 65px;
|
||
font-weight: 500;
|
||
padding-top: %1px;
|
||
padding-bottom: %1px;
|
||
}
|
||
QPushButton:checked {
|
||
color: white;
|
||
}
|
||
QPushButton:pressed {
|
||
color: #ADADAD;
|
||
}
|
||
)").arg(padding));
|
||
|
||
nav_btns->addButton(btn);
|
||
sidebar_layout->addWidget(btn, 0, Qt::AlignRight);
|
||
|
||
const int lr_margin = name != "Network" ? 50 : 0; // Network panel handles its own margins
|
||
panel->setContentsMargins(lr_margin, 25, lr_margin, 25);
|
||
|
||
ScrollView *panel_frame = new ScrollView(panel, this);
|
||
panel_widget->addWidget(panel_frame);
|
||
|
||
QObject::connect(btn, &QPushButton::clicked, [=, w = panel_frame]() {
|
||
btn->setChecked(true);
|
||
panel_widget->setCurrentWidget(w);
|
||
});
|
||
}
|
||
sidebar_layout->setContentsMargins(50, 50, 100, 50);
|
||
|
||
// main settings layout, sidebar + main panel
|
||
QHBoxLayout *main_layout = new QHBoxLayout(this);
|
||
|
||
sidebar_widget->setFixedWidth(500);
|
||
main_layout->addWidget(sidebar_widget);
|
||
main_layout->addWidget(panel_widget);
|
||
|
||
setStyleSheet(R"(
|
||
* {
|
||
color: white;
|
||
font-size: 50px;
|
||
}
|
||
SettingsWindow {
|
||
background-color: black;
|
||
}
|
||
)");
|
||
}
|
||
|
||
void SettingsWindow::hideEvent(QHideEvent *event) {
|
||
#ifdef QCOM
|
||
HardwareEon::close_activities();
|
||
#endif
|
||
}
|