10 #include "transformer.h" 14 std::shared_ptr<GlobalEvents> GlobalEvents::instance() {
15 static auto instance = std::make_shared<GlobalEvents>();
19 void Selection::operator=(
const std::shared_ptr<Object> &o) {
25 bool Selection::contains(
const std::shared_ptr<Object> &o)
const {
27 for (
auto id : _objects) {
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) {
47 void Selection::clear() { _objects.clear(); }
48 void Selection::add(
const std::shared_ptr<Object> &
object) {
50 for (
auto &
id : _objects) {
51 if (
id == object->id()) {
55 _objects.push_back(object->id());
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) {
62 if (contains(
object)) {
69 void Selection::erase(
const std::shared_ptr<Object> &
object) {
71 for (
auto it = _objects.begin(); it != _objects.end();) {
72 if (*it == object->id()) {
73 it = _objects.erase(it);
81 if (a._objects.size() != b._objects.size()) {
84 for (
size_t i = 0; i < a._objects.size(); i++) {
85 if (a._objects.at(i) != b._objects.at(i)) {
92 Workspace::Workspace() {
94 history = std::make_shared<History<std::shared_ptr<Workspace>>>();
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);
106 modified.connect([
this]() { document()->display()->refreshRecursive(); });
108 Workspace::~Workspace() { object_ptr_test.clear(); }
109 std::vector<std::string> Workspace::listTopics(
const std::string &type_name) {
111 return player->listTopics(type_name);
113 return TopicManager::instance()->listTopics(type_name);
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)) {
124 return std::vector<std::string>(ret.begin(), ret.end());
127 ObjectScope::ObjectScope() { Property::unlockScope(+1); }
128 ObjectScope::~ObjectScope() { Property::unlockScope(-1); }
130 LockScope::LockScope() {
131 if (!ws()->history->current) {
132 auto item = std::make_shared<History<std::shared_ptr<Workspace>>::Item>();
135 ws()->history->current = item;
138 LockScope::~LockScope() {}
140 std::shared_ptr<Workspace> &LockScope::workspace_instance() {
141 static std::shared_ptr<Workspace> instance = std::make_shared<Workspace>();
144 static std::recursive_mutex g_mutex_instance;
145 std::recursive_mutex &LockScope::mutex_instance() {
return g_mutex_instance; }
147 static std::vector<ActionScope *> g_current_action_scopes;
148 ActionScope::ActionScope(
const std::string &label,
149 const std::shared_ptr<const Object> &
object,
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>();
159 ws()->history->current = item;
162 ActionScope::~ActionScope() {
163 LOG_INFO(
"done " << label);
164 if (g_current_action_scopes.back() !=
this) {
165 throw std::runtime_error(
"overlapping actions");
167 g_current_action_scopes.pop_back();
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;
175 if (new_snapshot != ws()->history->current->snapshot) {
176 auto item = std::make_shared<History<std::shared_ptr<Workspace>>::Item>();
178 item->snapshot = new_snapshot;
179 if (aggregate && ws()->history->current->label == label &&
180 ws()->history->canUndo()) {
181 ws()->history->current = item;
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();
187 ws()->history->redo_stack.clear();
188 ws()->history->current = item;
192 ws()->history->locked =
false;
193 QTimer::singleShot(0, []() {