2020-03-01 23:14:37 +01:00
/**************************************************************************/
/* display_server_x11.cpp */
/**************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/**************************************************************************/
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
/* */
/* 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_x11.h"
# ifdef X11_ENABLED
2023-06-08 14:51:32 +02:00
# include "x11/detect_prime_x11.h"
# include "x11/key_mapping_x11.h"
2020-11-07 23:33:38 +01:00
# include "core/config/project_settings.h"
2022-02-04 17:21:07 +01:00
# include "core/math/math_funcs.h"
2020-11-07 23:33:38 +01:00
# include "core/string/print_string.h"
2020-11-18 19:11:30 +01:00
# include "core/string/ustring.h"
2023-09-15 02:58:13 +02:00
# include "drivers/png/png_driver_common.h"
2020-06-29 13:29:31 +02:00
# include "main/main.h"
2020-03-01 23:14:37 +01:00
# if defined(VULKAN_ENABLED)
2020-12-04 19:26:24 +01:00
# include "servers/rendering/renderer_rd/renderer_compositor_rd.h"
2020-03-01 23:14:37 +01:00
# endif
2021-10-26 17:18:39 +02:00
# if defined(GLES3_ENABLED)
# include "drivers/gles3/rasterizer_gles3.h"
2020-11-18 19:11:30 +01:00
# endif
2022-10-03 11:43:20 +02:00
# include <dlfcn.h>
2020-12-02 15:43:29 +01:00
# include <limits.h>
2020-03-01 23:14:37 +01:00
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
2022-10-03 11:43:20 +02:00
# include <sys/stat.h>
# include <sys/types.h>
# include <unistd.h>
2020-03-01 23:14:37 +01:00
2023-06-08 14:51:32 +02:00
# undef CursorShape
# include <X11/XKBlib.h>
2020-03-01 23:14:37 +01:00
// ICCCM
# define WM_NormalState 1L // window normal state
# define WM_IconicState 3L // window minimized
// EWMH
# define _NET_WM_STATE_REMOVE 0L // remove/unset property
# define _NET_WM_STATE_ADD 1L // add/set property
// 2.2 is the first release with multitouch
# define XINPUT_CLIENT_VERSION_MAJOR 2
# define XINPUT_CLIENT_VERSION_MINOR 2
# define VALUATOR_ABSX 0
# define VALUATOR_ABSY 1
# define VALUATOR_PRESSURE 2
# define VALUATOR_TILTX 3
# define VALUATOR_TILTY 4
2020-08-22 13:01:49 +02:00
//#define DISPLAY_SERVER_X11_DEBUG_LOGS_ENABLED
# ifdef DISPLAY_SERVER_X11_DEBUG_LOGS_ENABLED
# define DEBUG_LOG_X11(...) printf(__VA_ARGS__)
# else
# define DEBUG_LOG_X11(...)
# endif
2020-03-01 23:14:37 +01:00
static const double abs_resolution_mult = 10000.0 ;
static const double abs_resolution_range_mult = 10.0 ;
2020-12-16 08:39:17 +01:00
// Hints for X11 fullscreen
struct Hints {
unsigned long flags = 0 ;
unsigned long functions = 0 ;
unsigned long decorations = 0 ;
long inputMode = 0 ;
unsigned long status = 0 ;
} ;
2022-03-04 10:50:24 +01:00
static String get_atom_name ( Display * p_disp , Atom p_atom ) {
char * name = XGetAtomName ( p_disp , p_atom ) ;
ERR_FAIL_NULL_V_MSG ( name , String ( ) , " Atom is invalid. " ) ;
String ret ;
ret . parse_utf8 ( name ) ;
XFree ( name ) ;
return ret ;
}
2020-03-01 23:14:37 +01:00
bool DisplayServerX11 : : has_feature ( Feature p_feature ) const {
switch ( p_feature ) {
2024-01-19 18:41:01 +01:00
# ifndef DISABLE_DEPRECATED
case FEATURE_GLOBAL_MENU : {
return ( native_menu & & native_menu - > has_feature ( NativeMenu : : FEATURE_GLOBAL_MENU ) ) ;
} break ;
# endif
2020-03-01 23:14:37 +01:00
case FEATURE_SUBWINDOWS :
# ifdef TOUCH_ENABLED
case FEATURE_TOUCHSCREEN :
# endif
case FEATURE_MOUSE :
case FEATURE_MOUSE_WARP :
case FEATURE_CLIPBOARD :
case FEATURE_CURSOR_SHAPE :
case FEATURE_CUSTOM_CURSOR_SHAPE :
case FEATURE_IME :
case FEATURE_WINDOW_TRANSPARENCY :
//case FEATURE_HIDPI:
case FEATURE_ICON :
2023-08-01 09:18:41 +02:00
# ifdef DBUS_ENABLED
2024-03-26 14:18:06 +01:00
case FEATURE_NATIVE_DIALOG_FILE :
2023-08-01 09:18:41 +02:00
# endif
2024-03-26 14:18:06 +01:00
//case FEATURE_NATIVE_DIALOG:
//case FEATURE_NATIVE_DIALOG_INPUT:
2022-10-26 20:39:31 +02:00
//case FEATURE_NATIVE_ICON:
2020-03-01 23:14:37 +01:00
case FEATURE_SWAP_BUFFERS :
2020-04-02 23:46:34 +02:00
# ifdef DBUS_ENABLED
case FEATURE_KEEP_SCREEN_ON :
# endif
2021-10-20 21:25:09 +02:00
case FEATURE_CLIPBOARD_PRIMARY :
2021-11-04 13:33:37 +01:00
case FEATURE_TEXT_TO_SPEECH :
2020-03-01 23:14:37 +01:00
return true ;
2024-06-12 09:05:17 +02:00
case FEATURE_SCREEN_CAPTURE :
return ! xwayland ;
2020-03-01 23:14:37 +01:00
default : {
}
}
return false ;
}
2020-05-14 14:29:06 +02:00
2020-03-01 23:14:37 +01:00
String DisplayServerX11 : : get_name ( ) const {
return " X11 " ;
}
void DisplayServerX11 : : _update_real_mouse_position ( const WindowData & wd ) {
Window root_return , child_return ;
int root_x , root_y , win_x , win_y ;
unsigned int mask_return ;
Bool xquerypointer_result = XQueryPointer ( x11_display , wd . x11_window , & root_return , & child_return , & root_x , & root_y ,
& win_x , & win_y , & mask_return ) ;
if ( xquerypointer_result ) {
if ( win_x > 0 & & win_y > 0 & & win_x < = wd . size . width & & win_y < = wd . size . height ) {
last_mouse_pos . x = win_x ;
last_mouse_pos . y = win_y ;
last_mouse_pos_valid = true ;
2020-04-28 15:19:37 +02:00
Input : : get_singleton ( ) - > set_mouse_position ( last_mouse_pos ) ;
2020-03-01 23:14:37 +01:00
}
}
}
bool DisplayServerX11 : : _refresh_device_info ( ) {
int event_base , error_base ;
print_verbose ( " XInput: Refreshing devices. " ) ;
if ( ! XQueryExtension ( x11_display , " XInputExtension " , & xi . opcode , & event_base , & error_base ) ) {
print_verbose ( " XInput extension not available. Please upgrade your distribution. " ) ;
return false ;
}
int xi_major_query = XINPUT_CLIENT_VERSION_MAJOR ;
int xi_minor_query = XINPUT_CLIENT_VERSION_MINOR ;
if ( XIQueryVersion ( x11_display , & xi_major_query , & xi_minor_query ) ! = Success ) {
print_verbose ( vformat ( " XInput 2 not available (server supports %d.%d). " , xi_major_query , xi_minor_query ) ) ;
xi . opcode = 0 ;
return false ;
}
if ( xi_major_query < XINPUT_CLIENT_VERSION_MAJOR | | ( xi_major_query = = XINPUT_CLIENT_VERSION_MAJOR & & xi_minor_query < XINPUT_CLIENT_VERSION_MINOR ) ) {
print_verbose ( vformat ( " XInput %d.%d not available (server supports %d.%d). Touch input unavailable. " ,
XINPUT_CLIENT_VERSION_MAJOR , XINPUT_CLIENT_VERSION_MINOR , xi_major_query , xi_minor_query ) ) ;
}
xi . absolute_devices . clear ( ) ;
xi . touch_devices . clear ( ) ;
2022-06-19 11:58:24 +02:00
xi . pen_inverted_devices . clear ( ) ;
2023-05-10 20:27:14 +02:00
xi . last_relative_time = 0 ;
2020-03-01 23:14:37 +01:00
int dev_count ;
XIDeviceInfo * info = XIQueryDevice ( x11_display , XIAllDevices , & dev_count ) ;
for ( int i = 0 ; i < dev_count ; i + + ) {
XIDeviceInfo * dev = & info [ i ] ;
2020-05-14 16:41:43 +02:00
if ( ! dev - > enabled ) {
2020-03-01 23:14:37 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2022-06-19 11:58:24 +02:00
if ( ! ( dev - > use = = XISlavePointer | | dev - > use = = XIFloatingSlave ) ) {
2020-03-01 23:14:37 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
bool direct_touch = false ;
bool absolute_mode = false ;
int resolution_x = 0 ;
int resolution_y = 0 ;
2020-04-10 12:41:41 +02:00
double abs_x_min = 0 ;
double abs_x_max = 0 ;
double abs_y_min = 0 ;
double abs_y_max = 0 ;
double pressure_min = 0 ;
double pressure_max = 0 ;
double tilt_x_min = 0 ;
double tilt_x_max = 0 ;
double tilt_y_min = 0 ;
double tilt_y_max = 0 ;
2020-03-01 23:14:37 +01:00
for ( int j = 0 ; j < dev - > num_classes ; j + + ) {
# ifdef TOUCH_ENABLED
if ( dev - > classes [ j ] - > type = = XITouchClass & & ( ( XITouchClassInfo * ) dev - > classes [ j ] ) - > mode = = XIDirectTouch ) {
direct_touch = true ;
}
# endif
if ( dev - > classes [ j ] - > type = = XIValuatorClass ) {
XIValuatorClassInfo * class_info = ( XIValuatorClassInfo * ) dev - > classes [ j ] ;
if ( class_info - > number = = VALUATOR_ABSX & & class_info - > mode = = XIModeAbsolute ) {
resolution_x = class_info - > resolution ;
2020-04-10 12:41:41 +02:00
abs_x_min = class_info - > min ;
2022-01-02 12:50:16 +01:00
abs_x_max = class_info - > max ;
2020-03-01 23:14:37 +01:00
absolute_mode = true ;
} else if ( class_info - > number = = VALUATOR_ABSY & & class_info - > mode = = XIModeAbsolute ) {
resolution_y = class_info - > resolution ;
2020-04-10 12:41:41 +02:00
abs_y_min = class_info - > min ;
abs_y_max = class_info - > max ;
2020-03-01 23:14:37 +01:00
absolute_mode = true ;
} else if ( class_info - > number = = VALUATOR_PRESSURE & & class_info - > mode = = XIModeAbsolute ) {
2020-04-10 12:41:41 +02:00
pressure_min = class_info - > min ;
pressure_max = class_info - > max ;
2020-03-01 23:14:37 +01:00
} else if ( class_info - > number = = VALUATOR_TILTX & & class_info - > mode = = XIModeAbsolute ) {
2020-04-10 12:41:41 +02:00
tilt_x_min = class_info - > min ;
tilt_x_max = class_info - > max ;
2020-03-01 23:14:37 +01:00
} else if ( class_info - > number = = VALUATOR_TILTY & & class_info - > mode = = XIModeAbsolute ) {
2022-01-02 12:50:16 +01:00
tilt_y_min = class_info - > min ;
tilt_y_max = class_info - > max ;
2020-03-01 23:14:37 +01:00
}
}
}
if ( direct_touch ) {
xi . touch_devices . push_back ( dev - > deviceid ) ;
print_verbose ( " XInput: Using touch device: " + String ( dev - > name ) ) ;
}
if ( absolute_mode ) {
// If no resolution was reported, use the min/max ranges.
if ( resolution_x < = 0 ) {
2020-04-10 12:41:41 +02:00
resolution_x = ( abs_x_max - abs_x_min ) * abs_resolution_range_mult ;
2020-03-01 23:14:37 +01:00
}
if ( resolution_y < = 0 ) {
2020-04-10 12:41:41 +02:00
resolution_y = ( abs_y_max - abs_y_min ) * abs_resolution_range_mult ;
2020-03-01 23:14:37 +01:00
}
xi . absolute_devices [ dev - > deviceid ] = Vector2 ( abs_resolution_mult / resolution_x , abs_resolution_mult / resolution_y ) ;
print_verbose ( " XInput: Absolute pointing device: " + String ( dev - > name ) ) ;
}
xi . pressure = 0 ;
2020-04-10 12:41:41 +02:00
xi . pen_pressure_range [ dev - > deviceid ] = Vector2 ( pressure_min , pressure_max ) ;
xi . pen_tilt_x_range [ dev - > deviceid ] = Vector2 ( tilt_x_min , tilt_x_max ) ;
xi . pen_tilt_y_range [ dev - > deviceid ] = Vector2 ( tilt_y_min , tilt_y_max ) ;
2022-07-20 00:19:38 +02:00
xi . pen_inverted_devices [ dev - > deviceid ] = String ( dev - > name ) . findn ( " eraser " ) > 0 ;
2020-03-01 23:14:37 +01:00
}
XIFreeDeviceInfo ( info ) ;
# ifdef TOUCH_ENABLED
if ( ! xi . touch_devices . size ( ) ) {
print_verbose ( " XInput: No touch devices found. " ) ;
}
# endif
return true ;
}
void DisplayServerX11 : : _flush_mouse_motion ( ) {
2020-09-09 12:43:20 +02:00
// Block events polling while flushing motion events.
MutexLock mutex_lock ( events_mutex ) ;
for ( uint32_t event_index = 0 ; event_index < polled_events . size ( ) ; + + event_index ) {
XEvent & event = polled_events [ event_index ] ;
if ( XGetEventData ( x11_display , & event . xcookie ) & & event . xcookie . type = = GenericEvent & & event . xcookie . extension = = xi . opcode ) {
XIDeviceEvent * event_data = ( XIDeviceEvent * ) event . xcookie . data ;
if ( event_data - > evtype = = XI_RawMotion ) {
XFreeEventData ( x11_display , & event . xcookie ) ;
2021-07-04 00:17:03 +02:00
polled_events . remove_at ( event_index - - ) ;
2020-09-09 12:43:20 +02:00
continue ;
2020-03-01 23:14:37 +01:00
}
2020-09-09 12:43:20 +02:00
XFreeEventData ( x11_display , & event . xcookie ) ;
2020-03-01 23:14:37 +01:00
break ;
}
}
xi . relative_motion . x = 0 ;
xi . relative_motion . y = 0 ;
}
2021-11-04 13:33:37 +01:00
# ifdef SPEECHD_ENABLED
bool DisplayServerX11 : : tts_is_speaking ( ) const {
2023-09-09 17:46:44 +02:00
ERR_FAIL_NULL_V_MSG ( tts , false , " Enable the \" audio/general/text_to_speech \" project setting to use text-to-speech. " ) ;
2021-11-04 13:33:37 +01:00
return tts - > is_speaking ( ) ;
}
bool DisplayServerX11 : : tts_is_paused ( ) const {
2023-09-09 17:46:44 +02:00
ERR_FAIL_NULL_V_MSG ( tts , false , " Enable the \" audio/general/text_to_speech \" project setting to use text-to-speech. " ) ;
2021-11-04 13:33:37 +01:00
return tts - > is_paused ( ) ;
}
2022-08-05 03:41:48 +02:00
TypedArray < Dictionary > DisplayServerX11 : : tts_get_voices ( ) const {
2023-09-09 17:46:44 +02:00
ERR_FAIL_NULL_V_MSG ( tts , TypedArray < Dictionary > ( ) , " Enable the \" audio/general/text_to_speech \" project setting to use text-to-speech. " ) ;
2021-11-04 13:33:37 +01:00
return tts - > get_voices ( ) ;
}
void DisplayServerX11 : : tts_speak ( const String & p_text , const String & p_voice , int p_volume , float p_pitch , float p_rate , int p_utterance_id , bool p_interrupt ) {
2023-09-09 17:46:44 +02:00
ERR_FAIL_NULL_MSG ( tts , " Enable the \" audio/general/text_to_speech \" project setting to use text-to-speech. " ) ;
2021-11-04 13:33:37 +01:00
tts - > speak ( p_text , p_voice , p_volume , p_pitch , p_rate , p_utterance_id , p_interrupt ) ;
}
void DisplayServerX11 : : tts_pause ( ) {
2023-09-09 17:46:44 +02:00
ERR_FAIL_NULL_MSG ( tts , " Enable the \" audio/general/text_to_speech \" project setting to use text-to-speech. " ) ;
2021-11-04 13:33:37 +01:00
tts - > pause ( ) ;
}
void DisplayServerX11 : : tts_resume ( ) {
2023-09-09 17:46:44 +02:00
ERR_FAIL_NULL_MSG ( tts , " Enable the \" audio/general/text_to_speech \" project setting to use text-to-speech. " ) ;
2021-11-04 13:33:37 +01:00
tts - > resume ( ) ;
}
void DisplayServerX11 : : tts_stop ( ) {
2023-09-09 17:46:44 +02:00
ERR_FAIL_NULL_MSG ( tts , " Enable the \" audio/general/text_to_speech \" project setting to use text-to-speech. " ) ;
2021-11-04 13:33:37 +01:00
tts - > stop ( ) ;
}
# endif
2022-08-30 17:04:17 +02:00
# ifdef DBUS_ENABLED
bool DisplayServerX11 : : is_dark_mode_supported ( ) const {
return portal_desktop - > is_supported ( ) ;
}
bool DisplayServerX11 : : is_dark_mode ( ) const {
switch ( portal_desktop - > get_appearance_color_scheme ( ) ) {
case 1 :
// Prefers dark theme.
return true ;
case 2 :
// Prefers light theme.
return false ;
default :
// Preference unknown.
return false ;
}
}
2024-01-19 19:46:26 +01:00
void DisplayServerX11 : : set_system_theme_change_callback ( const Callable & p_callable ) {
portal_desktop - > set_system_theme_change_callback ( p_callable ) ;
}
2023-08-01 09:18:41 +02:00
Error DisplayServerX11 : : file_dialog_show ( const String & p_title , const String & p_current_directory , const String & p_filename , bool p_show_hidden , FileDialogMode p_mode , const Vector < String > & p_filters , const Callable & p_callback ) {
2023-08-24 07:49:18 +02:00
WindowID window_id = last_focused_window ;
2023-08-01 09:18:41 +02:00
if ( ! windows . has ( window_id ) ) {
window_id = MAIN_WINDOW_ID ;
}
String xid = vformat ( " x11:%x " , ( uint64_t ) windows [ window_id ] . x11_window ) ;
2023-10-13 11:37:46 +02:00
return portal_desktop - > file_dialog_show ( last_focused_window , xid , p_title , p_current_directory , String ( ) , p_filename , p_mode , p_filters , TypedArray < Dictionary > ( ) , p_callback , false ) ;
}
Error DisplayServerX11 : : file_dialog_with_options_show ( const String & p_title , const String & p_current_directory , const String & p_root , const String & p_filename , bool p_show_hidden , FileDialogMode p_mode , const Vector < String > & p_filters , const TypedArray < Dictionary > & p_options , const Callable & p_callback ) {
WindowID window_id = last_focused_window ;
if ( ! windows . has ( window_id ) ) {
window_id = MAIN_WINDOW_ID ;
}
String xid = vformat ( " x11:%x " , ( uint64_t ) windows [ window_id ] . x11_window ) ;
return portal_desktop - > file_dialog_show ( last_focused_window , xid , p_title , p_current_directory , p_root , p_filename , p_mode , p_filters , p_options , p_callback , true ) ;
2023-08-01 09:18:41 +02:00
}
2022-08-30 17:04:17 +02:00
# endif
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : mouse_set_mode ( MouseMode p_mode ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-05-14 16:41:43 +02:00
if ( p_mode = = mouse_mode ) {
2020-03-01 23:14:37 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
2021-03-31 00:35:08 +02:00
if ( mouse_mode = = MOUSE_MODE_CAPTURED | | mouse_mode = = MOUSE_MODE_CONFINED | | mouse_mode = = MOUSE_MODE_CONFINED_HIDDEN ) {
2020-03-01 23:14:37 +01:00
XUngrabPointer ( x11_display , CurrentTime ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
// The only modes that show a cursor are VISIBLE and CONFINED
2022-11-10 18:23:10 +01:00
bool show_cursor = ( p_mode = = MOUSE_MODE_VISIBLE | | p_mode = = MOUSE_MODE_CONFINED ) ;
bool previously_shown = ( mouse_mode = = MOUSE_MODE_VISIBLE | | mouse_mode = = MOUSE_MODE_CONFINED ) ;
if ( show_cursor & & ! previously_shown ) {
WindowID window_id = get_window_at_screen_position ( mouse_get_position ( ) ) ;
2023-08-05 01:52:59 +02:00
if ( window_id ! = INVALID_WINDOW_ID & & window_mouseover_id ! = window_id ) {
if ( window_mouseover_id ! = INVALID_WINDOW_ID ) {
_send_window_event ( windows [ window_mouseover_id ] , WINDOW_EVENT_MOUSE_EXIT ) ;
}
window_mouseover_id = window_id ;
2022-11-10 18:23:10 +01:00
_send_window_event ( windows [ window_id ] , WINDOW_EVENT_MOUSE_ENTER ) ;
}
}
2020-03-01 23:14:37 +01:00
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
2022-11-10 18:23:10 +01:00
if ( show_cursor ) {
2021-08-09 22:13:42 +02:00
XDefineCursor ( x11_display , E . value . x11_window , cursors [ current_cursor ] ) ; // show cursor
2020-03-01 23:14:37 +01:00
} else {
2021-08-09 22:13:42 +02:00
XDefineCursor ( x11_display , E . value . x11_window , null_cursor ) ; // hide cursor
2020-03-01 23:14:37 +01:00
}
}
mouse_mode = p_mode ;
2021-03-31 00:35:08 +02:00
if ( mouse_mode = = MOUSE_MODE_CAPTURED | | mouse_mode = = MOUSE_MODE_CONFINED | | mouse_mode = = MOUSE_MODE_CONFINED_HIDDEN ) {
2020-03-01 23:14:37 +01:00
//flush pending motion events
_flush_mouse_motion ( ) ;
2022-09-07 19:31:57 +02:00
WindowID window_id = _get_focused_window_or_popup ( ) ;
if ( ! windows . has ( window_id ) ) {
window_id = MAIN_WINDOW_ID ;
}
2022-02-01 03:49:51 +01:00
WindowData & window = windows [ window_id ] ;
2020-03-01 23:14:37 +01:00
if ( XGrabPointer (
2022-02-01 03:49:51 +01:00
x11_display , window . x11_window , True ,
2020-03-01 23:14:37 +01:00
ButtonPressMask | ButtonReleaseMask | PointerMotionMask ,
2022-02-01 03:49:51 +01:00
GrabModeAsync , GrabModeAsync , window . x11_window , None , CurrentTime ) ! = GrabSuccess ) {
2020-03-01 23:14:37 +01:00
ERR_PRINT ( " NO GRAB " ) ;
}
if ( mouse_mode = = MOUSE_MODE_CAPTURED ) {
2022-02-01 03:49:51 +01:00
center . x = window . size . width / 2 ;
center . y = window . size . height / 2 ;
2020-03-01 23:14:37 +01:00
2022-02-01 03:49:51 +01:00
XWarpPointer ( x11_display , None , window . x11_window ,
2020-03-01 23:14:37 +01:00
0 , 0 , 0 , 0 , ( int ) center . x , ( int ) center . y ) ;
2020-04-28 15:19:37 +02:00
Input : : get_singleton ( ) - > set_mouse_position ( center ) ;
2020-03-01 23:14:37 +01:00
}
} else {
do_mouse_warp = false ;
}
XFlush ( x11_display ) ;
}
2020-05-14 14:29:06 +02:00
2020-03-01 23:14:37 +01:00
DisplayServerX11 : : MouseMode DisplayServerX11 : : mouse_get_mode ( ) const {
return mouse_mode ;
}
2022-03-27 11:17:36 +02:00
void DisplayServerX11 : : warp_mouse ( const Point2i & p_position ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-01 23:14:37 +01:00
if ( mouse_mode = = MOUSE_MODE_CAPTURED ) {
2022-03-27 11:17:36 +02:00
last_mouse_pos = p_position ;
2020-03-01 23:14:37 +01:00
} else {
2022-09-07 19:31:57 +02:00
WindowID window_id = _get_focused_window_or_popup ( ) ;
if ( ! windows . has ( window_id ) ) {
window_id = MAIN_WINDOW_ID ;
}
2022-02-01 03:49:51 +01:00
XWarpPointer ( x11_display , None , windows [ window_id ] . x11_window ,
2022-03-27 11:17:36 +02:00
0 , 0 , 0 , 0 , ( int ) p_position . x , ( int ) p_position . y ) ;
2020-03-01 23:14:37 +01:00
}
}
2020-03-07 17:02:54 +01:00
2020-03-01 23:14:37 +01:00
Point2i DisplayServerX11 : : mouse_get_position ( ) const {
2020-03-07 17:02:54 +01:00
int number_of_screens = XScreenCount ( x11_display ) ;
for ( int i = 0 ; i < number_of_screens ; i + + ) {
Window root , child ;
int root_x , root_y , win_x , win_y ;
unsigned int mask ;
if ( XQueryPointer ( x11_display , XRootWindow ( x11_display , i ) , & root , & child , & root_x , & root_y , & win_x , & win_y , & mask ) ) {
XWindowAttributes root_attrs ;
XGetWindowAttributes ( x11_display , root , & root_attrs ) ;
return Vector2i ( root_attrs . x + root_x , root_attrs . y + root_y ) ;
}
}
return Vector2i ( ) ;
}
2023-01-08 00:55:54 +01:00
BitField < MouseButtonMask > DisplayServerX11 : : mouse_get_button_state ( ) const {
2024-05-27 09:52:59 +02:00
int number_of_screens = XScreenCount ( x11_display ) ;
for ( int i = 0 ; i < number_of_screens ; i + + ) {
Window root , child ;
int root_x , root_y , win_x , win_y ;
unsigned int mask ;
if ( XQueryPointer ( x11_display , XRootWindow ( x11_display , i ) , & root , & child , & root_x , & root_y , & win_x , & win_y , & mask ) ) {
BitField < MouseButtonMask > last_button_state = 0 ;
if ( mask & Button1Mask ) {
last_button_state . set_flag ( MouseButtonMask : : LEFT ) ;
}
if ( mask & Button2Mask ) {
last_button_state . set_flag ( MouseButtonMask : : MIDDLE ) ;
}
if ( mask & Button3Mask ) {
last_button_state . set_flag ( MouseButtonMask : : RIGHT ) ;
}
if ( mask & Button4Mask ) {
last_button_state . set_flag ( MouseButtonMask : : MB_XBUTTON1 ) ;
}
if ( mask & Button5Mask ) {
last_button_state . set_flag ( MouseButtonMask : : MB_XBUTTON2 ) ;
}
return last_button_state ;
}
}
return 0 ;
2020-03-01 23:14:37 +01:00
}
void DisplayServerX11 : : clipboard_set ( const String & p_text ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-09-25 16:40:04 +02:00
{
// The clipboard content can be accessed while polling for events.
MutexLock mutex_lock ( events_mutex ) ;
internal_clipboard = p_text ;
}
2020-03-01 23:14:37 +01:00
XSetSelectionOwner ( x11_display , XA_PRIMARY , windows [ MAIN_WINDOW_ID ] . x11_window , CurrentTime ) ;
XSetSelectionOwner ( x11_display , XInternAtom ( x11_display , " CLIPBOARD " , 0 ) , windows [ MAIN_WINDOW_ID ] . x11_window , CurrentTime ) ;
}
2021-10-12 09:43:50 +02:00
void DisplayServerX11 : : clipboard_set_primary ( const String & p_text ) {
_THREAD_SAFE_METHOD_
if ( ! p_text . is_empty ( ) ) {
{
// The clipboard content can be accessed while polling for events.
MutexLock mutex_lock ( events_mutex ) ;
internal_clipboard_primary = p_text ;
}
XSetSelectionOwner ( x11_display , XA_PRIMARY , windows [ MAIN_WINDOW_ID ] . x11_window , CurrentTime ) ;
XSetSelectionOwner ( x11_display , XInternAtom ( x11_display , " PRIMARY " , 0 ) , windows [ MAIN_WINDOW_ID ] . x11_window , CurrentTime ) ;
}
}
2020-09-09 12:43:20 +02:00
Bool DisplayServerX11 : : _predicate_clipboard_selection ( Display * display , XEvent * event , XPointer arg ) {
if ( event - > type = = SelectionNotify & & event - > xselection . requestor = = * ( Window * ) arg ) {
return True ;
} else {
return False ;
}
}
2020-10-09 16:54:36 +02:00
Bool DisplayServerX11 : : _predicate_clipboard_incr ( Display * display , XEvent * event , XPointer arg ) {
2023-09-15 02:58:13 +02:00
if ( event - > type = = PropertyNotify & & event - > xproperty . state = = PropertyNewValue & & event - > xproperty . atom = = * ( Atom * ) arg ) {
2020-10-09 16:54:36 +02:00
return True ;
} else {
return False ;
}
}
2020-09-09 12:43:20 +02:00
String DisplayServerX11 : : _clipboard_get_impl ( Atom p_source , Window x11_window , Atom target ) const {
2020-03-01 23:14:37 +01:00
String ret ;
2020-09-25 16:40:04 +02:00
Window selection_owner = XGetSelectionOwner ( x11_display , p_source ) ;
if ( selection_owner = = x11_window ) {
2021-10-12 09:43:50 +02:00
static const char * target_type = " PRIMARY " ;
2022-03-04 10:50:24 +01:00
if ( p_source ! = None & & get_atom_name ( x11_display , p_source ) = = target_type ) {
2021-10-12 09:43:50 +02:00
return internal_clipboard_primary ;
} else {
return internal_clipboard ;
}
2020-09-25 16:40:04 +02:00
}
2020-03-01 23:14:37 +01:00
2020-09-25 16:40:04 +02:00
if ( selection_owner ! = None ) {
2020-10-09 16:54:36 +02:00
// Block events polling while processing selection events.
MutexLock mutex_lock ( events_mutex ) ;
2020-09-09 12:43:20 +02:00
2020-10-09 16:54:36 +02:00
Atom selection = XA_PRIMARY ;
XConvertSelection ( x11_display , p_source , target , selection ,
x11_window , CurrentTime ) ;
2020-09-09 12:43:20 +02:00
2020-10-09 16:54:36 +02:00
XFlush ( x11_display ) ;
2020-09-09 12:43:20 +02:00
2020-10-09 16:54:36 +02:00
// Blocking wait for predicate to be True and remove the event from the queue.
XEvent event ;
XIfEvent ( x11_display , & event , _predicate_clipboard_selection , ( XPointer ) & x11_window ) ;
2020-03-01 23:14:37 +01:00
2020-10-09 16:54:36 +02:00
// Do not get any data, see how much data is there.
Atom type ;
int format , result ;
unsigned long len , bytes_left , dummy ;
unsigned char * data ;
2020-03-01 23:14:37 +01:00
XGetWindowProperty ( x11_display , x11_window ,
selection , // Tricky..
0 , 0 , // offset - len
0 , // Delete 0==FALSE
2020-10-09 16:54:36 +02:00
AnyPropertyType , // flag
2020-03-01 23:14:37 +01:00
& type , // return type
& format , // return format
2020-10-09 16:54:36 +02:00
& len , & bytes_left , // data length
2020-03-01 23:14:37 +01:00
& data ) ;
2020-10-09 16:54:36 +02:00
if ( data ) {
XFree ( data ) ;
}
if ( type = = XInternAtom ( x11_display , " INCR " , 0 ) ) {
// Data is going to be received incrementally.
DEBUG_LOG_X11 ( " INCR selection started. \n " ) ;
LocalVector < uint8_t > incr_data ;
uint32_t data_size = 0 ;
bool success = false ;
// Delete INCR property to notify the owner.
XDeleteProperty ( x11_display , x11_window , type ) ;
// Process events from the queue.
bool done = false ;
while ( ! done ) {
if ( ! _wait_for_events ( ) ) {
// Error or timeout, abort.
break ;
}
// Non-blocking wait for next event and remove it from the queue.
XEvent ev ;
2023-09-15 02:58:13 +02:00
while ( XCheckIfEvent ( x11_display , & ev , _predicate_clipboard_incr , ( XPointer ) & selection ) ) {
2020-10-09 16:54:36 +02:00
result = XGetWindowProperty ( x11_display , x11_window ,
selection , // selection type
0 , LONG_MAX , // offset - len
True , // delete property to notify the owner
AnyPropertyType , // flag
& type , // return type
& format , // return format
& len , & bytes_left , // data length
& data ) ;
DEBUG_LOG_X11 ( " PropertyNotify: len=%lu, format=%i \n " , len , format ) ;
if ( result = = Success ) {
if ( data & & ( len > 0 ) ) {
uint32_t prev_size = incr_data . size ( ) ;
if ( prev_size = = 0 ) {
// First property contains initial data size.
unsigned long initial_size = * ( unsigned long * ) data ;
incr_data . resize ( initial_size ) ;
} else {
// New chunk, resize to be safe and append data.
incr_data . resize ( MAX ( data_size + len , prev_size ) ) ;
memcpy ( incr_data . ptr ( ) + data_size , data , len ) ;
data_size + = len ;
}
} else {
// Last chunk, process finished.
done = true ;
success = true ;
}
} else {
2023-03-15 09:40:36 +01:00
print_verbose ( " Failed to get selection data chunk. " ) ;
2020-10-09 16:54:36 +02:00
done = true ;
}
if ( data ) {
XFree ( data ) ;
}
if ( done ) {
break ;
}
}
}
if ( success & & ( data_size > 0 ) ) {
ret . parse_utf8 ( ( const char * ) incr_data . ptr ( ) , data_size ) ;
}
} else if ( bytes_left > 0 ) {
// Data is ready and can be processed all at once.
2020-03-01 23:14:37 +01:00
result = XGetWindowProperty ( x11_display , x11_window ,
selection , 0 , bytes_left , 0 ,
AnyPropertyType , & type , & format ,
& len , & dummy , & data ) ;
2020-10-09 16:54:36 +02:00
2020-03-01 23:14:37 +01:00
if ( result = = Success ) {
ret . parse_utf8 ( ( const char * ) data ) ;
2020-05-14 16:41:43 +02:00
} else {
2023-03-15 09:40:36 +01:00
print_verbose ( " Failed to get selection data. " ) ;
2020-05-14 16:41:43 +02:00
}
2020-10-09 16:54:36 +02:00
2020-03-06 18:00:16 +01:00
if ( data ) {
XFree ( data ) ;
}
2020-03-01 23:14:37 +01:00
}
}
return ret ;
}
2023-09-15 02:58:13 +02:00
Atom DisplayServerX11 : : _clipboard_get_image_target ( Atom p_source , Window x11_window ) const {
Atom target = XInternAtom ( x11_display , " TARGETS " , 0 ) ;
Atom png = XInternAtom ( x11_display , " image/png " , 0 ) ;
Atom * valid_targets = nullptr ;
unsigned long atom_count = 0 ;
Window selection_owner = XGetSelectionOwner ( x11_display , p_source ) ;
2023-10-26 01:32:40 +02:00
if ( selection_owner ! = None & & selection_owner ! = x11_window ) {
2023-09-15 02:58:13 +02:00
// Block events polling while processing selection events.
MutexLock mutex_lock ( events_mutex ) ;
Atom selection = XA_PRIMARY ;
XConvertSelection ( x11_display , p_source , target , selection , x11_window , CurrentTime ) ;
XFlush ( x11_display ) ;
// Blocking wait for predicate to be True and remove the event from the queue.
XEvent event ;
XIfEvent ( x11_display , & event , _predicate_clipboard_selection , ( XPointer ) & x11_window ) ;
// Do not get any data, see how much data is there.
Atom type ;
int format , result ;
unsigned long len , bytes_left , dummy ;
XGetWindowProperty ( x11_display , x11_window ,
selection , // Tricky..
0 , 0 , // offset - len
0 , // Delete 0==FALSE
XA_ATOM , // flag
& type , // return type
& format , // return format
& len , & bytes_left , // data length
( unsigned char * * ) & valid_targets ) ;
if ( valid_targets ) {
XFree ( valid_targets ) ;
valid_targets = nullptr ;
}
if ( type = = XA_ATOM & & bytes_left > 0 ) {
// Data is ready and can be processed all at once.
result = XGetWindowProperty ( x11_display , x11_window ,
selection , 0 , bytes_left / 4 , 0 ,
XA_ATOM , & type , & format ,
& len , & dummy , ( unsigned char * * ) & valid_targets ) ;
if ( result = = Success ) {
atom_count = len ;
} else {
print_verbose ( " Failed to get selection data. " ) ;
return None ;
}
} else {
return None ;
}
} else {
return None ;
}
for ( unsigned long i = 0 ; i < atom_count ; i + + ) {
Atom atom = valid_targets [ i ] ;
if ( atom = = png ) {
XFree ( valid_targets ) ;
return png ;
}
}
XFree ( valid_targets ) ;
return None ;
}
2020-09-09 12:43:20 +02:00
String DisplayServerX11 : : _clipboard_get ( Atom p_source , Window x11_window ) const {
2020-03-01 23:14:37 +01:00
String ret ;
Atom utf8_atom = XInternAtom ( x11_display , " UTF8_STRING " , True ) ;
if ( utf8_atom ! = None ) {
2020-09-09 12:43:20 +02:00
ret = _clipboard_get_impl ( p_source , x11_window , utf8_atom ) ;
2020-03-01 23:14:37 +01:00
}
2020-12-15 13:04:21 +01:00
if ( ret . is_empty ( ) ) {
2020-09-09 12:43:20 +02:00
ret = _clipboard_get_impl ( p_source , x11_window , XA_STRING ) ;
2020-03-01 23:14:37 +01:00
}
return ret ;
}
String DisplayServerX11 : : clipboard_get ( ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-01 23:14:37 +01:00
String ret ;
2020-09-09 12:43:20 +02:00
ret = _clipboard_get ( XInternAtom ( x11_display , " CLIPBOARD " , 0 ) , windows [ MAIN_WINDOW_ID ] . x11_window ) ;
2020-03-01 23:14:37 +01:00
2020-12-15 13:04:21 +01:00
if ( ret . is_empty ( ) ) {
2020-09-09 12:43:20 +02:00
ret = _clipboard_get ( XA_PRIMARY , windows [ MAIN_WINDOW_ID ] . x11_window ) ;
}
2020-03-01 23:14:37 +01:00
return ret ;
}
2021-10-12 09:43:50 +02:00
String DisplayServerX11 : : clipboard_get_primary ( ) const {
_THREAD_SAFE_METHOD_
String ret ;
ret = _clipboard_get ( XInternAtom ( x11_display , " PRIMARY " , 0 ) , windows [ MAIN_WINDOW_ID ] . x11_window ) ;
if ( ret . is_empty ( ) ) {
ret = _clipboard_get ( XA_PRIMARY , windows [ MAIN_WINDOW_ID ] . x11_window ) ;
}
return ret ;
}
2023-09-15 02:58:13 +02:00
Ref < Image > DisplayServerX11 : : clipboard_get_image ( ) const {
_THREAD_SAFE_METHOD_
Atom clipboard = XInternAtom ( x11_display , " CLIPBOARD " , 0 ) ;
Window x11_window = windows [ MAIN_WINDOW_ID ] . x11_window ;
Ref < Image > ret ;
Atom target = _clipboard_get_image_target ( clipboard , x11_window ) ;
if ( target = = None ) {
return ret ;
}
Window selection_owner = XGetSelectionOwner ( x11_display , clipboard ) ;
2023-10-26 01:32:40 +02:00
if ( selection_owner ! = None & & selection_owner ! = x11_window ) {
2023-09-15 02:58:13 +02:00
// Block events polling while processing selection events.
MutexLock mutex_lock ( events_mutex ) ;
// Identifier for the property the other window
// will send the converted data to.
Atom transfer_prop = XA_PRIMARY ;
XConvertSelection ( x11_display ,
clipboard , // source selection
target , // format to convert to
transfer_prop , // output property
x11_window , CurrentTime ) ;
XFlush ( x11_display ) ;
// Blocking wait for predicate to be True and remove the event from the queue.
XEvent event ;
XIfEvent ( x11_display , & event , _predicate_clipboard_selection , ( XPointer ) & x11_window ) ;
// Do not get any data, see how much data is there.
Atom type ;
int format , result ;
unsigned long len , bytes_left , dummy ;
unsigned char * data ;
XGetWindowProperty ( x11_display , x11_window ,
transfer_prop , // Property data is transferred through
0 , 1 , // offset, len (4 so we can get the size if INCR is used)
0 , // Delete 0==FALSE
AnyPropertyType , // flag
& type , // return type
& format , // return format
& len , & bytes_left , // data length
& data ) ;
if ( type = = XInternAtom ( x11_display , " INCR " , 0 ) ) {
ERR_FAIL_COND_V_MSG ( len ! = 1 , ret , " Incremental transfer initial value was not length. " ) ;
// Data is going to be received incrementally.
DEBUG_LOG_X11 ( " INCR selection started. \n " ) ;
LocalVector < uint8_t > incr_data ;
uint32_t data_size = 0 ;
bool success = false ;
// Initial response is the lower bound of the length of the transferred data.
incr_data . resize ( * ( unsigned long * ) data ) ;
XFree ( data ) ;
data = nullptr ;
// Delete INCR property to notify the owner.
XDeleteProperty ( x11_display , x11_window , transfer_prop ) ;
// Process events from the queue.
bool done = false ;
while ( ! done ) {
if ( ! _wait_for_events ( ) ) {
// Error or timeout, abort.
break ;
}
// Non-blocking wait for next event and remove it from the queue.
XEvent ev ;
while ( XCheckIfEvent ( x11_display , & ev , _predicate_clipboard_incr , ( XPointer ) & transfer_prop ) ) {
result = XGetWindowProperty ( x11_display , x11_window ,
transfer_prop , // output property
0 , LONG_MAX , // offset - len
True , // delete property to notify the owner
AnyPropertyType , // flag
& type , // return type
& format , // return format
& len , & bytes_left , // data length
& data ) ;
DEBUG_LOG_X11 ( " PropertyNotify: len=%lu, format=%i \n " , len , format ) ;
if ( result = = Success ) {
if ( data & & ( len > 0 ) ) {
uint32_t prev_size = incr_data . size ( ) ;
// New chunk, resize to be safe and append data.
incr_data . resize ( MAX ( data_size + len , prev_size ) ) ;
memcpy ( incr_data . ptr ( ) + data_size , data , len ) ;
data_size + = len ;
} else if ( ! ( format = = 0 & & len = = 0 ) ) {
// For unclear reasons the first GetWindowProperty always returns a length and format of 0.
// Otherwise, last chunk, process finished.
done = true ;
success = true ;
}
} else {
print_verbose ( " Failed to get selection data chunk. " ) ;
done = true ;
}
if ( data ) {
XFree ( data ) ;
data = nullptr ;
}
if ( done ) {
break ;
}
}
}
if ( success & & ( data_size > 0 ) ) {
ret . instantiate ( ) ;
PNGDriverCommon : : png_to_image ( incr_data . ptr ( ) , incr_data . size ( ) , false , ret ) ;
}
} else if ( bytes_left > 0 ) {
if ( data ) {
XFree ( data ) ;
data = nullptr ;
}
// Data is ready and can be processed all at once.
result = XGetWindowProperty ( x11_display , x11_window ,
transfer_prop , 0 , bytes_left + 4 , 0 ,
AnyPropertyType , & type , & format ,
& len , & dummy , & data ) ;
if ( result = = Success ) {
ret . instantiate ( ) ;
PNGDriverCommon : : png_to_image ( ( uint8_t * ) data , bytes_left , false , ret ) ;
} else {
print_verbose ( " Failed to get selection data. " ) ;
}
if ( data ) {
XFree ( data ) ;
}
}
}
return ret ;
}
bool DisplayServerX11 : : clipboard_has_image ( ) const {
Atom target = _clipboard_get_image_target (
XInternAtom ( x11_display , " CLIPBOARD " , 0 ) ,
windows [ MAIN_WINDOW_ID ] . x11_window ) ;
return target ! = None ;
}
2020-10-08 17:57:54 +02:00
Bool DisplayServerX11 : : _predicate_clipboard_save_targets ( Display * display , XEvent * event , XPointer arg ) {
if ( event - > xany . window = = * ( Window * ) arg ) {
return ( event - > type = = SelectionRequest ) | |
2021-10-28 15:19:35 +02:00
( event - > type = = SelectionNotify ) ;
2020-10-08 17:57:54 +02:00
} else {
return False ;
}
}
void DisplayServerX11 : : _clipboard_transfer_ownership ( Atom p_source , Window x11_window ) const {
_THREAD_SAFE_METHOD_
Window selection_owner = XGetSelectionOwner ( x11_display , p_source ) ;
if ( selection_owner ! = x11_window ) {
return ;
}
// Block events polling while processing selection events.
MutexLock mutex_lock ( events_mutex ) ;
Atom clipboard_manager = XInternAtom ( x11_display , " CLIPBOARD_MANAGER " , False ) ;
Atom save_targets = XInternAtom ( x11_display , " SAVE_TARGETS " , False ) ;
XConvertSelection ( x11_display , clipboard_manager , save_targets , None ,
x11_window , CurrentTime ) ;
// Process events from the queue.
while ( true ) {
if ( ! _wait_for_events ( ) ) {
// Error or timeout, abort.
break ;
}
// Non-blocking wait for next event and remove it from the queue.
XEvent ev ;
while ( XCheckIfEvent ( x11_display , & ev , _predicate_clipboard_save_targets , ( XPointer ) & x11_window ) ) {
switch ( ev . type ) {
case SelectionRequest :
_handle_selection_request_event ( & ( ev . xselectionrequest ) ) ;
break ;
case SelectionNotify : {
if ( ev . xselection . target = = save_targets ) {
// Once SelectionNotify is received, we're done whether it succeeded or not.
return ;
}
break ;
}
}
}
}
}
2020-03-01 23:14:37 +01:00
int DisplayServerX11 : : get_screen_count ( ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2022-02-18 12:29:10 +01:00
int count = 0 ;
2020-03-07 16:46:50 +01:00
2020-03-01 23:14:37 +01:00
// Using Xinerama Extension
int event_base , error_base ;
2023-08-04 09:58:14 +02:00
if ( xinerama_ext_ok & & XineramaQueryExtension ( x11_display , & event_base , & error_base ) ) {
2022-02-18 12:29:10 +01:00
XineramaScreenInfo * xsi = XineramaQueryScreens ( x11_display , & count ) ;
XFree ( xsi ) ;
2024-05-14 09:32:24 +02:00
}
if ( count = = 0 ) {
2022-02-18 12:29:10 +01:00
count = XScreenCount ( x11_display ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
return count ;
}
2020-05-14 14:29:06 +02:00
2023-01-04 23:00:02 +01:00
int DisplayServerX11 : : get_primary_screen ( ) const {
2023-03-21 12:08:46 +01:00
int event_base , error_base ;
2023-08-04 09:58:14 +02:00
if ( xinerama_ext_ok & & XineramaQueryExtension ( x11_display , & event_base , & error_base ) ) {
2023-03-21 12:08:46 +01:00
return 0 ;
} else {
return XDefaultScreen ( x11_display ) ;
}
2023-01-04 23:00:02 +01:00
}
2023-03-21 12:08:46 +01:00
int DisplayServerX11 : : get_keyboard_focus_screen ( ) const {
int count = get_screen_count ( ) ;
if ( count < 2 ) {
// Early exit with single monitor.
return 0 ;
}
2020-03-07 16:46:50 +01:00
2023-03-21 12:08:46 +01:00
Window focus = 0 ;
int revert_to = 0 ;
XGetInputFocus ( x11_display , & focus , & revert_to ) ;
if ( focus ) {
Window focus_child = 0 ;
int x = 0 , y = 0 ;
XTranslateCoordinates ( x11_display , focus , DefaultRootWindow ( x11_display ) , 0 , 0 , & x , & y , & focus_child ) ;
XWindowAttributes xwa ;
XGetWindowAttributes ( x11_display , focus , & xwa ) ;
Rect2i window_rect = Rect2i ( x , y , xwa . width , xwa . height ) ;
// Find which monitor has the largest overlap with the given window.
int screen_index = 0 ;
int max_area = 0 ;
for ( int i = 0 ; i < count ; i + + ) {
Rect2i screen_rect = _screen_get_rect ( i ) ;
Rect2i intersection = screen_rect . intersection ( window_rect ) ;
int area = intersection . get_area ( ) ;
if ( area > max_area ) {
max_area = area ;
screen_index = i ;
}
}
return screen_index ;
2020-03-01 23:14:37 +01:00
}
2023-03-21 12:08:46 +01:00
return get_primary_screen ( ) ;
}
Rect2i DisplayServerX11 : : _screen_get_rect ( int p_screen ) const {
Rect2i rect ( 0 , 0 , 0 , 0 ) ;
p_screen = _get_screen_index ( p_screen ) ;
2020-09-16 16:33:33 +02:00
ERR_FAIL_COND_V ( p_screen < 0 , rect ) ;
2020-03-01 23:14:37 +01:00
2020-09-16 16:33:33 +02:00
// Using Xinerama Extension.
2024-05-14 09:32:24 +02:00
bool found = false ;
2020-09-16 16:33:33 +02:00
int event_base , error_base ;
2023-08-04 09:58:14 +02:00
if ( xinerama_ext_ok & & XineramaQueryExtension ( x11_display , & event_base , & error_base ) ) {
2020-09-16 16:33:33 +02:00
int count ;
XineramaScreenInfo * xsi = XineramaQueryScreens ( x11_display , & count ) ;
if ( xsi ) {
2024-05-14 09:32:24 +02:00
if ( count > 0 ) {
// Check if screen is valid.
if ( p_screen < count ) {
rect . position . x = xsi [ p_screen ] . x_org ;
rect . position . y = xsi [ p_screen ] . y_org ;
rect . size . width = xsi [ p_screen ] . width ;
rect . size . height = xsi [ p_screen ] . height ;
found = true ;
} else {
ERR_PRINT ( vformat ( " Invalid screen index: %d (count: %d). " , p_screen , count ) ) ;
}
}
2020-09-16 16:33:33 +02:00
XFree ( xsi ) ;
}
2024-05-14 09:32:24 +02:00
}
if ( ! found ) {
2022-02-18 12:29:10 +01:00
int count = XScreenCount ( x11_display ) ;
if ( p_screen < count ) {
Window root = XRootWindow ( x11_display , p_screen ) ;
XWindowAttributes xwa ;
XGetWindowAttributes ( x11_display , root , & xwa ) ;
rect . position . x = xwa . x ;
rect . position . y = xwa . y ;
rect . size . width = xwa . width ;
rect . size . height = xwa . height ;
} else {
2024-05-14 09:32:24 +02:00
ERR_PRINT ( vformat ( " Invalid screen index: %d (count: %d). " , p_screen , count ) ) ;
2022-02-18 12:29:10 +01:00
}
2020-09-16 16:33:33 +02:00
}
2020-03-01 23:14:37 +01:00
2020-09-16 16:33:33 +02:00
return rect ;
}
2020-03-01 23:14:37 +01:00
2020-09-16 16:33:33 +02:00
Point2i DisplayServerX11 : : screen_get_position ( int p_screen ) const {
_THREAD_SAFE_METHOD_
2020-03-01 23:14:37 +01:00
2020-09-16 16:33:33 +02:00
return _screen_get_rect ( p_screen ) . position ;
2020-03-01 23:14:37 +01:00
}
2020-03-20 03:32:09 +01:00
2020-03-01 23:14:37 +01:00
Size2i DisplayServerX11 : : screen_get_size ( int p_screen ) const {
2020-09-16 16:33:33 +02:00
_THREAD_SAFE_METHOD_
return _screen_get_rect ( p_screen ) . size ;
2020-03-20 03:32:09 +01:00
}
2020-03-01 23:14:37 +01:00
2023-03-19 13:54:24 +01:00
// A Handler to avoid crashing on non-fatal X errors by default.
//
// The original X11 error formatter `_XPrintDefaultError` is defined here:
// https://gitlab.freedesktop.org/xorg/lib/libx11/-/blob/e45ca7b41dcd3ace7681d6897505f85d374640f2/src/XlibInt.c#L1322
// It is not exposed through the API, accesses X11 internals,
// and is much more complex, so this is a less complete simplified error X11 printer.
int default_window_error_handler ( Display * display , XErrorEvent * error ) {
static char message [ 1024 ] ;
XGetErrorText ( display , error - > error_code , message , sizeof ( message ) ) ;
ERR_PRINT ( vformat ( " Unhandled XServer error: %s "
" \n Major opcode of failed request: %d "
" \n Serial number of failed request: %d "
" \n Current serial number in output stream: %d " ,
2023-06-07 14:57:51 +02:00
String : : utf8 ( message ) , ( uint64_t ) error - > request_code , ( uint64_t ) error - > minor_code , ( uint64_t ) error - > serial ) ) ;
2023-03-19 13:54:24 +01:00
return 0 ;
}
2021-11-04 17:47:18 +01:00
bool g_bad_window = false ;
int bad_window_error_handler ( Display * display , XErrorEvent * error ) {
if ( error - > error_code = = BadWindow ) {
g_bad_window = true ;
} else {
2023-03-19 13:54:24 +01:00
return default_window_error_handler ( display , error ) ;
2021-11-04 17:47:18 +01:00
}
return 0 ;
}
2020-03-20 03:32:09 +01:00
Rect2i DisplayServerX11 : : screen_get_usable_rect ( int p_screen ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2023-03-21 12:08:46 +01:00
p_screen = _get_screen_index ( p_screen ) ;
2020-08-27 17:22:24 +02:00
int screen_count = get_screen_count ( ) ;
// Check if screen is valid.
ERR_FAIL_INDEX_V ( p_screen , screen_count , Rect2i ( 0 , 0 , 0 , 0 ) ) ;
bool is_multiscreen = screen_count > 1 ;
// Use full monitor size as fallback.
Rect2i rect = _screen_get_rect ( p_screen ) ;
// There's generally only one screen reported by xlib even in multi-screen setup,
// in this case it's just one virtual screen composed of all physical monitors.
int x11_screen_count = ScreenCount ( x11_display ) ;
Window x11_window = RootWindow ( x11_display , p_screen < x11_screen_count ? p_screen : 0 ) ;
Atom type ;
int format = 0 ;
unsigned long remaining = 0 ;
// Find active desktop for the root window.
unsigned int desktop_index = 0 ;
Atom desktop_prop = XInternAtom ( x11_display , " _NET_CURRENT_DESKTOP " , True ) ;
if ( desktop_prop ! = None ) {
unsigned long desktop_len = 0 ;
unsigned char * desktop_data = nullptr ;
if ( XGetWindowProperty ( x11_display , x11_window , desktop_prop , 0 , LONG_MAX , False , XA_CARDINAL , & type , & format , & desktop_len , & remaining , & desktop_data ) = = Success ) {
if ( ( format = = 32 ) & & ( desktop_len > 0 ) & & desktop_data ) {
desktop_index = ( unsigned int ) desktop_data [ 0 ] ;
}
if ( desktop_data ) {
XFree ( desktop_data ) ;
}
}
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
2020-08-27 17:22:24 +02:00
bool use_simple_method = true ;
// First check for GTK work area, which is more accurate for multi-screen setup.
if ( is_multiscreen ) {
// Use already calculated work area when available.
Atom gtk_workareas_prop = XInternAtom ( x11_display , " _GTK_WORKAREAS " , False ) ;
if ( gtk_workareas_prop ! = None ) {
char gtk_workarea_prop_name [ 32 ] ;
snprintf ( gtk_workarea_prop_name , 32 , " _GTK_WORKAREAS_D%d " , desktop_index ) ;
Atom gtk_workarea_prop = XInternAtom ( x11_display , gtk_workarea_prop_name , True ) ;
if ( gtk_workarea_prop ! = None ) {
unsigned long workarea_len = 0 ;
unsigned char * workarea_data = nullptr ;
if ( XGetWindowProperty ( x11_display , x11_window , gtk_workarea_prop , 0 , LONG_MAX , False , XA_CARDINAL , & type , & format , & workarea_len , & remaining , & workarea_data ) = = Success ) {
if ( ( format = = 32 ) & & ( workarea_len % 4 = = 0 ) & & workarea_data ) {
long * rect_data = ( long * ) workarea_data ;
for ( uint32_t data_offset = 0 ; data_offset < workarea_len ; data_offset + = 4 ) {
Rect2i workarea_rect ;
workarea_rect . position . x = rect_data [ data_offset ] ;
workarea_rect . position . y = rect_data [ data_offset + 1 ] ;
workarea_rect . size . x = rect_data [ data_offset + 2 ] ;
workarea_rect . size . y = rect_data [ data_offset + 3 ] ;
// Intersect with actual monitor size to find the correct area,
// because areas are not in the same order as screens from Xinerama.
if ( rect . grow ( - 1 ) . intersects ( workarea_rect ) ) {
rect = rect . intersection ( workarea_rect ) ;
XFree ( workarea_data ) ;
return rect ;
}
}
}
}
if ( workarea_data ) {
XFree ( workarea_data ) ;
}
}
}
2021-02-28 17:43:46 +01:00
2020-08-27 17:22:24 +02:00
// Fallback to calculating work area by hand from struts.
Atom client_list_prop = XInternAtom ( x11_display , " _NET_CLIENT_LIST " , True ) ;
if ( client_list_prop ! = None ) {
unsigned long clients_len = 0 ;
unsigned char * clients_data = nullptr ;
if ( XGetWindowProperty ( x11_display , x11_window , client_list_prop , 0 , LONG_MAX , False , XA_WINDOW , & type , & format , & clients_len , & remaining , & clients_data ) = = Success ) {
if ( ( format = = 32 ) & & ( clients_len > 0 ) & & clients_data ) {
Window * windows_data = ( Window * ) clients_data ;
Rect2i desktop_rect ;
bool desktop_valid = false ;
// Get full desktop size.
{
Atom desktop_geometry_prop = XInternAtom ( x11_display , " _NET_DESKTOP_GEOMETRY " , True ) ;
if ( desktop_geometry_prop ! = None ) {
unsigned long geom_len = 0 ;
unsigned char * geom_data = nullptr ;
if ( XGetWindowProperty ( x11_display , x11_window , desktop_geometry_prop , 0 , LONG_MAX , False , XA_CARDINAL , & type , & format , & geom_len , & remaining , & geom_data ) = = Success ) {
if ( ( format = = 32 ) & & ( geom_len > = 2 ) & & geom_data ) {
desktop_valid = true ;
long * size_data = ( long * ) geom_data ;
desktop_rect . size . x = size_data [ 0 ] ;
desktop_rect . size . y = size_data [ 1 ] ;
}
}
if ( geom_data ) {
XFree ( geom_data ) ;
}
}
}
// Get full desktop position.
if ( desktop_valid ) {
Atom desktop_viewport_prop = XInternAtom ( x11_display , " _NET_DESKTOP_VIEWPORT " , True ) ;
if ( desktop_viewport_prop ! = None ) {
unsigned long viewport_len = 0 ;
unsigned char * viewport_data = nullptr ;
if ( XGetWindowProperty ( x11_display , x11_window , desktop_viewport_prop , 0 , LONG_MAX , False , XA_CARDINAL , & type , & format , & viewport_len , & remaining , & viewport_data ) = = Success ) {
if ( ( format = = 32 ) & & ( viewport_len > = 2 ) & & viewport_data ) {
desktop_valid = true ;
long * pos_data = ( long * ) viewport_data ;
desktop_rect . position . x = pos_data [ 0 ] ;
desktop_rect . position . y = pos_data [ 1 ] ;
}
}
if ( viewport_data ) {
XFree ( viewport_data ) ;
}
}
}
if ( desktop_valid ) {
use_simple_method = false ;
2021-11-04 17:47:18 +01:00
// Handle bad window errors silently because there's no other way to check
// that one of the windows has been destroyed in the meantime.
int ( * oldHandler ) ( Display * , XErrorEvent * ) = XSetErrorHandler ( & bad_window_error_handler ) ;
2020-08-27 17:22:24 +02:00
for ( unsigned long win_index = 0 ; win_index < clients_len ; + + win_index ) {
2021-11-04 17:47:18 +01:00
g_bad_window = false ;
2020-08-27 17:22:24 +02:00
// Remove strut size from desktop size to get a more accurate result.
bool strut_found = false ;
unsigned long strut_len = 0 ;
unsigned char * strut_data = nullptr ;
Atom strut_partial_prop = XInternAtom ( x11_display , " _NET_WM_STRUT_PARTIAL " , True ) ;
if ( strut_partial_prop ! = None ) {
if ( XGetWindowProperty ( x11_display , windows_data [ win_index ] , strut_partial_prop , 0 , LONG_MAX , False , XA_CARDINAL , & type , & format , & strut_len , & remaining , & strut_data ) = = Success ) {
strut_found = true ;
}
}
// Fallback to older strut property.
2021-11-04 17:47:18 +01:00
if ( ! g_bad_window & & ! strut_found ) {
2020-08-27 17:22:24 +02:00
Atom strut_prop = XInternAtom ( x11_display , " _NET_WM_STRUT " , True ) ;
if ( strut_prop ! = None ) {
if ( XGetWindowProperty ( x11_display , windows_data [ win_index ] , strut_prop , 0 , LONG_MAX , False , XA_CARDINAL , & type , & format , & strut_len , & remaining , & strut_data ) = = Success ) {
strut_found = true ;
}
}
}
2021-11-04 17:47:18 +01:00
if ( ! g_bad_window & & strut_found & & ( format = = 32 ) & & ( strut_len > = 4 ) & & strut_data ) {
2020-08-27 17:22:24 +02:00
long * struts = ( long * ) strut_data ;
long left = struts [ 0 ] ;
long right = struts [ 1 ] ;
long top = struts [ 2 ] ;
long bottom = struts [ 3 ] ;
long left_start_y , left_end_y , right_start_y , right_end_y ;
long top_start_x , top_end_x , bottom_start_x , bottom_end_x ;
if ( strut_len > = 12 ) {
left_start_y = struts [ 4 ] ;
left_end_y = struts [ 5 ] ;
right_start_y = struts [ 6 ] ;
right_end_y = struts [ 7 ] ;
top_start_x = struts [ 8 ] ;
top_end_x = struts [ 9 ] ;
bottom_start_x = struts [ 10 ] ;
bottom_end_x = struts [ 11 ] ;
} else {
left_start_y = 0 ;
left_end_y = desktop_rect . size . y ;
right_start_y = 0 ;
right_end_y = desktop_rect . size . y ;
top_start_x = 0 ;
top_end_x = desktop_rect . size . x ;
bottom_start_x = 0 ;
bottom_end_x = desktop_rect . size . x ;
}
const Point2i & pos = desktop_rect . position ;
const Size2i & size = desktop_rect . size ;
Rect2i left_rect ( pos . x , pos . y + left_start_y , left , left_end_y - left_start_y ) ;
if ( left_rect . size . x > 0 ) {
Rect2i intersection = rect . intersection ( left_rect ) ;
2022-08-15 04:51:45 +02:00
if ( intersection . has_area ( ) & & intersection . size . x < rect . size . x ) {
2020-08-27 17:22:24 +02:00
rect . position . x = left_rect . size . x ;
rect . size . x = rect . size . x - intersection . size . x ;
}
}
Rect2i right_rect ( pos . x + size . x - right , pos . y + right_start_y , right , right_end_y - right_start_y ) ;
if ( right_rect . size . x > 0 ) {
Rect2i intersection = rect . intersection ( right_rect ) ;
2022-08-15 04:51:45 +02:00
if ( intersection . has_area ( ) & & right_rect . size . x < rect . size . x ) {
2020-08-27 17:22:24 +02:00
rect . size . x = intersection . position . x - rect . position . x ;
}
}
Rect2i top_rect ( pos . x + top_start_x , pos . y , top_end_x - top_start_x , top ) ;
if ( top_rect . size . y > 0 ) {
Rect2i intersection = rect . intersection ( top_rect ) ;
2022-08-15 04:51:45 +02:00
if ( intersection . has_area ( ) & & intersection . size . y < rect . size . y ) {
2020-08-27 17:22:24 +02:00
rect . position . y = top_rect . size . y ;
rect . size . y = rect . size . y - intersection . size . y ;
}
}
Rect2i bottom_rect ( pos . x + bottom_start_x , pos . y + size . y - bottom , bottom_end_x - bottom_start_x , bottom ) ;
if ( bottom_rect . size . y > 0 ) {
Rect2i intersection = rect . intersection ( bottom_rect ) ;
2022-08-15 04:51:45 +02:00
if ( intersection . has_area ( ) & & right_rect . size . y < rect . size . y ) {
2020-08-27 17:22:24 +02:00
rect . size . y = intersection . position . y - rect . position . y ;
}
}
}
if ( strut_data ) {
XFree ( strut_data ) ;
}
}
2021-11-04 17:47:18 +01:00
// Restore default error handler.
XSetErrorHandler ( oldHandler ) ;
2020-08-27 17:22:24 +02:00
}
}
}
if ( clients_data ) {
XFree ( clients_data ) ;
}
}
}
// Single screen or fallback for multi screen.
if ( use_simple_method ) {
// Get desktop available size from the global work area.
Atom workarea_prop = XInternAtom ( x11_display , " _NET_WORKAREA " , True ) ;
if ( workarea_prop ! = None ) {
unsigned long workarea_len = 0 ;
unsigned char * workarea_data = nullptr ;
if ( XGetWindowProperty ( x11_display , x11_window , workarea_prop , 0 , LONG_MAX , False , XA_CARDINAL , & type , & format , & workarea_len , & remaining , & workarea_data ) = = Success ) {
if ( ( format = = 32 ) & & ( workarea_len > = ( ( desktop_index + 1 ) * 4 ) ) & & workarea_data ) {
long * rect_data = ( long * ) workarea_data ;
int data_offset = desktop_index * 4 ;
Rect2i workarea_rect ;
workarea_rect . position . x = rect_data [ data_offset ] ;
workarea_rect . position . y = rect_data [ data_offset + 1 ] ;
workarea_rect . size . x = rect_data [ data_offset + 2 ] ;
workarea_rect . size . y = rect_data [ data_offset + 3 ] ;
// Intersect with actual monitor size to get a proper approximation in multi-screen setup.
if ( ! is_multiscreen ) {
rect = workarea_rect ;
} else if ( rect . intersects ( workarea_rect ) ) {
rect = rect . intersection ( workarea_rect ) ;
}
}
}
if ( workarea_data ) {
XFree ( workarea_data ) ;
}
}
}
2020-03-01 23:14:37 +01:00
2020-03-20 03:32:09 +01:00
return rect ;
2020-03-01 23:14:37 +01:00
}
2020-03-20 03:32:09 +01:00
2020-03-01 23:14:37 +01:00
int DisplayServerX11 : : screen_get_dpi ( int p_screen ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2023-03-21 12:08:46 +01:00
p_screen = _get_screen_index ( p_screen ) ;
2020-03-01 23:14:37 +01:00
ERR_FAIL_INDEX_V ( p_screen , get_screen_count ( ) , 0 ) ;
//Get physical monitor Dimensions through XRandR and calculate dpi
Size2i sc = screen_get_size ( p_screen ) ;
if ( xrandr_ext_ok ) {
int count = 0 ;
if ( xrr_get_monitors ) {
xrr_monitor_info * monitors = xrr_get_monitors ( x11_display , windows [ MAIN_WINDOW_ID ] . x11_window , true , & count ) ;
if ( p_screen < count ) {
double xdpi = sc . width / ( double ) monitors [ p_screen ] . mwidth * 25.4 ;
double ydpi = sc . height / ( double ) monitors [ p_screen ] . mheight * 25.4 ;
xrr_free_monitors ( monitors ) ;
return ( xdpi + ydpi ) / 2 ;
}
xrr_free_monitors ( monitors ) ;
} else if ( p_screen = = 0 ) {
XRRScreenSize * sizes = XRRSizes ( x11_display , 0 , & count ) ;
if ( sizes ) {
double xdpi = sc . width / ( double ) sizes [ 0 ] . mwidth * 25.4 ;
double ydpi = sc . height / ( double ) sizes [ 0 ] . mheight * 25.4 ;
return ( xdpi + ydpi ) / 2 ;
}
}
}
int width_mm = DisplayWidthMM ( x11_display , p_screen ) ;
int height_mm = DisplayHeightMM ( x11_display , p_screen ) ;
double xdpi = ( width_mm ? sc . width / ( double ) width_mm * 25.4 : 0 ) ;
double ydpi = ( height_mm ? sc . height / ( double ) height_mm * 25.4 : 0 ) ;
2020-05-14 16:41:43 +02:00
if ( xdpi | | ydpi ) {
2020-03-01 23:14:37 +01:00
return ( xdpi + ydpi ) / ( xdpi & & ydpi ? 2 : 1 ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
//could not get dpi
return 96 ;
}
2020-05-14 14:29:06 +02:00
2024-06-12 09:05:17 +02:00
int get_image_errorhandler ( Display * dpy , XErrorEvent * ev ) {
return 0 ;
}
2023-02-27 19:36:13 +01:00
Color DisplayServerX11 : : screen_get_pixel ( const Point2i & p_position ) const {
Point2i pos = p_position ;
2024-06-12 09:05:17 +02:00
if ( xwayland ) {
return Color ( ) ;
}
int ( * old_handler ) ( Display * , XErrorEvent * ) = XSetErrorHandler ( & get_image_errorhandler ) ;
Color color ;
2023-02-27 19:36:13 +01:00
int number_of_screens = XScreenCount ( x11_display ) ;
for ( int i = 0 ; i < number_of_screens ; i + + ) {
Window root = XRootWindow ( x11_display , i ) ;
XWindowAttributes root_attrs ;
XGetWindowAttributes ( x11_display , root , & root_attrs ) ;
if ( ( pos . x > = root_attrs . x ) & & ( pos . x < = root_attrs . x + root_attrs . width ) & & ( pos . y > = root_attrs . y ) & & ( pos . y < = root_attrs . y + root_attrs . height ) ) {
XImage * image = XGetImage ( x11_display , root , pos . x , pos . y , 1 , 1 , AllPlanes , XYPixmap ) ;
if ( image ) {
XColor c ;
c . pixel = XGetPixel ( image , 0 , 0 ) ;
2024-07-17 13:25:28 +02:00
XDestroyImage ( image ) ;
2023-02-27 19:36:13 +01:00
XQueryColor ( x11_display , XDefaultColormap ( x11_display , i ) , & c ) ;
2024-06-12 09:05:17 +02:00
color = Color ( float ( c . red ) / 65535.0 , float ( c . green ) / 65535.0 , float ( c . blue ) / 65535.0 , 1.0 ) ;
break ;
2023-02-27 19:36:13 +01:00
}
}
}
2024-06-12 09:05:17 +02:00
XSetErrorHandler ( old_handler ) ;
return color ;
2023-02-27 19:36:13 +01:00
}
2023-03-16 22:26:09 +01:00
Ref < Image > DisplayServerX11 : : screen_get_image ( int p_screen ) const {
ERR_FAIL_INDEX_V ( p_screen , get_screen_count ( ) , Ref < Image > ( ) ) ;
switch ( p_screen ) {
case SCREEN_PRIMARY : {
p_screen = get_primary_screen ( ) ;
} break ;
case SCREEN_OF_MAIN_WINDOW : {
p_screen = window_get_current_screen ( MAIN_WINDOW_ID ) ;
} break ;
default :
break ;
}
ERR_FAIL_COND_V ( p_screen < 0 , Ref < Image > ( ) ) ;
2024-06-12 09:05:17 +02:00
if ( xwayland ) {
return Ref < Image > ( ) ;
}
int ( * old_handler ) ( Display * , XErrorEvent * ) = XSetErrorHandler ( & get_image_errorhandler ) ;
2023-03-16 22:26:09 +01:00
XImage * image = nullptr ;
2024-05-14 09:32:24 +02:00
bool found = false ;
2023-03-16 22:26:09 +01:00
int event_base , error_base ;
2023-08-04 09:58:14 +02:00
if ( xinerama_ext_ok & & XineramaQueryExtension ( x11_display , & event_base , & error_base ) ) {
2023-03-16 22:26:09 +01:00
int xin_count ;
XineramaScreenInfo * xsi = XineramaQueryScreens ( x11_display , & xin_count ) ;
2024-05-14 09:32:24 +02:00
if ( xsi ) {
if ( xin_count > 0 ) {
if ( p_screen < xin_count ) {
int x_count = XScreenCount ( x11_display ) ;
for ( int i = 0 ; i < x_count ; i + + ) {
Window root = XRootWindow ( x11_display , i ) ;
XWindowAttributes root_attrs ;
XGetWindowAttributes ( x11_display , root , & root_attrs ) ;
if ( ( xsi [ p_screen ] . x_org > = root_attrs . x ) & & ( xsi [ p_screen ] . x_org < = root_attrs . x + root_attrs . width ) & & ( xsi [ p_screen ] . y_org > = root_attrs . y ) & & ( xsi [ p_screen ] . y_org < = root_attrs . y + root_attrs . height ) ) {
found = true ;
image = XGetImage ( x11_display , root , xsi [ p_screen ] . x_org , xsi [ p_screen ] . y_org , xsi [ p_screen ] . width , xsi [ p_screen ] . height , AllPlanes , ZPixmap ) ;
break ;
}
}
} else {
ERR_PRINT ( vformat ( " Invalid screen index: %d (count: %d). " , p_screen , xin_count ) ) ;
2023-03-16 22:26:09 +01:00
}
}
2024-05-14 09:32:24 +02:00
XFree ( xsi ) ;
2023-03-16 22:26:09 +01:00
}
2024-05-14 09:32:24 +02:00
}
if ( ! found ) {
2023-03-16 22:26:09 +01:00
int x_count = XScreenCount ( x11_display ) ;
if ( p_screen < x_count ) {
Window root = XRootWindow ( x11_display , p_screen ) ;
XWindowAttributes root_attrs ;
XGetWindowAttributes ( x11_display , root , & root_attrs ) ;
image = XGetImage ( x11_display , root , root_attrs . x , root_attrs . y , root_attrs . width , root_attrs . height , AllPlanes , ZPixmap ) ;
} else {
2024-05-14 09:32:24 +02:00
ERR_PRINT ( vformat ( " Invalid screen index: %d (count: %d). " , p_screen , x_count ) ) ;
2023-03-16 22:26:09 +01:00
}
}
2024-06-12 09:05:17 +02:00
XSetErrorHandler ( old_handler ) ;
2023-03-16 22:26:09 +01:00
Ref < Image > img ;
if ( image ) {
int width = image - > width ;
int height = image - > height ;
Vector < uint8_t > img_data ;
img_data . resize ( height * width * 4 ) ;
uint8_t * sr = ( uint8_t * ) image - > data ;
uint8_t * wr = ( uint8_t * ) img_data . ptrw ( ) ;
if ( image - > bits_per_pixel = = 24 & & image - > red_mask = = 0xff0000 & & image - > green_mask = = 0x00ff00 & & image - > blue_mask = = 0x0000ff ) {
for ( int y = 0 ; y < height ; y + + ) {
for ( int x = 0 ; x < width ; x + + ) {
wr [ ( y * width + x ) * 4 + 0 ] = sr [ ( y * width + x ) * 3 + 2 ] ;
wr [ ( y * width + x ) * 4 + 1 ] = sr [ ( y * width + x ) * 3 + 1 ] ;
wr [ ( y * width + x ) * 4 + 2 ] = sr [ ( y * width + x ) * 3 + 0 ] ;
wr [ ( y * width + x ) * 4 + 3 ] = 255 ;
}
}
} else if ( image - > bits_per_pixel = = 24 & & image - > red_mask = = 0x0000ff & & image - > green_mask = = 0x00ff00 & & image - > blue_mask = = 0xff0000 ) {
for ( int y = 0 ; y < height ; y + + ) {
for ( int x = 0 ; x < width ; x + + ) {
wr [ ( y * width + x ) * 4 + 0 ] = sr [ ( y * width + x ) * 3 + 2 ] ;
wr [ ( y * width + x ) * 4 + 1 ] = sr [ ( y * width + x ) * 3 + 1 ] ;
wr [ ( y * width + x ) * 4 + 2 ] = sr [ ( y * width + x ) * 3 + 0 ] ;
wr [ ( y * width + x ) * 4 + 3 ] = 255 ;
}
}
} else if ( image - > bits_per_pixel = = 32 & & image - > red_mask = = 0xff0000 & & image - > green_mask = = 0x00ff00 & & image - > blue_mask = = 0x0000ff ) {
for ( int y = 0 ; y < height ; y + + ) {
for ( int x = 0 ; x < width ; x + + ) {
wr [ ( y * width + x ) * 4 + 0 ] = sr [ ( y * width + x ) * 4 + 2 ] ;
wr [ ( y * width + x ) * 4 + 1 ] = sr [ ( y * width + x ) * 4 + 1 ] ;
wr [ ( y * width + x ) * 4 + 2 ] = sr [ ( y * width + x ) * 4 + 0 ] ;
wr [ ( y * width + x ) * 4 + 3 ] = 255 ;
}
}
} else {
2024-07-17 13:25:28 +02:00
String msg = vformat ( " XImage with RGB mask %x %x %x and depth %d is not supported. " , ( uint64_t ) image - > red_mask , ( uint64_t ) image - > green_mask , ( uint64_t ) image - > blue_mask , ( int64_t ) image - > bits_per_pixel ) ;
XDestroyImage ( image ) ;
ERR_FAIL_V_MSG ( Ref < Image > ( ) , msg ) ;
2023-03-16 22:26:09 +01:00
}
img = Image : : create_from_data ( width , height , false , Image : : FORMAT_RGBA8 , img_data ) ;
2024-07-17 13:25:28 +02:00
XDestroyImage ( image ) ;
2023-03-16 22:26:09 +01:00
}
return img ;
}
2022-01-27 20:46:57 +01:00
float DisplayServerX11 : : screen_get_refresh_rate ( int p_screen ) const {
_THREAD_SAFE_METHOD_
2023-03-21 12:08:46 +01:00
p_screen = _get_screen_index ( p_screen ) ;
2022-01-27 20:46:57 +01:00
ERR_FAIL_INDEX_V ( p_screen , get_screen_count ( ) , SCREEN_REFRESH_RATE_FALLBACK ) ;
//Use xrandr to get screen refresh rate.
if ( xrandr_ext_ok ) {
2023-10-24 20:25:51 +02:00
XRRScreenResources * screen_info = XRRGetScreenResourcesCurrent ( x11_display , windows [ MAIN_WINDOW_ID ] . x11_window ) ;
2022-01-27 20:46:57 +01:00
if ( screen_info ) {
RRMode current_mode = 0 ;
xrr_monitor_info * monitors = nullptr ;
if ( xrr_get_monitors ) {
int count = 0 ;
monitors = xrr_get_monitors ( x11_display , windows [ MAIN_WINDOW_ID ] . x11_window , true , & count ) ;
ERR_FAIL_INDEX_V ( p_screen , count , SCREEN_REFRESH_RATE_FALLBACK ) ;
} else {
2022-02-10 12:00:11 +01:00
ERR_PRINT ( " An error occurred while trying to get the screen refresh rate. " ) ;
2022-01-27 20:46:57 +01:00
return SCREEN_REFRESH_RATE_FALLBACK ;
}
bool found_active_mode = false ;
for ( int crtc = 0 ; crtc < screen_info - > ncrtc ; crtc + + ) { // Loop through outputs to find which one is currently outputting.
XRRCrtcInfo * monitor_info = XRRGetCrtcInfo ( x11_display , screen_info , screen_info - > crtcs [ crtc ] ) ;
if ( monitor_info - > x ! = monitors [ p_screen ] . x | | monitor_info - > y ! = monitors [ p_screen ] . y ) { // If X and Y aren't the same as the monitor we're looking for, this isn't the right monitor. Continue.
continue ;
}
if ( monitor_info - > mode ! = None ) {
current_mode = monitor_info - > mode ;
found_active_mode = true ;
break ;
}
}
if ( found_active_mode ) {
for ( int mode = 0 ; mode < screen_info - > nmode ; mode + + ) {
XRRModeInfo m_info = screen_info - > modes [ mode ] ;
if ( m_info . id = = current_mode ) {
2022-02-04 17:21:07 +01:00
// Snap to nearest 0.01 to stay consistent with other platforms.
return Math : : snapped ( ( float ) m_info . dotClock / ( ( float ) m_info . hTotal * ( float ) m_info . vTotal ) , 0.01 ) ;
2022-01-27 20:46:57 +01:00
}
}
}
2022-02-10 12:00:11 +01:00
ERR_PRINT ( " An error occurred while trying to get the screen refresh rate. " ) ; // We should have returned the refresh rate by now. An error must have occurred.
2022-01-27 20:46:57 +01:00
return SCREEN_REFRESH_RATE_FALLBACK ;
} else {
2022-02-10 12:00:11 +01:00
ERR_PRINT ( " An error occurred while trying to get the screen refresh rate. " ) ;
2022-01-27 20:46:57 +01:00
return SCREEN_REFRESH_RATE_FALLBACK ;
}
}
2022-02-10 12:00:11 +01:00
ERR_PRINT ( " An error occurred while trying to get the screen refresh rate. " ) ;
2022-01-27 20:46:57 +01:00
return SCREEN_REFRESH_RATE_FALLBACK ;
}
2020-04-02 23:46:34 +02:00
# ifdef DBUS_ENABLED
void DisplayServerX11 : : screen_set_keep_on ( bool p_enable ) {
if ( screen_is_kept_on ( ) = = p_enable ) {
return ;
}
if ( p_enable ) {
screensaver - > inhibit ( ) ;
} else {
screensaver - > uninhibit ( ) ;
}
keep_screen_on = p_enable ;
}
bool DisplayServerX11 : : screen_is_kept_on ( ) const {
return keep_screen_on ;
}
# endif
2020-03-01 23:14:37 +01:00
Vector < DisplayServer : : WindowID > DisplayServerX11 : : get_window_list ( ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-01 23:14:37 +01:00
Vector < int > ret ;
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
ret . push_back ( E . key ) ;
2020-03-01 23:14:37 +01:00
}
return ret ;
}
2024-07-24 18:20:31 +02:00
DisplayServer : : WindowID DisplayServerX11 : : create_sub_window ( WindowMode p_mode , VSyncMode p_vsync_mode , uint32_t p_flags , const Rect2i & p_rect , bool p_exclusive , WindowID p_transient_parent ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2023-01-04 23:00:02 +01:00
WindowID id = _create_window ( p_mode , p_vsync_mode , p_flags , p_rect ) ;
2020-03-01 23:14:37 +01:00
for ( int i = 0 ; i < WINDOW_FLAG_MAX ; i + + ) {
if ( p_flags & ( 1 < < i ) ) {
window_set_flag ( WindowFlags ( i ) , true , id ) ;
}
}
2023-12-19 18:57:56 +01:00
# ifdef RD_ENABLED
if ( rendering_device ) {
rendering_device - > screen_create ( id ) ;
}
# endif
2024-07-24 18:20:31 +02:00
if ( p_transient_parent ! = INVALID_WINDOW_ID ) {
window_set_transient ( id , p_transient_parent ) ;
}
2020-03-01 23:14:37 +01:00
return id ;
}
2020-08-21 09:39:30 +02:00
void DisplayServerX11 : : show_window ( WindowID p_id ) {
_THREAD_SAFE_METHOD_
2021-07-21 21:16:53 +02:00
const WindowData & wd = windows [ p_id ] ;
2022-02-24 10:21:23 +01:00
popup_open ( p_id ) ;
2021-07-21 21:16:53 +02:00
DEBUG_LOG_X11 ( " show_window: %lu (%u) \n " , wd . x11_window , p_id ) ;
2020-08-21 09:39:30 +02:00
XMapWindow ( x11_display , wd . x11_window ) ;
2022-12-08 19:34:15 +01:00
XSync ( x11_display , False ) ;
_validate_mode_on_map ( p_id ) ;
2020-08-21 09:39:30 +02:00
}
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : delete_sub_window ( WindowID p_id ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-06 18:00:16 +01:00
ERR_FAIL_COND ( ! windows . has ( p_id ) ) ;
2022-02-24 10:21:23 +01:00
ERR_FAIL_COND_MSG ( p_id = = MAIN_WINDOW_ID , " Main window can't be deleted " ) ;
popup_close ( p_id ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_id ] ;
2020-08-22 13:01:49 +02:00
DEBUG_LOG_X11 ( " delete_sub_window: %lu (%u) \n " , wd . x11_window , p_id ) ;
2023-08-05 01:52:59 +02:00
if ( window_mouseover_id = = p_id ) {
window_mouseover_id = INVALID_WINDOW_ID ;
_send_window_event ( windows [ p_id ] , WINDOW_EVENT_MOUSE_EXIT ) ;
}
2023-08-02 00:17:25 +02:00
window_set_rect_changed_callback ( Callable ( ) , p_id ) ;
window_set_window_event_callback ( Callable ( ) , p_id ) ;
window_set_input_event_callback ( Callable ( ) , p_id ) ;
window_set_input_text_callback ( Callable ( ) , p_id ) ;
window_set_drop_files_callback ( Callable ( ) , p_id ) ;
2020-03-04 20:03:30 +01:00
while ( wd . transient_children . size ( ) ) {
2022-05-19 17:00:06 +02:00
window_set_transient ( * wd . transient_children . begin ( ) , INVALID_WINDOW_ID ) ;
2020-03-04 20:03:30 +01:00
}
if ( wd . transient_parent ! = INVALID_WINDOW_ID ) {
window_set_transient ( p_id , INVALID_WINDOW_ID ) ;
}
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
2023-12-19 18:57:56 +01:00
if ( rendering_device ) {
rendering_device - > screen_free ( p_id ) ;
}
if ( rendering_context ) {
rendering_context - > window_destroy ( p_id ) ;
2020-03-03 14:36:29 +01:00
}
# endif
2021-10-26 17:18:39 +02:00
# ifdef GLES3_ENABLED
2020-11-18 19:11:30 +01:00
if ( gl_manager ) {
gl_manager - > window_destroy ( p_id ) ;
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
gl_manager_egl - > window_destroy ( p_id ) ;
}
2020-11-18 19:11:30 +01:00
# endif
2020-03-01 23:14:37 +01:00
if ( wd . xic ) {
XDestroyIC ( wd . xic ) ;
2020-10-08 17:21:51 +02:00
wd . xic = nullptr ;
2020-03-01 23:14:37 +01:00
}
2023-01-30 09:59:49 +01:00
XDestroyWindow ( x11_display , wd . x11_xim_window ) ;
2023-02-15 13:13:56 +01:00
# ifdef XKB_ENABLED
2023-03-16 09:44:47 +01:00
if ( xkb_loaded_v05p ) {
2023-01-30 09:59:49 +01:00
if ( wd . xkb_state ) {
xkb_compose_state_unref ( wd . xkb_state ) ;
wd . xkb_state = nullptr ;
}
}
2023-02-15 13:13:56 +01:00
# endif
2023-01-30 09:59:49 +01:00
XUnmapWindow ( x11_display , wd . x11_window ) ;
XDestroyWindow ( x11_display , wd . x11_window ) ;
2020-03-01 23:14:37 +01:00
windows . erase ( p_id ) ;
}
2022-01-14 11:41:04 +01:00
int64_t DisplayServerX11 : : window_get_native_handle ( HandleType p_handle_type , WindowID p_window ) const {
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , 0 ) ;
switch ( p_handle_type ) {
case DISPLAY_HANDLE : {
return ( int64_t ) x11_display ;
}
case WINDOW_HANDLE : {
return ( int64_t ) windows [ p_window ] . x11_window ;
}
case WINDOW_VIEW : {
return 0 ; // Not supported.
}
2022-10-23 03:29:15 +02:00
# ifdef GLES3_ENABLED
case OPENGL_CONTEXT : {
2022-11-13 00:48:45 +01:00
if ( gl_manager ) {
return ( int64_t ) gl_manager - > get_glx_context ( p_window ) ;
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
return ( int64_t ) gl_manager_egl - > get_context ( p_window ) ;
}
2022-11-13 00:48:45 +01:00
return 0 ;
2022-10-23 03:29:15 +02:00
}
# endif
2022-01-14 11:41:04 +01:00
default : {
return 0 ;
}
}
}
2020-03-25 00:15:35 +01:00
void DisplayServerX11 : : window_attach_instance_id ( ObjectID p_instance , WindowID p_window ) {
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
WindowData & wd = windows [ p_window ] ;
wd . instance_id = p_instance ;
}
ObjectID DisplayServerX11 : : window_get_attached_instance_id ( WindowID p_window ) const {
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , ObjectID ( ) ) ;
const WindowData & wd = windows [ p_window ] ;
return wd . instance_id ;
}
DisplayServerX11 : : WindowID DisplayServerX11 : : get_window_at_screen_position ( const Point2i & p_position ) const {
2020-08-29 12:00:44 +02:00
WindowID found_window = INVALID_WINDOW_ID ;
WindowID parent_window = INVALID_WINDOW_ID ;
unsigned int focus_order = 0 ;
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
const WindowData & wd = E . value ;
2020-08-29 12:00:44 +02:00
// Discard windows with no focus.
if ( wd . focus_order = = 0 ) {
continue ;
}
// Find topmost window which contains the given position.
2021-08-09 22:13:42 +02:00
WindowID window_id = E . key ;
2020-08-29 12:00:44 +02:00
Rect2i win_rect = Rect2i ( window_get_position ( window_id ) , window_get_size ( window_id ) ) ;
2020-07-03 17:08:37 +02:00
if ( win_rect . has_point ( p_position ) ) {
2020-08-29 12:00:44 +02:00
// For siblings, pick the window which was focused last.
if ( ( parent_window ! = wd . transient_parent ) | | ( wd . focus_order > focus_order ) ) {
found_window = window_id ;
parent_window = wd . transient_parent ;
focus_order = wd . focus_order ;
}
2020-07-03 17:08:37 +02:00
}
}
2020-08-29 12:00:44 +02:00
return found_window ;
2020-03-25 00:15:35 +01:00
}
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : window_set_title ( const String & p_title , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
XStoreName ( x11_display , wd . x11_window , p_title . utf8 ( ) . get_data ( ) ) ;
Atom _net_wm_name = XInternAtom ( x11_display , " _NET_WM_NAME " , false ) ;
Atom utf8_string = XInternAtom ( x11_display , " UTF8_STRING " , false ) ;
2020-11-14 21:31:16 +01:00
if ( _net_wm_name ! = None & & utf8_string ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , _net_wm_name , utf8_string , 8 , PropModeReplace , ( unsigned char * ) p_title . utf8 ( ) . get_data ( ) , p_title . utf8 ( ) . length ( ) ) ;
}
2020-03-01 23:14:37 +01:00
}
2020-06-29 11:31:36 +02:00
void DisplayServerX11 : : window_set_mouse_passthrough ( const Vector < Vector2 > & p_region , WindowID p_window ) {
_THREAD_SAFE_METHOD_
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2023-01-15 11:05:25 +01:00
windows [ p_window ] . mpath = p_region ;
_update_window_mouse_passthrough ( p_window ) ;
}
void DisplayServerX11 : : _update_window_mouse_passthrough ( WindowID p_window ) {
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2023-08-04 09:58:14 +02:00
ERR_FAIL_COND ( ! xshaped_ext_ok ) ;
2023-01-15 11:05:25 +01:00
const Vector < Vector2 > region_path = windows [ p_window ] . mpath ;
2020-06-29 11:31:36 +02:00
int event_base , error_base ;
const Bool ext_okay = XShapeQueryExtension ( x11_display , & event_base , & error_base ) ;
if ( ext_okay ) {
2023-01-15 11:05:25 +01:00
if ( windows [ p_window ] . mpass ) {
2023-01-17 06:54:54 +01:00
Region region = XCreateRegion ( ) ;
XShapeCombineRegion ( x11_display , windows [ p_window ] . x11_window , ShapeInput , 0 , 0 , region , ShapeSet ) ;
XDestroyRegion ( region ) ;
2023-01-15 11:05:25 +01:00
} else if ( region_path . size ( ) = = 0 ) {
2023-01-17 06:54:54 +01:00
XShapeCombineMask ( x11_display , windows [ p_window ] . x11_window , ShapeInput , 0 , 0 , None , ShapeSet ) ;
2020-06-29 11:31:36 +02:00
} else {
2023-01-15 11:05:25 +01:00
XPoint * points = ( XPoint * ) memalloc ( sizeof ( XPoint ) * region_path . size ( ) ) ;
for ( int i = 0 ; i < region_path . size ( ) ; i + + ) {
points [ i ] . x = region_path [ i ] . x ;
points [ i ] . y = region_path [ i ] . y ;
2020-06-29 11:31:36 +02:00
}
2023-01-17 06:54:54 +01:00
Region region = XPolygonRegion ( points , region_path . size ( ) , EvenOddRule ) ;
2020-06-29 11:31:36 +02:00
memfree ( points ) ;
2023-01-17 06:54:54 +01:00
XShapeCombineRegion ( x11_display , windows [ p_window ] . x11_window , ShapeInput , 0 , 0 , region , ShapeSet ) ;
XDestroyRegion ( region ) ;
2020-06-29 11:31:36 +02:00
}
}
}
2020-03-14 17:06:39 +01:00
void DisplayServerX11 : : window_set_rect_changed_callback ( const Callable & p_callable , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-04 02:51:12 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
WindowData & wd = windows [ p_window ] ;
2020-03-14 17:06:39 +01:00
wd . rect_changed_callback = p_callable ;
2020-03-04 02:51:12 +01:00
}
2020-03-04 17:36:09 +01:00
void DisplayServerX11 : : window_set_window_event_callback ( const Callable & p_callable , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-04 17:36:09 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
WindowData & wd = windows [ p_window ] ;
wd . event_callback = p_callable ;
}
void DisplayServerX11 : : window_set_input_event_callback ( const Callable & p_callable , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-04 17:36:09 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
WindowData & wd = windows [ p_window ] ;
wd . input_event_callback = p_callable ;
}
2020-05-14 14:29:06 +02:00
2020-03-04 17:36:09 +01:00
void DisplayServerX11 : : window_set_input_text_callback ( const Callable & p_callable , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-04 17:36:09 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
WindowData & wd = windows [ p_window ] ;
wd . input_text_callback = p_callable ;
}
void DisplayServerX11 : : window_set_drop_files_callback ( const Callable & p_callable , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-04 17:36:09 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
WindowData & wd = windows [ p_window ] ;
wd . drop_files_callback = p_callable ;
}
2020-03-01 23:14:37 +01:00
int DisplayServerX11 : : window_get_current_screen ( WindowID p_window ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-09-16 16:33:33 +02:00
int count = get_screen_count ( ) ;
if ( count < 2 ) {
// Early exit with single monitor.
return 0 ;
}
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , 0 ) ;
2020-03-01 23:14:37 +01:00
const WindowData & wd = windows [ p_window ] ;
2020-09-16 16:33:33 +02:00
const Rect2i window_rect ( wd . position , wd . size ) ;
2020-03-01 23:14:37 +01:00
2020-09-16 16:33:33 +02:00
// Find which monitor has the largest overlap with the given window.
int screen_index = 0 ;
int max_area = 0 ;
2020-03-01 23:14:37 +01:00
for ( int i = 0 ; i < count ; i + + ) {
2020-09-16 16:33:33 +02:00
Rect2i screen_rect = _screen_get_rect ( i ) ;
Rect2i intersection = screen_rect . intersection ( window_rect ) ;
int area = intersection . get_area ( ) ;
if ( area > max_area ) {
max_area = area ;
screen_index = i ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
}
2020-09-16 16:33:33 +02:00
return screen_index ;
2020-03-01 23:14:37 +01:00
}
2020-05-14 14:29:06 +02:00
2020-11-18 19:11:30 +01:00
void DisplayServerX11 : : gl_window_make_current ( DisplayServer : : WindowID p_window_id ) {
2021-10-26 17:18:39 +02:00
# if defined(GLES3_ENABLED)
2022-01-27 17:34:33 +01:00
if ( gl_manager ) {
2020-11-18 19:11:30 +01:00
gl_manager - > window_make_current ( p_window_id ) ;
2022-01-27 17:34:33 +01:00
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
gl_manager_egl - > window_make_current ( p_window_id ) ;
}
2020-11-18 19:11:30 +01:00
# endif
}
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : window_set_current_screen ( int p_screen , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
2023-03-21 12:08:46 +01:00
p_screen = _get_screen_index ( p_screen ) ;
2021-02-28 17:43:46 +01:00
ERR_FAIL_INDEX ( p_screen , get_screen_count ( ) ) ;
2022-12-08 19:34:15 +01:00
if ( window_get_current_screen ( p_window ) = = p_screen ) {
return ;
}
2020-03-01 23:14:37 +01:00
if ( window_get_mode ( p_window ) = = WINDOW_MODE_FULLSCREEN ) {
Point2i position = screen_get_position ( p_screen ) ;
Size2i size = screen_get_size ( p_screen ) ;
XMoveResizeWindow ( x11_display , wd . x11_window , position . x , position . y , size . x , size . y ) ;
} else {
2022-12-08 19:34:15 +01:00
Rect2i srect = screen_get_usable_rect ( p_screen ) ;
Point2i wpos = window_get_position ( p_window ) - screen_get_position ( window_get_current_screen ( p_window ) ) ;
Size2i wsize = window_get_size ( p_window ) ;
wpos + = srect . position ;
2023-01-04 23:00:02 +01:00
if ( srect ! = Rect2i ( ) ) {
2024-03-03 12:49:08 +01:00
wpos = wpos . clamp ( srect . position , srect . position + srect . size - wsize / 3 ) ;
2023-01-04 23:00:02 +01:00
}
2022-12-08 19:34:15 +01:00
window_set_position ( wpos , p_window ) ;
2020-03-01 23:14:37 +01:00
}
}
2020-03-04 20:03:30 +01:00
void DisplayServerX11 : : window_set_transient ( WindowID p_window , WindowID p_parent ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-04 20:03:30 +01:00
ERR_FAIL_COND ( p_window = = p_parent ) ;
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
WindowData & wd_window = windows [ p_window ] ;
2020-08-22 13:01:49 +02:00
WindowID prev_parent = wd_window . transient_parent ;
ERR_FAIL_COND ( prev_parent = = p_parent ) ;
2020-03-04 20:03:30 +01:00
2021-07-21 21:16:53 +02:00
DEBUG_LOG_X11 ( " window_set_transient: %lu (%u), prev_parent=%u, parent=%u \n " , wd_window . x11_window , p_window , prev_parent , p_parent ) ;
2020-03-04 20:03:30 +01:00
ERR_FAIL_COND_MSG ( wd_window . on_top , " Windows with the 'on top' can't become transient. " ) ;
if ( p_parent = = INVALID_WINDOW_ID ) {
//remove transient
2020-08-22 13:01:49 +02:00
ERR_FAIL_COND ( prev_parent = = INVALID_WINDOW_ID ) ;
ERR_FAIL_COND ( ! windows . has ( prev_parent ) ) ;
2020-03-04 20:03:30 +01:00
2020-08-22 13:01:49 +02:00
WindowData & wd_parent = windows [ prev_parent ] ;
2020-03-04 20:03:30 +01:00
wd_window . transient_parent = INVALID_WINDOW_ID ;
wd_parent . transient_children . erase ( p_window ) ;
XSetTransientForHint ( x11_display , wd_window . x11_window , None ) ;
2020-08-22 13:01:49 +02:00
2022-08-26 22:13:58 +02:00
XWindowAttributes xwa ;
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd_parent . x11_window , & xwa ) ;
2021-07-21 21:16:53 +02:00
// Set focus to parent sub window to avoid losing all focus when closing a nested sub-menu.
2020-08-22 13:01:49 +02:00
// RevertToPointerRoot is used to make sure we don't lose all focus in case
// a subwindow and its parent are both destroyed.
2022-02-24 10:21:23 +01:00
if ( ! wd_window . no_focus & & ! wd_window . is_popup & & wd_window . focused ) {
2023-03-04 17:52:15 +01:00
if ( ( xwa . map_state = = IsViewable ) & & ! wd_parent . no_focus & & ! wd_window . is_popup & & _window_focus_check ( ) ) {
2023-12-31 19:53:27 +01:00
_set_input_focus ( wd_parent . x11_window , RevertToPointerRoot ) ;
2020-08-22 13:01:49 +02:00
}
}
2020-03-04 20:03:30 +01:00
} else {
ERR_FAIL_COND ( ! windows . has ( p_parent ) ) ;
2020-08-22 13:01:49 +02:00
ERR_FAIL_COND_MSG ( prev_parent ! = INVALID_WINDOW_ID , " Window already has a transient parent " ) ;
2020-03-04 20:03:30 +01:00
WindowData & wd_parent = windows [ p_parent ] ;
wd_window . transient_parent = p_parent ;
wd_parent . transient_children . insert ( p_window ) ;
XSetTransientForHint ( x11_display , wd_window . x11_window , wd_parent . x11_window ) ;
}
}
2020-07-31 17:27:16 +02:00
// Helper method. Assumes that the window id has already been checked and exists.
void DisplayServerX11 : : _update_size_hints ( WindowID p_window ) {
WindowData & wd = windows [ p_window ] ;
WindowMode window_mode = window_get_mode ( p_window ) ;
XSizeHints * xsh = XAllocSizeHints ( ) ;
// Always set the position and size hints - they should be synchronized with the actual values after the window is mapped anyway
xsh - > flags | = PPosition | PSize ;
xsh - > x = wd . position . x ;
xsh - > y = wd . position . y ;
xsh - > width = wd . size . width ;
xsh - > height = wd . size . height ;
2022-12-07 08:54:02 +01:00
if ( window_mode = = WINDOW_MODE_FULLSCREEN | | window_mode = = WINDOW_MODE_EXCLUSIVE_FULLSCREEN ) {
2020-07-31 17:27:16 +02:00
// Do not set any other hints to prevent the window manager from ignoring the fullscreen flags
} else if ( window_get_flag ( WINDOW_FLAG_RESIZE_DISABLED , p_window ) ) {
// If resizing is disabled, use the forced size
xsh - > flags | = PMinSize | PMaxSize ;
xsh - > min_width = wd . size . x ;
xsh - > max_width = wd . size . x ;
xsh - > min_height = wd . size . y ;
xsh - > max_height = wd . size . y ;
} else {
// Otherwise, just respect min_size and max_size
if ( wd . min_size ! = Size2i ( ) ) {
xsh - > flags | = PMinSize ;
xsh - > min_width = wd . min_size . x ;
xsh - > min_height = wd . min_size . y ;
}
if ( wd . max_size ! = Size2i ( ) ) {
xsh - > flags | = PMaxSize ;
xsh - > max_width = wd . max_size . x ;
xsh - > max_height = wd . max_size . y ;
}
}
XSetWMNormalHints ( x11_display , wd . x11_window , xsh ) ;
XFree ( xsh ) ;
}
2020-03-01 23:14:37 +01:00
Point2i DisplayServerX11 : : window_get_position ( WindowID p_window ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , Point2i ( ) ) ;
2020-03-01 23:14:37 +01:00
const WindowData & wd = windows [ p_window ] ;
2020-03-25 15:16:19 +01:00
return wd . position ;
2020-03-01 23:14:37 +01:00
}
2020-03-25 15:16:19 +01:00
2022-11-30 09:28:16 +01:00
Point2i DisplayServerX11 : : window_get_position_with_decorations ( WindowID p_window ) const {
_THREAD_SAFE_METHOD_
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , Size2i ( ) ) ;
const WindowData & wd = windows [ p_window ] ;
if ( wd . fullscreen ) {
return wd . position ;
}
XWindowAttributes xwa ;
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd . x11_window , & xwa ) ;
int x = wd . position . x ;
int y = wd . position . y ;
Atom prop = XInternAtom ( x11_display , " _NET_FRAME_EXTENTS " , True ) ;
if ( prop ! = None ) {
Atom type ;
int format ;
unsigned long len ;
unsigned long remaining ;
unsigned char * data = nullptr ;
if ( XGetWindowProperty ( x11_display , wd . x11_window , prop , 0 , 4 , False , AnyPropertyType , & type , & format , & len , & remaining , & data ) = = Success ) {
if ( format = = 32 & & len = = 4 & & data ) {
long * extents = ( long * ) data ;
x - = extents [ 0 ] ; // left
y - = extents [ 2 ] ; // top
}
XFree ( data ) ;
}
}
return Size2i ( x , y ) ;
}
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : window_set_position ( const Point2i & p_position , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
int x = 0 ;
int y = 0 ;
if ( ! window_get_flag ( WINDOW_FLAG_BORDERLESS , p_window ) ) {
//exclude window decorations
XSync ( x11_display , False ) ;
Atom prop = XInternAtom ( x11_display , " _NET_FRAME_EXTENTS " , True ) ;
if ( prop ! = None ) {
Atom type ;
int format ;
unsigned long len ;
unsigned long remaining ;
2020-04-02 01:20:12 +02:00
unsigned char * data = nullptr ;
2020-03-01 23:14:37 +01:00
if ( XGetWindowProperty ( x11_display , wd . x11_window , prop , 0 , 4 , False , AnyPropertyType , & type , & format , & len , & remaining , & data ) = = Success ) {
2020-03-06 18:00:16 +01:00
if ( format = = 32 & & len = = 4 & & data ) {
2020-03-01 23:14:37 +01:00
long * extents = ( long * ) data ;
x = extents [ 0 ] ;
y = extents [ 2 ] ;
}
XFree ( data ) ;
}
}
}
XMoveWindow ( x11_display , wd . x11_window , p_position . x - x , p_position . y - y ) ;
_update_real_mouse_position ( wd ) ;
}
void DisplayServerX11 : : window_set_max_size ( const Size2i p_size , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
if ( ( p_size ! = Size2i ( ) ) & & ( ( p_size . x < wd . min_size . x ) | | ( p_size . y < wd . min_size . y ) ) ) {
ERR_PRINT ( " Maximum window size can't be smaller than minimum window size! " ) ;
return ;
}
wd . max_size = p_size ;
2020-07-31 17:27:16 +02:00
_update_size_hints ( p_window ) ;
XFlush ( x11_display ) ;
2020-03-01 23:14:37 +01:00
}
2020-05-14 14:29:06 +02:00
2020-03-01 23:14:37 +01:00
Size2i DisplayServerX11 : : window_get_max_size ( WindowID p_window ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , Size2i ( ) ) ;
2020-03-01 23:14:37 +01:00
const WindowData & wd = windows [ p_window ] ;
return wd . max_size ;
}
void DisplayServerX11 : : window_set_min_size ( const Size2i p_size , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
if ( ( p_size ! = Size2i ( ) ) & & ( wd . max_size ! = Size2i ( ) ) & & ( ( p_size . x > wd . max_size . x ) | | ( p_size . y > wd . max_size . y ) ) ) {
ERR_PRINT ( " Minimum window size can't be larger than maximum window size! " ) ;
return ;
}
wd . min_size = p_size ;
2020-07-31 17:27:16 +02:00
_update_size_hints ( p_window ) ;
XFlush ( x11_display ) ;
2020-03-01 23:14:37 +01:00
}
2020-05-14 14:29:06 +02:00
2020-03-01 23:14:37 +01:00
Size2i DisplayServerX11 : : window_get_min_size ( WindowID p_window ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , Size2i ( ) ) ;
2020-03-01 23:14:37 +01:00
const WindowData & wd = windows [ p_window ] ;
return wd . min_size ;
}
void DisplayServerX11 : : window_set_size ( const Size2i p_size , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-25 15:16:19 +01:00
Size2i size = p_size ;
2024-03-03 14:37:52 +01:00
size = size . maxi ( 1 ) ;
2020-03-25 15:16:19 +01:00
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
2020-05-14 16:41:43 +02:00
if ( wd . size . width = = size . width & & wd . size . height = = size . height ) {
2020-03-01 23:14:37 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
XWindowAttributes xwa ;
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd . x11_window , & xwa ) ;
int old_w = xwa . width ;
int old_h = xwa . height ;
2020-07-31 17:27:16 +02:00
// Update our videomode width and height
wd . size = size ;
// Update the size hints first to make sure the window size can be set
_update_size_hints ( p_window ) ;
2020-03-01 23:14:37 +01:00
// Resize the window
2020-03-25 15:16:19 +01:00
XResizeWindow ( x11_display , wd . x11_window , size . x , size . y ) ;
2020-03-01 23:14:37 +01:00
for ( int timeout = 0 ; timeout < 50 ; + + timeout ) {
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd . x11_window , & xwa ) ;
2020-05-14 16:41:43 +02:00
if ( old_w ! = xwa . width | | old_h ! = xwa . height ) {
2020-03-01 23:14:37 +01:00
break ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
2024-05-21 17:32:14 +02:00
OS : : get_singleton ( ) - > delay_usec ( 10'000 ) ;
2020-03-01 23:14:37 +01:00
}
2022-10-11 23:04:48 +02:00
// Keep rendering context window size in sync
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
2023-12-19 18:57:56 +01:00
if ( rendering_context ) {
rendering_context - > window_set_size ( p_window , xwa . width , xwa . height ) ;
2022-10-11 23:04:48 +02:00
}
# endif
# if defined(GLES3_ENABLED)
if ( gl_manager ) {
gl_manager - > window_resize ( p_window , xwa . width , xwa . height ) ;
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
gl_manager_egl - > window_resize ( p_window , xwa . width , xwa . height ) ;
}
2022-10-11 23:04:48 +02:00
# endif
2020-03-01 23:14:37 +01:00
}
2020-05-14 14:29:06 +02:00
2020-03-01 23:14:37 +01:00
Size2i DisplayServerX11 : : window_get_size ( WindowID p_window ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , Size2i ( ) ) ;
2020-03-01 23:14:37 +01:00
const WindowData & wd = windows [ p_window ] ;
return wd . size ;
}
2020-05-14 14:29:06 +02:00
2022-11-30 09:28:16 +01:00
Size2i DisplayServerX11 : : window_get_size_with_decorations ( WindowID p_window ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , Size2i ( ) ) ;
2020-03-01 23:14:37 +01:00
const WindowData & wd = windows [ p_window ] ;
2022-11-30 09:28:16 +01:00
if ( wd . fullscreen ) {
return wd . size ;
}
2020-03-01 23:14:37 +01:00
XWindowAttributes xwa ;
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd . x11_window , & xwa ) ;
int w = xwa . width ;
int h = xwa . height ;
Atom prop = XInternAtom ( x11_display , " _NET_FRAME_EXTENTS " , True ) ;
if ( prop ! = None ) {
Atom type ;
int format ;
unsigned long len ;
unsigned long remaining ;
2020-04-02 01:20:12 +02:00
unsigned char * data = nullptr ;
2020-03-01 23:14:37 +01:00
if ( XGetWindowProperty ( x11_display , wd . x11_window , prop , 0 , 4 , False , AnyPropertyType , & type , & format , & len , & remaining , & data ) = = Success ) {
2020-03-06 18:00:16 +01:00
if ( format = = 32 & & len = = 4 & & data ) {
2020-03-01 23:14:37 +01:00
long * extents = ( long * ) data ;
w + = extents [ 0 ] + extents [ 1 ] ; // left, right
h + = extents [ 2 ] + extents [ 3 ] ; // top, bottom
}
XFree ( data ) ;
}
}
2020-03-13 00:12:34 +01:00
return Size2i ( w , h ) ;
2020-03-01 23:14:37 +01:00
}
2020-06-29 13:29:31 +02:00
// Just a helper to reduce code duplication in `window_is_maximize_allowed`
// and `_set_wm_maximized`.
bool DisplayServerX11 : : _window_maximize_check ( WindowID p_window , const char * p_atom_name ) const {
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , false ) ;
2020-03-01 23:14:37 +01:00
const WindowData & wd = windows [ p_window ] ;
2020-06-29 13:29:31 +02:00
Atom property = XInternAtom ( x11_display , p_atom_name , False ) ;
2020-03-01 23:14:37 +01:00
Atom type ;
int format ;
unsigned long len ;
unsigned long remaining ;
2020-04-02 01:20:12 +02:00
unsigned char * data = nullptr ;
2020-06-29 13:29:31 +02:00
bool retval = false ;
2020-03-01 23:14:37 +01:00
2020-11-12 14:38:54 +01:00
if ( property = = None ) {
return false ;
}
2020-03-01 23:14:37 +01:00
int result = XGetWindowProperty (
x11_display ,
wd . x11_window ,
property ,
0 ,
1024 ,
False ,
XA_ATOM ,
& type ,
& format ,
& len ,
& remaining ,
& data ) ;
2020-03-06 18:00:16 +01:00
if ( result = = Success & & data ) {
2020-03-01 23:14:37 +01:00
Atom * atoms = ( Atom * ) data ;
2022-03-31 23:21:32 +02:00
Atom wm_act_max_horz ;
Atom wm_act_max_vert ;
if ( strcmp ( p_atom_name , " _NET_WM_STATE " ) = = 0 ) {
wm_act_max_horz = XInternAtom ( x11_display , " _NET_WM_STATE_MAXIMIZED_HORZ " , False ) ;
wm_act_max_vert = XInternAtom ( x11_display , " _NET_WM_STATE_MAXIMIZED_VERT " , False ) ;
} else {
wm_act_max_horz = XInternAtom ( x11_display , " _NET_WM_ACTION_MAXIMIZE_HORZ " , False ) ;
wm_act_max_vert = XInternAtom ( x11_display , " _NET_WM_ACTION_MAXIMIZE_VERT " , False ) ;
}
2020-03-01 23:14:37 +01:00
bool found_wm_act_max_horz = false ;
bool found_wm_act_max_vert = false ;
for ( uint64_t i = 0 ; i < len ; i + + ) {
2020-05-14 16:41:43 +02:00
if ( atoms [ i ] = = wm_act_max_horz ) {
2020-03-01 23:14:37 +01:00
found_wm_act_max_horz = true ;
2020-05-14 16:41:43 +02:00
}
if ( atoms [ i ] = = wm_act_max_vert ) {
2020-03-01 23:14:37 +01:00
found_wm_act_max_vert = true ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
2020-05-14 16:41:43 +02:00
if ( found_wm_act_max_horz | | found_wm_act_max_vert ) {
2020-06-29 13:29:31 +02:00
retval = true ;
break ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
}
2020-06-29 13:29:31 +02:00
XFree ( data ) ;
2020-03-01 23:14:37 +01:00
}
2020-06-29 13:29:31 +02:00
return retval ;
}
2022-08-30 20:52:24 +02:00
bool DisplayServerX11 : : _window_minimize_check ( WindowID p_window ) const {
const WindowData & wd = windows [ p_window ] ;
2023-07-30 02:05:47 +02:00
// Using EWMH instead of ICCCM, might work better for Wayland users.
Atom property = XInternAtom ( x11_display , " _NET_WM_STATE " , True ) ;
Atom hidden = XInternAtom ( x11_display , " _NET_WM_STATE_HIDDEN " , True ) ;
if ( property = = None | | hidden = = None ) {
2022-08-30 20:52:24 +02:00
return false ;
}
Atom type ;
int format ;
unsigned long len ;
unsigned long remaining ;
2023-07-30 02:05:47 +02:00
Atom * atoms = nullptr ;
2022-08-30 20:52:24 +02:00
int result = XGetWindowProperty (
x11_display ,
wd . x11_window ,
property ,
0 ,
32 ,
False ,
2023-07-30 02:05:47 +02:00
XA_ATOM ,
2022-08-30 20:52:24 +02:00
& type ,
& format ,
& len ,
& remaining ,
2023-07-30 02:05:47 +02:00
( unsigned char * * ) & atoms ) ;
2022-08-30 20:52:24 +02:00
2023-07-30 02:05:47 +02:00
if ( result = = Success & & atoms ) {
for ( unsigned int i = 0 ; i < len ; i + + ) {
if ( atoms [ i ] = = hidden ) {
XFree ( atoms ) ;
return true ;
}
2022-08-30 20:52:24 +02:00
}
2023-07-30 02:05:47 +02:00
XFree ( atoms ) ;
2022-08-30 20:52:24 +02:00
}
return false ;
}
2022-06-30 03:55:40 +02:00
bool DisplayServerX11 : : _window_fullscreen_check ( WindowID p_window ) const {
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , false ) ;
const WindowData & wd = windows [ p_window ] ;
// Using EWMH -- Extended Window Manager Hints
Atom property = XInternAtom ( x11_display , " _NET_WM_STATE " , False ) ;
Atom type ;
int format ;
unsigned long len ;
unsigned long remaining ;
unsigned char * data = nullptr ;
bool retval = false ;
if ( property = = None ) {
return retval ;
}
int result = XGetWindowProperty (
x11_display ,
wd . x11_window ,
property ,
0 ,
1024 ,
False ,
XA_ATOM ,
& type ,
& format ,
& len ,
& remaining ,
& data ) ;
if ( result = = Success ) {
Atom * atoms = ( Atom * ) data ;
Atom wm_fullscreen = XInternAtom ( x11_display , " _NET_WM_STATE_FULLSCREEN " , False ) ;
for ( uint64_t i = 0 ; i < len ; i + + ) {
if ( atoms [ i ] = = wm_fullscreen ) {
retval = true ;
break ;
}
}
XFree ( data ) ;
}
return retval ;
}
2022-08-30 20:52:24 +02:00
void DisplayServerX11 : : _validate_mode_on_map ( WindowID p_window ) {
// Check if we applied any window modes that didn't take effect while unmapped
2022-08-27 23:29:25 +02:00
const WindowData & wd = windows [ p_window ] ;
if ( wd . fullscreen & & ! _window_fullscreen_check ( p_window ) ) {
2022-12-07 08:54:02 +01:00
_set_wm_fullscreen ( p_window , true , wd . exclusive_fullscreen ) ;
2022-08-30 20:52:24 +02:00
} else if ( wd . maximized & & ! _window_maximize_check ( p_window , " _NET_WM_STATE " ) ) {
_set_wm_maximized ( p_window , true ) ;
} else if ( wd . minimized & & ! _window_minimize_check ( p_window ) ) {
_set_wm_minimized ( p_window , true ) ;
2022-08-27 23:29:25 +02:00
}
2023-02-08 08:45:00 +01:00
if ( wd . on_top ) {
Atom wm_state = XInternAtom ( x11_display , " _NET_WM_STATE " , False ) ;
Atom wm_above = XInternAtom ( x11_display , " _NET_WM_STATE_ABOVE " , False ) ;
XClientMessageEvent xev ;
memset ( & xev , 0 , sizeof ( xev ) ) ;
xev . type = ClientMessage ;
xev . window = wd . x11_window ;
xev . message_type = wm_state ;
xev . format = 32 ;
xev . data . l [ 0 ] = _NET_WM_STATE_ADD ;
xev . data . l [ 1 ] = wm_above ;
xev . data . l [ 3 ] = 1 ;
XSendEvent ( x11_display , DefaultRootWindow ( x11_display ) , False , SubstructureRedirectMask | SubstructureNotifyMask , ( XEvent * ) & xev ) ;
}
2022-08-27 23:29:25 +02:00
}
2020-06-29 13:29:31 +02:00
bool DisplayServerX11 : : window_is_maximize_allowed ( WindowID p_window ) const {
_THREAD_SAFE_METHOD_
return _window_maximize_check ( p_window , " _NET_WM_ALLOWED_ACTIONS " ) ;
2020-03-01 23:14:37 +01:00
}
void DisplayServerX11 : : _set_wm_maximized ( WindowID p_window , bool p_enabled ) {
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
// Using EWMH -- Extended Window Manager Hints
XEvent xev ;
Atom wm_state = XInternAtom ( x11_display , " _NET_WM_STATE " , False ) ;
Atom wm_max_horz = XInternAtom ( x11_display , " _NET_WM_STATE_MAXIMIZED_HORZ " , False ) ;
Atom wm_max_vert = XInternAtom ( x11_display , " _NET_WM_STATE_MAXIMIZED_VERT " , False ) ;
memset ( & xev , 0 , sizeof ( xev ) ) ;
xev . type = ClientMessage ;
xev . xclient . window = wd . x11_window ;
xev . xclient . message_type = wm_state ;
xev . xclient . format = 32 ;
xev . xclient . data . l [ 0 ] = p_enabled ? _NET_WM_STATE_ADD : _NET_WM_STATE_REMOVE ;
xev . xclient . data . l [ 1 ] = wm_max_horz ;
xev . xclient . data . l [ 2 ] = wm_max_vert ;
XSendEvent ( x11_display , DefaultRootWindow ( x11_display ) , False , SubstructureRedirectMask | SubstructureNotifyMask , & xev ) ;
if ( p_enabled & & window_is_maximize_allowed ( p_window ) ) {
// Wait for effective resizing (so the GLX context is too).
// Give up after 0.5s, it's not going to happen on this WM.
// https://github.com/godotengine/godot/issues/19978
for ( int attempt = 0 ; window_get_mode ( p_window ) ! = WINDOW_MODE_MAXIMIZED & & attempt < 50 ; attempt + + ) {
2024-05-21 17:32:14 +02:00
OS : : get_singleton ( ) - > delay_usec ( 10'000 ) ;
2020-03-01 23:14:37 +01:00
}
}
2022-08-30 20:52:24 +02:00
wd . maximized = p_enabled ;
}
void DisplayServerX11 : : _set_wm_minimized ( WindowID p_window , bool p_enabled ) {
WindowData & wd = windows [ p_window ] ;
// Using ICCCM -- Inter-Client Communication Conventions Manual
XEvent xev ;
Atom wm_change = XInternAtom ( x11_display , " WM_CHANGE_STATE " , False ) ;
memset ( & xev , 0 , sizeof ( xev ) ) ;
xev . type = ClientMessage ;
xev . xclient . window = wd . x11_window ;
xev . xclient . message_type = wm_change ;
xev . xclient . format = 32 ;
xev . xclient . data . l [ 0 ] = p_enabled ? WM_IconicState : WM_NormalState ;
XSendEvent ( x11_display , DefaultRootWindow ( x11_display ) , False , SubstructureRedirectMask | SubstructureNotifyMask , & xev ) ;
Atom wm_state = XInternAtom ( x11_display , " _NET_WM_STATE " , False ) ;
Atom wm_hidden = XInternAtom ( x11_display , " _NET_WM_STATE_HIDDEN " , False ) ;
memset ( & xev , 0 , sizeof ( xev ) ) ;
xev . type = ClientMessage ;
xev . xclient . window = wd . x11_window ;
xev . xclient . message_type = wm_state ;
xev . xclient . format = 32 ;
xev . xclient . data . l [ 0 ] = p_enabled ? _NET_WM_STATE_ADD : _NET_WM_STATE_REMOVE ;
xev . xclient . data . l [ 1 ] = wm_hidden ;
XSendEvent ( x11_display , DefaultRootWindow ( x11_display ) , False , SubstructureRedirectMask | SubstructureNotifyMask , & xev ) ;
wd . minimized = p_enabled ;
2020-03-01 23:14:37 +01:00
}
2022-12-07 08:54:02 +01:00
void DisplayServerX11 : : _set_wm_fullscreen ( WindowID p_window , bool p_enabled , bool p_exclusive ) {
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
if ( p_enabled & & ! window_get_flag ( WINDOW_FLAG_BORDERLESS , p_window ) ) {
// remove decorations if the window is not already borderless
Hints hints ;
Atom property ;
hints . flags = 2 ;
hints . decorations = 0 ;
property = XInternAtom ( x11_display , " _MOTIF_WM_HINTS " , True ) ;
2020-11-14 21:31:16 +01:00
if ( property ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , property , property , 32 , PropModeReplace , ( unsigned char * ) & hints , 5 ) ;
}
2020-03-01 23:14:37 +01:00
}
2020-07-31 17:27:16 +02:00
if ( p_enabled ) {
2020-03-01 23:14:37 +01:00
// Set the window as resizable to prevent window managers to ignore the fullscreen state flag.
2020-07-31 17:27:16 +02:00
_update_size_hints ( p_window ) ;
2020-03-01 23:14:37 +01:00
}
// Using EWMH -- Extended Window Manager Hints
XEvent xev ;
Atom wm_state = XInternAtom ( x11_display , " _NET_WM_STATE " , False ) ;
Atom wm_fullscreen = XInternAtom ( x11_display , " _NET_WM_STATE_FULLSCREEN " , False ) ;
memset ( & xev , 0 , sizeof ( xev ) ) ;
xev . type = ClientMessage ;
xev . xclient . window = wd . x11_window ;
xev . xclient . message_type = wm_state ;
xev . xclient . format = 32 ;
xev . xclient . data . l [ 0 ] = p_enabled ? _NET_WM_STATE_ADD : _NET_WM_STATE_REMOVE ;
xev . xclient . data . l [ 1 ] = wm_fullscreen ;
xev . xclient . data . l [ 2 ] = 0 ;
XSendEvent ( x11_display , DefaultRootWindow ( x11_display ) , False , SubstructureRedirectMask | SubstructureNotifyMask , & xev ) ;
// set bypass compositor hint
Atom bypass_compositor = XInternAtom ( x11_display , " _NET_WM_BYPASS_COMPOSITOR " , False ) ;
2022-12-07 08:54:02 +01:00
unsigned long compositing_disable_on = 0 ; // Use default.
if ( p_enabled ) {
if ( p_exclusive ) {
compositing_disable_on = 1 ; // Force composition OFF to reduce overhead.
} else {
compositing_disable_on = 2 ; // Force composition ON to allow popup windows.
}
}
2020-11-14 21:31:16 +01:00
if ( bypass_compositor ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , bypass_compositor , XA_CARDINAL , 32 , PropModeReplace , ( unsigned char * ) & compositing_disable_on , 1 ) ;
}
2020-03-01 23:14:37 +01:00
XFlush ( x11_display ) ;
if ( ! p_enabled ) {
// Reset the non-resizable flags if we un-set these before.
2020-07-31 17:27:16 +02:00
_update_size_hints ( p_window ) ;
2020-03-01 23:14:37 +01:00
// put back or remove decorations according to the last set borderless state
Hints hints ;
Atom property ;
hints . flags = 2 ;
2022-02-08 10:01:24 +01:00
hints . decorations = wd . borderless ? 0 : 1 ;
2020-03-01 23:14:37 +01:00
property = XInternAtom ( x11_display , " _MOTIF_WM_HINTS " , True ) ;
2020-11-14 21:31:16 +01:00
if ( property ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , property , property , 32 , PropModeReplace , ( unsigned char * ) & hints , 5 ) ;
}
2020-03-01 23:14:37 +01:00
}
}
void DisplayServerX11 : : window_set_mode ( WindowMode p_mode , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
WindowMode old_mode = window_get_mode ( p_window ) ;
if ( old_mode = = p_mode ) {
return ; // do nothing
}
//remove all "extra" modes
switch ( old_mode ) {
case WINDOW_MODE_WINDOWED : {
//do nothing
} break ;
case WINDOW_MODE_MINIMIZED : {
2022-08-30 20:52:24 +02:00
_set_wm_minimized ( p_window , false ) ;
2020-03-01 23:14:37 +01:00
} break ;
2022-01-28 10:19:53 +01:00
case WINDOW_MODE_EXCLUSIVE_FULLSCREEN :
2020-03-01 23:14:37 +01:00
case WINDOW_MODE_FULLSCREEN : {
//Remove full-screen
2020-07-31 17:27:16 +02:00
wd . fullscreen = false ;
2022-12-07 08:54:02 +01:00
wd . exclusive_fullscreen = false ;
2020-07-31 17:27:16 +02:00
2022-12-07 08:54:02 +01:00
_set_wm_fullscreen ( p_window , false , false ) ;
2020-03-01 23:14:37 +01:00
//un-maximize required for always on top
bool on_top = window_get_flag ( WINDOW_FLAG_ALWAYS_ON_TOP , p_window ) ;
window_set_position ( wd . last_position_before_fs , p_window ) ;
if ( on_top ) {
_set_wm_maximized ( p_window , false ) ;
}
} break ;
case WINDOW_MODE_MAXIMIZED : {
_set_wm_maximized ( p_window , false ) ;
} break ;
}
switch ( p_mode ) {
case WINDOW_MODE_WINDOWED : {
//do nothing
} break ;
case WINDOW_MODE_MINIMIZED : {
2022-08-30 20:52:24 +02:00
_set_wm_minimized ( p_window , true ) ;
2020-03-01 23:14:37 +01:00
} break ;
2022-01-28 10:19:53 +01:00
case WINDOW_MODE_EXCLUSIVE_FULLSCREEN :
2020-03-01 23:14:37 +01:00
case WINDOW_MODE_FULLSCREEN : {
2020-03-25 15:16:19 +01:00
wd . last_position_before_fs = wd . position ;
2020-07-31 17:27:16 +02:00
2020-03-01 23:14:37 +01:00
if ( window_get_flag ( WINDOW_FLAG_ALWAYS_ON_TOP , p_window ) ) {
_set_wm_maximized ( p_window , true ) ;
}
2020-07-31 17:27:16 +02:00
2020-03-01 23:14:37 +01:00
wd . fullscreen = true ;
2022-12-07 08:54:02 +01:00
if ( p_mode = = WINDOW_MODE_EXCLUSIVE_FULLSCREEN ) {
wd . exclusive_fullscreen = true ;
_set_wm_fullscreen ( p_window , true , true ) ;
} else {
wd . exclusive_fullscreen = false ;
_set_wm_fullscreen ( p_window , true , false ) ;
}
2020-03-01 23:14:37 +01:00
} break ;
case WINDOW_MODE_MAXIMIZED : {
_set_wm_maximized ( p_window , true ) ;
} break ;
}
}
DisplayServer : : WindowMode DisplayServerX11 : : window_get_mode ( WindowID p_window ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , WINDOW_MODE_WINDOWED ) ;
2020-03-01 23:14:37 +01:00
const WindowData & wd = windows [ p_window ] ;
if ( wd . fullscreen ) { //if fullscreen, it's not in another mode
2022-12-07 08:54:02 +01:00
if ( wd . exclusive_fullscreen ) {
return WINDOW_MODE_EXCLUSIVE_FULLSCREEN ;
} else {
return WINDOW_MODE_FULLSCREEN ;
}
2020-03-01 23:14:37 +01:00
}
2020-06-29 13:29:31 +02:00
// Test maximized.
// Using EWMH -- Extended Window Manager Hints
if ( _window_maximize_check ( p_window , " _NET_WM_STATE " ) ) {
return WINDOW_MODE_MAXIMIZED ;
2020-03-01 23:14:37 +01:00
}
2022-08-30 20:52:24 +02:00
{
if ( _window_minimize_check ( p_window ) ) {
return WINDOW_MODE_MINIMIZED ;
2020-03-01 23:14:37 +01:00
}
}
2020-06-29 13:29:31 +02:00
// All other discarded, return windowed.
2020-03-01 23:14:37 +01:00
return WINDOW_MODE_WINDOWED ;
}
void DisplayServerX11 : : window_set_flag ( WindowFlags p_flag , bool p_enabled , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
switch ( p_flag ) {
case WINDOW_FLAG_RESIZE_DISABLED : {
wd . resize_disabled = p_enabled ;
2020-07-31 17:27:16 +02:00
_update_size_hints ( p_window ) ;
2020-03-01 23:14:37 +01:00
2020-07-31 17:27:16 +02:00
XFlush ( x11_display ) ;
2020-03-01 23:14:37 +01:00
} break ;
case WINDOW_FLAG_BORDERLESS : {
Hints hints ;
Atom property ;
hints . flags = 2 ;
hints . decorations = p_enabled ? 0 : 1 ;
property = XInternAtom ( x11_display , " _MOTIF_WM_HINTS " , True ) ;
2020-11-14 21:31:16 +01:00
if ( property ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , property , property , 32 , PropModeReplace , ( unsigned char * ) & hints , 5 ) ;
}
2020-03-01 23:14:37 +01:00
// Preserve window size
window_set_size ( window_get_size ( p_window ) , p_window ) ;
wd . borderless = p_enabled ;
2023-01-15 11:05:25 +01:00
_update_window_mouse_passthrough ( p_window ) ;
2020-03-01 23:14:37 +01:00
} break ;
case WINDOW_FLAG_ALWAYS_ON_TOP : {
2020-03-04 20:03:30 +01:00
ERR_FAIL_COND_MSG ( wd . transient_parent ! = INVALID_WINDOW_ID , " Can't make a window transient if the 'on top' flag is active. " ) ;
2020-03-01 23:14:37 +01:00
if ( p_enabled & & wd . fullscreen ) {
_set_wm_maximized ( p_window , true ) ;
}
Atom wm_state = XInternAtom ( x11_display , " _NET_WM_STATE " , False ) ;
Atom wm_above = XInternAtom ( x11_display , " _NET_WM_STATE_ABOVE " , False ) ;
XClientMessageEvent xev ;
memset ( & xev , 0 , sizeof ( xev ) ) ;
xev . type = ClientMessage ;
xev . window = wd . x11_window ;
xev . message_type = wm_state ;
xev . format = 32 ;
xev . data . l [ 0 ] = p_enabled ? _NET_WM_STATE_ADD : _NET_WM_STATE_REMOVE ;
xev . data . l [ 1 ] = wm_above ;
xev . data . l [ 3 ] = 1 ;
XSendEvent ( x11_display , DefaultRootWindow ( x11_display ) , False , SubstructureRedirectMask | SubstructureNotifyMask , ( XEvent * ) & xev ) ;
if ( ! p_enabled & & ! wd . fullscreen ) {
_set_wm_maximized ( p_window , false ) ;
}
wd . on_top = p_enabled ;
} break ;
case WINDOW_FLAG_TRANSPARENT : {
2022-09-03 12:40:53 +02:00
wd . layered_window = p_enabled ;
2020-03-01 23:14:37 +01:00
} break ;
2022-02-24 10:21:23 +01:00
case WINDOW_FLAG_NO_FOCUS : {
wd . no_focus = p_enabled ;
} break ;
2023-01-15 11:05:25 +01:00
case WINDOW_FLAG_MOUSE_PASSTHROUGH : {
wd . mpass = p_enabled ;
_update_window_mouse_passthrough ( p_window ) ;
} break ;
2022-02-24 10:21:23 +01:00
case WINDOW_FLAG_POPUP : {
XWindowAttributes xwa ;
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd . x11_window , & xwa ) ;
ERR_FAIL_COND_MSG ( p_window = = MAIN_WINDOW_ID , " Main window can't be popup. " ) ;
2022-03-08 10:46:44 +01:00
ERR_FAIL_COND_MSG ( ( xwa . map_state = = IsViewable ) & & ( wd . is_popup ! = p_enabled ) , " Popup flag can't changed while window is opened. " ) ;
2022-02-24 10:21:23 +01:00
wd . is_popup = p_enabled ;
} break ;
2020-03-01 23:14:37 +01:00
default : {
}
}
}
2020-05-14 14:29:06 +02:00
2020-03-01 23:14:37 +01:00
bool DisplayServerX11 : : window_get_flag ( WindowFlags p_flag , WindowID p_window ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , false ) ;
2020-03-01 23:14:37 +01:00
const WindowData & wd = windows [ p_window ] ;
switch ( p_flag ) {
case WINDOW_FLAG_RESIZE_DISABLED : {
return wd . resize_disabled ;
} break ;
case WINDOW_FLAG_BORDERLESS : {
bool borderless = wd . borderless ;
Atom prop = XInternAtom ( x11_display , " _MOTIF_WM_HINTS " , True ) ;
if ( prop ! = None ) {
Atom type ;
int format ;
unsigned long len ;
unsigned long remaining ;
2020-04-02 01:20:12 +02:00
unsigned char * data = nullptr ;
2020-03-01 23:14:37 +01:00
if ( XGetWindowProperty ( x11_display , wd . x11_window , prop , 0 , sizeof ( Hints ) , False , AnyPropertyType , & type , & format , & len , & remaining , & data ) = = Success ) {
if ( data & & ( format = = 32 ) & & ( len > = 5 ) ) {
2022-04-07 12:23:40 +02:00
borderless = ! ( reinterpret_cast < Hints * > ( data ) - > decorations ) ;
2020-03-01 23:14:37 +01:00
}
2020-03-06 18:00:16 +01:00
if ( data ) {
XFree ( data ) ;
}
2020-03-01 23:14:37 +01:00
}
}
return borderless ;
} break ;
case WINDOW_FLAG_ALWAYS_ON_TOP : {
return wd . on_top ;
} break ;
case WINDOW_FLAG_TRANSPARENT : {
2022-09-03 12:40:53 +02:00
return wd . layered_window ;
2020-03-01 23:14:37 +01:00
} break ;
2022-02-24 10:21:23 +01:00
case WINDOW_FLAG_NO_FOCUS : {
return wd . no_focus ;
} break ;
2023-01-15 11:05:25 +01:00
case WINDOW_FLAG_MOUSE_PASSTHROUGH : {
return wd . mpass ;
} break ;
2022-02-24 10:21:23 +01:00
case WINDOW_FLAG_POPUP : {
return wd . is_popup ;
} break ;
2020-03-01 23:14:37 +01:00
default : {
}
}
return false ;
}
void DisplayServerX11 : : window_request_attention ( WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2022-04-07 12:23:40 +02:00
const WindowData & wd = windows [ p_window ] ;
2020-03-01 23:14:37 +01:00
// Using EWMH -- Extended Window Manager Hints
//
// Sets the _NET_WM_STATE_DEMANDS_ATTENTION atom for WM_STATE
// Will be unset by the window manager after user react on the request for attention
XEvent xev ;
Atom wm_state = XInternAtom ( x11_display , " _NET_WM_STATE " , False ) ;
Atom wm_attention = XInternAtom ( x11_display , " _NET_WM_STATE_DEMANDS_ATTENTION " , False ) ;
memset ( & xev , 0 , sizeof ( xev ) ) ;
xev . type = ClientMessage ;
xev . xclient . window = wd . x11_window ;
xev . xclient . message_type = wm_state ;
xev . xclient . format = 32 ;
xev . xclient . data . l [ 0 ] = _NET_WM_STATE_ADD ;
xev . xclient . data . l [ 1 ] = wm_attention ;
XSendEvent ( x11_display , DefaultRootWindow ( x11_display ) , False , SubstructureRedirectMask | SubstructureNotifyMask , & xev ) ;
XFlush ( x11_display ) ;
}
void DisplayServerX11 : : window_move_to_foreground ( WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2022-04-07 12:23:40 +02:00
const WindowData & wd = windows [ p_window ] ;
2020-03-01 23:14:37 +01:00
XEvent xev ;
Atom net_active_window = XInternAtom ( x11_display , " _NET_ACTIVE_WINDOW " , False ) ;
memset ( & xev , 0 , sizeof ( xev ) ) ;
xev . type = ClientMessage ;
xev . xclient . window = wd . x11_window ;
xev . xclient . message_type = net_active_window ;
xev . xclient . format = 32 ;
xev . xclient . data . l [ 0 ] = 1 ;
xev . xclient . data . l [ 1 ] = CurrentTime ;
XSendEvent ( x11_display , DefaultRootWindow ( x11_display ) , False , SubstructureRedirectMask | SubstructureNotifyMask , & xev ) ;
XFlush ( x11_display ) ;
}
2023-12-22 17:50:21 +01:00
DisplayServerX11 : : WindowID DisplayServerX11 : : get_focused_window ( ) const {
return last_focused_window ;
}
2023-06-15 09:53:31 +02:00
bool DisplayServerX11 : : window_is_focused ( WindowID p_window ) const {
_THREAD_SAFE_METHOD_
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , false ) ;
2023-12-22 17:50:21 +01:00
2023-06-15 09:53:31 +02:00
const WindowData & wd = windows [ p_window ] ;
return wd . focused ;
}
2020-03-01 23:14:37 +01:00
bool DisplayServerX11 : : window_can_draw ( WindowID p_window ) const {
//this seems to be all that is provided by X11
return window_get_mode ( p_window ) ! = WINDOW_MODE_MINIMIZED ;
}
2020-05-14 14:29:06 +02:00
2020-03-03 14:36:29 +01:00
bool DisplayServerX11 : : can_any_window_draw ( ) const {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
if ( window_get_mode ( E . key ) ! = WINDOW_MODE_MINIMIZED ) {
2020-03-03 14:36:29 +01:00
return true ;
}
}
return false ;
}
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : window_set_ime_active ( const bool p_active , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
2020-05-14 16:41:43 +02:00
if ( ! wd . xic ) {
2020-03-01 23:14:37 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2022-12-11 00:21:22 +01:00
if ( ! wd . focused ) {
2023-01-25 09:21:27 +01:00
wd . ime_active = false ;
im_text = String ( ) ;
im_selection = Vector2i ( ) ;
2022-12-11 00:21:22 +01:00
return ;
}
2020-03-01 23:14:37 +01:00
2020-09-09 12:43:20 +02:00
// Block events polling while changing input focus
// because it triggers some event polling internally.
2020-03-01 23:14:37 +01:00
if ( p_active ) {
2022-12-11 00:21:22 +01:00
MutexLock mutex_lock ( events_mutex ) ;
wd . ime_active = true ;
XMapWindow ( x11_display , wd . x11_xim_window ) ;
XWindowAttributes xwa ;
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd . x11_xim_window , & xwa ) ;
2024-06-28 00:27:21 +02:00
if ( xwa . map_state = = IsViewable ) {
2023-12-31 19:53:27 +01:00
_set_input_focus ( wd . x11_xim_window , RevertToParent ) ;
2020-09-09 12:43:20 +02:00
}
2022-12-11 00:21:22 +01:00
XSetICFocus ( wd . xic ) ;
2020-03-01 23:14:37 +01:00
} else {
2020-09-09 12:43:20 +02:00
MutexLock mutex_lock ( events_mutex ) ;
2020-03-01 23:14:37 +01:00
XUnsetICFocus ( wd . xic ) ;
2022-12-11 00:21:22 +01:00
XUnmapWindow ( x11_display , wd . x11_xim_window ) ;
wd . ime_active = false ;
im_text = String ( ) ;
im_selection = Vector2i ( ) ;
2020-03-01 23:14:37 +01:00
}
}
2020-05-14 14:29:06 +02:00
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : window_set_ime_position ( const Point2i & p_pos , WindowID p_window ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ p_window ] ;
2020-05-14 16:41:43 +02:00
if ( ! wd . xic ) {
2020-03-01 23:14:37 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2022-12-11 00:21:22 +01:00
if ( ! wd . focused ) {
return ;
}
2020-03-01 23:14:37 +01:00
2022-12-11 00:21:22 +01:00
if ( wd . ime_active ) {
XWindowAttributes xwa ;
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd . x11_xim_window , & xwa ) ;
if ( xwa . map_state = = IsViewable ) {
XMoveWindow ( x11_display , wd . x11_xim_window , p_pos . x , p_pos . y ) ;
}
2020-09-09 12:43:20 +02:00
}
2022-12-11 00:21:22 +01:00
}
Point2i DisplayServerX11 : : ime_get_selection ( ) const {
return im_selection ;
}
2020-09-09 12:43:20 +02:00
2022-12-11 00:21:22 +01:00
String DisplayServerX11 : : ime_get_text ( ) const {
return im_text ;
2020-03-01 23:14:37 +01:00
}
void DisplayServerX11 : : cursor_set_shape ( CursorShape p_shape ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-01 23:14:37 +01:00
ERR_FAIL_INDEX ( p_shape , CURSOR_MAX ) ;
if ( p_shape = = current_cursor ) {
return ;
}
if ( mouse_mode = = MOUSE_MODE_VISIBLE | | mouse_mode = = MOUSE_MODE_CONFINED ) {
if ( cursors [ p_shape ] ! = None ) {
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
XDefineCursor ( x11_display , E . value . x11_window , cursors [ p_shape ] ) ;
2020-03-01 23:14:37 +01:00
}
} else if ( cursors [ CURSOR_ARROW ] ! = None ) {
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
XDefineCursor ( x11_display , E . value . x11_window , cursors [ CURSOR_ARROW ] ) ;
2020-03-01 23:14:37 +01:00
}
}
}
current_cursor = p_shape ;
}
2020-05-14 14:29:06 +02:00
2020-03-01 23:14:37 +01:00
DisplayServerX11 : : CursorShape DisplayServerX11 : : cursor_get_shape ( ) const {
return current_cursor ;
}
2020-05-14 14:29:06 +02:00
2022-05-03 01:43:50 +02:00
void DisplayServerX11 : : cursor_set_custom_image ( const Ref < Resource > & p_cursor , CursorShape p_shape , const Vector2 & p_hotspot ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2023-02-17 14:17:37 +01:00
ERR_FAIL_INDEX ( p_shape , CURSOR_MAX ) ;
2020-03-01 23:14:37 +01:00
if ( p_cursor . is_valid ( ) ) {
2022-05-13 15:04:37 +02:00
HashMap < CursorShape , Vector < Variant > > : : Iterator cursor_c = cursors_cache . find ( p_shape ) ;
2020-03-01 23:14:37 +01:00
if ( cursor_c ) {
2022-05-13 15:04:37 +02:00
if ( cursor_c - > value [ 0 ] = = p_cursor & & cursor_c - > value [ 1 ] = = p_hotspot ) {
2020-03-01 23:14:37 +01:00
cursor_set_shape ( p_shape ) ;
return ;
}
cursors_cache . erase ( p_shape ) ;
}
2024-06-02 22:05:21 +02:00
Ref < Image > image = _get_cursor_image_from_resource ( p_cursor , p_hotspot ) ;
2024-02-28 18:38:15 +01:00
ERR_FAIL_COND ( image . is_null ( ) ) ;
Vector2i texture_size = image - > get_size ( ) ;
2020-03-01 23:14:37 +01:00
// Create the cursor structure
XcursorImage * cursor_image = XcursorImageCreate ( texture_size . width , texture_size . height ) ;
XcursorUInt image_size = texture_size . width * texture_size . height ;
XcursorDim size = sizeof ( XcursorPixel ) * image_size ;
cursor_image - > version = 1 ;
cursor_image - > size = size ;
cursor_image - > xhot = p_hotspot . x ;
cursor_image - > yhot = p_hotspot . y ;
// allocate memory to contain the whole file
cursor_image - > pixels = ( XcursorPixel * ) memalloc ( size ) ;
for ( XcursorPixel index = 0 ; index < image_size ; index + + ) {
2024-06-02 22:05:21 +02:00
int row_index = floor ( index / texture_size . width ) ;
int column_index = index % int ( texture_size . width ) ;
2020-03-01 23:14:37 +01:00
* ( cursor_image - > pixels + index ) = image - > get_pixel ( column_index , row_index ) . to_argb32 ( ) ;
}
2023-09-28 11:40:18 +02:00
ERR_FAIL_NULL ( cursor_image - > pixels ) ;
2020-03-01 23:14:37 +01:00
// Save it for a further usage
cursors [ p_shape ] = XcursorImageLoadCursor ( x11_display , cursor_image ) ;
Vector < Variant > params ;
params . push_back ( p_cursor ) ;
params . push_back ( p_hotspot ) ;
cursors_cache . insert ( p_shape , params ) ;
if ( p_shape = = current_cursor ) {
if ( mouse_mode = = MOUSE_MODE_VISIBLE | | mouse_mode = = MOUSE_MODE_CONFINED ) {
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
XDefineCursor ( x11_display , E . value . x11_window , cursors [ p_shape ] ) ;
2020-03-01 23:14:37 +01:00
}
}
}
memfree ( cursor_image - > pixels ) ;
XcursorImageDestroy ( cursor_image ) ;
} else {
// Reset to default system cursor
2023-05-11 12:32:23 +02:00
if ( cursor_img [ p_shape ] ) {
cursors [ p_shape ] = XcursorImageLoadCursor ( x11_display , cursor_img [ p_shape ] ) ;
2020-03-01 23:14:37 +01:00
}
2023-03-06 22:00:34 +01:00
cursors_cache . erase ( p_shape ) ;
2020-03-01 23:14:37 +01:00
CursorShape c = current_cursor ;
current_cursor = CURSOR_MAX ;
cursor_set_shape ( c ) ;
}
}
2020-06-11 11:27:07 +02:00
int DisplayServerX11 : : keyboard_get_layout_count ( ) const {
int _group_count = 0 ;
XkbDescRec * kbd = XkbAllocKeyboard ( ) ;
if ( kbd ) {
kbd - > dpy = x11_display ;
XkbGetControls ( x11_display , XkbAllControlsMask , kbd ) ;
XkbGetNames ( x11_display , XkbSymbolsNameMask , kbd ) ;
const Atom * groups = kbd - > names - > groups ;
2021-04-05 14:02:50 +02:00
if ( kbd - > ctrls ! = nullptr ) {
2020-06-11 11:27:07 +02:00
_group_count = kbd - > ctrls - > num_groups ;
} else {
while ( _group_count < XkbNumKbdGroups & & groups [ _group_count ] ! = None ) {
_group_count + + ;
}
}
XkbFreeKeyboard ( kbd , 0 , true ) ;
}
return _group_count ;
}
2020-03-01 23:14:37 +01:00
2020-06-11 11:27:07 +02:00
int DisplayServerX11 : : keyboard_get_current_layout ( ) const {
XkbStateRec state ;
XkbGetState ( x11_display , XkbUseCoreKbd , & state ) ;
return state . group ;
}
2020-03-01 23:14:37 +01:00
2020-06-11 11:27:07 +02:00
void DisplayServerX11 : : keyboard_set_current_layout ( int p_index ) {
ERR_FAIL_INDEX ( p_index , keyboard_get_layout_count ( ) ) ;
XkbLockGroup ( x11_display , XkbUseCoreKbd , p_index ) ;
}
2020-03-01 23:14:37 +01:00
2020-06-11 11:27:07 +02:00
String DisplayServerX11 : : keyboard_get_layout_language ( int p_index ) const {
String ret ;
XkbDescRec * kbd = XkbAllocKeyboard ( ) ;
if ( kbd ) {
kbd - > dpy = x11_display ;
XkbGetControls ( x11_display , XkbAllControlsMask , kbd ) ;
XkbGetNames ( x11_display , XkbSymbolsNameMask , kbd ) ;
XkbGetNames ( x11_display , XkbGroupNamesMask , kbd ) ;
int _group_count = 0 ;
const Atom * groups = kbd - > names - > groups ;
2021-04-05 14:02:50 +02:00
if ( kbd - > ctrls ! = nullptr ) {
2020-06-11 11:27:07 +02:00
_group_count = kbd - > ctrls - > num_groups ;
} else {
while ( _group_count < XkbNumKbdGroups & & groups [ _group_count ] ! = None ) {
_group_count + + ;
}
}
2020-03-01 23:14:37 +01:00
2020-06-11 11:27:07 +02:00
Atom names = kbd - > names - > symbols ;
if ( names ! = None ) {
2022-03-04 10:50:24 +01:00
Vector < String > info = get_atom_name ( x11_display , names ) . split ( " + " ) ;
2020-06-11 11:27:07 +02:00
if ( p_index > = 0 & & p_index < _group_count ) {
if ( p_index + 1 < info . size ( ) ) {
ret = info [ p_index + 1 ] ; // Skip "pc" at the start and "inet"/"group" at the end of symbols.
} else {
ret = " en " ; // No symbol for layout fallback to "en".
}
} else {
ERR_PRINT ( " Index " + itos ( p_index ) + " is out of bounds ( " + itos ( _group_count ) + " ). " ) ;
}
}
XkbFreeKeyboard ( kbd , 0 , true ) ;
2020-03-01 23:14:37 +01:00
}
2020-06-11 11:27:07 +02:00
return ret . substr ( 0 , 2 ) ;
}
String DisplayServerX11 : : keyboard_get_layout_name ( int p_index ) const {
String ret ;
XkbDescRec * kbd = XkbAllocKeyboard ( ) ;
if ( kbd ) {
kbd - > dpy = x11_display ;
XkbGetControls ( x11_display , XkbAllControlsMask , kbd ) ;
XkbGetNames ( x11_display , XkbSymbolsNameMask , kbd ) ;
XkbGetNames ( x11_display , XkbGroupNamesMask , kbd ) ;
int _group_count = 0 ;
const Atom * groups = kbd - > names - > groups ;
2021-04-05 14:02:50 +02:00
if ( kbd - > ctrls ! = nullptr ) {
2020-06-11 11:27:07 +02:00
_group_count = kbd - > ctrls - > num_groups ;
} else {
while ( _group_count < XkbNumKbdGroups & & groups [ _group_count ] ! = None ) {
_group_count + + ;
}
}
2020-03-01 23:14:37 +01:00
2020-06-11 11:27:07 +02:00
if ( p_index > = 0 & & p_index < _group_count ) {
2022-03-04 10:50:24 +01:00
ret = get_atom_name ( x11_display , groups [ p_index ] ) ;
2020-06-11 11:27:07 +02:00
} else {
ERR_PRINT ( " Index " + itos ( p_index ) + " is out of bounds ( " + itos ( _group_count ) + " ). " ) ;
}
XkbFreeKeyboard ( kbd , 0 , true ) ;
}
return ret ;
2020-03-01 23:14:37 +01:00
}
2021-09-17 03:07:59 +02:00
Key DisplayServerX11 : : keyboard_get_keycode_from_physical ( Key p_keycode ) const {
2021-08-13 23:31:57 +02:00
Key modifiers = p_keycode & KeyModifierMask : : MODIFIER_MASK ;
Key keycode_no_mod = p_keycode & KeyModifierMask : : CODE_MASK ;
unsigned int xkeycode = KeyMappingX11 : : get_xlibcode ( keycode_no_mod ) ;
2023-03-29 15:18:31 +02:00
KeySym xkeysym = XkbKeycodeToKeysym ( x11_display , xkeycode , keyboard_get_current_layout ( ) , 0 ) ;
2022-02-04 09:32:20 +01:00
if ( is_ascii_lower_case ( xkeysym ) ) {
2021-09-17 03:07:59 +02:00
xkeysym - = ( ' a ' - ' A ' ) ;
}
Key key = KeyMappingX11 : : get_keycode ( xkeysym ) ;
// If not found, fallback to QWERTY.
// This should match the behavior of the event pump
2021-08-13 23:31:57 +02:00
if ( key = = Key : : NONE ) {
2021-09-17 03:07:59 +02:00
return p_keycode ;
}
return ( Key ) ( key | modifiers ) ;
}
2023-06-08 08:26:48 +02:00
Key DisplayServerX11 : : keyboard_get_label_from_physical ( Key p_keycode ) const {
Key modifiers = p_keycode & KeyModifierMask : : MODIFIER_MASK ;
Key keycode_no_mod = p_keycode & KeyModifierMask : : CODE_MASK ;
unsigned int xkeycode = KeyMappingX11 : : get_xlibcode ( keycode_no_mod ) ;
KeySym xkeysym = XkbKeycodeToKeysym ( x11_display , xkeycode , keyboard_get_current_layout ( ) , 0 ) ;
if ( is_ascii_lower_case ( xkeysym ) ) {
xkeysym - = ( ' a ' - ' A ' ) ;
}
Key key = KeyMappingX11 : : get_keycode ( xkeysym ) ;
# ifdef XKB_ENABLED
if ( xkb_loaded_v08p ) {
String keysym = String : : chr ( xkb_keysym_to_utf32 ( xkb_keysym_to_upper ( xkeysym ) ) ) ;
key = fix_key_label ( keysym [ 0 ] , KeyMappingX11 : : get_keycode ( xkeysym ) ) ;
}
# endif
// If not found, fallback to QWERTY.
// This should match the behavior of the event pump
if ( key = = Key : : NONE ) {
return p_keycode ;
}
return ( Key ) ( key | modifiers ) ;
}
2020-03-01 23:14:37 +01:00
DisplayServerX11 : : Property DisplayServerX11 : : _read_property ( Display * p_display , Window p_window , Atom p_property ) {
2020-11-12 14:38:54 +01:00
Atom actual_type = None ;
int actual_format = 0 ;
unsigned long nitems = 0 ;
unsigned long bytes_after = 0 ;
2020-04-02 01:20:12 +02:00
unsigned char * ret = nullptr ;
2020-03-01 23:14:37 +01:00
2020-11-12 14:38:54 +01:00
// Keep trying to read the property until there are no bytes unread.
if ( p_property ! = None ) {
2022-04-07 12:23:40 +02:00
int read_bytes = 1024 ;
2020-11-12 14:38:54 +01:00
do {
if ( ret ! = nullptr ) {
XFree ( ret ) ;
}
2020-03-01 23:14:37 +01:00
2020-11-12 14:38:54 +01:00
XGetWindowProperty ( p_display , p_window , p_property , 0 , read_bytes , False , AnyPropertyType ,
& actual_type , & actual_format , & nitems , & bytes_after ,
& ret ) ;
2020-03-01 23:14:37 +01:00
2020-11-12 14:38:54 +01:00
read_bytes * = 2 ;
2020-03-01 23:14:37 +01:00
2020-11-12 14:38:54 +01:00
} while ( bytes_after ! = 0 ) ;
}
2020-03-01 23:14:37 +01:00
Property p = { ret , actual_format , ( int ) nitems , actual_type } ;
return p ;
}
2022-04-07 12:23:40 +02:00
static Atom pick_target_from_list ( Display * p_display , const Atom * p_list , int p_count ) {
2020-03-01 23:14:37 +01:00
static const char * target_type = " text/uri-list " ;
for ( int i = 0 ; i < p_count ; i + + ) {
Atom atom = p_list [ i ] ;
2022-03-04 10:50:24 +01:00
if ( atom ! = None & & get_atom_name ( p_display , atom ) = = target_type ) {
2020-03-01 23:14:37 +01:00
return atom ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
}
return None ;
}
static Atom pick_target_from_atoms ( Display * p_disp , Atom p_t1 , Atom p_t2 , Atom p_t3 ) {
static const char * target_type = " text/uri-list " ;
2022-03-04 10:50:24 +01:00
if ( p_t1 ! = None & & get_atom_name ( p_disp , p_t1 ) = = target_type ) {
2020-03-01 23:14:37 +01:00
return p_t1 ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
2022-03-04 10:50:24 +01:00
if ( p_t2 ! = None & & get_atom_name ( p_disp , p_t2 ) = = target_type ) {
2020-03-01 23:14:37 +01:00
return p_t2 ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
2022-03-04 10:50:24 +01:00
if ( p_t3 ! = None & & get_atom_name ( p_disp , p_t3 ) = = target_type ) {
2020-03-01 23:14:37 +01:00
return p_t3 ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
return None ;
}
void DisplayServerX11 : : _get_key_modifier_state ( unsigned int p_x11_state , Ref < InputEventWithModifiers > state ) {
2021-04-24 22:33:50 +02:00
state - > set_shift_pressed ( ( p_x11_state & ShiftMask ) ) ;
state - > set_ctrl_pressed ( ( p_x11_state & ControlMask ) ) ;
state - > set_alt_pressed ( ( p_x11_state & Mod1Mask /*|| p_x11_state&Mod5Mask*/ ) ) ; //altgr should not count as alt
state - > set_meta_pressed ( ( p_x11_state & Mod4Mask ) ) ;
2020-03-01 23:14:37 +01:00
}
2020-09-09 12:43:20 +02:00
void DisplayServerX11 : : _handle_key_event ( WindowID p_window , XKeyEvent * p_event , LocalVector < XEvent > & p_events , uint32_t & p_event_index , bool p_echo ) {
2023-03-07 09:05:11 +01:00
WindowData & wd = windows [ p_window ] ;
2020-03-01 23:14:37 +01:00
// X11 functions don't know what const is
XKeyEvent * xkeyevent = p_event ;
2022-12-11 00:21:22 +01:00
if ( wd . ime_in_progress ) {
return ;
}
if ( wd . ime_suppress_next_keyup ) {
wd . ime_suppress_next_keyup = false ;
if ( xkeyevent - > type ! = KeyPress ) {
return ;
}
}
2020-03-01 23:14:37 +01:00
// This code was pretty difficult to write.
// The docs stink and every toolkit seems to
// do it in a different way.
/* Phase 1, obtain a proper keysym */
// This was also very difficult to figure out.
// You'd expect you could just use Keysym provided by
// XKeycodeToKeysym to obtain internationalized
// input.. WRONG!!
// you must use XLookupString (???) which not only wastes
// cycles generating an unnecessary string, but also
// still works in half the cases. (won't handle deadkeys)
// For more complex input methods (deadkeys and more advanced)
// you have to use XmbLookupString (??).
Fix various typos with codespell
Found via `codespell -q 3 -S ./thirdparty,*.po,./DONORS.md -L ackward,ang,ans,ba,beng,cas,childs,childrens,dof,doubleclick,fave,findn,hist,inout,leapyear,lod,nd,numer,ois,ony,paket,seeked,sinc,switchs,te,uint`
2021-07-07 17:17:32 +02:00
// So then you have to choose which of both results
2020-03-01 23:14:37 +01:00
// you want to keep.
// This is a real bizarreness and cpu waster.
KeySym keysym_keycode = 0 ; // keysym used to find a keycode
KeySym keysym_unicode = 0 ; // keysym used to find unicode
// XLookupString returns keysyms usable as nice keycodes.
2022-12-11 00:21:22 +01:00
char str [ 256 ] = { } ;
2020-03-01 23:14:37 +01:00
XKeyEvent xkeyevent_no_mod = * xkeyevent ;
xkeyevent_no_mod . state & = ~ ShiftMask ;
xkeyevent_no_mod . state & = ~ ControlMask ;
2022-12-11 00:21:22 +01:00
XLookupString ( xkeyevent , str , 255 , & keysym_unicode , nullptr ) ;
2023-01-25 12:04:02 +01:00
XLookupString ( & xkeyevent_no_mod , nullptr , 0 , & keysym_keycode , nullptr ) ;
2022-12-11 00:21:22 +01:00
String keysym ;
2023-02-15 13:13:56 +01:00
# ifdef XKB_ENABLED
2023-03-16 09:44:47 +01:00
if ( xkb_loaded_v08p ) {
2022-12-11 00:21:22 +01:00
KeySym keysym_unicode_nm = 0 ; // keysym used to find unicode
XLookupString ( & xkeyevent_no_mod , nullptr , 0 , & keysym_unicode_nm , nullptr ) ;
keysym = String : : chr ( xkb_keysym_to_utf32 ( xkb_keysym_to_upper ( keysym_unicode_nm ) ) ) ;
}
2023-02-15 13:13:56 +01:00
# endif
2020-03-01 23:14:37 +01:00
// Meanwhile, XLookupString returns keysyms useful for unicode.
if ( ! xmbstring ) {
// keep a temporary buffer for the string
xmbstring = ( char * ) memalloc ( sizeof ( char ) * 8 ) ;
xmblen = 8 ;
}
if ( xkeyevent - > type = = KeyPress & & wd . xic ) {
Status status ;
# ifdef X_HAVE_UTF8_STRING
int utf8len = 8 ;
char * utf8string = ( char * ) memalloc ( sizeof ( char ) * utf8len ) ;
int utf8bytes = Xutf8LookupString ( wd . xic , xkeyevent , utf8string ,
utf8len - 1 , & keysym_unicode , & status ) ;
if ( status = = XBufferOverflow ) {
utf8len = utf8bytes + 1 ;
utf8string = ( char * ) memrealloc ( utf8string , utf8len ) ;
utf8bytes = Xutf8LookupString ( wd . xic , xkeyevent , utf8string ,
utf8len - 1 , & keysym_unicode , & status ) ;
}
utf8string [ utf8bytes ] = ' \0 ' ;
if ( status = = XLookupChars ) {
bool keypress = xkeyevent - > type = = KeyPress ;
2021-06-20 19:12:33 +02:00
Key keycode = KeyMappingX11 : : get_keycode ( keysym_keycode ) ;
2021-08-13 23:31:57 +02:00
Key physical_keycode = KeyMappingX11 : : get_scancode ( xkeyevent - > keycode ) ;
2020-03-01 23:14:37 +01:00
2021-08-13 23:31:57 +02:00
if ( keycode > = Key : : A + 32 & & keycode < = Key : : Z + 32 ) {
2020-03-01 23:14:37 +01:00
keycode - = ' a ' - ' A ' ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
String tmp ;
tmp . parse_utf8 ( utf8string , utf8bytes ) ;
for ( int i = 0 ; i < tmp . length ( ) ; i + + ) {
Ref < InputEventKey > k ;
2021-06-18 00:03:09 +02:00
k . instantiate ( ) ;
2021-08-13 23:31:57 +02:00
if ( physical_keycode = = Key : : NONE & & keycode = = Key : : NONE & & tmp [ i ] = = 0 ) {
2020-03-01 23:14:37 +01:00
continue ;
}
2021-08-13 23:31:57 +02:00
if ( keycode = = Key : : NONE ) {
2021-06-20 19:12:33 +02:00
keycode = ( Key ) physical_keycode ;
2020-03-01 23:14:37 +01:00
}
_get_key_modifier_state ( xkeyevent - > state , k ) ;
2020-03-06 18:00:16 +01:00
k - > set_window_id ( p_window ) ;
2020-03-01 23:14:37 +01:00
k - > set_pressed ( keypress ) ;
k - > set_keycode ( keycode ) ;
2022-12-11 00:21:22 +01:00
k - > set_physical_keycode ( physical_keycode ) ;
if ( ! keysym . is_empty ( ) ) {
k - > set_key_label ( fix_key_label ( keysym [ 0 ] , keycode ) ) ;
} else {
k - > set_key_label ( keycode ) ;
}
if ( keypress ) {
k - > set_unicode ( fix_unicode ( tmp [ i ] ) ) ;
}
2020-03-01 23:14:37 +01:00
k - > set_echo ( false ) ;
2021-08-13 23:31:57 +02:00
if ( k - > get_keycode ( ) = = Key : : BACKTAB ) {
2020-03-01 23:14:37 +01:00
//make it consistent across platforms.
2021-08-13 23:31:57 +02:00
k - > set_keycode ( Key : : TAB ) ;
k - > set_physical_keycode ( Key : : TAB ) ;
2021-04-24 22:33:50 +02:00
k - > set_shift_pressed ( true ) ;
2020-03-01 23:14:37 +01:00
}
2021-08-13 00:31:16 +02:00
Input : : get_singleton ( ) - > parse_input_event ( k ) ;
2020-03-01 23:14:37 +01:00
}
memfree ( utf8string ) ;
return ;
}
memfree ( utf8string ) ;
# else
do {
int mnbytes = XmbLookupString ( xic , xkeyevent , xmbstring , xmblen - 1 , & keysym_unicode , & status ) ;
xmbstring [ mnbytes ] = ' \0 ' ;
if ( status = = XBufferOverflow ) {
xmblen = mnbytes + 1 ;
xmbstring = ( char * ) memrealloc ( xmbstring , xmblen ) ;
}
} while ( status = = XBufferOverflow ) ;
# endif
2023-02-15 13:13:56 +01:00
# ifdef XKB_ENABLED
2023-03-16 09:44:47 +01:00
} else if ( xkeyevent - > type = = KeyPress & & wd . xkb_state & & xkb_loaded_v05p ) {
2023-01-30 09:59:49 +01:00
xkb_compose_feed_result res = xkb_compose_state_feed ( wd . xkb_state , keysym_unicode ) ;
if ( res = = XKB_COMPOSE_FEED_ACCEPTED ) {
if ( xkb_compose_state_get_status ( wd . xkb_state ) = = XKB_COMPOSE_COMPOSED ) {
bool keypress = xkeyevent - > type = = KeyPress ;
Key keycode = KeyMappingX11 : : get_keycode ( keysym_keycode ) ;
Key physical_keycode = KeyMappingX11 : : get_scancode ( xkeyevent - > keycode ) ;
2023-08-03 15:18:26 +02:00
KeyLocation key_location = KeyMappingX11 : : get_location ( xkeyevent - > keycode ) ;
2023-01-30 09:59:49 +01:00
if ( keycode > = Key : : A + 32 & & keycode < = Key : : Z + 32 ) {
keycode - = ' a ' - ' A ' ;
}
char str_xkb [ 256 ] = { } ;
int str_xkb_size = xkb_compose_state_get_utf8 ( wd . xkb_state , str_xkb , 255 ) ;
String tmp ;
tmp . parse_utf8 ( str_xkb , str_xkb_size ) ;
for ( int i = 0 ; i < tmp . length ( ) ; i + + ) {
Ref < InputEventKey > k ;
k . instantiate ( ) ;
if ( physical_keycode = = Key : : NONE & & keycode = = Key : : NONE & & tmp [ i ] = = 0 ) {
continue ;
}
if ( keycode = = Key : : NONE ) {
keycode = ( Key ) physical_keycode ;
}
_get_key_modifier_state ( xkeyevent - > state , k ) ;
k - > set_window_id ( p_window ) ;
k - > set_pressed ( keypress ) ;
k - > set_keycode ( keycode ) ;
k - > set_physical_keycode ( physical_keycode ) ;
if ( ! keysym . is_empty ( ) ) {
k - > set_key_label ( fix_key_label ( keysym [ 0 ] , keycode ) ) ;
} else {
k - > set_key_label ( keycode ) ;
}
if ( keypress ) {
k - > set_unicode ( fix_unicode ( tmp [ i ] ) ) ;
}
2023-08-03 15:18:26 +02:00
k - > set_location ( key_location ) ;
2023-01-30 09:59:49 +01:00
k - > set_echo ( false ) ;
if ( k - > get_keycode ( ) = = Key : : BACKTAB ) {
//make it consistent across platforms.
k - > set_keycode ( Key : : TAB ) ;
k - > set_physical_keycode ( Key : : TAB ) ;
k - > set_shift_pressed ( true ) ;
}
Input : : get_singleton ( ) - > parse_input_event ( k ) ;
}
return ;
}
}
2023-02-15 13:13:56 +01:00
# endif
2020-03-01 23:14:37 +01:00
}
/* Phase 2, obtain a Godot keycode from the keysym */
// KeyMappingX11 just translated the X11 keysym to a PIGUI
// keysym, so it works in all platforms the same.
2021-06-20 19:12:33 +02:00
Key keycode = KeyMappingX11 : : get_keycode ( keysym_keycode ) ;
2021-08-13 23:31:57 +02:00
Key physical_keycode = KeyMappingX11 : : get_scancode ( xkeyevent - > keycode ) ;
2020-03-01 23:14:37 +01:00
2023-08-03 15:18:26 +02:00
KeyLocation key_location = KeyMappingX11 : : get_location ( xkeyevent - > keycode ) ;
2020-03-01 23:14:37 +01:00
/* Phase 3, obtain a unicode character from the keysym */
// KeyMappingX11 also translates keysym to unicode.
// It does a binary search on a table to translate
// most properly.
2022-12-11 00:21:22 +01:00
char32_t unicode = keysym_unicode > 0 ? KeyMappingX11 : : get_unicode_from_keysym ( keysym_unicode ) : 0 ;
2020-03-01 23:14:37 +01:00
/* Phase 4, determine if event must be filtered */
// This seems to be a side-effect of using XIM.
2020-09-09 12:43:20 +02:00
// XFilterEvent looks like a core X11 function,
2020-03-01 23:14:37 +01:00
// but it's actually just used to see if we must
// ignore a deadkey, or events XIM determines
// must not reach the actual gui.
// Guess it was a design problem of the extension
bool keypress = xkeyevent - > type = = KeyPress ;
2021-08-13 23:31:57 +02:00
if ( physical_keycode = = Key : : NONE & & keycode = = Key : : NONE & & unicode = = 0 ) {
2020-03-01 23:14:37 +01:00
return ;
}
2021-08-13 23:31:57 +02:00
if ( keycode = = Key : : NONE ) {
2021-06-20 19:12:33 +02:00
keycode = ( Key ) physical_keycode ;
2020-03-01 23:14:37 +01:00
}
/* Phase 5, determine modifier mask */
// No problems here, except I had no way to
// know Mod1 was ALT and Mod4 was META (applekey/winkey)
// just tried Mods until i found them.
//print_verbose("mod1: "+itos(xkeyevent->state&Mod1Mask)+" mod 5: "+itos(xkeyevent->state&Mod5Mask));
Ref < InputEventKey > k ;
2021-06-18 00:03:09 +02:00
k . instantiate ( ) ;
2020-03-06 18:00:16 +01:00
k - > set_window_id ( p_window ) ;
2020-03-01 23:14:37 +01:00
_get_key_modifier_state ( xkeyevent - > state , k ) ;
/* Phase 6, determine echo character */
// Echo characters in X11 are a keyrelease and a keypress
// one after the other with the (almot) same timestamp.
2020-09-09 12:43:20 +02:00
// To detect them, i compare to the next event in list and
// check that their difference in time is below a threshold.
2020-03-01 23:14:37 +01:00
if ( xkeyevent - > type ! = KeyPress ) {
p_echo = false ;
// make sure there are events pending,
// so this call won't block.
2020-09-09 12:43:20 +02:00
if ( p_event_index + 1 < p_events . size ( ) ) {
XEvent & peek_event = p_events [ p_event_index + 1 ] ;
2020-03-01 23:14:37 +01:00
// I'm using a threshold of 5 msecs,
// since sometimes there seems to be a little
// jitter. I'm still not convinced that all this approach
// is correct, but the xorg developers are
// not very helpful today.
# define ABSDIFF(x, y) (((x) < (y)) ? ((y) - (x)) : ((x) - (y)))
: : Time threshold = ABSDIFF ( peek_event . xkey . time , xkeyevent - > time ) ;
# undef ABSDIFF
if ( peek_event . type = = KeyPress & & threshold < 5 ) {
KeySym rk ;
2020-04-02 01:20:12 +02:00
XLookupString ( ( XKeyEvent * ) & peek_event , str , 256 , & rk , nullptr ) ;
2020-03-01 23:14:37 +01:00
if ( rk = = keysym_keycode ) {
2020-09-09 12:43:20 +02:00
// Consume to next event.
+ + p_event_index ;
_handle_key_event ( p_window , ( XKeyEvent * ) & peek_event , p_events , p_event_index , true ) ;
2020-03-01 23:14:37 +01:00
return ; //ignore current, echo next
}
}
// use the time from peek_event so it always works
}
// save the time to check for echo when keypress happens
}
/* Phase 7, send event to Window */
k - > set_pressed ( keypress ) ;
2021-08-13 23:31:57 +02:00
if ( keycode > = Key : : A + 32 & & keycode < = Key : : Z + 32 ) {
2021-06-20 19:12:33 +02:00
keycode - = int ( ' a ' - ' A ' ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
k - > set_keycode ( keycode ) ;
2021-06-20 19:12:33 +02:00
k - > set_physical_keycode ( ( Key ) physical_keycode ) ;
2022-12-11 00:21:22 +01:00
if ( ! keysym . is_empty ( ) ) {
k - > set_key_label ( fix_key_label ( keysym [ 0 ] , keycode ) ) ;
} else {
k - > set_key_label ( keycode ) ;
}
if ( keypress ) {
k - > set_unicode ( fix_unicode ( unicode ) ) ;
}
2023-08-03 15:18:26 +02:00
k - > set_location ( key_location ) ;
2020-03-01 23:14:37 +01:00
k - > set_echo ( p_echo ) ;
2021-08-13 23:31:57 +02:00
if ( k - > get_keycode ( ) = = Key : : BACKTAB ) {
2020-03-01 23:14:37 +01:00
//make it consistent across platforms.
2021-08-13 23:31:57 +02:00
k - > set_keycode ( Key : : TAB ) ;
k - > set_physical_keycode ( Key : : TAB ) ;
2021-04-24 22:33:50 +02:00
k - > set_shift_pressed ( true ) ;
2020-03-01 23:14:37 +01:00
}
//don't set mod state if modifier keys are released by themselves
//else event.is_action() will not work correctly here
if ( ! k - > is_pressed ( ) ) {
2021-08-13 23:31:57 +02:00
if ( k - > get_keycode ( ) = = Key : : SHIFT ) {
2021-04-24 22:33:50 +02:00
k - > set_shift_pressed ( false ) ;
2021-08-13 23:31:57 +02:00
} else if ( k - > get_keycode ( ) = = Key : : CTRL ) {
2021-04-24 22:33:50 +02:00
k - > set_ctrl_pressed ( false ) ;
2021-08-13 23:31:57 +02:00
} else if ( k - > get_keycode ( ) = = Key : : ALT ) {
2021-04-24 22:33:50 +02:00
k - > set_alt_pressed ( false ) ;
2021-08-13 23:31:57 +02:00
} else if ( k - > get_keycode ( ) = = Key : : META ) {
2021-04-24 22:33:50 +02:00
k - > set_meta_pressed ( false ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
}
2020-04-28 15:19:37 +02:00
bool last_is_pressed = Input : : get_singleton ( ) - > is_key_pressed ( k - > get_keycode ( ) ) ;
2020-03-01 23:14:37 +01:00
if ( k - > is_pressed ( ) ) {
if ( last_is_pressed ) {
k - > set_echo ( true ) ;
}
}
2021-08-13 00:31:16 +02:00
Input : : get_singleton ( ) - > parse_input_event ( k ) ;
2020-03-01 23:14:37 +01:00
}
2021-10-12 09:43:50 +02:00
Atom DisplayServerX11 : : _process_selection_request_target ( Atom p_target , Window p_requestor , Atom p_property , Atom p_selection ) const {
2020-10-08 17:57:54 +02:00
if ( p_target = = XInternAtom ( x11_display , " TARGETS " , 0 ) ) {
// Request to list all supported targets.
Atom data [ 9 ] ;
2020-09-25 16:40:04 +02:00
data [ 0 ] = XInternAtom ( x11_display , " TARGETS " , 0 ) ;
2020-10-08 17:57:54 +02:00
data [ 1 ] = XInternAtom ( x11_display , " SAVE_TARGETS " , 0 ) ;
data [ 2 ] = XInternAtom ( x11_display , " MULTIPLE " , 0 ) ;
data [ 3 ] = XInternAtom ( x11_display , " UTF8_STRING " , 0 ) ;
data [ 4 ] = XInternAtom ( x11_display , " COMPOUND_TEXT " , 0 ) ;
data [ 5 ] = XInternAtom ( x11_display , " TEXT " , 0 ) ;
data [ 6 ] = XA_STRING ;
data [ 7 ] = XInternAtom ( x11_display , " text/plain;charset=utf-8 " , 0 ) ;
data [ 8 ] = XInternAtom ( x11_display , " text/plain " , 0 ) ;
2020-09-25 16:40:04 +02:00
XChangeProperty ( x11_display ,
2020-10-08 17:57:54 +02:00
p_requestor ,
p_property ,
2020-09-25 16:40:04 +02:00
XA_ATOM ,
32 ,
PropModeReplace ,
( unsigned char * ) & data ,
sizeof ( data ) / sizeof ( data [ 0 ] ) ) ;
2020-10-08 17:57:54 +02:00
return p_property ;
} else if ( p_target = = XInternAtom ( x11_display , " SAVE_TARGETS " , 0 ) ) {
// Request to check if SAVE_TARGETS is supported, nothing special to do.
XChangeProperty ( x11_display ,
p_requestor ,
p_property ,
XInternAtom ( x11_display , " NULL " , False ) ,
32 ,
PropModeReplace ,
nullptr ,
0 ) ;
return p_property ;
} else if ( p_target = = XInternAtom ( x11_display , " UTF8_STRING " , 0 ) | |
2021-10-28 15:19:35 +02:00
p_target = = XInternAtom ( x11_display , " COMPOUND_TEXT " , 0 ) | |
p_target = = XInternAtom ( x11_display , " TEXT " , 0 ) | |
p_target = = XA_STRING | |
p_target = = XInternAtom ( x11_display , " text/plain;charset=utf-8 " , 0 ) | |
p_target = = XInternAtom ( x11_display , " text/plain " , 0 ) ) {
2020-10-08 17:57:54 +02:00
// Directly using internal clipboard because we know our window
// is the owner during a selection request.
2021-10-12 09:43:50 +02:00
CharString clip ;
static const char * target_type = " PRIMARY " ;
2022-03-04 10:50:24 +01:00
if ( p_selection ! = None & & get_atom_name ( x11_display , p_selection ) = = target_type ) {
2021-10-12 09:43:50 +02:00
clip = internal_clipboard_primary . utf8 ( ) ;
} else {
clip = internal_clipboard . utf8 ( ) ;
}
2020-10-08 17:57:54 +02:00
XChangeProperty ( x11_display ,
p_requestor ,
p_property ,
p_target ,
8 ,
PropModeReplace ,
( unsigned char * ) clip . get_data ( ) ,
clip . length ( ) ) ;
return p_property ;
2020-09-25 16:40:04 +02:00
} else {
2020-10-08 17:57:54 +02:00
char * target_name = XGetAtomName ( x11_display , p_target ) ;
2023-03-15 09:40:36 +01:00
print_verbose ( vformat ( " Target '%s' not supported. " , target_name ) ) ;
2020-10-08 17:57:54 +02:00
if ( target_name ) {
XFree ( target_name ) ;
2020-09-25 16:40:04 +02:00
}
2020-10-08 17:57:54 +02:00
return None ;
}
}
void DisplayServerX11 : : _handle_selection_request_event ( XSelectionRequestEvent * p_event ) const {
XEvent respond ;
if ( p_event - > target = = XInternAtom ( x11_display , " MULTIPLE " , 0 ) ) {
// Request for multiple target conversions at once.
Atom atom_pair = XInternAtom ( x11_display , " ATOM_PAIR " , False ) ;
2020-09-25 16:40:04 +02:00
respond . xselection . property = None ;
2020-10-08 17:57:54 +02:00
Atom type ;
int format ;
unsigned long len ;
unsigned long remaining ;
unsigned char * data = nullptr ;
if ( XGetWindowProperty ( x11_display , p_event - > requestor , p_event - > property , 0 , LONG_MAX , False , atom_pair , & type , & format , & len , & remaining , & data ) = = Success ) {
if ( ( len > = 2 ) & & data ) {
Atom * targets = ( Atom * ) data ;
for ( uint64_t i = 0 ; i < len ; i + = 2 ) {
Atom target = targets [ i ] ;
Atom & property = targets [ i + 1 ] ;
2021-10-12 09:43:50 +02:00
property = _process_selection_request_target ( target , p_event - > requestor , property , p_event - > selection ) ;
2020-10-08 17:57:54 +02:00
}
XChangeProperty ( x11_display ,
p_event - > requestor ,
p_event - > property ,
atom_pair ,
32 ,
PropModeReplace ,
( unsigned char * ) targets ,
len ) ;
respond . xselection . property = p_event - > property ;
}
XFree ( data ) ;
}
} else {
// Request for target conversion.
2021-10-12 09:43:50 +02:00
respond . xselection . property = _process_selection_request_target ( p_event - > target , p_event - > requestor , p_event - > property , p_event - > selection ) ;
2020-09-25 16:40:04 +02:00
}
respond . xselection . type = SelectionNotify ;
respond . xselection . display = p_event - > display ;
respond . xselection . requestor = p_event - > requestor ;
respond . xselection . selection = p_event - > selection ;
respond . xselection . target = p_event - > target ;
respond . xselection . time = p_event - > time ;
XSendEvent ( x11_display , p_event - > requestor , True , NoEventMask , & respond ) ;
XFlush ( x11_display ) ;
}
2022-12-11 00:21:22 +01:00
int DisplayServerX11 : : _xim_preedit_start_callback ( : : XIM xim , : : XPointer client_data ,
: : XPointer call_data ) {
DisplayServerX11 * ds = reinterpret_cast < DisplayServerX11 * > ( client_data ) ;
WindowID window_id = ds - > _get_focused_window_or_popup ( ) ;
WindowData & wd = ds - > windows [ window_id ] ;
if ( wd . ime_active ) {
wd . ime_in_progress = true ;
}
return - 1 ; // Allow preedit strings of any length (no limit).
}
void DisplayServerX11 : : _xim_preedit_done_callback ( : : XIM xim , : : XPointer client_data ,
: : XPointer call_data ) {
DisplayServerX11 * ds = reinterpret_cast < DisplayServerX11 * > ( client_data ) ;
WindowID window_id = ds - > _get_focused_window_or_popup ( ) ;
WindowData & wd = ds - > windows [ window_id ] ;
if ( wd . ime_active ) {
wd . ime_in_progress = false ;
wd . ime_suppress_next_keyup = true ;
}
}
void DisplayServerX11 : : _xim_preedit_draw_callback ( : : XIM xim , : : XPointer client_data ,
: : XIMPreeditDrawCallbackStruct * call_data ) {
DisplayServerX11 * ds = reinterpret_cast < DisplayServerX11 * > ( client_data ) ;
WindowID window_id = ds - > _get_focused_window_or_popup ( ) ;
WindowData & wd = ds - > windows [ window_id ] ;
XIMText * xim_text = call_data - > text ;
2023-01-25 09:21:27 +01:00
if ( wd . ime_active ) {
if ( xim_text ! = nullptr ) {
String changed_text ;
if ( xim_text - > encoding_is_wchar ) {
changed_text = String ( xim_text - > string . wide_char ) ;
} else {
changed_text . parse_utf8 ( xim_text - > string . multi_byte ) ;
}
2022-12-11 00:21:22 +01:00
2023-01-25 09:21:27 +01:00
if ( call_data - > chg_length < 0 ) {
ds - > im_text = ds - > im_text . substr ( 0 , call_data - > chg_first ) + changed_text ;
} else {
ds - > im_text = ds - > im_text . substr ( 0 , call_data - > chg_first ) + changed_text + ds - > im_text . substr ( call_data - > chg_length ) ;
}
2022-12-11 00:21:22 +01:00
2023-01-25 09:21:27 +01:00
// Find the start and end of the selection.
int start = 0 , count = 0 ;
for ( int i = 0 ; i < xim_text - > length ; i + + ) {
if ( xim_text - > feedback [ i ] & XIMReverse ) {
if ( count = = 0 ) {
start = i ;
count = 1 ;
} else {
count + + ;
}
2022-12-11 00:21:22 +01:00
}
}
2023-01-25 09:21:27 +01:00
if ( count > 0 ) {
ds - > im_selection = Point2i ( start + call_data - > chg_first , count ) ;
} else {
ds - > im_selection = Point2i ( call_data - > caret , 0 ) ;
}
2022-12-11 00:21:22 +01:00
} else {
2023-01-25 09:21:27 +01:00
ds - > im_text = String ( ) ;
ds - > im_selection = Point2i ( ) ;
2022-12-11 00:21:22 +01:00
}
2023-01-25 09:21:27 +01:00
2023-12-18 15:46:56 +01:00
callable_mp ( ( Object * ) OS_Unix : : get_singleton ( ) - > get_main_loop ( ) , & Object : : notification ) . call_deferred ( MainLoop : : NOTIFICATION_OS_IME_UPDATE , false ) ;
2022-12-11 00:21:22 +01:00
}
}
void DisplayServerX11 : : _xim_preedit_caret_callback ( : : XIM xim , : : XPointer client_data ,
: : XIMPreeditCaretCallbackStruct * call_data ) {
}
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : _xim_destroy_callback ( : : XIM im , : : XPointer client_data ,
: : XPointer call_data ) {
WARN_PRINT ( " Input method stopped " ) ;
DisplayServerX11 * ds = reinterpret_cast < DisplayServerX11 * > ( client_data ) ;
2020-04-02 01:20:12 +02:00
ds - > xim = nullptr ;
2020-03-01 23:14:37 +01:00
2021-08-09 22:13:42 +02:00
for ( KeyValue < WindowID , WindowData > & E : ds - > windows ) {
E . value . xic = nullptr ;
2020-03-01 23:14:37 +01:00
}
}
void DisplayServerX11 : : _window_changed ( XEvent * event ) {
WindowID window_id = MAIN_WINDOW_ID ;
2020-03-13 00:12:34 +01:00
// Assign the event to the relevant window
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
if ( event - > xany . window = = E . value . x11_window ) {
window_id = E . key ;
2020-03-01 23:14:37 +01:00
break ;
}
}
2020-03-25 15:16:19 +01:00
Rect2i new_rect ;
2020-03-01 23:14:37 +01:00
WindowData & wd = windows [ window_id ] ;
2020-03-25 15:16:19 +01:00
if ( wd . x11_window ! = event - > xany . window ) { // Check if the correct window, in case it was not main window or anything else
2020-03-13 00:12:34 +01:00
return ;
}
2020-03-01 23:14:37 +01:00
2022-10-02 00:01:28 +02:00
// Query display server about a possible new window state.
wd . fullscreen = _window_fullscreen_check ( window_id ) ;
2023-05-09 02:23:45 +02:00
wd . maximized = _window_maximize_check ( window_id , " _NET_WM_STATE " ) & & ! wd . fullscreen ;
wd . minimized = _window_minimize_check ( window_id ) & & ! wd . fullscreen & & ! wd . maximized ;
2022-10-02 00:01:28 +02:00
2023-04-13 20:06:43 +02:00
// Readjusting the window position if the window is being reparented by the window manager for decoration
Window root , parent , * children ;
unsigned int nchildren ;
if ( XQueryTree ( x11_display , wd . x11_window , & root , & parent , & children , & nchildren ) & & wd . parent ! = parent ) {
wd . parent = parent ;
window_set_position ( wd . position , window_id ) ;
}
XFree ( children ) ;
2020-03-25 15:16:19 +01:00
{
//the position in xconfigure is not useful here, obtain it manually
2022-10-01 21:09:22 +02:00
int x = 0 , y = 0 ;
2020-03-25 15:16:19 +01:00
Window child ;
XTranslateCoordinates ( x11_display , wd . x11_window , DefaultRootWindow ( x11_display ) , 0 , 0 , & x , & y , & child ) ;
new_rect . position . x = x ;
new_rect . position . y = y ;
new_rect . size . width = event - > xconfigure . width ;
new_rect . size . height = event - > xconfigure . height ;
}
if ( new_rect = = Rect2i ( wd . position , wd . size ) ) {
return ;
}
2020-03-06 18:00:16 +01:00
2020-03-25 15:16:19 +01:00
wd . position = new_rect . position ;
wd . size = new_rect . size ;
2020-03-01 23:14:37 +01:00
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
2023-12-19 18:57:56 +01:00
if ( rendering_context ) {
rendering_context - > window_set_size ( window_id , wd . size . width , wd . size . height ) ;
2020-03-01 23:14:37 +01:00
}
# endif
2021-10-26 17:18:39 +02:00
# if defined(GLES3_ENABLED)
2020-11-18 19:11:30 +01:00
if ( gl_manager ) {
gl_manager - > window_resize ( window_id , wd . size . width , wd . size . height ) ;
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
gl_manager_egl - > window_resize ( window_id , wd . size . width , wd . size . height ) ;
}
2020-11-18 19:11:30 +01:00
# endif
2020-03-04 02:51:12 +01:00
2023-07-11 16:18:10 +02:00
if ( wd . rect_changed_callback . is_valid ( ) ) {
wd . rect_changed_callback . call ( new_rect ) ;
2020-03-04 02:51:12 +01:00
}
2020-03-01 23:14:37 +01:00
}
2022-09-07 19:31:57 +02:00
DisplayServer : : WindowID DisplayServerX11 : : _get_focused_window_or_popup ( ) const {
const List < WindowID > : : Element * E = popup_list . back ( ) ;
if ( E ) {
return E - > get ( ) ;
}
return last_focused_window ;
}
2020-03-04 17:36:09 +01:00
void DisplayServerX11 : : _dispatch_input_events ( const Ref < InputEvent > & p_event ) {
2022-04-05 12:40:26 +02:00
static_cast < DisplayServerX11 * > ( get_singleton ( ) ) - > _dispatch_input_event ( p_event ) ;
2020-03-04 17:36:09 +01:00
}
void DisplayServerX11 : : _dispatch_input_event ( const Ref < InputEvent > & p_event ) {
2022-02-24 10:21:23 +01:00
{
2022-04-13 08:58:38 +02:00
List < WindowID > : : Element * E = popup_list . back ( ) ;
2022-02-24 10:21:23 +01:00
if ( E & & Object : : cast_to < InputEventKey > ( * p_event ) ) {
// Redirect keyboard input to active popup.
if ( windows . has ( E - > get ( ) ) ) {
Callable callable = windows [ E - > get ( ) ] . input_event_callback ;
if ( callable . is_valid ( ) ) {
2023-07-11 16:18:10 +02:00
callable . call ( p_event ) ;
2022-02-24 10:21:23 +01:00
}
}
return ;
}
}
2020-03-04 17:36:09 +01:00
Ref < InputEventFromWindow > event_from_window = p_event ;
if ( event_from_window . is_valid ( ) & & event_from_window - > get_window_id ( ) ! = INVALID_WINDOW_ID ) {
2022-02-24 10:21:23 +01:00
// Send to a single window.
if ( windows . has ( event_from_window - > get_window_id ( ) ) ) {
Callable callable = windows [ event_from_window - > get_window_id ( ) ] . input_event_callback ;
if ( callable . is_valid ( ) ) {
2023-07-11 16:18:10 +02:00
callable . call ( p_event ) ;
2022-02-24 10:21:23 +01:00
}
2020-03-04 17:36:09 +01:00
}
} else {
2022-02-24 10:21:23 +01:00
// Send to all windows.
2021-08-09 22:13:42 +02:00
for ( KeyValue < WindowID , WindowData > & E : windows ) {
Callable callable = E . value . input_event_callback ;
2022-02-24 10:21:23 +01:00
if ( callable . is_valid ( ) ) {
2023-07-11 16:18:10 +02:00
callable . call ( p_event ) ;
2020-03-04 17:36:09 +01:00
}
}
}
}
void DisplayServerX11 : : _send_window_event ( const WindowData & wd , WindowEvent p_event ) {
2023-07-11 16:18:10 +02:00
if ( wd . event_callback . is_valid ( ) ) {
2020-03-04 17:36:09 +01:00
Variant event = int ( p_event ) ;
2023-07-11 16:18:10 +02:00
wd . event_callback . call ( event ) ;
2020-03-04 17:36:09 +01:00
}
}
2020-05-14 14:29:06 +02:00
2023-12-31 19:53:27 +01:00
void DisplayServerX11 : : _set_input_focus ( Window p_window , int p_revert_to ) {
Window focused_window ;
int focus_ret_state ;
XGetInputFocus ( x11_display , & focused_window , & focus_ret_state ) ;
// Only attempt to change focus if the window isn't already focused, in order to
// prevent issues with Godot stealing input focus with alternative window managers.
if ( p_window ! = focused_window ) {
XSetInputFocus ( x11_display , p_window , p_revert_to , CurrentTime ) ;
}
}
2020-09-09 12:43:20 +02:00
void DisplayServerX11 : : _poll_events_thread ( void * ud ) {
2022-04-05 12:40:26 +02:00
DisplayServerX11 * display_server = static_cast < DisplayServerX11 * > ( ud ) ;
2020-09-09 12:43:20 +02:00
display_server - > _poll_events ( ) ;
}
Bool DisplayServerX11 : : _predicate_all_events ( Display * display , XEvent * event , XPointer arg ) {
// Just accept all events.
return True ;
}
2020-10-08 17:57:54 +02:00
bool DisplayServerX11 : : _wait_for_events ( ) const {
2020-09-09 12:43:20 +02:00
int x11_fd = ConnectionNumber ( x11_display ) ;
fd_set in_fds ;
2020-10-08 17:57:54 +02:00
XFlush ( x11_display ) ;
FD_ZERO ( & in_fds ) ;
FD_SET ( x11_fd , & in_fds ) ;
2020-09-09 12:43:20 +02:00
2020-10-08 17:57:54 +02:00
struct timeval tv ;
tv . tv_usec = 0 ;
tv . tv_sec = 1 ;
2020-09-09 12:43:20 +02:00
2020-10-08 17:57:54 +02:00
// Wait for next event or timeout.
2021-04-05 14:02:50 +02:00
int num_ready_fds = select ( x11_fd + 1 , & in_fds , nullptr , nullptr , & tv ) ;
2020-09-09 12:43:20 +02:00
2020-10-08 17:57:54 +02:00
if ( num_ready_fds > 0 ) {
// Event received.
return true ;
} else {
// Error or timeout.
2020-09-09 12:43:20 +02:00
if ( num_ready_fds < 0 ) {
2020-10-08 17:57:54 +02:00
ERR_PRINT ( " _wait_for_events: select error: " + itos ( errno ) ) ;
2020-09-09 12:43:20 +02:00
}
2020-10-08 17:57:54 +02:00
return false ;
}
}
void DisplayServerX11 : : _poll_events ( ) {
2021-02-10 19:22:13 +01:00
while ( ! events_thread_done . is_set ( ) ) {
2020-10-08 17:57:54 +02:00
_wait_for_events ( ) ;
2020-09-09 12:43:20 +02:00
// Process events from the queue.
{
MutexLock mutex_lock ( events_mutex ) ;
2021-10-28 00:23:40 +02:00
_check_pending_events ( polled_events ) ;
}
}
}
2020-09-25 16:40:04 +02:00
2021-10-28 00:23:40 +02:00
void DisplayServerX11 : : _check_pending_events ( LocalVector < XEvent > & r_events ) {
// Flush to make sure to gather all pending events.
XFlush ( x11_display ) ;
2020-09-25 16:40:04 +02:00
2021-10-28 00:23:40 +02:00
// Non-blocking wait for next event and remove it from the queue.
2022-10-01 21:09:22 +02:00
XEvent ev = { } ;
2021-10-28 00:23:40 +02:00
while ( XCheckIfEvent ( x11_display , & ev , _predicate_all_events , nullptr ) ) {
// Check if the input manager wants to process the event.
if ( XFilterEvent ( & ev , None ) ) {
// Event has been filtered by the Input Manager,
// it has to be ignored and a new one will be received.
continue ;
}
// Handle selection request events directly in the event thread, because
// communication through the x server takes several events sent back and forth
// and we don't want to block other programs while processing only one each frame.
if ( ev . type = = SelectionRequest ) {
_handle_selection_request_event ( & ( ev . xselectionrequest ) ) ;
continue ;
2020-09-09 12:43:20 +02:00
}
2021-10-28 00:23:40 +02:00
r_events . push_back ( ev ) ;
2020-09-09 12:43:20 +02:00
}
}
2022-02-24 10:21:23 +01:00
DisplayServer : : WindowID DisplayServerX11 : : window_get_active_popup ( ) const {
const List < WindowID > : : Element * E = popup_list . back ( ) ;
if ( E ) {
return E - > get ( ) ;
} else {
return INVALID_WINDOW_ID ;
}
}
void DisplayServerX11 : : window_set_popup_safe_rect ( WindowID p_window , const Rect2i & p_rect ) {
_THREAD_SAFE_METHOD_
ERR_FAIL_COND ( ! windows . has ( p_window ) ) ;
WindowData & wd = windows [ p_window ] ;
wd . parent_safe_rect = p_rect ;
}
Rect2i DisplayServerX11 : : window_get_popup_safe_rect ( WindowID p_window ) const {
_THREAD_SAFE_METHOD_
ERR_FAIL_COND_V ( ! windows . has ( p_window ) , Rect2i ( ) ) ;
const WindowData & wd = windows [ p_window ] ;
return wd . parent_safe_rect ;
}
void DisplayServerX11 : : popup_open ( WindowID p_window ) {
2022-04-05 11:34:27 +02:00
_THREAD_SAFE_METHOD_
2021-04-15 20:22:58 +02:00
bool has_popup_ancestor = false ;
WindowID transient_root = p_window ;
while ( true ) {
WindowID parent = windows [ transient_root ] . transient_parent ;
if ( parent = = INVALID_WINDOW_ID ) {
break ;
} else {
transient_root = parent ;
if ( windows [ parent ] . is_popup ) {
has_popup_ancestor = true ;
break ;
}
}
}
2024-06-06 19:00:05 +02:00
// Detect tooltips and other similar popups that shouldn't block input to their parent.
bool ignores_input = window_get_flag ( WINDOW_FLAG_NO_FOCUS , p_window ) & & window_get_flag ( WINDOW_FLAG_MOUSE_PASSTHROUGH , p_window ) ;
2022-02-24 10:21:23 +01:00
WindowData & wd = windows [ p_window ] ;
2024-06-06 19:00:05 +02:00
if ( wd . is_popup | | ( has_popup_ancestor & & ! ignores_input ) ) {
2022-04-05 11:34:27 +02:00
// Find current popup parent, or root popup if new window is not transient.
List < WindowID > : : Element * C = nullptr ;
2022-02-24 10:21:23 +01:00
List < WindowID > : : Element * E = popup_list . back ( ) ;
while ( E ) {
if ( wd . transient_parent ! = E - > get ( ) | | wd . transient_parent = = INVALID_WINDOW_ID ) {
2022-04-05 11:34:27 +02:00
C = E ;
E = E - > prev ( ) ;
2022-02-24 10:21:23 +01:00
} else {
break ;
}
}
2022-04-05 11:34:27 +02:00
if ( C ) {
_send_window_event ( windows [ C - > get ( ) ] , DisplayServerX11 : : WINDOW_EVENT_CLOSE_REQUEST ) ;
}
2022-02-24 10:21:23 +01:00
time_since_popup = OS : : get_singleton ( ) - > get_ticks_msec ( ) ;
popup_list . push_back ( p_window ) ;
}
}
void DisplayServerX11 : : popup_close ( WindowID p_window ) {
2022-04-05 11:34:27 +02:00
_THREAD_SAFE_METHOD_
2022-02-24 10:21:23 +01:00
List < WindowID > : : Element * E = popup_list . find ( p_window ) ;
while ( E ) {
List < WindowID > : : Element * F = E - > next ( ) ;
2022-04-05 11:34:27 +02:00
WindowID win_id = E - > get ( ) ;
2022-02-24 10:21:23 +01:00
popup_list . erase ( E ) ;
2022-04-05 11:34:27 +02:00
2024-05-28 08:41:04 +02:00
if ( win_id ! = p_window ) {
// Only request close on related windows, not this window. We are already processing it.
_send_window_event ( windows [ win_id ] , DisplayServerX11 : : WINDOW_EVENT_CLOSE_REQUEST ) ;
}
2022-02-24 10:21:23 +01:00
E = F ;
}
}
2022-05-30 10:06:08 +02:00
bool DisplayServerX11 : : mouse_process_popups ( ) {
2022-04-05 11:34:27 +02:00
_THREAD_SAFE_METHOD_
2022-02-24 10:21:23 +01:00
if ( popup_list . is_empty ( ) ) {
2022-05-30 10:06:08 +02:00
return false ;
2022-02-24 10:21:23 +01:00
}
uint64_t delta = OS : : get_singleton ( ) - > get_ticks_msec ( ) - time_since_popup ;
if ( delta < 250 ) {
2022-05-30 10:06:08 +02:00
return false ;
2022-02-24 10:21:23 +01:00
}
int number_of_screens = XScreenCount ( x11_display ) ;
2022-05-30 10:06:08 +02:00
bool closed = false ;
2022-02-24 10:21:23 +01:00
for ( int i = 0 ; i < number_of_screens ; i + + ) {
Window root , child ;
int root_x , root_y , win_x , win_y ;
unsigned int mask ;
if ( XQueryPointer ( x11_display , XRootWindow ( x11_display , i ) , & root , & child , & root_x , & root_y , & win_x , & win_y , & mask ) ) {
XWindowAttributes root_attrs ;
XGetWindowAttributes ( x11_display , root , & root_attrs ) ;
Vector2i pos = Vector2i ( root_attrs . x + root_x , root_attrs . y + root_y ) ;
2022-10-24 01:47:48 +02:00
if ( mask ! = last_mouse_monitor_mask ) {
2022-02-24 10:21:23 +01:00
if ( ( ( mask & Button1Mask ) | | ( mask & Button2Mask ) | | ( mask & Button3Mask ) | | ( mask & Button4Mask ) | | ( mask & Button5Mask ) ) ) {
2022-04-05 11:34:27 +02:00
List < WindowID > : : Element * C = nullptr ;
2022-02-24 10:21:23 +01:00
List < WindowID > : : Element * E = popup_list . back ( ) ;
2022-04-05 11:34:27 +02:00
// Find top popup to close.
2022-02-24 10:21:23 +01:00
while ( E ) {
// Popup window area.
2023-07-14 08:16:06 +02:00
Rect2i win_rect = Rect2i ( window_get_position_with_decorations ( E - > get ( ) ) , window_get_size_with_decorations ( E - > get ( ) ) ) ;
2022-02-24 10:21:23 +01:00
// Area of the parent window, which responsible for opening sub-menu.
Rect2i safe_rect = window_get_popup_safe_rect ( E - > get ( ) ) ;
if ( win_rect . has_point ( pos ) ) {
break ;
} else if ( safe_rect ! = Rect2i ( ) & & safe_rect . has_point ( pos ) ) {
break ;
} else {
2022-04-05 11:34:27 +02:00
C = E ;
E = E - > prev ( ) ;
2022-02-24 10:21:23 +01:00
}
}
2022-04-05 11:34:27 +02:00
if ( C ) {
_send_window_event ( windows [ C - > get ( ) ] , DisplayServerX11 : : WINDOW_EVENT_CLOSE_REQUEST ) ;
2022-05-30 10:06:08 +02:00
closed = true ;
2022-04-05 11:34:27 +02:00
}
2022-02-24 10:21:23 +01:00
}
}
last_mouse_monitor_mask = mask ;
}
}
2022-05-30 10:06:08 +02:00
return closed ;
2022-02-24 10:21:23 +01:00
}
2023-03-04 17:52:15 +01:00
bool DisplayServerX11 : : _window_focus_check ( ) {
Window focused_window ;
int focus_ret_state ;
XGetInputFocus ( x11_display , & focused_window , & focus_ret_state ) ;
bool has_focus = false ;
for ( const KeyValue < int , DisplayServerX11 : : WindowData > & wid : windows ) {
if ( wid . value . x11_window = = focused_window ) {
has_focus = true ;
break ;
}
}
return has_focus ;
}
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : process_events ( ) {
2024-04-15 18:17:09 +02:00
ERR_FAIL_COND ( ! Thread : : is_main_thread ( ) ) ;
2024-04-09 11:47:06 +02:00
_THREAD_SAFE_LOCK_
2020-03-07 16:46:50 +01:00
2020-08-22 13:01:49 +02:00
# ifdef DISPLAY_SERVER_X11_DEBUG_LOGS_ENABLED
static int frame = 0 ;
+ + frame ;
# endif
2022-05-30 10:06:08 +02:00
bool ignore_events = mouse_process_popups ( ) ;
2022-02-24 10:21:23 +01:00
2020-07-01 14:27:43 +02:00
if ( app_focused ) {
//verify that one of the windows has focus, else send focus out notification
bool focus_found = false ;
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
if ( E . value . focused ) {
2020-07-01 14:27:43 +02:00
focus_found = true ;
2020-07-04 00:07:26 +02:00
break ;
2020-07-01 14:27:43 +02:00
}
}
if ( ! focus_found ) {
2020-07-01 21:14:46 +02:00
uint64_t delta = OS : : get_singleton ( ) - > get_ticks_msec ( ) - time_since_no_focus ;
2020-07-01 14:27:43 +02:00
2020-07-01 21:14:46 +02:00
if ( delta > 250 ) {
2020-07-16 18:54:15 +02:00
//X11 can go between windows and have no focus for a while, when creating them or something else. Use this as safety to avoid unnecessary focus in/outs.
2020-07-01 21:14:46 +02:00
if ( OS : : get_singleton ( ) - > get_main_loop ( ) ) {
2020-08-22 13:01:49 +02:00
DEBUG_LOG_X11 ( " All focus lost, triggering NOTIFICATION_APPLICATION_FOCUS_OUT \n " ) ;
2020-07-01 21:14:46 +02:00
OS : : get_singleton ( ) - > get_main_loop ( ) - > notification ( MainLoop : : NOTIFICATION_APPLICATION_FOCUS_OUT ) ;
}
app_focused = false ;
}
} else {
time_since_no_focus = OS : : get_singleton ( ) - > get_ticks_msec ( ) ;
2020-07-01 14:27:43 +02:00
}
}
2020-03-01 23:14:37 +01:00
do_mouse_warp = false ;
// Is the current mouse mode one where it needs to be grabbed.
2021-03-31 00:35:08 +02:00
bool mouse_mode_grab = mouse_mode = = MOUSE_MODE_CAPTURED | | mouse_mode = = MOUSE_MODE_CONFINED | | mouse_mode = = MOUSE_MODE_CONFINED_HIDDEN ;
2020-03-01 23:14:37 +01:00
2020-04-10 12:41:41 +02:00
xi . pressure = 0 ;
xi . tilt = Vector2 ( ) ;
2020-05-03 14:38:58 +02:00
xi . pressure_supported = false ;
2020-04-10 12:41:41 +02:00
2020-09-09 12:43:20 +02:00
LocalVector < XEvent > events ;
{
// Block events polling while flushing events.
MutexLock mutex_lock ( events_mutex ) ;
events = polled_events ;
polled_events . clear ( ) ;
2021-10-28 00:23:40 +02:00
// Check for more pending events to avoid an extra frame delay.
_check_pending_events ( events ) ;
2020-09-09 12:43:20 +02:00
}
2020-03-01 23:14:37 +01:00
2020-09-09 12:43:20 +02:00
for ( uint32_t event_index = 0 ; event_index < events . size ( ) ; + + event_index ) {
XEvent & event = events [ event_index ] ;
2020-08-22 13:01:49 +02:00
2022-12-11 00:21:22 +01:00
bool ime_window_event = false ;
2020-03-01 23:14:37 +01:00
WindowID window_id = MAIN_WINDOW_ID ;
2020-03-13 00:12:34 +01:00
// Assign the event to the relevant window
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
if ( event . xany . window = = E . value . x11_window ) {
window_id = E . key ;
2020-03-01 23:14:37 +01:00
break ;
}
2022-12-11 00:21:22 +01:00
if ( event . xany . window = = E . value . x11_xim_window ) {
window_id = E . key ;
ime_window_event = true ;
break ;
}
2020-03-01 23:14:37 +01:00
}
if ( XGetEventData ( x11_display , & event . xcookie ) ) {
if ( event . xcookie . type = = GenericEvent & & event . xcookie . extension = = xi . opcode ) {
XIDeviceEvent * event_data = ( XIDeviceEvent * ) event . xcookie . data ;
switch ( event_data - > evtype ) {
case XI_HierarchyChanged :
case XI_DeviceChanged : {
_refresh_device_info ( ) ;
} break ;
case XI_RawMotion : {
2023-02-07 11:13:34 +01:00
if ( ime_window_event | | ignore_events ) {
2022-12-11 00:21:22 +01:00
break ;
}
2020-03-01 23:14:37 +01:00
XIRawEvent * raw_event = ( XIRawEvent * ) event_data ;
2022-06-19 11:58:24 +02:00
int device_id = raw_event - > sourceid ;
2020-03-01 23:14:37 +01:00
// Determine the axis used (called valuators in XInput for some forsaken reason)
// Mask is a bitmask indicating which axes are involved.
// We are interested in the values of axes 0 and 1.
if ( raw_event - > valuators . mask_len < = 0 ) {
break ;
}
const double * values = raw_event - > raw_values ;
double rel_x = 0.0 ;
double rel_y = 0.0 ;
if ( XIMaskIsSet ( raw_event - > valuators . mask , VALUATOR_ABSX ) ) {
rel_x = * values ;
values + + ;
}
if ( XIMaskIsSet ( raw_event - > valuators . mask , VALUATOR_ABSY ) ) {
rel_y = * values ;
values + + ;
}
if ( XIMaskIsSet ( raw_event - > valuators . mask , VALUATOR_PRESSURE ) ) {
2022-05-13 15:04:37 +02:00
HashMap < int , Vector2 > : : Iterator pen_pressure = xi . pen_pressure_range . find ( device_id ) ;
2020-04-10 12:41:41 +02:00
if ( pen_pressure ) {
2022-05-13 15:04:37 +02:00
Vector2 pen_pressure_range = pen_pressure - > value ;
2020-05-03 14:38:58 +02:00
if ( pen_pressure_range ! = Vector2 ( ) ) {
xi . pressure_supported = true ;
2020-04-10 12:41:41 +02:00
xi . pressure = ( * values - pen_pressure_range [ 0 ] ) /
2021-10-28 15:19:35 +02:00
( pen_pressure_range [ 1 ] - pen_pressure_range [ 0 ] ) ;
2020-05-03 14:38:58 +02:00
}
2020-04-10 12:41:41 +02:00
}
2020-03-01 23:14:37 +01:00
values + + ;
}
if ( XIMaskIsSet ( raw_event - > valuators . mask , VALUATOR_TILTX ) ) {
2022-05-13 15:04:37 +02:00
HashMap < int , Vector2 > : : Iterator pen_tilt_x = xi . pen_tilt_x_range . find ( device_id ) ;
2020-04-10 12:41:41 +02:00
if ( pen_tilt_x ) {
2022-05-13 15:04:37 +02:00
Vector2 pen_tilt_x_range = pen_tilt_x - > value ;
2022-01-02 12:50:16 +01:00
if ( pen_tilt_x_range [ 0 ] ! = 0 & & * values < 0 ) {
xi . tilt . x = * values / - pen_tilt_x_range [ 0 ] ;
} else if ( pen_tilt_x_range [ 1 ] ! = 0 ) {
xi . tilt . x = * values / pen_tilt_x_range [ 1 ] ;
2020-04-10 12:41:41 +02:00
}
}
2020-03-01 23:14:37 +01:00
values + + ;
}
if ( XIMaskIsSet ( raw_event - > valuators . mask , VALUATOR_TILTY ) ) {
2022-05-13 15:04:37 +02:00
HashMap < int , Vector2 > : : Iterator pen_tilt_y = xi . pen_tilt_y_range . find ( device_id ) ;
2020-04-10 12:41:41 +02:00
if ( pen_tilt_y ) {
2022-05-13 15:04:37 +02:00
Vector2 pen_tilt_y_range = pen_tilt_y - > value ;
2022-01-02 12:50:16 +01:00
if ( pen_tilt_y_range [ 0 ] ! = 0 & & * values < 0 ) {
xi . tilt . y = * values / - pen_tilt_y_range [ 0 ] ;
} else if ( pen_tilt_y_range [ 1 ] ! = 0 ) {
xi . tilt . y = * values / pen_tilt_y_range [ 1 ] ;
2020-04-10 12:41:41 +02:00
}
}
2020-03-01 23:14:37 +01:00
2020-04-10 12:41:41 +02:00
values + + ;
2020-03-01 23:14:37 +01:00
}
2022-06-19 11:58:24 +02:00
HashMap < int , bool > : : Iterator pen_inverted = xi . pen_inverted_devices . find ( device_id ) ;
if ( pen_inverted ) {
xi . pen_inverted = pen_inverted - > value ;
}
2020-03-01 23:14:37 +01:00
// https://bugs.freedesktop.org/show_bug.cgi?id=71609
// http://lists.libsdl.org/pipermail/commits-libsdl.org/2015-June/000282.html
if ( raw_event - > time = = xi . last_relative_time & & rel_x = = xi . relative_motion . x & & rel_y = = xi . relative_motion . y ) {
break ; // Flush duplicate to avoid overly fast motion
}
xi . old_raw_pos . x = xi . raw_pos . x ;
xi . old_raw_pos . y = xi . raw_pos . y ;
xi . raw_pos . x = rel_x ;
xi . raw_pos . y = rel_y ;
2022-05-13 15:04:37 +02:00
HashMap < int , Vector2 > : : Iterator abs_info = xi . absolute_devices . find ( device_id ) ;
2020-03-01 23:14:37 +01:00
if ( abs_info ) {
// Absolute mode device
2022-05-13 15:04:37 +02:00
Vector2 mult = abs_info - > value ;
2020-03-01 23:14:37 +01:00
xi . relative_motion . x + = ( xi . raw_pos . x - xi . old_raw_pos . x ) * mult . x ;
xi . relative_motion . y + = ( xi . raw_pos . y - xi . old_raw_pos . y ) * mult . y ;
} else {
// Relative mode device
xi . relative_motion . x = xi . raw_pos . x ;
xi . relative_motion . y = xi . raw_pos . y ;
}
xi . last_relative_time = raw_event - > time ;
} break ;
# ifdef TOUCH_ENABLED
2021-10-28 15:43:36 +02:00
case XI_TouchBegin :
2020-03-01 23:14:37 +01:00
case XI_TouchEnd : {
2023-02-07 11:13:34 +01:00
if ( ime_window_event | | ignore_events ) {
2022-12-11 00:21:22 +01:00
break ;
}
2020-03-01 23:14:37 +01:00
bool is_begin = event_data - > evtype = = XI_TouchBegin ;
2023-10-13 14:38:04 +02:00
int index = event_data - > detail ;
Vector2 pos = Vector2 ( event_data - > event_x , event_data - > event_y ) ;
2020-03-01 23:14:37 +01:00
Ref < InputEventScreenTouch > st ;
2021-06-18 00:03:09 +02:00
st . instantiate ( ) ;
2020-03-01 23:14:37 +01:00
st - > set_window_id ( window_id ) ;
st - > set_index ( index ) ;
st - > set_position ( pos ) ;
st - > set_pressed ( is_begin ) ;
if ( is_begin ) {
2020-05-14 16:41:43 +02:00
if ( xi . state . has ( index ) ) { // Defensive
2020-03-01 23:14:37 +01:00
break ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
xi . state [ index ] = pos ;
if ( xi . state . size ( ) = = 1 ) {
// X11 may send a motion event when a touch gesture begins, that would result
// in a spurious mouse motion event being sent to Godot; remember it to be able to filter it out
xi . mouse_pos_to_filter = pos ;
}
2021-08-13 00:31:16 +02:00
Input : : get_singleton ( ) - > parse_input_event ( st ) ;
2020-03-01 23:14:37 +01:00
} else {
2020-05-14 16:41:43 +02:00
if ( ! xi . state . has ( index ) ) { // Defensive
2020-03-01 23:14:37 +01:00
break ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
xi . state . erase ( index ) ;
2021-08-13 00:31:16 +02:00
Input : : get_singleton ( ) - > parse_input_event ( st ) ;
2020-03-01 23:14:37 +01:00
}
} break ;
case XI_TouchUpdate : {
2023-02-07 11:13:34 +01:00
if ( ime_window_event | | ignore_events ) {
2022-12-11 00:21:22 +01:00
break ;
}
2023-10-13 14:38:04 +02:00
int index = event_data - > detail ;
Vector2 pos = Vector2 ( event_data - > event_x , event_data - > event_y ) ;
2022-05-13 15:04:37 +02:00
HashMap < int , Vector2 > : : Iterator curr_pos_elem = xi . state . find ( index ) ;
2020-03-01 23:14:37 +01:00
if ( ! curr_pos_elem ) { // Defensive
break ;
}
2022-05-13 15:04:37 +02:00
if ( curr_pos_elem - > value ! = pos ) {
2020-03-01 23:14:37 +01:00
Ref < InputEventScreenDrag > sd ;
2021-06-18 00:03:09 +02:00
sd . instantiate ( ) ;
2020-03-01 23:14:37 +01:00
sd - > set_window_id ( window_id ) ;
sd - > set_index ( index ) ;
sd - > set_position ( pos ) ;
2022-05-13 15:04:37 +02:00
sd - > set_relative ( pos - curr_pos_elem - > value ) ;
2023-10-04 19:20:01 +02:00
sd - > set_relative_screen_position ( sd - > get_relative ( ) ) ;
2021-08-13 00:31:16 +02:00
Input : : get_singleton ( ) - > parse_input_event ( sd ) ;
2020-03-01 23:14:37 +01:00
2022-05-13 15:04:37 +02:00
curr_pos_elem - > value = pos ;
2020-03-01 23:14:37 +01:00
}
} break ;
# endif
}
}
}
XFreeEventData ( x11_display , & event . xcookie ) ;
switch ( event . type ) {
2020-08-22 13:01:49 +02:00
case MapNotify : {
DEBUG_LOG_X11 ( " [%u] MapNotify window=%lu (%u) \n " , frame , event . xmap . window , window_id ) ;
2022-12-11 00:21:22 +01:00
if ( ime_window_event ) {
break ;
}
2020-08-22 13:01:49 +02:00
const WindowData & wd = windows [ window_id ] ;
2022-08-26 22:13:58 +02:00
XWindowAttributes xwa ;
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd . x11_window , & xwa ) ;
2020-08-22 13:01:49 +02:00
// Set focus when menu window is started.
// RevertToPointerRoot is used to make sure we don't lose all focus in case
// a subwindow and its parent are both destroyed.
2023-03-04 17:52:15 +01:00
if ( ( xwa . map_state = = IsViewable ) & & ! wd . no_focus & & ! wd . is_popup & & _window_focus_check ( ) ) {
2023-12-31 19:53:27 +01:00
_set_input_focus ( wd . x11_window , RevertToPointerRoot ) ;
2020-08-22 13:01:49 +02:00
}
2022-08-30 20:52:24 +02:00
// Have we failed to set fullscreen while the window was unmapped?
_validate_mode_on_map ( window_id ) ;
2020-08-22 13:01:49 +02:00
} break ;
case Expose : {
DEBUG_LOG_X11 ( " [%u] Expose window=%lu (%u), count='%u' \n " , frame , event . xexpose . window , window_id , event . xexpose . count ) ;
2022-12-11 00:21:22 +01:00
if ( ime_window_event ) {
break ;
}
2020-08-22 13:01:49 +02:00
2022-06-30 03:55:40 +02:00
windows [ window_id ] . fullscreen = _window_fullscreen_check ( window_id ) ;
2020-03-01 23:14:37 +01:00
Main : : force_redraw ( ) ;
2020-08-22 13:01:49 +02:00
} break ;
case NoExpose : {
DEBUG_LOG_X11 ( " [%u] NoExpose drawable=%lu (%u) \n " , frame , event . xnoexpose . drawable , window_id ) ;
2022-12-11 00:21:22 +01:00
if ( ime_window_event ) {
break ;
}
2020-03-01 23:14:37 +01:00
2020-07-01 14:27:43 +02:00
windows [ window_id ] . minimized = true ;
2020-08-22 13:01:49 +02:00
} break ;
2020-03-01 23:14:37 +01:00
case VisibilityNotify : {
2020-08-22 13:01:49 +02:00
DEBUG_LOG_X11 ( " [%u] VisibilityNotify window=%lu (%u), state=%u \n " , frame , event . xvisibility . window , window_id , event . xvisibility . state ) ;
2022-12-11 00:21:22 +01:00
if ( ime_window_event ) {
break ;
}
2020-08-22 13:01:49 +02:00
2022-08-30 20:52:24 +02:00
windows [ window_id ] . minimized = _window_minimize_check ( window_id ) ;
2020-03-01 23:14:37 +01:00
} break ;
2020-08-22 13:01:49 +02:00
2020-03-01 23:14:37 +01:00
case LeaveNotify : {
2020-08-22 13:01:49 +02:00
DEBUG_LOG_X11 ( " [%u] LeaveNotify window=%lu (%u), mode='%u' \n " , frame , event . xcrossing . window , window_id , event . xcrossing . mode ) ;
2022-12-11 00:21:22 +01:00
if ( ime_window_event ) {
break ;
}
2020-08-22 13:01:49 +02:00
2023-08-05 01:52:59 +02:00
if ( ! mouse_mode_grab & & window_mouseover_id = = window_id ) {
window_mouseover_id = INVALID_WINDOW_ID ;
2020-03-04 17:36:09 +01:00
_send_window_event ( windows [ window_id ] , WINDOW_EVENT_MOUSE_EXIT ) ;
}
2020-03-01 23:14:37 +01:00
} break ;
2020-08-22 13:01:49 +02:00
2020-03-01 23:14:37 +01:00
case EnterNotify : {
2020-08-22 13:01:49 +02:00
DEBUG_LOG_X11 ( " [%u] EnterNotify window=%lu (%u), mode='%u' \n " , frame , event . xcrossing . window , window_id , event . xcrossing . mode ) ;
2022-12-11 00:21:22 +01:00
if ( ime_window_event ) {
break ;
}
2020-08-22 13:01:49 +02:00
2023-08-05 01:52:59 +02:00
if ( ! mouse_mode_grab & & window_mouseover_id ! = window_id ) {
if ( window_mouseover_id ! = INVALID_WINDOW_ID ) {
_send_window_event ( windows [ window_mouseover_id ] , WINDOW_EVENT_MOUSE_EXIT ) ;
}
window_mouseover_id = window_id ;
2020-03-04 17:36:09 +01:00
_send_window_event ( windows [ window_id ] , WINDOW_EVENT_MOUSE_ENTER ) ;
}
2020-03-01 23:14:37 +01:00
} break ;
2020-08-22 13:01:49 +02:00
case FocusIn : {
DEBUG_LOG_X11 ( " [%u] FocusIn window=%lu (%u), mode='%u' \n " , frame , event . xfocus . window , window_id , event . xfocus . mode ) ;
2023-02-06 08:18:16 +01:00
if ( ime_window_event | | ( event . xfocus . detail = = NotifyInferior ) ) {
2022-12-11 00:21:22 +01:00
break ;
}
2020-08-22 13:01:49 +02:00
WindowData & wd = windows [ window_id ] ;
2022-02-01 03:49:51 +01:00
last_focused_window = window_id ;
2020-08-22 13:01:49 +02:00
wd . focused = true ;
2020-08-29 12:00:44 +02:00
// Keep track of focus order for overlapping windows.
static unsigned int focus_order = 0 ;
wd . focus_order = + + focus_order ;
2020-08-22 13:01:49 +02:00
_send_window_event ( wd , WINDOW_EVENT_FOCUS_IN ) ;
2020-03-01 23:14:37 +01:00
if ( mouse_mode_grab ) {
// Show and update the cursor if confined and the window regained focus.
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
2020-05-14 16:41:43 +02:00
if ( mouse_mode = = MOUSE_MODE_CONFINED ) {
2021-08-09 22:13:42 +02:00
XUndefineCursor ( x11_display , E . value . x11_window ) ;
2021-03-31 00:35:08 +02:00
} else if ( mouse_mode = = MOUSE_MODE_CAPTURED | | mouse_mode = = MOUSE_MODE_CONFINED_HIDDEN ) { // Or re-hide it.
2021-08-09 22:13:42 +02:00
XDefineCursor ( x11_display , E . value . x11_window , null_cursor ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
XGrabPointer (
2021-08-09 22:13:42 +02:00
x11_display , E . value . x11_window , True ,
2020-03-01 23:14:37 +01:00
ButtonPressMask | ButtonReleaseMask | PointerMotionMask ,
2021-08-09 22:13:42 +02:00
GrabModeAsync , GrabModeAsync , E . value . x11_window , None , CurrentTime ) ;
2020-03-01 23:14:37 +01:00
}
}
# ifdef TOUCH_ENABLED
// Grab touch devices to avoid OS gesture interference
/*for (int i = 0; i < xi.touch_devices.size(); ++i) {
XIGrabDevice ( x11_display , xi . touch_devices [ i ] , x11_window , CurrentTime , None , XIGrabModeAsync , XIGrabModeAsync , False , & xi . touch_event_mask ) ;
} */
# endif
2020-07-01 14:27:43 +02:00
if ( ! app_focused ) {
if ( OS : : get_singleton ( ) - > get_main_loop ( ) ) {
OS : : get_singleton ( ) - > get_main_loop ( ) - > notification ( MainLoop : : NOTIFICATION_APPLICATION_FOCUS_IN ) ;
}
app_focused = true ;
}
2020-08-22 13:01:49 +02:00
} break ;
case FocusOut : {
DEBUG_LOG_X11 ( " [%u] FocusOut window=%lu (%u), mode='%u' \n " , frame , event . xfocus . window , window_id , event . xfocus . mode ) ;
WindowData & wd = windows [ window_id ] ;
2023-02-06 08:18:16 +01:00
if ( ime_window_event | | ( event . xfocus . detail = = NotifyInferior ) ) {
2022-12-11 00:21:22 +01:00
break ;
}
if ( wd . ime_active ) {
2020-10-08 17:21:51 +02:00
MutexLock mutex_lock ( events_mutex ) ;
XUnsetICFocus ( wd . xic ) ;
2022-12-11 00:21:22 +01:00
XUnmapWindow ( x11_display , wd . x11_xim_window ) ;
wd . ime_active = false ;
im_text = String ( ) ;
im_selection = Vector2i ( ) ;
OS_Unix : : get_singleton ( ) - > get_main_loop ( ) - > notification ( MainLoop : : NOTIFICATION_OS_IME_UPDATE ) ;
2020-10-08 17:21:51 +02:00
}
2022-12-11 00:21:22 +01:00
wd . focused = false ;
2020-10-08 17:21:51 +02:00
2020-04-28 15:19:37 +02:00
Input : : get_singleton ( ) - > release_pressed_events ( ) ;
2020-08-22 13:01:49 +02:00
_send_window_event ( wd , WINDOW_EVENT_FOCUS_OUT ) ;
2020-03-01 23:14:37 +01:00
if ( mouse_mode_grab ) {
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
Fix various typos
Found via `codespell -q 3 -S ./thirdparty,*.po,./DONORS.md -L ackward,ang,ans,ba,beng,cas,childs,childrens,dof,doubleclick,expct,fave,findn,gird,hist,inh,inout,leapyear,lod,nd,numer,ois,ony,paket,ro,seeked,sinc,switchs,te,uint,varn,vew`
2022-01-02 19:47:52 +01:00
//dear X11, I try, I really try, but you never work, you do whatever you want.
2020-03-01 23:14:37 +01:00
if ( mouse_mode = = MOUSE_MODE_CAPTURED ) {
// Show the cursor if we're in captured mode so it doesn't look weird.
2021-08-09 22:13:42 +02:00
XUndefineCursor ( x11_display , E . value . x11_window ) ;
2020-03-01 23:14:37 +01:00
}
}
XUngrabPointer ( x11_display , CurrentTime ) ;
}
# ifdef TOUCH_ENABLED
// Ungrab touch devices so input works as usual while we are unfocused
/*for (int i = 0; i < xi.touch_devices.size(); ++i) {
XIUngrabDevice ( x11_display , xi . touch_devices [ i ] , CurrentTime ) ;
} */
// Release every pointer to avoid sticky points
2021-08-09 22:13:42 +02:00
for ( const KeyValue < int , Vector2 > & E : xi . state ) {
2020-03-01 23:14:37 +01:00
Ref < InputEventScreenTouch > st ;
2021-06-18 00:03:09 +02:00
st . instantiate ( ) ;
2021-08-09 22:13:42 +02:00
st - > set_index ( E . key ) ;
2020-03-01 23:14:37 +01:00
st - > set_window_id ( window_id ) ;
2021-08-09 22:13:42 +02:00
st - > set_position ( E . value ) ;
2021-08-13 00:31:16 +02:00
Input : : get_singleton ( ) - > parse_input_event ( st ) ;
2020-03-01 23:14:37 +01:00
}
xi . state . clear ( ) ;
# endif
2020-08-22 13:01:49 +02:00
} break ;
case ConfigureNotify : {
DEBUG_LOG_X11 ( " [%u] ConfigureNotify window=%lu (%u), event=%lu, above=%lu, override_redirect=%u \n " , frame , event . xconfigure . window , window_id , event . xconfigure . event , event . xconfigure . above , event . xconfigure . override_redirect ) ;
2022-12-11 00:21:22 +01:00
if ( event . xconfigure . window = = windows [ window_id ] . x11_xim_window ) {
break ;
}
2020-08-22 13:01:49 +02:00
2020-03-01 23:14:37 +01:00
_window_changed ( & event ) ;
2020-08-22 13:01:49 +02:00
} break ;
2020-03-01 23:14:37 +01:00
case ButtonPress :
case ButtonRelease : {
2023-02-07 11:13:34 +01:00
if ( ime_window_event | | ignore_events ) {
2022-12-11 00:21:22 +01:00
break ;
}
2020-03-01 23:14:37 +01:00
/* exit in case of a mouse button press */
last_timestamp = event . xbutton . time ;
if ( mouse_mode = = MOUSE_MODE_CAPTURED ) {
event . xbutton . x = last_mouse_pos . x ;
event . xbutton . y = last_mouse_pos . y ;
}
Ref < InputEventMouseButton > mb ;
2021-06-18 00:03:09 +02:00
mb . instantiate ( ) ;
2020-03-01 23:14:37 +01:00
mb - > set_window_id ( window_id ) ;
_get_key_modifier_state ( event . xbutton . state , mb ) ;
2021-03-25 21:56:12 +01:00
mb - > set_button_index ( ( MouseButton ) event . xbutton . button ) ;
2021-08-13 23:31:57 +02:00
if ( mb - > get_button_index ( ) = = MouseButton : : RIGHT ) {
mb - > set_button_index ( MouseButton : : MIDDLE ) ;
} else if ( mb - > get_button_index ( ) = = MouseButton : : MIDDLE ) {
mb - > set_button_index ( MouseButton : : RIGHT ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
mb - > set_position ( Vector2 ( event . xbutton . x , event . xbutton . y ) ) ;
mb - > set_global_position ( mb - > get_position ( ) ) ;
mb - > set_pressed ( ( event . type = = ButtonPress ) ) ;
2024-05-27 09:52:59 +02:00
if ( mb - > is_pressed ( ) & & mb - > get_button_index ( ) > = MouseButton : : WHEEL_UP & & mb - > get_button_index ( ) < = MouseButton : : WHEEL_RIGHT ) {
MouseButtonMask mask = mouse_button_to_mask ( mb - > get_button_index ( ) ) ;
BitField < MouseButtonMask > scroll_mask = mouse_get_button_state ( ) ;
scroll_mask . set_flag ( mask ) ;
mb - > set_button_mask ( scroll_mask ) ;
} else {
mb - > set_button_mask ( mouse_get_button_state ( ) ) ;
}
2020-08-29 12:00:44 +02:00
const WindowData & wd = windows [ window_id ] ;
2020-03-01 23:14:37 +01:00
if ( event . type = = ButtonPress ) {
2020-08-22 13:01:49 +02:00
DEBUG_LOG_X11 ( " [%u] ButtonPress window=%lu (%u), button_index=%u \n " , frame , event . xbutton . window , window_id , mb - > get_button_index ( ) ) ;
// Ensure window focus on click.
// RevertToPointerRoot is used to make sure we don't lose all focus in case
// a subwindow and its parent are both destroyed.
2022-02-24 10:21:23 +01:00
if ( ! wd . no_focus & & ! wd . is_popup ) {
2023-12-31 19:53:27 +01:00
_set_input_focus ( wd . x11_window , RevertToPointerRoot ) ;
2020-08-22 13:01:49 +02:00
}
2020-03-01 23:14:37 +01:00
uint64_t diff = OS : : get_singleton ( ) - > get_ticks_usec ( ) / 1000 - last_click_ms ;
if ( mb - > get_button_index ( ) = = last_click_button_index ) {
if ( diff < 400 & & Vector2 ( last_click_pos ) . distance_to ( Vector2 ( event . xbutton . x , event . xbutton . y ) ) < 5 ) {
last_click_ms = 0 ;
last_click_pos = Point2i ( - 100 , - 100 ) ;
2021-08-13 23:31:57 +02:00
last_click_button_index = MouseButton : : NONE ;
2021-04-13 10:25:44 +02:00
mb - > set_double_click ( true ) ;
2020-03-01 23:14:37 +01:00
}
2021-08-13 23:31:57 +02:00
} else if ( mb - > get_button_index ( ) < MouseButton : : WHEEL_UP | | mb - > get_button_index ( ) > MouseButton : : WHEEL_RIGHT ) {
2020-03-01 23:14:37 +01:00
last_click_button_index = mb - > get_button_index ( ) ;
}
2021-04-13 10:25:44 +02:00
if ( ! mb - > is_double_click ( ) ) {
2020-03-01 23:14:37 +01:00
last_click_ms + = diff ;
last_click_pos = Point2i ( event . xbutton . x , event . xbutton . y ) ;
}
2020-08-22 13:01:49 +02:00
} else {
DEBUG_LOG_X11 ( " [%u] ButtonRelease window=%lu (%u), button_index=%u \n " , frame , event . xbutton . window , window_id , mb - > get_button_index ( ) ) ;
2020-08-29 12:00:44 +02:00
2022-12-05 20:50:20 +01:00
WindowID window_id_other = INVALID_WINDOW_ID ;
Window wd_other_x11_window ;
if ( wd . focused ) {
// Handle cases where an unfocused popup is open that needs to receive button-up events.
WindowID popup_id = _get_focused_window_or_popup ( ) ;
if ( popup_id ! = INVALID_WINDOW_ID & & popup_id ! = window_id ) {
window_id_other = popup_id ;
wd_other_x11_window = windows [ popup_id ] . x11_window ;
}
} else {
2020-08-29 12:00:44 +02:00
// Propagate the event to the focused window,
// because it's received only on the topmost window.
// Note: This is needed for drag & drop to work between windows,
// because the engine expects events to keep being processed
// on the same window dragging started.
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
2022-12-05 20:50:20 +01:00
if ( E . value . focused ) {
if ( E . key ! = window_id ) {
window_id_other = E . key ;
wd_other_x11_window = E . value . x11_window ;
2020-08-29 12:00:44 +02:00
}
break ;
}
}
}
2022-12-05 20:50:20 +01:00
if ( window_id_other ! = INVALID_WINDOW_ID ) {
int x , y ;
Window child ;
XTranslateCoordinates ( x11_display , wd . x11_window , wd_other_x11_window , event . xbutton . x , event . xbutton . y , & x , & y , & child ) ;
mb - > set_window_id ( window_id_other ) ;
mb - > set_position ( Vector2 ( x , y ) ) ;
mb - > set_global_position ( mb - > get_position ( ) ) ;
}
2020-03-01 23:14:37 +01:00
}
2021-08-13 00:31:16 +02:00
Input : : get_singleton ( ) - > parse_input_event ( mb ) ;
2020-03-01 23:14:37 +01:00
} break ;
case MotionNotify : {
2023-02-07 11:13:34 +01:00
if ( ime_window_event | | ignore_events ) {
2022-12-11 00:21:22 +01:00
break ;
}
2020-03-01 23:14:37 +01:00
// The X11 API requires filtering one-by-one through the motion
// notify events, in order to figure out which event is the one
// generated by warping the mouse pointer.
2022-09-07 19:31:57 +02:00
WindowID focused_window_id = _get_focused_window_or_popup ( ) ;
if ( ! windows . has ( focused_window_id ) ) {
focused_window_id = MAIN_WINDOW_ID ;
}
2020-03-01 23:14:37 +01:00
while ( true ) {
2022-02-01 03:49:51 +01:00
if ( mouse_mode = = MOUSE_MODE_CAPTURED & & event . xmotion . x = = windows [ focused_window_id ] . size . width / 2 & & event . xmotion . y = = windows [ focused_window_id ] . size . height / 2 ) {
2020-03-01 23:14:37 +01:00
//this is likely the warp event since it was warped here
center = Vector2 ( event . xmotion . x , event . xmotion . y ) ;
break ;
}
2020-09-09 12:43:20 +02:00
if ( event_index + 1 < events . size ( ) ) {
const XEvent & next_event = events [ event_index + 1 ] ;
if ( next_event . type = = MotionNotify ) {
+ + event_index ;
event = next_event ;
2020-03-01 23:14:37 +01:00
} else {
break ;
}
} else {
break ;
}
}
last_timestamp = event . xmotion . time ;
// Motion is also simple.
// A little hack is in order
// to be able to send relative motion events.
Point2i pos ( event . xmotion . x , event . xmotion . y ) ;
// Avoidance of spurious mouse motion (see handling of touch)
bool filter = false ;
// Adding some tolerance to match better Point2i to Vector2
if ( xi . state . size ( ) & & Vector2 ( pos ) . distance_squared_to ( xi . mouse_pos_to_filter ) < 2 ) {
filter = true ;
}
// Invalidate to avoid filtering a possible legitimate similar event coming later
xi . mouse_pos_to_filter = Vector2 ( 1e10 , 1e10 ) ;
if ( filter ) {
break ;
}
2020-08-29 12:00:44 +02:00
const WindowData & wd = windows [ window_id ] ;
bool focused = wd . focused ;
2020-03-01 23:14:37 +01:00
if ( mouse_mode = = MOUSE_MODE_CAPTURED ) {
if ( xi . relative_motion . x = = 0 & & xi . relative_motion . y = = 0 ) {
break ;
}
Point2i new_center = pos ;
pos = last_mouse_pos + xi . relative_motion ;
center = new_center ;
2020-08-29 12:00:44 +02:00
do_mouse_warp = focused ; // warp the cursor if we're focused in
2020-03-01 23:14:37 +01:00
}
if ( ! last_mouse_pos_valid ) {
last_mouse_pos = pos ;
last_mouse_pos_valid = true ;
}
// Hackish but relative mouse motion is already handled in the RawMotion event.
// RawMotion does not provide the absolute mouse position (whereas MotionNotify does).
// Therefore, RawMotion cannot be the authority on absolute mouse position.
// RawMotion provides more precision than MotionNotify, which doesn't sense subpixel motion.
// Therefore, MotionNotify cannot be the authority on relative mouse motion.
// This means we need to take a combined approach...
Point2i rel ;
// Only use raw input if in capture mode. Otherwise use the classic behavior.
if ( mouse_mode = = MOUSE_MODE_CAPTURED ) {
rel = xi . relative_motion ;
} else {
rel = pos - last_mouse_pos ;
}
// Reset to prevent lingering motion
xi . relative_motion . x = 0 ;
xi . relative_motion . y = 0 ;
if ( mouse_mode = = MOUSE_MODE_CAPTURED ) {
2022-02-01 03:49:51 +01:00
pos = Point2i ( windows [ focused_window_id ] . size . width / 2 , windows [ focused_window_id ] . size . height / 2 ) ;
2020-03-01 23:14:37 +01:00
}
2024-08-01 09:07:07 +02:00
BitField < MouseButtonMask > last_button_state = 0 ;
if ( event . xmotion . state & Button1Mask ) {
last_button_state . set_flag ( MouseButtonMask : : LEFT ) ;
}
if ( event . xmotion . state & Button2Mask ) {
last_button_state . set_flag ( MouseButtonMask : : MIDDLE ) ;
}
if ( event . xmotion . state & Button3Mask ) {
last_button_state . set_flag ( MouseButtonMask : : RIGHT ) ;
}
if ( event . xmotion . state & Button4Mask ) {
last_button_state . set_flag ( MouseButtonMask : : MB_XBUTTON1 ) ;
}
if ( event . xmotion . state & Button5Mask ) {
last_button_state . set_flag ( MouseButtonMask : : MB_XBUTTON2 ) ;
}
2020-03-01 23:14:37 +01:00
Ref < InputEventMouseMotion > mm ;
2021-06-18 00:03:09 +02:00
mm . instantiate ( ) ;
2020-03-01 23:14:37 +01:00
mm - > set_window_id ( window_id ) ;
2020-05-03 14:38:58 +02:00
if ( xi . pressure_supported ) {
mm - > set_pressure ( xi . pressure ) ;
} else {
2024-08-01 09:07:07 +02:00
mm - > set_pressure ( bool ( last_button_state . has_flag ( MouseButtonMask : : LEFT ) ) ? 1.0f : 0.0f ) ;
2020-05-03 14:38:58 +02:00
}
2020-03-01 23:14:37 +01:00
mm - > set_tilt ( xi . tilt ) ;
2022-06-19 11:58:24 +02:00
mm - > set_pen_inverted ( xi . pen_inverted ) ;
2020-03-01 23:14:37 +01:00
_get_key_modifier_state ( event . xmotion . state , mm ) ;
2024-08-01 09:07:07 +02:00
mm - > set_button_mask ( last_button_state ) ;
2020-08-29 12:00:44 +02:00
mm - > set_position ( pos ) ;
mm - > set_global_position ( pos ) ;
2021-12-29 14:22:22 +01:00
mm - > set_velocity ( Input : : get_singleton ( ) - > get_last_mouse_velocity ( ) ) ;
2023-10-04 19:20:01 +02:00
mm - > set_screen_velocity ( mm - > get_velocity ( ) ) ;
2020-03-01 23:14:37 +01:00
mm - > set_relative ( rel ) ;
2023-10-04 19:20:01 +02:00
mm - > set_relative_screen_position ( rel ) ;
2020-03-01 23:14:37 +01:00
last_mouse_pos = pos ;
// printf("rel: %d,%d\n", rel.x, rel.y );
// Don't propagate the motion event unless we have focus
// this is so that the relative motion doesn't get messed up
// after we regain focus.
2020-08-29 12:00:44 +02:00
if ( focused ) {
2021-08-13 00:31:16 +02:00
Input : : get_singleton ( ) - > parse_input_event ( mm ) ;
2020-08-29 12:00:44 +02:00
} else {
// Propagate the event to the focused window,
// because it's received only on the topmost window.
// Note: This is needed for drag & drop to work between windows,
// because the engine expects events to keep being processed
// on the same window dragging started.
2021-08-09 22:13:42 +02:00
for ( const KeyValue < WindowID , WindowData > & E : windows ) {
const WindowData & wd_other = E . value ;
2020-08-29 12:00:44 +02:00
if ( wd_other . focused ) {
int x , y ;
Window child ;
XTranslateCoordinates ( x11_display , wd . x11_window , wd_other . x11_window , event . xmotion . x , event . xmotion . y , & x , & y , & child ) ;
Point2i pos_focused ( x , y ) ;
2021-08-09 22:13:42 +02:00
mm - > set_window_id ( E . key ) ;
2020-08-29 12:00:44 +02:00
mm - > set_position ( pos_focused ) ;
mm - > set_global_position ( pos_focused ) ;
2021-12-29 14:22:22 +01:00
mm - > set_velocity ( Input : : get_singleton ( ) - > get_last_mouse_velocity ( ) ) ;
2021-08-13 00:31:16 +02:00
Input : : get_singleton ( ) - > parse_input_event ( mm ) ;
2020-08-29 12:00:44 +02:00
break ;
}
}
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
} break ;
case KeyPress :
case KeyRelease : {
2023-02-07 11:13:34 +01:00
if ( ignore_events ) {
break ;
}
2021-10-27 19:08:09 +02:00
# ifdef DISPLAY_SERVER_X11_DEBUG_LOGS_ENABLED
if ( event . type = = KeyPress ) {
DEBUG_LOG_X11 ( " [%u] KeyPress window=%lu (%u), keycode=%u, time=%lu \n " , frame , event . xkey . window , window_id , event . xkey . keycode , event . xkey . time ) ;
} else {
DEBUG_LOG_X11 ( " [%u] KeyRelease window=%lu (%u), keycode=%u, time=%lu \n " , frame , event . xkey . window , window_id , event . xkey . keycode , event . xkey . time ) ;
}
# endif
2020-03-01 23:14:37 +01:00
last_timestamp = event . xkey . time ;
// key event is a little complex, so
// it will be handled in its own function.
2021-10-27 19:08:09 +02:00
_handle_key_event ( window_id , & event . xkey , events , event_index ) ;
2020-03-01 23:14:37 +01:00
} break ;
case SelectionNotify :
2022-12-11 00:21:22 +01:00
if ( ime_window_event ) {
break ;
}
2020-03-01 23:14:37 +01:00
if ( event . xselection . target = = requested ) {
Property p = _read_property ( x11_display , windows [ window_id ] . x11_window , XInternAtom ( x11_display , " PRIMARY " , 0 ) ) ;
2022-10-21 01:40:32 +02:00
Vector < String > files = String ( ( char * ) p . data ) . split ( " \r \n " , false ) ;
2022-03-04 10:50:24 +01:00
XFree ( p . data ) ;
2020-03-01 23:14:37 +01:00
for ( int i = 0 ; i < files . size ( ) ; i + + ) {
2022-10-21 01:40:32 +02:00
files . write [ i ] = files [ i ] . replace ( " file:// " , " " ) . uri_decode ( ) ;
2020-03-01 23:14:37 +01:00
}
2020-03-04 17:36:09 +01:00
2024-04-27 11:56:39 +02:00
if ( windows [ window_id ] . drop_files_callback . is_valid ( ) ) {
2024-03-01 08:52:21 +01:00
Variant v_files = files ;
const Variant * v_args [ 1 ] = { & v_files } ;
Variant ret ;
Callable : : CallError ce ;
windows [ window_id ] . drop_files_callback . callp ( ( const Variant * * ) & v_args , 1 , ret , ce ) ;
if ( ce . error ! = Callable : : CallError : : CALL_OK ) {
ERR_PRINT ( vformat ( " Failed to execute drop files callback: %s. " , Variant : : get_callable_error_text ( windows [ window_id ] . drop_files_callback , v_args , 1 , ce ) ) ) ;
}
2020-03-04 17:36:09 +01:00
}
2020-03-01 23:14:37 +01:00
//Reply that all is well.
XClientMessageEvent m ;
memset ( & m , 0 , sizeof ( m ) ) ;
m . type = ClientMessage ;
m . display = x11_display ;
m . window = xdnd_source_window ;
m . message_type = xdnd_finished ;
m . format = 32 ;
m . data . l [ 0 ] = windows [ window_id ] . x11_window ;
m . data . l [ 1 ] = 1 ;
m . data . l [ 2 ] = xdnd_action_copy ; //We only ever copy.
XSendEvent ( x11_display , xdnd_source_window , False , NoEventMask , ( XEvent * ) & m ) ;
}
break ;
case ClientMessage :
2022-12-11 00:21:22 +01:00
if ( ime_window_event ) {
break ;
}
2020-03-04 17:36:09 +01:00
if ( ( unsigned int ) event . xclient . data . l [ 0 ] = = ( unsigned int ) wm_delete ) {
_send_window_event ( windows [ window_id ] , WINDOW_EVENT_CLOSE_REQUEST ) ;
}
2020-03-01 23:14:37 +01:00
else if ( ( unsigned int ) event . xclient . message_type = = ( unsigned int ) xdnd_enter ) {
//File(s) have been dragged over the window, check for supported target (text/uri-list)
xdnd_version = ( event . xclient . data . l [ 1 ] > > 24 ) ;
Window source = event . xclient . data . l [ 0 ] ;
bool more_than_3 = event . xclient . data . l [ 1 ] & 1 ;
if ( more_than_3 ) {
Property p = _read_property ( x11_display , source , XInternAtom ( x11_display , " XdndTypeList " , False ) ) ;
requested = pick_target_from_list ( x11_display , ( Atom * ) p . data , p . nitems ) ;
2022-03-04 10:50:24 +01:00
XFree ( p . data ) ;
2020-05-14 16:41:43 +02:00
} else {
2020-03-01 23:14:37 +01:00
requested = pick_target_from_atoms ( x11_display , event . xclient . data . l [ 2 ] , event . xclient . data . l [ 3 ] , event . xclient . data . l [ 4 ] ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
} else if ( ( unsigned int ) event . xclient . message_type = = ( unsigned int ) xdnd_position ) {
//xdnd position event, reply with an XDND status message
//just depending on type of data for now
XClientMessageEvent m ;
memset ( & m , 0 , sizeof ( m ) ) ;
m . type = ClientMessage ;
m . display = event . xclient . display ;
m . window = event . xclient . data . l [ 0 ] ;
m . message_type = xdnd_status ;
m . format = 32 ;
m . data . l [ 0 ] = windows [ window_id ] . x11_window ;
m . data . l [ 1 ] = ( requested ! = None ) ;
m . data . l [ 2 ] = 0 ; //empty rectangle
m . data . l [ 3 ] = 0 ;
m . data . l [ 4 ] = xdnd_action_copy ;
XSendEvent ( x11_display , event . xclient . data . l [ 0 ] , False , NoEventMask , ( XEvent * ) & m ) ;
XFlush ( x11_display ) ;
} else if ( ( unsigned int ) event . xclient . message_type = = ( unsigned int ) xdnd_drop ) {
if ( requested ! = None ) {
xdnd_source_window = event . xclient . data . l [ 0 ] ;
2020-05-14 16:41:43 +02:00
if ( xdnd_version > = 1 ) {
2020-03-01 23:14:37 +01:00
XConvertSelection ( x11_display , xdnd_selection , requested , XInternAtom ( x11_display , " PRIMARY " , 0 ) , windows [ window_id ] . x11_window , event . xclient . data . l [ 2 ] ) ;
2020-05-14 16:41:43 +02:00
} else {
2020-03-01 23:14:37 +01:00
XConvertSelection ( x11_display , xdnd_selection , requested , XInternAtom ( x11_display , " PRIMARY " , 0 ) , windows [ window_id ] . x11_window , CurrentTime ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-01 23:14:37 +01:00
} else {
//Reply that we're not interested.
XClientMessageEvent m ;
memset ( & m , 0 , sizeof ( m ) ) ;
m . type = ClientMessage ;
m . display = event . xclient . display ;
m . window = event . xclient . data . l [ 0 ] ;
m . message_type = xdnd_finished ;
m . format = 32 ;
m . data . l [ 0 ] = windows [ window_id ] . x11_window ;
m . data . l [ 1 ] = 0 ;
m . data . l [ 2 ] = None ; //Failed.
XSendEvent ( x11_display , event . xclient . data . l [ 0 ] , False , NoEventMask , ( XEvent * ) & m ) ;
}
}
break ;
default :
break ;
}
}
XFlush ( x11_display ) ;
if ( do_mouse_warp ) {
XWarpPointer ( x11_display , None , windows [ MAIN_WINDOW_ID ] . x11_window ,
0 , 0 , 0 , 0 , ( int ) windows [ MAIN_WINDOW_ID ] . size . width / 2 , ( int ) windows [ MAIN_WINDOW_ID ] . size . height / 2 ) ;
/*
Window root , child ;
int root_x , root_y ;
int win_x , win_y ;
unsigned int mask ;
XQueryPointer ( x11_display , x11_window , & root , & child , & root_x , & root_y , & win_x , & win_y , & mask ) ;
printf ( " Root: %d,%d \n " , root_x , root_y ) ;
printf ( " Win: %d,%d \n " , win_x , win_y ) ;
*/
}
2024-04-23 17:33:33 +02:00
# ifdef DBUS_ENABLED
if ( portal_desktop ) {
portal_desktop - > process_file_dialog_callbacks ( ) ;
}
# endif
2024-04-09 11:47:06 +02:00
_THREAD_SAFE_UNLOCK_
2021-08-13 00:38:41 +02:00
Input : : get_singleton ( ) - > flush_buffered_events ( ) ;
2020-03-01 23:14:37 +01:00
}
void DisplayServerX11 : : release_rendering_thread ( ) {
2021-10-26 17:18:39 +02:00
# if defined(GLES3_ENABLED)
2022-07-06 08:56:47 +02:00
if ( gl_manager ) {
gl_manager - > release_current ( ) ;
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
gl_manager_egl - > release_current ( ) ;
}
2020-11-18 19:11:30 +01:00
# endif
2020-03-01 23:14:37 +01:00
}
2020-03-03 14:36:29 +01:00
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : swap_buffers ( ) {
2021-10-26 17:18:39 +02:00
# if defined(GLES3_ENABLED)
2020-11-18 19:11:30 +01:00
if ( gl_manager ) {
gl_manager - > swap_buffers ( ) ;
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
gl_manager_egl - > swap_buffers ( ) ;
}
2020-11-18 19:11:30 +01:00
# endif
2020-03-01 23:14:37 +01:00
}
2020-03-03 14:36:29 +01:00
void DisplayServerX11 : : _update_context ( WindowData & wd ) {
XClassHint * classHint = XAllocClassHint ( ) ;
if ( classHint ) {
CharString name_str ;
switch ( context ) {
case CONTEXT_EDITOR :
name_str = " Godot_Editor " ;
break ;
case CONTEXT_PROJECTMAN :
name_str = " Godot_ProjectList " ;
break ;
case CONTEXT_ENGINE :
name_str = " Godot_Engine " ;
break ;
}
CharString class_str ;
if ( context = = CONTEXT_ENGINE ) {
String config_name = GLOBAL_GET ( " application/config/name " ) ;
if ( config_name . length ( ) = = 0 ) {
class_str = " Godot_Engine " ;
} else {
class_str = config_name . utf8 ( ) ;
}
} else {
class_str = " Godot " ;
}
classHint - > res_class = class_str . ptrw ( ) ;
classHint - > res_name = name_str . ptrw ( ) ;
XSetClassHint ( x11_display , wd . x11_window , classHint ) ;
XFree ( classHint ) ;
}
}
2020-05-14 14:29:06 +02:00
2020-03-03 14:36:29 +01:00
void DisplayServerX11 : : set_context ( Context p_context ) {
2020-03-07 16:46:50 +01:00
_THREAD_SAFE_METHOD_
2020-03-03 14:36:29 +01:00
context = p_context ;
2021-08-09 22:13:42 +02:00
for ( KeyValue < WindowID , WindowData > & E : windows ) {
_update_context ( E . value ) ;
2020-03-03 14:36:29 +01:00
}
}
2020-05-14 14:29:06 +02:00
2024-05-03 10:48:46 +02:00
bool DisplayServerX11 : : is_window_transparency_available ( ) const {
CharString net_wm_cm_name = vformat ( " _NET_WM_CM_S%d " , XDefaultScreen ( x11_display ) ) . ascii ( ) ;
Atom net_wm_cm = XInternAtom ( x11_display , net_wm_cm_name . get_data ( ) , False ) ;
if ( net_wm_cm = = None ) {
return false ;
}
if ( XGetSelectionOwner ( x11_display , net_wm_cm ) = = None ) {
return false ;
}
2024-05-25 20:08:50 +02:00
# if defined(RD_ENABLED)
2024-05-03 10:48:46 +02:00
if ( rendering_device & & ! rendering_device - > is_composite_alpha_supported ( ) ) {
return false ;
}
2024-05-25 20:08:50 +02:00
# endif
2024-05-03 10:48:46 +02:00
return OS : : get_singleton ( ) - > is_layered_allowed ( ) ;
}
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : set_native_icon ( const String & p_filename ) {
WARN_PRINT ( " Native icon not supported by this display server. " ) ;
}
2020-03-13 00:12:34 +01:00
bool g_set_icon_error = false ;
int set_icon_errorhandler ( Display * dpy , XErrorEvent * ev ) {
g_set_icon_error = true ;
return 0 ;
}
2020-03-01 23:14:37 +01:00
void DisplayServerX11 : : set_icon ( const Ref < Image > & p_icon ) {
2020-03-13 00:12:34 +01:00
_THREAD_SAFE_METHOD_
WindowData & wd = windows [ MAIN_WINDOW_ID ] ;
int ( * oldHandler ) ( Display * , XErrorEvent * ) = XSetErrorHandler ( & set_icon_errorhandler ) ;
Atom net_wm_icon = XInternAtom ( x11_display , " _NET_WM_ICON " , False ) ;
if ( p_icon . is_valid ( ) ) {
2023-06-19 15:18:39 +02:00
ERR_FAIL_COND ( p_icon - > get_width ( ) < = 0 | | p_icon - > get_height ( ) < = 0 ) ;
2020-03-13 00:12:34 +01:00
Ref < Image > img = p_icon - > duplicate ( ) ;
img - > convert ( Image : : FORMAT_RGBA8 ) ;
while ( true ) {
int w = img - > get_width ( ) ;
int h = img - > get_height ( ) ;
if ( g_set_icon_error ) {
g_set_icon_error = false ;
2024-03-20 18:00:54 +01:00
WARN_PRINT ( vformat ( " Icon too large (%dx%d), attempting to downscale icon. " , w , h ) ) ;
2020-03-13 00:12:34 +01:00
int new_width , new_height ;
if ( w > h ) {
new_width = w / 2 ;
new_height = h * new_width / w ;
} else {
new_height = h / 2 ;
new_width = w * new_height / h ;
}
w = new_width ;
h = new_height ;
if ( ! w | | ! h ) {
WARN_PRINT ( " Unable to set icon. " ) ;
break ;
}
img - > resize ( w , h , Image : : INTERPOLATE_CUBIC ) ;
}
// We're using long to have wordsize (32Bit build -> 32 Bits, 64 Bit build -> 64 Bits
Vector < long > pd ;
pd . resize ( 2 + w * h ) ;
pd . write [ 0 ] = w ;
pd . write [ 1 ] = h ;
const uint8_t * r = img - > get_data ( ) . ptr ( ) ;
long * wr = & pd . write [ 2 ] ;
uint8_t const * pr = r ;
for ( int i = 0 ; i < w * h ; i + + ) {
long v = 0 ;
// A R G B
v | = pr [ 3 ] < < 24 | pr [ 0 ] < < 16 | pr [ 1 ] < < 8 | pr [ 2 ] ;
* wr + + = v ;
pr + = 4 ;
}
2020-11-14 21:31:16 +01:00
if ( net_wm_icon ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , net_wm_icon , XA_CARDINAL , 32 , PropModeReplace , ( unsigned char * ) pd . ptr ( ) , pd . size ( ) ) ;
}
2020-03-13 00:12:34 +01:00
2020-05-14 16:41:43 +02:00
if ( ! g_set_icon_error ) {
2020-03-13 00:12:34 +01:00
break ;
2020-05-14 16:41:43 +02:00
}
2020-03-13 00:12:34 +01:00
}
} else {
XDeleteProperty ( x11_display , wd . x11_window , net_wm_icon ) ;
}
XFlush ( x11_display ) ;
XSetErrorHandler ( oldHandler ) ;
2020-03-01 23:14:37 +01:00
}
2021-06-19 17:44:59 +02:00
void DisplayServerX11 : : window_set_vsync_mode ( DisplayServer : : VSyncMode p_vsync_mode , WindowID p_window ) {
_THREAD_SAFE_METHOD_
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
2023-12-19 18:57:56 +01:00
if ( rendering_context ) {
rendering_context - > window_set_vsync_mode ( p_window , p_vsync_mode ) ;
2020-11-18 19:11:30 +01:00
}
# endif
2021-10-26 17:18:39 +02:00
# if defined(GLES3_ENABLED)
2020-11-18 19:11:30 +01:00
if ( gl_manager ) {
2022-11-15 23:14:21 +01:00
gl_manager - > set_use_vsync ( p_vsync_mode ! = DisplayServer : : VSYNC_DISABLED ) ;
2020-11-18 19:11:30 +01:00
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
gl_manager_egl - > set_use_vsync ( p_vsync_mode ! = DisplayServer : : VSYNC_DISABLED ) ;
}
2021-06-19 17:44:59 +02:00
# endif
}
DisplayServer : : VSyncMode DisplayServerX11 : : window_get_vsync_mode ( WindowID p_window ) const {
_THREAD_SAFE_METHOD_
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
2023-12-19 18:57:56 +01:00
if ( rendering_context ) {
return rendering_context - > window_get_vsync_mode ( p_window ) ;
2020-11-18 19:11:30 +01:00
}
# endif
2021-10-26 17:18:39 +02:00
# if defined(GLES3_ENABLED)
2020-11-18 19:11:30 +01:00
if ( gl_manager ) {
return gl_manager - > is_using_vsync ( ) ? DisplayServer : : VSYNC_ENABLED : DisplayServer : : VSYNC_DISABLED ;
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
return gl_manager_egl - > is_using_vsync ( ) ? DisplayServer : : VSYNC_ENABLED : DisplayServer : : VSYNC_DISABLED ;
}
2021-06-19 17:44:59 +02:00
# endif
2020-11-18 19:11:30 +01:00
return DisplayServer : : VSYNC_ENABLED ;
2021-06-19 17:44:59 +02:00
}
2020-03-03 14:36:29 +01:00
Vector < String > DisplayServerX11 : : get_rendering_drivers_func ( ) {
Vector < String > drivers ;
# ifdef VULKAN_ENABLED
drivers . push_back ( " vulkan " ) ;
# endif
2021-10-26 17:18:39 +02:00
# ifdef GLES3_ENABLED
drivers . push_back ( " opengl3 " ) ;
2023-09-22 09:55:55 +02:00
drivers . push_back ( " opengl3_es " ) ;
2020-03-03 14:36:29 +01:00
# endif
return drivers ;
}
2024-05-22 14:06:04 +02:00
DisplayServer * DisplayServerX11 : : create_func ( const String & p_rendering_driver , WindowMode p_mode , VSyncMode p_vsync_mode , uint32_t p_flags , const Vector2i * p_position , const Vector2i & p_resolution , int p_screen , Context p_context , Error & r_error ) {
DisplayServer * ds = memnew ( DisplayServerX11 ( p_rendering_driver , p_mode , p_vsync_mode , p_flags , p_position , p_resolution , p_screen , p_context , r_error ) ) ;
2020-07-13 18:24:04 +02:00
if ( r_error ! = OK ) {
2022-10-17 19:47:32 +02:00
if ( p_rendering_driver = = " vulkan " ) {
String executable_name = OS : : get_singleton ( ) - > get_executable_path ( ) . get_file ( ) ;
2023-01-17 15:26:10 +01:00
OS : : get_singleton ( ) - > alert (
vformat ( " Your video card drivers seem not to support the required Vulkan version. \n \n "
" If possible, consider updating your video card drivers or using the OpenGL 3 driver. \n \n "
" You can enable the OpenGL 3 driver by starting the engine from the \n "
2023-03-08 22:27:42 +01:00
" command line with the command: \n \n \" %s \" --rendering-driver opengl3 \n \n "
2023-01-17 15:26:10 +01:00
" If you recently updated your video card drivers, try rebooting. " ,
executable_name ) ,
" Unable to initialize Vulkan video driver " ) ;
2022-10-17 19:47:32 +02:00
} else {
2023-01-17 15:26:10 +01:00
OS : : get_singleton ( ) - > alert (
" Your video card drivers seem not to support the required OpenGL 3.3 version. \n \n "
" If possible, consider updating your video card drivers. \n \n "
" If you recently updated your video card drivers, try rebooting. " ,
" Unable to initialize OpenGL video driver " ) ;
2022-10-17 19:47:32 +02:00
}
2020-07-13 18:24:04 +02:00
}
return ds ;
2020-03-01 23:14:37 +01:00
}
2023-01-04 23:00:02 +01:00
DisplayServerX11 : : WindowID DisplayServerX11 : : _create_window ( WindowMode p_mode , VSyncMode p_vsync_mode , uint32_t p_flags , const Rect2i & p_rect ) {
2020-03-01 23:14:37 +01:00
//Create window
2022-09-03 12:40:53 +02:00
XVisualInfo visualInfo ;
bool vi_selected = false ;
2020-03-01 23:14:37 +01:00
2022-09-03 12:40:53 +02:00
# ifdef GLES3_ENABLED
if ( gl_manager ) {
2023-03-07 18:24:05 +01:00
Error err ;
visualInfo = gl_manager - > get_vi ( x11_display , err ) ;
ERR_FAIL_COND_V_MSG ( err ! = OK , INVALID_WINDOW_ID , " Can't acquire visual info from display. " ) ;
2022-09-03 12:40:53 +02:00
vi_selected = true ;
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
XVisualInfo visual_info_template ;
int visual_id = gl_manager_egl - > display_get_native_visual_id ( x11_display ) ;
ERR_FAIL_COND_V_MSG ( visual_id < 0 , INVALID_WINDOW_ID , " Unable to get a visual id. " ) ;
visual_info_template . visualid = ( VisualID ) visual_id ;
int number_of_visuals = 0 ;
XVisualInfo * vi_list = XGetVisualInfo ( x11_display , VisualIDMask , & visual_info_template , & number_of_visuals ) ;
ERR_FAIL_COND_V ( number_of_visuals < = 0 , INVALID_WINDOW_ID ) ;
visualInfo = vi_list [ 0 ] ;
XFree ( vi_list ) ;
}
2022-09-03 12:40:53 +02:00
# endif
if ( ! vi_selected ) {
long visualMask = VisualScreenMask ;
int numberOfVisuals ;
XVisualInfo vInfoTemplate = { } ;
vInfoTemplate . screen = DefaultScreen ( x11_display ) ;
XVisualInfo * vi_list = XGetVisualInfo ( x11_display , visualMask , & vInfoTemplate , & numberOfVisuals ) ;
2023-09-09 17:46:44 +02:00
ERR_FAIL_NULL_V ( vi_list , INVALID_WINDOW_ID ) ;
2022-09-03 12:40:53 +02:00
visualInfo = vi_list [ 0 ] ;
if ( OS : : get_singleton ( ) - > is_layered_allowed ( ) ) {
for ( int i = 0 ; i < numberOfVisuals ; i + + ) {
XRenderPictFormat * pict_format = XRenderFindVisualFormat ( x11_display , vi_list [ i ] . visual ) ;
if ( ! pict_format ) {
continue ;
}
visualInfo = vi_list [ i ] ;
if ( pict_format - > direct . alphaMask > 0 ) {
break ;
}
}
}
XFree ( vi_list ) ;
}
Colormap colormap = XCreateColormap ( x11_display , RootWindow ( x11_display , visualInfo . screen ) , visualInfo . visual , AllocNone ) ;
2020-03-01 23:14:37 +01:00
XSetWindowAttributes windowAttributes = { } ;
windowAttributes . colormap = colormap ;
windowAttributes . background_pixel = 0xFFFFFFFF ;
windowAttributes . border_pixel = 0 ;
windowAttributes . event_mask = KeyPressMask | KeyReleaseMask | StructureNotifyMask | ExposureMask ;
unsigned long valuemask = CWBorderPixel | CWColormap | CWEventMask ;
2022-09-03 12:40:53 +02:00
if ( OS : : get_singleton ( ) - > is_layered_allowed ( ) ) {
windowAttributes . background_pixmap = None ;
windowAttributes . background_pixel = 0 ;
windowAttributes . border_pixmap = None ;
valuemask | = CWBackPixel ;
}
2020-08-22 13:01:49 +02:00
WindowID id = window_id_counter + + ;
WindowData & wd = windows [ id ] ;
if ( p_flags & WINDOW_FLAG_NO_FOCUS_BIT ) {
wd . no_focus = true ;
}
2022-02-24 10:21:23 +01:00
if ( p_flags & WINDOW_FLAG_POPUP_BIT ) {
wd . is_popup = true ;
}
2020-08-22 13:01:49 +02:00
// Setup for menu subwindows:
// - override_redirect forces the WM not to interfere with the window, to avoid delays due to
// handling decorations and placement.
// On the other hand, focus changes need to be handled manually when this is set.
// - save_under is a hint for the WM to keep the content of windows behind to avoid repaint.
2023-02-01 09:51:03 +01:00
if ( wd . no_focus ) {
2020-08-22 13:01:49 +02:00
windowAttributes . override_redirect = True ;
windowAttributes . save_under = True ;
valuemask | = CWOverrideRedirect | CWSaveUnder ;
}
2023-01-04 23:00:02 +01:00
int rq_screen = get_screen_from_rect ( p_rect ) ;
if ( rq_screen < 0 ) {
rq_screen = get_primary_screen ( ) ; // Requested window rect is outside any screen bounds.
}
2022-12-08 19:34:15 +01:00
Rect2i win_rect = p_rect ;
if ( p_mode = = WINDOW_MODE_FULLSCREEN | | p_mode = = WINDOW_MODE_EXCLUSIVE_FULLSCREEN ) {
2023-01-04 23:00:02 +01:00
Rect2i screen_rect = Rect2i ( screen_get_position ( rq_screen ) , screen_get_size ( rq_screen ) ) ;
2022-12-08 19:34:15 +01:00
win_rect = screen_rect ;
} else {
2023-01-04 23:00:02 +01:00
Rect2i srect = screen_get_usable_rect ( rq_screen ) ;
Point2i wpos = p_rect . position ;
2024-03-03 12:49:08 +01:00
wpos = wpos . clamp ( srect . position , srect . position + srect . size - p_rect . size / 3 ) ;
2022-12-08 19:34:15 +01:00
win_rect . position = wpos ;
}
2023-02-14 12:15:14 +01:00
// Position and size hints are set from these values before they are updated to the actual
// window size, so we need to initialize them here.
wd . position = win_rect . position ;
wd . size = win_rect . size ;
2020-03-01 23:14:37 +01:00
{
2022-12-08 19:34:15 +01:00
wd . x11_window = XCreateWindow ( x11_display , RootWindow ( x11_display , visualInfo . screen ) , win_rect . position . x , win_rect . position . y , win_rect . size . width > 0 ? win_rect . size . width : 1 , win_rect . size . height > 0 ? win_rect . size . height : 1 , 0 , visualInfo . depth , InputOutput , visualInfo . visual , valuemask , & windowAttributes ) ;
2023-01-25 11:31:42 +01:00
2023-04-13 20:06:43 +02:00
wd . parent = RootWindow ( x11_display , visualInfo . screen ) ;
2023-01-25 11:31:42 +01:00
XSetWindowAttributes window_attributes_ime = { } ;
window_attributes_ime . event_mask = KeyPressMask | KeyReleaseMask | StructureNotifyMask | ExposureMask ;
wd . x11_xim_window = XCreateWindow ( x11_display , wd . x11_window , 0 , 0 , 1 , 1 , 0 , CopyFromParent , InputOnly , CopyFromParent , CWEventMask , & window_attributes_ime ) ;
2023-02-15 13:13:56 +01:00
# ifdef XKB_ENABLED
2023-03-16 09:44:47 +01:00
if ( dead_tbl & & xkb_loaded_v05p ) {
2023-01-30 09:59:49 +01:00
wd . xkb_state = xkb_compose_state_new ( dead_tbl , XKB_COMPOSE_STATE_NO_FLAGS ) ;
}
2023-02-15 13:13:56 +01:00
# endif
2020-08-22 13:01:49 +02:00
// Enable receiving notification when the window is initialized (MapNotify)
// so the focus can be set at the right time.
2022-02-24 10:21:23 +01:00
if ( ! wd . no_focus & & ! wd . is_popup ) {
2020-08-22 13:01:49 +02:00
XSelectInput ( x11_display , wd . x11_window , StructureNotifyMask ) ;
}
2020-03-01 23:14:37 +01:00
//associate PID
// make PID known to X11
{
const long pid = OS : : get_singleton ( ) - > get_process_id ( ) ;
Atom net_wm_pid = XInternAtom ( x11_display , " _NET_WM_PID " , False ) ;
2020-11-14 21:31:16 +01:00
if ( net_wm_pid ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , net_wm_pid , XA_CARDINAL , 32 , PropModeReplace , ( unsigned char * ) & pid , 1 ) ;
}
2020-03-01 23:14:37 +01:00
}
long im_event_mask = 0 ;
{
XIEventMask all_event_mask ;
XSetWindowAttributes new_attr ;
new_attr . event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask |
2021-10-28 15:19:35 +02:00
ButtonReleaseMask | EnterWindowMask |
LeaveWindowMask | PointerMotionMask |
Button1MotionMask |
Button2MotionMask | Button3MotionMask |
Button4MotionMask | Button5MotionMask |
ButtonMotionMask | KeymapStateMask |
ExposureMask | VisibilityChangeMask |
StructureNotifyMask |
SubstructureNotifyMask | SubstructureRedirectMask |
FocusChangeMask | PropertyChangeMask |
ColormapChangeMask | OwnerGrabButtonMask |
im_event_mask ;
2020-03-01 23:14:37 +01:00
XChangeWindowAttributes ( x11_display , wd . x11_window , CWEventMask , & new_attr ) ;
static unsigned char all_mask_data [ XIMaskLen ( XI_LASTEVENT ) ] = { } ;
all_event_mask . deviceid = XIAllDevices ;
all_event_mask . mask_len = sizeof ( all_mask_data ) ;
all_event_mask . mask = all_mask_data ;
XISetMask ( all_event_mask . mask , XI_HierarchyChanged ) ;
# ifdef TOUCH_ENABLED
if ( xi . touch_devices . size ( ) ) {
XISetMask ( all_event_mask . mask , XI_TouchBegin ) ;
XISetMask ( all_event_mask . mask , XI_TouchUpdate ) ;
XISetMask ( all_event_mask . mask , XI_TouchEnd ) ;
XISetMask ( all_event_mask . mask , XI_TouchOwnership ) ;
}
# endif
XISelectEvents ( x11_display , wd . x11_window , & all_event_mask , 1 ) ;
}
/* set the titlebar name */
XStoreName ( x11_display , wd . x11_window , " Godot " ) ;
XSetWMProtocols ( x11_display , wd . x11_window , & wm_delete , 1 ) ;
2020-11-14 21:31:16 +01:00
if ( xdnd_aware ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , xdnd_aware , XA_ATOM , 32 , PropModeReplace , ( unsigned char * ) & xdnd_version , 1 ) ;
}
2020-03-01 23:14:37 +01:00
if ( xim & & xim_style ) {
2020-09-09 12:43:20 +02:00
// Block events polling while changing input focus
// because it triggers some event polling internally.
MutexLock mutex_lock ( events_mutex ) ;
2022-12-11 00:21:22 +01:00
// Force on-the-spot for the over-the-spot style.
if ( ( xim_style & XIMPreeditPosition ) ! = 0 ) {
xim_style & = ~ XIMPreeditPosition ;
xim_style | = XIMPreeditCallbacks ;
}
if ( ( xim_style & XIMPreeditCallbacks ) ! = 0 ) {
: : XIMCallback preedit_start_callback ;
preedit_start_callback . client_data = ( : : XPointer ) ( this ) ;
preedit_start_callback . callback = ( : : XIMProc ) ( void * ) ( _xim_preedit_start_callback ) ;
: : XIMCallback preedit_done_callback ;
preedit_done_callback . client_data = ( : : XPointer ) ( this ) ;
preedit_done_callback . callback = ( : : XIMProc ) ( _xim_preedit_done_callback ) ;
: : XIMCallback preedit_draw_callback ;
preedit_draw_callback . client_data = ( : : XPointer ) ( this ) ;
preedit_draw_callback . callback = ( : : XIMProc ) ( _xim_preedit_draw_callback ) ;
: : XIMCallback preedit_caret_callback ;
preedit_caret_callback . client_data = ( : : XPointer ) ( this ) ;
preedit_caret_callback . callback = ( : : XIMProc ) ( _xim_preedit_caret_callback ) ;
: : XVaNestedList preedit_attributes = XVaCreateNestedList ( 0 ,
XNPreeditStartCallback , & preedit_start_callback ,
XNPreeditDoneCallback , & preedit_done_callback ,
XNPreeditDrawCallback , & preedit_draw_callback ,
XNPreeditCaretCallback , & preedit_caret_callback ,
( char * ) nullptr ) ;
wd . xic = XCreateIC ( xim ,
XNInputStyle , xim_style ,
XNClientWindow , wd . x11_xim_window ,
XNFocusWindow , wd . x11_xim_window ,
XNPreeditAttributes , preedit_attributes ,
( char * ) nullptr ) ;
XFree ( preedit_attributes ) ;
} else {
wd . xic = XCreateIC ( xim ,
XNInputStyle , xim_style ,
XNClientWindow , wd . x11_xim_window ,
XNFocusWindow , wd . x11_xim_window ,
( char * ) nullptr ) ;
}
2020-04-02 01:20:12 +02:00
if ( XGetICValues ( wd . xic , XNFilterEvents , & im_event_mask , nullptr ) ! = nullptr ) {
2020-03-01 23:14:37 +01:00
WARN_PRINT ( " XGetICValues couldn't obtain XNFilterEvents value " ) ;
XDestroyIC ( wd . xic ) ;
2020-04-02 01:20:12 +02:00
wd . xic = nullptr ;
2020-03-01 23:14:37 +01:00
}
if ( wd . xic ) {
XUnsetICFocus ( wd . xic ) ;
} else {
WARN_PRINT ( " XCreateIC couldn't create wd.xic " ) ;
}
} else {
2020-04-02 01:20:12 +02:00
wd . xic = nullptr ;
2020-03-01 23:14:37 +01:00
WARN_PRINT ( " XCreateIC couldn't create wd.xic " ) ;
}
2020-03-03 14:36:29 +01:00
_update_context ( wd ) ;
2020-08-22 13:01:49 +02:00
if ( p_flags & WINDOW_FLAG_BORDERLESS_BIT ) {
Hints hints ;
Atom property ;
hints . flags = 2 ;
hints . decorations = 0 ;
property = XInternAtom ( x11_display , " _MOTIF_WM_HINTS " , True ) ;
2020-11-14 21:31:16 +01:00
if ( property ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , property , property , 32 , PropModeReplace , ( unsigned char * ) & hints , 5 ) ;
}
2020-08-22 13:01:49 +02:00
}
2020-03-25 15:16:19 +01:00
2022-02-24 10:21:23 +01:00
if ( wd . is_popup | | wd . no_focus ) {
2020-08-22 13:01:49 +02:00
// Set Utility type to disable fade animations.
Atom type_atom = XInternAtom ( x11_display , " _NET_WM_WINDOW_TYPE_UTILITY " , False ) ;
Atom wt_atom = XInternAtom ( x11_display , " _NET_WM_WINDOW_TYPE " , False ) ;
2020-11-14 21:31:16 +01:00
if ( wt_atom ! = None & & type_atom ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , wt_atom , XA_ATOM , 32 , PropModeReplace , ( unsigned char * ) & type_atom , 1 ) ;
}
2020-08-22 13:01:49 +02:00
} else {
Atom type_atom = XInternAtom ( x11_display , " _NET_WM_WINDOW_TYPE_NORMAL " , False ) ;
Atom wt_atom = XInternAtom ( x11_display , " _NET_WM_WINDOW_TYPE " , False ) ;
2020-03-25 15:16:19 +01:00
2020-11-14 21:31:16 +01:00
if ( wt_atom ! = None & & type_atom ! = None ) {
XChangeProperty ( x11_display , wd . x11_window , wt_atom , XA_ATOM , 32 , PropModeReplace , ( unsigned char * ) & type_atom , 1 ) ;
}
2020-03-25 15:16:19 +01:00
}
2020-07-31 17:27:16 +02:00
_update_size_hints ( id ) ;
2020-03-13 00:12:34 +01:00
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
2023-12-19 18:57:56 +01:00
if ( rendering_context ) {
2023-12-19 12:48:02 +01:00
union {
# ifdef VULKAN_ENABLED
2023-12-19 18:57:56 +01:00
RenderingContextDriverVulkanX11 : : WindowPlatformData vulkan ;
2023-12-19 12:48:02 +01:00
# endif
} wpd ;
# ifdef VULKAN_ENABLED
if ( rendering_driver = = " vulkan " ) {
wpd . vulkan . window = wd . x11_window ;
wpd . vulkan . display = x11_display ;
}
# endif
2023-12-19 18:57:56 +01:00
Error err = rendering_context - > window_create ( id , & wpd ) ;
ERR_FAIL_COND_V_MSG ( err ! = OK , INVALID_WINDOW_ID , vformat ( " Can't create a %s window " , rendering_driver ) ) ;
rendering_context - > window_set_size ( id , win_rect . size . width , win_rect . size . height ) ;
rendering_context - > window_set_vsync_mode ( id , p_vsync_mode ) ;
2020-03-13 00:12:34 +01:00
}
# endif
2021-10-26 17:18:39 +02:00
# ifdef GLES3_ENABLED
2020-11-18 19:11:30 +01:00
if ( gl_manager ) {
2022-12-08 19:34:15 +01:00
Error err = gl_manager - > window_create ( id , wd . x11_window , x11_display , win_rect . size . width , win_rect . size . height ) ;
2021-09-27 01:07:10 +02:00
ERR_FAIL_COND_V_MSG ( err ! = OK , INVALID_WINDOW_ID , " Can't create an OpenGL window " ) ;
2020-11-18 19:11:30 +01:00
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
Error err = gl_manager_egl - > window_create ( id , x11_display , & wd . x11_window , win_rect . size . width , win_rect . size . height ) ;
ERR_FAIL_COND_V_MSG ( err ! = OK , INVALID_WINDOW_ID , " Failed to create an OpenGLES window. " ) ;
}
window_set_vsync_mode ( p_vsync_mode , id ) ;
2020-11-18 19:11:30 +01:00
# endif
2020-03-13 00:12:34 +01:00
//set_class_hint(x11_display, wd.x11_window);
XFlush ( x11_display ) ;
XSync ( x11_display , False ) ;
//XSetErrorHandler(oldHandler);
2020-03-01 23:14:37 +01:00
}
window_set_mode ( p_mode , id ) ;
2020-03-06 18:00:16 +01:00
//sync size
{
XWindowAttributes xwa ;
XSync ( x11_display , False ) ;
XGetWindowAttributes ( x11_display , wd . x11_window , & xwa ) ;
2020-03-25 15:16:19 +01:00
wd . position . x = xwa . x ;
wd . position . y = xwa . y ;
2020-03-06 18:00:16 +01:00
wd . size . width = xwa . width ;
wd . size . height = xwa . height ;
}
2020-03-01 23:14:37 +01:00
//set cursor
if ( cursors [ current_cursor ] ! = None ) {
XDefineCursor ( x11_display , wd . x11_window , cursors [ current_cursor ] ) ;
}
2020-08-21 09:39:30 +02:00
2020-03-01 23:14:37 +01:00
return id ;
}
2022-12-11 00:21:22 +01:00
static bool _is_xim_style_supported ( const : : XIMStyle & p_style ) {
const : : XIMStyle supported_preedit = XIMPreeditCallbacks | XIMPreeditPosition | XIMPreeditNothing | XIMPreeditNone ;
const : : XIMStyle supported_status = XIMStatusNothing | XIMStatusNone ;
// Check preedit style is supported
if ( ( p_style & supported_preedit ) = = 0 ) {
return false ;
}
// Check status style is supported
if ( ( p_style & supported_status ) = = 0 ) {
return false ;
}
return true ;
}
static : : XIMStyle _get_best_xim_style ( const : : XIMStyle & p_style_a , const : : XIMStyle & p_style_b ) {
if ( p_style_a = = 0 ) {
return p_style_b ;
}
if ( p_style_b = = 0 ) {
return p_style_a ;
}
const : : XIMStyle preedit = XIMPreeditArea | XIMPreeditCallbacks | XIMPreeditPosition | XIMPreeditNothing | XIMPreeditNone ;
const : : XIMStyle status = XIMStatusArea | XIMStatusCallbacks | XIMStatusNothing | XIMStatusNone ;
: : XIMStyle a = p_style_a & preedit ;
: : XIMStyle b = p_style_b & preedit ;
if ( a ! = b ) {
// Compare preedit styles.
if ( ( a | b ) & XIMPreeditCallbacks ) {
return a = = XIMPreeditCallbacks ? p_style_a : p_style_b ;
} else if ( ( a | b ) & XIMPreeditPosition ) {
return a = = XIMPreeditPosition ? p_style_a : p_style_b ;
} else if ( ( a | b ) & XIMPreeditArea ) {
return a = = XIMPreeditArea ? p_style_a : p_style_b ;
} else if ( ( a | b ) & XIMPreeditNothing ) {
return a = = XIMPreeditNothing ? p_style_a : p_style_b ;
}
} else {
// Preedit styles are the same, compare status styles.
a = p_style_a & status ;
b = p_style_b & status ;
if ( ( a | b ) & XIMStatusCallbacks ) {
return a = = XIMStatusCallbacks ? p_style_a : p_style_b ;
} else if ( ( a | b ) & XIMStatusArea ) {
return a = = XIMStatusArea ? p_style_a : p_style_b ;
} else if ( ( a | b ) & XIMStatusNothing ) {
return a = = XIMStatusNothing ? p_style_a : p_style_b ;
}
}
return p_style_a ;
}
2024-05-22 14:06:04 +02:00
DisplayServerX11 : : DisplayServerX11 ( const String & p_rendering_driver , WindowMode p_mode , VSyncMode p_vsync_mode , uint32_t p_flags , const Vector2i * p_position , const Vector2i & p_resolution , int p_screen , Context p_context , Error & r_error ) {
2022-12-11 00:21:22 +01:00
KeyMappingX11 : : initialize ( ) ;
2024-06-12 09:05:17 +02:00
xwayland = OS : : get_singleton ( ) - > get_environment ( " XDG_SESSION_TYPE " ) . to_lower ( ) = = " wayland " ;
2024-01-19 18:41:01 +01:00
native_menu = memnew ( NativeMenu ) ;
2024-05-22 14:06:04 +02:00
context = p_context ;
2024-01-19 18:41:01 +01:00
2023-02-15 13:13:56 +01:00
# ifdef SOWRAP_ENABLED
2022-11-29 12:03:06 +01:00
# ifdef DEBUG_ENABLED
int dylibloader_verbose = 1 ;
# else
int dylibloader_verbose = 0 ;
# endif
if ( initialize_xlib ( dylibloader_verbose ) ! = 0 ) {
r_error = ERR_UNAVAILABLE ;
ERR_FAIL_MSG ( " Can't load Xlib dynamically. " ) ;
}
if ( initialize_xcursor ( dylibloader_verbose ) ! = 0 ) {
r_error = ERR_UNAVAILABLE ;
ERR_FAIL_MSG ( " Can't load XCursor dynamically. " ) ;
}
2023-02-15 13:13:56 +01:00
# ifdef XKB_ENABLED
2023-03-16 09:44:47 +01:00
bool xkb_loaded = ( initialize_xkbcommon ( dylibloader_verbose ) = = 0 ) ;
xkb_loaded_v05p = xkb_loaded ;
if ( ! xkb_context_new | | ! xkb_compose_table_new_from_locale | | ! xkb_compose_table_unref | | ! xkb_context_unref | | ! xkb_compose_state_feed | | ! xkb_compose_state_unref | | ! xkb_compose_state_new | | ! xkb_compose_state_get_status | | ! xkb_compose_state_get_utf8 ) {
xkb_loaded_v05p = false ;
print_verbose ( " Detected XKBcommon library version older than 0.5, dead key composition and Unicode key labels disabled. " ) ;
}
xkb_loaded_v08p = xkb_loaded ;
if ( ! xkb_keysym_to_utf32 | | ! xkb_keysym_to_upper ) {
xkb_loaded_v08p = false ;
print_verbose ( " Detected XKBcommon library version older than 0.8, Unicode key labels disabled. " ) ;
2023-03-02 09:25:50 +01:00
}
2023-02-15 13:13:56 +01:00
# endif
2022-11-29 12:03:06 +01:00
if ( initialize_xext ( dylibloader_verbose ) ! = 0 ) {
r_error = ERR_UNAVAILABLE ;
ERR_FAIL_MSG ( " Can't load Xext dynamically. " ) ;
}
if ( initialize_xinerama ( dylibloader_verbose ) ! = 0 ) {
2023-08-04 09:58:14 +02:00
xinerama_ext_ok = false ;
2022-11-29 12:03:06 +01:00
}
if ( initialize_xrandr ( dylibloader_verbose ) ! = 0 ) {
2023-08-04 09:58:14 +02:00
xrandr_ext_ok = false ;
2022-11-29 12:03:06 +01:00
}
if ( initialize_xrender ( dylibloader_verbose ) ! = 0 ) {
r_error = ERR_UNAVAILABLE ;
ERR_FAIL_MSG ( " Can't load Xrender dynamically. " ) ;
}
if ( initialize_xinput2 ( dylibloader_verbose ) ! = 0 ) {
r_error = ERR_UNAVAILABLE ;
ERR_FAIL_MSG ( " Can't load Xinput2 dynamically. " ) ;
}
2023-02-15 13:13:56 +01:00
# else
# ifdef XKB_ENABLED
2023-07-06 14:05:28 +02:00
bool xkb_loaded = true ;
xkb_loaded_v05p = true ;
xkb_loaded_v08p = true ;
2023-02-15 13:13:56 +01:00
# endif
# endif
2022-11-29 12:03:06 +01:00
2023-02-15 13:13:56 +01:00
# ifdef XKB_ENABLED
2023-01-30 09:59:49 +01:00
if ( xkb_loaded ) {
xkb_ctx = xkb_context_new ( XKB_CONTEXT_NO_FLAGS ) ;
if ( xkb_ctx ) {
const char * locale = getenv ( " LC_ALL " ) ;
if ( ! locale | | ! * locale ) {
locale = getenv ( " LC_CTYPE " ) ;
}
if ( ! locale | | ! * locale ) {
locale = getenv ( " LANG " ) ;
}
if ( ! locale | | ! * locale ) {
locale = " C " ;
}
dead_tbl = xkb_compose_table_new_from_locale ( xkb_ctx , locale , XKB_COMPOSE_COMPILE_NO_FLAGS ) ;
}
}
2023-02-15 13:13:56 +01:00
# endif
2023-01-30 09:59:49 +01:00
2020-04-28 15:19:37 +02:00
Input : : get_singleton ( ) - > set_event_dispatch_function ( _dispatch_input_events ) ;
2020-03-04 17:36:09 +01:00
2020-03-01 23:14:37 +01:00
r_error = OK ;
2023-07-06 14:05:28 +02:00
# ifdef SOWRAP_ENABLED
2023-03-16 09:44:47 +01:00
{
if ( ! XcursorImageCreate | | ! XcursorImageLoadCursor | | ! XcursorImageDestroy | | ! XcursorGetDefaultSize | | ! XcursorGetTheme | | ! XcursorLibraryLoadImage ) {
// There's no API to check version, check if functions are available instead.
ERR_PRINT ( " Unsupported Xcursor library version. " ) ;
r_error = ERR_UNAVAILABLE ;
return ;
}
}
2023-07-06 14:05:28 +02:00
# endif
2023-03-16 09:44:47 +01:00
2020-04-01 19:29:35 +02:00
for ( int i = 0 ; i < CURSOR_MAX ; i + + ) {
cursors [ i ] = None ;
2023-05-11 12:32:23 +02:00
cursor_img [ i ] = nullptr ;
2020-04-01 19:29:35 +02:00
}
2020-03-07 16:46:50 +01:00
XInitThreads ( ) ; //always use threads
2020-03-01 23:14:37 +01:00
/** XLIB INITIALIZATION **/
2020-04-02 01:20:12 +02:00
x11_display = XOpenDisplay ( nullptr ) ;
2020-03-01 23:14:37 +01:00
if ( ! x11_display ) {
ERR_PRINT ( " X11 Display is not available " ) ;
r_error = ERR_UNAVAILABLE ;
return ;
}
2023-08-04 09:58:14 +02:00
if ( xshaped_ext_ok ) {
2023-03-16 09:44:47 +01:00
int version_major = 0 ;
int version_minor = 0 ;
int rc = XShapeQueryVersion ( x11_display , & version_major , & version_minor ) ;
print_verbose ( vformat ( " Xshape %d.%d detected. " , version_major , version_minor ) ) ;
if ( rc ! = 1 | | version_major < 1 ) {
2023-08-04 09:58:14 +02:00
xshaped_ext_ok = false ;
print_verbose ( " Unsupported Xshape library version. " ) ;
2023-03-16 09:44:47 +01:00
}
}
2023-08-04 09:58:14 +02:00
if ( xinerama_ext_ok ) {
2023-03-16 09:44:47 +01:00
int version_major = 0 ;
int version_minor = 0 ;
int rc = XineramaQueryVersion ( x11_display , & version_major , & version_minor ) ;
print_verbose ( vformat ( " Xinerama %d.%d detected. " , version_major , version_minor ) ) ;
if ( rc ! = 1 | | version_major < 1 ) {
2023-08-04 09:58:14 +02:00
xinerama_ext_ok = false ;
print_verbose ( " Unsupported Xinerama library version. " ) ;
2023-03-16 09:44:47 +01:00
}
}
2023-08-04 09:58:14 +02:00
if ( xrandr_ext_ok ) {
2023-03-16 09:44:47 +01:00
int version_major = 0 ;
int version_minor = 0 ;
int rc = XRRQueryVersion ( x11_display , & version_major , & version_minor ) ;
print_verbose ( vformat ( " Xrandr %d.%d detected. " , version_major , version_minor ) ) ;
if ( rc ! = 1 | | ( version_major = = 1 & & version_minor < 3 ) | | ( version_major < 1 ) ) {
2023-08-04 09:58:14 +02:00
xrandr_ext_ok = false ;
print_verbose ( " Unsupported Xrandr library version. " ) ;
2023-03-16 09:44:47 +01:00
}
}
{
int version_major = 0 ;
int version_minor = 0 ;
int rc = XRenderQueryVersion ( x11_display , & version_major , & version_minor ) ;
print_verbose ( vformat ( " Xrender %d.%d detected. " , version_major , version_minor ) ) ;
if ( rc ! = 1 | | ( version_major = = 0 & & version_minor < 11 ) ) {
ERR_PRINT ( " Unsupported Xrender library version. " ) ;
r_error = ERR_UNAVAILABLE ;
XCloseDisplay ( x11_display ) ;
return ;
}
}
{
int version_major = 2 ; // Report 2.2 as supported by engine, but should work with 2.1 or 2.0 library as well.
int version_minor = 2 ;
int rc = XIQueryVersion ( x11_display , & version_major , & version_minor ) ;
print_verbose ( vformat ( " Xinput %d.%d detected. " , version_major , version_minor ) ) ;
if ( rc ! = Success | | ( version_major < 2 ) ) {
ERR_PRINT ( " Unsupported Xinput2 library version. " ) ;
r_error = ERR_UNAVAILABLE ;
XCloseDisplay ( x11_display ) ;
return ;
}
}
2020-04-02 01:20:12 +02:00
char * modifiers = nullptr ;
2020-03-01 23:14:37 +01:00
Bool xkb_dar = False ;
XAutoRepeatOn ( x11_display ) ;
2020-04-02 01:20:12 +02:00
xkb_dar = XkbSetDetectableAutoRepeat ( x11_display , True , nullptr ) ;
2020-03-01 23:14:37 +01:00
// Try to support IME if detectable auto-repeat is supported
if ( xkb_dar = = True ) {
# ifdef X_HAVE_UTF8_STRING
// Xutf8LookupString will be used later instead of XmbLookupString before
// the multibyte sequences can be converted to unicode string.
modifiers = XSetLocaleModifiers ( " " ) ;
# endif
}
2020-04-02 01:20:12 +02:00
if ( modifiers = = nullptr ) {
2020-03-01 23:14:37 +01:00
if ( OS : : get_singleton ( ) - > is_stdout_verbose ( ) ) {
WARN_PRINT ( " IME is disabled " ) ;
}
XSetLocaleModifiers ( " @im=none " ) ;
WARN_PRINT ( " Error setting locale modifiers " ) ;
}
const char * err ;
int xrandr_major = 0 ;
int xrandr_minor = 0 ;
int event_base , error_base ;
xrandr_ext_ok = XRRQueryExtension ( x11_display , & event_base , & error_base ) ;
xrandr_handle = dlopen ( " libXrandr.so.2 " , RTLD_LAZY ) ;
if ( ! xrandr_handle ) {
err = dlerror ( ) ;
2020-09-18 11:55:12 +02:00
// For some arcane reason, NetBSD now ships libXrandr.so.3 while the rest of the world has libXrandr.so.2...
// In case this happens for other X11 platforms in the future, let's give it a try too before failing.
xrandr_handle = dlopen ( " libXrandr.so.3 " , RTLD_LAZY ) ;
if ( ! xrandr_handle ) {
fprintf ( stderr , " could not load libXrandr.so.2, Error: %s \n " , err ) ;
}
2023-08-04 09:58:14 +02:00
}
if ( xrandr_handle ) {
2020-03-01 23:14:37 +01:00
XRRQueryVersion ( x11_display , & xrandr_major , & xrandr_minor ) ;
if ( ( ( xrandr_major < < 8 ) | xrandr_minor ) > = 0x0105 ) {
xrr_get_monitors = ( xrr_get_monitors_t ) dlsym ( xrandr_handle , " XRRGetMonitors " ) ;
if ( ! xrr_get_monitors ) {
err = dlerror ( ) ;
fprintf ( stderr , " could not find symbol XRRGetMonitors \n Error: %s \n " , err ) ;
} else {
xrr_free_monitors = ( xrr_free_monitors_t ) dlsym ( xrandr_handle , " XRRFreeMonitors " ) ;
if ( ! xrr_free_monitors ) {
err = dlerror ( ) ;
fprintf ( stderr , " could not find XRRFreeMonitors \n Error: %s \n " , err ) ;
2020-04-02 01:20:12 +02:00
xrr_get_monitors = nullptr ;
2020-03-01 23:14:37 +01:00
}
}
}
}
if ( ! _refresh_device_info ( ) ) {
2021-07-22 18:23:48 +02:00
OS : : get_singleton ( ) - > alert ( " Your system does not support XInput 2. \n "
" Please upgrade your distribution. " ,
2020-03-01 23:14:37 +01:00
" Unable to initialize XInput " ) ;
r_error = ERR_UNAVAILABLE ;
return ;
}
2020-04-02 01:20:12 +02:00
xim = XOpenIM ( x11_display , nullptr , nullptr , nullptr ) ;
2020-03-01 23:14:37 +01:00
2020-04-02 01:20:12 +02:00
if ( xim = = nullptr ) {
2020-03-01 23:14:37 +01:00
WARN_PRINT ( " XOpenIM failed " ) ;
xim_style = 0L ;
} else {
: : XIMCallback im_destroy_callback ;
im_destroy_callback . client_data = ( : : XPointer ) ( this ) ;
im_destroy_callback . callback = ( : : XIMProc ) ( _xim_destroy_callback ) ;
if ( XSetIMValues ( xim , XNDestroyCallback , & im_destroy_callback ,
2020-04-02 01:20:12 +02:00
nullptr ) ! = nullptr ) {
2020-03-01 23:14:37 +01:00
WARN_PRINT ( " Error setting XIM destroy callback " ) ;
}
2020-04-02 01:20:12 +02:00
: : XIMStyles * xim_styles = nullptr ;
2020-03-01 23:14:37 +01:00
xim_style = 0L ;
2020-04-02 01:20:12 +02:00
char * imvalret = XGetIMValues ( xim , XNQueryInputStyle , & xim_styles , nullptr ) ;
if ( imvalret ! = nullptr | | xim_styles = = nullptr ) {
2020-03-01 23:14:37 +01:00
fprintf ( stderr , " Input method doesn't support any styles \n " ) ;
}
if ( xim_styles ) {
xim_style = 0L ;
for ( int i = 0 ; i < xim_styles - > count_styles ; i + + ) {
2022-12-11 00:21:22 +01:00
const : : XIMStyle & style = xim_styles - > supported_styles [ i ] ;
if ( ! _is_xim_style_supported ( style ) ) {
continue ;
2020-03-01 23:14:37 +01:00
}
2022-12-11 00:21:22 +01:00
xim_style = _get_best_xim_style ( xim_style , style ) ;
2020-03-01 23:14:37 +01:00
}
XFree ( xim_styles ) ;
}
XFree ( imvalret ) ;
}
2022-05-02 16:28:25 +02:00
/* Atom internment */
2020-03-01 23:14:37 +01:00
wm_delete = XInternAtom ( x11_display , " WM_DELETE_WINDOW " , true ) ;
2022-05-02 16:28:25 +02:00
// Set Xdnd (drag & drop) support.
2020-03-01 23:14:37 +01:00
xdnd_aware = XInternAtom ( x11_display , " XdndAware " , False ) ;
xdnd_enter = XInternAtom ( x11_display , " XdndEnter " , False ) ;
xdnd_position = XInternAtom ( x11_display , " XdndPosition " , False ) ;
xdnd_status = XInternAtom ( x11_display , " XdndStatus " , False ) ;
xdnd_action_copy = XInternAtom ( x11_display , " XdndActionCopy " , False ) ;
xdnd_drop = XInternAtom ( x11_display , " XdndDrop " , False ) ;
xdnd_finished = XInternAtom ( x11_display , " XdndFinished " , False ) ;
xdnd_selection = XInternAtom ( x11_display , " XdndSelection " , False ) ;
2021-11-04 13:33:37 +01:00
# ifdef SPEECHD_ENABLED
// Init TTS
2023-05-16 13:18:12 +02:00
bool tts_enabled = GLOBAL_GET ( " audio/general/text_to_speech " ) ;
if ( tts_enabled ) {
tts = memnew ( TTS_Linux ) ;
}
2021-11-04 13:33:37 +01:00
# endif
2020-03-01 23:14:37 +01:00
//!!!!!!!!!!!!!!!!!!!!!!!!!!
2021-09-27 01:07:10 +02:00
//TODO - do Vulkan and OpenGL support checks, driver selection and fallback
2020-03-03 14:36:29 +01:00
rendering_driver = p_rendering_driver ;
2020-03-01 23:14:37 +01:00
2020-11-18 19:11:30 +01:00
bool driver_found = false ;
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
2020-03-01 23:14:37 +01:00
# if defined(VULKAN_ENABLED)
2020-03-03 14:36:29 +01:00
if ( rendering_driver = = " vulkan " ) {
2023-12-19 18:57:56 +01:00
rendering_context = memnew ( RenderingContextDriverVulkanX11 ) ;
2023-12-19 12:48:02 +01:00
}
# endif
2023-12-19 18:57:56 +01:00
if ( rendering_context ) {
if ( rendering_context - > initialize ( ) ! = OK ) {
ERR_PRINT ( vformat ( " Could not initialize %s " , rendering_driver ) ) ;
memdelete ( rendering_context ) ;
rendering_context = nullptr ;
2020-03-01 23:14:37 +01:00
r_error = ERR_CANT_CREATE ;
2024-01-05 18:27:10 +01:00
return ;
2020-03-01 23:14:37 +01:00
}
2020-11-18 19:11:30 +01:00
driver_found = true ;
2020-03-01 23:14:37 +01:00
}
# endif
2021-10-26 17:18:39 +02:00
// Initialize context and rendering device.
# if defined(GLES3_ENABLED)
2023-09-22 09:55:55 +02:00
if ( rendering_driver = = " opengl3 " | | rendering_driver = = " opengl3_es " ) {
2020-04-02 01:20:12 +02:00
if ( getenv ( " DRI_PRIME " ) = = nullptr ) {
2020-03-01 23:14:37 +01:00
int use_prime = - 1 ;
if ( getenv ( " PRIMUS_DISPLAY " ) | |
getenv ( " PRIMUS_libGLd " ) | |
getenv ( " PRIMUS_libGLa " ) | |
getenv ( " PRIMUS_libGL " ) | |
getenv ( " PRIMUS_LOAD_GLOBAL " ) | |
getenv ( " BUMBLEBEE_SOCKET " ) ) {
print_verbose ( " Optirun/primusrun detected. Skipping GPU detection " ) ;
use_prime = 0 ;
}
2021-03-08 15:42:07 +01:00
// Some tools use fake libGL libraries and have them override the real one using
// LD_LIBRARY_PATH, so we skip them. *But* Steam also sets LD_LIBRARY_PATH for its
// runtime and includes system `/lib` and `/lib64`... so ignore Steam.
if ( use_prime = = - 1 & & getenv ( " LD_LIBRARY_PATH " ) & & ! getenv ( " STEAM_RUNTIME_LIBRARY_PATH " ) ) {
2020-03-01 23:14:37 +01:00
String ld_library_path ( getenv ( " LD_LIBRARY_PATH " ) ) ;
Vector < String > libraries = ld_library_path . split ( " : " ) ;
for ( int i = 0 ; i < libraries . size ( ) ; + + i ) {
if ( FileAccess : : exists ( libraries [ i ] + " /libGL.so.1 " ) | |
FileAccess : : exists ( libraries [ i ] + " /libGL.so " ) ) {
print_verbose ( " Custom libGL override detected. Skipping GPU detection " ) ;
use_prime = 0 ;
}
}
}
if ( use_prime = = - 1 ) {
print_verbose ( " Detecting GPUs, set DRI_PRIME in the environment to override GPU detection logic. " ) ;
use_prime = detect_prime ( ) ;
}
if ( use_prime ) {
print_line ( " Found discrete GPU, setting DRI_PRIME=1 to use it. " ) ;
print_line ( " Note: Set DRI_PRIME=0 in the environment to disable Godot from using the discrete GPU. " ) ;
setenv ( " DRI_PRIME " , " 1 " , 1 ) ;
}
}
2023-09-22 09:55:55 +02:00
}
if ( rendering_driver = = " opengl3 " ) {
2023-11-06 11:17:14 +01:00
gl_manager = memnew ( GLManager_X11 ( p_resolution , GLManager_X11 : : GLES_3_0_COMPATIBLE ) ) ;
if ( gl_manager - > initialize ( x11_display ) ! = OK | | gl_manager - > open_display ( x11_display ) ! = OK ) {
2020-11-18 19:11:30 +01:00
memdelete ( gl_manager ) ;
gl_manager = nullptr ;
2023-11-06 11:17:14 +01:00
bool fallback = GLOBAL_GET ( " rendering/gl_compatibility/fallback_to_gles " ) ;
if ( fallback ) {
WARN_PRINT ( " Your video card drivers seem not to support the required OpenGL version, switching to OpenGLES. " ) ;
rendering_driver = " opengl3_es " ;
} else {
r_error = ERR_UNAVAILABLE ;
ERR_FAIL_MSG ( " Could not initialize OpenGL. " ) ;
}
} else {
driver_found = true ;
RasterizerGLES3 : : make_current ( true ) ;
2020-03-01 23:14:37 +01:00
}
2023-09-22 09:55:55 +02:00
}
2023-11-06 11:17:14 +01:00
2023-09-22 09:55:55 +02:00
if ( rendering_driver = = " opengl3_es " ) {
gl_manager_egl = memnew ( GLManagerEGL_X11 ) ;
if ( gl_manager_egl - > initialize ( ) ! = OK ) {
memdelete ( gl_manager_egl ) ;
gl_manager_egl = nullptr ;
2020-11-18 19:11:30 +01:00
r_error = ERR_UNAVAILABLE ;
2023-11-06 11:17:14 +01:00
ERR_FAIL_MSG ( " Could not initialize OpenGLES. " ) ;
2020-03-01 23:14:37 +01:00
}
2023-09-22 09:55:55 +02:00
driver_found = true ;
RasterizerGLES3 : : make_current ( false ) ;
2020-03-01 23:14:37 +01:00
}
2023-11-06 11:17:14 +01:00
2020-03-01 23:14:37 +01:00
# endif
2020-11-18 19:11:30 +01:00
if ( ! driver_found ) {
r_error = ERR_UNAVAILABLE ;
ERR_FAIL_MSG ( " Video driver not found " ) ;
}
2023-01-04 23:00:02 +01:00
Point2i window_position ;
2022-10-23 12:34:41 +02:00
if ( p_position ! = nullptr ) {
window_position = * p_position ;
2023-01-04 23:00:02 +01:00
} else {
if ( p_screen = = SCREEN_OF_MAIN_WINDOW ) {
p_screen = SCREEN_PRIMARY ;
}
2023-03-30 10:37:15 +02:00
Rect2i scr_rect = screen_get_usable_rect ( p_screen ) ;
window_position = scr_rect . position + ( scr_rect . size - p_resolution ) / 2 ;
2022-10-23 12:34:41 +02:00
}
2023-01-04 23:00:02 +01:00
WindowID main_window = _create_window ( p_mode , p_vsync_mode , p_flags , Rect2i ( window_position , p_resolution ) ) ;
2020-11-14 12:41:13 +01:00
if ( main_window = = INVALID_WINDOW_ID ) {
r_error = ERR_CANT_CREATE ;
return ;
}
2020-03-01 23:14:37 +01:00
for ( int i = 0 ; i < WINDOW_FLAG_MAX ; i + + ) {
if ( p_flags & ( 1 < < i ) ) {
window_set_flag ( WindowFlags ( i ) , true , main_window ) ;
}
}
2020-08-21 09:39:30 +02:00
show_window ( main_window ) ;
2020-03-01 23:14:37 +01:00
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
2023-12-19 18:57:56 +01:00
if ( rendering_context ) {
2024-06-29 14:03:18 +02:00
rendering_device = memnew ( RenderingDevice ) ;
2024-06-28 17:17:20 +02:00
if ( rendering_device - > initialize ( rendering_context , MAIN_WINDOW_ID ) ! = OK ) {
memdelete ( rendering_device ) ;
rendering_device = nullptr ;
memdelete ( rendering_context ) ;
rendering_context = nullptr ;
r_error = ERR_UNAVAILABLE ;
return ;
}
2023-12-19 18:57:56 +01:00
rendering_device - > screen_create ( MAIN_WINDOW_ID ) ;
2020-03-03 14:36:29 +01:00
2020-12-04 19:26:24 +01:00
RendererCompositorRD : : make_current ( ) ;
2020-03-03 14:36:29 +01:00
}
# endif
2020-03-01 23:14:37 +01:00
{
//set all event master mask
XIEventMask all_master_event_mask ;
static unsigned char all_master_mask_data [ XIMaskLen ( XI_LASTEVENT ) ] = { } ;
all_master_event_mask . deviceid = XIAllMasterDevices ;
all_master_event_mask . mask_len = sizeof ( all_master_mask_data ) ;
all_master_event_mask . mask = all_master_mask_data ;
XISetMask ( all_master_event_mask . mask , XI_DeviceChanged ) ;
XISetMask ( all_master_event_mask . mask , XI_RawMotion ) ;
XISelectEvents ( x11_display , DefaultRootWindow ( x11_display ) , & all_master_event_mask , 1 ) ;
}
cursor_size = XcursorGetDefaultSize ( x11_display ) ;
cursor_theme = XcursorGetTheme ( x11_display ) ;
if ( ! cursor_theme ) {
print_verbose ( " XcursorGetTheme could not get cursor theme " ) ;
cursor_theme = " default " ;
}
for ( int i = 0 ; i < CURSOR_MAX ; i + + ) {
static const char * cursor_file [ ] = {
" left_ptr " ,
" xterm " ,
" hand2 " ,
" cross " ,
" watch " ,
" left_ptr_watch " ,
" fleur " ,
" dnd-move " ,
" crossed_circle " ,
" v_double_arrow " ,
" h_double_arrow " ,
" size_bdiag " ,
" size_fdiag " ,
" move " ,
" row_resize " ,
" col_resize " ,
" question_arrow "
} ;
2023-05-11 12:32:23 +02:00
cursor_img [ i ] = XcursorLibraryLoadImage ( cursor_file [ i ] , cursor_theme , cursor_size ) ;
if ( ! cursor_img [ i ] ) {
2020-04-02 01:20:12 +02:00
const char * fallback = nullptr ;
2020-03-01 23:14:37 +01:00
switch ( i ) {
case CURSOR_POINTING_HAND :
fallback = " pointer " ;
break ;
case CURSOR_CROSS :
fallback = " crosshair " ;
break ;
case CURSOR_WAIT :
fallback = " wait " ;
break ;
case CURSOR_BUSY :
fallback = " progress " ;
break ;
case CURSOR_DRAG :
fallback = " grabbing " ;
break ;
case CURSOR_CAN_DROP :
fallback = " hand1 " ;
break ;
case CURSOR_FORBIDDEN :
fallback = " forbidden " ;
break ;
case CURSOR_VSIZE :
fallback = " ns-resize " ;
break ;
case CURSOR_HSIZE :
fallback = " ew-resize " ;
break ;
case CURSOR_BDIAGSIZE :
fallback = " fd_double_arrow " ;
break ;
case CURSOR_FDIAGSIZE :
fallback = " bd_double_arrow " ;
break ;
case CURSOR_MOVE :
2023-05-11 12:32:23 +02:00
cursor_img [ i ] = cursor_img [ CURSOR_DRAG ] ;
2020-03-01 23:14:37 +01:00
break ;
case CURSOR_VSPLIT :
fallback = " sb_v_double_arrow " ;
break ;
case CURSOR_HSPLIT :
fallback = " sb_h_double_arrow " ;
break ;
case CURSOR_HELP :
fallback = " help " ;
break ;
}
2020-04-02 01:20:12 +02:00
if ( fallback ! = nullptr ) {
2023-05-11 12:32:23 +02:00
cursor_img [ i ] = XcursorLibraryLoadImage ( fallback , cursor_theme , cursor_size ) ;
2020-03-01 23:14:37 +01:00
}
}
2023-05-11 12:32:23 +02:00
if ( cursor_img [ i ] ) {
cursors [ i ] = XcursorImageLoadCursor ( x11_display , cursor_img [ i ] ) ;
2020-03-01 23:14:37 +01:00
} else {
print_verbose ( " Failed loading custom cursor: " + String ( cursor_file [ i ] ) ) ;
}
}
{
// Creating an empty/transparent cursor
// Create 1x1 bitmap
Pixmap cursormask = XCreatePixmap ( x11_display ,
RootWindow ( x11_display , DefaultScreen ( x11_display ) ) , 1 , 1 , 1 ) ;
// Fill with zero
XGCValues xgc ;
xgc . function = GXclear ;
GC gc = XCreateGC ( x11_display , cursormask , GCFunction , & xgc ) ;
XFillRectangle ( x11_display , cursormask , gc , 0 , 0 , 1 , 1 ) ;
// Color value doesn't matter. Mask zero means no foreground or background will be drawn
XColor col = { } ;
Cursor cursor = XCreatePixmapCursor ( x11_display ,
cursormask , // source (using cursor mask as placeholder, since it'll all be ignored)
cursormask , // mask
& col , & col , 0 , 0 ) ;
XFreePixmap ( x11_display , cursormask ) ;
XFreeGC ( x11_display , gc ) ;
if ( cursor = = None ) {
ERR_PRINT ( " FAILED CREATING CURSOR " ) ;
}
null_cursor = cursor ;
}
cursor_set_shape ( CURSOR_BUSY ) ;
2022-09-05 00:46:37 +02:00
// Search the X11 event queue for ConfigureNotify events and process all
// that are currently queued early, so we can get the final window size
// for correctly drawing of the bootsplash.
XEvent config_event ;
while ( XCheckTypedEvent ( x11_display , ConfigureNotify , & config_event ) ) {
_window_changed ( & config_event ) ;
2022-08-30 16:35:19 +02:00
}
2021-01-19 13:29:41 +01:00
events_thread . start ( _poll_events_thread , this ) ;
2020-09-09 12:43:20 +02:00
2020-03-01 23:14:37 +01:00
_update_real_mouse_position ( windows [ MAIN_WINDOW_ID ] ) ;
2020-04-02 23:46:34 +02:00
# ifdef DBUS_ENABLED
screensaver = memnew ( FreeDesktopScreenSaver ) ;
2022-08-06 22:10:24 +02:00
screen_set_keep_on ( GLOBAL_GET ( " display/window/energy_saving/keep_screen_on " ) ) ;
2022-08-30 17:04:17 +02:00
portal_desktop = memnew ( FreeDesktopPortalDesktop ) ;
2020-04-02 23:46:34 +02:00
# endif
2023-03-19 13:54:24 +01:00
XSetErrorHandler ( & default_window_error_handler ) ;
2020-04-02 23:46:34 +02:00
2020-03-01 23:14:37 +01:00
r_error = OK ;
}
2020-05-14 14:29:06 +02:00
2020-03-01 23:14:37 +01:00
DisplayServerX11 : : ~ DisplayServerX11 ( ) {
2020-10-08 17:57:54 +02:00
// Send owned clipboard data to clipboard manager before exit.
Window x11_main_window = windows [ MAIN_WINDOW_ID ] . x11_window ;
_clipboard_transfer_ownership ( XA_PRIMARY , x11_main_window ) ;
_clipboard_transfer_ownership ( XInternAtom ( x11_display , " CLIPBOARD " , 0 ) , x11_main_window ) ;
2021-02-10 19:22:13 +01:00
events_thread_done . set ( ) ;
2021-01-19 13:29:41 +01:00
events_thread . wait_to_finish ( ) ;
2020-09-09 12:43:20 +02:00
2024-01-19 18:41:01 +01:00
if ( native_menu ) {
memdelete ( native_menu ) ;
native_menu = nullptr ;
}
2020-03-03 14:36:29 +01:00
//destroy all windows
2021-08-09 22:13:42 +02:00
for ( KeyValue < WindowID , WindowData > & E : windows ) {
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
2023-12-19 18:57:56 +01:00
if ( rendering_device ) {
rendering_device - > screen_free ( E . key ) ;
}
if ( rendering_context ) {
rendering_context - > window_destroy ( E . key ) ;
2020-03-03 14:36:29 +01:00
}
# endif
2021-10-26 17:18:39 +02:00
# ifdef GLES3_ENABLED
2021-12-14 02:49:29 +01:00
if ( gl_manager ) {
2021-09-27 01:07:10 +02:00
gl_manager - > window_destroy ( E . key ) ;
2020-11-18 19:11:30 +01:00
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
gl_manager_egl - > window_destroy ( E . key ) ;
}
2020-11-18 19:11:30 +01:00
# endif
2020-03-03 14:36:29 +01:00
2021-08-09 22:13:42 +02:00
WindowData & wd = E . value ;
2020-10-08 17:21:51 +02:00
if ( wd . xic ) {
XDestroyIC ( wd . xic ) ;
wd . xic = nullptr ;
2020-03-03 14:36:29 +01:00
}
2023-01-30 09:59:49 +01:00
XDestroyWindow ( x11_display , wd . x11_xim_window ) ;
2023-02-15 13:13:56 +01:00
# ifdef XKB_ENABLED
2023-03-16 09:44:47 +01:00
if ( xkb_loaded_v05p ) {
2023-01-30 09:59:49 +01:00
if ( wd . xkb_state ) {
xkb_compose_state_unref ( wd . xkb_state ) ;
wd . xkb_state = nullptr ;
}
}
2023-02-15 13:13:56 +01:00
# endif
2020-10-08 17:21:51 +02:00
XUnmapWindow ( x11_display , wd . x11_window ) ;
XDestroyWindow ( x11_display , wd . x11_window ) ;
2020-03-03 14:36:29 +01:00
}
2023-02-15 13:13:56 +01:00
# ifdef XKB_ENABLED
2023-03-16 09:44:47 +01:00
if ( xkb_loaded_v05p ) {
2023-01-30 09:59:49 +01:00
if ( dead_tbl ) {
xkb_compose_table_unref ( dead_tbl ) ;
}
if ( xkb_ctx ) {
xkb_context_unref ( xkb_ctx ) ;
}
}
2023-02-15 13:13:56 +01:00
# endif
2023-01-30 09:59:49 +01:00
2020-03-03 14:36:29 +01:00
//destroy drivers
2023-12-19 12:48:02 +01:00
# if defined(RD_ENABLED)
if ( rendering_device ) {
memdelete ( rendering_device ) ;
rendering_device = nullptr ;
2021-12-14 02:49:29 +01:00
}
2020-03-03 14:36:29 +01:00
2023-12-19 18:57:56 +01:00
if ( rendering_context ) {
memdelete ( rendering_context ) ;
rendering_context = nullptr ;
2020-03-03 14:36:29 +01:00
}
# endif
2021-10-26 17:18:39 +02:00
# ifdef GLES3_ENABLED
2020-11-18 19:11:30 +01:00
if ( gl_manager ) {
memdelete ( gl_manager ) ;
gl_manager = nullptr ;
}
2023-09-22 09:55:55 +02:00
if ( gl_manager_egl ) {
memdelete ( gl_manager_egl ) ;
gl_manager_egl = nullptr ;
}
2020-11-18 19:11:30 +01:00
# endif
2020-05-14 16:41:43 +02:00
if ( xrandr_handle ) {
2020-03-03 14:36:29 +01:00
dlclose ( xrandr_handle ) ;
2020-05-14 16:41:43 +02:00
}
2020-03-03 14:36:29 +01:00
for ( int i = 0 ; i < CURSOR_MAX ; i + + ) {
2020-05-14 16:41:43 +02:00
if ( cursors [ i ] ! = None ) {
2020-03-03 14:36:29 +01:00
XFreeCursor ( x11_display , cursors [ i ] ) ;
2020-05-14 16:41:43 +02:00
}
2023-05-11 12:32:23 +02:00
if ( cursor_img [ i ] ! = nullptr ) {
XcursorImageDestroy ( cursor_img [ i ] ) ;
2020-05-14 16:41:43 +02:00
}
2022-02-16 13:56:32 +01:00
}
2020-03-03 14:36:29 +01:00
if ( xim ) {
XCloseIM ( xim ) ;
}
XCloseDisplay ( x11_display ) ;
2020-05-14 16:41:43 +02:00
if ( xmbstring ) {
2020-03-03 14:36:29 +01:00
memfree ( xmbstring ) ;
2020-05-14 16:41:43 +02:00
}
2020-04-02 23:46:34 +02:00
2021-11-04 13:33:37 +01:00
# ifdef SPEECHD_ENABLED
2023-05-16 13:18:12 +02:00
if ( tts ) {
memdelete ( tts ) ;
}
2021-11-04 13:33:37 +01:00
# endif
2020-04-02 23:46:34 +02:00
# ifdef DBUS_ENABLED
memdelete ( screensaver ) ;
2022-08-30 17:04:17 +02:00
memdelete ( portal_desktop ) ;
2020-04-02 23:46:34 +02:00
# endif
2020-03-03 14:36:29 +01:00
}
void DisplayServerX11 : : register_x11_driver ( ) {
register_create_function ( " x11 " , create_func , get_rendering_drivers_func ) ;
2020-03-01 23:14:37 +01:00
}
# endif // X11 enabled