Tue, 28 Sep 2021 23:07:23 +0300
Use QSaveFile to save the file more safely
24 | 1 | /* |
2 | * LDForge: LDraw parts authoring CAD | |
3 | * Copyright (C) 2013 - 2020 Teemu Piippo | |
4 | * | |
5 | * This program is free software: you can redistribute it and/or modify | |
6 | * it under the terms of the GNU General Public License as published by | |
7 | * the Free Software Foundation, either version 3 of the License, or | |
8 | * (at your option) any later version. | |
9 | * | |
10 | * This program is distributed in the hope that it will be useful, | |
11 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
12 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
13 | * GNU General Public License for more details. | |
14 | * | |
15 | * You should have received a copy of the GNU General Public License | |
16 | * along with this program. If not, see <http://www.gnu.org/licenses/>. | |
17 | */ | |
18 | ||
3 | 19 | #pragma once |
8
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
20 | #include <QAbstractListModel> |
3 | 21 | #include <memory> |
22 | #include "main.h" | |
23 | #include "header.h" | |
14 | 24 | #include "linetypes/object.h" |
141 | 25 | #include "linetypes/metacommand.h" |
21 | 26 | #include "gl/common.h" |
3 | 27 | |
8
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
28 | enum class HeaderProperty |
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
29 | { |
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
30 | Name |
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
31 | }; |
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
32 | |
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
33 | class Model : public QAbstractListModel |
3 | 34 | { |
35 | Q_OBJECT | |
36 | public: | |
37 | class EditContext; | |
140 | 38 | Model(const QString& path, QObject* parent = nullptr); |
21 | 39 | Model(QObject* parent = nullptr); |
5 | 40 | Model(const Model&) = delete; |
3 | 41 | int size() const; |
133
e39326ee48dc
Begin work on edit history
Teemu Piippo <teemu@hecknology.net>
parents:
117
diff
changeset
|
42 | ldraw::id_t at(int index) const; |
3 | 43 | EditContext edit(); |
8
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
44 | int rowCount(const QModelIndex&) const override; |
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
45 | QVariant data(const QModelIndex& index, int role) const override; |
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
46 | QVariant getHeaderProperty(const HeaderProperty property); |
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
47 | const QString& getName() const; |
35
98906a94732f
renamed the linetypes namespace to ldraw namespace and added more structures to it
Teemu Piippo <teemu@hecknology.net>
parents:
26
diff
changeset
|
48 | QVariant getObjectProperty(const int index, const ldraw::Property property) const; |
21 | 49 | std::vector<gl::Polygon> getPolygons(class DocumentManager* documents) const; |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
50 | QModelIndex lookup(ldraw::id_t id) const; |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
51 | ldraw::id_t resolve(const QModelIndex& index) const; |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
52 | template<typename R> |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
53 | ldraw::Id<R> checkType(ldraw::id_t id) const; |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
54 | template<typename R> |
141 | 55 | bool isA(ldraw::id_t id) const; |
56 | template<typename R> | |
116 | 57 | const R* get(ldraw::Id<R> id) const; |
58 | template<typename R> | |
59 | struct Get2Result | |
60 | { | |
61 | QModelIndex index; | |
62 | const R* object; | |
63 | }; | |
64 | template<typename R> | |
65 | Get2Result<R> get2(ldraw::Id<R> id) const; | |
117 | 66 | template<typename R, typename Fn> |
67 | void apply(Fn f) const; | |
141 | 68 | const QString& path() const; |
69 | void setPath(const QString& path); | |
70 | bool save(QTextStream& errors) const; | |
71 | void makeUnofficial(); | |
112 | 72 | Q_SIGNALS: |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
73 | void objectAdded(ldraw::id_t id, int position); |
136 | 74 | void objectModified(ldraw::id_t id, int position); |
3 | 75 | private: |
35
98906a94732f
renamed the linetypes namespace to ldraw namespace and added more structures to it
Teemu Piippo <teemu@hecknology.net>
parents:
26
diff
changeset
|
76 | using ModelObjectPointer = std::unique_ptr<ldraw::Object>; |
3 | 77 | template<typename T, typename... Args> |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
78 | ldraw::Id<T> append(Args&&... args); |
8
44679e468ba9
major update with many things
Teemu Piippo <teemu@hecknology.net>
parents:
6
diff
changeset
|
79 | void append(ModelObjectPointer&& object); |
3 | 80 | template<typename T, typename... Args> |
76
7c4a63a02632
finished splitQuadrilateral theoretically (untested)
Teemu Piippo <teemu@hecknology.net>
parents:
73
diff
changeset
|
81 | ldraw::Id<T> insert(std::size_t position, Args&&... args); |
7c4a63a02632
finished splitQuadrilateral theoretically (untested)
Teemu Piippo <teemu@hecknology.net>
parents:
73
diff
changeset
|
82 | void remove(int position); |
51 | 83 | ldraw::Object* objectAt(const QModelIndex& index); |
84 | const ldraw::Object* objectAt(const QModelIndex& index) const; | |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
85 | template<typename T> |
138 | 86 | T* objectAt(ldraw::Id<T> id); |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
87 | template<typename T> |
138 | 88 | const T* objectAt(ldraw::Id<T> id) const; |
51 | 89 | void getObjectPolygons( |
90 | const int index, | |
91 | std::vector<gl::Polygon>& polygons_out, | |
92 | ldraw::GetPolygonsContext* context) const; | |
133
e39326ee48dc
Begin work on edit history
Teemu Piippo <teemu@hecknology.net>
parents:
117
diff
changeset
|
93 | void editFinished(); |
e39326ee48dc
Begin work on edit history
Teemu Piippo <teemu@hecknology.net>
parents:
117
diff
changeset
|
94 | void objectModified(ldraw::id_t id); |
3 | 95 | bool modified = false; |
141 | 96 | QString storedPath; |
3 | 97 | LDHeader header; |
98 | std::vector<ModelObjectPointer> body; | |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
99 | std::map<ldraw::id_t, ldraw::Object*> objectsById; |
21 | 100 | mutable std::vector<gl::Polygon> cachedPolygons; |
101 | mutable bool needRecache = true; | |
136 | 102 | /** |
103 | * @brief Amount of model edit contexts active | |
104 | */ | |
133
e39326ee48dc
Begin work on edit history
Teemu Piippo <teemu@hecknology.net>
parents:
117
diff
changeset
|
105 | int editCounter = 0; |
3 | 106 | }; |
107 | ||
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
108 | /** |
141 | 109 | * @brief Checks whether the id is exactly of the specified type |
110 | * @tparam R Type of LDraw line type object to test for | |
111 | * @param id Id of object to test | |
112 | * @returns whether the type of the object specified by @c id is the same type as R. Returns false if it is a subclass. | |
113 | */ | |
114 | template<typename R> | |
115 | bool Model::isA(ldraw::id_t id) const | |
116 | { | |
117 | const ldraw::Object* object = this->objectAt(this->lookup(id)); | |
118 | const std::type_info& a = typeid(*object); | |
119 | const std::type_info& b = typeid(R); | |
120 | return a == b; | |
121 | } | |
122 | ||
123 | /** | |
138 | 124 | * @brief Calls the specified function to all matching objects in the model |
125 | * @tparam R Type of LDraw line type object to filter by | |
126 | * @param fn Function to call. | |
117 | 127 | */ |
128 | template<typename R, typename Fn> | |
129 | void Model::apply(Fn f) const | |
130 | { | |
131 | for (const ModelObjectPointer& object : this->body) | |
132 | { | |
133 | const R* subobject = dynamic_cast<const R*>(object.get()); | |
134 | if (subobject != nullptr) | |
135 | { | |
136 | f(subobject); | |
137 | } | |
138 | } | |
139 | } | |
140 | ||
141 | /** | |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
142 | * \brief Checks type of object behind id |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
143 | * Checks whether the specified id refers to an object of the specified type. |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
144 | * \returns id casted to subclass if appropriate, null id otherwise |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
145 | */ |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
146 | template<typename R> |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
147 | ldraw::Id<R> Model::checkType(ldraw::id_t id) const |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
148 | { |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
149 | if (dynamic_cast<const R*>(this->objectAt(this->lookup(id))) != nullptr) |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
150 | { |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
151 | return ldraw::Id<R>{id.value}; |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
152 | } |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
153 | else |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
154 | { |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
155 | return ldraw::NULL_ID; |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
156 | } |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
157 | } |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
158 | |
3 | 159 | template<typename T, typename... Args> |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
160 | ldraw::Id<T> Model::append(Args&&... args) |
3 | 161 | { |
111
1f42c03fafca
Draw tool actually adds objects now
Teemu Piippo <teemu@hecknology.net>
parents:
81
diff
changeset
|
162 | const int position = static_cast<int>(this->body.size()); |
112 | 163 | Q_EMIT beginInsertRows({}, position, position); |
3 | 164 | this->body.push_back(std::make_unique<T>(args...)); |
35
98906a94732f
renamed the linetypes namespace to ldraw namespace and added more structures to it
Teemu Piippo <teemu@hecknology.net>
parents:
26
diff
changeset
|
165 | ldraw::Object* pointer = this->body.back().get(); |
6 | 166 | this->objectsById[pointer->id] = pointer; |
112 | 167 | Q_EMIT objectAdded(pointer->id, static_cast<int>(this->body.size() - 1)); |
168 | Q_EMIT endInsertRows(); | |
111
1f42c03fafca
Draw tool actually adds objects now
Teemu Piippo <teemu@hecknology.net>
parents:
81
diff
changeset
|
169 | this->needRecache = true; |
76
7c4a63a02632
finished splitQuadrilateral theoretically (untested)
Teemu Piippo <teemu@hecknology.net>
parents:
73
diff
changeset
|
170 | return ldraw::Id<T>{pointer->id.value}; |
3 | 171 | } |
172 | ||
173 | template<typename T, typename... Args> | |
111
1f42c03fafca
Draw tool actually adds objects now
Teemu Piippo <teemu@hecknology.net>
parents:
81
diff
changeset
|
174 | ldraw::Id<T> Model::insert(const std::size_t position, Args&&... args) |
3 | 175 | { |
112 | 176 | Q_EMIT beginInsertRows({}, position, position); |
76
7c4a63a02632
finished splitQuadrilateral theoretically (untested)
Teemu Piippo <teemu@hecknology.net>
parents:
73
diff
changeset
|
177 | this->body.insert(std::begin(this->body) + position, std::make_unique<T>(args...)); |
35
98906a94732f
renamed the linetypes namespace to ldraw namespace and added more structures to it
Teemu Piippo <teemu@hecknology.net>
parents:
26
diff
changeset
|
178 | ldraw::Object* pointer = this->body[position].get(); |
6 | 179 | this->objectsById[pointer->id] = pointer; |
112 | 180 | Q_EMIT objectAdded(pointer->id, static_cast<int>(position)); |
181 | Q_EMIT endInsertRows(); | |
111
1f42c03fafca
Draw tool actually adds objects now
Teemu Piippo <teemu@hecknology.net>
parents:
81
diff
changeset
|
182 | this->needRecache = true; |
76
7c4a63a02632
finished splitQuadrilateral theoretically (untested)
Teemu Piippo <teemu@hecknology.net>
parents:
73
diff
changeset
|
183 | return ldraw::Id<T>{pointer->id.value}; |
3 | 184 | } |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
185 | |
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
186 | template<typename R> |
116 | 187 | const R* Model::get(ldraw::Id<R> id) const |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
188 | { |
116 | 189 | return this->get2(id).object; |
190 | } | |
191 | ||
192 | template<typename R> | |
193 | Model::Get2Result<R> Model::get2(const ldraw::Id<R> id) const | |
194 | { | |
195 | Get2Result<R> result; | |
196 | result.index = this->lookup(id); | |
197 | if (result.index.isValid()) | |
76
7c4a63a02632
finished splitQuadrilateral theoretically (untested)
Teemu Piippo <teemu@hecknology.net>
parents:
73
diff
changeset
|
198 | { |
116 | 199 | result.object = static_cast<const R*>(this->objectAt(result.index)); |
81
62373840e33a
object editor widgets start to form up
Teemu Piippo <teemu@hecknology.net>
parents:
76
diff
changeset
|
200 | } |
62373840e33a
object editor widgets start to form up
Teemu Piippo <teemu@hecknology.net>
parents:
76
diff
changeset
|
201 | else |
62373840e33a
object editor widgets start to form up
Teemu Piippo <teemu@hecknology.net>
parents:
76
diff
changeset
|
202 | { |
116 | 203 | result.object = nullptr; |
81
62373840e33a
object editor widgets start to form up
Teemu Piippo <teemu@hecknology.net>
parents:
76
diff
changeset
|
204 | } |
116 | 205 | return result; |
73
97df974b5ed5
ldraw::Id is now templated for extra type safety
Teemu Piippo <teemu@hecknology.net>
parents:
51
diff
changeset
|
206 | } |
138 | 207 | |
208 | /** | |
209 | * @brief Gets an object pointer by id. Used by the editing context to actually modify objects. | |
210 | * @param id | |
211 | * @return object pointer | |
212 | */ | |
213 | template<typename T> | |
214 | T* Model::objectAt(ldraw::Id<T> id) | |
215 | { | |
216 | return static_cast<T*>(this->objectAt(this->lookup(id))); | |
217 | } | |
218 | ||
219 | template<typename T> | |
220 | const T* Model::objectAt(ldraw::Id<T> id) const | |
221 | { | |
222 | return static_cast<const T*>(this->objectAt(this->lookup(id))); | |
223 | } |