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.
		
		
		
		
		
			
		
			
				
					
					
						
							507 lines
						
					
					
						
							19 KiB
						
					
					
				
			
		
		
	
	
							507 lines
						
					
					
						
							19 KiB
						
					
					
				| #include "tools/cabana/binaryview.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| 
 | |
| #include <QDebug>
 | |
| #include <QFontDatabase>
 | |
| #include <QHeaderView>
 | |
| #include <QMouseEvent>
 | |
| #include <QPainter>
 | |
| #include <QScrollBar>
 | |
| #include <QShortcut>
 | |
| #include <QToolTip>
 | |
| 
 | |
| #include "tools/cabana/commands.h"
 | |
| 
 | |
| // BinaryView
 | |
| 
 | |
| const int CELL_HEIGHT = 36;
 | |
| const int VERTICAL_HEADER_WIDTH = 30;
 | |
| inline int get_bit_pos(const QModelIndex &index) { return flipBitPos(index.row() * 8 + index.column()); }
 | |
| 
 | |
| BinaryView::BinaryView(QWidget *parent) : QTableView(parent) {
 | |
|   model = new BinaryViewModel(this);
 | |
|   setModel(model);
 | |
|   delegate = new BinaryItemDelegate(this);
 | |
|   setItemDelegate(delegate);
 | |
|   horizontalHeader()->setSectionResizeMode(QHeaderView::Stretch);
 | |
|   horizontalHeader()->setFont(QFontDatabase::systemFont(QFontDatabase::FixedFont));
 | |
|   verticalHeader()->setSectionsClickable(false);
 | |
|   verticalHeader()->setSectionResizeMode(QHeaderView::Fixed);
 | |
|   verticalHeader()->setDefaultSectionSize(CELL_HEIGHT);
 | |
|   horizontalHeader()->hide();
 | |
|   setShowGrid(false);
 | |
|   setMouseTracking(true);
 | |
|   setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
 | |
| 
 | |
|   QObject::connect(dbc(), &DBCManager::DBCFileChanged, this, &BinaryView::refresh);
 | |
|   QObject::connect(UndoStack::instance(), &QUndoStack::indexChanged, this, &BinaryView::refresh);
 | |
| 
 | |
|   addShortcuts();
 | |
|   setWhatsThis(R"(
 | |
|     <b>Binary View</b><br/>
 | |
|     <!-- TODO: add descprition here -->
 | |
|     <span style="color:gray">Shortcuts</span><br />
 | |
|     Delete Signal:
 | |
|       <span style="background-color:lightGray;color:gray"> x </span>,
 | |
|       <span style="background-color:lightGray;color:gray"> Backspace </span>,
 | |
|       <span style="background-color:lightGray;color:gray"> Delete </span><br />
 | |
|     Change endianness: <span style="background-color:lightGray;color:gray"> e  </span><br />
 | |
|     Change singedness: <span style="background-color:lightGray;color:gray"> s </span><br />
 | |
|     Open chart:
 | |
|       <span style="background-color:lightGray;color:gray"> c </span>,
 | |
|       <span style="background-color:lightGray;color:gray"> p </span>,
 | |
|       <span style="background-color:lightGray;color:gray"> g </span>
 | |
|   )");
 | |
| }
 | |
| 
 | |
| void BinaryView::addShortcuts() {
 | |
|   // Delete (x, backspace, delete)
 | |
|   QShortcut *shortcut_delete_x = new QShortcut(QKeySequence(Qt::Key_X), this);
 | |
|   QShortcut *shortcut_delete_backspace = new QShortcut(QKeySequence(Qt::Key_Backspace), this);
 | |
|   QShortcut *shortcut_delete_delete = new QShortcut(QKeySequence(Qt::Key_Delete), this);
 | |
|   QObject::connect(shortcut_delete_delete, &QShortcut::activated, shortcut_delete_x, &QShortcut::activated);
 | |
|   QObject::connect(shortcut_delete_backspace, &QShortcut::activated, shortcut_delete_x, &QShortcut::activated);
 | |
|   QObject::connect(shortcut_delete_x, &QShortcut::activated, [=]{
 | |
|     if (hovered_sig != nullptr) {
 | |
|       UndoStack::push(new RemoveSigCommand(model->msg_id, hovered_sig));
 | |
|       hovered_sig = nullptr;
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   // Change endianness (e)
 | |
|   QShortcut *shortcut_endian = new QShortcut(QKeySequence(Qt::Key_E), this);
 | |
|   QObject::connect(shortcut_endian, &QShortcut::activated, [=]{
 | |
|     if (hovered_sig != nullptr) {
 | |
|       cabana::Signal s = *hovered_sig;
 | |
|       s.is_little_endian = !s.is_little_endian;
 | |
|       emit editSignal(hovered_sig, s);
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   // Change signedness (s)
 | |
|   QShortcut *shortcut_sign = new QShortcut(QKeySequence(Qt::Key_S), this);
 | |
|   QObject::connect(shortcut_sign, &QShortcut::activated, [=]{
 | |
|     if (hovered_sig != nullptr) {
 | |
|       cabana::Signal s = *hovered_sig;
 | |
|       s.is_signed = !s.is_signed;
 | |
|       emit editSignal(hovered_sig, s);
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   // Open chart (c, p, g)
 | |
|   QShortcut *shortcut_plot = new QShortcut(QKeySequence(Qt::Key_P), this);
 | |
|   QShortcut *shortcut_plot_g = new QShortcut(QKeySequence(Qt::Key_G), this);
 | |
|   QShortcut *shortcut_plot_c = new QShortcut(QKeySequence(Qt::Key_C), this);
 | |
|   QObject::connect(shortcut_plot_g, &QShortcut::activated, shortcut_plot, &QShortcut::activated);
 | |
|   QObject::connect(shortcut_plot_c, &QShortcut::activated, shortcut_plot, &QShortcut::activated);
 | |
|   QObject::connect(shortcut_plot, &QShortcut::activated, [=]{
 | |
|     if (hovered_sig != nullptr) {
 | |
|       emit showChart(model->msg_id, hovered_sig, true, false);
 | |
|     }
 | |
|   });
 | |
| }
 | |
| 
 | |
| QSize BinaryView::minimumSizeHint() const {
 | |
|   return {(horizontalHeader()->minimumSectionSize() + 1) * 9 + VERTICAL_HEADER_WIDTH + 2,
 | |
|           CELL_HEIGHT * std::min(model->rowCount(), 10) + 2};
 | |
| }
 | |
| 
 | |
| void BinaryView::highlight(const cabana::Signal *sig) {
 | |
|   if (sig != hovered_sig) {
 | |
|     for (int i = 0; i < model->items.size(); ++i) {
 | |
|       auto &item_sigs = model->items[i].sigs;
 | |
|       if ((sig && item_sigs.contains(sig)) || (hovered_sig && item_sigs.contains(hovered_sig))) {
 | |
|         auto index = model->index(i / model->columnCount(), i % model->columnCount());
 | |
|         emit model->dataChanged(index, index, {Qt::DisplayRole});
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     hovered_sig = sig;
 | |
|     emit signalHovered(hovered_sig);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void BinaryView::setSelection(const QRect &rect, QItemSelectionModel::SelectionFlags flags) {
 | |
|   auto index = indexAt(viewport()->mapFromGlobal(QCursor::pos()));
 | |
|   if (!anchor_index.isValid() || !index.isValid())
 | |
|     return;
 | |
| 
 | |
|   QItemSelection selection;
 | |
|   auto [start, size, is_lb] = getSelection(index);
 | |
|   for (int i = 0; i < size; ++i) {
 | |
|     int pos = is_lb ? flipBitPos(start + i) : flipBitPos(start) + i;
 | |
|     selection << QItemSelectionRange{model->index(pos / 8, pos % 8)};
 | |
|   }
 | |
|   selectionModel()->select(selection, flags);
 | |
| }
 | |
| 
 | |
| void BinaryView::mousePressEvent(QMouseEvent *event) {
 | |
|   resize_sig = nullptr;
 | |
|   if (auto index = indexAt(event->pos()); index.isValid() && index.column() != 8) {
 | |
|     anchor_index = index;
 | |
|     auto item = (const BinaryViewModel::Item *)anchor_index.internalPointer();
 | |
|     int bit_pos = get_bit_pos(anchor_index);
 | |
|     for (auto s : item->sigs) {
 | |
|       if (bit_pos == s->lsb || bit_pos == s->msb) {
 | |
|         int idx = flipBitPos(bit_pos == s->lsb ? s->msb : s->lsb);
 | |
|         anchor_index = model->index(idx / 8, idx % 8);
 | |
|         resize_sig = s;
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   event->accept();
 | |
| }
 | |
| 
 | |
| void BinaryView::highlightPosition(const QPoint &pos) {
 | |
|   if (auto index = indexAt(viewport()->mapFromGlobal(pos)); index.isValid()) {
 | |
|     auto item = (BinaryViewModel::Item *)index.internalPointer();
 | |
|     const cabana::Signal *sig = item->sigs.isEmpty() ? nullptr : item->sigs.back();
 | |
|     highlight(sig);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void BinaryView::mouseMoveEvent(QMouseEvent *event) {
 | |
|   highlightPosition(event->globalPos());
 | |
|   QTableView::mouseMoveEvent(event);
 | |
| }
 | |
| 
 | |
| void BinaryView::mouseReleaseEvent(QMouseEvent *event) {
 | |
|   QTableView::mouseReleaseEvent(event);
 | |
| 
 | |
|   auto release_index = indexAt(event->pos());
 | |
|   if (release_index.isValid() && anchor_index.isValid()) {
 | |
|     if (selectionModel()->hasSelection()) {
 | |
|       auto sig = resize_sig ? *resize_sig : cabana::Signal{};
 | |
|       std::tie(sig.start_bit, sig.size, sig.is_little_endian) = getSelection(release_index);
 | |
|       resize_sig ? emit editSignal(resize_sig, sig)
 | |
|                  : UndoStack::push(new AddSigCommand(model->msg_id, sig));
 | |
|     } else {
 | |
|       auto item = (const BinaryViewModel::Item *)anchor_index.internalPointer();
 | |
|       if (item && item->sigs.size() > 0)
 | |
|         emit signalClicked(item->sigs.back());
 | |
|     }
 | |
|   }
 | |
|   clearSelection();
 | |
|   anchor_index = QModelIndex();
 | |
|   resize_sig = nullptr;
 | |
| }
 | |
| 
 | |
| void BinaryView::leaveEvent(QEvent *event) {
 | |
|   highlight(nullptr);
 | |
|   QTableView::leaveEvent(event);
 | |
| }
 | |
| 
 | |
| void BinaryView::setMessage(const MessageId &message_id) {
 | |
|   model->msg_id = message_id;
 | |
|   verticalScrollBar()->setValue(0);
 | |
|   refresh();
 | |
| }
 | |
| 
 | |
| void BinaryView::refresh() {
 | |
|   clearSelection();
 | |
|   anchor_index = QModelIndex();
 | |
|   resize_sig = nullptr;
 | |
|   hovered_sig = nullptr;
 | |
|   model->refresh();
 | |
|   highlightPosition(QCursor::pos());
 | |
| }
 | |
| 
 | |
| QSet<const cabana::Signal *> BinaryView::getOverlappingSignals() const {
 | |
|   QSet<const cabana::Signal *> overlapping;
 | |
|   for (const auto &item : model->items) {
 | |
|     if (item.sigs.size() > 1) {
 | |
|       for (auto s : item.sigs) {
 | |
|         if (s->type == cabana::Signal::Type::Normal) overlapping += s;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return overlapping;
 | |
| }
 | |
| 
 | |
| std::tuple<int, int, bool> BinaryView::getSelection(QModelIndex index) {
 | |
|   if (index.column() == 8) {
 | |
|     index = model->index(index.row(), 7);
 | |
|   }
 | |
|   bool is_lb = true;
 | |
|   if (resize_sig) {
 | |
|     is_lb = resize_sig->is_little_endian;
 | |
|   } else if (settings.drag_direction == Settings::DragDirection::MsbFirst) {
 | |
|     is_lb = index < anchor_index;
 | |
|   } else if (settings.drag_direction == Settings::DragDirection::LsbFirst) {
 | |
|     is_lb = !(index < anchor_index);
 | |
|   } else if (settings.drag_direction == Settings::DragDirection::AlwaysLE) {
 | |
|     is_lb = true;
 | |
|   } else if (settings.drag_direction == Settings::DragDirection::AlwaysBE) {
 | |
|     is_lb = false;
 | |
|   }
 | |
| 
 | |
|   int cur_bit_pos = get_bit_pos(index);
 | |
|   int anchor_bit_pos = get_bit_pos(anchor_index);
 | |
|   int start_bit = is_lb ? std::min(cur_bit_pos, anchor_bit_pos) : get_bit_pos(std::min(index, anchor_index));
 | |
|   int size = is_lb ? std::abs(cur_bit_pos - anchor_bit_pos) + 1 : std::abs(flipBitPos(cur_bit_pos) - flipBitPos(anchor_bit_pos)) + 1;
 | |
|   return {start_bit, size, is_lb};
 | |
| }
 | |
| 
 | |
| // BinaryViewModel
 | |
| 
 | |
| void BinaryViewModel::refresh() {
 | |
|   beginResetModel();
 | |
|   bit_flip_tracker = {};
 | |
|   items.clear();
 | |
|   if (auto dbc_msg = dbc()->msg(msg_id)) {
 | |
|     row_count = dbc_msg->size;
 | |
|     items.resize(row_count * column_count);
 | |
|     for (auto sig : dbc_msg->getSignals()) {
 | |
|       for (int j = 0; j < sig->size; ++j) {
 | |
|         int pos = sig->is_little_endian ? flipBitPos(sig->start_bit + j) : flipBitPos(sig->start_bit) + j;
 | |
|         int idx = column_count * (pos / 8) + pos % 8;
 | |
|         if (idx >= items.size()) {
 | |
|           qWarning() << "signal " << sig->name << "out of bounds.start_bit:" << sig->start_bit << "size:" << sig->size;
 | |
|           break;
 | |
|         }
 | |
|         if (j == 0) sig->is_little_endian ? items[idx].is_lsb = true : items[idx].is_msb = true;
 | |
|         if (j == sig->size - 1) sig->is_little_endian ? items[idx].is_msb = true : items[idx].is_lsb = true;
 | |
| 
 | |
|         auto &sigs = items[idx].sigs;
 | |
|         sigs.push_back(sig);
 | |
|         if (sigs.size() > 1) {
 | |
|           std::sort(sigs.begin(), sigs.end(), [](auto l, auto r) { return l->size > r->size; });
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   } else {
 | |
|     row_count = can->lastMessage(msg_id).dat.size();
 | |
|     items.resize(row_count * column_count);
 | |
|   }
 | |
|   int valid_rows = std::min<int>(can->lastMessage(msg_id).dat.size(), row_count);
 | |
|   for (int i = 0; i < valid_rows * column_count; ++i) {
 | |
|     items[i].valid = true;
 | |
|   }
 | |
|   endResetModel();
 | |
|   updateState();
 | |
| }
 | |
| 
 | |
| void BinaryViewModel::updateItem(int row, int col, uint8_t val, const QColor &color) {
 | |
|   auto &item = items[row * column_count + col];
 | |
|   if (item.val != val || item.bg_color != color) {
 | |
|     item.val = val;
 | |
|     item.bg_color = color;
 | |
|     auto idx = index(row, col);
 | |
|     emit dataChanged(idx, idx, {Qt::DisplayRole});
 | |
|   }
 | |
| }
 | |
| 
 | |
| void BinaryViewModel::updateState() {
 | |
|   const auto &last_msg = can->lastMessage(msg_id);
 | |
|   const auto &binary = last_msg.dat;
 | |
|   // Handle size changes in binary data
 | |
|   if (binary.size() > row_count) {
 | |
|     beginInsertRows({}, row_count, binary.size() - 1);
 | |
|     row_count = binary.size();
 | |
|     items.resize(row_count * column_count);
 | |
|     endInsertRows();
 | |
|   }
 | |
| 
 | |
|   auto &bit_flips = heatmap_live_mode ? last_msg.bit_flip_counts : getBitFlipChanges(binary.size());
 | |
|   // Find the maximum bit flip count across the message
 | |
|   uint32_t max_bit_flip_count = 1;  // Default to 1 to avoid division by zero
 | |
|   for (const auto &row : bit_flips) {
 | |
|     for (uint32_t count : row) {
 | |
|       max_bit_flip_count = std::max(max_bit_flip_count, count);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const double max_alpha = 255.0;
 | |
|   const double min_alpha_with_signal = 25.0;  // Base alpha for small flip counts
 | |
|   const double min_alpha_no_signal = 10.0;    // Base alpha for small flip counts for no signal bits
 | |
|   const double log_factor = 1.0 + 0.2;        // Factor for logarithmic scaling
 | |
|   const double log_scaler = max_alpha / log2(log_factor * max_bit_flip_count);
 | |
| 
 | |
|   for (size_t i = 0; i < binary.size(); ++i) {
 | |
|     for (int j = 0; j < 8; ++j) {
 | |
|       auto &item = items[i * column_count + j];
 | |
|       int bit_val = (binary[i] >> (7 - j)) & 1;
 | |
| 
 | |
|       double alpha = item.sigs.empty() ? 0 : min_alpha_with_signal;
 | |
|       uint32_t flip_count = bit_flips[i][j];
 | |
|       if (flip_count > 0) {
 | |
|         double normalized_alpha = log2(1.0 + flip_count * log_factor) * log_scaler;
 | |
|         double min_alpha = item.sigs.empty() ? min_alpha_no_signal : min_alpha_with_signal;
 | |
|         alpha = std::clamp(normalized_alpha, min_alpha, max_alpha);
 | |
|       }
 | |
| 
 | |
|       auto color = item.bg_color;
 | |
|       color.setAlpha(alpha);
 | |
|       updateItem(i, j, bit_val, color);
 | |
|     }
 | |
|     updateItem(i, 8, binary[i], last_msg.colors[i]);
 | |
|   }
 | |
| }
 | |
| 
 | |
| const std::vector<std::array<uint32_t, 8>> &BinaryViewModel::getBitFlipChanges(size_t msg_size) {
 | |
|   // Return cached results if time range and data are unchanged
 | |
|   auto time_range = can->timeRange();
 | |
|   if (bit_flip_tracker.time_range == time_range && !bit_flip_tracker.flip_counts.empty())
 | |
|     return bit_flip_tracker.flip_counts;
 | |
| 
 | |
|   bit_flip_tracker.time_range = time_range;
 | |
|   bit_flip_tracker.flip_counts.assign(msg_size, std::array<uint32_t, 8>{});
 | |
| 
 | |
|   // Iterate over events within the specified time range and calculate bit flips
 | |
|   auto [first, last] = can->eventsInRange(msg_id, time_range);
 | |
|   if (std::distance(first, last) <= 1) return bit_flip_tracker.flip_counts;
 | |
| 
 | |
|   std::vector<uint8_t> prev_values((*first)->dat, (*first)->dat + (*first)->size);
 | |
|   for (auto it = std::next(first); it != last; ++it) {
 | |
|     const CanEvent *event = *it;
 | |
|     int size = std::min<int>(msg_size, event->size);
 | |
|     for (int i = 0; i < size; ++i) {
 | |
|       const uint8_t diff = event->dat[i] ^ prev_values[i];
 | |
|       if (!diff) continue;
 | |
| 
 | |
|       auto &bit_flips = bit_flip_tracker.flip_counts[i];
 | |
|       for (int bit = 0; bit < 8; ++bit) {
 | |
|         if (diff & (1u << bit)) ++bit_flips[7 - bit];
 | |
|       }
 | |
|       prev_values[i] = event->dat[i];
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return bit_flip_tracker.flip_counts;
 | |
| }
 | |
| 
 | |
| QVariant BinaryViewModel::headerData(int section, Qt::Orientation orientation, int role) const {
 | |
|   if (orientation == Qt::Vertical) {
 | |
|     switch (role) {
 | |
|       case Qt::DisplayRole: return section;
 | |
|       case Qt::SizeHintRole: return QSize(VERTICAL_HEADER_WIDTH, 0);
 | |
|       case Qt::TextAlignmentRole: return Qt::AlignCenter;
 | |
|     }
 | |
|   }
 | |
|   return {};
 | |
| }
 | |
| 
 | |
| QVariant BinaryViewModel::data(const QModelIndex &index, int role) const {
 | |
|   auto item = (const BinaryViewModel::Item *)index.internalPointer();
 | |
|   return role == Qt::ToolTipRole && item && !item->sigs.empty() ? signalToolTip(item->sigs.back()) : QVariant();
 | |
| }
 | |
| 
 | |
| // BinaryItemDelegate
 | |
| 
 | |
| BinaryItemDelegate::BinaryItemDelegate(QObject *parent) : QStyledItemDelegate(parent) {
 | |
|   small_font.setPixelSize(8);
 | |
|   hex_font = QFontDatabase::systemFont(QFontDatabase::FixedFont);
 | |
|   hex_font.setBold(true);
 | |
| 
 | |
|   bin_text_table[0].setText("0");
 | |
|   bin_text_table[1].setText("1");
 | |
|   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({}, hex_font);
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool BinaryItemDelegate::hasSignal(const QModelIndex &index, int dx, int dy, const cabana::Signal *sig) const {
 | |
|   if (!index.isValid()) return false;
 | |
|   auto model = (const BinaryViewModel*)(index.model());
 | |
|   int idx = (index.row() + dy) * model->columnCount() + index.column() + dx;
 | |
|   return (idx >=0 && idx < model->items.size()) ? model->items[idx].sigs.contains(sig) : false;
 | |
| }
 | |
| 
 | |
| void BinaryItemDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const {
 | |
|   auto item = (const BinaryViewModel::Item *)index.internalPointer();
 | |
|   BinaryView *bin_view = (BinaryView *)parent();
 | |
|   painter->save();
 | |
| 
 | |
|   if (index.column() == 8) {
 | |
|     if (item->valid) {
 | |
|       painter->setFont(hex_font);
 | |
|       painter->fillRect(option.rect, item->bg_color);
 | |
|     }
 | |
|   } else if (option.state & QStyle::State_Selected) {
 | |
|     auto color = bin_view->resize_sig ? bin_view->resize_sig->color : option.palette.color(QPalette::Active, QPalette::Highlight);
 | |
|     painter->fillRect(option.rect, color);
 | |
|     painter->setPen(option.palette.color(QPalette::BrightText));
 | |
|   } else if (!bin_view->selectionModel()->hasSelection() || !item->sigs.contains(bin_view->resize_sig)) {  // not resizing
 | |
|     if (item->sigs.size() > 0) {
 | |
|       for (auto &s : item->sigs) {
 | |
|         if (s == bin_view->hovered_sig) {
 | |
|           painter->fillRect(option.rect, s->color.darker(125));  // 4/5x brightness
 | |
|         } else {
 | |
|           drawSignalCell(painter, option, index, s);
 | |
|         }
 | |
|       }
 | |
|     } else if (item->valid && item->bg_color.alpha() > 0) {
 | |
|       painter->fillRect(option.rect, item->bg_color);
 | |
|     }
 | |
|     auto color_role = item->sigs.contains(bin_view->hovered_sig) ? QPalette::BrightText : QPalette::Text;
 | |
|     painter->setPen(option.palette.color(bin_view->is_message_active ? QPalette::Normal : QPalette::Disabled, color_role));
 | |
|   }
 | |
| 
 | |
|   if (item->sigs.size() > 1) {
 | |
|     painter->fillRect(option.rect, QBrush(Qt::darkGray, Qt::Dense7Pattern));
 | |
|   } else if (!item->valid) {
 | |
|     painter->fillRect(option.rect, QBrush(Qt::darkGray, Qt::BDiagPattern));
 | |
|   }
 | |
|   if (item->valid) {
 | |
|     utils::drawStaticText(painter, option.rect, index.column() == 8 ? hex_text_table[item->val] : bin_text_table[item->val]);
 | |
|   }
 | |
|   if (item->is_msb || item->is_lsb) {
 | |
|     painter->setFont(small_font);
 | |
|     painter->drawText(option.rect.adjusted(8, 0, -8, -3), Qt::AlignRight | Qt::AlignBottom, item->is_msb ? "M" : "L");
 | |
|   }
 | |
|   painter->restore();
 | |
| }
 | |
| 
 | |
| // Draw border on edge of signal
 | |
| void BinaryItemDelegate::drawSignalCell(QPainter *painter, const QStyleOptionViewItem &option,
 | |
|                                         const QModelIndex &index, const cabana::Signal *sig) const {
 | |
|   bool draw_left = !hasSignal(index, -1, 0, sig);
 | |
|   bool draw_top = !hasSignal(index, 0, -1, sig);
 | |
|   bool draw_right = !hasSignal(index, 1, 0, sig);
 | |
|   bool draw_bottom = !hasSignal(index, 0, 1, sig);
 | |
| 
 | |
|   const int spacing = 2;
 | |
|   QRect rc = option.rect.adjusted(draw_left * 3, draw_top * spacing, draw_right * -3, draw_bottom * -spacing);
 | |
|   QRegion subtract;
 | |
|   if (!draw_top) {
 | |
|     if (!draw_left && !hasSignal(index, -1, -1, sig)) {
 | |
|       subtract += QRect{rc.left(), rc.top(), 3, spacing};
 | |
|     } else if (!draw_right && !hasSignal(index, 1, -1, sig)) {
 | |
|       subtract += QRect{rc.right() - 2, rc.top(), 3, spacing};
 | |
|     }
 | |
|   }
 | |
|   if (!draw_bottom) {
 | |
|     if (!draw_left && !hasSignal(index, -1, 1, sig)) {
 | |
|       subtract += QRect{rc.left(), rc.bottom() - (spacing - 1), 3, spacing};
 | |
|     } else if (!draw_right && !hasSignal(index, 1, 1, sig)) {
 | |
|       subtract += QRect{rc.right() - 2, rc.bottom() - (spacing - 1), 3, spacing};
 | |
|     }
 | |
|   }
 | |
|   painter->setClipRegion(QRegion(rc).subtracted(subtract));
 | |
| 
 | |
|   auto item = (const BinaryViewModel::Item *)index.internalPointer();
 | |
|   QColor color = sig->color;
 | |
|   color.setAlpha(item->bg_color.alpha());
 | |
|   // Mixing the signal color with the Base background color to fade it
 | |
|   painter->fillRect(rc, option.palette.color(QPalette::Base));
 | |
|   painter->fillRect(rc, color);
 | |
| 
 | |
|   // Draw edges
 | |
|   color = sig->color.darker(125);
 | |
|   painter->setPen(QPen(color, 1));
 | |
|   if (draw_left) painter->drawLine(rc.topLeft(), rc.bottomLeft());
 | |
|   if (draw_right) painter->drawLine(rc.topRight(), rc.bottomRight());
 | |
|   if (draw_bottom) painter->drawLine(rc.bottomLeft(), rc.bottomRight());
 | |
|   if (draw_top) painter->drawLine(rc.topLeft(), rc.topRight());
 | |
| 
 | |
|   if (!subtract.isEmpty()) {
 | |
|     // fill gaps inside corners.
 | |
|     painter->setPen(QPen(color, 2, Qt::SolidLine, Qt::SquareCap, Qt::MiterJoin));
 | |
|     for (auto &r : subtract) {
 | |
|       painter->drawRect(r);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 |