| 1 | /**************************************************************************/ |
| 2 | /* editor_export.cpp */ |
| 3 | /**************************************************************************/ |
| 4 | /* This file is part of: */ |
| 5 | /* GODOT ENGINE */ |
| 6 | /* https://godotengine.org */ |
| 7 | /**************************************************************************/ |
| 8 | /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ |
| 9 | /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */ |
| 10 | /* */ |
| 11 | /* Permission is hereby granted, free of charge, to any person obtaining */ |
| 12 | /* a copy of this software and associated documentation files (the */ |
| 13 | /* "Software"), to deal in the Software without restriction, including */ |
| 14 | /* without limitation the rights to use, copy, modify, merge, publish, */ |
| 15 | /* distribute, sublicense, and/or sell copies of the Software, and to */ |
| 16 | /* permit persons to whom the Software is furnished to do so, subject to */ |
| 17 | /* the following conditions: */ |
| 18 | /* */ |
| 19 | /* The above copyright notice and this permission notice shall be */ |
| 20 | /* included in all copies or substantial portions of the Software. */ |
| 21 | /* */ |
| 22 | /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ |
| 23 | /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ |
| 24 | /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */ |
| 25 | /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ |
| 26 | /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ |
| 27 | /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ |
| 28 | /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ |
| 29 | /**************************************************************************/ |
| 30 | |
| 31 | #include "editor_export.h" |
| 32 | |
| 33 | #include "core/config/project_settings.h" |
| 34 | #include "core/io/config_file.h" |
| 35 | |
| 36 | EditorExport *EditorExport::singleton = nullptr; |
| 37 | |
| 38 | void EditorExport::_save() { |
| 39 | Ref<ConfigFile> config; |
| 40 | Ref<ConfigFile> credentials; |
| 41 | config.instantiate(); |
| 42 | credentials.instantiate(); |
| 43 | for (int i = 0; i < export_presets.size(); i++) { |
| 44 | Ref<EditorExportPreset> preset = export_presets[i]; |
| 45 | String section = "preset." + itos(i); |
| 46 | |
| 47 | config->set_value(section, "name" , preset->get_name()); |
| 48 | config->set_value(section, "platform" , preset->get_platform()->get_name()); |
| 49 | config->set_value(section, "runnable" , preset->is_runnable()); |
| 50 | config->set_value(section, "dedicated_server" , preset->is_dedicated_server()); |
| 51 | config->set_value(section, "custom_features" , preset->get_custom_features()); |
| 52 | |
| 53 | bool save_files = false; |
| 54 | switch (preset->get_export_filter()) { |
| 55 | case EditorExportPreset::EXPORT_ALL_RESOURCES: { |
| 56 | config->set_value(section, "export_filter" , "all_resources" ); |
| 57 | } break; |
| 58 | case EditorExportPreset::EXPORT_SELECTED_SCENES: { |
| 59 | config->set_value(section, "export_filter" , "scenes" ); |
| 60 | save_files = true; |
| 61 | } break; |
| 62 | case EditorExportPreset::EXPORT_SELECTED_RESOURCES: { |
| 63 | config->set_value(section, "export_filter" , "resources" ); |
| 64 | save_files = true; |
| 65 | } break; |
| 66 | case EditorExportPreset::EXCLUDE_SELECTED_RESOURCES: { |
| 67 | config->set_value(section, "export_filter" , "exclude" ); |
| 68 | save_files = true; |
| 69 | } break; |
| 70 | case EditorExportPreset::EXPORT_CUSTOMIZED: { |
| 71 | config->set_value(section, "export_filter" , "customized" ); |
| 72 | config->set_value(section, "customized_files" , preset->get_customized_files()); |
| 73 | save_files = false; |
| 74 | }; |
| 75 | } |
| 76 | |
| 77 | if (save_files) { |
| 78 | Vector<String> export_files = preset->get_files_to_export(); |
| 79 | config->set_value(section, "export_files" , export_files); |
| 80 | } |
| 81 | config->set_value(section, "include_filter" , preset->get_include_filter()); |
| 82 | config->set_value(section, "exclude_filter" , preset->get_exclude_filter()); |
| 83 | config->set_value(section, "export_path" , preset->get_export_path()); |
| 84 | config->set_value(section, "encryption_include_filters" , preset->get_enc_in_filter()); |
| 85 | config->set_value(section, "encryption_exclude_filters" , preset->get_enc_ex_filter()); |
| 86 | config->set_value(section, "encrypt_pck" , preset->get_enc_pck()); |
| 87 | config->set_value(section, "encrypt_directory" , preset->get_enc_directory()); |
| 88 | credentials->set_value(section, "script_encryption_key" , preset->get_script_encryption_key()); |
| 89 | |
| 90 | String option_section = "preset." + itos(i) + ".options" ; |
| 91 | |
| 92 | for (const KeyValue<StringName, Variant> &E : preset->values) { |
| 93 | PropertyInfo *prop = preset->properties.getptr(E.key); |
| 94 | if (prop && prop->usage & PROPERTY_USAGE_SECRET) { |
| 95 | credentials->set_value(option_section, E.key, E.value); |
| 96 | } else { |
| 97 | config->set_value(option_section, E.key, E.value); |
| 98 | } |
| 99 | } |
| 100 | } |
| 101 | |
| 102 | config->save("res://export_presets.cfg" ); |
| 103 | credentials->save("res://.godot/export_credentials.cfg" ); |
| 104 | } |
| 105 | |
| 106 | void EditorExport::save_presets() { |
| 107 | if (block_save) { |
| 108 | return; |
| 109 | } |
| 110 | save_timer->start(); |
| 111 | } |
| 112 | |
| 113 | void EditorExport::_bind_methods() { |
| 114 | ADD_SIGNAL(MethodInfo("export_presets_updated" )); |
| 115 | } |
| 116 | |
| 117 | void EditorExport::add_export_platform(const Ref<EditorExportPlatform> &p_platform) { |
| 118 | export_platforms.push_back(p_platform); |
| 119 | should_update_presets = true; |
| 120 | } |
| 121 | |
| 122 | int EditorExport::get_export_platform_count() { |
| 123 | return export_platforms.size(); |
| 124 | } |
| 125 | |
| 126 | Ref<EditorExportPlatform> EditorExport::get_export_platform(int p_idx) { |
| 127 | ERR_FAIL_INDEX_V(p_idx, export_platforms.size(), Ref<EditorExportPlatform>()); |
| 128 | |
| 129 | return export_platforms[p_idx]; |
| 130 | } |
| 131 | |
| 132 | void EditorExport::add_export_preset(const Ref<EditorExportPreset> &p_preset, int p_at_pos) { |
| 133 | if (p_at_pos < 0) { |
| 134 | export_presets.push_back(p_preset); |
| 135 | } else { |
| 136 | export_presets.insert(p_at_pos, p_preset); |
| 137 | } |
| 138 | } |
| 139 | |
| 140 | int EditorExport::get_export_preset_count() const { |
| 141 | return export_presets.size(); |
| 142 | } |
| 143 | |
| 144 | Ref<EditorExportPreset> EditorExport::get_export_preset(int p_idx) { |
| 145 | ERR_FAIL_INDEX_V(p_idx, export_presets.size(), Ref<EditorExportPreset>()); |
| 146 | return export_presets[p_idx]; |
| 147 | } |
| 148 | |
| 149 | void EditorExport::remove_export_preset(int p_idx) { |
| 150 | export_presets.remove_at(p_idx); |
| 151 | save_presets(); |
| 152 | } |
| 153 | |
| 154 | void EditorExport::add_export_plugin(const Ref<EditorExportPlugin> &p_plugin) { |
| 155 | if (!export_plugins.has(p_plugin)) { |
| 156 | export_plugins.push_back(p_plugin); |
| 157 | should_update_presets = true; |
| 158 | } |
| 159 | } |
| 160 | |
| 161 | void EditorExport::remove_export_plugin(const Ref<EditorExportPlugin> &p_plugin) { |
| 162 | export_plugins.erase(p_plugin); |
| 163 | should_update_presets = true; |
| 164 | } |
| 165 | |
| 166 | Vector<Ref<EditorExportPlugin>> EditorExport::get_export_plugins() { |
| 167 | return export_plugins; |
| 168 | } |
| 169 | |
| 170 | void EditorExport::_notification(int p_what) { |
| 171 | switch (p_what) { |
| 172 | case NOTIFICATION_ENTER_TREE: { |
| 173 | load_config(); |
| 174 | } break; |
| 175 | |
| 176 | case NOTIFICATION_PROCESS: { |
| 177 | update_export_presets(); |
| 178 | } break; |
| 179 | |
| 180 | case NOTIFICATION_EXIT_TREE: { |
| 181 | for (int i = 0; i < export_platforms.size(); i++) { |
| 182 | export_platforms.write[i]->cleanup(); |
| 183 | } |
| 184 | } break; |
| 185 | } |
| 186 | } |
| 187 | |
| 188 | void EditorExport::load_config() { |
| 189 | Ref<ConfigFile> config; |
| 190 | config.instantiate(); |
| 191 | Error err = config->load("res://export_presets.cfg" ); |
| 192 | if (err != OK) { |
| 193 | return; |
| 194 | } |
| 195 | |
| 196 | Ref<ConfigFile> credentials; |
| 197 | credentials.instantiate(); |
| 198 | err = credentials->load("res://.godot/export_credentials.cfg" ); |
| 199 | if (!(err == OK || err == ERR_FILE_NOT_FOUND)) { |
| 200 | return; |
| 201 | } |
| 202 | |
| 203 | block_save = true; |
| 204 | |
| 205 | int index = 0; |
| 206 | while (true) { |
| 207 | String section = "preset." + itos(index); |
| 208 | if (!config->has_section(section)) { |
| 209 | break; |
| 210 | } |
| 211 | |
| 212 | String platform = config->get_value(section, "platform" ); |
| 213 | |
| 214 | Ref<EditorExportPreset> preset; |
| 215 | |
| 216 | for (int i = 0; i < export_platforms.size(); i++) { |
| 217 | if (export_platforms[i]->get_name() == platform) { |
| 218 | preset = export_platforms.write[i]->create_preset(); |
| 219 | break; |
| 220 | } |
| 221 | } |
| 222 | |
| 223 | if (!preset.is_valid()) { |
| 224 | index++; |
| 225 | ERR_CONTINUE(!preset.is_valid()); |
| 226 | } |
| 227 | |
| 228 | preset->set_name(config->get_value(section, "name" )); |
| 229 | preset->set_runnable(config->get_value(section, "runnable" )); |
| 230 | preset->set_dedicated_server(config->get_value(section, "dedicated_server" , false)); |
| 231 | |
| 232 | if (config->has_section_key(section, "custom_features" )) { |
| 233 | preset->set_custom_features(config->get_value(section, "custom_features" )); |
| 234 | } |
| 235 | |
| 236 | String export_filter = config->get_value(section, "export_filter" ); |
| 237 | |
| 238 | bool get_files = false; |
| 239 | |
| 240 | if (export_filter == "all_resources" ) { |
| 241 | preset->set_export_filter(EditorExportPreset::EXPORT_ALL_RESOURCES); |
| 242 | } else if (export_filter == "scenes" ) { |
| 243 | preset->set_export_filter(EditorExportPreset::EXPORT_SELECTED_SCENES); |
| 244 | get_files = true; |
| 245 | } else if (export_filter == "resources" ) { |
| 246 | preset->set_export_filter(EditorExportPreset::EXPORT_SELECTED_RESOURCES); |
| 247 | get_files = true; |
| 248 | } else if (export_filter == "exclude" ) { |
| 249 | preset->set_export_filter(EditorExportPreset::EXCLUDE_SELECTED_RESOURCES); |
| 250 | get_files = true; |
| 251 | } else if (export_filter == "customized" ) { |
| 252 | preset->set_export_filter(EditorExportPreset::EXPORT_CUSTOMIZED); |
| 253 | preset->set_customized_files(config->get_value(section, "customized_files" , Dictionary())); |
| 254 | get_files = false; |
| 255 | } |
| 256 | |
| 257 | if (get_files) { |
| 258 | Vector<String> files = config->get_value(section, "export_files" ); |
| 259 | |
| 260 | for (int i = 0; i < files.size(); i++) { |
| 261 | if (!FileAccess::exists(files[i])) { |
| 262 | preset->remove_export_file(files[i]); |
| 263 | } else { |
| 264 | preset->add_export_file(files[i]); |
| 265 | } |
| 266 | } |
| 267 | } |
| 268 | |
| 269 | preset->set_include_filter(config->get_value(section, "include_filter" )); |
| 270 | preset->set_exclude_filter(config->get_value(section, "exclude_filter" )); |
| 271 | preset->set_export_path(config->get_value(section, "export_path" , "" )); |
| 272 | |
| 273 | if (config->has_section_key(section, "encrypt_pck" )) { |
| 274 | preset->set_enc_pck(config->get_value(section, "encrypt_pck" )); |
| 275 | } |
| 276 | if (config->has_section_key(section, "encrypt_directory" )) { |
| 277 | preset->set_enc_directory(config->get_value(section, "encrypt_directory" )); |
| 278 | } |
| 279 | if (config->has_section_key(section, "encryption_include_filters" )) { |
| 280 | preset->set_enc_in_filter(config->get_value(section, "encryption_include_filters" )); |
| 281 | } |
| 282 | if (config->has_section_key(section, "encryption_exclude_filters" )) { |
| 283 | preset->set_enc_ex_filter(config->get_value(section, "encryption_exclude_filters" )); |
| 284 | } |
| 285 | if (credentials->has_section_key(section, "script_encryption_key" )) { |
| 286 | preset->set_script_encryption_key(credentials->get_value(section, "script_encryption_key" )); |
| 287 | } |
| 288 | |
| 289 | String option_section = "preset." + itos(index) + ".options" ; |
| 290 | |
| 291 | List<String> options; |
| 292 | config->get_section_keys(option_section, &options); |
| 293 | |
| 294 | for (const String &E : options) { |
| 295 | Variant value = config->get_value(option_section, E); |
| 296 | preset->set(E, value); |
| 297 | } |
| 298 | |
| 299 | if (credentials->has_section(option_section)) { |
| 300 | options.clear(); |
| 301 | credentials->get_section_keys(option_section, &options); |
| 302 | |
| 303 | for (const String &E : options) { |
| 304 | // Drop values for secret properties that no longer exist, or during the next save they would end up in the regular config file. |
| 305 | if (preset->get_properties().has(E)) { |
| 306 | Variant value = credentials->get_value(option_section, E); |
| 307 | preset->set(E, value); |
| 308 | } |
| 309 | } |
| 310 | } |
| 311 | |
| 312 | add_export_preset(preset); |
| 313 | index++; |
| 314 | } |
| 315 | |
| 316 | block_save = false; |
| 317 | } |
| 318 | |
| 319 | void EditorExport::update_export_presets() { |
| 320 | HashMap<StringName, List<EditorExportPlatform::ExportOption>> platform_options; |
| 321 | |
| 322 | for (int i = 0; i < export_platforms.size(); i++) { |
| 323 | Ref<EditorExportPlatform> platform = export_platforms[i]; |
| 324 | |
| 325 | bool should_update = should_update_presets; |
| 326 | should_update |= platform->should_update_export_options(); |
| 327 | for (int j = 0; j < export_plugins.size(); j++) { |
| 328 | should_update |= export_plugins.write[j]->_should_update_export_options(platform); |
| 329 | } |
| 330 | |
| 331 | if (should_update) { |
| 332 | List<EditorExportPlatform::ExportOption> options; |
| 333 | platform->get_export_options(&options); |
| 334 | |
| 335 | for (int j = 0; j < export_plugins.size(); j++) { |
| 336 | export_plugins[j]->_get_export_options(platform, &options); |
| 337 | } |
| 338 | |
| 339 | platform_options[platform->get_name()] = options; |
| 340 | } |
| 341 | } |
| 342 | should_update_presets = false; |
| 343 | |
| 344 | bool export_presets_updated = false; |
| 345 | for (int i = 0; i < export_presets.size(); i++) { |
| 346 | Ref<EditorExportPreset> preset = export_presets[i]; |
| 347 | if (platform_options.has(preset->get_platform()->get_name())) { |
| 348 | export_presets_updated = true; |
| 349 | |
| 350 | List<EditorExportPlatform::ExportOption> options = platform_options[preset->get_platform()->get_name()]; |
| 351 | |
| 352 | // Clear the preset properties prior to reloading, keep the values to preserve options from plugins that may be currently disabled. |
| 353 | preset->properties.clear(); |
| 354 | preset->update_visibility.clear(); |
| 355 | |
| 356 | for (const EditorExportPlatform::ExportOption &E : options) { |
| 357 | StringName option_name = E.option.name; |
| 358 | preset->properties[option_name] = E.option; |
| 359 | if (!preset->has(option_name)) { |
| 360 | preset->values[option_name] = E.default_value; |
| 361 | } |
| 362 | preset->update_visibility[option_name] = E.update_visibility; |
| 363 | } |
| 364 | } |
| 365 | } |
| 366 | |
| 367 | if (export_presets_updated) { |
| 368 | emit_signal(_export_presets_updated); |
| 369 | } |
| 370 | } |
| 371 | |
| 372 | bool EditorExport::poll_export_platforms() { |
| 373 | bool changed = false; |
| 374 | for (int i = 0; i < export_platforms.size(); i++) { |
| 375 | if (export_platforms.write[i]->poll_export()) { |
| 376 | changed = true; |
| 377 | } |
| 378 | } |
| 379 | |
| 380 | return changed; |
| 381 | } |
| 382 | |
| 383 | EditorExport::EditorExport() { |
| 384 | save_timer = memnew(Timer); |
| 385 | add_child(save_timer); |
| 386 | save_timer->set_wait_time(0.8); |
| 387 | save_timer->set_one_shot(true); |
| 388 | save_timer->connect("timeout" , callable_mp(this, &EditorExport::_save)); |
| 389 | |
| 390 | _export_presets_updated = "export_presets_updated" ; |
| 391 | |
| 392 | singleton = this; |
| 393 | set_process(true); |
| 394 | } |
| 395 | |
| 396 | EditorExport::~EditorExport() { |
| 397 | } |
| 398 | |