Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
godotengine
GitHub Repository: godotengine/godot
Path: blob/master/editor/export/editor_export.cpp
20957 views
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
#include "editor/settings/editor_settings.h"
36
#include "scene/main/timer.h"
37
38
EditorExport *EditorExport::singleton = nullptr;
39
40
void EditorExport::_save() {
41
Ref<ConfigFile> config;
42
Ref<ConfigFile> credentials;
43
config.instantiate();
44
credentials.instantiate();
45
46
for (const KeyValue<Ref<EditorExportPlatform>, Ref<EditorExportPreset>> &E : runnable_presets) {
47
config->set_value(RUNNABLE_SECTION_NAME, E.key->get_name(), E.value->get_name());
48
}
49
50
for (int i = 0; i < export_presets.size(); i++) {
51
Ref<EditorExportPreset> preset = export_presets[i];
52
String section = "preset." + itos(i);
53
54
config->set_value(section, "name", preset->get_name());
55
config->set_value(section, "platform", preset->get_platform()->get_name());
56
config->set_value(section, "dedicated_server", preset->is_dedicated_server());
57
config->set_value(section, "custom_features", preset->get_custom_features());
58
59
bool save_files = false;
60
switch (preset->get_export_filter()) {
61
case EditorExportPreset::EXPORT_ALL_RESOURCES: {
62
config->set_value(section, "export_filter", "all_resources");
63
} break;
64
case EditorExportPreset::EXPORT_SELECTED_SCENES: {
65
config->set_value(section, "export_filter", "scenes");
66
save_files = true;
67
} break;
68
case EditorExportPreset::EXPORT_SELECTED_RESOURCES: {
69
config->set_value(section, "export_filter", "resources");
70
save_files = true;
71
} break;
72
case EditorExportPreset::EXCLUDE_SELECTED_RESOURCES: {
73
config->set_value(section, "export_filter", "exclude");
74
save_files = true;
75
} break;
76
case EditorExportPreset::EXPORT_CUSTOMIZED: {
77
config->set_value(section, "export_filter", "customized");
78
config->set_value(section, "customized_files", preset->get_customized_files());
79
save_files = false;
80
};
81
}
82
83
if (save_files) {
84
Vector<String> export_files = preset->get_files_to_export();
85
config->set_value(section, "export_files", export_files);
86
}
87
config->set_value(section, "include_filter", preset->get_include_filter());
88
config->set_value(section, "exclude_filter", preset->get_exclude_filter());
89
config->set_value(section, "export_path", preset->get_export_path());
90
91
config->set_value(section, "patches", preset->get_patches());
92
config->set_value(section, "patch_delta_encoding", preset->is_patch_delta_encoding_enabled());
93
config->set_value(section, "patch_delta_compression_level_zstd", preset->get_patch_delta_zstd_level());
94
config->set_value(section, "patch_delta_min_reduction", preset->get_patch_delta_min_reduction());
95
config->set_value(section, "patch_delta_include_filters", preset->get_patch_delta_include_filter());
96
config->set_value(section, "patch_delta_exclude_filters", preset->get_patch_delta_exclude_filter());
97
98
config->set_value(section, "encryption_include_filters", preset->get_enc_in_filter());
99
config->set_value(section, "encryption_exclude_filters", preset->get_enc_ex_filter());
100
config->set_value(section, "seed", preset->get_seed());
101
102
config->set_value(section, "encrypt_pck", preset->get_enc_pck());
103
config->set_value(section, "encrypt_directory", preset->get_enc_directory());
104
config->set_value(section, "script_export_mode", preset->get_script_export_mode());
105
credentials->set_value(section, "script_encryption_key", preset->get_script_encryption_key());
106
107
String option_section = "preset." + itos(i) + ".options";
108
109
for (const KeyValue<StringName, Variant> &E : preset->values) {
110
PropertyInfo *prop = preset->properties.getptr(E.key);
111
if (prop && prop->usage & PROPERTY_USAGE_SECRET) {
112
credentials->set_value(option_section, E.key, E.value);
113
} else {
114
config->set_value(option_section, E.key, E.value);
115
}
116
}
117
}
118
119
config->save("res://export_presets.cfg");
120
credentials->save("res://.godot/export_credentials.cfg");
121
}
122
123
void EditorExport::save_presets() {
124
if (block_save) {
125
return;
126
}
127
save_timer->start();
128
}
129
130
void EditorExport::emit_presets_runnable_changed() {
131
emit_signal(_export_presets_runnable_updated);
132
}
133
134
void EditorExport::_bind_methods() {
135
ADD_SIGNAL(MethodInfo(_export_presets_updated));
136
ADD_SIGNAL(MethodInfo(_export_presets_runnable_updated));
137
}
138
139
void EditorExport::add_export_platform(const Ref<EditorExportPlatform> &p_platform) {
140
p_platform->initialize();
141
export_platforms.push_back(p_platform);
142
143
should_update_presets = true;
144
should_reload_presets = true;
145
}
146
147
void EditorExport::remove_export_platform(const Ref<EditorExportPlatform> &p_platform) {
148
export_platforms.erase(p_platform);
149
p_platform->cleanup();
150
151
should_update_presets = true;
152
should_reload_presets = true;
153
}
154
155
int EditorExport::get_export_platform_count() const {
156
return export_platforms.size();
157
}
158
159
int EditorExport::get_export_platform_index_by_name(const String &p_name) {
160
for (int j = 0; j < get_export_platform_count(); j++) {
161
Ref<EditorExportPlatform> plat = get_export_platform(j);
162
if (!plat.is_null() && plat->get_name().nocasecmp_to(p_name) == 0) {
163
return j;
164
}
165
}
166
return -1;
167
}
168
169
bool EditorExport::has_preset_with_name(const String &p_name, int p_exclude_index) const {
170
for (int i = 0; i < export_presets.size(); i++) {
171
if (i == p_exclude_index) {
172
continue;
173
}
174
if (export_presets[i]->get_name() == p_name) {
175
return true;
176
}
177
}
178
179
return false;
180
}
181
182
Ref<EditorExportPlatform> EditorExport::get_export_platform(int p_idx) {
183
ERR_FAIL_INDEX_V(p_idx, export_platforms.size(), Ref<EditorExportPlatform>());
184
185
return export_platforms[p_idx];
186
}
187
188
void EditorExport::add_export_preset(const Ref<EditorExportPreset> &p_preset, int p_at_pos) {
189
if (p_at_pos < 0) {
190
export_presets.push_back(p_preset);
191
} else {
192
export_presets.insert(p_at_pos, p_preset);
193
}
194
emit_presets_runnable_changed();
195
}
196
197
int EditorExport::get_export_preset_count() const {
198
return export_presets.size();
199
}
200
201
Ref<EditorExportPreset> EditorExport::get_export_preset(int p_idx) {
202
ERR_FAIL_INDEX_V(p_idx, export_presets.size(), Ref<EditorExportPreset>());
203
return export_presets[p_idx];
204
}
205
206
void EditorExport::remove_export_preset(int p_idx) {
207
export_presets.remove_at(p_idx);
208
save_presets();
209
emit_presets_runnable_changed();
210
}
211
212
void EditorExport::add_export_plugin(const Ref<EditorExportPlugin> &p_plugin) {
213
if (!export_plugins.has(p_plugin)) {
214
export_plugins.push_back(p_plugin);
215
should_update_presets = true;
216
}
217
}
218
219
void EditorExport::remove_export_plugin(const Ref<EditorExportPlugin> &p_plugin) {
220
export_plugins.erase(p_plugin);
221
should_update_presets = true;
222
}
223
224
Vector<Ref<EditorExportPlugin>> EditorExport::get_export_plugins() {
225
return export_plugins;
226
}
227
228
void EditorExport::set_runnable_preset(const Ref<EditorExportPreset> &p_preset) {
229
runnable_presets[p_preset->get_platform()] = p_preset;
230
emit_presets_runnable_changed();
231
save_presets();
232
}
233
234
void EditorExport::unset_runnable_preset(const Ref<EditorExportPreset> &p_preset) {
235
const Ref<EditorExportPreset> *current = runnable_presets.getptr(p_preset->get_platform());
236
if (current && *current == p_preset) {
237
runnable_presets.erase(p_preset->get_platform());
238
emit_presets_runnable_changed();
239
save_presets();
240
}
241
}
242
243
Ref<EditorExportPreset> EditorExport::get_runnable_preset_for_platform(const Ref<EditorExportPlatform> &p_for_platform) const {
244
const Ref<EditorExportPreset> *preset = runnable_presets.getptr(p_for_platform);
245
return preset ? *preset : Ref<EditorExportPreset>();
246
}
247
248
void EditorExport::_notification(int p_what) {
249
switch (p_what) {
250
case NOTIFICATION_ENTER_TREE: {
251
load_config();
252
} break;
253
254
case NOTIFICATION_PROCESS: {
255
update_export_presets();
256
} break;
257
258
case NOTIFICATION_EXIT_TREE: {
259
for (int i = 0; i < export_platforms.size(); i++) {
260
export_platforms.write[i]->cleanup();
261
}
262
} break;
263
264
case EditorSettings::NOTIFICATION_EDITOR_SETTINGS_CHANGED: {
265
for (int i = 0; i < export_platforms.size(); i++) {
266
export_platforms.write[i]->notification(p_what);
267
}
268
} break;
269
}
270
}
271
272
void EditorExport::load_config() {
273
Ref<ConfigFile> config;
274
config.instantiate();
275
Error err = config->load("res://export_presets.cfg");
276
if (err != OK) {
277
return;
278
}
279
280
Ref<ConfigFile> credentials;
281
credentials.instantiate();
282
err = credentials->load("res://.godot/export_credentials.cfg");
283
if (!(err == OK || err == ERR_FILE_NOT_FOUND)) {
284
return;
285
}
286
287
HashMap<String, String> runnable_loading;
288
if (config->has_section(RUNNABLE_SECTION_NAME)) {
289
for (const String &platform_name : config->get_section_keys(RUNNABLE_SECTION_NAME)) {
290
runnable_loading[platform_name] = config->get_value(RUNNABLE_SECTION_NAME, platform_name);
291
}
292
}
293
294
block_save = true;
295
296
int index = 0;
297
while (true) {
298
String section = "preset." + itos(index);
299
if (!config->has_section(section)) {
300
break;
301
}
302
303
String platform = config->get_value(section, "platform");
304
#ifndef DISABLE_DEPRECATED
305
// Compatibility with Linux platform before 4.3.
306
if (platform == "Linux/X11") {
307
platform = "Linux";
308
}
309
#endif
310
311
Ref<EditorExportPreset> preset;
312
313
for (Ref<EditorExportPlatform> &export_platform : export_platforms) {
314
if (export_platform->get_name() == platform) {
315
preset = export_platform->create_preset();
316
317
const String preset_name = config->get_value(section, "name");
318
preset->set_name(preset_name);
319
320
const String *runnable_preset = runnable_loading.getptr(export_platform->get_name());
321
if (runnable_preset && *runnable_preset == preset_name) {
322
runnable_presets[export_platform] = preset;
323
}
324
break;
325
}
326
}
327
328
if (preset.is_null()) {
329
index++;
330
continue; // Unknown platform, skip without error (platform might be loaded later).
331
}
332
333
#ifndef DISABLE_DEPRECATED
334
bool legacy_runnable = config->get_value(section, "runnable", false);
335
if (legacy_runnable) {
336
preset->set_runnable(true);
337
}
338
#endif
339
preset->set_dedicated_server(config->get_value(section, "dedicated_server", false));
340
341
if (config->has_section_key(section, "custom_features")) {
342
preset->set_custom_features(config->get_value(section, "custom_features"));
343
}
344
345
String export_filter = config->get_value(section, "export_filter");
346
347
bool get_files = false;
348
349
if (export_filter == "all_resources") {
350
preset->set_export_filter(EditorExportPreset::EXPORT_ALL_RESOURCES);
351
} else if (export_filter == "scenes") {
352
preset->set_export_filter(EditorExportPreset::EXPORT_SELECTED_SCENES);
353
get_files = true;
354
} else if (export_filter == "resources") {
355
preset->set_export_filter(EditorExportPreset::EXPORT_SELECTED_RESOURCES);
356
get_files = true;
357
} else if (export_filter == "exclude") {
358
preset->set_export_filter(EditorExportPreset::EXCLUDE_SELECTED_RESOURCES);
359
get_files = true;
360
} else if (export_filter == "customized") {
361
preset->set_export_filter(EditorExportPreset::EXPORT_CUSTOMIZED);
362
preset->set_customized_files(config->get_value(section, "customized_files", Dictionary()));
363
get_files = false;
364
}
365
366
if (get_files) {
367
Vector<String> files = config->get_value(section, "export_files");
368
369
for (int i = 0; i < files.size(); i++) {
370
if (!FileAccess::exists(files[i])) {
371
preset->remove_export_file(files[i]);
372
} else {
373
preset->add_export_file(files[i]);
374
}
375
}
376
}
377
378
preset->set_include_filter(config->get_value(section, "include_filter"));
379
preset->set_exclude_filter(config->get_value(section, "exclude_filter"));
380
preset->set_export_path(config->get_value(section, "export_path", ""));
381
preset->set_script_export_mode(config->get_value(section, "script_export_mode", EditorExportPreset::MODE_SCRIPT_BINARY_TOKENS_COMPRESSED));
382
preset->set_patches(config->get_value(section, "patches", Vector<String>()));
383
384
if (config->has_section_key(section, "patch_delta_encoding")) {
385
preset->set_patch_delta_encoding_enabled(config->get_value(section, "patch_delta_encoding"));
386
}
387
if (config->has_section_key(section, "patch_delta_compression_level_zstd")) {
388
preset->set_patch_delta_zstd_level(config->get_value(section, "patch_delta_compression_level_zstd"));
389
}
390
if (config->has_section_key(section, "patch_delta_min_reduction")) {
391
preset->set_patch_delta_min_reduction(config->get_value(section, "patch_delta_min_reduction"));
392
}
393
if (config->has_section_key(section, "patch_delta_include_filters")) {
394
preset->set_patch_delta_include_filter(config->get_value(section, "patch_delta_include_filters"));
395
}
396
if (config->has_section_key(section, "patch_delta_exclude_filters")) {
397
preset->set_patch_delta_exclude_filter(config->get_value(section, "patch_delta_exclude_filters"));
398
}
399
400
if (config->has_section_key(section, "seed")) {
401
preset->set_seed(config->get_value(section, "seed"));
402
}
403
if (config->has_section_key(section, "encrypt_pck")) {
404
preset->set_enc_pck(config->get_value(section, "encrypt_pck"));
405
}
406
if (config->has_section_key(section, "encrypt_directory")) {
407
preset->set_enc_directory(config->get_value(section, "encrypt_directory"));
408
}
409
if (config->has_section_key(section, "encryption_include_filters")) {
410
preset->set_enc_in_filter(config->get_value(section, "encryption_include_filters"));
411
}
412
if (config->has_section_key(section, "encryption_exclude_filters")) {
413
preset->set_enc_ex_filter(config->get_value(section, "encryption_exclude_filters"));
414
}
415
if (credentials->has_section_key(section, "script_encryption_key")) {
416
preset->set_script_encryption_key(credentials->get_value(section, "script_encryption_key"));
417
}
418
419
String option_section = "preset." + itos(index) + ".options";
420
421
Vector<String> options = config->get_section_keys(option_section);
422
423
for (const String &E : options) {
424
Variant value = config->get_value(option_section, E);
425
preset->set(E, value);
426
}
427
428
if (credentials->has_section(option_section)) {
429
options = credentials->get_section_keys(option_section);
430
431
for (const String &E : options) {
432
// Drop values for secret properties that no longer exist, or during the next save they would end up in the regular config file.
433
if (preset->get_properties().has(E)) {
434
Variant value = credentials->get_value(option_section, E);
435
preset->set(E, value);
436
}
437
}
438
}
439
440
add_export_preset(preset);
441
index++;
442
}
443
444
block_save = false;
445
}
446
447
void EditorExport::update_export_presets() {
448
HashMap<StringName, List<EditorExportPlatform::ExportOption>> platform_options;
449
450
if (should_reload_presets) {
451
should_reload_presets = false;
452
export_presets.clear();
453
load_config();
454
}
455
456
for (int i = 0; i < export_platforms.size(); i++) {
457
Ref<EditorExportPlatform> platform = export_platforms[i];
458
459
bool should_update = should_update_presets;
460
should_update |= platform->should_update_export_options();
461
for (int j = 0; j < export_plugins.size(); j++) {
462
should_update |= export_plugins.write[j]->_should_update_export_options(platform);
463
}
464
465
if (should_update) {
466
List<EditorExportPlatform::ExportOption> options;
467
platform->get_export_options(&options);
468
469
for (int j = 0; j < export_plugins.size(); j++) {
470
export_plugins[j]->_get_export_options(platform, &options);
471
}
472
473
platform_options[platform->get_name()] = options;
474
}
475
}
476
should_update_presets = false;
477
478
bool export_presets_updated = false;
479
for (int i = 0; i < export_presets.size(); i++) {
480
Ref<EditorExportPreset> preset = export_presets[i];
481
if (platform_options.has(preset->get_platform()->get_name())) {
482
export_presets_updated = true;
483
484
bool update_value_overrides = false;
485
List<EditorExportPlatform::ExportOption> options = platform_options[preset->get_platform()->get_name()];
486
487
// Clear the preset properties prior to reloading, keep the values to preserve options from plugins that may be currently disabled.
488
preset->properties.clear();
489
preset->update_visibility.clear();
490
491
for (const EditorExportPlatform::ExportOption &E : options) {
492
StringName option_name = E.option.name;
493
preset->properties[option_name] = E.option;
494
if (!preset->has(option_name)) {
495
preset->values[option_name] = E.default_value;
496
}
497
preset->update_visibility[option_name] = E.update_visibility;
498
if (E.update_visibility) {
499
update_value_overrides = true;
500
}
501
}
502
503
if (update_value_overrides) {
504
preset->update_value_overrides();
505
}
506
}
507
}
508
509
if (export_presets_updated) {
510
emit_signal(_export_presets_updated);
511
}
512
}
513
514
bool EditorExport::poll_export_platforms() {
515
bool changed = false;
516
for (int i = 0; i < export_platforms.size(); i++) {
517
if (export_platforms.write[i]->poll_export()) {
518
changed = true;
519
}
520
}
521
522
return changed;
523
}
524
525
void EditorExport::connect_presets_runnable_updated(const Callable &p_target) {
526
connect(_export_presets_runnable_updated, p_target);
527
}
528
529
EditorExport::EditorExport() {
530
save_timer = memnew(Timer);
531
add_child(save_timer);
532
save_timer->set_wait_time(0.8);
533
save_timer->set_one_shot(true);
534
save_timer->connect("timeout", callable_mp(this, &EditorExport::_save));
535
536
_export_presets_updated = StringName("export_presets_updated", true);
537
_export_presets_runnable_updated = StringName("export_presets_runnable_updated", true);
538
539
singleton = this;
540
set_process(true);
541
}
542
543