diff options
26 files changed, 666 insertions, 10 deletions
diff --git a/core/io/zip_io.cpp b/core/io/zip_io.cpp index e573e8de19..200e5f5e83 100644 --- a/core/io/zip_io.cpp +++ b/core/io/zip_io.cpp @@ -37,11 +37,17 @@ void *zipio_open(voidpf opaque, const char *p_fname, int mode) {  	String fname;  	fname.parse_utf8(p_fname); +	int file_access_mode = 0;  	if (mode & ZLIB_FILEFUNC_MODE_WRITE) { -		(*fa) = FileAccess::open(fname, FileAccess::WRITE); -	} else { -		(*fa) = FileAccess::open(fname, FileAccess::READ); +		file_access_mode |= FileAccess::WRITE;  	} +	if (mode & ZLIB_FILEFUNC_MODE_READ) { +		file_access_mode |= FileAccess::READ; +	} +	if (mode & ZLIB_FILEFUNC_MODE_CREATE) { +		file_access_mode |= FileAccess::WRITE_READ; +	} +	(*fa) = FileAccess::open(fname, file_access_mode);  	if (fa->is_null()) {  		return nullptr; diff --git a/doc/classes/ProjectSettings.xml b/doc/classes/ProjectSettings.xml index 4b2bb3b6e9..ff66affeab 100644 --- a/doc/classes/ProjectSettings.xml +++ b/doc/classes/ProjectSettings.xml @@ -413,6 +413,9 @@  		<member name="debug/gdscript/warnings/standalone_ternary" type="int" setter="" getter="" default="1">  			When set to [code]warn[/code] or [code]error[/code], produces a warning or an error respectively when calling a ternary expression that has no effect on the surrounding code, such as writing [code]42 if active else 0[/code] as a statement.  		</member> +		<member name="debug/gdscript/warnings/static_called_on_instance" type="int" setter="" getter="" default="1"> +			When set to [code]warn[/code] or [code]error[/code], produces a warning or an error respectively when calling a static method from an instance of a class instead of from the class directly. +		</member>  		<member name="debug/gdscript/warnings/treat_warnings_as_errors" type="bool" setter="" getter="" default="false">  			If [code]true[/code], all warnings will be reported as if they are errors.  		</member> diff --git a/drivers/gles3/rasterizer_canvas_gles3.cpp b/drivers/gles3/rasterizer_canvas_gles3.cpp index 1a923729fa..b407670098 100644 --- a/drivers/gles3/rasterizer_canvas_gles3.cpp +++ b/drivers/gles3/rasterizer_canvas_gles3.cpp @@ -130,7 +130,10 @@ void RasterizerCanvasGLES3::canvas_render_items(RID p_to_render_target, Item *p_  		if (syncStatus == GL_UNSIGNALED) {  			// If older than 2 frames, wait for sync OpenGL can have up to 3 frames in flight, any more and we need to sync anyway.  			if (state.canvas_instance_data_buffers[state.current_buffer].last_frame_used < RSG::rasterizer->get_frame_number() - 2) { +#ifndef WEB_ENABLED +				// On web, we do nothing as the glSubBufferData will force a sync anyway and WebGL does not like waiting.  				glClientWaitSync(state.canvas_instance_data_buffers[state.current_buffer].fence, 0, 100000000); // wait for up to 100ms +#endif  			} else {  				// Used in last frame or frame before that. OpenGL can get up to two frames behind, so these buffers may still be in use  				// Allocate a new buffer and use that. diff --git a/drivers/gles3/shaders/canvas.glsl b/drivers/gles3/shaders/canvas.glsl index 22a11cdc29..23db41802e 100644 --- a/drivers/gles3/shaders/canvas.glsl +++ b/drivers/gles3/shaders/canvas.glsl @@ -211,7 +211,9 @@ void main() {  #include "canvas_uniforms_inc.glsl"  #include "stdlib_inc.glsl" +#ifndef DISABLE_LIGHTING  uniform sampler2D atlas_texture; //texunit:-2 +#endif // DISABLE_LIGHTING  //uniform sampler2D shadow_atlas_texture; //texunit:-3  uniform sampler2D screen_texture; //texunit:-4  uniform sampler2D sdf_texture; //texunit:-5 diff --git a/drivers/gles3/shaders/canvas_uniforms_inc.glsl b/drivers/gles3/shaders/canvas_uniforms_inc.glsl index 43d275205f..dd5ebecb1a 100644 --- a/drivers/gles3/shaders/canvas_uniforms_inc.glsl +++ b/drivers/gles3/shaders/canvas_uniforms_inc.glsl @@ -82,6 +82,7 @@ layout(std140) uniform CanvasData { //ubo:0  	uint pad2;  }; +#ifndef DISABLE_LIGHTING  #define LIGHT_FLAGS_BLEND_MASK uint(3 << 16)  #define LIGHT_FLAGS_BLEND_MODE_ADD uint(0 << 16)  #define LIGHT_FLAGS_BLEND_MODE_SUB uint(1 << 16) @@ -114,7 +115,7 @@ struct Light {  layout(std140) uniform LightData { //ubo:2  	Light light_array[MAX_LIGHTS];  }; - +#endif // DISABLE_LIGHTING  layout(std140) uniform DrawDataInstances { //ubo:3  	DrawData draw_data[MAX_DRAW_DATA_INSTANCES]; diff --git a/editor/editor_properties.cpp b/editor/editor_properties.cpp index 332e47dc52..899fa69be1 100644 --- a/editor/editor_properties.cpp +++ b/editor/editor_properties.cpp @@ -90,7 +90,9 @@ void EditorPropertyText::update_property() {  	String s = get_edited_object()->get(get_edited_property());  	updating = true;  	if (text->get_text() != s) { +		int caret = text->get_caret_column();  		text->set_text(s); +		text->set_caret_column(caret);  	}  	text->set_editable(!is_read_only());  	updating = false; diff --git a/editor/event_listener_line_edit.cpp b/editor/event_listener_line_edit.cpp index 14e482432a..e4c35a5b81 100644 --- a/editor/event_listener_line_edit.cpp +++ b/editor/event_listener_line_edit.cpp @@ -59,8 +59,9 @@ void EventListenerLineEdit::gui_input(const Ref<InputEvent> &p_event) {  	// First event will be an event which is used to focus this control - i.e. a mouse click, or a tab press.  	// Ignore the first one so that clicking into the LineEdit does not override the current event.  	// Ignore is reset to true when the control is unfocused. -	if (ignore) { -		ignore = false; +	// This class also specially handles grab_focus() calls. +	if (ignore_next_event) { +		ignore_next_event = false;  		return;  	} @@ -85,7 +86,7 @@ void EventListenerLineEdit::_on_focus() {  }  void EventListenerLineEdit::_on_unfocus() { -	ignore = true; +	ignore_next_event = true;  	set_placeholder(TTR("Filter by event..."));  } @@ -109,6 +110,12 @@ int EventListenerLineEdit::get_allowed_input_types() const {  	return allowed_input_types;  } +void EventListenerLineEdit::grab_focus() { +	// If we grab focus through code, we don't need to ignore the first event! +	ignore_next_event = false; +	Control::grab_focus(); +} +  void EventListenerLineEdit::_notification(int p_what) {  	switch (p_what) {  		case NOTIFICATION_ENTER_TREE: { diff --git a/editor/event_listener_line_edit.h b/editor/event_listener_line_edit.h index 5b415e8a2d..c4cd5e4511 100644 --- a/editor/event_listener_line_edit.h +++ b/editor/event_listener_line_edit.h @@ -44,7 +44,7 @@ class EventListenerLineEdit : public LineEdit {  	GDCLASS(EventListenerLineEdit, LineEdit)  	int allowed_input_types = INPUT_KEY | INPUT_MOUSE_BUTTON | INPUT_JOY_BUTTON | INPUT_JOY_MOTION; -	bool ignore = true; +	bool ignore_next_event = true;  	bool share_keycodes = false;  	Ref<InputEvent> event; @@ -67,6 +67,8 @@ public:  	void set_allowed_input_types(int input_types);  	int get_allowed_input_types() const; +	void grab_focus(); +  public:  	EventListenerLineEdit();  }; diff --git a/editor/icons/Line.svg b/editor/icons/Line.svg new file mode 100644 index 0000000000..c7b1f8a701 --- /dev/null +++ b/editor/icons/Line.svg @@ -0,0 +1 @@ +<svg height="12" viewBox="0 0 12 12" width="12" xmlns="http://www.w3.org/2000/svg"><path d="m2 1050.4 8-8" fill="none" stroke="#ffffff" stroke-linecap="round" stroke-width="2" transform="translate(0 -1040.4)"/></svg> diff --git a/editor/input_event_configuration_dialog.cpp b/editor/input_event_configuration_dialog.cpp index ed9a5898fb..c577c61db7 100644 --- a/editor/input_event_configuration_dialog.cpp +++ b/editor/input_event_configuration_dialog.cpp @@ -533,6 +533,10 @@ String InputEventConfigurationDialog::_get_device_string(int p_device) const {  void InputEventConfigurationDialog::_notification(int p_what) {  	switch (p_what) { +		case NOTIFICATION_VISIBILITY_CHANGED: { +			event_listener->grab_focus(); +		} break; +  		case NOTIFICATION_ENTER_TREE:  		case NOTIFICATION_THEME_CHANGED: {  			input_list_search->set_right_icon(input_list_search->get_theme_icon(SNAME("Search"), SNAME("EditorIcons"))); diff --git a/editor/plugins/tiles/tile_map_editor.cpp b/editor/plugins/tiles/tile_map_editor.cpp index dd16d4ffea..cbc95ef6d6 100644 --- a/editor/plugins/tiles/tile_map_editor.cpp +++ b/editor/plugins/tiles/tile_map_editor.cpp @@ -460,7 +460,7 @@ void TileMapEditorTilesPlugin::_update_theme() {  	source_sort_button->set_icon(tiles_bottom_panel->get_theme_icon(SNAME("Sort"), SNAME("EditorIcons")));  	select_tool_button->set_icon(tiles_bottom_panel->get_theme_icon(SNAME("ToolSelect"), SNAME("EditorIcons")));  	paint_tool_button->set_icon(tiles_bottom_panel->get_theme_icon(SNAME("Edit"), SNAME("EditorIcons"))); -	line_tool_button->set_icon(tiles_bottom_panel->get_theme_icon(SNAME("CurveLinear"), SNAME("EditorIcons"))); +	line_tool_button->set_icon(tiles_bottom_panel->get_theme_icon(SNAME("Line"), SNAME("EditorIcons")));  	rect_tool_button->set_icon(tiles_bottom_panel->get_theme_icon(SNAME("Rectangle"), SNAME("EditorIcons")));  	bucket_tool_button->set_icon(tiles_bottom_panel->get_theme_icon(SNAME("Bucket"), SNAME("EditorIcons"))); @@ -3277,7 +3277,7 @@ void TileMapEditorTerrainsPlugin::_update_tiles_list() {  void TileMapEditorTerrainsPlugin::_update_theme() {  	paint_tool_button->set_icon(main_vbox_container->get_theme_icon(SNAME("Edit"), SNAME("EditorIcons"))); -	line_tool_button->set_icon(main_vbox_container->get_theme_icon(SNAME("CurveLinear"), SNAME("EditorIcons"))); +	line_tool_button->set_icon(main_vbox_container->get_theme_icon(SNAME("Line"), SNAME("EditorIcons")));  	rect_tool_button->set_icon(main_vbox_container->get_theme_icon(SNAME("Rectangle"), SNAME("EditorIcons")));  	bucket_tool_button->set_icon(main_vbox_container->get_theme_icon(SNAME("Bucket"), SNAME("EditorIcons"))); diff --git a/modules/gdscript/gdscript_analyzer.cpp b/modules/gdscript/gdscript_analyzer.cpp index b3ca8ff00a..6fbdec863f 100644 --- a/modules/gdscript/gdscript_analyzer.cpp +++ b/modules/gdscript/gdscript_analyzer.cpp @@ -2552,6 +2552,10 @@ void GDScriptAnalyzer::reduce_call(GDScriptParser::CallNode *p_call, bool p_is_a  		if (p_is_root && return_type.kind != GDScriptParser::DataType::UNRESOLVED && return_type.builtin_type != Variant::NIL) {  			parser->push_warning(p_call, GDScriptWarning::RETURN_VALUE_DISCARDED, p_call->function_name);  		} + +		if (is_static && !base_type.is_meta_type && !(callee_type != GDScriptParser::Node::SUBSCRIPT && parser->current_function != nullptr && parser->current_function->is_static)) { +			parser->push_warning(p_call, GDScriptWarning::STATIC_CALLED_ON_INSTANCE, p_call->function_name, base_type.to_string()); +		}  #endif // DEBUG_ENABLED  		call_type = return_type; diff --git a/modules/gdscript/gdscript_warning.cpp b/modules/gdscript/gdscript_warning.cpp index 1cae7bdfac..a0c107aa53 100644 --- a/modules/gdscript/gdscript_warning.cpp +++ b/modules/gdscript/gdscript_warning.cpp @@ -155,6 +155,10 @@ String GDScriptWarning::get_message() const {  		case INT_ASSIGNED_TO_ENUM: {  			return "Integer used when an enum value is expected. If this is intended cast the integer to the enum type.";  		} +		case STATIC_CALLED_ON_INSTANCE: { +			CHECK_SYMBOLS(2); +			return vformat(R"(The function '%s()' is a static function but was called from an instance. Instead, it should be directly called from the type: '%s.%s()'.)", symbols[0], symbols[1], symbols[0]); +		}  		case WARNING_MAX:  			break; // Can't happen, but silences warning  	} @@ -215,6 +219,7 @@ String GDScriptWarning::get_name_from_code(Code p_code) {  		"EMPTY_FILE",  		"SHADOWED_GLOBAL_IDENTIFIER",  		"INT_ASSIGNED_TO_ENUM", +		"STATIC_CALLED_ON_INSTANCE",  	};  	static_assert((sizeof(names) / sizeof(*names)) == WARNING_MAX, "Amount of warning types don't match the amount of warning names."); diff --git a/modules/gdscript/gdscript_warning.h b/modules/gdscript/gdscript_warning.h index a639e7b44e..7e4e975510 100644 --- a/modules/gdscript/gdscript_warning.h +++ b/modules/gdscript/gdscript_warning.h @@ -78,6 +78,7 @@ public:  		EMPTY_FILE, // A script file is empty.  		SHADOWED_GLOBAL_IDENTIFIER, // A global class or function has the same name as variable.  		INT_ASSIGNED_TO_ENUM, // An integer value was assigned to an enum-typed variable without casting. +		STATIC_CALLED_ON_INSTANCE, // A static method was called on an instance of a class instead of on the class itself.  		WARNING_MAX,  	}; diff --git a/modules/gdscript/tests/scripts/parser/warnings/static_called_on_instance.gd b/modules/gdscript/tests/scripts/parser/warnings/static_called_on_instance.gd new file mode 100644 index 0000000000..29d8501b78 --- /dev/null +++ b/modules/gdscript/tests/scripts/parser/warnings/static_called_on_instance.gd @@ -0,0 +1,11 @@ +class Player: +	var x = 3 + +func test(): +	# These should not emit a warning. +	var _player = Player.new() +	print(String.num_uint64(8589934592)) # 2 ^ 33 + +	# This should emit a warning. +	var some_string = String() +	print(some_string.num_uint64(8589934592)) # 2 ^ 33 diff --git a/modules/gdscript/tests/scripts/parser/warnings/static_called_on_instance.out b/modules/gdscript/tests/scripts/parser/warnings/static_called_on_instance.out new file mode 100644 index 0000000000..3933a35178 --- /dev/null +++ b/modules/gdscript/tests/scripts/parser/warnings/static_called_on_instance.out @@ -0,0 +1,7 @@ +GDTEST_OK +>> WARNING +>> Line: 11 +>> STATIC_CALLED_ON_INSTANCE +>> The function 'num_uint64()' is a static function but was called from an instance. Instead, it should be directly called from the type: 'String.num_uint64()'. +8589934592 +8589934592 diff --git a/modules/zip/SCsub b/modules/zip/SCsub new file mode 100644 index 0000000000..b7710123fd --- /dev/null +++ b/modules/zip/SCsub @@ -0,0 +1,9 @@ +#!/usr/bin/env python + +Import("env") +Import("env_modules") + +env_zip = env_modules.Clone() + +# Module files +env_zip.add_source_files(env.modules_sources, "*.cpp") diff --git a/modules/zip/config.py b/modules/zip/config.py new file mode 100644 index 0000000000..96cd2fc5bd --- /dev/null +++ b/modules/zip/config.py @@ -0,0 +1,17 @@ +def can_build(env, platform): +    return env["minizip"] + + +def configure(env): +    pass + + +def get_doc_classes(): +    return [ +        "ZIPReader", +        "ZIPPacker", +    ] + + +def get_doc_path(): +    return "doc_classes" diff --git a/modules/zip/doc_classes/ZIPPacker.xml b/modules/zip/doc_classes/ZIPPacker.xml new file mode 100644 index 0000000000..95d7ef50f9 --- /dev/null +++ b/modules/zip/doc_classes/ZIPPacker.xml @@ -0,0 +1,72 @@ +<?xml version="1.0" encoding="UTF-8" ?> +<class name="ZIPPacker" inherits="RefCounted" version="4.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="../../../doc/class.xsd"> +	<brief_description> +		Allows the creation of zip files. +	</brief_description> +	<description> +		This class implements a writer that allows storing the multiple blobs in a zip archive. +		[codeblock] +		func write_zip_file(): +		    var writer := ZIPPacker.new() +		    var err := writer.open("user://archive.zip") +		    if err != OK: +		        return err +		    writer.start_file("hello.txt") +		    writer.write_file("Hello World".to_utf8_buffer()) +		    writer.close_file() + +		    writer.close() +		    return OK +		[/codeblock] +	</description> +	<tutorials> +	</tutorials> +	<methods> +		<method name="close"> +			<return type="int" enum="Error" /> +			<description> +				Closes the underlying resources used by this instance. +			</description> +		</method> +		<method name="close_file"> +			<return type="int" enum="Error" /> +			<description> +				Stops writing to a file within the archive. +				It will fail if there is no open file. +			</description> +		</method> +		<method name="open"> +			<return type="int" enum="Error" /> +			<param index="0" name="path" type="String" /> +			<param index="1" name="append" type="int" enum="ZIPPacker.ZipAppend" default="0" /> +			<description> +				Opens a zip file for writing at the given path using the specified write mode. +				This must be called before everything else. +			</description> +		</method> +		<method name="start_file"> +			<return type="int" enum="Error" /> +			<param index="0" name="path" type="String" /> +			<description> +				Starts writing to a file within the archive. Only one file can be written at the same time. +				Must be called after [method open]. +			</description> +		</method> +		<method name="write_file"> +			<return type="int" enum="Error" /> +			<param index="0" name="data" type="PackedByteArray" /> +			<description> +				Write the given [param data] to the file. +				Needs to be called after [method start_file]. +			</description> +		</method> +	</methods> +	<constants> +		<constant name="APPEND_CREATE" value="0" enum="ZipAppend"> +		</constant> +		<constant name="APPEND_CREATEAFTER" value="1" enum="ZipAppend"> +		</constant> +		<constant name="APPEND_ADDINZIP" value="2" enum="ZipAppend"> +		</constant> +	</constants> +</class> diff --git a/modules/zip/doc_classes/ZIPReader.xml b/modules/zip/doc_classes/ZIPReader.xml new file mode 100644 index 0000000000..717116a531 --- /dev/null +++ b/modules/zip/doc_classes/ZIPReader.xml @@ -0,0 +1,52 @@ +<?xml version="1.0" encoding="UTF-8" ?> +<class name="ZIPReader" inherits="RefCounted" version="4.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="../../../doc/class.xsd"> +	<brief_description> +		Allows reading the content of a zip file. +	</brief_description> +	<description> +		This class implements a reader that can extract the content of individual files inside a zip archive. +		[codeblock] +		func read_zip_file(): +		    var reader := ZIPReader.new() +		    var err := reader.open("user://archive.zip") +		    if err == OK: +		        return PackedByteArray() +		    var res := reader.read_file("hello.txt") +		    reader.close() +		    return res +		[/codeblock] +	</description> +	<tutorials> +	</tutorials> +	<methods> +		<method name="close"> +			<return type="int" enum="Error" /> +			<description> +				Closes the underlying resources used by this instance. +			</description> +		</method> +		<method name="get_files"> +			<return type="PackedStringArray" /> +			<description> +				Returns the list of names of all files in the loaded archive. +				Must be called after [method open]. +			</description> +		</method> +		<method name="open"> +			<return type="int" enum="Error" /> +			<param index="0" name="path" type="String" /> +			<description> +				Opens the zip archive at the given [param path] and reads its file index. +			</description> +		</method> +		<method name="read_file"> +			<return type="PackedByteArray" /> +			<param index="0" name="path" type="String" /> +			<param index="1" name="case_sensitive" type="bool" default="true" /> +			<description> +				Loads the whole content of a file in the loaded zip archive into memory and returns it. +				Must be called after [method open]. +			</description> +		</method> +	</methods> +</class> diff --git a/modules/zip/register_types.cpp b/modules/zip/register_types.cpp new file mode 100644 index 0000000000..20fb484cfe --- /dev/null +++ b/modules/zip/register_types.cpp @@ -0,0 +1,50 @@ +/*************************************************************************/ +/*  register_types.cpp                                                   */ +/*************************************************************************/ +/*                       This file is part of:                           */ +/*                           GODOT ENGINE                                */ +/*                      https://godotengine.org                          */ +/*************************************************************************/ +/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */ +/* Copyright (c) 2014-2022 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 "register_types.h" + +#include "core/object/class_db.h" +#include "zip_packer.h" +#include "zip_reader.h" + +void initialize_zip_module(ModuleInitializationLevel p_level) { +	if (p_level != MODULE_INITIALIZATION_LEVEL_SCENE) { +		return; +	} + +	GDREGISTER_CLASS(ZIPPacker); +	GDREGISTER_CLASS(ZIPReader); +} + +void uninitialize_zip_module(ModuleInitializationLevel p_level) { +	if (p_level != MODULE_INITIALIZATION_LEVEL_SCENE) { +		return; +	} +} diff --git a/modules/zip/register_types.h b/modules/zip/register_types.h new file mode 100644 index 0000000000..2640be12b8 --- /dev/null +++ b/modules/zip/register_types.h @@ -0,0 +1,39 @@ +/*************************************************************************/ +/*  register_types.h                                                     */ +/*************************************************************************/ +/*                       This file is part of:                           */ +/*                           GODOT ENGINE                                */ +/*                      https://godotengine.org                          */ +/*************************************************************************/ +/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */ +/* Copyright (c) 2014-2022 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.                */ +/*************************************************************************/ + +#ifndef ZIP_REGISTER_TYPES_H +#define ZIP_REGISTER_TYPES_H + +#include "modules/register_module_types.h" + +void initialize_zip_module(ModuleInitializationLevel p_level); +void uninitialize_zip_module(ModuleInitializationLevel p_level); + +#endif // ZIP_REGISTER_TYPES_H diff --git a/modules/zip/zip_packer.cpp b/modules/zip/zip_packer.cpp new file mode 100644 index 0000000000..c37fc0945e --- /dev/null +++ b/modules/zip/zip_packer.cpp @@ -0,0 +1,108 @@ +/*************************************************************************/ +/*  zip_packer.cpp                                                       */ +/*************************************************************************/ +/*                       This file is part of:                           */ +/*                           GODOT ENGINE                                */ +/*                      https://godotengine.org                          */ +/*************************************************************************/ +/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */ +/* Copyright (c) 2014-2022 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 "zip_packer.h" + +#include "core/io/zip_io.h" +#include "core/os/os.h" + +Error ZIPPacker::open(String p_path, ZipAppend p_append) { +	if (fa.is_valid()) { +		close(); +	} + +	zlib_filefunc_def io = zipio_create_io(&fa); +	zf = zipOpen2(p_path.utf8().get_data(), p_append, NULL, &io); +	return zf != NULL ? OK : FAILED; +} + +Error ZIPPacker::close() { +	ERR_FAIL_COND_V_MSG(fa.is_null(), FAILED, "ZIPPacker cannot be closed because it is not open."); + +	return zipClose(zf, NULL) == ZIP_OK ? OK : FAILED; +} + +Error ZIPPacker::start_file(String p_path) { +	ERR_FAIL_COND_V_MSG(zf != NULL, FAILED, "ZIPPacker is already in use."); +	ERR_FAIL_COND_V_MSG(fa.is_null(), FAILED, "ZIPPacker must be opened before use."); + +	zip_fileinfo zipfi; + +	OS::DateTime time = OS::get_singleton()->get_datetime(); + +	zipfi.tmz_date.tm_hour = time.hour; +	zipfi.tmz_date.tm_mday = time.day; +	zipfi.tmz_date.tm_min = time.minute; +	zipfi.tmz_date.tm_mon = time.month - 1; +	zipfi.tmz_date.tm_sec = time.second; +	zipfi.tmz_date.tm_year = time.year; +	zipfi.dosDate = 0; +	zipfi.external_fa = 0; +	zipfi.internal_fa = 0; + +	int ret = zipOpenNewFileInZip(zf, p_path.utf8().get_data(), &zipfi, NULL, 0, NULL, 0, NULL, Z_DEFLATED, Z_DEFAULT_COMPRESSION); +	return ret == ZIP_OK ? OK : FAILED; +} + +Error ZIPPacker::write_file(Vector<uint8_t> p_data) { +	ERR_FAIL_COND_V_MSG(fa.is_null(), FAILED, "ZIPPacker must be opened before use."); + +	return zipWriteInFileInZip(zf, p_data.ptr(), p_data.size()) == ZIP_OK ? OK : FAILED; +} + +Error ZIPPacker::close_file() { +	ERR_FAIL_COND_V_MSG(fa.is_null(), FAILED, "ZIPPacker must be opened before use."); + +	Error err = zipCloseFileInZip(zf) == ZIP_OK ? OK : FAILED; +	if (err == OK) { +		zf = NULL; +	} +	return err; +} + +void ZIPPacker::_bind_methods() { +	ClassDB::bind_method(D_METHOD("open", "path", "append"), &ZIPPacker::open, DEFVAL(Variant(APPEND_CREATE))); +	ClassDB::bind_method(D_METHOD("start_file", "path"), &ZIPPacker::start_file); +	ClassDB::bind_method(D_METHOD("write_file", "data"), &ZIPPacker::write_file); +	ClassDB::bind_method(D_METHOD("close_file"), &ZIPPacker::close_file); +	ClassDB::bind_method(D_METHOD("close"), &ZIPPacker::close); + +	BIND_ENUM_CONSTANT(APPEND_CREATE); +	BIND_ENUM_CONSTANT(APPEND_CREATEAFTER); +	BIND_ENUM_CONSTANT(APPEND_ADDINZIP); +} + +ZIPPacker::ZIPPacker() {} + +ZIPPacker::~ZIPPacker() { +	if (fa.is_valid()) { +		close(); +	} +} diff --git a/modules/zip/zip_packer.h b/modules/zip/zip_packer.h new file mode 100644 index 0000000000..23e96b5ad2 --- /dev/null +++ b/modules/zip/zip_packer.h @@ -0,0 +1,68 @@ +/*************************************************************************/ +/*  zip_packer.h                                                         */ +/*************************************************************************/ +/*                       This file is part of:                           */ +/*                           GODOT ENGINE                                */ +/*                      https://godotengine.org                          */ +/*************************************************************************/ +/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */ +/* Copyright (c) 2014-2022 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.                */ +/*************************************************************************/ + +#ifndef ZIP_PACKER_H +#define ZIP_PACKER_H + +#include "core/io/file_access.h" +#include "core/object/ref_counted.h" + +#include "thirdparty/minizip/zip.h" + +class ZIPPacker : public RefCounted { +	GDCLASS(ZIPPacker, RefCounted); + +	Ref<FileAccess> fa; +	zipFile zf; + +protected: +	static void _bind_methods(); + +public: +	enum ZipAppend { +		APPEND_CREATE = 0, +		APPEND_CREATEAFTER = 1, +		APPEND_ADDINZIP = 2, +	}; + +	Error open(String p_path, ZipAppend p_append); +	Error close(); + +	Error start_file(String p_path); +	Error write_file(Vector<uint8_t> p_data); +	Error close_file(); + +	ZIPPacker(); +	~ZIPPacker(); +}; + +VARIANT_ENUM_CAST(ZIPPacker::ZipAppend) + +#endif // ZIP_PACKER_H diff --git a/modules/zip/zip_reader.cpp b/modules/zip/zip_reader.cpp new file mode 100644 index 0000000000..f35b947cef --- /dev/null +++ b/modules/zip/zip_reader.cpp @@ -0,0 +1,123 @@ +/*************************************************************************/ +/*  zip_reader.cpp                                                       */ +/*************************************************************************/ +/*                       This file is part of:                           */ +/*                           GODOT ENGINE                                */ +/*                      https://godotengine.org                          */ +/*************************************************************************/ +/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */ +/* Copyright (c) 2014-2022 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 "zip_reader.h" + +#include "core/error/error_macros.h" +#include "core/io/zip_io.h" + +Error ZIPReader::open(String p_path) { +	if (fa.is_valid()) { +		close(); +	} + +	zlib_filefunc_def io = zipio_create_io(&fa); +	uzf = unzOpen2(p_path.utf8().get_data(), &io); +	return uzf != NULL ? OK : FAILED; +} + +Error ZIPReader::close() { +	ERR_FAIL_COND_V_MSG(fa.is_null(), FAILED, "ZIPReader cannot be closed because it is not open."); + +	return unzClose(uzf) == UNZ_OK ? OK : FAILED; +} + +PackedStringArray ZIPReader::get_files() { +	ERR_FAIL_COND_V_MSG(fa.is_null(), PackedStringArray(), "ZIPReader must be opened before use."); + +	List<String> s; + +	if (unzGoToFirstFile(uzf) != UNZ_OK) { +		return PackedStringArray(); +	} + +	do { +		unz_file_info64 file_info; +		char filename[256]; // Note filename is a path ! +		int err = unzGetCurrentFileInfo64(uzf, &file_info, filename, sizeof(filename), NULL, 0, NULL, 0); +		if (err == UNZ_OK) { +			s.push_back(filename); +		} else { +			// Assume filename buffer was too small +			char *long_filename_buff = (char *)memalloc(file_info.size_filename); +			int err2 = unzGetCurrentFileInfo64(uzf, NULL, long_filename_buff, sizeof(long_filename_buff), NULL, 0, NULL, 0); +			if (err2 == UNZ_OK) { +				s.push_back(long_filename_buff); +				memfree(long_filename_buff); +			} +		} +	} while (unzGoToNextFile(uzf) == UNZ_OK); + +	PackedStringArray arr; +	arr.resize(s.size()); +	int idx = 0; +	for (const List<String>::Element *E = s.front(); E; E = E->next()) { +		arr.set(idx++, E->get()); +	} +	return arr; +} + +PackedByteArray ZIPReader::read_file(String p_path, bool p_case_sensitive) { +	ERR_FAIL_COND_V_MSG(fa.is_null(), PackedByteArray(), "ZIPReader must be opened before use."); + +	int cs = p_case_sensitive ? 1 : 2; +	if (unzLocateFile(uzf, p_path.utf8().get_data(), cs) != UNZ_OK) { +		ERR_FAIL_V_MSG(PackedByteArray(), "File does not exist in zip archive: " + p_path); +	} +	if (unzOpenCurrentFile(uzf) != UNZ_OK) { +		ERR_FAIL_V_MSG(PackedByteArray(), "Could not open file within zip archive."); +	} + +	unz_file_info info; +	unzGetCurrentFileInfo(uzf, &info, NULL, 0, NULL, 0, NULL, 0); +	PackedByteArray data; +	data.resize(info.uncompressed_size); + +	uint8_t *w = data.ptrw(); +	unzReadCurrentFile(uzf, &w[0], info.uncompressed_size); + +	unzCloseCurrentFile(uzf); +	return data; +} + +ZIPReader::ZIPReader() {} + +ZIPReader::~ZIPReader() { +	if (fa.is_valid()) { +		close(); +	} +} + +void ZIPReader::_bind_methods() { +	ClassDB::bind_method(D_METHOD("open", "path"), &ZIPReader::open); +	ClassDB::bind_method(D_METHOD("close"), &ZIPReader::close); +	ClassDB::bind_method(D_METHOD("get_files"), &ZIPReader::get_files); +	ClassDB::bind_method(D_METHOD("read_file", "path", "case_sensitive"), &ZIPReader::read_file, DEFVAL(Variant(true))); +} diff --git a/modules/zip/zip_reader.h b/modules/zip/zip_reader.h new file mode 100644 index 0000000000..fbc2fc0409 --- /dev/null +++ b/modules/zip/zip_reader.h @@ -0,0 +1,59 @@ +/*************************************************************************/ +/*  zip_reader.h                                                         */ +/*************************************************************************/ +/*                       This file is part of:                           */ +/*                           GODOT ENGINE                                */ +/*                      https://godotengine.org                          */ +/*************************************************************************/ +/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */ +/* Copyright (c) 2014-2022 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.                */ +/*************************************************************************/ + +#ifndef ZIP_READER_H +#define ZIP_READER_H + +#include "core/io/file_access.h" +#include "core/object/ref_counted.h" + +#include "thirdparty/minizip/unzip.h" + +class ZIPReader : public RefCounted { +	GDCLASS(ZIPReader, RefCounted) + +	Ref<FileAccess> fa; +	unzFile uzf; + +protected: +	static void _bind_methods(); + +public: +	Error open(String p_path); +	Error close(); + +	PackedStringArray get_files(); +	PackedByteArray read_file(String p_path, bool p_case_sensitive); + +	ZIPReader(); +	~ZIPReader(); +}; + +#endif // ZIP_READER_H  |