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.
		
		
		
		
		
			
		
			
				
					
					
						
							283 lines
						
					
					
						
							9.8 KiB
						
					
					
				
			
		
		
	
	
							283 lines
						
					
					
						
							9.8 KiB
						
					
					
				| #include "tools/cabana/util.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <csignal>
 | |
| #include <limits>
 | |
| #include <memory>
 | |
| #include <string>
 | |
| #include <sys/socket.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #include <QColor>
 | |
| #include <QDateTime>
 | |
| #include <QFontDatabase>
 | |
| #include <QLocale>
 | |
| #include <QPainter>
 | |
| #include <QPixmapCache>
 | |
| 
 | |
| #include "selfdrive/ui/qt/util.h"
 | |
| 
 | |
| // SegmentTree
 | |
| 
 | |
| void SegmentTree::build(const std::vector<QPointF> &arr) {
 | |
|   size = arr.size();
 | |
|   tree.resize(4 * size);  // size of the tree is 4 times the size of the array
 | |
|   if (size > 0) {
 | |
|     build_tree(arr, 1, 0, size - 1);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void SegmentTree::build_tree(const std::vector<QPointF> &arr, int n, int left, int right) {
 | |
|   if (left == right) {
 | |
|     const double y = arr[left].y();
 | |
|     tree[n] = {y, y};
 | |
|   } else {
 | |
|     const int mid = (left + right) >> 1;
 | |
|     build_tree(arr, 2 * n, left, mid);
 | |
|     build_tree(arr, 2 * n + 1, mid + 1, right);
 | |
|     tree[n] = {std::min(tree[2 * n].first, tree[2 * n + 1].first), std::max(tree[2 * n].second, tree[2 * n + 1].second)};
 | |
|   }
 | |
| }
 | |
| 
 | |
| std::pair<double, double> SegmentTree::get_minmax(int n, int left, int right, int range_left, int range_right) const {
 | |
|   if (range_left > right || range_right < left)
 | |
|     return {std::numeric_limits<double>::max(), std::numeric_limits<double>::lowest()};
 | |
|   if (range_left <= left && range_right >= right)
 | |
|     return tree[n];
 | |
|   int mid = (left + right) >> 1;
 | |
|   auto l = get_minmax(2 * n, left, mid, range_left, range_right);
 | |
|   auto r = get_minmax(2 * n + 1, mid + 1, right, range_left, range_right);
 | |
|   return {std::min(l.first, r.first), std::max(l.second, r.second)};
 | |
| }
 | |
| 
 | |
| // MessageBytesDelegate
 | |
| 
 | |
| MessageBytesDelegate::MessageBytesDelegate(QObject *parent, bool multiple_lines) : multiple_lines(multiple_lines), QStyledItemDelegate(parent) {
 | |
|   fixed_font = QFontDatabase::systemFont(QFontDatabase::FixedFont);
 | |
|   byte_size = QFontMetrics(fixed_font).size(Qt::TextSingleLine, "00 ") + QSize(0, 2);
 | |
|   for (int i = 0; i < 256; ++i) {
 | |
|     hex_text_table[i].setText(QStringLiteral("%1").arg(i, 2, 16, QLatin1Char('0')).toUpper());
 | |
|     hex_text_table[i].prepare({}, fixed_font);
 | |
|   }
 | |
|   h_margin = QApplication::style()->pixelMetric(QStyle::PM_FocusFrameHMargin) + 1;
 | |
|   v_margin = QApplication::style()->pixelMetric(QStyle::PM_FocusFrameVMargin) + 1;
 | |
| }
 | |
| 
 | |
| QSize MessageBytesDelegate::sizeForBytes(int n) const {
 | |
|   int rows = multiple_lines ? std::max(1, n / 8) : 1;
 | |
|   return {(n / rows) * byte_size.width() + h_margin * 2, rows * byte_size.height() + v_margin * 2};
 | |
| }
 | |
| 
 | |
| QSize MessageBytesDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const {
 | |
|   auto data = index.data(BytesRole);
 | |
|   return sizeForBytes(data.isValid() ? static_cast<std::vector<uint8_t> *>(data.value<void *>())->size() : 0);
 | |
| }
 | |
| 
 | |
| void MessageBytesDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const {
 | |
|   auto data = index.data(BytesRole);
 | |
|   if (!data.isValid()) {
 | |
|     return QStyledItemDelegate::paint(painter, option, index);
 | |
|   }
 | |
| 
 | |
|   QFont old_font = painter->font();
 | |
|   QPen old_pen = painter->pen();
 | |
|   if (option.state & QStyle::State_Selected) {
 | |
|     painter->fillRect(option.rect, option.palette.brush(QPalette::Normal, QPalette::Highlight));
 | |
|   }
 | |
|   const QPoint pt{option.rect.left() + h_margin, option.rect.top() + v_margin};
 | |
|   painter->setFont(fixed_font);
 | |
| 
 | |
|   const auto &bytes = *static_cast<std::vector<uint8_t>*>(data.value<void*>());
 | |
|   const auto &colors = *static_cast<std::vector<QColor>*>(index.data(ColorsRole).value<void*>());
 | |
|   for (int i = 0; i < bytes.size(); ++i) {
 | |
|     int row = !multiple_lines ? 0 : i / 8;
 | |
|     int column = !multiple_lines ? i : i % 8;
 | |
|     QRect r = QRect({pt.x() + column * byte_size.width(), pt.y() + row * byte_size.height()}, byte_size);
 | |
|     if (i < colors.size() && colors[i].alpha() > 0) {
 | |
|       if (option.state & QStyle::State_Selected) {
 | |
|         painter->setPen(option.palette.color(QPalette::Text));
 | |
|         painter->fillRect(r, option.palette.color(QPalette::Window));
 | |
|       }
 | |
|       painter->fillRect(r, colors[i]);
 | |
|     } else if (option.state & QStyle::State_Selected) {
 | |
|       painter->setPen(option.palette.color(QPalette::HighlightedText));
 | |
|     }
 | |
|     utils::drawStaticText(painter, r, hex_text_table[bytes[i]]);
 | |
|   }
 | |
|   painter->setFont(old_font);
 | |
|   painter->setPen(old_pen);
 | |
| }
 | |
| 
 | |
| // TabBar
 | |
| 
 | |
| int TabBar::addTab(const QString &text) {
 | |
|   int index = QTabBar::addTab(text);
 | |
|   QToolButton *btn = new ToolButton("x", tr("Close Tab"));
 | |
|   int width = style()->pixelMetric(QStyle::PM_TabCloseIndicatorWidth, nullptr, btn);
 | |
|   int height = style()->pixelMetric(QStyle::PM_TabCloseIndicatorHeight, nullptr, btn);
 | |
|   btn->setFixedSize({width, height});
 | |
|   setTabButton(index, QTabBar::RightSide, btn);
 | |
|   QObject::connect(btn, &QToolButton::clicked, this, &TabBar::closeTabClicked);
 | |
|   return index;
 | |
| }
 | |
| 
 | |
| void TabBar::closeTabClicked() {
 | |
|   QObject *object = sender();
 | |
|   for (int i = 0; i < count(); ++i) {
 | |
|     if (tabButton(i, QTabBar::RightSide) == object) {
 | |
|       emit tabCloseRequested(i);
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // UnixSignalHandler
 | |
| 
 | |
| UnixSignalHandler::UnixSignalHandler(QObject *parent) : QObject(nullptr) {
 | |
|   if (::socketpair(AF_UNIX, SOCK_STREAM, 0, sig_fd)) {
 | |
|     qFatal("Couldn't create TERM socketpair");
 | |
|   }
 | |
| 
 | |
|   sn = new QSocketNotifier(sig_fd[1], QSocketNotifier::Read, this);
 | |
|   connect(sn, &QSocketNotifier::activated, this, &UnixSignalHandler::handleSigTerm);
 | |
|   std::signal(SIGINT, signalHandler);
 | |
|   std::signal(SIGTERM, UnixSignalHandler::signalHandler);
 | |
| }
 | |
| 
 | |
| UnixSignalHandler::~UnixSignalHandler() {
 | |
|   ::close(sig_fd[0]);
 | |
|   ::close(sig_fd[1]);
 | |
| }
 | |
| 
 | |
| void UnixSignalHandler::signalHandler(int s) {
 | |
|   ::write(sig_fd[0], &s, sizeof(s));
 | |
| }
 | |
| 
 | |
| void UnixSignalHandler::handleSigTerm() {
 | |
|   sn->setEnabled(false);
 | |
|   int tmp;
 | |
|   ::read(sig_fd[1], &tmp, sizeof(tmp));
 | |
| 
 | |
|   printf("\nexiting...\n");
 | |
|   qApp->closeAllWindows();
 | |
|   qApp->exit();
 | |
| }
 | |
| 
 | |
| // NameValidator
 | |
| 
 | |
| NameValidator::NameValidator(QObject *parent) : QRegExpValidator(QRegExp("^(\\w+)"), parent) {}
 | |
| 
 | |
| QValidator::State NameValidator::validate(QString &input, int &pos) const {
 | |
|   input.replace(' ', '_');
 | |
|   return QRegExpValidator::validate(input, pos);
 | |
| }
 | |
| 
 | |
| DoubleValidator::DoubleValidator(QObject *parent) : QDoubleValidator(parent) {
 | |
|   // Match locale of QString::toDouble() instead of system
 | |
|   QLocale locale(QLocale::C);
 | |
|   locale.setNumberOptions(QLocale::RejectGroupSeparator);
 | |
|   setLocale(locale);
 | |
| }
 | |
| 
 | |
| namespace utils {
 | |
| QPixmap icon(const QString &id) {
 | |
|   bool dark_theme = settings.theme == DARK_THEME;
 | |
|   QPixmap pm;
 | |
|   QString key = "bootstrap_" % id % (dark_theme ? "1" : "0");
 | |
|   if (!QPixmapCache::find(key, &pm)) {
 | |
|     pm = bootstrapPixmap(id);
 | |
|     if (dark_theme) {
 | |
|       QPainter p(&pm);
 | |
|       p.setCompositionMode(QPainter::CompositionMode_SourceIn);
 | |
|       p.fillRect(pm.rect(), QColor("#bbbbbb"));
 | |
|     }
 | |
|     QPixmapCache::insert(key, pm);
 | |
|   }
 | |
|   return pm;
 | |
| }
 | |
| 
 | |
| void setTheme(int theme) {
 | |
|   auto style = QApplication::style();
 | |
|   if (!style) return;
 | |
| 
 | |
|   static int prev_theme = 0;
 | |
|   if (theme != prev_theme) {
 | |
|     prev_theme = theme;
 | |
|     QPalette new_palette;
 | |
|     if (theme == DARK_THEME) {
 | |
|       // "Darcula" like dark theme
 | |
|       new_palette.setColor(QPalette::Window, QColor("#353535"));
 | |
|       new_palette.setColor(QPalette::WindowText, QColor("#bbbbbb"));
 | |
|       new_palette.setColor(QPalette::Base, QColor("#3c3f41"));
 | |
|       new_palette.setColor(QPalette::AlternateBase, QColor("#3c3f41"));
 | |
|       new_palette.setColor(QPalette::ToolTipBase, QColor("#3c3f41"));
 | |
|       new_palette.setColor(QPalette::ToolTipText, QColor("#bbb"));
 | |
|       new_palette.setColor(QPalette::Text, QColor("#bbbbbb"));
 | |
|       new_palette.setColor(QPalette::Button, QColor("#3c3f41"));
 | |
|       new_palette.setColor(QPalette::ButtonText, QColor("#bbbbbb"));
 | |
|       new_palette.setColor(QPalette::Highlight, QColor("#2f65ca"));
 | |
|       new_palette.setColor(QPalette::HighlightedText, QColor("#bbbbbb"));
 | |
|       new_palette.setColor(QPalette::BrightText, QColor("#f0f0f0"));
 | |
|       new_palette.setColor(QPalette::Disabled, QPalette::ButtonText, QColor("#777777"));
 | |
|       new_palette.setColor(QPalette::Disabled, QPalette::WindowText, QColor("#777777"));
 | |
|       new_palette.setColor(QPalette::Disabled, QPalette::Text, QColor("#777777"));
 | |
|       new_palette.setColor(QPalette::Light, QColor("#777777"));
 | |
|       new_palette.setColor(QPalette::Dark, QColor("#353535"));
 | |
|     } else {
 | |
|       new_palette = style->standardPalette();
 | |
|     }
 | |
|     qApp->setPalette(new_palette);
 | |
|     style->polish(qApp);
 | |
|     for (auto w : QApplication::allWidgets()) {
 | |
|       w->setPalette(new_palette);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| QString formatSeconds(double sec, bool include_milliseconds, bool absolute_time) {
 | |
|   QString format = absolute_time ? "yyyy-MM-dd hh:mm:ss"
 | |
|                                  : (sec > 60 * 60 ? "hh:mm:ss" : "mm:ss");
 | |
|   if (include_milliseconds) format += ".zzz";
 | |
|   return QDateTime::fromMSecsSinceEpoch(sec * 1000).toString(format);
 | |
| }
 | |
| 
 | |
| }  // namespace utils
 | |
| 
 | |
| int num_decimals(double num) {
 | |
|   const QString string = QString::number(num);
 | |
|   auto dot_pos = string.indexOf('.');
 | |
|   return dot_pos == -1 ? 0 : string.size() - dot_pos - 1;
 | |
| }
 | |
| 
 | |
| QString signalToolTip(const cabana::Signal *sig) {
 | |
|   return QObject::tr(R"(
 | |
|     %1<br /><span font-size:small">
 | |
|     Start Bit: %2 Size: %3<br />
 | |
|     MSB: %4 LSB: %5<br />
 | |
|     Little Endian: %6 Signed: %7</span>
 | |
|   )").arg(sig->name).arg(sig->start_bit).arg(sig->size).arg(sig->msb).arg(sig->lsb)
 | |
|      .arg(sig->is_little_endian ? "Y" : "N").arg(sig->is_signed ? "Y" : "N");
 | |
| }
 | |
| 
 | |
| // MonotonicBuffer
 | |
| 
 | |
| void *MonotonicBuffer::allocate(size_t bytes, size_t alignment) {
 | |
|   assert(bytes > 0);
 | |
|   void *p = std::align(alignment, bytes, current_buf, available);
 | |
|   if (p == nullptr) {
 | |
|     available = next_buffer_size = std::max(next_buffer_size, bytes);
 | |
|     current_buf = buffers.emplace_back(std::aligned_alloc(alignment, next_buffer_size));
 | |
|     next_buffer_size *= growth_factor;
 | |
|     p = current_buf;
 | |
|   }
 | |
| 
 | |
|   current_buf = (char *)current_buf + bytes;
 | |
|   available -= bytes;
 | |
|   return p;
 | |
| }
 | |
| 
 | |
| MonotonicBuffer::~MonotonicBuffer() {
 | |
|   for (auto buf : buffers) {
 | |
|     free(buf);
 | |
|   }
 | |
| }
 | |
| 
 |