6 #include "../annotations/image.h" 7 #include "../core/bagplayer.h" 8 #include "../core/log.h" 9 #include "../core/topic.h" 10 #include "../core/tracks.h" 11 #include "../core/workspace.h" 12 #include "mainwindow.h" 15 #include <rosbag/bag.h> 16 #include <rosbag/view.h> 17 #include <sensor_msgs/Image.h> 18 #include <std_msgs/Float64.h> 20 static const double track_height = 42;
21 static const double track_label_width = 200;
22 static const double track_padding_right = 100;
25 static void updateEnabled(QLayout *layout,
const FNC &callback) {
26 for (
size_t i = 0; i < layout->count(); i++) {
27 if (
auto *widget = layout->itemAt(i)->widget()) {
30 if (
auto *l = layout->itemAt(i)->layout()) {
31 updateEnabled(l, callback);
36 static size_t trackCount(
const std::shared_ptr<Workspace> &ws) {
37 return (ws->document() && ws->document()->timeline())
38 ? ws->document()->timeline()->tracks().size()
42 static double timelineDuration(
const std::shared_ptr<Workspace> &ws) {
43 return std::max(0.001, ws->player ? ws->player->duration() : 1.0) + 1.0;
46 static std::shared_ptr<TrackBase> trackAt(
const std::shared_ptr<Workspace> &ws,
48 if (ws->document() && ws->document()->timeline() &&
49 index < ws->document()->timeline()->tracks().size()) {
50 return ws->document()->timeline()->tracks()[index];
60 virtual void sync(
const std::shared_ptr<Workspace> &ws) {}
62 LockScope()->modified.connect(
this, [
this]() {
64 throw std::runtime_error(
"timeline item already destroyed");
74 QGraphicsProxyWidget *_proxy =
nullptr;
75 QLineEdit *_edit =
nullptr;
76 QMargins margins()
const {
77 int m = std::round(std::min(rect().width(), rect().height()) * 0.25);
78 if (_alignment & Qt::AlignHCenter) {
81 return QMargins(m, 0, m, 0);
83 QPen _textPen = QPen(QApplication::palette().text(), 1);
84 Qt::Alignment _alignment = (Qt::AlignLeft | Qt::AlignVCenter);
85 QFont _font = QApplication::font();
88 EditableText(QGraphicsItem *parent =
nullptr) : QGraphicsRectItem(parent) {
89 setPen(QPen(Qt::NoPen));
91 void setFont(
const QFont &font) { _font = font; }
93 std::string text()
const {
return _text.toStdString(); }
94 void setText(
const std::string &t) {
98 void setTextAlignment(Qt::Alignment a) {
102 void setTextPen(
const QPen &pen) {
106 virtual void paint(QPainter *painter,
const QStyleOptionGraphicsItem *option,
107 QWidget *widget)
override {
112 QGraphicsRectItem::paint(painter, option, widget);
113 painter->setFont(_font);
114 painter->setPen(_textPen);
115 painter->drawText(rect().marginsRemoved(margins()), _text,
116 QTextOption(_alignment));
119 virtual void mouseDoubleClickEvent(QGraphicsSceneMouseEvent *event)
override {
121 scene()->views().front()->setFocusPolicy(Qt::StrongFocus);
122 scene()->views().front()->setFocus();
123 _edit =
new QLineEdit();
124 _edit->setFixedSize(rect().width(), rect().height());
125 _edit->setAlignment(_alignment);
126 _edit->setTextMargins(margins());
127 _edit->setText(_text);
128 connect(_edit, &QLineEdit::editingFinished,
this, [
this]() {
129 if (_text != _edit->text()) {
130 _text = _edit->text();
131 LOG_DEBUG(
"text changed " << text());
134 LOG_DEBUG(
"text unchanged");
136 scene()->views().front()->setFocusPolicy(Qt::NoFocus);
138 _proxy->clearFocus();
139 _edit->deleteLater();
141 _proxy->deleteLater();
143 scene()->clearFocus();
144 scene()->views().front()->window()->setFocus();
146 _proxy =
new QGraphicsProxyWidget(
this);
147 _proxy->setWidget(_edit);
148 _proxy->setPos(rect().x(), rect().y());
149 _edit->setFocus(Qt::MouseFocusReason);
157 class ResizeHandle :
public QGraphicsRectItem {
160 double _drag_offset = 0.0;
161 QRectF _drag_start_rect;
164 double timeToPosition(
double t) {
166 double bag_duration = _parent->bag_duration;
167 double track_length = _parent->track_length;
168 return t * track_length / bag_duration;
170 double positionToTime(
double p) {
172 double bag_duration = _parent->bag_duration;
173 double track_length = _parent->track_length;
174 return p / track_length * bag_duration;
177 : _parent(parent), _side(side) {
178 setCursor(Qt::SizeHorCursor);
179 setParentItem(parent);
180 setBrush(QBrush(Qt::transparent));
181 setPen(QPen(Qt::NoPen));
184 auto rect = _parent->rect();
185 double w = std::min(5.0, rect.width() * 0.25);
187 rect.setRight(rect.left() + w);
190 rect.setLeft(rect.right() - w);
193 setCursor(Qt::SizeHorCursor);
195 virtual void mousePressEvent(QGraphicsSceneMouseEvent *event)
override {
196 if (event->button() == Qt::LeftButton) {
198 LOG_DEBUG(
"begin resize span");
199 _drag_start_rect = rect();
201 _drag_offset =
event->scenePos().x() - _parent->rect().left();
204 _drag_offset =
event->scenePos().x() - _parent->rect().right();
208 virtual void mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
override {
209 if (event->button() == Qt::LeftButton) {
210 if (_drag_start_rect != rect()) {
211 if (rect().width() < 2.5) {
213 auto &spans = _parent->_track->branch(ws(),
true)->spans();
215 std::remove(spans.begin(), spans.end(), _parent->_annotation),
226 virtual void mouseMoveEvent(QGraphicsSceneMouseEvent *event)
override {
227 if (event->buttons() == Qt::LeftButton) {
229 LOG_DEBUG(
"resize span");
230 auto rect = _parent->rect();
233 double x =
event->scenePos().x() - _drag_offset;
235 if (
auto branch = _parent->_track->branch(ws(),
false)) {
236 for (
auto &span : branch->spans()) {
237 if (timeToPosition(span->start() + span->duration()) <
238 _drag_start_rect.center().x()) {
240 timeToPosition(span->start() + span->duration()));
244 rect.setLeft(std::max(0.0, std::min(rect.right(), x)));
247 double x =
event->scenePos().x() - _drag_offset;
249 if (
auto branch = _parent->_track->branch(ws(),
false)) {
250 for (
auto &span : branch->spans()) {
251 if (timeToPosition(span->start()) >
252 _drag_start_rect.center().x()) {
253 x = std::min(x, timeToPosition(span->start()));
257 rect.setRight(std::min(scene()->sceneRect().width() -
258 track_label_width - track_padding_right,
259 std::max(rect.left(), x)));
261 _parent->setItemRect(rect.x(), rect.y(), rect.width());
267 std::shared_ptr<AnnotationTrack> _track;
268 std::shared_ptr<AnnotationSpan> _annotation;
269 double track_length = 0;
270 double bag_duration = 0.0;
271 double timeToPosition(
double t) {
return t * track_length / bag_duration; }
272 double positionToTime(
double p) {
return p / track_length * bag_duration; }
273 QRectF _drag_start_rect;
274 ResizeHandle *left_handle =
new ResizeHandle(
this, -1);
275 ResizeHandle *right_handle =
new ResizeHandle(
this, +1);
276 bool _dragged =
false;
277 std::unordered_set<std::shared_ptr<AnnotationSpan>> _dragged_spans;
278 bool _selected =
false;
279 double _track_color = 0;
281 bool snap(
double &x) {
282 snap(x, [
this](
const std::shared_ptr<AnnotationSpan> &span) {
283 return span != _annotation;
286 template <
class F>
bool snap(
double &x,
const F &filter) {
287 double snap_position = 0;
288 bool snapped =
false;
291 double l = timeToPosition(ws->player->time());
292 if (!snapped || std::abs(x - l) < std::abs(x - snap_position)) {
297 for (
size_t itrack = 0; itrack < trackCount(ws()); itrack++) {
298 if (
auto track = std::dynamic_pointer_cast<AnnotationTrack>(
299 trackAt(ws(), itrack))) {
300 if (
auto branch = track->branch(ws(),
false)) {
301 for (
auto &other_span : branch->spans()) {
302 if (!filter(other_span)) {
305 double l = timeToPosition(other_span->start());
307 timeToPosition(other_span->start() + other_span->duration());
308 if (!snapped || std::abs(x - l) < std::abs(x - snap_position)) {
312 if (!snapped || std::abs(x - r) < std::abs(x - snap_position)) {
320 if (std::abs(snap_position - x) < 6.1) {
328 static std::unordered_set<AnnotationSpanItem *> &instances() {
329 static std::unordered_set<AnnotationSpanItem *> instances;
333 QPointF &last_clicked_point = *[]() {
341 setTextAlignment(Qt::AlignHCenter | Qt::AlignVCenter);
343 setTextPen(QPen(QBrush(QColor(0, 0, 0)), 1));
345 changed.connect([
this](
const std::string &label) {
347 if (label == _track->label()) {
348 _annotation->label().clear();
350 _annotation->label() = label;
355 instances().insert(
this);
359 void setItemRect(
double x,
double y,
double width) {
360 setRect(x, y, width, track_height);
361 left_handle->update();
362 right_handle->update();
366 size_t itrack = std::max(0.0, std::round(y / track_height) - 1.0);
367 if (
auto track = trackAt(ws(), itrack)) {
369 setTextPen(QPen(QBrush(QColor(0, 0, 0)), 1));
370 QFont font = QApplication::font();
371 if (_annotation->label().empty()) {
372 setText(track->label());
373 font.setItalic(
true);
375 setText(_annotation->label());
377 _track_color = track->color();
378 if (_selected = ws->selection().contains(_annotation)) {
379 setBrush(QBrush(QColor::fromHsvF(track->color(), 0.8, 0.6)));
380 setTextPen(QPen(QBrush(QColor(Qt::white)), 1));
382 setBrush(QBrush(QColor::fromHsvF(track->color(), 0.2, 1.0)));
383 setTextPen(QPen(QBrush(QColor(Qt::black)), 1));
385 setPen(QPen(QBrush(Qt::black), 1));
390 virtual void paint(QPainter *painter,
const QStyleOptionGraphicsItem *option,
391 QWidget *widget)
override {
392 EditableText::paint(painter, option, widget);
393 painter->setBrush(QBrush(Qt::transparent));
394 painter->setPen(QPen(QBrush(QColor(255, 255, 255, 200)), 1, Qt::SolidLine,
395 Qt::SquareCap, Qt::MiterJoin));
396 painter->drawRect(rect().marginsRemoved(QMarginsF(1, 1, 1, 1)));
398 void update(
const std::shared_ptr<Workspace> &ws,
size_t track_index,
399 const std::shared_ptr<AnnotationTrack> &track,
400 const std::shared_ptr<AnnotationSpan> &annotation,
401 double track_length,
double bag_duration) {
403 _annotation = annotation;
404 this->track_length = track_length;
405 this->bag_duration = bag_duration;
406 setItemRect(track_length * annotation->start() / bag_duration,
407 (track_index + 1) * track_height,
408 track_length * annotation->duration() / bag_duration);
410 virtual void mousePressEvent(QGraphicsSceneMouseEvent *event)
override {
411 QGraphicsRectItem::mousePressEvent(event);
413 if (event->button() == Qt::LeftButton) {
414 LOG_DEBUG(
"begin dragging annotation span");
418 _dragged_spans.clear();
420 LOG_DEBUG(
"select " << _annotation);
421 ws->currentAnnotationTrack() = _track;
422 if (event->modifiers() & Qt::ShiftModifier) {
423 if (!last_clicked_point.isNull()) {
425 std::min(last_clicked_point.x(),
event->scenePos().x());
427 std::max(last_clicked_point.x(),
event->scenePos().x());
429 std::min(std::floor(last_clicked_point.y() / track_height) *
432 double bottom = std::max(
433 std::ceil(last_clicked_point.y() / track_height) * track_height,
435 for (
auto *instance : instances()) {
436 if (instance->rect().right() - 1e-6 > left &&
437 instance->rect().left() + 1e-6 < right &&
438 instance->rect().center().y() >= top &&
439 instance->rect().center().y() <= bottom) {
440 ws->selection().add(instance->_annotation);
443 ws->selection().add(_annotation);
445 ws->selection().toggle(_annotation);
446 if (ws->selection().contains(_annotation)) {
447 last_clicked_point =
event->scenePos();
449 last_clicked_point = QPointF();
452 }
else if (event->modifiers() & Qt::ControlModifier) {
453 ws->selection().toggle(_annotation);
454 if (ws->selection().contains(_annotation)) {
455 last_clicked_point =
event->scenePos();
457 last_clicked_point = QPointF();
460 if (!ws->selection().contains(_annotation)) {
461 ws->selection() = _annotation;
462 last_clicked_point =
event->scenePos();
464 for (
auto object : ws->selection().resolve(ws())) {
465 if (
auto span = std::dynamic_pointer_cast<AnnotationSpan>(
object)) {
466 _dragged_spans.insert(span);
469 for (
auto &view : instances()) {
470 if (_dragged_spans.find(view->_annotation) !=
471 _dragged_spans.end()) {
472 view->_drag_start_rect = view->rect();
476 left_handle->unsetCursor();
477 right_handle->unsetCursor();
482 virtual void mouseMoveEvent(QGraphicsSceneMouseEvent *event)
override {
483 QGraphicsRectItem::mouseMoveEvent(event);
484 if (event->buttons() == Qt::LeftButton) {
485 if (!_dragged_spans.empty()) {
488 double timeline_duration = timelineDuration(ws());
489 double max_x = timeToPosition(timeline_duration);
490 size_t track_count = trackCount(ws());
491 double dx =
event->scenePos().x() -
492 event->buttonDownScenePos(Qt::LeftButton).x();
493 double di = std::round((event->scenePos().y() -
494 event->buttonDownScenePos(Qt::LeftButton).y()) /
496 for (
auto *view : instances()) {
497 if (_dragged_spans.find(view->_annotation) != _dragged_spans.end()) {
499 std::round(view->_drag_start_rect.y() / track_height) - 1;
500 di = std::min(track_count - 1.0, std::max(0.0, di + vi)) - vi;
504 bool snapped =
false;
505 double best_snap = 0;
506 for (
auto *view : instances()) {
507 if (_dragged_spans.find(view->_annotation) !=
508 _dragged_spans.end()) {
509 for (
double offset : {0.0, view->rect().width()}) {
510 double side = view->_drag_start_rect.x() + offset + dx;
513 const std::shared_ptr<AnnotationSpan> &span) {
514 return _dragged_spans.find(span) == _dragged_spans.end();
517 if (!snapped || std::abs(x) < std::abs(best_snap)) {
527 for (
auto *view : instances()) {
528 if (_dragged_spans.find(view->_annotation) != _dragged_spans.end()) {
529 dx = std::max(dx, -view->_drag_start_rect.x());
530 dx = std::min(max_x - view->_drag_start_rect.right(), dx);
533 for (
auto *view : instances()) {
534 if (_dragged_spans.find(view->_annotation) != _dragged_spans.end()) {
535 for (
auto *other : instances()) {
536 if (_dragged_spans.find(other->_annotation) ==
537 _dragged_spans.end()) {
538 if ((std::round(view->_drag_start_rect.y() / track_height) +
539 di) == std::round(other->rect().y() / track_height)) {
540 double min_distance =
541 (view->rect().width() + other->rect().width()) * 0.5;
542 double distance = (view->_drag_start_rect.center().x() + dx) -
543 other->rect().center().x();
544 if (std::abs(distance) < min_distance) {
545 dx += (std::signbit(distance) ? -1.0 : 1.0) * min_distance -
553 for (
auto *view : instances()) {
554 if (_dragged_spans.find(view->_annotation) != _dragged_spans.end()) {
555 for (
auto *other : instances()) {
556 if (_dragged_spans.find(other->_annotation) ==
557 _dragged_spans.end()) {
558 if ((std::round(view->_drag_start_rect.y() / track_height) +
559 di) == std::round(other->rect().y() / track_height)) {
560 double min_distance =
561 (view->rect().width() + other->rect().width()) * 0.5;
562 double distance = (view->_drag_start_rect.center().x() + dx) -
563 other->rect().center().x();
564 if (std::abs(distance) < min_distance - 1e-6) {
572 for (
auto *view : instances()) {
573 if (_dragged_spans.find(view->_annotation) != _dragged_spans.end()) {
574 if (view->_drag_start_rect.x() + dx < -1e-6) {
577 if (view->_drag_start_rect.right() + dx > max_x + 1e-6) {
582 for (
auto *view : instances()) {
583 if (_dragged_spans.find(view->_annotation) != _dragged_spans.end()) {
584 auto rect = view->rect();
585 view->setItemRect(view->_drag_start_rect.x() + dx,
586 view->_drag_start_rect.y() + di * track_height,
593 virtual void mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
override {
594 QGraphicsRectItem::mouseReleaseEvent(event);
595 if (instances().find(
this) == instances().end()) {
596 throw std::runtime_error(
"already deleted");
598 LOG_DEBUG(
"mouse release event " <<
this);
599 if (event->button() == Qt::LeftButton) {
600 if (!_dragged_spans.empty() && _dragged) {
601 LOG_DEBUG(
"finished dragging annotation span");
605 std::vector<QPointer<AnnotationSpanItem>> insts;
606 for (
auto &view : instances()) {
607 insts.emplace_back(view);
609 LOG_DEBUG(
"count " << insts.size());
610 for (
auto &view : insts) {
612 if (_dragged_spans.find(view->_annotation) !=
613 _dragged_spans.end()) {
618 LOG_DEBUG(
"view gone");
621 QTimer::singleShot(0, []() {
626 _dragged_spans.clear();
628 if (event->modifiers() == 0) {
630 ws->selection() = _annotation;
632 last_clicked_point =
event->buttonDownPos(Qt::LeftButton);
640 if (
auto annotation = _annotation) {
641 double start = positionToTime(rect().x());
642 double duration = positionToTime(rect().width());
644 std::max(0.0, std::round(rect().y() / track_height) - 1.0);
645 if (annotation->start() != start || annotation->duration() != duration ||
646 trackAt(ws(), itrack) != _track) {
647 if (itrack >= 0 && itrack < trackCount(ws())) {
649 if (trackAt(ws(), itrack) != _track) {
650 auto &spans = _track->branch(ws(),
true)->spans();
651 spans.erase(std::remove(spans.begin(), spans.end(), _annotation),
653 if (
auto track = std::dynamic_pointer_cast<AnnotationTrack>(
654 trackAt(ws(), itrack))) {
655 track->branch(ws(),
true)->spans().push_back(_annotation);
658 annotation->start() = start;
659 annotation->duration() = duration;
661 LOG_ERROR(
"failed to move annotation");
664 LOG_DEBUG(
"annotation not moved");
674 QGraphicsRectItem *_reorder_marker =
new QGraphicsRectItem(
this);
677 virtual void mousePressEvent(QGraphicsSceneMouseEvent *event)
override {
678 static std::weak_ptr<TrackBase> last_clicked_track;
679 EditableText::mousePressEvent(event);
680 if (event->button() == Qt::LeftButton) {
683 LOG_DEBUG(
"select " << _parent->_track->label());
684 if (event->modifiers() & Qt::ControlModifier) {
685 ws->selection().toggle(_parent->_track);
686 }
else if (event->modifiers() & Qt::ShiftModifier) {
687 if (
auto last_clicked = last_clicked_track.lock()) {
688 size_t track_count = trackCount(ws());
689 bool selection_flag =
false;
690 for (
size_t i = 0; i < track_count; i++) {
691 auto track = trackAt(ws(), i);
692 bool range_boundary =
693 (track == last_clicked || track == _parent->_track);
694 if (selection_flag || range_boundary) {
695 ws->selection().add(track);
697 if (range_boundary) {
698 selection_flag = !selection_flag;
702 ws->selection().toggle(_parent->_track);
705 ws->selection() = _parent->_track;
707 if (
auto annotation_track =
708 std::dynamic_pointer_cast<AnnotationTrack>(_parent->_track)) {
709 if (ws->selection().contains(annotation_track)) {
710 ws->currentAnnotationTrack() = annotation_track;
711 last_clicked_track = annotation_track;
717 _reorder_marker->setBrush(QBrush(QColor(100, 100, 100)));
718 _reorder_marker->setPen(QPen(Qt::NoPen));
721 size_t y2i(
double y) {
723 return std::max((int64_t)0,
724 std::min((int64_t)trackCount(ws()),
725 (int64_t)std::round(y / track_height) - 1));
727 virtual void mouseMoveEvent(QGraphicsSceneMouseEvent *event)
override {
728 EditableText::mouseMoveEvent(event);
730 if (event->buttons() == Qt::LeftButton) {
731 LOG_DEBUG(
"drag track");
732 double thickness = 6;
734 (y2i(event->scenePos().y()) + 1) * track_height - thickness * 0.5;
735 y = std::max(y, track_height);
736 _reorder_marker->setRect(rect().x(), y, rect().width(), thickness);
737 _reorder_marker->show();
740 virtual void mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
override {
741 EditableText::mouseReleaseEvent(event);
743 _reorder_marker->hide();
744 if (event->button() == Qt::LeftButton &&
745 event->buttonDownScenePos(Qt::LeftButton).y() !=
746 event->scenePos().y()) {
748 size_t new_index = y2i(event->scenePos().y());
749 LOG_DEBUG(
"dragged from " << _parent->_track_index <<
" to " 751 if (new_index != _parent->_track_index) {
753 auto &tracks = ws->document()->timeline()->tracks();
754 auto new_position_it =
755 tracks.insert(tracks.begin() + new_index, _parent->_track);
756 for (
auto it = tracks.begin(); it < tracks.end(); it++) {
757 if (*it == _parent->_track && it != new_position_it) {
762 QTimer::singleShot(0, []() {
772 virtual void paint(QPainter *painter,
773 const QStyleOptionGraphicsItem *option,
774 QWidget *widget)
override {
775 EditableText::paint(painter, option, widget);
776 painter->setBrush(QBrush(Qt::transparent));
777 painter->setPen(QPen(QBrush(QColor(255, 255, 255, 200)), 1, Qt::SolidLine,
778 Qt::SquareCap, Qt::MiterJoin));
779 painter->drawRect(rect().marginsRemoved(QMarginsF(1, 1, 1, 1)));
782 LabelItem *_text =
nullptr;
783 std::shared_ptr<TrackBase> _track;
784 bool _selected =
false;
787 double scrollPositionX()
const {
788 auto *view = scene()->views().first();
789 QPointF p = view->mapToScene(view->viewport()->rect().topLeft());
794 const size_t _track_index = 0;
797 std::shared_ptr<TrackBase> track()
const {
return _track; }
798 TrackViewBase(
size_t track_index, std::shared_ptr<TrackBase> track)
799 : _track(track), _track_index(track_index) {
801 setBrush(QBrush(Qt::transparent));
802 setPen(QPen(Qt::NoPen));
804 _text =
new LabelItem(
this);
805 _text->changed.connect(
this, [track](
const std::string &str) {
807 track->label() = str;
810 _text->setZValue(100000);
812 virtual void update(
const std::shared_ptr<Workspace> &ws,
double width) {
813 QRectF rect = scene()->sceneRect();
814 double y = track_height * (_track_index + 1);
815 setRect(QRectF(scrollPositionX() + track_label_width, y,
816 rect.width() - track_padding_right - track_label_width,
818 _text->setRect(scrollPositionX() - 1, y, track_label_width + 1,
820 setZValue(ws->selection().contains(_track) ? 2 : 1);
821 _text->setText(_track->label().c_str());
822 if (_selected = ws->selection().contains(_track)) {
824 _text->setZValue(100002);
825 _text->setBrush(QBrush(QColor::fromHsvF(_track->color(), 0.8, 0.6)));
826 _text->setPen(QPen(QApplication::palette().brush(QPalette::Mid), 1));
827 _text->setTextPen(QPen(QBrush(Qt::white), 1));
831 _text->setBrush(QBrush(QColor::fromHsvF(_track->color(), 0.2, 1)));
832 _text->setTextPen(QPen(QBrush(Qt::black), 1));
833 _text->setPen(QPen(QApplication::palette().brush(QPalette::Mid), 1));
834 _text->setZValue(100001);
837 QFont font = QApplication::font();
838 if (_track->id() == ws->currentAnnotationTrack().id()) {
841 _text->setFont(font);
848 double xmin = 0, xmax = 0, ymin = 0, ymax = 0;
849 void merge(
const Item &other) {
850 xmin = std::min(xmin, other.xmin);
851 ymin = std::min(ymin, other.ymin);
852 xmax = std::max(xmax, other.xmax);
853 ymax = std::max(ymax, other.ymax);
857 std::vector<Item> items;
860 std::deque<Level> levels;
862 std::shared_ptr<Data> _data;
872 void sample(
size_t ilevel,
size_t iitem,
double xmin,
double xmax,
873 double xstep, std::vector<Sample> &samples) {
874 auto &item = _data->levels[ilevel].items[iitem];
875 double item_end = item.xmax;
876 if (iitem + 1 < _data->levels[ilevel].items.size()) {
877 auto &item2 = _data->levels[ilevel].items[iitem + 1];
878 item_end = item2.xmin;
880 double prev_x = item.xmin;
882 prev_x = _data->levels[ilevel].items[iitem - 1].xmax;
884 if (item_end < xmin || prev_x > xmax) {
887 if (ilevel >= 1 && (item_end - item.xmin) > xstep) {
888 sample(ilevel - 1, iitem * 2, xmin, xmax, xstep, samples);
889 if (iitem * 2 + 1 < _data->levels[ilevel - 1].items.size()) {
890 sample(ilevel - 1, iitem * 2 + 1, xmin, xmax, xstep, samples);
894 sample.x = (item.xmin + item.xmax) / 2;
895 sample.ymin = item.ymin;
896 sample.ymax = item.ymax;
897 samples.push_back(sample);
902 void clear() { _data.reset(); }
903 void push(
double x,
double y) {
906 _data = std::make_shared<Data>();
908 if (_data.use_count() > 1) {
909 _data = std::make_shared<Data>(*_data);
913 item.xmin = item.xmax = x;
914 item.ymin = item.ymax = y;
915 if (_data->levels.empty()) {
916 _data->levels.emplace_back();
918 _data->levels.front().items.push_back(item);
920 for (
size_t ilevel = 1;; ilevel++) {
921 if (_data->levels[ilevel - 1].items.size() <= 1) {
924 if (_data->levels.size() <= ilevel) {
925 _data->levels.emplace_back();
927 size_t iitem = (_data->levels[ilevel - 1].items.size() - 1) / 2;
928 _data->levels[ilevel].items.resize(iitem + 1);
929 _data->levels[ilevel].items[iitem] =
930 _data->levels[ilevel - 1].items[iitem * 2];
931 if (iitem * 2 + 1 < _data->levels[ilevel - 1].items.size()) {
932 _data->levels[ilevel].items[iitem].merge(
933 _data->levels[ilevel - 1].items[iitem * 2 + 1]);
938 throw std::runtime_error(
"graph tree overflow");
942 void sample(
double xmin,
double xmax,
double xstep,
943 std::vector<Sample> &samples) {
944 if (_data && !_data->levels.empty()) {
945 sample(_data->levels.size() - 1, 0, xmin, xmax, xstep, samples);
948 void range(
double &ymin,
double &ymax) {
949 if (_data && !_data->levels.empty()) {
950 ymin = _data->levels.back().items.front().ymin;
951 ymax = _data->levels.back().items.front().ymax;
957 std::shared_ptr<GraphTrack> _track;
958 size_t _track_height = 0;
959 double _track_length = 0;
960 double _bag_duration = 0.0;
962 double timeToPosition(
double t) {
return t * _track_length / _bag_duration; }
963 double positionToTime(
double p) {
return p / _track_length * _bag_duration; }
964 volatile bool _exit_flag =
false;
965 std::mutex _load_mutex;
967 std::string _load_query, _load_topic;
968 double _load_duration = 0;
969 std::condition_variable _load_condition;
970 std::shared_ptr<BagPlayer> _load_player;
972 bool _load_flag =
false;
975 QObject::connect(&o, &QObject::destroyed,
this,
976 [
this](QObject *o) {
LockScope()->modified(); });
978 std::thread _load_thread{[
this]() {
983 std::shared_ptr<BagPlayer> player;
986 std::unique_lock<std::mutex> lock(_load_mutex);
995 duration = _load_duration;
997 player = _load_player;
1000 if (topic.empty() || query.str().empty() || player ==
nullptr) {
1008 _load_condition.wait(lock);
1011 LOG_DEBUG(
"load graph track data");
1013 LOG_DEBUG(
"begin loading graph track data");
1015 auto t0 = std::chrono::steady_clock::now();
1016 player->readMessageSamples(
1017 topic, 0, duration, [&](
const std::shared_ptr<const Message> &msg) {
1020 double x = (msg->time() - player->startTime()).toSec() / duration;
1021 double y = query(parser).toDouble();
1023 if (std::chrono::steady_clock::now() >
1024 t0 + std::chrono::milliseconds(250)) {
1026 std::unique_lock<std::mutex> lock(_load_mutex);
1030 t0 = std::chrono::steady_clock::now();
1032 return !_exit_flag && !_load_flag;
1037 std::unique_lock<std::mutex> lock(_load_mutex);
1042 LOG_DEBUG(
"graph track data loaded");
1048 GraphTrackView(
size_t track_index, std::shared_ptr<GraphTrack> track)
1049 :
TrackViewBase(track_index, track), _track_height(track_index),
1054 std::unique_lock<std::mutex> lock(_load_mutex);
1056 _load_condition.notify_all();
1059 _load_thread.join();
1062 virtual void update(
const std::shared_ptr<Workspace> &ws,
1063 double width)
override {
1064 TrackViewBase::update(ws, width);
1065 _track_length = width;
1066 _bag_duration = timelineDuration(ws);
1067 _pen = QPen(QBrush(QColor::fromHsvF(_track->color(), 1.0, 0.5)), 1.5,
1068 Qt::SolidLine, Qt::SquareCap, Qt::RoundJoin);
1069 if (_watcher.changed(ws->player, _bag_duration, _track->query().str(),
1070 _track->query().str())) {
1071 LOG_DEBUG(
"load graph track");
1072 std::unique_lock<std::mutex> lock(_load_mutex);
1074 _load_player = ws->player;
1075 _load_duration = _bag_duration;
1076 _load_query = _track->query().str();
1077 _load_topic = _track->topic().topic();
1078 _load_condition.notify_all();
1082 virtual void paint(QPainter *painter,
const QStyleOptionGraphicsItem *option,
1083 QWidget *widget)
override {
1088 std::unique_lock<std::mutex> lock(_load_mutex);
1091 std::vector<GraphTree::Sample> samples;
1092 auto *view = scene()->views().first();
1093 auto view_top_left = view->mapToScene(view->viewport()->rect().topLeft());
1094 auto view_top_right =
1095 view->mapToScene(view->viewport()->rect().topRight());
1096 data.sample(view_top_left.x() / _track_length,
1097 view_top_right.x() / _track_length, 1.5 / _track_length,
1099 LOG_DEBUG(
"graph track samples " << samples.size());
1100 if (!samples.empty()) {
1101 QVector<QPointF> points;
1110 double hi = 0, lo = 0;
1112 for (
size_t isample = 0; isample < samples.size(); isample++) {
1113 auto &sample = samples[isample];
1114 points.push_back(QPointF(
1115 sample.x * _track_length,
1116 (0.9 - (sample.ymax - lo) / (hi - lo) * 0.8) * track_height +
1117 track_height + _track_index * track_height));
1119 for (ssize_t isample = ssize_t(samples.size()) - 1; isample >= 0;
1121 auto &sample = samples[isample];
1122 points.push_back(QPointF(
1123 sample.x * _track_length,
1124 (0.9 - (sample.ymin - lo) / (hi - lo) * 0.8) * track_height +
1125 track_height + _track_index * track_height));
1127 polygon = QPolygonF(points);
1131 painter->setRenderHints(QPainter::Antialiasing);
1132 painter->setPen(_pen);
1134 painter->setBrush(_pen.brush());
1136 painter->drawPolygon(polygon);
1175 const std::shared_ptr<AnnotationTrack> _track;
1176 QGraphicsRectItem *_item_prototype =
nullptr;
1177 double _new_item_start = 0;
1178 std::vector<AnnotationSpanItem *> _span_items;
1179 QGraphicsRectItem *_clip_rect =
new QGraphicsRectItem(
this);
1182 void updateItemPrototype(QGraphicsSceneMouseEvent *event) {
1184 double track_length =
1185 scene()->sceneRect().width() - track_label_width - track_padding_right;
1186 double x = std::max(0.0, std::min(track_length, event->scenePos().x()));
1188 if (x == _new_item_start) {
1191 x = std::min(x, track_length);
1192 if (
auto branch = _track->branch(ws(),
false)) {
1193 for (
auto &span : branch->spans()) {
1194 double left = span->start() * track_length / timelineDuration(ws());
1195 double right = (span->start() + span->duration()) * track_length /
1196 timelineDuration(ws());
1197 if (_new_item_start > right) {
1198 x = std::max(x, right);
1200 if (_new_item_start < left) {
1201 x = std::min(x, left);
1205 double x0 = std::min(x, _new_item_start);
1206 double x1 = std::max(x, _new_item_start);
1207 _item_prototype->setRect(x0, rect().y(), x1 - x0, rect().height());
1211 virtual void mousePressEvent(QGraphicsSceneMouseEvent *event)
override {
1212 QGraphicsItem::mousePressEvent(event);
1214 double track_length =
1215 scene()->sceneRect().width() - track_label_width - track_padding_right;
1216 if (event->button() == Qt::LeftButton) {
1219 std::max(0.0, std::min(scene()->sceneRect().width() -
1220 track_padding_right - track_label_width,
1221 event->scenePos().x()));
1223 double t = _new_item_start * timelineDuration(ws()) / track_length;
1224 if (
auto branch = _track->branch(ws(),
false)) {
1225 for (
auto &span : branch->spans()) {
1226 if (t > span->start() && t < span->start() + span->duration()) {
1232 updateItemPrototype(event);
1233 _item_prototype->setBrush(
1234 QBrush(QColor::fromHsvF(_track->color(), 1.0, 0.5)));
1235 _item_prototype->show();
1237 if (!ws->selection().empty()) {
1238 ws->selection().clear();
1242 virtual void mouseMoveEvent(QGraphicsSceneMouseEvent *event)
override {
1243 QGraphicsItem::mouseMoveEvent(event);
1245 if (event->buttons() & Qt::LeftButton) {
1247 updateItemPrototype(event);
1250 virtual void mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
override {
1251 QGraphicsItem::mouseReleaseEvent(event);
1253 if (event->button() == Qt::LeftButton) {
1255 if (_item_prototype->isVisible()) {
1256 if (_item_prototype->rect().width() > 3.2) {
1258 double track_length = scene()->sceneRect().width() -
1259 track_padding_right - track_label_width;
1260 auto span = std::make_shared<AnnotationSpan>();
1261 span->start() = _item_prototype->rect().x() * timelineDuration(ws()) /
1263 span->duration() = _item_prototype->rect().width() *
1264 timelineDuration(ws()) / track_length;
1265 _track->branch(ws(),
true)->spans().push_back(span);
1266 _item_prototype->hide();
1267 ws->selection() = span;
1268 ws->currentAnnotationTrack() = _track;
1271 _item_prototype->hide();
1278 std::shared_ptr<AnnotationTrack> track)
1282 setBrush(QBrush(Qt::transparent));
1283 setPen(QPen(Qt::NoPen));
1285 _item_prototype =
new QGraphicsRectItem();
1286 _item_prototype->setParentItem(
this);
1287 _item_prototype->setBrush(QBrush(Qt::blue));
1288 _item_prototype->setPen(QPen(Qt::NoPen));
1289 _item_prototype->hide();
1291 _clip_rect->setFlag(ItemClipsChildrenToShape,
true);
1292 _clip_rect->setBrush(QBrush(Qt::transparent));
1293 _clip_rect->setPen(QPen(Qt::NoPen));
1295 virtual void update(
const std::shared_ptr<Workspace> &ws,
1296 double width)
override {
1297 TrackViewBase::update(ws, width);
1299 auto *view = scene()->views().first();
1300 auto view_top_left = view->mapToScene(view->viewport()->rect().topLeft());
1301 auto view_top_right = view->mapToScene(view->viewport()->rect().topRight());
1302 _clip_rect->setRect(
1303 QRect(view_top_left.x() + track_label_width, scene()->sceneRect().y(),
1304 view_top_right.x() - view_top_left.x() - track_label_width,
1305 scene()->sceneRect().height() + track_height));
1307 QRectF rect = scene()->sceneRect();
1308 double y = track_height * (_track_index + 1);
1309 if (
auto branch = _track->branch(ws,
false)) {
1310 for (
size_t i = 0; i < branch->spans().size(); i++) {
1311 if (_span_items.size() <= i) {
1313 item->setParentItem(_clip_rect);
1314 _span_items.push_back(item);
1316 auto *item = _span_items[i];
1317 auto &annotation = branch->spans()[i];
1318 item->update(ws, _track_index, _track, annotation, width,
1319 timelineDuration(ws));
1321 while (_span_items.size() > branch->spans().size()) {
1322 delete _span_items.back();
1323 _span_items.pop_back();
1326 while (_span_items.size() > 0) {
1327 delete _span_items.back();
1328 _span_items.pop_back();
1335 double _width = 300;
1336 bool _enabled =
false;
1337 std::vector<TrackViewBase *> _tracks;
1341 const std::shared_ptr<TrackBase> &track) {
1342 if (
auto t = std::dynamic_pointer_cast<AnnotationTrack>(track)) {
1345 if (
auto t = std::dynamic_pointer_cast<GraphTrack>(track)) {
1348 throw std::runtime_error(
"unknown track type");
1352 virtual void sync(
const std::shared_ptr<Workspace> &ws)
override {
1356 size_t track_count = trackCount(ws);
1357 LOG_DEBUG(
"annotation tracks " << track_count);
1358 while (_tracks.size() > track_count) {
1359 delete _tracks.back();
1362 while (_tracks.size() < track_count) {
1364 createTrackView(_tracks.size(), trackAt(ws, _tracks.size()));
1366 _tracks.push_back(track);
1370 for (
size_t i = 0; i < _tracks.size(); i++) {
1371 if (_tracks[i]->track() != trackAt(ws, i)) {
1373 _tracks[i] =
nullptr;
1374 _tracks[i] = createTrackView(i, trackAt(ws, i));
1375 addItem(_tracks[i]);
1377 _tracks[i]->update(ws, _width);
1380 void updateSceneRect() {
1382 double height = track_height * (trackCount(ws()) + 1);
1383 setSceneRect(-track_label_width, 0,
1384 track_label_width + _width + track_padding_right, height);
1386 Scene(QWidget *parent) : QGraphicsScene(parent) {
1389 setBackgroundBrush(QApplication::palette().brush(QPalette::Window));
1391 void wheelEvent(QGraphicsSceneWheelEvent *wheel) {
1398 auto *view = views().first();
1400 QPointF center = view->mapToScene(view->viewport()->rect().center());
1401 double center2mouse = wheel->scenePos().x() - center.x();
1402 LOG_DEBUG(
"zoom " << wheel->scenePos().x() <<
" " << center.x() <<
" " 1404 center.setX(center.x() + center2mouse);
1405 center.setX(center.x() / _width);
1407 double degrees = wheel->delta() * (1.0 / 8);
1408 double exponent = degrees / 90;
1409 double factor = std::pow(0.5, -exponent);
1411 LOG_DEBUG(
"scene " << -track_label_width <<
" " << track_label_width <<
" " 1412 << _width <<
" " << track_label_width + _width);
1416 center.setX(center.x() * _width);
1417 center.setX(center.x() - center2mouse);
1418 view->centerOn(center);
1422 void handleSeek(QGraphicsSceneMouseEvent *event) {
1427 ws->player->seek(std::max(
1429 std::min(timelineDuration(ws()),
1430 event->scenePos().x() * timelineDuration(ws()) / _width)));
1433 virtual void mousePressEvent(QGraphicsSceneMouseEvent *event)
override {
1434 QGraphicsScene::mousePressEvent(event);
1435 if (event->isAccepted()) {
1438 if (event->button() == Qt::MiddleButton) {
1441 if (event->button() == Qt::LeftButton &&
1442 event->scenePos().y() < track_height) {
1446 virtual void mouseMoveEvent(QGraphicsSceneMouseEvent *event)
override {
1447 QGraphicsScene::mouseMoveEvent(event);
1448 if (event->isAccepted()) {
1451 if (event->buttons() == Qt::MiddleButton) {
1453 static int recursion_depth = 0;
1454 if (recursion_depth == 0) {
1456 if (
auto *view = views().first()) {
1457 double dx =
event->scenePos().x() -
event->lastScenePos().x();
1458 LOG_DEBUG(
"dx " << dx);
1459 view->translate(dx, 0);
1465 if (event->buttons() & Qt::LeftButton &&
1466 event->buttonDownScenePos(Qt::LeftButton).y() < track_height) {
1470 virtual void mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
override {
1471 QGraphicsScene::mouseReleaseEvent(event);
1474 void render(QPainter *painter,
const QRectF &target,
const QRectF &source,
1475 Qt::AspectRatioMode aspectRatioMode) {
1476 QGraphicsScene::render(painter, target, source, aspectRatioMode);
1479 void drawBackground(QPainter *painter,
const QRectF &rect) {
1480 QGraphicsScene::drawBackground(painter, rect);
1482 painter->fillRect(rect, QApplication::palette().brush(QPalette::Base));
1483 painter->setPen(QPen(QApplication::palette().brush(
1484 _enabled ? QPalette::Normal : QPalette::Disabled,
1485 QPalette::ButtonText),
1487 for (
double y = track_height; y < sceneRect().height() + track_height * 0.5;
1488 y += track_height) {
1489 painter->drawLine(rect.x(), y, rect.x() + rect.width(), y);
1490 if (y == track_height) {
1491 auto pen = painter->pen();
1492 auto brush = pen.brush();
1493 auto color = brush.color();
1494 color.setAlpha(color.alpha() / 4);
1495 brush.setColor(color);
1496 pen.setBrush(brush);
1497 painter->setPen(pen);
1507 setBrush(QBrush(Qt::transparent));
1508 setPen(QPen(Qt::NoPen));
1511 virtual void sync(
const std::shared_ptr<Workspace> &ws)
override {
1512 QRectF rect = scene()->sceneRect();
1513 setRect(QRectF(0, 0, rect.width() - track_label_width - track_padding_right,
1519 double _duration = 0.001;
1523 virtual void sync(
const std::shared_ptr<Workspace> &ws)
override {
1524 _duration = timelineDuration(ws);
1526 virtual QRectF boundingRect()
const override {
1527 auto rect = scene()->sceneRect();
1528 auto *graphics_view = scene()->views().first();
1529 if (graphics_view) {
1531 graphics_view->mapToScene(graphics_view->viewport()->geometry())
1533 rect.setHeight(clip.y() + clip.height());
1537 double timeToPosition(
double t) {
1538 auto r = parentItem()->boundingRect();
1539 return r.x() + r.width() * t / _duration;
1541 double positionToTime(
double p) {
1542 auto r = parentItem()->boundingRect();
1543 return (p - r.x()) * _duration / r.width();
1545 void drawTime(QPainter *painter,
double t) {
1546 double p = timeToPosition(t);
1547 painter->drawLine(p, 0.0, p, track_height * 0.15);
1549 tq = tq.addMSecs(std::round(t * 1000));
1550 QString tstr = tq.toString(
"hh:mm:ss.zzz");
1551 painter->drawText(QRectF(p - 100, 0, 200, track_height),
1552 Qt::AlignCenter | Qt::TextSingleLine, tstr);
1554 void drawTick(QPainter *painter,
double t) {
1555 double p = timeToPosition(t);
1556 painter->drawLine(p, track_height * 0.85, p, track_height);
1558 virtual void paint(QPainter *painter,
const QStyleOptionGraphicsItem *option,
1559 QWidget *widget)
override {
1563 QColor color = QApplication::palette().color(
1564 isEnabled() ? QPalette::Normal : QPalette::Disabled,
1565 QPalette::ButtonText);
1566 color.setAlpha(color.alpha() / 4);
1567 painter->setPen(QPen(QBrush(color), 1.0));
1570 auto *graphics_view = scene()->views().first();
1571 auto clip = graphics_view->mapToScene(graphics_view->viewport()->geometry())
1577 rect.setHeight(track_height);
1578 rect.setX(rect.x() - 1);
1579 rect.setWidth(rect.width() + 2);
1580 painter->fillRect(rect, QApplication::palette().brush(QPalette::Button));
1584 QPen(QApplication::palette().brush(isEnabled() ? QPalette::Normal
1585 : QPalette::Disabled,
1586 QPalette::ButtonText),
1588 painter->drawLine(rect.x(), track_height, rect.x() + rect.width(),
1592 painter->drawLine(clip.x() + track_label_width - 1, track_height,
1593 clip.x() + track_label_width - 1,
1594 clip.y() + clip.height());
1597 auto r = parentItem()->boundingRect();
1598 painter->drawLine(r.x() + r.width(), clip.y(), r.x() + r.width(),
1602 QPen(QApplication::palette().brush(isEnabled() ? QPalette::Normal
1603 : QPalette::Disabled,
1604 QPalette::ButtonText),
1607 double min_width = QFontMetrics(painter->font()).width(
" 00:00:00.000");
1608 double desired_time_step = min_width * _duration / r.width();
1631 double time_step = *step_sizes.begin();
1632 double tick_step = *step_sizes.begin();
1633 for (
auto &s : step_sizes) {
1634 tick_step = time_step;
1636 if (s > desired_time_step) {
1641 double t0 = std::max(0.0, std::round(positionToTime(clip.x()) / time_step) *
1645 std::min(_duration, std::round(positionToTime(clip.x() + clip.width()) /
1649 for (
double t = t0; t == t0 || t < tn; t += tick_step) {
1650 drawTick(painter, t);
1652 for (
double t = t0; t == t0 || t < tn - desired_time_step; t += time_step) {
1653 double p = timeToPosition(t);
1654 drawTime(painter, t);
1656 drawTick(painter, _duration);
1657 drawTime(painter, _duration);
1663 bool _dragged =
false;
1664 double _drag_offset = 0.0;
1665 QTimer *timer =
new QTimer((
ItemBase *)
this);
1669 setRect(0, -1, 11, track_height);
1670 auto brush = QBrush(QColor(255, 0, 0));
1672 setPen(QPen(QColor(130, 0, 0), 1.0));
1673 auto *seek_line =
new QGraphicsLineItem(5, rect().height() * 0.5, 5, 1000);
1674 seek_line->setPen(QPen(brush, 3.0));
1675 seek_line->setParentItem(
this);
1676 timer->setInterval(10);
1677 QObject::connect(timer, &QTimer::timeout, (
ItemBase *)
this,
1679 auto *fx =
new QGraphicsDropShadowEffect();
1680 fx->setBlurRadius(4);
1681 fx->setOffset(1, 2);
1682 setGraphicsEffect(fx);
1684 seek_line->setZValue(1000000);
1685 setCursor(Qt::OpenHandCursor);
1687 virtual void mousePressEvent(QGraphicsSceneMouseEvent *event)
override {
1688 if (event->button() == Qt::LeftButton) {
1691 setCursor(Qt::ClosedHandCursor);
1692 _drag_offset = pos().x() + rect().width() * 0.5 -
event->scenePos().x();
1696 virtual void mouseMoveEvent(QGraphicsSceneMouseEvent *event)
override {
1697 auto parent_rect = parentItem()->sceneBoundingRect();
1698 double x = _drag_offset +
event->scenePos().x();
1699 x = std::max(parent_rect.x(),
1700 std::min(parent_rect.x() + parent_rect.width(), x));
1701 setX(x - rect().width() * 0.5);
1704 if (
auto player = ws->player) {
1705 player->seek((x - parent_rect.x()) * timelineDuration(ws.ws()) /
1706 parent_rect.width());
1710 virtual void mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
override {
1711 if (event->button() == Qt::LeftButton) {
1713 setCursor(Qt::OpenHandCursor);
1716 virtual void sync(
const std::shared_ptr<Workspace> &ws)
override {
1722 auto r = parentItem()->sceneBoundingRect();
1723 auto px = ws->player->time() * r.width() / timelineDuration(ws) + r.x() -
1724 rect().width() * 0.5;
1732 if (!timer->isActive()) {
1739 if (timer->isActive()) {
1746 TimelineWidget::TimelineWidget() : QDockWidget(
"Timeline") {
1750 QWidget *content_widget =
new QWidget();
1752 auto *main_layout =
new QVBoxLayout(content_widget);
1753 main_layout->setSpacing(0);
1754 main_layout->setContentsMargins(0, 0, 0, 0);
1756 auto *playback_bar =
new QHBoxLayout(content_widget);
1757 playback_bar->setSpacing(0);
1758 playback_bar->setContentsMargins(3, 3, 3, 3);
1760 QWidget *playback_bar_widget =
new QWidget();
1761 playback_bar_widget->setAutoFillBackground(
true);
1762 playback_bar_widget->setLayout(playback_bar);
1763 setTitleBarWidget(playback_bar_widget);
1765 connect(
this, &QDockWidget::dockLocationChanged,
this,
1766 [
this](Qt::DockWidgetArea area) {
1767 LOG_DEBUG(
"timeline dock location changed");
1772 auto *playback_bar_left =
new QHBoxLayout(content_widget);
1773 playback_bar_left->setSpacing(0);
1774 playback_bar->addLayout(playback_bar_left, 4);
1776 std::unordered_set<QWidget *> always_active_widgets;
1779 auto *title =
new QLabel(tr(
"Timeline"));
1780 playback_bar_left->addWidget(title);
1781 always_active_widgets.insert(title);
1786 QMenu *menu =
new QMenu(
this);
1787 auto types = Type::find<TrackBase>()->list();
1788 for (
auto &type : types) {
1789 if (!type->constructable()) {
1792 QString label = type->name().c_str();
1793 connect(menu->addAction(label), &QAction::triggered,
this,
1794 [type,
this](
bool checked) {
1796 auto track = type->instantiate<
TrackBase>();
1797 for (
size_t i = 1;; i++) {
1798 auto label = std::string() + type->name() + std::to_string(i);
1799 bool label_taken =
false;
1800 for (
auto &t : ws->document()->timeline()->tracks()) {
1801 if (t->label() == label) {
1809 track->label() = label;
1814 std::vector<double> colors;
1815 for (
auto &t : ws->document()->timeline()->tracks()) {
1816 colors.push_back(t->color());
1818 if (colors.size() == 1) {
1819 double color = colors.at(0) + 0.5;
1820 color -= std::floor(color);
1821 track->color() = color;
1823 if (colors.size() >= 2) {
1824 std::sort(colors.begin(), colors.end());
1825 double best_color = 0;
1826 double largest_distance = 0;
1827 for (
size_t i = 0; i < colors.size(); i++) {
1828 size_t j = (i + 1) % colors.size();
1829 double distance = colors[j] - colors[i];
1830 distance -= std::floor(distance);
1831 if (distance > largest_distance) {
1832 largest_distance = distance;
1833 best_color = colors[i] + distance * 0.5;
1836 track->color() = best_color - std::floor(best_color);
1839 ws->document()->timeline()->tracks().push_back(track);
1843 button->setMenu(menu);
1844 always_active_widgets.insert(button);
1845 playback_bar_left->addWidget(button);
1850 always_active_widgets.insert(open_button);
1851 connect(open_button, &QPushButton::clicked,
this, [
this]() {
1852 QString path = QFileDialog::getOpenFileName(
1853 this, tr(
"Open Bag"), QString(),
1854 tr(
"Bag files (*.bag);;All files (*.*)"));
1855 if (path.isNull()) {
1858 MainWindow::instance()->openBag(path);
1860 playback_bar_left->addWidget(open_button);
1865 QMenu *menu =
new QMenu(
this);
1866 connect(menu, &QMenu::aboutToShow, [menu,
this]() {
1867 LOG_DEBUG(
"update timeline bag menu");
1869 connect(menu->addAction(
"Open Bag"), &QAction::triggered,
this,
1870 [
this](
bool checked) {
1871 QString path = QFileDialog::getOpenFileName(
1872 this, tr(
"Open Bag"), QString(),
1873 tr(
"Bag files (*.bag);;All files (*.*)"));
1874 if (path.isNull()) {
1877 MainWindow::instance()->openBag(path);
1881 auto action = menu->addAction(
"Close Bag");
1882 connect(action, &QAction::triggered,
this,
1883 [
this](
bool checked) { MainWindow::instance()->closeBag(); });
1884 action->setEnabled(ws->player !=
nullptr);
1886 std::set<std::string> branches;
1887 for (
auto &track : ws->document()->timeline()->tracks()) {
1888 LOG_DEBUG(
"track " << track->label());
1889 if (
auto annotation_track =
1890 std::dynamic_pointer_cast<AnnotationTrack>(track)) {
1891 for (
auto &branch : annotation_track->branches()) {
1892 branches.insert(branch->name());
1899 if (!branches.empty()) {
1900 menu->addSeparator();
1901 for (
auto &branch : branches) {
1902 auto *action = menu->addAction(QString::fromStdString(branch));
1903 connect(action, &QAction::triggered,
this,
1904 [
this, branch](
bool checked) {
1906 MainWindow::instance()->findAndOpenBag(branch);
1908 MainWindow::instance()->closeBag();
1911 action->setCheckable(
true);
1912 if (
auto player = ws->player) {
1913 action->setChecked(branch == ws->player->fileName());
1965 open_button->setMenu(menu);
1966 always_active_widgets.insert(open_button);
1967 playback_bar_left->addWidget(open_button);
1972 QMenu *menu =
new QMenu(
this);
1974 menu->addAction(
"Annotated Bag"), &QAction::triggered,
this,
1975 [
this](
bool checked) {
1978 auto player = ws->player;
1980 throw std::runtime_error(
"No bag player");
1982 std::string src_path(player->path());
1983 std::string dst_path(src_path +
".annotated.bag");
1984 QFile::remove(dst_path.c_str());
1985 if (!QFile::copy(src_path.c_str(), dst_path.c_str())) {
1986 throw std::runtime_error(
"Failed to copy bag file");
1988 rosbag::Bag src_bag(src_path, rosbag::bagmode::Read);
1989 rosbag::Bag dst_bag(dst_path, rosbag::bagmode::Append);
1990 rosbag::View view(src_bag);
1992 for (
auto &message : view) {
1993 double t = (message.getTime() - view.getBeginTime()).toSec();
1994 if (message.getDataType() ==
"sensor_msgs/Image") {
1995 if (
auto image_message =
1996 message.instantiate<sensor_msgs::Image>()) {
1997 QImage annotation_image(image_message->width,
1998 image_message->height,
1999 QImage::Format_Grayscale8);
2000 annotation_image.fill(Qt::black);
2001 QPainter annotation_painter(&annotation_image);
2002 for (
auto &track : ws->document()->timeline()->tracks()) {
2003 if (
auto annotation_track =
2004 std::dynamic_pointer_cast<AnnotationTrack>(track)) {
2005 for (
auto &branch : annotation_track->branches()) {
2006 if (branch->name() == player->fileName()) {
2007 for (
auto &span : branch->spans()) {
2008 for (
auto &annotation : span->annotations()) {
2009 if (
auto image_annotation =
2010 std::dynamic_pointer_cast<
2012 if (image_annotation->topic() ==
2013 message.getTopic()) {
2014 if (span->start() <= t + 1e-6 &&
2015 span->start() + span->duration() >
2017 auto label = span->label().empty()
2020 LOG_DEBUG(
"annotation span " 2022 << image_annotation->topic()
2023 <<
" " << span->start() <<
" " 2024 << span->duration());
2025 image_annotation->render();
2026 if (
auto &shape = image_annotation->shape) {
2027 annotation_painter.setPen(Qt::NoPen);
2028 annotation_painter.setBrush(Qt::white);
2029 annotation_painter.drawPath(*shape);
2040 annotation_painter.end();
2041 sensor_msgs::Image annotation_message;
2042 annotation_message.header = image_message->header;
2043 annotation_message.width = annotation_image.width();
2044 annotation_message.height = annotation_image.height();
2045 annotation_message.step = annotation_message.width;
2046 annotation_message.encoding =
"mono8";
2047 for (
size_t y = 0; y < annotation_message.height; y++) {
2048 auto *line = annotation_image.constScanLine(y);
2049 for (
size_t x = 0; x < annotation_message.width; x++) {
2050 annotation_message.data.push_back(line[x]);
2053 std::string annotation_topic = message.getTopic();
2054 if (!annotation_topic.empty() && annotation_topic[0] !=
'/') {
2055 annotation_topic =
"/" + annotation_topic;
2057 annotation_topic =
"/annotations" + annotation_topic;
2058 LOG_DEBUG(
"annotation_topic " << annotation_topic);
2059 dst_bag.write(annotation_topic, message.getTime(),
2060 annotation_message);
2066 }
catch (
const std::exception &ex) {
2067 QMessageBox::critical(
nullptr,
"Error", ex.what());
2070 connect(menu->addAction(
"Time Spans / CSV"), &QAction::triggered,
this,
2071 [
this](
bool checked) {
2074 auto player = ws->player;
2076 throw std::runtime_error(
"No bag player");
2078 std::string dst_path(player->path() +
".annotations.csv");
2079 LOG_INFO(
"writing csv " << dst_path);
2080 QFile::remove(dst_path.c_str());
2081 std::ofstream stream(dst_path);
2082 stream <<
"start,duration,label" << std::endl;
2083 for (
auto &track : ws->document()->timeline()->tracks()) {
2084 if (
auto annotation_track =
2085 std::dynamic_pointer_cast<AnnotationTrack>(track)) {
2086 for (
auto &branch : annotation_track->branches()) {
2087 if (branch->name() == player->fileName()) {
2088 for (
auto &span : branch->spans()) {
2089 auto label = span->label().empty() ? track->label()
2091 std::string label_escaped;
2092 for (
auto &c : label) {
2094 label_escaped.append(
"\"\"");
2096 label_escaped.push_back(c);
2099 stream << span->start() <<
"," << span->duration()
2100 <<
",\"" << label_escaped <<
"\"" << std::endl;
2106 LOG_SUCCESS(
"csv written " << dst_path);
2107 }
catch (
const std::exception &ex) {
2108 QMessageBox::critical(
nullptr,
"Error", ex.what());
2111 button->setMenu(menu);
2112 playback_bar_left->addWidget(button);
2116 auto *close_button =
new FlatButton(
"Close");
2117 connect(close_button, &QPushButton::clicked,
this,
2118 [
this]() { MainWindow::instance()->closeBag(); });
2119 playback_bar_left->addWidget(close_button);
2122 playback_bar_left->addStretch(1);
2125 auto *button =
new FlatButton(MATERIAL_ICON(
"fast_rewind", -0.13));
2126 connect(button, &QPushButton::clicked,
this, [
this]() {
2129 ws->player->rewind();
2132 playback_bar->addWidget(button);
2136 auto *button =
new FlatButton(MATERIAL_ICON(
"skip_previous", -0.1));
2137 connect(button, &QPushButton::clicked,
this, [
this]() {
2140 double current_time = ws->player->time();
2141 double seek_time = -std::numeric_limits<double>::max();
2142 for (size_t i = 0; i < trackCount(ws()); i++) {
2143 if (auto track = std::dynamic_pointer_cast<AnnotationTrack>(
2144 trackAt(ws(), i))) {
2145 if (auto branch = track->branch(ws(), false)) {
2146 for (auto &span : branch->spans()) {
2147 for (double span_time : {
2149 span->start() + span->duration(),
2151 if (span_time < current_time - 1e-6 &&
2152 std::abs(current_time - span_time) <
2153 std::abs(current_time - seek_time)) {
2154 seek_time = span_time;
2162 std::min(timelineDuration(ws()), std::max(0.0, seek_time)));
2165 playback_bar->addWidget(button);
2169 auto *button =
new FlatButton(MATERIAL_ICON(
"stop", -0.1));
2170 connect(button, &QPushButton::clicked,
this, [
this]() {
2176 playback_bar->addWidget(button);
2179 auto *button =
new FlatButton(MATERIAL_ICON(
"play_arrow", -0.12));
2180 connect(button, &QPushButton::clicked,
this, [
this]() {
2182 if (ws->player && ws->document()->timeline()) {
2183 std::vector<double> annotation_times;
2184 for (
auto &track : ws->document()->timeline()->tracks()) {
2185 if (
auto annotation_track =
2186 std::dynamic_pointer_cast<AnnotationTrack>(track)) {
2187 if (
auto branch = annotation_track->branch(ws(),
false)) {
2188 for (
auto &span : branch->spans()) {
2189 annotation_times.push_back(span->start());
2190 annotation_times.push_back(span->start() + span->duration());
2195 std::sort(annotation_times.begin(), annotation_times.end());
2196 annotation_times.erase(
2197 std::unique(annotation_times.begin(), annotation_times.end()),
2198 annotation_times.end());
2199 ws->player->play(annotation_times);
2202 playback_bar->addWidget(button);
2206 auto *button =
new FlatButton(MATERIAL_ICON(
"skip_next", -0.1));
2207 connect(button, &QPushButton::clicked,
this, [
this]() {
2210 double current_time = ws->player->time();
2211 double seek_time = std::numeric_limits<double>::max();
2212 for (size_t i = 0; i < trackCount(ws()); i++) {
2213 if (auto track = std::dynamic_pointer_cast<AnnotationTrack>(
2214 trackAt(ws(), i))) {
2215 if (auto branch = track->branch(ws(), false)) {
2216 for (auto &span : branch->spans()) {
2217 for (double span_time : {
2219 span->start() + span->duration(),
2221 if (span_time > current_time + 1e-6 &&
2222 std::abs(current_time - span_time) <
2223 std::abs(current_time - seek_time)) {
2224 seek_time = span_time;
2232 std::min(timelineDuration(ws()), std::max(0.0, seek_time)));
2235 playback_bar->addWidget(button);
2238 auto *playback_bar_right =
new QHBoxLayout(content_widget);
2239 playback_bar_right->setSpacing(0);
2240 playback_bar->addLayout(playback_bar_right, 4);
2242 playback_bar_right->addStretch(1);
2246 new FlatButton(style()->standardIcon(QStyle::SP_TitleBarCloseButton));
2247 connect(button, &QPushButton::clicked,
this, [
this]() {
2251 playback_bar->addWidget(button);
2252 always_active_widgets.insert(button);
2257 auto *time_bar =
new TimeBar();
2258 scene->addItem(time_bar);
2261 scale_item->setParentItem(time_bar);
2262 scene->addItem(scale_item);
2265 seek_head->setParentItem(time_bar);
2266 scene->addItem(seek_head);
2268 class GraphicsView :
public QGraphicsView {
2271 virtual void mouseMoveEvent(QMouseEvent *event)
override {
2272 QGraphicsView::mouseMoveEvent(event);
2274 virtual void mousePressEvent(QMouseEvent *event)
override {
2275 QGraphicsView::mousePressEvent(event);
2277 virtual void mouseReleaseEvent(QMouseEvent *event)
override {
2278 QGraphicsView::mouseReleaseEvent(event);
2280 virtual void scrollContentsBy(
int dx,
int dy)
override {
2281 QGraphicsView::scrollContentsBy(dx, dy);
2286 GraphicsView() { setMouseTracking(
true); }
2289 auto *view =
new GraphicsView();
2290 view->setParent(
this);
2291 view->setViewport(
new QOpenGLWidget(
this));
2292 view->setScene(scene);
2293 view->setCacheMode(QGraphicsView::CacheNone);
2294 view->setViewportUpdateMode(QGraphicsView::FullViewportUpdate);
2295 view->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOn);
2296 view->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOn);
2297 view->setAlignment(Qt::AlignLeft | Qt::AlignTop);
2298 view->setFocusPolicy(Qt::NoFocus);
2299 view->setTransformationAnchor(QGraphicsView::NoAnchor);
2301 main_layout->addWidget(view,
true);
2304 bool enabled =
false;
2307 enabled = (ws->player !=
nullptr);
2310 updateEnabled(playback_bar, [enabled, always_active_widgets](QWidget *w) {
2311 if (always_active_widgets.find(w) == always_active_widgets.end()) {
2312 w->setEnabled(enabled);
2317 ws->modified.connect(playback_bar, sync);
2319 setWidget(content_widget);
2322 TimelineWidget::~TimelineWidget() {}