|  |  |  | @ -62,7 +62,7 @@ void AbstractStream::updateEvent(const MessageId &id, double sec, const uint8_t | 
			
		
	
		
			
				
					|  |  |  |  |   std::lock_guard lk(mutex); | 
			
		
	
		
			
				
					|  |  |  |  |   auto mask_it = masks.find(id); | 
			
		
	
		
			
				
					|  |  |  |  |   std::vector<uint8_t> *mask = mask_it == masks.end() ? nullptr : &mask_it->second; | 
			
		
	
		
			
				
					|  |  |  |  |   all_msgs[id].compute((const char *)data, size, sec, getSpeed(), mask); | 
			
		
	
		
			
				
					|  |  |  |  |   all_msgs[id].compute(id, (const char *)data, size, sec, getSpeed(), mask); | 
			
		
	
		
			
				
					|  |  |  |  |   if (!new_msgs->contains(id)) { | 
			
		
	
		
			
				
					|  |  |  |  |     new_msgs->insert(id, {}); | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
	
		
			
				
					|  |  |  | @ -113,9 +113,8 @@ void AbstractStream::updateLastMsgsTo(double sec) { | 
			
		
	
		
			
				
					|  |  |  |  |     if (it != ev.crend()) { | 
			
		
	
		
			
				
					|  |  |  |  |       double ts = (*it)->mono_time / 1e9 - routeStartTime(); | 
			
		
	
		
			
				
					|  |  |  |  |       auto &m = all_msgs[id]; | 
			
		
	
		
			
				
					|  |  |  |  |       m.compute((const char *)(*it)->dat, (*it)->size, ts, getSpeed(), mask); | 
			
		
	
		
			
				
					|  |  |  |  |       m.compute(id, (const char *)(*it)->dat, (*it)->size, ts, getSpeed(), mask); | 
			
		
	
		
			
				
					|  |  |  |  |       m.count = std::distance(it, ev.crend()); | 
			
		
	
		
			
				
					|  |  |  |  |       m.freq = m.count / std::max(1.0, ts); | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -165,18 +164,20 @@ void AbstractStream::mergeEvents(std::vector<Event *>::const_iterator first, std | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   auto compare = [](const CanEvent *l, const CanEvent *r) { | 
			
		
	
		
			
				
					|  |  |  |  |     return l->mono_time < r->mono_time; | 
			
		
	
		
			
				
					|  |  |  |  |   }; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   bool append = new_events.front()->mono_time > lastest_event_ts; | 
			
		
	
		
			
				
					|  |  |  |  |   for (auto &[id, new_e] : new_events_map) { | 
			
		
	
		
			
				
					|  |  |  |  |     auto &e = events_[id]; | 
			
		
	
		
			
				
					|  |  |  |  |     auto pos = append ? e.end() : std::upper_bound(e.cbegin(), e.cend(), new_e.front(), [](const CanEvent *l, const CanEvent *r) { | 
			
		
	
		
			
				
					|  |  |  |  |       return l->mono_time < r->mono_time; | 
			
		
	
		
			
				
					|  |  |  |  |     }); | 
			
		
	
		
			
				
					|  |  |  |  |     auto pos = append ? e.end() | 
			
		
	
		
			
				
					|  |  |  |  |                       : std::upper_bound(e.cbegin(), e.cend(), new_e.front(), compare); | 
			
		
	
		
			
				
					|  |  |  |  |     e.insert(pos, new_e.cbegin(), new_e.cend()); | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   auto pos = append ? all_events_.end() : std::upper_bound(all_events_.begin(), all_events_.end(), new_events.front(), [](auto l, auto r) { | 
			
		
	
		
			
				
					|  |  |  |  |     return l->mono_time < r->mono_time; | 
			
		
	
		
			
				
					|  |  |  |  |   }); | 
			
		
	
		
			
				
					|  |  |  |  |   auto pos = append ? all_events_.end() | 
			
		
	
		
			
				
					|  |  |  |  |                     : std::upper_bound(all_events_.begin(), all_events_.end(), new_events.front(), compare); | 
			
		
	
		
			
				
					|  |  |  |  |   all_events_.insert(pos, new_events.cbegin(), new_events.cend()); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   lastest_event_ts = all_events_.back()->mono_time; | 
			
		
	
	
		
			
				
					|  |  |  | @ -185,6 +186,8 @@ void AbstractStream::mergeEvents(std::vector<Event *>::const_iterator first, std | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | // CanData
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | namespace { | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | constexpr int periodic_threshold = 10; | 
			
		
	
		
			
				
					|  |  |  |  | constexpr int start_alpha = 128; | 
			
		
	
		
			
				
					|  |  |  |  | constexpr float fade_time = 2.0; | 
			
		
	
	
		
			
				
					|  |  |  | @ -195,15 +198,41 @@ const QColor CYAN_LIGHTER = QColor(0, 187, 255, start_alpha).lighter(135); | 
			
		
	
		
			
				
					|  |  |  |  | const QColor RED_LIGHTER = QColor(255, 0, 0, start_alpha).lighter(135); | 
			
		
	
		
			
				
					|  |  |  |  | const QColor GREYISH_BLUE_LIGHTER = QColor(102, 86, 169, start_alpha / 2).lighter(135); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | static inline QColor blend(const QColor &a, const QColor &b) { | 
			
		
	
		
			
				
					|  |  |  |  | inline QColor blend(const QColor &a, const QColor &b) { | 
			
		
	
		
			
				
					|  |  |  |  |   return QColor((a.red() + b.red()) / 2, (a.green() + b.green()) / 2, (a.blue() + b.blue()) / 2, (a.alpha() + b.alpha()) / 2); | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | void CanData::compute(const char *can_data, const int size, double current_sec, double playback_speed, const std::vector<uint8_t> *mask, uint32_t in_freq) { | 
			
		
	
		
			
				
					|  |  |  |  | // Calculate the frequency of the past minute.
 | 
			
		
	
		
			
				
					|  |  |  |  | double calc_freq(const MessageId &msg_id, double current_sec) { | 
			
		
	
		
			
				
					|  |  |  |  |   auto compare = [](const CanEvent *e, uint64_t mono_time) { | 
			
		
	
		
			
				
					|  |  |  |  |     return e->mono_time < mono_time; | 
			
		
	
		
			
				
					|  |  |  |  |   }; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   const auto &events = can->events(msg_id); | 
			
		
	
		
			
				
					|  |  |  |  |   uint64_t cur_mono_time = (can->routeStartTime() + current_sec) * 1e9; | 
			
		
	
		
			
				
					|  |  |  |  |   uint64_t first_mono_time = std::max<int64_t>(0, cur_mono_time - 59 * 1e9); | 
			
		
	
		
			
				
					|  |  |  |  |   auto first = std::lower_bound(events.begin(), events.end(), first_mono_time, compare); | 
			
		
	
		
			
				
					|  |  |  |  |   auto second = std::lower_bound(first, events.end(), cur_mono_time, compare); | 
			
		
	
		
			
				
					|  |  |  |  |   if (first != events.end() && second != events.end()) { | 
			
		
	
		
			
				
					|  |  |  |  |     double duration = ((*second)->mono_time - (*first)->mono_time) / 1e9; | 
			
		
	
		
			
				
					|  |  |  |  |     uint32_t count = std::distance(first, second); | 
			
		
	
		
			
				
					|  |  |  |  |     return count / std::max(1.0, duration); | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  |   return 0; | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | }  // namespace
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | void CanData::compute(const MessageId &msg_id, const char *can_data, const int size, double current_sec, | 
			
		
	
		
			
				
					|  |  |  |  |                       double playback_speed, const std::vector<uint8_t> *mask, double in_freq) { | 
			
		
	
		
			
				
					|  |  |  |  |   ts = current_sec; | 
			
		
	
		
			
				
					|  |  |  |  |   ++count; | 
			
		
	
		
			
				
					|  |  |  |  |   const double sec_to_first_event = current_sec - (can->allEvents().front()->mono_time / 1e9 - can->routeStartTime()); | 
			
		
	
		
			
				
					|  |  |  |  |   freq = in_freq == 0 ? count / std::max(1.0, sec_to_first_event) : in_freq; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   if (auto sec = seconds_since_boot(); (sec - last_freq_update_ts) >= 1) { | 
			
		
	
		
			
				
					|  |  |  |  |     last_freq_update_ts = sec; | 
			
		
	
		
			
				
					|  |  |  |  |     freq = !in_freq ? calc_freq(msg_id, ts) : in_freq; | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   if (dat.size() != size) { | 
			
		
	
		
			
				
					|  |  |  |  |     dat.resize(size); | 
			
		
	
		
			
				
					|  |  |  |  |     bit_change_counts.resize(size); | 
			
		
	
	
		
			
				
					|  |  |  | 
 |