1 |
/* |
2 |
* |
3 |
Copyright (C) 2017 Fábio Bento (random-guy) |
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 |
*/ |
19 |
|
20 |
#include "projectfilevago.h" |
21 |
|
22 |
const QString ProjectFileVago::XMLTableName = "XML"; |
23 |
const QString ProjectFileVago::TexturesTableName = "Textures"; |
24 |
const QString ProjectFileVago::CharactersTableName = "Characters"; |
25 |
const QString ProjectFileVago::ObjectsTableName = "Objects"; |
26 |
const QString ProjectFileVago::LevelsTableName = "Levels"; |
27 |
const QString ProjectFileVago::MiscTableName = "Misc"; |
28 |
|
29 |
ProjectFileVago::ProjectData ProjectFileVago::readProjectDataFromFile(const QString &fileFullPath){ |
30 |
|
31 |
ProjectFileVago::ProjectData currentProjectData; |
32 |
|
33 |
upgradeProjectFileIfNecessary(fileFullPath); |
34 |
|
35 |
pugi::xml_document doc; |
36 |
|
37 |
pugi::xml_parse_result result = doc.load_file(QSTR_TO_CSTR(fileFullPath)); |
38 |
|
39 |
if(result.status!=pugi::status_ok){ |
40 |
throw std::runtime_error(QSTR_TO_CSTR(QString("An error ocurred while loading project file.\n") + result.description())); |
41 |
} |
42 |
|
43 |
if(QString(doc.root().first_child().name()) != "VagoProject"){ |
44 |
throw std::runtime_error(QSTR_TO_CSTR(QString(doc.root().name()) + "The file opened is not a valid Vago project file. Load aborted.")); |
45 |
} |
46 |
|
47 |
QString projVagoVersion; |
48 |
|
49 |
try{ |
50 |
projVagoVersion = QString(doc.select_node("/VagoProject/@vagoVersion").attribute().value()); |
51 |
} |
52 |
catch (const pugi::xpath_exception& e) |
53 |
{ |
54 |
throw std::runtime_error(QSTR_TO_CSTR(QString("Couldn't find the vagoVersion of the current project. Load aborted.\n") + e.what())); |
55 |
} |
56 |
|
57 |
if(!projVagoVersion.startsWith(GlobalVars::LastCompatibleVersion)){ |
58 |
throw std::runtime_error("The project that you are trying to load seems it is not compatible with your Vago Version. Please update Vago and try again."); |
59 |
} |
60 |
|
61 |
// After the initial validations begin loading the project data |
62 |
|
63 |
auto fFetchFromToForTab = [](pugi::xml_document &doc, const QString &tableName, ProjectTable &tableToInjectData) -> void{ |
64 |
tableToInjectData.from = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+tableName+"/@from")).attribute().value(); |
65 |
tableToInjectData.to = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+tableName+"/@to")).attribute().value(); |
66 |
}; |
67 |
|
68 |
auto fFetchTabTableRows = [](pugi::xml_document &doc, const QString &tableName) -> QVector<ProjectTableRow>{ |
69 |
|
70 |
QVector<ProjectTableRow> rows; |
71 |
|
72 |
for(const pugi::xpath_node &xPathNode : doc.select_nodes(QSTR_TO_CSTR("/VagoProject/"+tableName+"/Row"))){ |
73 |
|
74 |
ProjectTableRow currentRow; |
75 |
|
76 |
pugi::xml_node currNode = xPathNode.node(); |
77 |
|
78 |
currentRow.fileFolder = currNode.attribute("fileFolder").value(); |
79 |
currentRow.fromTo = currNode.attribute("fromTo").value(); |
80 |
currentRow.command = currNode.attribute("command").value(); |
81 |
|
82 |
pugi::xml_attribute disabledAttr = currNode.attribute("disabled"); |
83 |
currentRow.isDisabled = disabledAttr.empty() ? false : disabledAttr.as_bool(); |
84 |
|
85 |
rows.append(currentRow); |
86 |
} |
87 |
|
88 |
return rows; |
89 |
}; |
90 |
|
91 |
QString currentTableName = XMLTableName; |
92 |
|
93 |
// XML tab |
94 |
|
95 |
fFetchFromToForTab(doc, currentTableName, currentProjectData.xmlTable); |
96 |
currentProjectData.xmlTable.rows = fFetchTabTableRows(doc, currentTableName); |
97 |
|
98 |
// Textures tab |
99 |
|
100 |
currentTableName = TexturesTableName; |
101 |
|
102 |
fFetchFromToForTab(doc, currentTableName, currentProjectData.texturesTable); |
103 |
|
104 |
currentProjectData.texturesTable.rbTexturesType = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@type")).attribute().value(); |
105 |
currentProjectData.texturesTable.cbGenMipMaps = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@genMipMaps")).attribute().as_bool(); |
106 |
currentProjectData.texturesTable.cbNoUwrap = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@noUwrap")).attribute().as_bool(); |
107 |
currentProjectData.texturesTable.cbNoUwrap = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@noVwrap")).attribute().as_bool(); |
108 |
currentProjectData.texturesTable.cbLarge = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@large")).attribute().as_bool(); |
109 |
currentProjectData.texturesTable.cbEnvMap = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@envMap")).attribute().as_bool(); |
110 |
currentProjectData.texturesTable.leEnvMapTexture = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@envMapValue")).attribute().value(); |
111 |
|
112 |
currentProjectData.texturesTable.rows = fFetchTabTableRows(doc, currentTableName); |
113 |
|
114 |
// Characters tab |
115 |
|
116 |
currentTableName = CharactersTableName; |
117 |
|
118 |
fFetchFromToForTab(doc, currentTableName, currentProjectData.charactersTable); |
119 |
|
120 |
currentProjectData.charactersTable.cbCellShading = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@cellShading")).attribute().as_bool(); |
121 |
currentProjectData.charactersTable.cbNormals = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@normals")).attribute().as_bool(); |
122 |
currentProjectData.charactersTable.cbStandingPose = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@standingPose")).attribute().as_bool(); |
123 |
currentProjectData.charactersTable.cbWithTRBS_ONCC = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@extractTRBSONCC")).attribute().as_bool(); |
124 |
currentProjectData.charactersTable.leTRBS_ONCC = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@extractTRBSONCCValue")).attribute().value(); |
125 |
|
126 |
currentProjectData.charactersTable.rows = fFetchTabTableRows(doc, currentTableName); |
127 |
|
128 |
// Objects tab |
129 |
|
130 |
currentTableName = ObjectsTableName; |
131 |
|
132 |
fFetchFromToForTab(doc, currentTableName, currentProjectData.objectsTable); |
133 |
|
134 |
currentProjectData.objectsTable.cbTexture = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@texture")).attribute().as_bool(); |
135 |
currentProjectData.objectsTable.leTextureName = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@textureValue")).attribute().value(); |
136 |
currentProjectData.objectsTable.cbWithAnimation = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@withAnimation")).attribute().as_bool(); |
137 |
currentProjectData.objectsTable.leAnimationName = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@withAnimationValue")).attribute().value(); |
138 |
|
139 |
currentProjectData.objectsTable.rows = fFetchTabTableRows(doc, currentTableName); |
140 |
|
141 |
// Levels tab |
142 |
|
143 |
currentTableName = LevelsTableName; |
144 |
|
145 |
fFetchFromToForTab(doc, currentTableName, currentProjectData.levelsTable); |
146 |
|
147 |
currentProjectData.levelsTable.cbSpecificFilesLevels = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@extractWithFiles")).attribute().as_bool(); |
148 |
currentProjectData.levelsTable.leSpecificFilesLevels = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@extractWithFilesValue")).attribute().value(); |
149 |
currentProjectData.levelsTable.cbDatLevels = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@datFilename")).attribute().as_bool(); |
150 |
currentProjectData.levelsTable.leTargetDatLevels = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@datFilenameValue")).attribute().value(); |
151 |
currentProjectData.levelsTable.cbBnvLevels = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@bnvSource")).attribute().as_bool(); |
152 |
currentProjectData.levelsTable.leBnvLevels = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@bnvSourceValue")).attribute().value(); |
153 |
currentProjectData.levelsTable.cbGridsLevels = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@generateGrids")).attribute().as_bool(); |
154 |
currentProjectData.levelsTable.cbAdditionalSourcesLevels = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@additionalSources")).attribute().as_bool(); |
155 |
currentProjectData.levelsTable.leAdditSourcesLevels = doc.select_node(QSTR_TO_CSTR("/VagoProject/"+currentTableName+"/Options/@additionalSourcesValue")).attribute().value(); |
156 |
|
157 |
currentProjectData.levelsTable.rows = fFetchTabTableRows(doc, currentTableName); |
158 |
|
159 |
// Misc tab |
160 |
|
161 |
currentTableName = MiscTableName; |
162 |
|
163 |
fFetchFromToForTab(doc, currentTableName, currentProjectData.miscTable); |
164 |
|
165 |
currentProjectData.miscTable.rows = fFetchTabTableRows(doc, currentTableName); |
166 |
|
167 |
return currentProjectData; |
168 |
} |
169 |
|
170 |
void ProjectFileVago::upgradeProjectFileIfNecessary(const QString &filePath){ |
171 |
|
172 |
pugi::xml_document doc; |
173 |
|
174 |
pugi::xml_parse_result result = doc.load_file(QSTR_TO_CSTR(filePath)); |
175 |
|
176 |
if(result.status!=pugi::status_ok){ |
177 |
throw std::runtime_error(QSTR_TO_CSTR(QString("An error ocurred while loading project file.\n") + result.description())); |
178 |
} |
179 |
|
180 |
QString projectVersion = QString(doc.select_single_node("/VagoProject").node().attribute("vagoVersion").as_string()); |
181 |
|
182 |
// 1.4 added standing pose in characters tab (we need to add it in 1.0 projects) |
183 |
if(projectVersion == "1.0"){ |
184 |
|
185 |
if(!Util::FileSystem::backupFile(filePath, filePath + UtilVago::getDateTimeFormatForFilename(QDateTime::currentDateTime()))){ |
186 |
QString errorMessage = "Couldn't backup the existing project file for version upgrade, program can't proceed."; |
187 |
Util::Dialogs::showError(errorMessage); |
188 |
LOG_FATAL << errorMessage; |
189 |
exit(1); |
190 |
} |
191 |
|
192 |
// Update version |
193 |
doc.select_single_node("/VagoProject").node().attribute("vagoVersion").set_value(QSTR_TO_CSTR(GlobalVars::LastCompatibleVersion)); |
194 |
|
195 |
// Add standing pose to characters options |
196 |
doc.select_node("/VagoProject/Characters/Options").node().append_attribute("standingPose").set_value("false"); |
197 |
|
198 |
if(!doc.save_file(QSTR_TO_CSTR(filePath), PUGIXML_TEXT("\t"), pugi::format_default | pugi::format_write_bom, pugi::xml_encoding::encoding_utf8)){ |
199 |
throw std::runtime_error(QSTR_TO_CSTR("Error while saving: '" + filePath + "'. After file version upgrade.")); |
200 |
} |
201 |
} |
202 |
else if(projectVersion != GlobalVars::LastCompatibleVersion){ |
203 |
throw std::runtime_error("Can't load the project file, it is from an incompatible version. Probably newer?"); |
204 |
} |
205 |
} |
206 |
|
207 |
// Right now it always replaces totally the old file with new content (old content is not modified but replaced) |
208 |
void ProjectFileVago::saveProjectDataToFile(const QString &fileFullPath, const ProjectFileVago::ProjectData &newProjectData){ |
209 |
pugi::xml_document doc; |
210 |
pugi::xml_node rootNode; |
211 |
|
212 |
rootNode = doc.append_child("VagoProject"); // create |
213 |
|
214 |
rootNode.append_attribute("vagoVersion").set_value(QSTR_TO_CSTR(GlobalVars::LastCompatibleVersion)); |
215 |
|
216 |
// Let's starting writting the GUI data |
217 |
|
218 |
QString currentTableName; |
219 |
|
220 |
// Returns node of the table |
221 |
auto fWriteTabGenericData = [](pugi::xml_node &rootNode, const ProjectTable &table, const QString &tableName) -> pugi::xml_node{ |
222 |
|
223 |
pugi::xml_node rowsNode = rootNode.append_child(QSTR_TO_CSTR(tableName)); |
224 |
|
225 |
rowsNode.append_attribute("from").set_value(QSTR_TO_CSTR(table.from)); |
226 |
rowsNode.append_attribute("to").set_value(QSTR_TO_CSTR(table.to)); |
227 |
|
228 |
for(const ProjectTableRow &currRow : table.rows){ |
229 |
|
230 |
pugi::xml_node currentRow = rowsNode.append_child("Row"); |
231 |
|
232 |
currentRow.append_attribute("fileFolder").set_value(QSTR_TO_CSTR(currRow.fileFolder)); |
233 |
currentRow.append_attribute("fromTo").set_value(QSTR_TO_CSTR(currRow.fromTo)); |
234 |
currentRow.append_attribute("command").set_value(QSTR_TO_CSTR(currRow.command)); |
235 |
|
236 |
// Only necessary to add if it is infact disabled |
237 |
if(currRow.isDisabled){ |
238 |
currentRow.append_attribute("disabled").set_value(currRow.isDisabled); |
239 |
} |
240 |
} |
241 |
|
242 |
return rowsNode; |
243 |
}; |
244 |
|
245 |
// XML tab |
246 |
|
247 |
currentTableName = XMLTableName; |
248 |
|
249 |
fWriteTabGenericData(rootNode, newProjectData.xmlTable, currentTableName); |
250 |
|
251 |
// Textures tab |
252 |
|
253 |
currentTableName = TexturesTableName; |
254 |
|
255 |
pugi::xml_node currentNodeTable = fWriteTabGenericData(rootNode, newProjectData.texturesTable, currentTableName); |
256 |
|
257 |
pugi::xml_node options = currentNodeTable.append_child("Options"); |
258 |
options.append_attribute("type").set_value(QSTR_TO_CSTR(newProjectData.texturesTable.rbTexturesType)); |
259 |
options.append_attribute("genMipMaps").set_value(Util::String::boolToCstr(newProjectData.texturesTable.cbGenMipMaps)); |
260 |
options.append_attribute("noUwrap").set_value(Util::String::boolToCstr(newProjectData.texturesTable.cbNoUwrap)); |
261 |
options.append_attribute("noVwrap").set_value(Util::String::boolToCstr(newProjectData.texturesTable.cbNoVwrap)); |
262 |
options.append_attribute("large").set_value(Util::String::boolToCstr(newProjectData.texturesTable.cbLarge)); |
263 |
options.append_attribute("envMap").set_value(Util::String::boolToCstr(newProjectData.texturesTable.cbEnvMap)); |
264 |
options.append_attribute("envMapValue").set_value(QSTR_TO_CSTR(newProjectData.texturesTable.leEnvMapTexture)); |
265 |
|
266 |
// Characters tab |
267 |
|
268 |
currentTableName = CharactersTableName; |
269 |
|
270 |
currentNodeTable = fWriteTabGenericData(rootNode, newProjectData.charactersTable, currentTableName); |
271 |
|
272 |
options = currentNodeTable.append_child("Options"); |
273 |
options.append_attribute("cellShading").set_value(Util::String::boolToCstr(newProjectData.charactersTable.cbCellShading)); |
274 |
options.append_attribute("normals").set_value(Util::String::boolToCstr(newProjectData.charactersTable.cbNormals)); |
275 |
options.append_attribute("standingPose").set_value(Util::String::boolToCstr(newProjectData.charactersTable.cbStandingPose)); |
276 |
options.append_attribute("extractTRBSONCC").set_value(Util::String::boolToCstr(newProjectData.charactersTable.cbWithTRBS_ONCC)); |
277 |
options.append_attribute("extractTRBSONCCValue").set_value(QSTR_TO_CSTR(newProjectData.charactersTable.leTRBS_ONCC)); |
278 |
|
279 |
// Objects tab |
280 |
|
281 |
currentTableName = ObjectsTableName; |
282 |
|
283 |
currentNodeTable = fWriteTabGenericData(rootNode, newProjectData.objectsTable, currentTableName); |
284 |
|
285 |
options = currentNodeTable.append_child("Options"); |
286 |
options.append_attribute("texture").set_value(Util::String::boolToCstr(newProjectData.objectsTable.cbTexture)); |
287 |
options.append_attribute("textureValue").set_value(QSTR_TO_CSTR(newProjectData.objectsTable.leTextureName)); |
288 |
options.append_attribute("withAnimation").set_value(Util::String::boolToCstr(newProjectData.objectsTable.cbWithAnimation)); |
289 |
options.append_attribute("withAnimationValue").set_value(QSTR_TO_CSTR(newProjectData.objectsTable.leAnimationName)); |
290 |
|
291 |
// Levels tab |
292 |
|
293 |
currentTableName = ObjectsTableName; |
294 |
|
295 |
currentNodeTable = fWriteTabGenericData(rootNode, newProjectData.levelsTable, currentTableName); |
296 |
|
297 |
options = currentNodeTable.append_child("Options"); |
298 |
options.append_attribute("extractWithFiles").set_value(Util::String::boolToCstr(newProjectData.levelsTable.cbSpecificFilesLevels)); |
299 |
options.append_attribute("extractWithFilesValue").set_value(QSTR_TO_CSTR(newProjectData.levelsTable.leSpecificFilesLevels)); |
300 |
options.append_attribute("datFilename").set_value(Util::String::boolToCstr(newProjectData.levelsTable.cbDatLevels)); |
301 |
options.append_attribute("datFilenameValue").set_value(QSTR_TO_CSTR(newProjectData.levelsTable.leTargetDatLevels)); |
302 |
options.append_attribute("bnvSource").set_value(Util::String::boolToCstr(newProjectData.levelsTable.cbBnvLevels)); |
303 |
options.append_attribute("bnvSourceValue").set_value(QSTR_TO_CSTR(newProjectData.levelsTable.leBnvLevels)); |
304 |
options.append_attribute("generateGrids").set_value(Util::String::boolToCstr(newProjectData.levelsTable.cbGridsLevels)); |
305 |
options.append_attribute("additionalSources").set_value(Util::String::boolToCstr(newProjectData.levelsTable.cbAdditionalSourcesLevels)); |
306 |
options.append_attribute("additionalSourcesValue").set_value(QSTR_TO_CSTR(newProjectData.levelsTable.leAdditSourcesLevels)); |
307 |
|
308 |
// Misc tab |
309 |
|
310 |
currentTableName = MiscTableName; |
311 |
|
312 |
fWriteTabGenericData(rootNode, newProjectData.miscTable, currentTableName); |
313 |
|
314 |
if(!doc.save_file(fileFullPath.toUtf8().constData(), PUGIXML_TEXT("\t"), pugi::format_default | pugi::format_write_bom, pugi::xml_encoding::encoding_utf8)){ |
315 |
throw std::runtime_error("An error ocurred while trying to save the project file. Please try another path."); |
316 |
} |
317 |
|
318 |
} |