TAMSVIZ
Visualization and annotation tool for ROS
workspace.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "workspace.h"
5 
6 #include "bagplayer.h"
7 #include "history.h"
8 #include "log.h"
9 #include "topic.h"
10 #include "transformer.h"
11 
12 #include <QTimer>
13 
14 std::shared_ptr<GlobalEvents> GlobalEvents::instance() {
15  static auto instance = std::make_shared<GlobalEvents>();
16  return instance;
17 }
18 
19 void Selection::operator=(const std::shared_ptr<Object> &o) {
20  _objects.clear();
21  if (o) {
22  _objects = {o->id()};
23  }
24 }
25 bool Selection::contains(const std::shared_ptr<Object> &o) const {
26  if (o) {
27  for (auto id : _objects) {
28  if (id == o->id()) {
29  return true;
30  }
31  }
32  }
33  return false;
34 }
35 std::vector<std::shared_ptr<Object>>
36 Selection::resolve(const std::shared_ptr<Object> &root) const {
37  std::vector<std::shared_ptr<Object>> ret;
38  root->recurse([&](const std::shared_ptr<Object> &o) {
39  for (auto id : _objects) {
40  if (o->id() == id) {
41  ret.push_back(o);
42  }
43  }
44  });
45  return ret;
46 }
47 void Selection::clear() { _objects.clear(); }
48 void Selection::add(const std::shared_ptr<Object> &object) {
49  if (object) {
50  for (auto &id : _objects) {
51  if (id == object->id()) {
52  return;
53  }
54  }
55  _objects.push_back(object->id());
56  }
57 }
58 bool Selection::empty() const { return _objects.empty(); }
59 size_t Selection::size() const { return _objects.size(); }
60 void Selection::toggle(const std::shared_ptr<Object> &object) {
61  if (object) {
62  if (contains(object)) {
63  erase(object);
64  } else {
65  add(object);
66  }
67  }
68 }
69 void Selection::erase(const std::shared_ptr<Object> &object) {
70  if (object) {
71  for (auto it = _objects.begin(); it != _objects.end();) {
72  if (*it == object->id()) {
73  it = _objects.erase(it);
74  } else {
75  it++;
76  }
77  }
78  }
79 }
80 bool operator==(const Selection &a, const Selection &b) {
81  if (a._objects.size() != b._objects.size()) {
82  return false;
83  }
84  for (size_t i = 0; i < a._objects.size(); i++) {
85  if (a._objects.at(i) != b._objects.at(i)) {
86  return false;
87  }
88  }
89 }
90 bool operator!=(const Selection &a, const Selection &b) { return !(a == b); }
91 
92 Workspace::Workspace() {
93 
94  history = std::make_shared<History<std::shared_ptr<Workspace>>>();
95 
96  if (0) {
97  modified.connect([this]() {
98  recurse([this](const std::shared_ptr<Object> &o) {
99  if (o && o.get() != this) {
100  object_ptr_test.insert(o);
101  }
102  });
103  });
104  }
105 
106  modified.connect([this]() { document()->display()->refreshRecursive(); });
107 }
108 Workspace::~Workspace() { object_ptr_test.clear(); }
109 std::vector<std::string> Workspace::listTopics(const std::string &type_name) {
110  if (player) {
111  return player->listTopics(type_name);
112  } else {
113  return TopicManager::instance()->listTopics(type_name);
114  }
115 }
116 std::vector<std::string>
117 Workspace::listTopics(const std::initializer_list<std::string> &type_names) {
118  std::set<std::string> ret;
119  for (auto &type_name : type_names) {
120  for (auto &v : listTopics(type_name)) {
121  ret.insert(v);
122  }
123  }
124  return std::vector<std::string>(ret.begin(), ret.end());
125 }
126 
127 ObjectScope::ObjectScope() { Property::unlockScope(+1); }
128 ObjectScope::~ObjectScope() { Property::unlockScope(-1); }
129 
130 LockScope::LockScope() {
131  if (!ws()->history->current) {
132  auto item = std::make_shared<History<std::shared_ptr<Workspace>>::Item>();
133  item->snapshot =
134  Snapshot<std::shared_ptr<Workspace>>::save(ws(), nullptr, nullptr);
135  ws()->history->current = item;
136  }
137 }
138 LockScope::~LockScope() {}
139 
140 std::shared_ptr<Workspace> &LockScope::workspace_instance() {
141  static std::shared_ptr<Workspace> instance = std::make_shared<Workspace>();
142  return instance;
143 }
144 static std::recursive_mutex g_mutex_instance;
145 std::recursive_mutex &LockScope::mutex_instance() { return g_mutex_instance; }
146 
147 static std::vector<ActionScope *> g_current_action_scopes;
148 ActionScope::ActionScope(const std::string &label,
149  const std::shared_ptr<const Object> &object,
150  bool aggregate)
151  : label(label), object(object), aggregate(aggregate) {
152  LOG_INFO("begin " << label);
153  ws()->history->locked = true;
154  g_current_action_scopes.push_back(this);
155  if (!ws()->history->current) {
156  auto item = std::make_shared<History<std::shared_ptr<Workspace>>::Item>();
157  item->snapshot =
158  Snapshot<std::shared_ptr<Workspace>>::save(ws(), nullptr, nullptr);
159  ws()->history->current = item;
160  }
161 }
162 ActionScope::~ActionScope() {
163  LOG_INFO("done " << label);
164  if (g_current_action_scopes.back() != this) {
165  throw std::runtime_error("overlapping actions");
166  }
167  g_current_action_scopes.pop_back();
168  {
169  std::shared_ptr<const Snapshot<std::shared_ptr<Workspace>>> new_snapshot =
171  ws(), ws()->history->current->snapshot,
172  [&](const std::shared_ptr<Object> &o) {
173  return object == nullptr || o == ws() || o == object;
174  });
175  if (new_snapshot != ws()->history->current->snapshot) {
176  auto item = std::make_shared<History<std::shared_ptr<Workspace>>::Item>();
177  item->label = label;
178  item->snapshot = new_snapshot;
179  if (aggregate && ws()->history->current->label == label &&
180  ws()->history->canUndo()) {
181  ws()->history->current = item;
182  } else {
183  ws()->history->undo_stack.push_back(ws()->history->current);
184  if (ws()->history->undo_stack.size() > ws()->history->limit) {
185  ws()->history->undo_stack.pop_front();
186  }
187  ws()->history->redo_stack.clear();
188  ws()->history->current = item;
189  }
190  }
191  }
192  ws()->history->locked = false;
193  QTimer::singleShot(0, []() {
194  LockScope ws;
195  ws->modified();
196  });
197 }