| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*  display_server_android.cpp                                           */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*                       This file is part of:                           */ | 
					
						
							|  |  |  | /*                           GODOT ENGINE                                */ | 
					
						
							|  |  |  | /*                      https://godotengine.org                          */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2021-01-01 20:13:46 +01:00
										 |  |  | /* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur.                 */ | 
					
						
							|  |  |  | /* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md).   */ | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | /*                                                                       */ | 
					
						
							|  |  |  | /* 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 "display_server_android.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "android_keys_utils.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-07 19:33:38 -03:00
										 |  |  | #include "core/config/project_settings.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | #include "java_godot_io_wrapper.h"
 | 
					
						
							|  |  |  | #include "java_godot_wrapper.h"
 | 
					
						
							|  |  |  | #include "os_android.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | #include <android/input.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | #if defined(VULKAN_ENABLED)
 | 
					
						
							|  |  |  | #include "drivers/vulkan/rendering_device_vulkan.h"
 | 
					
						
							|  |  |  | #include "platform/android/vulkan/vulkan_context_android.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-04 15:26:24 -03:00
										 |  |  | #include "servers/rendering/renderer_rd/renderer_compositor_rd.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DisplayServerAndroid *DisplayServerAndroid::get_singleton() { | 
					
						
							|  |  |  | 	return (DisplayServerAndroid *)DisplayServer::get_singleton(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DisplayServerAndroid::has_feature(Feature p_feature) const { | 
					
						
							|  |  |  | 	switch (p_feature) { | 
					
						
							|  |  |  | 		//case FEATURE_CONSOLE_WINDOW:
 | 
					
						
							|  |  |  | 		//case FEATURE_CURSOR_SHAPE:
 | 
					
						
							|  |  |  | 		//case FEATURE_CUSTOM_CURSOR_SHAPE:
 | 
					
						
							|  |  |  | 		//case FEATURE_GLOBAL_MENU:
 | 
					
						
							|  |  |  | 		//case FEATURE_HIDPI:
 | 
					
						
							|  |  |  | 		//case FEATURE_ICON:
 | 
					
						
							|  |  |  | 		//case FEATURE_IME:
 | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 		case FEATURE_MOUSE: | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 		//case FEATURE_MOUSE_WARP:
 | 
					
						
							|  |  |  | 		//case FEATURE_NATIVE_DIALOG:
 | 
					
						
							|  |  |  | 		//case FEATURE_NATIVE_ICON:
 | 
					
						
							|  |  |  | 		//case FEATURE_NATIVE_VIDEO:
 | 
					
						
							|  |  |  | 		//case FEATURE_WINDOW_TRANSPARENCY:
 | 
					
						
							|  |  |  | 		case FEATURE_CLIPBOARD: | 
					
						
							|  |  |  | 		case FEATURE_KEEP_SCREEN_ON: | 
					
						
							|  |  |  | 		case FEATURE_ORIENTATION: | 
					
						
							|  |  |  | 		case FEATURE_TOUCHSCREEN: | 
					
						
							|  |  |  | 		case FEATURE_VIRTUAL_KEYBOARD: | 
					
						
							|  |  |  | 			return true; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String DisplayServerAndroid::get_name() const { | 
					
						
							|  |  |  | 	return "Android"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::clipboard_set(const String &p_text) { | 
					
						
							|  |  |  | 	GodotJavaWrapper *godot_java = OS_Android::get_singleton()->get_godot_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!godot_java); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (godot_java->has_set_clipboard()) { | 
					
						
							|  |  |  | 		godot_java->set_clipboard(p_text); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		DisplayServer::clipboard_set(p_text); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String DisplayServerAndroid::clipboard_get() const { | 
					
						
							|  |  |  | 	GodotJavaWrapper *godot_java = OS_Android::get_singleton()->get_godot_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND_V(!godot_java, String()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (godot_java->has_get_clipboard()) { | 
					
						
							|  |  |  | 		return godot_java->get_clipboard(); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return DisplayServer::clipboard_get(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::screen_set_keep_on(bool p_enable) { | 
					
						
							|  |  |  | 	GodotJavaWrapper *godot_java = OS_Android::get_singleton()->get_godot_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!godot_java); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	godot_java->set_keep_screen_on(p_enable); | 
					
						
							|  |  |  | 	keep_screen_on = p_enable; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DisplayServerAndroid::screen_is_kept_on() const { | 
					
						
							|  |  |  | 	return keep_screen_on; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::screen_set_orientation(DisplayServer::ScreenOrientation p_orientation, int p_screen) { | 
					
						
							|  |  |  | 	GodotIOJavaWrapper *godot_io_java = OS_Android::get_singleton()->get_godot_io_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!godot_io_java); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	godot_io_java->set_screen_orientation(p_orientation); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DisplayServer::ScreenOrientation DisplayServerAndroid::screen_get_orientation(int p_screen) const { | 
					
						
							|  |  |  | 	GodotIOJavaWrapper *godot_io_java = OS_Android::get_singleton()->get_godot_io_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND_V(!godot_io_java, SCREEN_LANDSCAPE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return (ScreenOrientation)godot_io_java->get_screen_orientation(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int DisplayServerAndroid::get_screen_count() const { | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Point2i DisplayServerAndroid::screen_get_position(int p_screen) const { | 
					
						
							|  |  |  | 	return Point2i(0, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Size2i DisplayServerAndroid::screen_get_size(int p_screen) const { | 
					
						
							|  |  |  | 	return OS_Android::get_singleton()->get_display_size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Rect2i DisplayServerAndroid::screen_get_usable_rect(int p_screen) const { | 
					
						
							| 
									
										
										
										
											2020-10-26 15:07:54 -04:00
										 |  |  | 	GodotIOJavaWrapper *godot_io_java = OS_Android::get_singleton()->get_godot_io_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND_V(!godot_io_java, Rect2i()); | 
					
						
							|  |  |  | 	int xywh[4]; | 
					
						
							|  |  |  | 	godot_io_java->screen_get_usable_rect(xywh); | 
					
						
							|  |  |  | 	return Rect2i(xywh[0], xywh[1], xywh[2], xywh[3]); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int DisplayServerAndroid::screen_get_dpi(int p_screen) const { | 
					
						
							|  |  |  | 	GodotIOJavaWrapper *godot_io_java = OS_Android::get_singleton()->get_godot_io_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND_V(!godot_io_java, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return godot_io_java->get_screen_dpi(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DisplayServerAndroid::screen_is_touchscreen(int p_screen) const { | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-17 17:44:13 +02:00
										 |  |  | void DisplayServerAndroid::virtual_keyboard_show(const String &p_existing_text, const Rect2 &p_screen_rect, bool p_multiline, int p_max_length, int p_cursor_start, int p_cursor_end) { | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	GodotIOJavaWrapper *godot_io_java = OS_Android::get_singleton()->get_godot_io_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!godot_io_java); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (godot_io_java->has_vk()) { | 
					
						
							| 
									
										
										
										
											2020-07-17 17:44:13 +02:00
										 |  |  | 		godot_io_java->show_vk(p_existing_text, p_multiline, p_max_length, p_cursor_start, p_cursor_end); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		ERR_PRINT("Virtual keyboard not available"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::virtual_keyboard_hide() { | 
					
						
							|  |  |  | 	GodotIOJavaWrapper *godot_io_java = OS_Android::get_singleton()->get_godot_io_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!godot_io_java); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (godot_io_java->has_vk()) { | 
					
						
							|  |  |  | 		godot_io_java->hide_vk(); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ERR_PRINT("Virtual keyboard not available"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int DisplayServerAndroid::virtual_keyboard_get_height() const { | 
					
						
							|  |  |  | 	GodotIOJavaWrapper *godot_io_java = OS_Android::get_singleton()->get_godot_io_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND_V(!godot_io_java, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return godot_io_java->get_vk_height(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_window_event_callback(const Callable &p_callable, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	window_event_callback = p_callable; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_input_event_callback(const Callable &p_callable, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	input_event_callback = p_callable; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_input_text_callback(const Callable &p_callable, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	input_text_callback = p_callable; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_rect_changed_callback(const Callable &p_callable, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_drop_files_callback(const Callable &p_callable, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::_window_callback(const Callable &p_callable, const Variant &p_arg) const { | 
					
						
							|  |  |  | 	if (!p_callable.is_null()) { | 
					
						
							|  |  |  | 		const Variant *argp = &p_arg; | 
					
						
							|  |  |  | 		Variant ret; | 
					
						
							|  |  |  | 		Callable::CallError ce; | 
					
						
							|  |  |  | 		p_callable.call((const Variant **)&argp, 1, ret, ce); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::send_window_event(DisplayServer::WindowEvent p_event) const { | 
					
						
							|  |  |  | 	_window_callback(window_event_callback, int(p_event)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::send_input_event(const Ref<InputEvent> &p_event) const { | 
					
						
							|  |  |  | 	_window_callback(input_event_callback, p_event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::send_input_text(const String &p_text) const { | 
					
						
							|  |  |  | 	_window_callback(input_text_callback, p_text); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::_dispatch_input_events(const Ref<InputEvent> &p_event) { | 
					
						
							|  |  |  | 	DisplayServerAndroid::get_singleton()->send_input_event(p_event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Vector<DisplayServer::WindowID> DisplayServerAndroid::get_window_list() const { | 
					
						
							|  |  |  | 	Vector<WindowID> ret; | 
					
						
							|  |  |  | 	ret.push_back(MAIN_WINDOW_ID); | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DisplayServer::WindowID DisplayServerAndroid::get_window_at_screen_position(const Point2i &p_position) const { | 
					
						
							|  |  |  | 	return MAIN_WINDOW_ID; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_attach_instance_id(ObjectID p_instance, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	window_attached_instance_id = p_instance; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ObjectID DisplayServerAndroid::window_get_attached_instance_id(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return window_attached_instance_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_title(const String &p_title, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int DisplayServerAndroid::window_get_current_screen(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return SCREEN_OF_MAIN_WINDOW; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_current_screen(int p_screen, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Point2i DisplayServerAndroid::window_get_position(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return Point2i(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_position(const Point2i &p_position, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_transient(DisplayServer::WindowID p_window, DisplayServer::WindowID p_parent) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_max_size(const Size2i p_size, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Size2i DisplayServerAndroid::window_get_max_size(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return Size2i(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_min_size(const Size2i p_size, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Size2i DisplayServerAndroid::window_get_min_size(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return Size2i(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_size(const Size2i p_size, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Size2i DisplayServerAndroid::window_get_size(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return OS_Android::get_singleton()->get_display_size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Size2i DisplayServerAndroid::window_get_real_size(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return OS_Android::get_singleton()->get_display_size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_mode(DisplayServer::WindowMode p_mode, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DisplayServer::WindowMode DisplayServerAndroid::window_get_mode(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return WINDOW_MODE_FULLSCREEN; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DisplayServerAndroid::window_is_maximize_allowed(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_set_flag(DisplayServer::WindowFlags p_flag, bool p_enabled, DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DisplayServerAndroid::window_get_flag(DisplayServer::WindowFlags p_flag, DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_request_attention(DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::window_move_to_foreground(DisplayServer::WindowID p_window) { | 
					
						
							|  |  |  | 	// Not supported on Android.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DisplayServerAndroid::window_can_draw(DisplayServer::WindowID p_window) const { | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DisplayServerAndroid::can_any_window_draw() const { | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::alert(const String &p_alert, const String &p_title) { | 
					
						
							|  |  |  | 	GodotJavaWrapper *godot_java = OS_Android::get_singleton()->get_godot_java(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!godot_java); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	godot_java->alert(p_alert, p_title); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::process_events() { | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	Input::get_singleton()->flush_accumulated_events(); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Vector<String> DisplayServerAndroid::get_rendering_drivers_func() { | 
					
						
							|  |  |  | 	Vector<String> drivers; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef OPENGL_ENABLED
 | 
					
						
							|  |  |  | 	drivers.push_back("opengl"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #ifdef VULKAN_ENABLED
 | 
					
						
							|  |  |  | 	drivers.push_back("vulkan"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return drivers; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DisplayServer *DisplayServerAndroid::create_func(const String &p_rendering_driver, DisplayServer::WindowMode p_mode, uint32_t p_flags, const Vector2i &p_resolution, Error &r_error) { | 
					
						
							| 
									
										
										
										
											2020-07-13 19:24:04 +03:00
										 |  |  | 	DisplayServer *ds = memnew(DisplayServerAndroid(p_rendering_driver, p_mode, p_flags, p_resolution, r_error)); | 
					
						
							|  |  |  | 	if (r_error != OK) { | 
					
						
							|  |  |  | 		ds->alert("Your video card driver does not support any of the supported Vulkan versions.", "Unable to initialize Video driver"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ds; | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::register_android_driver() { | 
					
						
							|  |  |  | 	register_create_function("android", create_func, get_rendering_drivers_func); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-24 10:04:59 +02:00
										 |  |  | void DisplayServerAndroid::reset_window() { | 
					
						
							|  |  |  | #if defined(VULKAN_ENABLED)
 | 
					
						
							|  |  |  | 	if (rendering_driver == "vulkan") { | 
					
						
							|  |  |  | 		ANativeWindow *native_window = OS_Android::get_singleton()->get_native_window(); | 
					
						
							|  |  |  | 		ERR_FAIL_COND(!native_window); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ERR_FAIL_COND(!context_vulkan); | 
					
						
							|  |  |  | 		context_vulkan->window_destroy(MAIN_WINDOW_ID); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Size2i display_size = OS_Android::get_singleton()->get_display_size(); | 
					
						
							|  |  |  | 		if (context_vulkan->window_create(native_window, display_size.width, display_size.height) == -1) { | 
					
						
							|  |  |  | 			memdelete(context_vulkan); | 
					
						
							|  |  |  | 			context_vulkan = nullptr; | 
					
						
							|  |  |  | 			ERR_FAIL_MSG("Failed to reset Vulkan window."); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | DisplayServerAndroid::DisplayServerAndroid(const String &p_rendering_driver, DisplayServer::WindowMode p_mode, uint32_t p_flags, const Vector2i &p_resolution, Error &r_error) { | 
					
						
							|  |  |  | 	rendering_driver = p_rendering_driver; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// TODO: rendering_driver is broken, change when different drivers are supported again
 | 
					
						
							|  |  |  | 	rendering_driver = "vulkan"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	keep_screen_on = GLOBAL_GET("display/window/energy_saving/keep_screen_on"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	buttons_state = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | #if defined(OPENGL_ENABLED)
 | 
					
						
							|  |  |  | 	if (rendering_driver == "opengl") { | 
					
						
							|  |  |  | 		bool gl_initialization_error = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (RasterizerGLES2::is_viable() == OK) { | 
					
						
							|  |  |  | 			RasterizerGLES2::register_config(); | 
					
						
							|  |  |  | 			RasterizerGLES2::make_current(); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			gl_initialization_error = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (gl_initialization_error) { | 
					
						
							|  |  |  | 			OS::get_singleton()->alert("Your device does not support any of the supported OpenGL versions.\n" | 
					
						
							|  |  |  | 									   "Please try updating your Android version.", | 
					
						
							|  |  |  | 					"Unable to initialize video driver"); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(VULKAN_ENABLED)
 | 
					
						
							|  |  |  | 	context_vulkan = nullptr; | 
					
						
							|  |  |  | 	rendering_device_vulkan = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (rendering_driver == "vulkan") { | 
					
						
							|  |  |  | 		ANativeWindow *native_window = OS_Android::get_singleton()->get_native_window(); | 
					
						
							|  |  |  | 		ERR_FAIL_COND(!native_window); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		context_vulkan = memnew(VulkanContextAndroid); | 
					
						
							|  |  |  | 		if (context_vulkan->initialize() != OK) { | 
					
						
							|  |  |  | 			memdelete(context_vulkan); | 
					
						
							|  |  |  | 			context_vulkan = nullptr; | 
					
						
							|  |  |  | 			ERR_FAIL_MSG("Failed to initialize Vulkan context"); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Size2i display_size = OS_Android::get_singleton()->get_display_size(); | 
					
						
							|  |  |  | 		if (context_vulkan->window_create(native_window, display_size.width, display_size.height) == -1) { | 
					
						
							|  |  |  | 			memdelete(context_vulkan); | 
					
						
							|  |  |  | 			context_vulkan = nullptr; | 
					
						
							|  |  |  | 			ERR_FAIL_MSG("Failed to create Vulkan window."); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		rendering_device_vulkan = memnew(RenderingDeviceVulkan); | 
					
						
							|  |  |  | 		rendering_device_vulkan->initialize(context_vulkan); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 15:26:24 -03:00
										 |  |  | 		RendererCompositorRD::make_current(); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-28 15:19:37 +02:00
										 |  |  | 	Input::get_singleton()->set_event_dispatch_function(_dispatch_input_events); | 
					
						
							| 
									
										
										
										
											2020-07-14 19:37:36 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	r_error = OK; | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DisplayServerAndroid::~DisplayServerAndroid() { | 
					
						
							|  |  |  | #if defined(VULKAN_ENABLED)
 | 
					
						
							|  |  |  | 	if (rendering_driver == "vulkan") { | 
					
						
							|  |  |  | 		if (rendering_device_vulkan) { | 
					
						
							|  |  |  | 			rendering_device_vulkan->finalize(); | 
					
						
							|  |  |  | 			memdelete(rendering_device_vulkan); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (context_vulkan) { | 
					
						
							|  |  |  | 			memdelete(context_vulkan); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::process_joy_event(DisplayServerAndroid::JoypadEvent p_event) { | 
					
						
							|  |  |  | 	switch (p_event.type) { | 
					
						
							|  |  |  | 		case JOY_EVENT_BUTTON: | 
					
						
							| 
									
										
										
										
											2020-04-28 15:19:37 +02:00
										 |  |  | 			Input::get_singleton()->joy_button(p_event.device, p_event.index, p_event.pressed); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case JOY_EVENT_AXIS: | 
					
						
							| 
									
										
										
										
											2020-05-13 18:27:34 -04:00
										 |  |  | 			Input::JoyAxisValue value; | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			value.min = -1; | 
					
						
							|  |  |  | 			value.value = p_event.value; | 
					
						
							| 
									
										
										
										
											2020-04-28 15:19:37 +02:00
										 |  |  | 			Input::get_singleton()->joy_axis(p_event.device, p_event.index, value); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case JOY_EVENT_HAT: | 
					
						
							| 
									
										
										
										
											2020-04-28 15:19:37 +02:00
										 |  |  | 			Input::get_singleton()->joy_hat(p_event.device, p_event.hat); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-15 02:47:24 +03:00
										 |  |  | void DisplayServerAndroid::_set_key_modifier_state(Ref<InputEventWithModifiers> ev) { | 
					
						
							|  |  |  | 	ev->set_shift(shift_mem); | 
					
						
							|  |  |  | 	ev->set_alt(alt_mem); | 
					
						
							|  |  |  | 	ev->set_metakey(meta_mem); | 
					
						
							|  |  |  | 	ev->set_control(control_mem); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | void DisplayServerAndroid::process_key_event(int p_keycode, int p_scancode, int p_unicode_char, bool p_pressed) { | 
					
						
							| 
									
										
										
										
											2020-09-03 14:22:16 +03:00
										 |  |  | 	static char32_t prev_wc = 0; | 
					
						
							|  |  |  | 	char32_t unicode = p_unicode_char; | 
					
						
							|  |  |  | 	if ((p_unicode_char & 0xfffffc00) == 0xd800) { | 
					
						
							|  |  |  | 		if (prev_wc != 0) { | 
					
						
							|  |  |  | 			ERR_PRINT("invalid utf16 surrogate input"); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		prev_wc = unicode; | 
					
						
							|  |  |  | 		return; // Skip surrogate.
 | 
					
						
							|  |  |  | 	} else if ((unicode & 0xfffffc00) == 0xdc00) { | 
					
						
							|  |  |  | 		if (prev_wc == 0) { | 
					
						
							|  |  |  | 			ERR_PRINT("invalid utf16 surrogate input"); | 
					
						
							|  |  |  | 			return; // Skip invalid surrogate.
 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		unicode = (prev_wc << 10UL) + unicode - ((0xd800 << 10UL) + 0xdc00 - 0x10000); | 
					
						
							|  |  |  | 		prev_wc = 0; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		prev_wc = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	Ref<InputEventKey> ev; | 
					
						
							|  |  |  | 	ev.instance(); | 
					
						
							| 
									
										
										
										
											2020-09-03 14:22:16 +03:00
										 |  |  | 	int val = unicode; | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	int keycode = android_get_keysym(p_keycode); | 
					
						
							|  |  |  | 	int phy_keycode = android_get_keysym(p_scancode); | 
					
						
							| 
									
										
										
										
											2020-07-15 02:47:24 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (keycode == KEY_SHIFT) { | 
					
						
							|  |  |  | 		shift_mem = p_pressed; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (keycode == KEY_ALT) { | 
					
						
							|  |  |  | 		alt_mem = p_pressed; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (keycode == KEY_CONTROL) { | 
					
						
							|  |  |  | 		control_mem = p_pressed; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (keycode == KEY_META) { | 
					
						
							|  |  |  | 		meta_mem = p_pressed; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	ev->set_keycode(keycode); | 
					
						
							|  |  |  | 	ev->set_physical_keycode(phy_keycode); | 
					
						
							|  |  |  | 	ev->set_unicode(val); | 
					
						
							|  |  |  | 	ev->set_pressed(p_pressed); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-15 02:47:24 +03:00
										 |  |  | 	_set_key_modifier_state(ev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	if (val == '\n') { | 
					
						
							|  |  |  | 		ev->set_keycode(KEY_ENTER); | 
					
						
							|  |  |  | 	} else if (val == 61448) { | 
					
						
							|  |  |  | 		ev->set_keycode(KEY_BACKSPACE); | 
					
						
							|  |  |  | 		ev->set_unicode(KEY_BACKSPACE); | 
					
						
							|  |  |  | 	} else if (val == 61453) { | 
					
						
							|  |  |  | 		ev->set_keycode(KEY_ENTER); | 
					
						
							|  |  |  | 		ev->set_unicode(KEY_ENTER); | 
					
						
							|  |  |  | 	} else if (p_keycode == 4) { | 
					
						
							|  |  |  | 		OS_Android::get_singleton()->main_loop_request_go_back(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | void DisplayServerAndroid::process_touch(int p_event, int p_pointer, const Vector<DisplayServerAndroid::TouchPos> &p_points) { | 
					
						
							|  |  |  | 	switch (p_event) { | 
					
						
							|  |  |  | 		case AMOTION_EVENT_ACTION_DOWN: { //gesture begin
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			if (touch.size()) { | 
					
						
							|  |  |  | 				//end all if exist
 | 
					
						
							|  |  |  | 				for (int i = 0; i < touch.size(); i++) { | 
					
						
							|  |  |  | 					Ref<InputEventScreenTouch> ev; | 
					
						
							|  |  |  | 					ev.instance(); | 
					
						
							|  |  |  | 					ev->set_index(touch[i].id); | 
					
						
							|  |  |  | 					ev->set_pressed(false); | 
					
						
							|  |  |  | 					ev->set_position(touch[i].pos); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 					Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			touch.resize(p_points.size()); | 
					
						
							|  |  |  | 			for (int i = 0; i < p_points.size(); i++) { | 
					
						
							|  |  |  | 				touch.write[i].id = p_points[i].id; | 
					
						
							|  |  |  | 				touch.write[i].pos = p_points[i].pos; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			//send touch
 | 
					
						
							|  |  |  | 			for (int i = 0; i < touch.size(); i++) { | 
					
						
							|  |  |  | 				Ref<InputEventScreenTouch> ev; | 
					
						
							|  |  |  | 				ev.instance(); | 
					
						
							|  |  |  | 				ev->set_index(touch[i].id); | 
					
						
							|  |  |  | 				ev->set_pressed(true); | 
					
						
							|  |  |  | 				ev->set_position(touch[i].pos); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 				Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 		case AMOTION_EVENT_ACTION_MOVE: { //motion
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			ERR_FAIL_COND(touch.size() != p_points.size()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for (int i = 0; i < touch.size(); i++) { | 
					
						
							|  |  |  | 				int idx = -1; | 
					
						
							|  |  |  | 				for (int j = 0; j < p_points.size(); j++) { | 
					
						
							|  |  |  | 					if (touch[i].id == p_points[j].id) { | 
					
						
							|  |  |  | 						idx = j; | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				ERR_CONTINUE(idx == -1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (touch[i].pos == p_points[idx].pos) | 
					
						
							|  |  |  | 					continue; //no move unncesearily
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Ref<InputEventScreenDrag> ev; | 
					
						
							|  |  |  | 				ev.instance(); | 
					
						
							|  |  |  | 				ev->set_index(touch[i].id); | 
					
						
							|  |  |  | 				ev->set_position(p_points[idx].pos); | 
					
						
							|  |  |  | 				ev->set_relative(p_points[idx].pos - touch[i].pos); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 				Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 				touch.write[i].pos = p_points[idx].pos; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 		case AMOTION_EVENT_ACTION_CANCEL: | 
					
						
							|  |  |  | 		case AMOTION_EVENT_ACTION_UP: { //release
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			if (touch.size()) { | 
					
						
							|  |  |  | 				//end all if exist
 | 
					
						
							|  |  |  | 				for (int i = 0; i < touch.size(); i++) { | 
					
						
							|  |  |  | 					Ref<InputEventScreenTouch> ev; | 
					
						
							|  |  |  | 					ev.instance(); | 
					
						
							|  |  |  | 					ev->set_index(touch[i].id); | 
					
						
							|  |  |  | 					ev->set_pressed(false); | 
					
						
							|  |  |  | 					ev->set_position(touch[i].pos); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 					Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				touch.clear(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 		case AMOTION_EVENT_ACTION_POINTER_DOWN: { // add touch
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			for (int i = 0; i < p_points.size(); i++) { | 
					
						
							|  |  |  | 				if (p_points[i].id == p_pointer) { | 
					
						
							|  |  |  | 					TouchPos tp = p_points[i]; | 
					
						
							|  |  |  | 					touch.push_back(tp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					Ref<InputEventScreenTouch> ev; | 
					
						
							|  |  |  | 					ev.instance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					ev->set_index(tp.id); | 
					
						
							|  |  |  | 					ev->set_pressed(true); | 
					
						
							|  |  |  | 					ev->set_position(tp.pos); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 					Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 		case AMOTION_EVENT_ACTION_POINTER_UP: { // remove touch
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			for (int i = 0; i < touch.size(); i++) { | 
					
						
							|  |  |  | 				if (touch[i].id == p_pointer) { | 
					
						
							|  |  |  | 					Ref<InputEventScreenTouch> ev; | 
					
						
							|  |  |  | 					ev.instance(); | 
					
						
							|  |  |  | 					ev->set_index(touch[i].id); | 
					
						
							|  |  |  | 					ev->set_pressed(false); | 
					
						
							|  |  |  | 					ev->set_position(touch[i].pos); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 					Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 					touch.remove(i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::process_hover(int p_type, Point2 p_pos) { | 
					
						
							|  |  |  | 	// https://developer.android.com/reference/android/view/MotionEvent.html#ACTION_HOVER_ENTER
 | 
					
						
							|  |  |  | 	switch (p_type) { | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 		case AMOTION_EVENT_ACTION_HOVER_MOVE: // hover move
 | 
					
						
							|  |  |  | 		case AMOTION_EVENT_ACTION_HOVER_ENTER: // hover enter
 | 
					
						
							|  |  |  | 		case AMOTION_EVENT_ACTION_HOVER_EXIT: { // hover exit
 | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			Ref<InputEventMouseMotion> ev; | 
					
						
							|  |  |  | 			ev.instance(); | 
					
						
							| 
									
										
										
										
											2020-07-15 02:47:24 +03:00
										 |  |  | 			_set_key_modifier_state(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			ev->set_position(p_pos); | 
					
						
							|  |  |  | 			ev->set_global_position(p_pos); | 
					
						
							|  |  |  | 			ev->set_relative(p_pos - hover_prev_pos); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 			Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 			hover_prev_pos = p_pos; | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-01 03:30:40 +02:00
										 |  |  | void DisplayServerAndroid::process_mouse_event(int input_device, int event_action, int event_android_buttons_mask, Point2 event_pos, float event_vertical_factor, float event_horizontal_factor) { | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	int event_buttons_mask = _android_button_mask_to_godot_button_mask(event_android_buttons_mask); | 
					
						
							|  |  |  | 	switch (event_action) { | 
					
						
							|  |  |  | 		case AMOTION_EVENT_ACTION_BUTTON_PRESS: | 
					
						
							|  |  |  | 		case AMOTION_EVENT_ACTION_BUTTON_RELEASE: { | 
					
						
							|  |  |  | 			Ref<InputEventMouseButton> ev; | 
					
						
							|  |  |  | 			ev.instance(); | 
					
						
							|  |  |  | 			_set_key_modifier_state(ev); | 
					
						
							| 
									
										
										
										
											2020-11-01 03:30:40 +02:00
										 |  |  | 			if ((input_device & AINPUT_SOURCE_MOUSE) == AINPUT_SOURCE_MOUSE) { | 
					
						
							|  |  |  | 				ev->set_position(event_pos); | 
					
						
							|  |  |  | 				ev->set_global_position(event_pos); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				ev->set_position(hover_prev_pos); | 
					
						
							|  |  |  | 				ev->set_global_position(hover_prev_pos); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 			ev->set_pressed(event_action == AMOTION_EVENT_ACTION_BUTTON_PRESS); | 
					
						
							|  |  |  | 			int changed_button_mask = buttons_state ^ event_buttons_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			buttons_state = event_buttons_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ev->set_button_index(_button_index_from_mask(changed_button_mask)); | 
					
						
							|  |  |  | 			ev->set_button_mask(event_buttons_mask); | 
					
						
							|  |  |  | 			Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case AMOTION_EVENT_ACTION_MOVE: { | 
					
						
							|  |  |  | 			Ref<InputEventMouseMotion> ev; | 
					
						
							|  |  |  | 			ev.instance(); | 
					
						
							|  |  |  | 			_set_key_modifier_state(ev); | 
					
						
							| 
									
										
										
										
											2020-11-01 03:30:40 +02:00
										 |  |  | 			if ((input_device & AINPUT_SOURCE_MOUSE) == AINPUT_SOURCE_MOUSE) { | 
					
						
							|  |  |  | 				ev->set_position(event_pos); | 
					
						
							|  |  |  | 				ev->set_global_position(event_pos); | 
					
						
							|  |  |  | 				ev->set_relative(event_pos - hover_prev_pos); | 
					
						
							|  |  |  | 				hover_prev_pos = event_pos; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				ev->set_position(hover_prev_pos); | 
					
						
							|  |  |  | 				ev->set_global_position(hover_prev_pos); | 
					
						
							|  |  |  | 				ev->set_relative(event_pos); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 			ev->set_button_mask(event_buttons_mask); | 
					
						
							|  |  |  | 			Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 		case AMOTION_EVENT_ACTION_SCROLL: { | 
					
						
							|  |  |  | 			Ref<InputEventMouseButton> ev; | 
					
						
							|  |  |  | 			ev.instance(); | 
					
						
							| 
									
										
										
										
											2020-11-01 03:30:40 +02:00
										 |  |  | 			if ((input_device & AINPUT_SOURCE_MOUSE) == AINPUT_SOURCE_MOUSE) { | 
					
						
							|  |  |  | 				ev->set_position(event_pos); | 
					
						
							|  |  |  | 				ev->set_global_position(event_pos); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				ev->set_position(hover_prev_pos); | 
					
						
							|  |  |  | 				ev->set_global_position(hover_prev_pos); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 			ev->set_pressed(true); | 
					
						
							|  |  |  | 			buttons_state = event_buttons_mask; | 
					
						
							|  |  |  | 			if (event_vertical_factor > 0) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 				_wheel_button_click(event_buttons_mask, ev, MOUSE_BUTTON_WHEEL_UP, event_vertical_factor); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 			} else if (event_vertical_factor < 0) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 				_wheel_button_click(event_buttons_mask, ev, MOUSE_BUTTON_WHEEL_DOWN, -event_vertical_factor); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (event_horizontal_factor > 0) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 				_wheel_button_click(event_buttons_mask, ev, MOUSE_BUTTON_WHEEL_RIGHT, event_horizontal_factor); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 			} else if (event_horizontal_factor < 0) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 				_wheel_button_click(event_buttons_mask, ev, MOUSE_BUTTON_WHEEL_LEFT, -event_horizontal_factor); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::_wheel_button_click(int event_buttons_mask, const Ref<InputEventMouseButton> &ev, int wheel_button, float factor) { | 
					
						
							|  |  |  | 	Ref<InputEventMouseButton> evd = ev->duplicate(); | 
					
						
							|  |  |  | 	_set_key_modifier_state(evd); | 
					
						
							|  |  |  | 	evd->set_button_index(wheel_button); | 
					
						
							|  |  |  | 	evd->set_button_mask(event_buttons_mask ^ (1 << (wheel_button - 1))); | 
					
						
							|  |  |  | 	evd->set_factor(factor); | 
					
						
							|  |  |  | 	Input::get_singleton()->accumulate_input_event(evd); | 
					
						
							|  |  |  | 	Ref<InputEventMouseButton> evdd = evd->duplicate(); | 
					
						
							|  |  |  | 	evdd->set_pressed(false); | 
					
						
							|  |  |  | 	evdd->set_button_mask(event_buttons_mask); | 
					
						
							|  |  |  | 	Input::get_singleton()->accumulate_input_event(evdd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::process_double_tap(int event_android_button_mask, Point2 p_pos) { | 
					
						
							|  |  |  | 	int event_button_mask = _android_button_mask_to_godot_button_mask(event_android_button_mask); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	Ref<InputEventMouseButton> ev; | 
					
						
							|  |  |  | 	ev.instance(); | 
					
						
							| 
									
										
										
										
											2020-07-15 02:47:24 +03:00
										 |  |  | 	_set_key_modifier_state(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	ev->set_position(p_pos); | 
					
						
							|  |  |  | 	ev->set_global_position(p_pos); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	ev->set_pressed(event_button_mask != 0); | 
					
						
							|  |  |  | 	ev->set_button_index(_button_index_from_mask(event_button_mask)); | 
					
						
							|  |  |  | 	ev->set_button_mask(event_button_mask); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	ev->set_doubleclick(true); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int DisplayServerAndroid::_button_index_from_mask(int button_mask) { | 
					
						
							|  |  |  | 	switch (button_mask) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 		case MOUSE_BUTTON_MASK_LEFT: | 
					
						
							|  |  |  | 			return MOUSE_BUTTON_LEFT; | 
					
						
							|  |  |  | 		case MOUSE_BUTTON_MASK_RIGHT: | 
					
						
							|  |  |  | 			return MOUSE_BUTTON_RIGHT; | 
					
						
							|  |  |  | 		case MOUSE_BUTTON_MASK_MIDDLE: | 
					
						
							|  |  |  | 			return MOUSE_BUTTON_MIDDLE; | 
					
						
							|  |  |  | 		case MOUSE_BUTTON_MASK_XBUTTON1: | 
					
						
							|  |  |  | 			return MOUSE_BUTTON_XBUTTON1; | 
					
						
							|  |  |  | 		case MOUSE_BUTTON_MASK_XBUTTON2: | 
					
						
							|  |  |  | 			return MOUSE_BUTTON_XBUTTON2; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 		default: | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::process_scroll(Point2 p_pos) { | 
					
						
							|  |  |  | 	Ref<InputEventPanGesture> ev; | 
					
						
							|  |  |  | 	ev.instance(); | 
					
						
							| 
									
										
										
										
											2020-07-15 02:47:24 +03:00
										 |  |  | 	_set_key_modifier_state(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	ev->set_position(p_pos); | 
					
						
							|  |  |  | 	ev->set_delta(p_pos - scroll_prev_pos); | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	Input::get_singleton()->accumulate_input_event(ev); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | 	scroll_prev_pos = p_pos; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::process_accelerometer(const Vector3 &p_accelerometer) { | 
					
						
							| 
									
										
										
										
											2020-04-28 15:19:37 +02:00
										 |  |  | 	Input::get_singleton()->set_accelerometer(p_accelerometer); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::process_gravity(const Vector3 &p_gravity) { | 
					
						
							| 
									
										
										
										
											2020-04-28 15:19:37 +02:00
										 |  |  | 	Input::get_singleton()->set_gravity(p_gravity); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::process_magnetometer(const Vector3 &p_magnetometer) { | 
					
						
							| 
									
										
										
										
											2020-04-28 15:19:37 +02:00
										 |  |  | 	Input::get_singleton()->set_magnetometer(p_magnetometer); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayServerAndroid::process_gyroscope(const Vector3 &p_gyroscope) { | 
					
						
							| 
									
										
										
										
											2020-04-28 15:19:37 +02:00
										 |  |  | 	Input::get_singleton()->set_gyroscope(p_gyroscope); | 
					
						
							| 
									
										
										
										
											2020-03-27 17:30:18 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-01 03:30:40 +02:00
										 |  |  | void DisplayServerAndroid::mouse_set_mode(MouseMode p_mode) { | 
					
						
							|  |  |  | 	if (mouse_mode == p_mode) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p_mode == MouseMode::MOUSE_MODE_CAPTURED) { | 
					
						
							|  |  |  | 		OS_Android::get_singleton()->get_godot_java()->get_godot_view()->request_pointer_capture(); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		OS_Android::get_singleton()->get_godot_java()->get_godot_view()->release_pointer_capture(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mouse_mode = p_mode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DisplayServer::MouseMode DisplayServerAndroid::mouse_get_mode() const { | 
					
						
							|  |  |  | 	return mouse_mode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | Point2i DisplayServerAndroid::mouse_get_position() const { | 
					
						
							|  |  |  | 	return hover_prev_pos; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int DisplayServerAndroid::mouse_get_button_state() const { | 
					
						
							|  |  |  | 	return buttons_state; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int DisplayServerAndroid::_android_button_mask_to_godot_button_mask(int android_button_mask) { | 
					
						
							|  |  |  | 	int godot_button_mask = 0; | 
					
						
							|  |  |  | 	if (android_button_mask & AMOTION_EVENT_BUTTON_PRIMARY) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 		godot_button_mask |= MOUSE_BUTTON_MASK_LEFT; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (android_button_mask & AMOTION_EVENT_BUTTON_SECONDARY) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 		godot_button_mask |= MOUSE_BUTTON_MASK_RIGHT; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (android_button_mask & AMOTION_EVENT_BUTTON_TERTIARY) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 		godot_button_mask |= MOUSE_BUTTON_MASK_MIDDLE; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (android_button_mask & AMOTION_EVENT_BUTTON_BACK) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 		godot_button_mask |= MOUSE_BUTTON_MASK_XBUTTON1; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (android_button_mask & AMOTION_EVENT_BUTTON_SECONDARY) { | 
					
						
							| 
									
										
										
										
											2021-01-07 22:37:37 -05:00
										 |  |  | 		godot_button_mask |= MOUSE_BUTTON_MASK_XBUTTON2; | 
					
						
							| 
									
										
										
										
											2020-10-08 02:37:58 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return godot_button_mask; | 
					
						
							|  |  |  | } |