#include #include #include #include #include #include #include #include #include #include #include "wifi.hpp" #include "settings.hpp" #include "widgets/toggle.hpp" #include "widgets/offroad_alerts.hpp" #include "common/params.h" #include "common/utilpp.h" ParamsToggle::ParamsToggle(QString param, QString title, QString description, QString icon_path, QWidget *parent): QFrame(parent) , param(param) { QHBoxLayout *hlayout = new QHBoxLayout; // Parameter image hlayout->addSpacing(25); if (icon_path.length()){ QPixmap pix(icon_path); QLabel *icon = new QLabel(); icon->setPixmap(pix.scaledToWidth(100, Qt::SmoothTransformation)); icon->setSizePolicy(QSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed)); hlayout->addWidget(icon); } else{ hlayout->addSpacing(100); } hlayout->addSpacing(25); // Name of the parameter QLabel *label = new QLabel(title); label->setWordWrap(true); // toggle switch Toggle* toggle_switch = new Toggle(this); toggle_switch->setFixedSize(150, 100); // TODO: show descriptions on tap hlayout->addWidget(label); hlayout->addSpacing(50); hlayout->addWidget(toggle_switch); hlayout->addSpacing(20); setLayout(hlayout); if(Params().read_db_bool(param.toStdString().c_str())){ toggle_switch->togglePosition(); } setStyleSheet(R"( QLabel { font-size: 50px; } * { background-color: #114265; } )"); QObject::connect(toggle_switch, SIGNAL(stateChanged(int)), this, SLOT(checkboxClicked(int))); } void ParamsToggle::checkboxClicked(int state){ char value = state ? '1': '0'; Params().write_db_value(param.toStdString().c_str(), &value, 1); } QWidget * toggles_panel() { QVBoxLayout *toggles_list = new QVBoxLayout(); toggles_list->setSpacing(25); toggles_list->addWidget(new ParamsToggle("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" )); toggles_list->addWidget(new ParamsToggle("LaneChangeEnabled", "Enable Lane Change Assist", "Perform assisted lane changes with openpilot by checking your surroundings for safety, activating the turn signal and gently nudging the steering wheel towards your desired lane. openpilot is not capable of checking if a lane change is safe. You must continuously observe your surroundings to use this feature.", "../assets/offroad/icon_road.png" )); toggles_list->addWidget(new ParamsToggle("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 31mph (50kph).", "../assets/offroad/icon_warning.png" )); toggles_list->addWidget(new ParamsToggle("RecordFront", "Record and Upload Driver Camera", "Upload data from the driver facing camera and help improve the driver monitoring algorithm.", "../assets/offroad/icon_network.png" )); toggles_list->addWidget(new ParamsToggle("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" )); toggles_list->addWidget(new ParamsToggle("IsMetric", "Use Metric System", "Display speed in km/h instead of mp/h.", "../assets/offroad/icon_metric.png" )); toggles_list->addWidget(new ParamsToggle("CommunityFeaturesToggle", "Enable Community Features", "Use features from the open source community that are not maintained or supported by comma.ai and have not been confirmed to meet the standard safety model. These features include community supported cars and community supported hardware. Be extra cautious when using these features", "../assets/offroad/icon_shell.png" )); QWidget *widget = new QWidget; widget->setLayout(toggles_list); return widget; } QWidget * device_panel() { QVBoxLayout *device_layout = new QVBoxLayout; device_layout->setSpacing(50); Params params = Params(); std::vector> labels = { {"Dongle ID", params.get("DongleId", false)}, //{"Serial Number", "abcdefghijk"}, }; for (auto l : labels) { QString text = QString::fromStdString(l.first + ": " + l.second); device_layout->addWidget(new QLabel(text)); } QPushButton *clear_cal_btn = new QPushButton("Reset Calibration"); device_layout->addWidget(clear_cal_btn); QObject::connect(clear_cal_btn, &QPushButton::released, [=]() { Params().delete_db_value("CalibrationParams"); }); std::map power_btns = { {"Power Off", "sudo poweroff"}, {"Reboot", "sudo reboot"}, }; for (auto b : power_btns) { QPushButton *btn = new QPushButton(QString::fromStdString(b.first)); device_layout->addWidget(btn); #ifdef __aarch64__ QObject::connect(btn, &QPushButton::released, [=]() {std::system(b.second);}); #endif } QWidget *widget = new QWidget; widget->setLayout(device_layout); widget->setStyleSheet(R"( QPushButton { padding: 60px; } )"); return widget; } QWidget * developer_panel() { QVBoxLayout *main_layout = new QVBoxLayout; // TODO: enable SSH toggle and github keys Params params = Params(); std::string brand = params.read_db_bool("Passive") ? "dashcam" : "openpilot"; std::vector> labels = { {"Version", brand + " v" + params.get("Version", false)}, {"Git Branch", params.get("GitBranch", false)}, {"Git Commit", params.get("GitCommit", false).substr(0, 10)}, {"Panda Firmware", params.get("PandaFirmwareHex", false)}, }; std::string os_version = util::read_file("/VERSION"); if (os_version.size()) { labels.push_back({"OS Version", "AGNOS " + os_version}); } for (auto l : labels) { QString text = QString::fromStdString(l.first + ": " + l.second); main_layout->addWidget(new QLabel(text)); } QWidget *widget = new QWidget; widget->setLayout(main_layout); widget->setStyleSheet(R"( QLabel { font-size: 50px; } )"); return widget; } QWidget * network_panel(QWidget * parent) { WifiUI *w = new WifiUI(); QObject::connect(w, SIGNAL(openKeyboard()), parent, SLOT(closeSidebar())); QObject::connect(w, SIGNAL(closeKeyboard()), parent, SLOT(openSidebar())); return w; } void SettingsWindow::setActivePanel() { QPushButton *btn = qobject_cast(sender()); panel_layout->setCurrentWidget(panels[btn->text()]); } SettingsWindow::SettingsWindow(QWidget *parent) : QWidget(parent) { // sidebar QVBoxLayout *sidebar_layout = new QVBoxLayout(); panel_layout = new QStackedLayout(); // close button QPushButton *close_button = new QPushButton("X"); close_button->setStyleSheet(R"( QPushButton { padding: 50px; font-weight: bold; font-size: 100px; } )"); sidebar_layout->addWidget(close_button); QObject::connect(close_button, SIGNAL(released()), this, SIGNAL(closeSettings())); // offroad alerts alerts_widget = new OffroadAlert(); QObject::connect(alerts_widget, SIGNAL(closeAlerts()), this, SLOT(closeAlerts())); panel_layout->addWidget(alerts_widget); sidebar_alert_widget = new QPushButton("");//Should get text when it is visible QObject::connect(sidebar_alert_widget, SIGNAL(released()), this, SLOT(openAlerts())); sidebar_layout->addWidget(sidebar_alert_widget); // setup panels panels = { {"developer", developer_panel()}, {"device", device_panel()}, {"network", network_panel(this)}, {"toggles", toggles_panel()}, }; for (auto &panel : panels) { QPushButton *btn = new QPushButton(panel.first); btn->setStyleSheet(R"( QPushButton { padding-top: 35px; padding-bottom: 35px; font-size: 60px; text-align: right; border: none; background: none; font-weight: bold; } )"); sidebar_layout->addWidget(btn); panel_layout->addWidget(panel.second); QObject::connect(btn, SIGNAL(released()), this, SLOT(setActivePanel())); } // We either show the alerts, or the developer panel if (alerts_widget->show_alert){ panel_layout->setCurrentWidget(alerts_widget); } QHBoxLayout *settings_layout = new QHBoxLayout(); settings_layout->addSpacing(45); sidebar_widget = new QWidget; sidebar_widget->setLayout(sidebar_layout); settings_layout->addWidget(sidebar_widget); settings_layout->addSpacing(45); settings_layout->addLayout(panel_layout); settings_layout->addSpacing(45); setLayout(settings_layout); setStyleSheet(R"( * { color: white; font-size: 50px; } )"); } // Refreshes the offroad alerts from the params folder and sets up the sidebar alerts widget. // The function gets called every time a user opens the settings page void SettingsWindow::refreshParams() { alerts_widget->refresh(); if (!alerts_widget->show_alert){ sidebar_alert_widget->setFixedHeight(0); panel_layout->setCurrentIndex(1); return; } // Panel 0 contains the alerts or release notes. panel_layout->setCurrentIndex(0); sidebar_alert_widget->setFixedHeight(100); sidebar_alert_widget->setStyleSheet(R"( background-color: #114267; )"); // light blue // Check for alerts int alerts = alerts_widget->alerts.size(); if (!alerts){ // There is a new release sidebar_alert_widget->setText("UPDATE"); return; } // Check if there is an important alert bool existsImportantAlert = false; for (auto alert : alerts_widget->alerts){ if (alert.severity){ existsImportantAlert = true; } } sidebar_alert_widget->setText(QString::number(alerts) + " ALERT" + (alerts == 1 ? "" : "S")); if (existsImportantAlert){ sidebar_alert_widget->setStyleSheet(R"( background-color: #661111; )"); //dark red } } void SettingsWindow::closeAlerts() { panel_layout->setCurrentIndex(1); } void SettingsWindow::openAlerts() { panel_layout->setCurrentIndex(0); } void SettingsWindow::closeSidebar() { sidebar_widget->setVisible(false); } void SettingsWindow::openSidebar() { sidebar_widget->setVisible(true); }