Add project manager / editor initial screen settings, implement DisplayServer.get_keyboard_focus_screen method.
This commit is contained in:
parent
1b4b8934e0
commit
520a8d8ed2
@ -1284,7 +1284,7 @@ ProjectSettings::ProjectSettings() {
|
||||
GLOBAL_DEF_BASIC(PropertyInfo(Variant::INT, "display/window/size/mode", PROPERTY_HINT_ENUM, "Windowed,Minimized,Maximized,Fullscreen,Exclusive Fullscreen"), 0);
|
||||
|
||||
// Keep the enum values in sync with the `DisplayServer::SCREEN_` enum.
|
||||
GLOBAL_DEF_BASIC(PropertyInfo(Variant::INT, "display/window/size/initial_position_type", PROPERTY_HINT_ENUM, "Absolute,Primary Screen Center,Other Screen Center"), 1);
|
||||
GLOBAL_DEF_BASIC(PropertyInfo(Variant::INT, "display/window/size/initial_position_type", PROPERTY_HINT_ENUM, "Absolute,Center of Primary Screen,Center of Other Screen,Center of Screen With Mouse Pointer,Center of Screen With Keyboard Focus"), 1);
|
||||
GLOBAL_DEF_BASIC(PropertyInfo(Variant::VECTOR2I, "display/window/size/initial_position"), Vector2i());
|
||||
GLOBAL_DEF_BASIC(PropertyInfo(Variant::INT, "display/window/size/initial_screen", PROPERTY_HINT_RANGE, "0,64,1,or_greater"), 0);
|
||||
|
||||
|
@ -123,6 +123,12 @@
|
||||
Returns the unobscured area of the display where interactive controls should be rendered. See also [method get_display_cutouts].
|
||||
</description>
|
||||
</method>
|
||||
<method name="get_keyboard_focus_screen" qualifiers="const">
|
||||
<return type="int" />
|
||||
<description>
|
||||
Returns the index of the screen containing the window with the keyboard focus, or the primary screen if there's no focused window.
|
||||
</description>
|
||||
</method>
|
||||
<method name="get_name" qualifiers="const">
|
||||
<return type="String" />
|
||||
<description>
|
||||
@ -1571,6 +1577,12 @@
|
||||
<constant name="MOUSE_MODE_CONFINED_HIDDEN" value="4" enum="MouseMode">
|
||||
Confines the mouse cursor to the game window, and make it hidden.
|
||||
</constant>
|
||||
<constant name="SCREEN_WITH_MOUSE_FOCUS" value="-4">
|
||||
Represents the screen containing the mouse pointer.
|
||||
</constant>
|
||||
<constant name="SCREEN_WITH_KEYBOARD_FOCUS" value="-3">
|
||||
Represents the screen containing the window with the keyboard focus.
|
||||
</constant>
|
||||
<constant name="SCREEN_PRIMARY" value="-2">
|
||||
Represents the primary screen.
|
||||
</constant>
|
||||
|
@ -513,6 +513,9 @@
|
||||
The language to use for the editor interface.
|
||||
Translations are provided by the community. If you spot a mistake, [url=$DOCS_URL/contributing/documentation/editor_and_docs_localization.html]contribute to editor translations on Weblate![/url]
|
||||
</member>
|
||||
<member name="interface/editor/editor_screen" type="int" setter="" getter="">
|
||||
The preferred monitor to display the editor.
|
||||
</member>
|
||||
<member name="interface/editor/expand_to_title" type="bool" setter="" getter="">
|
||||
Expanding main editor window content to the title, if supported by [DisplayServer]. See [constant DisplayServer.WINDOW_FLAG_EXTEND_TO_TITLE].
|
||||
Specific to the macOS platform.
|
||||
@ -545,6 +548,9 @@
|
||||
<member name="interface/editor/mouse_extra_buttons_navigate_history" type="bool" setter="" getter="">
|
||||
If [code]true[/code], the mouse's additional side buttons will be usable to navigate in the script editor's file history. Set this to [code]false[/code] if you're using the side buttons for other purposes (such as a push-to-talk button in a VoIP program).
|
||||
</member>
|
||||
<member name="interface/editor/project_manager_screen" type="int" setter="" getter="">
|
||||
The preferred monitor to display the project manager.
|
||||
</member>
|
||||
<member name="interface/editor/save_each_scene_on_quit" type="bool" setter="" getter="">
|
||||
If [code]false[/code], the editor will save all scenes when confirming the [b]Save[/b] action when quitting the editor or quitting to the project list. If [code]true[/code], the editor will ask to save each scene individually.
|
||||
</member>
|
||||
|
@ -782,13 +782,19 @@
|
||||
Initial window position is determined by [member position].
|
||||
</constant>
|
||||
<constant name="WINDOW_INITIAL_POSITION_CENTER_PRIMARY_SCREEN" value="1" enum="WindowInitialPosition">
|
||||
Initial window position is a center of the primary screen.
|
||||
Initial window position is the center of the primary screen.
|
||||
</constant>
|
||||
<constant name="WINDOW_INITIAL_POSITION_CENTER_MAIN_WINDOW_SCREEN" value="2" enum="WindowInitialPosition">
|
||||
Initial window position is a center of the main window screen.
|
||||
Initial window position is the center of the main window screen.
|
||||
</constant>
|
||||
<constant name="WINDOW_INITIAL_POSITION_CENTER_OTHER_SCREEN" value="3" enum="WindowInitialPosition">
|
||||
Initial window position is a center of [member current_screen] screen.
|
||||
Initial window position is the center of [member current_screen] screen.
|
||||
</constant>
|
||||
<constant name="WINDOW_INITIAL_POSITION_CENTER_SCREEN_WITH_MOUSE_FOCUS" value="4" enum="WindowInitialPosition">
|
||||
Initial window position is the center of the screen containing the mouse pointer.
|
||||
</constant>
|
||||
<constant name="WINDOW_INITIAL_POSITION_CENTER_SCREEN_WITH_KEYBOARD_FOCUS" value="5" enum="WindowInitialPosition">
|
||||
Initial window position is the center of the screen containing the window with the keyboard focus.
|
||||
</constant>
|
||||
</constants>
|
||||
<theme_items>
|
||||
|
@ -400,6 +400,13 @@ void EditorSettings::_load_defaults(Ref<ConfigFile> p_extra_config) {
|
||||
const String display_scale_hint_string = vformat("Auto (%d%%),75%%,100%%,125%%,150%%,175%%,200%%,Custom", Math::round(get_auto_display_scale() * 100));
|
||||
EDITOR_SETTING_USAGE(Variant::INT, PROPERTY_HINT_ENUM, "interface/editor/display_scale", 0, display_scale_hint_string, PROPERTY_USAGE_DEFAULT | PROPERTY_USAGE_RESTART_IF_CHANGED)
|
||||
|
||||
String ed_screen_hints = "Screen With Mouse Pointer:-4,Screen With Keyboard Focus:-3,Primary Screen:-2"; // Note: Main Window Screen:-1 is not used for the main window.
|
||||
for (int i = 0; i < DisplayServer::get_singleton()->get_screen_count(); i++) {
|
||||
ed_screen_hints += ",Screen " + itos(i + 1) + ":" + itos(i);
|
||||
}
|
||||
EDITOR_SETTING(Variant::INT, PROPERTY_HINT_ENUM, "interface/editor/editor_screen", -2, ed_screen_hints)
|
||||
EDITOR_SETTING(Variant::INT, PROPERTY_HINT_ENUM, "interface/editor/project_manager_screen", -2, ed_screen_hints)
|
||||
|
||||
_initial_set("interface/editor/debug/enable_pseudolocalization", false);
|
||||
set_restart_if_changed("interface/editor/debug/enable_pseudolocalization", true);
|
||||
// Use pseudolocalization in editor.
|
||||
@ -709,9 +716,9 @@ void EditorSettings::_load_defaults(Ref<ConfigFile> p_extra_config) {
|
||||
// Window placement
|
||||
EDITOR_SETTING(Variant::INT, PROPERTY_HINT_ENUM, "run/window_placement/rect", 1, "Top Left,Centered,Custom Position,Force Maximized,Force Fullscreen")
|
||||
// Keep the enum values in sync with the `DisplayServer::SCREEN_` enum.
|
||||
String screen_hints = "Same as Editor:-5,Previous Monitor:-4,Next Monitor:-3,Primary Monitor:-2"; // Note: Main Window Screen:-1 is not used for the main window.
|
||||
String screen_hints = "Same as Editor:-5,Previous Screen:-4,Next Screen:-3,Primary Screen:-2"; // Note: Main Window Screen:-1 is not used for the main window.
|
||||
for (int i = 0; i < DisplayServer::get_singleton()->get_screen_count(); i++) {
|
||||
screen_hints += ",Monitor " + itos(i + 1) + ":" + itos(i);
|
||||
screen_hints += ",Screen " + itos(i + 1) + ":" + itos(i);
|
||||
}
|
||||
_initial_set("run/window_placement/rect_custom_position", Vector2());
|
||||
EDITOR_SETTING(Variant::INT, PROPERTY_HINT_ENUM, "run/window_placement/screen", -5, screen_hints)
|
||||
|
@ -1969,6 +1969,53 @@ Error Main::setup2(Thread::ID p_main_tid_override) {
|
||||
#ifdef TOOLS_ENABLED
|
||||
if (editor || project_manager || cmdline_tool) {
|
||||
EditorPaths::create();
|
||||
|
||||
// Editor setting class is not available, load config directly.
|
||||
if (!init_use_custom_screen && (editor || project_manager) && EditorPaths::get_singleton()->are_paths_valid()) {
|
||||
Ref<DirAccess> dir = DirAccess::open(EditorPaths::get_singleton()->get_config_dir());
|
||||
String config_file_name = "editor_settings-" + itos(VERSION_MAJOR) + ".tres";
|
||||
String config_file_path = EditorPaths::get_singleton()->get_config_dir().path_join(config_file_name);
|
||||
if (dir->file_exists(config_file_name)) {
|
||||
Error err;
|
||||
Ref<FileAccess> f = FileAccess::open(config_file_path, FileAccess::READ, &err);
|
||||
if (f.is_valid()) {
|
||||
VariantParser::StreamFile stream;
|
||||
stream.f = f;
|
||||
|
||||
String assign;
|
||||
Variant value;
|
||||
VariantParser::Tag next_tag;
|
||||
|
||||
int lines = 0;
|
||||
String error_text;
|
||||
|
||||
while (true) {
|
||||
assign = Variant();
|
||||
next_tag.fields.clear();
|
||||
next_tag.name = String();
|
||||
|
||||
err = VariantParser::parse_tag_assign_eof(&stream, lines, error_text, next_tag, assign, value, nullptr, true);
|
||||
if (err == ERR_FILE_EOF) {
|
||||
break;
|
||||
}
|
||||
if (err == OK && !assign.is_empty()) {
|
||||
if (project_manager) {
|
||||
if (assign == "interface/editor/project_manager_screen") {
|
||||
init_screen = value;
|
||||
break;
|
||||
}
|
||||
} else if (editor) {
|
||||
if (assign == "interface/editor/editor_screen") {
|
||||
init_screen = value;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (found_project && EditorPaths::get_singleton()->is_self_contained()) {
|
||||
if (ProjectSettings::get_singleton()->get_resource_path() == OS::get_singleton()->get_executable_path().get_base_dir()) {
|
||||
ERR_PRINT("You are trying to run a self-contained editor at the same location as a project. This is not allowed, since editor files will mix with project files.");
|
||||
|
@ -753,23 +753,56 @@ int DisplayServerX11::get_screen_count() const {
|
||||
}
|
||||
|
||||
int DisplayServerX11::get_primary_screen() const {
|
||||
return XDefaultScreen(x11_display);
|
||||
int event_base, error_base;
|
||||
if (XineramaQueryExtension(x11_display, &event_base, &error_base)) {
|
||||
return 0;
|
||||
} else {
|
||||
return XDefaultScreen(x11_display);
|
||||
}
|
||||
}
|
||||
|
||||
int DisplayServerX11::get_keyboard_focus_screen() const {
|
||||
int count = get_screen_count();
|
||||
if (count < 2) {
|
||||
// Early exit with single monitor.
|
||||
return 0;
|
||||
}
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
return get_primary_screen();
|
||||
}
|
||||
|
||||
Rect2i DisplayServerX11::_screen_get_rect(int p_screen) const {
|
||||
Rect2i rect(0, 0, 0, 0);
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
ERR_FAIL_COND_V(p_screen < 0, rect);
|
||||
|
||||
// Using Xinerama Extension.
|
||||
@ -834,17 +867,7 @@ int bad_window_error_handler(Display *display, XErrorEvent *error) {
|
||||
Rect2i DisplayServerX11::screen_get_usable_rect(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
int screen_count = get_screen_count();
|
||||
|
||||
// Check if screen is valid.
|
||||
@ -1121,18 +1144,7 @@ Rect2i DisplayServerX11::screen_get_usable_rect(int p_screen) const {
|
||||
int DisplayServerX11::screen_get_dpi(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
//invalid screen?
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
ERR_FAIL_INDEX_V(p_screen, get_screen_count(), 0);
|
||||
|
||||
//Get physical monitor Dimensions through XRandR and calculate dpi
|
||||
@ -1196,18 +1208,7 @@ Color DisplayServerX11::screen_get_pixel(const Point2i &p_position) const {
|
||||
float DisplayServerX11::screen_get_refresh_rate(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
//invalid screen?
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
ERR_FAIL_INDEX_V(p_screen, get_screen_count(), SCREEN_REFRESH_RATE_FALLBACK);
|
||||
|
||||
//Use xrandr to get screen refresh rate.
|
||||
@ -1570,18 +1571,7 @@ void DisplayServerX11::window_set_current_screen(int p_screen, WindowID p_window
|
||||
ERR_FAIL_COND(!windows.has(p_window));
|
||||
WindowData &wd = windows[p_window];
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
// Check if screen is valid
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
ERR_FAIL_INDEX(p_screen, get_screen_count());
|
||||
|
||||
if (window_get_current_screen(p_window) == p_screen) {
|
||||
|
@ -406,6 +406,7 @@ public:
|
||||
|
||||
virtual int get_screen_count() const override;
|
||||
virtual int get_primary_screen() const override;
|
||||
virtual int get_keyboard_focus_screen() const override;
|
||||
virtual Point2i screen_get_position(int p_screen = SCREEN_OF_MAIN_WINDOW) const override;
|
||||
virtual Size2i screen_get_size(int p_screen = SCREEN_OF_MAIN_WINDOW) const override;
|
||||
virtual Rect2i screen_get_usable_rect(int p_screen = SCREEN_OF_MAIN_WINDOW) const override;
|
||||
|
@ -327,6 +327,7 @@ public:
|
||||
|
||||
virtual int get_screen_count() const override;
|
||||
virtual int get_primary_screen() const override;
|
||||
virtual int get_keyboard_focus_screen() const override;
|
||||
virtual Point2i screen_get_position(int p_screen = SCREEN_OF_MAIN_WINDOW) const override;
|
||||
virtual Size2i screen_get_size(int p_screen = SCREEN_OF_MAIN_WINDOW) const override;
|
||||
virtual int screen_get_dpi(int p_screen = SCREEN_OF_MAIN_WINDOW) const override;
|
||||
|
@ -2112,20 +2112,15 @@ int DisplayServerMacOS::get_primary_screen() const {
|
||||
return 0;
|
||||
}
|
||||
|
||||
int DisplayServerMacOS::get_keyboard_focus_screen() const {
|
||||
const NSUInteger index = [[NSScreen screens] indexOfObject:[NSScreen mainScreen]];
|
||||
return (index == NSNotFound) ? 0 : index;
|
||||
}
|
||||
|
||||
Point2i DisplayServerMacOS::screen_get_position(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
Point2i position = _get_native_screen_position(p_screen) - _get_screens_origin();
|
||||
// macOS native y-coordinate relative to _get_screens_origin() is negative,
|
||||
// Godot expects a positive value.
|
||||
@ -2136,17 +2131,7 @@ Point2i DisplayServerMacOS::screen_get_position(int p_screen) const {
|
||||
Size2i DisplayServerMacOS::screen_get_size(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
NSArray *screenArray = [NSScreen screens];
|
||||
if ((NSUInteger)p_screen < [screenArray count]) {
|
||||
// Note: Use frame to get the whole screen size.
|
||||
@ -2160,17 +2145,7 @@ Size2i DisplayServerMacOS::screen_get_size(int p_screen) const {
|
||||
int DisplayServerMacOS::screen_get_dpi(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
NSArray *screenArray = [NSScreen screens];
|
||||
if ((NSUInteger)p_screen < [screenArray count]) {
|
||||
NSDictionary *description = [[screenArray objectAtIndex:p_screen] deviceDescription];
|
||||
@ -2191,17 +2166,7 @@ int DisplayServerMacOS::screen_get_dpi(int p_screen) const {
|
||||
float DisplayServerMacOS::screen_get_scale(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
if (OS::get_singleton()->is_hidpi_allowed()) {
|
||||
NSArray *screenArray = [NSScreen screens];
|
||||
if ((NSUInteger)p_screen < [screenArray count]) {
|
||||
@ -2224,17 +2189,7 @@ float DisplayServerMacOS::screen_get_max_scale() const {
|
||||
Rect2i DisplayServerMacOS::screen_get_usable_rect(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
NSArray *screenArray = [NSScreen screens];
|
||||
if ((NSUInteger)p_screen < [screenArray count]) {
|
||||
const float scale = screen_get_max_scale();
|
||||
@ -2282,17 +2237,7 @@ Color DisplayServerMacOS::screen_get_pixel(const Point2i &p_position) const {
|
||||
float DisplayServerMacOS::screen_get_refresh_rate(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
NSArray *screenArray = [NSScreen screens];
|
||||
if ((NSUInteger)p_screen < [screenArray count]) {
|
||||
NSDictionary *description = [[screenArray objectAtIndex:p_screen] deviceDescription];
|
||||
|
@ -388,6 +388,17 @@ int DisplayServerWindows::get_primary_screen() const {
|
||||
return data.screen;
|
||||
}
|
||||
|
||||
int DisplayServerWindows::get_keyboard_focus_screen() const {
|
||||
HWND hwnd = GetForegroundWindow();
|
||||
if (hwnd) {
|
||||
EnumScreenData data = { 0, 0, MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST) };
|
||||
EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcScreen, (LPARAM)&data);
|
||||
return data.screen;
|
||||
} else {
|
||||
return get_primary_screen();
|
||||
}
|
||||
}
|
||||
|
||||
typedef struct {
|
||||
int count;
|
||||
int screen;
|
||||
@ -424,17 +435,7 @@ Point2i DisplayServerWindows::_get_screens_origin() const {
|
||||
Point2i DisplayServerWindows::screen_get_position(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
EnumPosData data = { 0, p_screen, Point2() };
|
||||
EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcPos, (LPARAM)&data);
|
||||
return data.pos - _get_screens_origin();
|
||||
@ -472,17 +473,7 @@ static BOOL CALLBACK _MonitorEnumProcSize(HMONITOR hMonitor, HDC hdcMonitor, LPR
|
||||
Size2i DisplayServerWindows::screen_get_size(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
EnumSizeData data = { 0, p_screen, Size2() };
|
||||
EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcSize, (LPARAM)&data);
|
||||
return data.size;
|
||||
@ -529,17 +520,7 @@ static BOOL CALLBACK _MonitorEnumProcRefreshRate(HMONITOR hMonitor, HDC hdcMonit
|
||||
Rect2i DisplayServerWindows::screen_get_usable_rect(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
EnumRectData data = { 0, p_screen, Rect2i() };
|
||||
EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcUsableSize, (LPARAM)&data);
|
||||
data.rect.position -= _get_screens_origin();
|
||||
@ -617,17 +598,7 @@ static BOOL CALLBACK _MonitorEnumProcDpi(HMONITOR hMonitor, HDC hdcMonitor, LPRE
|
||||
int DisplayServerWindows::screen_get_dpi(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
EnumDpiData data = { 0, p_screen, 72 };
|
||||
EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcDpi, (LPARAM)&data);
|
||||
return data.dpi;
|
||||
@ -655,17 +626,7 @@ Color DisplayServerWindows::screen_get_pixel(const Point2i &p_position) const {
|
||||
float DisplayServerWindows::screen_get_refresh_rate(int p_screen) const {
|
||||
_THREAD_SAFE_METHOD_
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
p_screen = _get_screen_index(p_screen);
|
||||
EnumRefreshRateData data = { 0, p_screen, SCREEN_REFRESH_RATE_FALLBACK };
|
||||
EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcRefreshRate, (LPARAM)&data);
|
||||
return data.rate;
|
||||
|
@ -523,6 +523,7 @@ public:
|
||||
|
||||
virtual int get_screen_count() const override;
|
||||
virtual int get_primary_screen() const override;
|
||||
virtual int get_keyboard_focus_screen() const override;
|
||||
virtual Point2i screen_get_position(int p_screen = SCREEN_OF_MAIN_WINDOW) const override;
|
||||
virtual Size2i screen_get_size(int p_screen = SCREEN_OF_MAIN_WINDOW) const override;
|
||||
virtual Rect2i screen_get_usable_rect(int p_screen = SCREEN_OF_MAIN_WINDOW) const override;
|
||||
|
@ -516,7 +516,12 @@ void Window::_make_window() {
|
||||
window_rect = Rect2i(DisplayServer::get_singleton()->screen_get_position(DisplayServer::SCREEN_OF_MAIN_WINDOW) + (DisplayServer::get_singleton()->screen_get_size(DisplayServer::SCREEN_OF_MAIN_WINDOW) - size) / 2, size);
|
||||
} else if (initial_position == WINDOW_INITIAL_POSITION_CENTER_OTHER_SCREEN) {
|
||||
window_rect = Rect2i(DisplayServer::get_singleton()->screen_get_position(current_screen) + (DisplayServer::get_singleton()->screen_get_size(current_screen) - size) / 2, size);
|
||||
} else if (initial_position == WINDOW_INITIAL_POSITION_CENTER_SCREEN_WITH_MOUSE_FOCUS) {
|
||||
window_rect = Rect2i(DisplayServer::get_singleton()->screen_get_position(DisplayServer::SCREEN_WITH_MOUSE_FOCUS) + (DisplayServer::get_singleton()->screen_get_size(DisplayServer::SCREEN_WITH_MOUSE_FOCUS) - size) / 2, size);
|
||||
} else if (initial_position == WINDOW_INITIAL_POSITION_CENTER_SCREEN_WITH_KEYBOARD_FOCUS) {
|
||||
window_rect = Rect2i(DisplayServer::get_singleton()->screen_get_position(DisplayServer::SCREEN_WITH_KEYBOARD_FOCUS) + (DisplayServer::get_singleton()->screen_get_size(DisplayServer::SCREEN_WITH_KEYBOARD_FOCUS) - size) / 2, size);
|
||||
}
|
||||
|
||||
window_id = DisplayServer::get_singleton()->create_sub_window(DisplayServer::WindowMode(mode), vsync_mode, f, window_rect);
|
||||
ERR_FAIL_COND(window_id == DisplayServer::INVALID_WINDOW_ID);
|
||||
DisplayServer::get_singleton()->window_set_max_size(Size2i(), window_id);
|
||||
@ -2427,7 +2432,7 @@ void Window::_bind_methods() {
|
||||
ClassDB::bind_method(D_METHOD("popup_centered", "minsize"), &Window::popup_centered, DEFVAL(Size2i()));
|
||||
ClassDB::bind_method(D_METHOD("popup_centered_clamped", "minsize", "fallback_ratio"), &Window::popup_centered_clamped, DEFVAL(Size2i()), DEFVAL(0.75));
|
||||
|
||||
ADD_PROPERTY(PropertyInfo(Variant::INT, "initial_position", PROPERTY_HINT_ENUM, "Absolute,Primary Screen Center,Main Window Screen Center,Other Screen Center"), "set_initial_position", "get_initial_position");
|
||||
ADD_PROPERTY(PropertyInfo(Variant::INT, "initial_position", PROPERTY_HINT_ENUM, "Absolute,Center of Primary Screen,Center of Other Screen,Center of Screen With Mouse Pointer,Center of Screen With Keyboard Focus"), "set_initial_position", "get_initial_position");
|
||||
ADD_PROPERTY(PropertyInfo(Variant::STRING, "title"), "set_title", "get_title");
|
||||
ADD_PROPERTY(PropertyInfo(Variant::VECTOR2I, "position", PROPERTY_HINT_NONE, "suffix:px"), "set_position", "get_position");
|
||||
ADD_PROPERTY(PropertyInfo(Variant::VECTOR2I, "size", PROPERTY_HINT_NONE, "suffix:px"), "set_size", "get_size");
|
||||
@ -2521,6 +2526,8 @@ void Window::_bind_methods() {
|
||||
BIND_ENUM_CONSTANT(WINDOW_INITIAL_POSITION_CENTER_PRIMARY_SCREEN);
|
||||
BIND_ENUM_CONSTANT(WINDOW_INITIAL_POSITION_CENTER_MAIN_WINDOW_SCREEN);
|
||||
BIND_ENUM_CONSTANT(WINDOW_INITIAL_POSITION_CENTER_OTHER_SCREEN);
|
||||
BIND_ENUM_CONSTANT(WINDOW_INITIAL_POSITION_CENTER_SCREEN_WITH_MOUSE_FOCUS);
|
||||
BIND_ENUM_CONSTANT(WINDOW_INITIAL_POSITION_CENTER_SCREEN_WITH_KEYBOARD_FOCUS);
|
||||
}
|
||||
|
||||
Window::Window() {
|
||||
|
@ -93,6 +93,8 @@ public:
|
||||
WINDOW_INITIAL_POSITION_CENTER_PRIMARY_SCREEN,
|
||||
WINDOW_INITIAL_POSITION_CENTER_MAIN_WINDOW_SCREEN,
|
||||
WINDOW_INITIAL_POSITION_CENTER_OTHER_SCREEN,
|
||||
WINDOW_INITIAL_POSITION_CENTER_SCREEN_WITH_MOUSE_FOCUS,
|
||||
WINDOW_INITIAL_POSITION_CENTER_SCREEN_WITH_KEYBOARD_FOCUS,
|
||||
};
|
||||
|
||||
private:
|
||||
|
@ -649,6 +649,7 @@ void DisplayServer::_bind_methods() {
|
||||
|
||||
ClassDB::bind_method(D_METHOD("get_screen_count"), &DisplayServer::get_screen_count);
|
||||
ClassDB::bind_method(D_METHOD("get_primary_screen"), &DisplayServer::get_primary_screen);
|
||||
ClassDB::bind_method(D_METHOD("get_keyboard_focus_screen"), &DisplayServer::get_keyboard_focus_screen);
|
||||
ClassDB::bind_method(D_METHOD("get_screen_from_rect", "rect"), &DisplayServer::get_screen_from_rect);
|
||||
ClassDB::bind_method(D_METHOD("screen_get_position", "screen"), &DisplayServer::screen_get_position, DEFVAL(SCREEN_OF_MAIN_WINDOW));
|
||||
ClassDB::bind_method(D_METHOD("screen_get_size", "screen"), &DisplayServer::screen_get_size, DEFVAL(SCREEN_OF_MAIN_WINDOW));
|
||||
@ -794,6 +795,8 @@ void DisplayServer::_bind_methods() {
|
||||
BIND_ENUM_CONSTANT(MOUSE_MODE_CONFINED);
|
||||
BIND_ENUM_CONSTANT(MOUSE_MODE_CONFINED_HIDDEN);
|
||||
|
||||
BIND_CONSTANT(SCREEN_WITH_MOUSE_FOCUS);
|
||||
BIND_CONSTANT(SCREEN_WITH_KEYBOARD_FOCUS);
|
||||
BIND_CONSTANT(SCREEN_PRIMARY);
|
||||
BIND_CONSTANT(SCREEN_OF_MAIN_WINDOW);
|
||||
|
||||
|
@ -253,14 +253,38 @@ public:
|
||||
virtual Rect2i get_display_safe_area() const { return screen_get_usable_rect(); }
|
||||
|
||||
enum {
|
||||
SCREEN_WITH_MOUSE_FOCUS = -4,
|
||||
SCREEN_WITH_KEYBOARD_FOCUS = -3,
|
||||
SCREEN_PRIMARY = -2,
|
||||
SCREEN_OF_MAIN_WINDOW = -1, // Note: for the main window, determine screen from position.
|
||||
};
|
||||
|
||||
const float SCREEN_REFRESH_RATE_FALLBACK = -1.0; // Returned by screen_get_refresh_rate if the method fails.
|
||||
|
||||
int _get_screen_index(int p_screen) const {
|
||||
switch (p_screen) {
|
||||
case SCREEN_WITH_MOUSE_FOCUS: {
|
||||
const Rect2i rect = Rect2i(mouse_get_position(), Vector2i(1, 1));
|
||||
return get_screen_from_rect(rect);
|
||||
} break;
|
||||
case SCREEN_WITH_KEYBOARD_FOCUS: {
|
||||
return get_keyboard_focus_screen();
|
||||
} break;
|
||||
case SCREEN_PRIMARY: {
|
||||
return get_primary_screen();
|
||||
} break;
|
||||
case SCREEN_OF_MAIN_WINDOW: {
|
||||
return window_get_current_screen(MAIN_WINDOW_ID);
|
||||
} break;
|
||||
default: {
|
||||
return p_screen;
|
||||
} break;
|
||||
}
|
||||
}
|
||||
|
||||
virtual int get_screen_count() const = 0;
|
||||
virtual int get_primary_screen() const = 0;
|
||||
virtual int get_keyboard_focus_screen() const { return get_primary_screen(); }
|
||||
virtual int get_screen_from_rect(const Rect2 &p_rect) const;
|
||||
virtual Point2i screen_get_position(int p_screen = SCREEN_OF_MAIN_WINDOW) const = 0;
|
||||
virtual Size2i screen_get_size(int p_screen = SCREEN_OF_MAIN_WINDOW) const = 0;
|
||||
|
Loading…
Reference in New Issue
Block a user