/*************************************************************************/ /* editor_import_export.cpp */ /*************************************************************************/ /* This file is part of: */ /* GODOT ENGINE */ /* http://www.godotengine.org */ /*************************************************************************/ /* Copyright (c) 2007-2017 Juan Linietsky, Ariel Manzur. */ /* Copyright (c) 2014-2017 Godot Engine contributors (cf. AUTHORS.md) */ /* */ /* Permission is hereby granted, free of charge, to any person obtaining */ /* a copy of this software and associated documentation files (the */ /* "Software"), to deal in the Software without restriction, including */ /* without limitation the rights to use, copy, modify, merge, publish, */ /* distribute, sublicense, and/or sell copies of the Software, and to */ /* permit persons to whom the Software is furnished to do so, subject to */ /* the following conditions: */ /* */ /* The above copyright notice and this permission notice shall be */ /* included in all copies or substantial portions of the Software. */ /* */ /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /*************************************************************************/ #include "editor_export.h" #include "editor/editor_file_system.h" #include "editor/plugins/script_editor_plugin.h" #include "editor_node.h" #include "editor_settings.h" #include "io/config_file.h" #include "io/resource_loader.h" #include "io/resource_saver.h" #include "io/zip_io.h" #include "os/file_access.h" #include "project_settings.h" #include "script_language.h" #include "version.h" #include "thirdparty/misc/md5.h" static int _get_pad(int p_alignment, int p_n) { int rest = p_n % p_alignment; int pad = 0; if (rest > 0) { pad = p_alignment - rest; }; return pad; } #define PCK_PADDING 16 bool EditorExportPreset::_set(const StringName &p_name, const Variant &p_value) { if (values.has(p_name)) { values[p_name] = p_value; EditorExport::singleton->save_presets(); return true; } return false; } bool EditorExportPreset::_get(const StringName &p_name, Variant &r_ret) const { if (values.has(p_name)) { r_ret = values[p_name]; return true; } return false; } void EditorExportPreset::_get_property_list(List *p_list) const { for (const List::Element *E = properties.front(); E; E = E->next()) { if (platform->get_option_visibility(E->get().name, values)) { p_list->push_back(E->get()); } } } Ref EditorExportPreset::get_platform() const { return platform; } Vector EditorExportPreset::get_files_to_export() const { Vector files; for (Set::Element *E = selected_files.front(); E; E = E->next()) { files.push_back(E->get()); } return files; } void EditorExportPreset::set_name(const String &p_name) { name = p_name; EditorExport::singleton->save_presets(); } String EditorExportPreset::get_name() const { return name; } void EditorExportPreset::set_runnable(bool p_enable) { runnable = p_enable; EditorExport::singleton->save_presets(); } bool EditorExportPreset::is_runnable() const { return runnable; } void EditorExportPreset::set_export_filter(ExportFilter p_filter) { export_filter = p_filter; EditorExport::singleton->save_presets(); } EditorExportPreset::ExportFilter EditorExportPreset::get_export_filter() const { return export_filter; } void EditorExportPreset::set_include_filter(const String &p_include) { include_filter = p_include; EditorExport::singleton->save_presets(); } String EditorExportPreset::get_include_filter() const { return include_filter; } void EditorExportPreset::set_exclude_filter(const String &p_exclude) { exclude_filter = p_exclude; EditorExport::singleton->save_presets(); } String EditorExportPreset::get_exclude_filter() const { return exclude_filter; } void EditorExportPreset::add_export_file(const String &p_path) { selected_files.insert(p_path); EditorExport::singleton->save_presets(); } void EditorExportPreset::remove_export_file(const String &p_path) { selected_files.erase(p_path); EditorExport::singleton->save_presets(); } bool EditorExportPreset::has_export_file(const String &p_path) { return selected_files.has(p_path); } void EditorExportPreset::add_patch(const String &p_path, int p_at_pos) { if (p_at_pos < 0) patches.push_back(p_path); else patches.insert(p_at_pos, p_path); EditorExport::singleton->save_presets(); } void EditorExportPreset::remove_patch(int p_idx) { patches.remove(p_idx); EditorExport::singleton->save_presets(); } void EditorExportPreset::set_patch(int p_index, const String &p_path) { ERR_FAIL_INDEX(p_index, patches.size()); patches[p_index] = p_path; EditorExport::singleton->save_presets(); } String EditorExportPreset::get_patch(int p_index) { ERR_FAIL_INDEX_V(p_index, patches.size(), String()); return patches[p_index]; } Vector EditorExportPreset::get_patches() const { return patches; } void EditorExportPreset::set_custom_features(const String &p_custom_features) { custom_features = p_custom_features; EditorExport::singleton->save_presets(); } String EditorExportPreset::get_custom_features() const { return custom_features; } EditorExportPreset::EditorExportPreset() { export_filter = EXPORT_ALL_RESOURCES; runnable = false; } /////////////////////////////////// void EditorExportPlatform::gen_debug_flags(Vector &r_flags, int p_flags) { String host = EditorSettings::get_singleton()->get("network/debug/remote_host"); int remote_port = (int)EditorSettings::get_singleton()->get("network/debug/remote_port"); if (p_flags & DEBUG_FLAG_REMOTE_DEBUG_LOCALHOST) host = "localhost"; if (p_flags & DEBUG_FLAG_DUMB_CLIENT) { int port = EditorSettings::get_singleton()->get("filesystem/file_server/port"); String passwd = EditorSettings::get_singleton()->get("filesystem/file_server/password"); r_flags.push_back("-rfs"); r_flags.push_back(host + ":" + itos(port)); if (passwd != "") { r_flags.push_back("-rfs_pass"); r_flags.push_back(passwd); } } if (p_flags & DEBUG_FLAG_REMOTE_DEBUG) { r_flags.push_back("-rdebug"); r_flags.push_back(host + ":" + String::num(remote_port)); List breakpoints; ScriptEditor::get_singleton()->get_breakpoints(&breakpoints); if (breakpoints.size()) { r_flags.push_back("-bp"); String bpoints; for (const List::Element *E = breakpoints.front(); E; E = E->next()) { bpoints += E->get().replace(" ", "%20"); if (E->next()) bpoints += ","; } r_flags.push_back(bpoints); } } if (p_flags & DEBUG_FLAG_VIEW_COLLISONS) { r_flags.push_back("-debugcol"); } if (p_flags & DEBUG_FLAG_VIEW_NAVIGATION) { r_flags.push_back("-debugnav"); } } Error EditorExportPlatform::_save_pack_file(void *p_userdata, const String &p_path, const Vector &p_data, int p_file, int p_total) { PackData *pd = (PackData *)p_userdata; SavedData sd; sd.path_utf8 = p_path.utf8(); sd.ofs = pd->f->get_pos(); sd.size = p_data.size(); pd->f->store_buffer(p_data.ptr(), p_data.size()); int pad = _get_pad(PCK_PADDING, sd.size); for (int i = 0; i < pad; i++) { pd->f->store_8(0); } { MD5_CTX ctx; MD5Init(&ctx); MD5Update(&ctx, (unsigned char *)p_data.ptr(), p_data.size()); MD5Final(&ctx); sd.md5.resize(16); for (int i = 0; i < 16; i++) { sd.md5[i] = ctx.digest[i]; } } pd->file_ofs.push_back(sd); pd->ep->step(TTR("Storing File:") + " " + p_path, 2 + p_file * 100 / p_total, false); return OK; } Error EditorExportPlatform::_save_zip_file(void *p_userdata, const String &p_path, const Vector &p_data, int p_file, int p_total) { String path = p_path.replace_first("res://", ""); ZipData *zd = (ZipData *)p_userdata; zipFile zip = (zipFile)zd->zip; zipOpenNewFileInZip(zip, path.utf8().get_data(), NULL, NULL, 0, NULL, 0, NULL, Z_DEFLATED, Z_DEFAULT_COMPRESSION); zipWriteInFileInZip(zip, p_data.ptr(), p_data.size()); zipCloseFileInZip(zip); zd->ep->step(TTR("Storing File:") + " " + p_path, 2 + p_file * 100 / p_total, false); return OK; } String EditorExportPlatform::find_export_template(String template_file_name, String *err) const { String base_name = itos(VERSION_MAJOR) + "." + itos(VERSION_MINOR) + "-" + _MKSTR(VERSION_STATUS) + "/" + template_file_name; String user_file = EditorSettings::get_singleton()->get_settings_path() + "/templates/" + base_name; String system_file = OS::get_singleton()->get_installed_templates_path(); bool has_system_path = (system_file != ""); system_file = system_file.plus_file(base_name); print_line("test user file: " + user_file); // Prefer user file if (FileAccess::exists(user_file)) { return user_file; } print_line("test system file: " + system_file); // Now check system file if (has_system_path) { if (FileAccess::exists(system_file)) { return system_file; } } // Not found if (err) { *err += "No export template found at \"" + user_file + "\""; if (has_system_path) *err += "\n or \"" + system_file + "\"."; else *err += "."; } return String(); // not found } bool EditorExportPlatform::exists_export_template(String template_file_name, String *err) const { return find_export_template(template_file_name, err) != ""; } Ref EditorExportPlatform::create_preset() { Ref preset; preset.instance(); preset->platform = Ref(this); List options; get_export_options(&options); for (List::Element *E = options.front(); E; E = E->next()) { preset->properties.push_back(E->get().option); preset->values[E->get().option.name] = E->get().default_value; } return preset; } void EditorExportPlatform::_export_find_resources(EditorFileSystemDirectory *p_dir, Set &p_paths) { for (int i = 0; i < p_dir->get_subdir_count(); i++) { _export_find_resources(p_dir->get_subdir(i), p_paths); } for (int i = 0; i < p_dir->get_file_count(); i++) { p_paths.insert(p_dir->get_file_path(i)); } } void EditorExportPlatform::_export_find_dependencies(const String &p_path, Set &p_paths) { if (p_paths.has(p_path)) return; p_paths.insert(p_path); EditorFileSystemDirectory *dir; int file_idx; dir = EditorFileSystem::get_singleton()->find_file(p_path, &file_idx); if (!dir) return; Vector deps = dir->get_file_deps(file_idx); for (int i = 0; i < deps.size(); i++) { _export_find_dependencies(deps[i], p_paths); } } void EditorExportPlatform::_edit_files_with_filter(DirAccess *da, const Vector &p_filters, Set &r_list, bool exclude) { da->list_dir_begin(); String cur_dir = da->get_current_dir().replace("\\", "/"); if (!cur_dir.ends_with("/")) cur_dir += "/"; Vector dirs; String f; while ((f = da->get_next()) != "") { if (da->current_is_dir()) dirs.push_back(f); else { String fullpath = cur_dir + f; for (int i = 0; i < p_filters.size(); ++i) { if (fullpath.matchn(p_filters[i])) { if (!exclude) { r_list.insert(fullpath); } else { r_list.erase(fullpath); } } } } } da->list_dir_end(); for (int i = 0; i < dirs.size(); ++i) { String dir = dirs[i]; if (dir.begins_with(".")) continue; da->change_dir(dir); _edit_files_with_filter(da, p_filters, r_list, exclude); da->change_dir(".."); } } void EditorExportPlatform::_edit_filter_list(Set &r_list, const String &p_filter, bool exclude) { if (p_filter == "") return; Vector split = p_filter.split(","); Vector filters; for (int i = 0; i < split.size(); i++) { String f = split[i].strip_edges(); if (f.empty()) continue; filters.push_back(f); } DirAccess *da = DirAccess::open("res://"); ERR_FAIL_NULL(da); _edit_files_with_filter(da, filters, r_list, exclude); memdelete(da); } Error EditorExportPlatform::export_project_files(const Ref &p_preset, EditorExportSaveFunction p_func, void *p_udata) { Ref platform = p_preset->get_platform(); List feature_list; platform->get_preset_features(p_preset, &feature_list); //figure out features Set features; for (List::Element *E = feature_list.front(); E; E = E->next()) { features.insert(E->get()); } //figure out paths of files that will be exported Set paths; if (p_preset->get_export_filter() == EditorExportPreset::EXPORT_ALL_RESOURCES) { //find stuff _export_find_resources(EditorFileSystem::get_singleton()->get_filesystem(), paths); } else { bool scenes_only = p_preset->get_export_filter() == EditorExportPreset::EXPORT_SELECTED_SCENES; Vector files = p_preset->get_files_to_export(); for (int i = 0; i < files.size(); i++) { if (scenes_only && ResourceLoader::get_resource_type(files[i]) != "PackedScene") continue; _export_find_dependencies(files[i], paths); } } _edit_filter_list(paths, p_preset->get_include_filter(), false); _edit_filter_list(paths, p_preset->get_exclude_filter(), true); //store everything in the export medium int idx = 0; int total = paths.size(); for (Set::Element *E = paths.front(); E; E = E->next()) { String path = E->get(); if (FileAccess::exists(path + ".import")) { //file is imported, replace by what it imports Ref config; config.instance(); Error err = config->load(path + ".import"); if (err != OK) { ERR_PRINTS("Could not parse: '" + path + "', not exported."); continue; } List remaps; config->get_section_keys("remap", &remaps); for (List::Element *F = remaps.front(); F; F = F->next()) { String remap = F->get(); if (remap == "path") { String remapped_path = config->get_value("remap", remap); Vector array = FileAccess::get_file_as_array(remapped_path); p_func(p_udata, remapped_path, array, idx, total); } else if (remap.begins_with("path.")) { String feature = remap.get_slice(".", 1); if (features.has(feature)) { String remapped_path = config->get_value("remap", remap); Vector array = FileAccess::get_file_as_array(remapped_path); p_func(p_udata, remapped_path, array, idx, total); } } } //also save the .import file Vector array = FileAccess::get_file_as_array(path + ".import"); p_func(p_udata, path + ".import", array, idx, total); } else { //just store it as it comes Vector array = FileAccess::get_file_as_array(path); p_func(p_udata, path, array, idx, total); } idx++; } //save config! Vector custom_list; if (p_preset->get_custom_features() != String()) { Vector tmp_custom_list = p_preset->get_custom_features().split(","); for (int i = 0; i < tmp_custom_list.size(); i++) { String f = tmp_custom_list[i].strip_edges(); if (f != String()) { custom_list.push_back(f); } } } String config_file = "project.binary"; String engine_cfb = EditorSettings::get_singleton()->get_settings_path() + "/tmp/tmp" + config_file; ProjectSettings::get_singleton()->save_custom(engine_cfb, ProjectSettings::CustomMap(), custom_list); Vector data = FileAccess::get_file_as_array(engine_cfb); p_func(p_udata, "res://" + config_file, data, idx, total); return OK; } Error EditorExportPlatform::save_pack(const Ref &p_preset, const String &p_path) { EditorProgress ep("savepack", TTR("Packing"), 102); String tmppath = EditorSettings::get_singleton()->get_settings_path() + "/tmp/packtmp"; FileAccess *ftmp = FileAccess::open(tmppath, FileAccess::WRITE); ERR_FAIL_COND_V(!ftmp, ERR_CANT_CREATE) PackData pd; pd.ep = &ep; pd.f = ftmp; Error err = export_project_files(p_preset, _save_pack_file, &pd); memdelete(ftmp); //close tmp file if (err) return err; pd.file_ofs.sort(); //do sort, so we can do binary search later FileAccess *f = FileAccess::open(p_path, FileAccess::WRITE); ERR_FAIL_COND_V(!f, ERR_CANT_CREATE) f->store_32(0x43504447); //GDPK f->store_32(1); //pack version f->store_32(VERSION_MAJOR); f->store_32(VERSION_MINOR); f->store_32(0); //hmph for (int i = 0; i < 16; i++) { //reserved f->store_32(0); } f->store_32(pd.file_ofs.size()); //amount of files size_t header_size = f->get_pos(); //precalculate header size for (int i = 0; i < pd.file_ofs.size(); i++) { header_size += 4; // size of path string (32 bits is enough) uint32_t string_len = pd.file_ofs[i].path_utf8.length(); header_size += string_len + _get_pad(4, string_len); ///size of path string header_size += 8; // offset to file _with_ header size included header_size += 8; // size of file header_size += 16; // md5 } size_t header_padding = _get_pad(PCK_PADDING, header_size); for (int i = 0; i < pd.file_ofs.size(); i++) { uint32_t string_len = pd.file_ofs[i].path_utf8.length(); uint32_t pad = _get_pad(4, string_len); ; f->store_32(string_len + pad); f->store_buffer((const uint8_t *)pd.file_ofs[i].path_utf8.get_data(), string_len); for (uint32_t j = 0; j < pad; j++) { f->store_8(0); } f->store_64(pd.file_ofs[i].ofs + header_padding + header_size); f->store_64(pd.file_ofs[i].size); // pay attention here, this is where file is f->store_buffer(pd.file_ofs[i].md5.ptr(), 16); //also save md5 for file } for (uint32_t j = 0; j < header_padding; j++) { f->store_8(0); } //save the rest of the data ftmp = FileAccess::open(tmppath, FileAccess::READ); if (!ftmp) { memdelete(f); ERR_FAIL_COND_V(!ftmp, ERR_CANT_CREATE) } const int bufsize = 16384; uint8_t buf[bufsize]; while (true) { int got = ftmp->get_buffer(buf, bufsize); if (got <= 0) break; f->store_buffer(buf, got); } memdelete(ftmp); f->store_32(0x43504447); //GDPK memdelete(f); return OK; } Error EditorExportPlatform::save_zip(const Ref &p_preset, const String &p_path) { EditorProgress ep("savezip", TTR("Packing"), 102); //FileAccess *tmp = FileAccess::open(tmppath,FileAccess::WRITE); FileAccess *src_f; zlib_filefunc_def io = zipio_create_io_from_file(&src_f); zipFile zip = zipOpen2(p_path.utf8().get_data(), APPEND_STATUS_CREATE, NULL, &io); ZipData zd; zd.ep = &ep; zd.zip = zip; Error err = export_project_files(p_preset, _save_zip_file, &zd); zipClose(zip, NULL); return OK; } void EditorExportPlatform::gen_export_flags(Vector &r_flags, int p_flags) { String host = EditorSettings::get_singleton()->get("network/debug/remote_host"); int remote_port = (int)EditorSettings::get_singleton()->get("network/debug/remote_port"); if (p_flags & DEBUG_FLAG_REMOTE_DEBUG_LOCALHOST) host = "localhost"; if (p_flags & DEBUG_FLAG_DUMB_CLIENT) { int port = EditorSettings::get_singleton()->get("filesystem/file_server/port"); String passwd = EditorSettings::get_singleton()->get("filesystem/file_server/password"); r_flags.push_back("-rfs"); r_flags.push_back(host + ":" + itos(port)); if (passwd != "") { r_flags.push_back("-rfs_pass"); r_flags.push_back(passwd); } } if (p_flags & DEBUG_FLAG_REMOTE_DEBUG) { r_flags.push_back("-rdebug"); r_flags.push_back(host + ":" + String::num(remote_port)); List breakpoints; ScriptEditor::get_singleton()->get_breakpoints(&breakpoints); if (breakpoints.size()) { r_flags.push_back("-bp"); String bpoints; for (const List::Element *E = breakpoints.front(); E; E = E->next()) { bpoints += E->get().replace(" ", "%20"); if (E->next()) bpoints += ","; } r_flags.push_back(bpoints); } } if (p_flags & DEBUG_FLAG_VIEW_COLLISONS) { r_flags.push_back("-debugcol"); } if (p_flags & DEBUG_FLAG_VIEW_NAVIGATION) { r_flags.push_back("-debugnav"); } } EditorExportPlatform::EditorExportPlatform() { } //// EditorExport *EditorExport::singleton = NULL; void EditorExport::_save() { Ref config; config.instance(); for (int i = 0; i < export_presets.size(); i++) { Ref preset = export_presets[i]; String section = "preset." + itos(i); config->set_value(section, "name", preset->get_name()); config->set_value(section, "platform", preset->get_platform()->get_name()); config->set_value(section, "runnable", preset->is_runnable()); config->set_value(section, "custom_features", preset->get_custom_features()); bool save_files = false; switch (preset->get_export_filter()) { case EditorExportPreset::EXPORT_ALL_RESOURCES: { config->set_value(section, "export_filter", "all_resources"); } break; case EditorExportPreset::EXPORT_SELECTED_SCENES: { config->set_value(section, "export_filter", "scenes"); save_files = true; } break; case EditorExportPreset::EXPORT_SELECTED_RESOURCES: { config->set_value(section, "export_filter", "resources"); save_files = true; } break; } if (save_files) { Vector export_files = preset->get_files_to_export(); config->set_value(section, "export_files", export_files); } config->set_value(section, "include_filter", preset->get_include_filter()); config->set_value(section, "exclude_filter", preset->get_exclude_filter()); config->set_value(section, "patch_list", preset->get_patches()); String option_section = "preset." + itos(i) + ".options"; for (const List::Element *E = preset->get_properties().front(); E; E = E->next()) { config->set_value(option_section, E->get().name, preset->get(E->get().name)); } } config->save("res://export_presets.cfg"); print_line("saved ok"); } void EditorExport::save_presets() { print_line("save presets"); if (block_save) return; save_timer->start(); } void EditorExport::_bind_methods() { ClassDB::bind_method("_save", &EditorExport::_save); } void EditorExport::add_export_platform(const Ref &p_platform) { export_platforms.push_back(p_platform); } int EditorExport::get_export_platform_count() { return export_platforms.size(); } Ref EditorExport::get_export_platform(int p_idx) { ERR_FAIL_INDEX_V(p_idx, export_platforms.size(), Ref()); return export_platforms[p_idx]; } void EditorExport::add_export_preset(const Ref &p_preset, int p_at_pos) { if (p_at_pos < 0) export_presets.push_back(p_preset); else export_presets.insert(p_at_pos, p_preset); } int EditorExport::get_export_preset_count() const { return export_presets.size(); } Ref EditorExport::get_export_preset(int p_idx) { ERR_FAIL_INDEX_V(p_idx, export_presets.size(), Ref()); return export_presets[p_idx]; } void EditorExport::remove_export_preset(int p_idx) { export_presets.remove(p_idx); } void EditorExport::_notification(int p_what) { if (p_what == NOTIFICATION_ENTER_TREE) { load_config(); } } void EditorExport::load_config() { Ref config; config.instance(); Error err = config->load("res://export_presets.cfg"); if (err != OK) return; block_save = true; int index = 0; while (true) { String section = "preset." + itos(index); if (!config->has_section(section)) break; String platform = config->get_value(section, "platform"); Ref preset; for (int i = 0; i < export_platforms.size(); i++) { if (export_platforms[i]->get_name() == platform) { preset = export_platforms[i]->create_preset(); break; } } if (!preset.is_valid()) { index++; ERR_CONTINUE(!preset.is_valid()); } preset->set_name(config->get_value(section, "name")); preset->set_runnable(config->get_value(section, "runnable")); if (config->has_section_key(section, "custom_features")) { preset->set_custom_features(config->get_value(section, "custom_features")); } String export_filter = config->get_value(section, "export_filter"); bool get_files = false; if (export_filter == "all_resources") { preset->set_export_filter(EditorExportPreset::EXPORT_ALL_RESOURCES); } else if (export_filter == "scenes") { preset->set_export_filter(EditorExportPreset::EXPORT_SELECTED_SCENES); get_files = true; } else if (export_filter == "resources") { preset->set_export_filter(EditorExportPreset::EXPORT_SELECTED_RESOURCES); get_files = true; } if (get_files) { Vector files = config->get_value(section, "export_files"); for (int i = 0; i < files.size(); i++) { preset->add_export_file(files[i]); } } preset->set_include_filter(config->get_value(section, "include_filter")); preset->set_exclude_filter(config->get_value(section, "exclude_filter")); Vector patch_list = config->get_value(section, "patch_list"); for (int i = 0; i < patch_list.size(); i++) { preset->add_patch(patch_list[i]); } String option_section = "preset." + itos(index) + ".options"; List options; config->get_section_keys(option_section, &options); for (List::Element *E = options.front(); E; E = E->next()) { Variant value = config->get_value(option_section, E->get()); preset->set(E->get(), value); } add_export_preset(preset); index++; } block_save = false; } bool EditorExport::poll_export_platforms() { bool changed = false; for (int i = 0; i < export_platforms.size(); i++) { if (export_platforms[i]->poll_devices()) { changed = true; } } return changed; } EditorExport::EditorExport() { save_timer = memnew(Timer); add_child(save_timer); save_timer->set_wait_time(0.8); save_timer->set_one_shot(true); save_timer->connect("timeout", this, "_save"); block_save = false; singleton = this; } EditorExport::~EditorExport() { } ////////// void EditorExportPlatformPC::get_preset_features(const Ref &p_preset, List *r_features) { if (p_preset->get("texture_format/s3tc")) { r_features->push_back("s3tc"); } if (p_preset->get("texture_format/etc")) { r_features->push_back("etc"); } if (p_preset->get("texture_format/etc2")) { r_features->push_back("etc2"); } } void EditorExportPlatformPC::get_export_options(List *r_options) { r_options->push_back(ExportOption(PropertyInfo(Variant::BOOL, "texture_format/s3tc"), true)); r_options->push_back(ExportOption(PropertyInfo(Variant::BOOL, "texture_format/etc"), false)); r_options->push_back(ExportOption(PropertyInfo(Variant::BOOL, "texture_format/etc2"), false)); r_options->push_back(ExportOption(PropertyInfo(Variant::BOOL, "binary_format/64_bits"), true)); r_options->push_back(ExportOption(PropertyInfo(Variant::STRING, "custom_template/release", PROPERTY_HINT_GLOBAL_FILE), "")); r_options->push_back(ExportOption(PropertyInfo(Variant::STRING, "custom_template/debug", PROPERTY_HINT_GLOBAL_FILE), "")); } String EditorExportPlatformPC::get_name() const { return name; } String EditorExportPlatformPC::get_os_name() const { return os_name; } Ref EditorExportPlatformPC::get_logo() const { return logo; } bool EditorExportPlatformPC::can_export(const Ref &p_preset, String &r_error, bool &r_missing_templates) const { String err; bool valid = true; if (use64 && (!exists_export_template(debug_file_64, &err) || !exists_export_template(release_file_64, &err))) { valid = false; } if (!use64 && (!exists_export_template(debug_file_32, &err) || !exists_export_template(release_file_32, &err))) { valid = false; } String custom_debug_binary = p_preset->get("custom_template/debug"); String custom_release_binary = p_preset->get("custom_template/release"); if (custom_debug_binary == "" && custom_release_binary == "") { if (!err.empty()) r_error = err; return valid; } bool dvalid = true; bool rvalid = true; if (!FileAccess::exists(custom_debug_binary)) { dvalid = false; err = "Custom debug binary not found.\n"; } if (!FileAccess::exists(custom_release_binary)) { rvalid = false; err += "Custom release binary not found.\n"; } if (dvalid || rvalid) valid = true; else valid = false; if (!err.empty()) r_error = err; return valid; } String EditorExportPlatformPC::get_binary_extension() const { return extension; } Error EditorExportPlatformPC::export_project(const Ref &p_preset, bool p_debug, const String &p_path, int p_flags) { String custom_debug = p_preset->get("custom_template/debug"); String custom_release = p_preset->get("custom_template/release"); String template_path = p_debug ? custom_debug : custom_release; template_path = template_path.strip_edges(); if (template_path == String()) { if (p_preset->get("binary_format/64_bits")) { if (p_debug) { template_path = find_export_template(debug_file_64); } else { template_path = find_export_template(release_file_64); } } else { if (p_debug) { template_path = find_export_template(debug_file_32); } else { template_path = find_export_template(release_file_32); } } } if (template_path != String() && !FileAccess::exists(template_path)) { EditorNode::get_singleton()->show_warning(TTR("Template file not found:\n") + template_path); return ERR_FILE_NOT_FOUND; } DirAccess *da = DirAccess::create(DirAccess::ACCESS_FILESYSTEM); da->copy(template_path, p_path); memdelete(da); String pck_path = p_path.get_basename() + ".pck"; return save_pack(p_preset, pck_path); } void EditorExportPlatformPC::set_extension(const String &p_extension) { extension = p_extension; } void EditorExportPlatformPC::set_name(const String &p_name) { name = p_name; } void EditorExportPlatformPC::set_os_name(const String &p_name) { os_name = p_name; } void EditorExportPlatformPC::set_logo(const Ref &p_logo) { logo = p_logo; } void EditorExportPlatformPC::set_release_64(const String &p_file) { release_file_64 = p_file; } void EditorExportPlatformPC::set_release_32(const String &p_file) { release_file_32 = p_file; } void EditorExportPlatformPC::set_debug_64(const String &p_file) { debug_file_64 = p_file; } void EditorExportPlatformPC::set_debug_32(const String &p_file) { debug_file_32 = p_file; } void EditorExportPlatformPC::add_platform_feature(const String &p_feature) { extra_features.insert(p_feature); } void EditorExportPlatformPC::get_platform_features(List *r_features) { r_features->push_back("pc"); //all pcs support "pc" r_features->push_back("s3tc"); //all pcs support "s3tc" compression r_features->push_back(get_os_name()); //OS name is a feature for (Set::Element *E = extra_features.front(); E; E = E->next()) { r_features->push_back(E->get()); } } EditorExportPlatformPC::EditorExportPlatformPC() { } //////// #if 0 #include "editor/editor_file_system.h" #include "editor/plugins/script_editor_plugin.h" #include "editor_node.h" #include "editor_settings.h" #include "io/config_file.h" #include "io/md5.h" #include "io/resource_loader.h" #include "io/resource_saver.h" #include "io/zip_io.h" #include "io_plugins/editor_texture_import_plugin.h" #include "os/dir_access.h" #include "os/file_access.h" #include "project_settings.h" #include "script_language.h" #include "version.h" String EditorImportPlugin::validate_source_path(const String& p_path) { String gp = ProjectSettings::get_singleton()->globalize_path(p_path); String rp = ProjectSettings::get_singleton()->get_resource_path(); if (!rp.ends_with("/")) rp+="/"; return rp.path_to_file(gp); } String EditorImportPlugin::expand_source_path(const String& p_path) { if (p_path.is_rel_path()) { return ProjectSettings::get_singleton()->get_resource_path().plus_file(p_path).simplify_path(); } else { return p_path; } } String EditorImportPlugin::_validate_source_path(const String& p_path) { return validate_source_path(p_path); } String EditorImportPlugin::_expand_source_path(const String& p_path) { return expand_source_path(p_path); } void EditorImportPlugin::_bind_methods() { ClassDB::bind_method(D_METHOD("validate_source_path","path"),&EditorImportPlugin::_validate_source_path); ClassDB::bind_method(D_METHOD("expand_source_path","path"),&EditorImportPlugin::_expand_source_path); ClassDB::add_virtual_method(get_class_static(),MethodInfo(Variant::STRING,"get_name")); ClassDB::add_virtual_method(get_class_static(),MethodInfo(Variant::STRING,"get_visible_name")); ClassDB::add_virtual_method(get_class_static(),MethodInfo("import_dialog",PropertyInfo(Variant::STRING,"from"))); ClassDB::add_virtual_method(get_class_static(),MethodInfo(Variant::INT,"import",PropertyInfo(Variant::STRING,"path"),PropertyInfo(Variant::OBJECT,"from",PROPERTY_HINT_RESOURCE_TYPE,"ResourceImportMetadata"))); ClassDB::add_virtual_method(get_class_static(),MethodInfo(Variant::POOL_BYTE_ARRAY,"custom_export",PropertyInfo(Variant::STRING,"path"),PropertyInfo(Variant::OBJECT,"platform",PROPERTY_HINT_RESOURCE_TYPE,"EditorExportPlatform"))); ClassDB::add_virtual_method(get_class_static(),MethodInfo("import_from_drop",PropertyInfo(Variant::POOL_STRING_ARRAY,"files"),PropertyInfo(Variant::STRING,"dest_path"))); ClassDB::add_virtual_method(get_class_static(),MethodInfo("reimport_multiple_files",PropertyInfo(Variant::POOL_STRING_ARRAY,"files"))); ClassDB::add_virtual_method(get_class_static(),MethodInfo(Variant::BOOL,"can_reimport_multiple_files")); //BIND_VMETHOD( mi ); } String EditorImportPlugin::get_name() const { if (get_script_instance() && get_script_instance()->has_method("get_name")) { return get_script_instance()->call("get_name"); } ERR_FAIL_V(""); } String EditorImportPlugin::get_visible_name() const { if (get_script_instance() && get_script_instance()->has_method("get_visible_name")) { return get_script_instance()->call("get_visible_name"); } ERR_FAIL_V(""); } void EditorImportPlugin::import_dialog(const String& p_from) { if (get_script_instance() && get_script_instance()->has_method("import_dialog")) { get_script_instance()->call("import_dialog",p_from); return; } ERR_FAIL(); } Error EditorImportPlugin::import(const String& p_path, const Ref& p_from) { if (get_script_instance() && get_script_instance()->has_method("import")) { return Error(get_script_instance()->call("import",p_path,p_from).operator int()); } ERR_FAIL_V(ERR_UNAVAILABLE); } Vector EditorImportPlugin::custom_export(const String& p_path, const Ref &p_platform) { if (get_script_instance() && get_script_instance()->has_method("custom_export")) { get_script_instance()->call("custom_export",p_path,p_platform); } return Vector(); } bool EditorImportPlugin::can_reimport_multiple_files() const { if (get_script_instance() && get_script_instance()->has_method("can_reimport_multiple_files")) { return get_script_instance()->call("can_reimport_multiple_files"); } return false; } void EditorImportPlugin::reimport_multiple_files(const Vector& p_list) { if (get_script_instance() && get_script_instance()->has_method("reimport_multiple_files")) { get_script_instance()->call("reimport_multiple_files",p_list); } } void EditorImportPlugin::import_from_drop(const Vector& p_drop, const String &p_dest_path) { if (get_script_instance() && get_script_instance()->has_method("import_from_drop")) { get_script_instance()->call("import_from_drop",p_drop,p_dest_path); } } EditorImportPlugin::EditorImportPlugin() { } ///////////////////////////////////////////////////////////////////////////////////////////////////// void EditorExportPlugin::_bind_methods() { MethodInfo mi = MethodInfo("custom_export:Variant",PropertyInfo(Variant::STRING,"name"),PropertyInfo(Variant::OBJECT,"platform",PROPERTY_HINT_RESOURCE_TYPE,"EditorExportPlatform")); mi.return_val.type=Variant::POOL_BYTE_ARRAY; BIND_VMETHOD( mi ); } Vector EditorExportPlugin::custom_export(String& p_path,const Ref &p_platform) { if (get_script_instance()) { Variant d = get_script_instance()->call("custom_export",p_path,p_platform); if (d.get_type()==Variant::NIL) return Vector(); if (d.get_type()==Variant::POOL_BYTE_ARRAY) return d; ERR_FAIL_COND_V(d.get_type()!=Variant::DICTIONARY,Vector()); Dictionary dict=d; ERR_FAIL_COND_V(!dict.has("name"),Vector()); ERR_FAIL_COND_V(!dict.has("data"),Vector()); p_path=dict["name"]; return dict["data"]; } return Vector(); } EditorExportPlugin::EditorExportPlugin() { } ///////////////////////////////////////////////////////////////////////////////////////////////////// static void _add_to_list(EditorFileSystemDirectory *p_efsd,Set& r_list) { for(int i=0;iget_subdir_count();i++) { _add_to_list(p_efsd->get_subdir(i),r_list); } for(int i=0;iget_file_count();i++) { r_list.insert(p_efsd->get_file_path(i)); } } struct __EESortDepCmp { _FORCE_INLINE_ bool operator()(const StringName& p_l,const StringName& p_r) const { return p_l.operator String() < p_r.operator String(); } }; static void _edit_files_with_filter(DirAccess *da,const List& p_filters,Set& r_list,bool exclude) { List files; List dirs; da->list_dir_begin(); String f = da->get_next(); while(f!="") { print_line("HOHO: "+f); if (da->current_is_dir()) dirs.push_back(f); else files.push_back(f); f=da->get_next(); } String r = da->get_current_dir().replace("\\","/"); if (!r.ends_with("/")) r+="/"; print_line("AT: "+r); for(List::Element *E=files.front();E;E=E->next()) { String fullpath=r+E->get(); for(const List::Element *F=p_filters.front();F;F=F->next()) { if (fullpath.matchn(F->get())) { String act = TTR("Added:")+" "; if (!exclude) { r_list.insert(fullpath); } else { act = TTR("Removed:")+" "; r_list.erase(fullpath); } print_line(act+fullpath); } } } da->list_dir_end(); for(List::Element *E=dirs.front();E;E=E->next()) { if (E->get().begins_with(".")) continue; da->change_dir(E->get()); _edit_files_with_filter(da,p_filters,r_list,exclude); da->change_dir(".."); } } static void _edit_filter_list(Set& r_list,const String& p_filter,bool exclude) { if (p_filter=="") return; Vector split = p_filter.split(","); List filters; for(int i=0;i& r_list,const String& p_filter) { _edit_filter_list(r_list,p_filter,false); } static void _remove_filter_from_list(Set& r_list,const String& p_filter) { _edit_filter_list(r_list,p_filter,true); } bool EditorExportPlatform::_set(const StringName& p_name, const Variant& p_value) { String n = p_name; if (n=="debug/debugging_enabled") { set_debugging_enabled(p_value); } else { return false; } return true; } bool EditorExportPlatform::_get(const StringName& p_name,Variant &r_ret) const { String n = p_name; if (n=="debug/debugging_enabled") { r_ret=is_debugging_enabled(); } else { return false; } return true; } void EditorExportPlatform::_get_property_list( List *p_list) const { p_list->push_front( PropertyInfo( Variant::BOOL, "debug/debugging_enabled")); } Vector EditorExportPlatform::get_exported_file_default(String& p_fname) const { FileAccess *f = FileAccess::open(p_fname,FileAccess::READ); ERR_FAIL_COND_V(!f,Vector()); Vector ret; ret.resize(f->get_len()); int rbs = f->get_buffer(ret.ptr(),ret.size()); memdelete(f); return ret; } Vector EditorExportPlatform::get_exported_file(String& p_fname) const { Ref ep=EditorImportExport::get_singleton()->get_export_platform(get_name()); for(int i=0;iget_export_plugin_count();i++) { Vector data = EditorImportExport::get_singleton()->get_export_plugin(i)->custom_export(p_fname,ep); if (data.size()) return data; } return get_exported_file_default(p_fname); } Vector EditorExportPlatform::get_dependencies(bool p_bundles) const { Set exported; if (FileAccess::exists("res://project.godot")) exported.insert("res://project.godot"); if (EditorImportExport::get_singleton()->get_export_filter()!=EditorImportExport::EXPORT_SELECTED) { String filter; if (EditorImportExport::get_singleton()->get_export_filter()==EditorImportExport::EXPORT_ALL) { _add_filter_to_list(exported,"*"); } else { _add_to_list(EditorFileSystem::get_singleton()->get_filesystem(),exported); String cf = EditorImportExport::get_singleton()->get_export_custom_filter(); if (cf!="") cf+=","; cf+="*.flags"; _add_filter_to_list(exported,cf); cf = EditorImportExport::get_singleton()->get_export_custom_filter_exclude(); _remove_filter_from_list(exported,cf); } } else { Map > remapped_paths; Set scene_extensions; Set resource_extensions; { List l; /* SceneLoader::get_recognized_extensions(&l); for(List::Element *E=l.front();E;E=E->next()) { scene_extensions.insert(E->get()); } */ ResourceLoader::get_recognized_extensions_for_type("",&l); for(List::Element *E=l.front();E;E=E->next()) { resource_extensions.insert(E->get()); } } List toexport; EditorImportExport::get_singleton()->get_export_file_list(&toexport); print_line("TO EXPORT: "+itos(toexport.size())); for (List::Element *E=toexport.front();E;E=E->next()) { print_line("DEP: "+String(E->get())); exported.insert(E->get()); if (p_bundles && EditorImportExport::get_singleton()->get_export_file_action(E->get())==EditorImportExport::ACTION_BUNDLE) { print_line("NO BECAUSE OF BUNDLE!"); continue; //no dependencies needed to be copied } List testsubs; testsubs.push_back(E->get()); while(testsubs.size()) { //recursive subdep search! List deplist; ResourceLoader::get_dependencies(testsubs.front()->get(),&deplist); testsubs.pop_front(); List subdeps; for (List::Element *F=deplist.front();F;F=F->next()) { StringName dep = F->get(); if (exported.has(dep) || EditorImportExport::get_singleton()->get_export_file_action(dep)!=EditorImportExport::ACTION_NONE) continue; //dependency added or to be added print_line(" SUBDEP: "+String(dep)); exported.insert(dep); testsubs.push_back(dep); } } } String cf = EditorImportExport::get_singleton()->get_export_custom_filter(); if (cf!="") cf+=","; cf+="*.flags"; _add_filter_to_list(exported,cf); cf = EditorImportExport::get_singleton()->get_export_custom_filter_exclude(); _remove_filter_from_list(exported,cf); } Vector ret; ret.resize(exported.size()); int idx=0; for(Set::Element *E=exported.front();E;E=E->next()) { ret[idx++]=E->get(); } SortArray sort; //some platforms work better if this is sorted sort.sort(ret.ptr(),ret.size()); return ret; } /////////////////////////////////////// bool EditorExportPlatform::is_debugging_enabled() const { return debugging_enabled; } void EditorExportPlatform::set_debugging_enabled(bool p_enabled) { debugging_enabled = p_enabled; } bool EditorExportPlatformPC::_set(const StringName& p_name, const Variant& p_value) { String n = p_name; if (n=="custom_binary/release") { custom_release_binary=p_value; } else if (n=="custom_binary/debug") { custom_debug_binary=p_value; } else if (n=="resources/pack_mode") { export_mode=ExportMode(int(p_value)); } else if (n=="resources/bundle_dependencies_(for_optical_disc)") { bundle=p_value; } else if (n=="binary/64_bits") { use64=p_value; } else return false; return true; } bool EditorExportPlatformPC::_get(const StringName& p_name,Variant &r_ret) const { String n = p_name; if (n=="custom_binary/release") { r_ret=custom_release_binary; } else if (n=="custom_binary/debug") { r_ret=custom_debug_binary; } else if (n=="resources/pack_mode") { r_ret=export_mode; } else if (n=="resources/bundle_dependencies_(for_optical_disc)") { r_ret=bundle; } else if (n=="binary/64_bits") { r_ret=use64; } else return false; return true; } void EditorExportPlatformPC::_get_property_list( List *p_list) const { p_list->push_back( PropertyInfo( Variant::STRING, "custom_binary/debug", PROPERTY_HINT_GLOBAL_FILE,binary_extension)); p_list->push_back( PropertyInfo( Variant::STRING, "custom_binary/release", PROPERTY_HINT_GLOBAL_FILE,binary_extension)); p_list->push_back( PropertyInfo( Variant::INT, "resources/pack_mode", PROPERTY_HINT_ENUM,"Pack into executable,Pack into binary file (.pck),Pack into archive file (.zip)")); p_list->push_back( PropertyInfo( Variant::BOOL, "resources/bundle_dependencies_(for_optical_disc)")); p_list->push_back( PropertyInfo( Variant::BOOL, "binary/64_bits")); } static void _exp_add_dep(Map > &deps,const StringName& p_path) { if (deps.has(p_path)) return; //already done deps.insert(p_path,List()); List &deplist=deps[p_path]; Set depset; List dl; ResourceLoader::get_dependencies(p_path,&dl); //added in order so child dependencies are always added bfore parent dependencies for (List::Element *E=dl.front();E;E=E->next()) { if (!deps.has(E->get())) _exp_add_dep(deps,E->get()); for(List::Element *F=deps[E->get()].front();F;F=F->next()) { if (!depset.has(F->get())) { depset.insert(F->get()); deplist.push_back(F->get()); } } if (!depset.has(E->get())) { depset.insert(E->get()); deplist.push_back(E->get()); } } } Error EditorExportPlatform::export_project_files(EditorExportSaveFunction p_func, void* p_udata,bool p_make_bundles) { /* ALL FILES AND DEPENDENCIES */ Vector files=get_dependencies(p_make_bundles); Map > deps; if (false) { for(int i=0;i groups; EditorImportExport::get_singleton()->image_export_get_groups(&groups); Map remap_files; Set saved; int counter=0; for(List::Element *E=groups.front();E;E=E->next()) { if (!EditorImportExport::get_singleton()->image_export_group_get_make_atlas(E->get())) continue; //uninterested, only process for atlas! List atlas_images; EditorImportExport::get_singleton()->image_export_get_images_in_group(E->get(),&atlas_images); atlas_images.sort_custom(); for (List::Element *F=atlas_images.front();F;) { List::Element *N=F->next(); if (!FileAccess::exists(F->get())) { atlas_images.erase(F); } F=N; } if (atlas_images.size()<=1) continue; int group_format=0; float group_lossy_quality=EditorImportExport::get_singleton()->image_export_group_get_lossy_quality(E->get()); int group_shrink=EditorImportExport::get_singleton()->image_export_group_get_shrink(E->get()); group_shrink*=EditorImportExport::get_singleton()->get_export_image_shrink(); switch(EditorImportExport::get_singleton()->image_export_group_get_image_action(E->get())) { case EditorImportExport::IMAGE_ACTION_KEEP: case EditorImportExport::IMAGE_ACTION_NONE: { switch(EditorImportExport::get_singleton()->get_export_image_action()) { case EditorImportExport::IMAGE_ACTION_NONE: { group_format=EditorTextureImportPlugin::IMAGE_FORMAT_COMPRESS_DISK_LOSSLESS; //? } break; //use default case EditorImportExport::IMAGE_ACTION_COMPRESS_DISK: { group_format=EditorTextureImportPlugin::IMAGE_FORMAT_COMPRESS_DISK_LOSSY; } break; //use default case EditorImportExport::IMAGE_ACTION_COMPRESS_RAM: { group_format=EditorTextureImportPlugin::IMAGE_FORMAT_COMPRESS_RAM; } break; //use default } group_lossy_quality=EditorImportExport::get_singleton()->get_export_image_quality(); } break; //use default case EditorImportExport::IMAGE_ACTION_COMPRESS_DISK: { group_format=EditorTextureImportPlugin::IMAGE_FORMAT_COMPRESS_DISK_LOSSY; } break; //use default case EditorImportExport::IMAGE_ACTION_COMPRESS_RAM: { group_format=EditorTextureImportPlugin::IMAGE_FORMAT_COMPRESS_RAM; } break; //use default } String image_list_md5; { MD5_CTX ctx; MD5Init(&ctx); for (List::Element *F=atlas_images.front();F;F=F->next()) { String p = F->get(); MD5Update(&ctx,(unsigned char*)p.utf8().get_data(),p.utf8().length()); } MD5Final(&ctx); image_list_md5=String::md5(ctx.digest); } //ok see if cached String md5; bool atlas_valid=true; String atlas_name; { MD5_CTX ctx; MD5Init(&ctx); String path = ProjectSettings::get_singleton()->get_resource_path()+"::"+String(E->get())+"::"+get_name(); MD5Update(&ctx,(unsigned char*)path.utf8().get_data(),path.utf8().length()); MD5Final(&ctx); md5 = String::md5(ctx.digest); } FileAccess *f=NULL; if (!FileAccess::exists(EditorSettings::get_singleton()->get_settings_path()+"/tmp/atlas-"+md5)) { print_line("NO MD5 INVALID"); atlas_valid=false; } if (atlas_valid) f=FileAccess::open(EditorSettings::get_singleton()->get_settings_path()+"/tmp/atlas-"+md5,FileAccess::READ); if (atlas_valid) { //compare options /*Dictionary options; options.parse_json(f->get_line()); if (!options.has("lossy_quality") || float(options["lossy_quality"])!=group_lossy_quality) atlas_valid=false; else if (!options.has("shrink") || int(options["shrink"])!=group_shrink) atlas_valid=false; else if (!options.has("image_format") || int(options["image_format"])!=group_format) atlas_valid=false; if (!atlas_valid) print_line("JSON INVALID"); */ } if (atlas_valid) { //check md5 of list of image /names/ if (f->get_line().strip_edges()!=image_list_md5) { atlas_valid=false; print_line("IMAGE MD5 INVALID!"); } } Vector rects; bool resave_deps=false; if (atlas_valid) { //check if images were not modified for (List::Element *F=atlas_images.front();F;F=F->next()) { Vector slices = f->get_line().strip_edges().split("::"); if (slices.size()!=10) { atlas_valid=false; print_line("CAN'T SLICE IN 10"); break; } uint64_t mod_time = slices[0].to_int64(); uint64_t file_mod_time = FileAccess::get_modified_time(F->get()); if (mod_time!=file_mod_time) { String image_md5 = slices[1]; String file_md5 = FileAccess::get_md5(F->get()); if (image_md5!=file_md5) { atlas_valid=false; print_line("IMAGE INVALID "+slices[0]); break; } else { resave_deps=true; } } if (atlas_valid) { //push back region and margin rects.push_back(Rect2(slices[2].to_float(),slices[3].to_float(),slices[4].to_float(),slices[5].to_float())); rects.push_back(Rect2(slices[6].to_float(),slices[7].to_float(),slices[8].to_float(),slices[9].to_float())); } } } if (f) { memdelete(f); f=NULL; } print_line("ATLAS VALID? "+itos(atlas_valid)+" RESAVE DEPS? "+itos(resave_deps)); if (!atlas_valid) { rects.clear(); //oh well, atlas is not valid. need to make new one.... String dst_file = EditorSettings::get_singleton()->get_settings_path()+"/tmp/atlas-"+md5+".tex"; Ref imd = memnew( ResourceImportMetadata ); //imd->set_editor(); for (List::Element *F=atlas_images.front();F;F=F->next()) { imd->add_source(EditorImportPlugin::validate_source_path(F->get()),FileAccess::get_md5(F->get())); } imd->set_option("format",group_format); int flags=0; if (ProjectSettings::get_singleton()->get("image_loader/filter")) flags|=EditorTextureImportPlugin::IMAGE_FLAG_FILTER; if (!ProjectSettings::get_singleton()->get("image_loader/gen_mipmaps")) flags|=EditorTextureImportPlugin::IMAGE_FLAG_NO_MIPMAPS; if (!ProjectSettings::get_singleton()->get("image_loader/repeat")) flags|=EditorTextureImportPlugin::IMAGE_FLAG_REPEAT; flags|=EditorTextureImportPlugin::IMAGE_FLAG_FIX_BORDER_ALPHA; imd->set_option("flags",flags); imd->set_option("quality",group_lossy_quality); imd->set_option("atlas",true); imd->set_option("crop",true); imd->set_option("shrink",group_shrink); Ref plugin = EditorImportExport::get_singleton()->get_import_plugin_by_name("texture"); Error err = plugin->import2(dst_file,imd,get_image_compression(),true); if (err) { EditorNode::add_io_error(TTR("Error saving atlas:")+" "+dst_file.get_file()); return ERR_CANT_CREATE; } ERR_FAIL_COND_V(imd->get_option("rects")==Variant(),ERR_BUG); Array r_rects=imd->get_option("rects"); rects.resize(r_rects.size()); for(int i=0;iget_settings_path()+"/tmp/atlas-"+md5,FileAccess::WRITE); Dictionary options; options["lossy_quality"]=group_lossy_quality; options["shrink"]=EditorImportExport::get_singleton()->image_export_group_get_shrink(E->get()); options["image_format"]=group_format; //f->store_line(options.to_json()); f->store_line(image_list_md5); } //go through all ATEX files { Ref atlas = memnew( ImageTexture ); //fake atlas! String atlas_path="res://atlas-"+md5+".tex"; atlas->set_path(atlas_path); int idx=0; for (List::Element *F=atlas_images.front();F;F=F->next()) { String p = F->get(); Ref atex = memnew(AtlasTexture); atex->set_atlas(atlas); Rect2 region=rects[idx++]; Rect2 margin=rects[idx++]; atex->set_region(region); atex->set_margin(margin); String path = EditorSettings::get_singleton()->get_settings_path()+"/tmp/tmpatlas.atlastex"; Error err = ResourceSaver::save(path,atex); if (err!=OK) { EditorNode::add_io_error(TTR("Could not save atlas subtexture:")+" "+path); return ERR_CANT_CREATE; } Vector data = FileAccess::get_file_as_array(path); String dst_path = F->get().operator String().get_basename()+".atlastex"; err = p_func(p_udata,dst_path,data,counter++,files.size()); saved.insert(dst_path); if (err) return err; if (f) { //recreating deps.. String depline; //depline=String(F->get())+"::"+itos(FileAccess::get_modified_time(F->get()))+"::"+FileAccess::get_md5(F->get()); name unnecessary by top md5 depline=itos(FileAccess::get_modified_time(F->get()))+"::"+FileAccess::get_md5(F->get()); depline+="::"+itos(region.pos.x)+"::"+itos(region.pos.y)+"::"+itos(region.size.x)+"::"+itos(region.size.y); depline+="::"+itos(margin.pos.x)+"::"+itos(margin.pos.y)+"::"+itos(margin.size.x)+"::"+itos(margin.size.y); f->store_line(depline); } remap_files[F->get()]=dst_path; } Vector atlas_data = FileAccess::get_file_as_array(EditorSettings::get_singleton()->get_settings_path()+"/tmp/atlas-"+md5+".tex"); Error err = p_func(p_udata,atlas_path,atlas_data,counter,files.size()); saved.insert(atlas_path); if (err) return err; } if (f) { memdelete(f); } } StringName engine_cfg="res://project.godot"; StringName boot_splash; { String splash=ProjectSettings::get_singleton()->get("application/boot_splash"); //avoid splash from being converted splash=splash.strip_edges(); if (splash!=String()) { if (!splash.begins_with("res://")) splash="res://"+splash; splash=splash.simplify_path(); boot_splash=splash; } } StringName custom_cursor; { String splash=ProjectSettings::get_singleton()->get("display/custom_mouse_cursor"); //avoid splash from being converted splash=splash.strip_edges(); if (splash!=String()) { if (!splash.begins_with("res://")) splash="res://"+splash; splash=splash.simplify_path(); custom_cursor=splash; } } for(int i=0;i buf; if (src==boot_splash || src==custom_cursor) buf = get_exported_file_default(src); //bootsplash must be kept if used else buf = get_exported_file(src); ERR_CONTINUE( saved.has(src) ); Error err = p_func(p_udata,src,buf,counter++,files.size()); if (err) return err; saved.insert(src); if (src!=String(files[i])) remap_files[files[i]]=src; } { //make binary project.godot config Map custom; if (remap_files.size()) { Vector remapsprop; for(Map::Element *E=remap_files.front();E;E=E->next()) { print_line("REMAP: "+String(E->key())+" -> "+E->get()); remapsprop.push_back(E->key()); remapsprop.push_back(E->get()); } custom["remap/all"]=remapsprop; } //add presaved dependencies for(Map >::Element *E=deps.front();E;E=E->next()) { if (E->get().size()==0) continue; //no deps String key; Vector deps; //if bundle continue (when bundles supported obviously) if (remap_files.has(E->key())) { key=remap_files[E->key()]; } else { key=E->key(); } deps.resize(E->get().size()); int i=0; for(List::Element *F=E->get().front();F;F=F->next()) { deps[i++]=F->get(); print_line(" -"+String(F->get())); } NodePath prop(deps,true,String()); //seems best to use this for performance custom["deps/"+key.md5_text()]=prop; } String remap_file="project.binary"; String engine_cfb =EditorSettings::get_singleton()->get_settings_path()+"/tmp/tmp"+remap_file; ProjectSettings::get_singleton()->save_custom(engine_cfb,custom); Vector data = FileAccess::get_file_as_array(engine_cfb); Error err = p_func(p_udata,"res://"+remap_file,data,counter,files.size()); if (err) return err; } return OK; } static int _get_pad(int p_alignment, int p_n) { int rest = p_n % p_alignment; int pad = 0; if (rest > 0) { pad = p_alignment - rest; }; return pad; }; void EditorExportPlatform::gen_export_flags(Vector &r_flags, int p_flags) { String host = EditorSettings::get_singleton()->get("network/debug/remote_host"); int remote_port = (int)EditorSettings::get_singleton()->get("network/debug/remote_port"); if (p_flags&EXPORT_REMOTE_DEBUG_LOCALHOST) host="localhost"; if (p_flags&EXPORT_DUMB_CLIENT) { int port = EditorSettings::get_singleton()->get("filesystem/file_server/port"); String passwd = EditorSettings::get_singleton()->get("filesystem/file_server/password"); r_flags.push_back("-rfs"); r_flags.push_back(host+":"+itos(port)); if (passwd!="") { r_flags.push_back("-rfs_pass"); r_flags.push_back(passwd); } } if (p_flags&EXPORT_REMOTE_DEBUG) { r_flags.push_back("-rdebug"); r_flags.push_back(host+":"+String::num(remote_port)); List breakpoints; ScriptEditor::get_singleton()->get_breakpoints(&breakpoints); if (breakpoints.size()) { r_flags.push_back("-bp"); String bpoints; for(const List::Element *E=breakpoints.front();E;E=E->next()) { bpoints+=E->get().replace(" ","%20"); if (E->next()) bpoints+=","; } r_flags.push_back(bpoints); } } if (p_flags&EXPORT_VIEW_COLLISONS) { r_flags.push_back("-debugcol"); } if (p_flags&EXPORT_VIEW_NAVIGATION) { r_flags.push_back("-debugnav"); } } Error EditorExportPlatform::save_pack_file(void *p_userdata,const String& p_path, const Vector& p_data,int p_file,int p_total) { PackData *pd = (PackData*)p_userdata; CharString cs=p_path.utf8(); pd->f->store_32(cs.length()); pd->f->store_buffer((uint8_t*)cs.get_data(),cs.length()); TempData td; td.pos=pd->f->get_pos(); td.ofs=pd->ftmp->get_pos(); td.size=p_data.size(); pd->file_ofs.push_back(td); pd->f->store_64(0); //ofs pd->f->store_64(0); //size { MD5_CTX ctx; MD5Init(&ctx); MD5Update(&ctx,(unsigned char*)p_data.ptr(),p_data.size()); MD5Final(&ctx); pd->f->store_buffer(ctx.digest,16); } pd->ep->step(TTR("Storing File:")+" "+p_path,2+p_file*100/p_total,false); pd->count++; pd->ftmp->store_buffer(p_data.ptr(),p_data.size()); if (pd->alignment > 1) { int pad = _get_pad(pd->alignment, pd->ftmp->get_pos()); for (int i=0; iftmp->store_8(0); }; }; return OK; } Error EditorExportPlatform::save_zip_file(void *p_userdata,const String& p_path, const Vector& p_data,int p_file,int p_total) { String path=p_path.replace_first("res://",""); ZipData *zd = (ZipData*)p_userdata; zipFile zip=(zipFile)zd->zip; zipOpenNewFileInZip(zip, path.utf8().get_data(), NULL, NULL, 0, NULL, 0, NULL, Z_DEFLATED, Z_DEFAULT_COMPRESSION); zipWriteInFileInZip(zip,p_data.ptr(),p_data.size()); zipCloseFileInZip(zip); zd->ep->step(TTR("Storing File:")+" "+p_path,2+p_file*100/p_total,false); zd->count++; return OK; } Error EditorExportPlatform::save_zip(const String& p_path, bool p_make_bundles) { EditorProgress ep("savezip",TTR("Packing"),102); //FileAccess *tmp = FileAccess::open(tmppath,FileAccess::WRITE); FileAccess *src_f; zlib_filefunc_def io = zipio_create_io_from_file(&src_f); zipFile zip=zipOpen2(p_path.utf8().get_data(),APPEND_STATUS_CREATE,NULL,&io); ZipData zd; zd.count=0; zd.ep=&ep; zd.zip=zip; Error err = export_project_files(save_zip_file,&zd,p_make_bundles); zipClose(zip,NULL); return err; } Error EditorExportPlatform::save_pack(FileAccess *dst,bool p_make_bundles, int p_alignment) { EditorProgress ep("savepack",TTR("Packing"),102); String tmppath = EditorSettings::get_singleton()->get_settings_path()+"/tmp/packtmp"; FileAccess *tmp = FileAccess::open(tmppath,FileAccess::WRITE); uint64_t ofs_begin = dst->get_pos(); dst->store_32(0x43504447); //GDPK dst->store_32(0); //pack version dst->store_32(VERSION_MAJOR); dst->store_32(VERSION_MINOR); dst->store_32(0); //hmph for(int i=0;i<16;i++) { //reserved dst->store_32(0); } size_t fcountpos = dst->get_pos(); dst->store_32(0); PackData pd; pd.ep=&ep; pd.f=dst; pd.ftmp=tmp; pd.count=0; pd.alignment = p_alignment; Error err = export_project_files(save_pack_file,&pd,p_make_bundles); memdelete(tmp); if (err) return err; if (p_alignment > 1) { int pad = _get_pad(p_alignment, dst->get_pos()); for (int i=0; istore_8(0); }; }; size_t ofsplus = dst->get_pos(); //append file tmp = FileAccess::open(tmppath,FileAccess::READ); ERR_FAIL_COND_V(!tmp,ERR_CANT_OPEN;) const int bufsize=16384; uint8_t buf[bufsize]; while(true) { int got = tmp->get_buffer(buf,bufsize); if (got<=0) break; dst->store_buffer(buf,got); } memdelete(tmp); dst->store_64(dst->get_pos()-ofs_begin); dst->store_32(0x43504447); //GDPK //fix offsets dst->seek(fcountpos); dst->store_32(pd.count); for(int i=0;iseek(pd.file_ofs[i].pos); dst->store_64(pd.file_ofs[i].ofs+ofsplus); dst->store_64(pd.file_ofs[i].size); } return OK; } EditorExportPlatform::EditorExportPlatform() { debugging_enabled = true; } Error EditorExportPlatformPC::export_project(const String& p_path, bool p_debug, int p_flags) { EditorProgress ep("export",vformat(TTR("Exporting for %s"),get_name()),102); const int BUFSIZE = 32768; ep.step(TTR("Setting Up.."),0); String exe_path=""; if (p_debug) exe_path=custom_debug_binary; else exe_path=custom_release_binary; if (exe_path=="") { String fname; if (use64) { if (p_debug) fname=debug_binary64; else fname=release_binary64; } else { if (p_debug) fname=debug_binary32; else fname=release_binary32; } String err=""; exe_path=find_export_template(fname,&err); if (exe_path=="") { EditorNode::add_io_error(err); return ERR_FILE_CANT_READ; } } FileAccess *src_exe=FileAccess::open(exe_path,FileAccess::READ); if (!src_exe) { EditorNode::add_io_error("Couldn't read source executable at:\n "+exe_path); return ERR_FILE_CANT_READ; } FileAccess *dst=FileAccess::open(p_path,FileAccess::WRITE); if (!dst) { EditorNode::add_io_error("Can't copy executable file to:\n "+p_path); return ERR_FILE_CANT_WRITE; } uint8_t buff[32768]; while(true) { int c = src_exe->get_buffer(buff,BUFSIZE); if (c>0) { dst->store_buffer(buff,c); } else { break; } } String dstfile = p_path.replace_first("res://","").replace("\\","/"); if (export_mode!=EXPORT_EXE) { String dstfile_extension=export_mode==EXPORT_ZIP?".zip":".pck"; if (dstfile.find("/")!=-1) dstfile=dstfile.get_base_dir()+"/data"+dstfile_extension; else dstfile="data"+dstfile_extension; if (export_mode==EXPORT_PACK) { memdelete(dst); dst=FileAccess::open(dstfile,FileAccess::WRITE); if (!dst) { EditorNode::add_io_error("Can't write data pack to:\n "+p_path); return ERR_FILE_CANT_WRITE; } } } memdelete(src_exe); Error err = export_mode==EXPORT_ZIP?save_zip(dstfile,bundle):save_pack(dst,bundle); memdelete(dst); return err; } void EditorExportPlatformPC::set_binary_extension(const String& p_extension) { binary_extension=p_extension; } EditorExportPlatformPC::EditorExportPlatformPC() { export_mode=EXPORT_PACK; use64=true; } /////////////////////////////////////////////////////////////////////////////////////////////////////// EditorImportExport* EditorImportExport::singleton=NULL; void EditorImportExport::add_import_plugin(const Ref& p_plugin) { // Need to make sure the name is unique if we are going to lookup by it ERR_FAIL_COND(by_idx.has(p_plugin->get_name())); by_idx[ p_plugin->get_name() ]=plugins.size(); plugins.push_back(p_plugin); } void EditorImportExport::remove_import_plugin(const Ref& p_plugin) { String plugin_name = p_plugin->get_name(); // Keep the indices the same // Find the index of the target plugin ERR_FAIL_COND(!by_idx.has(plugin_name)); int idx = by_idx[plugin_name]; int last_idx = plugins.size() - 1; // Swap the last plugin and the target one SWAP(plugins[idx], plugins[last_idx]); // Update the index of the old last one by_idx[plugins[idx]->get_name()] = idx; // Remove the target plugin's by_idx entry by_idx.erase(plugin_name); // Erase the plugin plugins.remove(last_idx); } int EditorImportExport::get_import_plugin_count() const{ return plugins.size(); } Ref EditorImportExport::get_import_plugin(int p_idx) const{ ERR_FAIL_INDEX_V(p_idx,plugins.size(),Ref()); return plugins[p_idx]; } Ref EditorImportExport::get_import_plugin_by_name(const String& p_string) const{ ERR_FAIL_COND_V( !by_idx.has(p_string), Ref()); return plugins[ by_idx[p_string] ]; } void EditorImportExport::add_export_plugin(const Ref& p_plugin) { ERR_FAIL_COND( p_plugin.is_null() ); export_plugins.push_back(p_plugin); } void EditorImportExport::remove_export_plugin(const Ref& p_plugin) { ERR_FAIL_COND( p_plugin.is_null() ); export_plugins.erase(p_plugin); } int EditorImportExport::get_export_plugin_count() const{ return export_plugins.size(); } Ref EditorImportExport::get_export_plugin(int p_idx) const{ ERR_FAIL_INDEX_V(p_idx,export_plugins.size(),Ref()); return export_plugins[p_idx]; } void EditorImportExport::set_export_file_action(const StringName& p_file, FileAction p_action) { if (p_action==ACTION_NONE) { files.erase(p_file); } else { files[p_file]=p_action; } } EditorImportExport::FileAction EditorImportExport::get_export_file_action(const StringName& p_file) const{ if (files.has(p_file)) return files[p_file]; return ACTION_NONE; } void EditorImportExport::get_export_file_list(List *p_files){ for (Map::Element *E=files.front();E;E=E->next()) { p_files->push_back(E->key()); } } void EditorImportExport::add_export_platform(const Ref& p_export) { exporters[p_export->get_name()]=p_export; } void EditorImportExport::get_export_platforms(List *r_platforms) { for (Map >::Element *E=exporters.front();E;E=E->next()) { r_platforms->push_back(E->key()); } } Ref EditorImportExport::get_export_platform(const StringName& p_platform) { if (exporters.has(p_platform)) { return exporters[p_platform]; } else { return Ref(); } } bool EditorImportExport::poll_export_platforms() { bool changed=false; for (Map >::Element *E=exporters.front();E;E=E->next()) { if (E->get()->poll_devices()) changed=true; } return changed; } void EditorImportExport::set_export_filter(ExportFilter p_enable) { export_filter=p_enable; } EditorImportExport::ExportFilter EditorImportExport::get_export_filter() const{ return export_filter; } void EditorImportExport::set_export_custom_filter(const String& p_custom_filter){ export_custom_filter=p_custom_filter; } void EditorImportExport::set_export_custom_filter_exclude(const String& p_custom_filter){ export_custom_filter_exclude=p_custom_filter; } String EditorImportExport::get_export_custom_filter() const{ return export_custom_filter; } String EditorImportExport::get_export_custom_filter_exclude() const{ return export_custom_filter_exclude; } void EditorImportExport::set_export_image_action(ImageAction p_action) { image_action=p_action; } EditorImportExport::ImageAction EditorImportExport::get_export_image_action() const{ return image_action; } void EditorImportExport::set_export_image_shrink(float p_shrink) { image_shrink=p_shrink; } float EditorImportExport::get_export_image_shrink() const{ return image_shrink; } void EditorImportExport::set_export_image_quality(float p_quality){ image_action_compress_quality=p_quality; } float EditorImportExport::get_export_image_quality() const{ return image_action_compress_quality; } void EditorImportExport::image_export_group_create(const StringName& p_name) { ERR_FAIL_COND(image_groups.has(p_name)); ImageGroup ig; ig.action=IMAGE_ACTION_NONE; //default ig.make_atlas=false; ig.shrink=1; ig.lossy_quality=0.7; image_groups[p_name]=ig; } bool EditorImportExport::image_export_has_group(const StringName& p_name) const { return image_groups.has(p_name); } void EditorImportExport::image_export_get_groups(List *r_name) const { for (Map::Element *E=image_groups.front();E;E=E->next()) { r_name->push_back(E->key()); } } void EditorImportExport::image_export_group_remove(const StringName& p_name){ ERR_FAIL_COND(!image_groups.has(p_name)); image_groups.erase(p_name); } void EditorImportExport::image_export_group_set_image_action(const StringName& p_export_group,ImageAction p_action){ ERR_FAIL_COND(!image_groups.has(p_export_group)); image_groups[p_export_group].action=p_action; } EditorImportExport::ImageAction EditorImportExport::image_export_group_get_image_action(const StringName& p_export_group) const{ ERR_FAIL_COND_V(!image_groups.has(p_export_group),IMAGE_ACTION_NONE); return image_groups[p_export_group].action; } void EditorImportExport::image_export_group_set_make_atlas(const StringName& p_export_group,bool p_make){ ERR_FAIL_COND(!image_groups.has(p_export_group)); image_groups[p_export_group].make_atlas=p_make; } bool EditorImportExport::image_export_group_get_make_atlas(const StringName& p_export_group) const{ ERR_FAIL_COND_V(!image_groups.has(p_export_group),false); return image_groups[p_export_group].make_atlas; } void EditorImportExport::image_export_group_set_shrink(const StringName& p_export_group,float p_amount){ ERR_FAIL_COND(!image_groups.has(p_export_group)); image_groups[p_export_group].shrink=p_amount; } float EditorImportExport::image_export_group_get_shrink(const StringName& p_export_group) const{ ERR_FAIL_COND_V(!image_groups.has(p_export_group),1); return image_groups[p_export_group].shrink; } void EditorImportExport::image_export_group_set_lossy_quality(const StringName& p_export_group,float p_amount){ ERR_FAIL_COND(!image_groups.has(p_export_group)); image_groups[p_export_group].lossy_quality=p_amount; } float EditorImportExport::image_export_group_get_lossy_quality(const StringName& p_export_group) const{ ERR_FAIL_COND_V(!image_groups.has(p_export_group),1); return image_groups[p_export_group].lossy_quality; } StringName EditorImportExport::image_get_export_group(const StringName& p_image) const { if (image_group_files.has(p_image)) return image_group_files[p_image]; else return StringName(); } void EditorImportExport::image_add_to_export_group(const StringName& p_image,const StringName& p_export_group) { bool emptygroup = String(p_export_group)==String(); ERR_FAIL_COND(!emptygroup && !image_groups.has(p_export_group)); if (emptygroup) image_group_files.erase(p_image); else image_group_files[p_image]=p_export_group; } void EditorImportExport::image_export_get_images_in_group(const StringName& p_group,List *r_images) const { for (Map::Element *E=image_group_files.front();E;E=E->next()) { if (p_group==E->get()) r_images->push_back(E->key()); } } void EditorImportExport::set_convert_text_scenes(bool p_convert) { convert_text_scenes=p_convert; } bool EditorImportExport::get_convert_text_scenes() const{ return convert_text_scenes; } void EditorImportExport::load_config() { Ref cf = memnew( ConfigFile ); Error err = cf->load("res://export.cfg"); if (err!=OK) return; //no export config to be loaded! export_custom_filter=cf->get_value("export_filter","filter"); export_custom_filter_exclude=cf->get_value("export_filter","filter_exclude"); String t=cf->get_value("export_filter","type"); if (t=="selected") export_filter=EXPORT_SELECTED; else if (t=="resources") export_filter=EXPORT_RESOURCES; else if (t=="all") export_filter=EXPORT_ALL; if (cf->has_section("convert_images")) { String ci = "convert_images"; String action = cf->get_value(ci,"action"); if (action=="none") image_action=IMAGE_ACTION_NONE; else if (action=="compress_ram") image_action=IMAGE_ACTION_COMPRESS_RAM; else if (action=="compress_disk") image_action=IMAGE_ACTION_COMPRESS_DISK; image_action_compress_quality = cf->get_value(ci,"compress_quality"); if (cf->has_section_key(ci,"shrink")) image_shrink = cf->get_value(ci,"shrink"); else image_shrink=1; String formats=cf->get_value(ci,"formats"); Vector f = formats.split(","); image_formats.clear(); for(int i=0;ihas_section("convert_scenes")) { convert_text_scenes = cf->get_value("convert_scenes","convert_text_scenes"); } if (cf->has_section("export_filter_files")) { String eff = "export_filter_files"; List k; cf->get_section_keys(eff,&k); for(List::Element *E=k.front();E;E=E->next()) { String val = cf->get_value(eff,E->get()); if (val=="copy") { files[E->get()]=ACTION_COPY; } else if (val=="bundle") { files[E->get()]=ACTION_BUNDLE; } } } List sect; cf->get_sections(§); for(List::Element *E=sect.front();E;E=E->next()) { String s = E->get(); if (!s.begins_with("platform:")) continue; String p = s.substr(s.find(":")+1,s.length()); if (!exporters.has(p)) continue; Ref ep = exporters[p]; if (!ep.is_valid()) { continue; } List keys; cf->get_section_keys(s,&keys); for(List::Element *F=keys.front();F;F=F->next()) { ep->set(F->get(),cf->get_value(s,F->get())); } } //save image groups if (cf->has_section("image_groups")) { sect.clear(); cf->get_section_keys("image_groups",§); for(List::Element *E=sect.front();E;E=E->next()) { Dictionary d = cf->get_value("image_groups",E->get()); ImageGroup g; g.action=IMAGE_ACTION_NONE; g.make_atlas=false; g.lossy_quality=0.7; g.shrink=1; if (d.has("action")) { String action=d["action"]; if (action=="compress_ram") g.action=IMAGE_ACTION_COMPRESS_RAM; else if (action=="compress_disk") g.action=IMAGE_ACTION_COMPRESS_DISK; else if (action=="keep") g.action=IMAGE_ACTION_KEEP; } if (d.has("atlas")) g.make_atlas=d["atlas"]; if (d.has("lossy_quality")) g.lossy_quality=d["lossy_quality"]; if (d.has("shrink")) { g.shrink=d["shrink"]; g.shrink=CLAMP(g.shrink,1,8); } image_groups[E->get()]=g; } if (cf->has_section_key("image_group_files","files")) { Vector sa=cf->get_value("image_group_files","files"); if (sa.size()%2==0) { for(int i=0;ihas_section("script")) { if (cf->has_section_key("script","action")) { String action = cf->get_value("script","action"); if (action=="compile") script_action=SCRIPT_ACTION_COMPILE; else if (action=="encrypt") script_action=SCRIPT_ACTION_ENCRYPT; else script_action=SCRIPT_ACTION_NONE; } if (cf->has_section_key("script","encrypt_key")) { script_key = cf->get_value("script","encrypt_key"); } } if (cf->has_section("convert_samples")) { if (cf->has_section_key("convert_samples","action")) { String action = cf->get_value("convert_samples","action"); if (action=="none") { sample_action=SAMPLE_ACTION_NONE; } else if (action=="compress_ram") { sample_action=SAMPLE_ACTION_COMPRESS_RAM; } } if (cf->has_section_key("convert_samples","max_hz")) sample_action_max_hz=cf->get_value("convert_samples","max_hz"); if (cf->has_section_key("convert_samples","trim")) sample_action_trim=cf->get_value("convert_samples","trim"); } } void EditorImportExport::save_config() { Ref cf = memnew( ConfigFile ); switch(export_filter) { case EXPORT_SELECTED: cf->set_value("export_filter","type","selected"); break; case EXPORT_RESOURCES: cf->set_value("export_filter","type","resources"); break; case EXPORT_ALL: cf->set_value("export_filter","type","all"); break; } cf->set_value("export_filter","filter",export_custom_filter); cf->set_value("export_filter", "filter_exclude",export_custom_filter_exclude); String file_action_section = "export_filter_files"; for (Map::Element *E=files.front();E;E=E->next()) { String f=E->key(); String a; switch (E->get()) { case ACTION_NONE: {} case ACTION_COPY: a="copy"; break; case ACTION_BUNDLE: a="bundle"; break; } cf->set_value(file_action_section,f,a); } for (Map >::Element *E=exporters.front();E;E=E->next()) { String pname = "platform:"+String(E->key()); Ref ep = E->get(); List pl; ep->get_property_list(&pl); for (List::Element *F=pl.front();F;F=F->next()) { cf->set_value(pname,F->get().name,ep->get(F->get().name)); } } switch(image_action) { case IMAGE_ACTION_NONE: cf->set_value("convert_images","action","none"); break; case IMAGE_ACTION_COMPRESS_RAM: cf->set_value("convert_images","action","compress_ram"); break; case IMAGE_ACTION_COMPRESS_DISK: cf->set_value("convert_images","action","compress_disk"); break; } cf->set_value("convert_images","shrink",image_shrink); cf->set_value("convert_images","compress_quality",image_action_compress_quality); String formats; for(Set::Element *E=image_formats.front();E;E=E->next()) { if (E!=image_formats.front()) formats+=","; formats+=E->get(); } cf->set_value("convert_images","formats",formats); //save image groups for(Map::Element *E=image_groups.front();E;E=E->next()) { Dictionary d; switch(E->get().action) { case IMAGE_ACTION_NONE: d["action"]="default"; break; case IMAGE_ACTION_COMPRESS_RAM: d["action"]="compress_ram"; break; case IMAGE_ACTION_COMPRESS_DISK: d["action"]="compress_disk"; break; case IMAGE_ACTION_KEEP: d["action"]="keep"; break; } d["atlas"]=E->get().make_atlas; d["shrink"]=E->get().shrink; d["lossy_quality"]=E->get().lossy_quality; cf->set_value("image_groups",E->key(),d); } if (image_groups.size() && image_group_files.size()){ Vector igfkeys; igfkeys.resize(image_group_files.size()); int idx=0; for (Map::Element *E=image_group_files.front();E;E=E->next()) { igfkeys[idx++]=E->key(); } igfkeys.sort(); Vector igfsave; igfsave.resize(image_group_files.size()*2); idx=0; for (int i=0;iset_value("image_group_files","files",igfsave); } switch(script_action) { case SCRIPT_ACTION_NONE: cf->set_value("script","action","none"); break; case SCRIPT_ACTION_COMPILE: cf->set_value("script","action","compile"); break; case SCRIPT_ACTION_ENCRYPT: cf->set_value("script","action","encrypt"); break; } cf->set_value("convert_scenes","convert_text_scenes",convert_text_scenes); cf->set_value("script","encrypt_key",script_key); switch(sample_action) { case SAMPLE_ACTION_NONE: cf->set_value("convert_samples","action","none"); break; case SAMPLE_ACTION_COMPRESS_RAM: cf->set_value("convert_samples","action","compress_ram"); break; } cf->set_value("convert_samples","max_hz",sample_action_max_hz); cf->set_value("convert_samples","trim",sample_action_trim); cf->save("res://export.cfg"); } void EditorImportExport::script_set_action(ScriptAction p_action) { script_action=p_action; } EditorImportExport::ScriptAction EditorImportExport::script_get_action() const{ return script_action; } void EditorImportExport::script_set_encryption_key(const String& p_key){ script_key=p_key; } String EditorImportExport::script_get_encryption_key() const{ return script_key; } void EditorImportExport::sample_set_action(SampleAction p_action) { sample_action=p_action; } EditorImportExport::SampleAction EditorImportExport::sample_get_action() const{ return sample_action; } void EditorImportExport::sample_set_max_hz(int p_hz){ sample_action_max_hz=p_hz; } int EditorImportExport::sample_get_max_hz() const{ return sample_action_max_hz; } void EditorImportExport::sample_set_trim(bool p_trim){ sample_action_trim=p_trim; } bool EditorImportExport::sample_get_trim() const{ return sample_action_trim; } PoolVector EditorImportExport::_get_export_file_list() { PoolVector fl; for (Map::Element *E=files.front();E;E=E->next()) { fl.push_back(E->key()); } return fl; } PoolVector EditorImportExport::_get_export_platforms() { PoolVector ep; for (Map >::Element *E=exporters.front();E;E=E->next()) { ep.push_back(E->key()); } return ep; } void EditorImportExport::_bind_methods() { ClassDB::bind_method(D_METHOD("add_import_plugin","plugin"),&EditorImportExport::add_import_plugin); ClassDB::bind_method(D_METHOD("remove_import_plugin","plugin"),&EditorImportExport::remove_import_plugin); ClassDB::bind_method(D_METHOD("get_import_plugin_count"),&EditorImportExport::get_import_plugin_count); ClassDB::bind_method(D_METHOD("get_import_plugin","idx"),&EditorImportExport::get_import_plugin); ClassDB::bind_method(D_METHOD("get_import_plugin_by_name","name"),&EditorImportExport::get_import_plugin_by_name); ClassDB::bind_method(D_METHOD("add_export_plugin","plugin"),&EditorImportExport::add_export_plugin); ClassDB::bind_method(D_METHOD("remove_export_plugin","plugin"),&EditorImportExport::remove_export_plugin); ClassDB::bind_method(D_METHOD("get_export_plugin_count"),&EditorImportExport::get_export_plugin_count); ClassDB::bind_method(D_METHOD("get_export_plugin","idx"),&EditorImportExport::get_export_plugin); ClassDB::bind_method(D_METHOD("set_export_file_action","file","action"),&EditorImportExport::set_export_file_action); ClassDB::bind_method(D_METHOD("get_export_file_action","file"),&EditorImportExport::get_export_file_action); ClassDB::bind_method(D_METHOD("get_export_file_list"),&EditorImportExport::_get_export_file_list); ClassDB::bind_method(D_METHOD("add_export_platform","platform"),&EditorImportExport::add_export_platform); //ClassDB::bind_method(D_METHOD("remove_export_platform","platform"),&EditorImportExport::add_export_platform); ClassDB::bind_method(D_METHOD("get_export_platform","name"),&EditorImportExport::get_export_platform); ClassDB::bind_method(D_METHOD("get_export_platforms"),&EditorImportExport::_get_export_platforms); ClassDB::bind_method(D_METHOD("set_export_filter","filter"),&EditorImportExport::set_export_filter); ClassDB::bind_method(D_METHOD("get_export_filter"),&EditorImportExport::get_export_filter); ClassDB::bind_method(D_METHOD("set_export_custom_filter","filter"),&EditorImportExport::set_export_custom_filter); ClassDB::bind_method(D_METHOD("get_export_custom_filter"),&EditorImportExport::get_export_custom_filter); ClassDB::bind_method(D_METHOD("set_export_custom_filter_exclude","filter_exclude"),&EditorImportExport::set_export_custom_filter_exclude); ClassDB::bind_method(D_METHOD("get_export_custom_filter_exclude"),&EditorImportExport::get_export_custom_filter_exclude); ClassDB::bind_method(D_METHOD("image_export_group_create"),&EditorImportExport::image_export_group_create); ClassDB::bind_method(D_METHOD("image_export_group_remove"),&EditorImportExport::image_export_group_remove); ClassDB::bind_method(D_METHOD("image_export_group_set_image_action"),&EditorImportExport::image_export_group_set_image_action); ClassDB::bind_method(D_METHOD("image_export_group_set_make_atlas"),&EditorImportExport::image_export_group_set_make_atlas); ClassDB::bind_method(D_METHOD("image_export_group_set_shrink"),&EditorImportExport::image_export_group_set_shrink); ClassDB::bind_method(D_METHOD("image_export_group_get_image_action"),&EditorImportExport::image_export_group_get_image_action); ClassDB::bind_method(D_METHOD("image_export_group_get_make_atlas"),&EditorImportExport::image_export_group_get_make_atlas); ClassDB::bind_method(D_METHOD("image_export_group_get_shrink"),&EditorImportExport::image_export_group_get_shrink); ClassDB::bind_method(D_METHOD("image_add_to_export_group"),&EditorImportExport::image_add_to_export_group); ClassDB::bind_method(D_METHOD("script_set_action"),&EditorImportExport::script_set_action); ClassDB::bind_method(D_METHOD("script_set_encryption_key"),&EditorImportExport::script_set_encryption_key); ClassDB::bind_method(D_METHOD("script_get_action"),&EditorImportExport::script_get_action); ClassDB::bind_method(D_METHOD("script_get_encryption_key"),&EditorImportExport::script_get_encryption_key); BIND_CONSTANT( ACTION_NONE ); BIND_CONSTANT( ACTION_COPY ); BIND_CONSTANT( ACTION_BUNDLE ); BIND_CONSTANT( EXPORT_SELECTED ); BIND_CONSTANT( EXPORT_RESOURCES ); BIND_CONSTANT( EXPORT_ALL ); BIND_CONSTANT( IMAGE_ACTION_NONE ); BIND_CONSTANT( IMAGE_ACTION_COMPRESS_DISK ); BIND_CONSTANT( IMAGE_ACTION_COMPRESS_RAM ); BIND_CONSTANT( IMAGE_ACTION_KEEP ); BIND_CONSTANT( SCRIPT_ACTION_NONE ); BIND_CONSTANT( SCRIPT_ACTION_COMPILE ); BIND_CONSTANT( SCRIPT_ACTION_ENCRYPT ); }; EditorImportExport::EditorImportExport() { export_filter=EXPORT_RESOURCES; singleton=this; image_action=IMAGE_ACTION_NONE; image_action_compress_quality=0.7; image_formats.insert("png"); image_shrink=1; script_action=SCRIPT_ACTION_COMPILE; sample_action=SAMPLE_ACTION_NONE; sample_action_max_hz=44100; sample_action_trim=false; convert_text_scenes=true; } EditorImportExport::~EditorImportExport() { } #endif