Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(core): add an option for preventing window overflow #9687

Open
wants to merge 72 commits into
base: dev
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
72 commits
Select commit Hold shift + click to select a range
d9e2626
Add an option for preventing window overflow
Legend-Master May 7, 2024
02ddb4f
Fix shadow counted as part of the width
Legend-Master May 7, 2024
6a3b1c4
Enable prevent overflow by default
Legend-Master May 7, 2024
9a8c4ab
Fix crashing when margin is bigger than screen
Legend-Master May 7, 2024
ae1d90d
Merge branch 'dev' of https://github.com/tauri-apps/tauri into preven…
Legend-Master May 7, 2024
01a8e7a
Config file support
Legend-Master May 7, 2024
5aa3eda
Add to ts type
Legend-Master May 7, 2024
b4f61c0
Add mac implementation
Legend-Master May 8, 2024
3451755
Should be let some
Legend-Master May 8, 2024
d5c2352
Apply suggestions from code review
Legend-Master May 8, 2024
097a5ed
Merge branch 'dev' into prevent-window-overflow
Legend-Master May 8, 2024
a5a028a
Merge branch 'dev' into prevent-window-overflow
Legend-Master May 10, 2024
e8026f7
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master May 14, 2024
2100bda
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master May 14, 2024
a9283cb
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master May 25, 2024
b92a7e7
checked_sub -> saturating_sub
Legend-Master May 25, 2024
b548734
Merge branch 'dev' into prevent-window-overflow
Legend-Master Jul 1, 2024
7c519f2
Revert license header
Legend-Master Jul 1, 2024
d7fd244
Merge branch 'dev' into prevent-window-overflow
Legend-Master Aug 8, 2024
ec6e483
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Aug 28, 2024
aab7893
Migrate windows 0.58
Legend-Master Aug 28, 2024
b7c5e23
Generate schema
Legend-Master Aug 28, 2024
2bbb42a
Merge branch 'dev' into prevent-window-overflow
Legend-Master Nov 5, 2024
03ed8c5
Fix merge conflict
Legend-Master Nov 5, 2024
ef94548
Try casting to objc2_app_kit::NSScreen
Legend-Master Nov 5, 2024
de09f3a
Add NSScreen feature
Legend-Master Nov 5, 2024
7656966
Use cast instead of as
Legend-Master Nov 5, 2024
1b4d7b0
fmt
Legend-Master Nov 5, 2024
5083149
Deref first
Legend-Master Nov 5, 2024
fe638b1
Wrong unsafe block
Legend-Master Nov 5, 2024
b6661bb
Add change file
Legend-Master Nov 5, 2024
13b4a1d
Add serde alias
Legend-Master Nov 5, 2024
c7714fb
More docs
Legend-Master Nov 5, 2024
aeeaae8
Disable prevent_overflow by default
Legend-Master Nov 5, 2024
ab7a48f
Remove unused default impl
Legend-Master Nov 5, 2024
86ef220
Merge branch 'dev' into prevent-window-overflow
Legend-Master Nov 5, 2024
e880df2
Missing pub
Legend-Master Nov 5, 2024
e59669f
Fix mock_runtime on ios
Legend-Master Nov 5, 2024
dde6abc
Merge branch 'dev' into prevent-window-overflow
Legend-Master Nov 6, 2024
07255da
Match PreventOverflowMargin's description
Legend-Master Nov 6, 2024
035e6bc
Merge branch 'dev' into prevent-window-overflow
Legend-Master Nov 7, 2024
52b9c7a
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Nov 7, 2024
4c944ee
Merge branch 'dev' into prevent-window-overflow
Legend-Master Nov 10, 2024
d492e9c
Typo
Legend-Master Nov 12, 2024
16f4895
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Nov 12, 2024
d887510
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Nov 13, 2024
7e84cb1
Apply suggestions from code review
Legend-Master Nov 13, 2024
a5a8b81
Move get_work_area_size to seperate files
Legend-Master Nov 13, 2024
ade88e8
Unused imports
Legend-Master Nov 13, 2024
a7bd0d2
Add prevent_overflow to WebviewWindowBuilder
Legend-Master Nov 13, 2024
deddb12
Fix mac compile
Legend-Master Nov 13, 2024
edf6132
MonitorExt is only for desktop
Legend-Master Nov 13, 2024
657b479
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Nov 13, 2024
a01d5c9
Rename to work_area
Legend-Master Nov 15, 2024
8bfd36e
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Nov 15, 2024
4e6cded
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Dec 9, 2024
cadb7a7
Use workarea for linux
Legend-Master Dec 9, 2024
984cc79
Missing `()`
Legend-Master Dec 9, 2024
9726b8c
Convert size
Legend-Master Dec 9, 2024
ab75756
Import MonitorExt
Legend-Master Dec 9, 2024
dc0eb0a
as u32
Legend-Master Dec 9, 2024
87941e9
Re-build API js
Legend-Master Dec 9, 2024
5bd115a
Fix wrong docs for work_area
Legend-Master Dec 9, 2024
5f30b49
Remove linux platform specific note
Legend-Master Dec 9, 2024
d5c2e8c
Remove left over linux platform specific note
Legend-Master Dec 10, 2024
8e1ca44
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Dec 11, 2024
bdb8d2e
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Dec 11, 2024
e87436e
Merge branch 'dev' into prevent-window-overflow
Legend-Master Jan 6, 2025
65f4a6b
Use work area API for center as well
Legend-Master Jan 6, 2025
0103bf3
Fix mobile
Legend-Master Jan 6, 2025
e67f344
Clean up
Legend-Master Jan 6, 2025
7850991
Merge remote-tracking branch 'upstream/dev' into prevent-window-overflow
Legend-Master Jan 17, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
51 changes: 51 additions & 0 deletions core/tauri-config-schema/schema.json
Original file line number Diff line number Diff line change
Expand Up @@ -291,6 +291,17 @@
],
"format": "double"
},
"preventOverflow": {
"description": "Whether or not to prevent window overflow",
"anyOf": [
{
"$ref": "#/definitions/PreventOverflowMarginConfig"
},
{
"type": "null"
}
]
},
"resizable": {
"description": "Whether the window is resizable or not. When resizable is set to false, native window's maximize button is automatically disabled.",
"default": true,
Expand Down Expand Up @@ -478,6 +489,46 @@
}
]
},
"PreventOverflowMarginConfig": {
"description": "Prevent overflow with a margin",
"anyOf": [
{
"description": "Enable prevent overflow or not",
"type": "boolean"
},
{
"description": "Enable prevent overflow with a margin",
"allOf": [
{
"$ref": "#/definitions/PreventOverflowMargin"
}
]
}
]
},
"PreventOverflowMargin": {
"description": "Enable prevent overflow with a margin",
"type": "object",
"required": [
"height",
"width"
],
"properties": {
"width": {
"description": "Horizontal margin in physical unit",
"type": "integer",
"format": "uint32",
"minimum": 0.0
},
"height": {
"description": "Vertical margin in physical unit",
"type": "integer",
"format": "uint32",
"minimum": 0.0
}
},
"additionalProperties": false
},
"Theme": {
"description": "System theme.",
"oneOf": [
Expand Down
139 changes: 116 additions & 23 deletions core/tauri-runtime-wry/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,10 @@ use tao::{
};
#[cfg(target_os = "macos")]
use tauri_utils::TitleBarStyle;
use tauri_utils::{config::WindowConfig, Theme};
use tauri_utils::{
config::{PreventOverflowMarginConfig, WindowConfig},
Theme,
};
use url::Url;
use wry::{
DragDropEvent as WryDragDropEvent, ProxyConfig, ProxyEndpoint, WebContext, WebView,
Expand Down Expand Up @@ -691,18 +694,33 @@ impl From<ProgressBarState> for ProgressBarStateWrapper {
}
}

#[derive(Clone, Default)]
#[derive(Clone)]
pub struct WindowBuilderWrapper {
inner: TaoWindowBuilder,
center: bool,
prevent_overflow: Option<Size>,
#[cfg(target_os = "macos")]
tabbing_identifier: Option<String>,
}

impl Default for WindowBuilderWrapper {
fn default() -> Self {
Self {
inner: Default::default(),
center: Default::default(),
prevent_overflow: Some(PhysicalSize::new(0, 0).into()),
#[cfg(target_os = "macos")]
tabbing_identifier: Default::default(),
}
}
}

impl std::fmt::Debug for WindowBuilderWrapper {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let mut s = f.debug_struct("WindowBuilderWrapper");
s.field("inner", &self.inner).field("center", &self.center);
s.field("inner", &self.inner)
.field("center", &self.center)
.field("prevent_overflow", &self.prevent_overflow);
#[cfg(target_os = "macos")]
{
s.field("tabbing_identifier", &self.tabbing_identifier);
Expand Down Expand Up @@ -790,6 +808,16 @@ impl WindowBuilder for WindowBuilderWrapper {
if config.center {
window = window.center();
}

if let Some(prevent_overflow) = &config.prevent_overflow {
window = match prevent_overflow {
PreventOverflowMarginConfig::Enable(false) => window.prevent_overflow(None),
PreventOverflowMarginConfig::Margin(margin) => window.prevent_overflow(Some(
TaoPhysicalSize::new(margin.width, margin.height).into(),
)),
_ => window,
};
}
}

window
Expand Down Expand Up @@ -826,6 +854,13 @@ impl WindowBuilder for WindowBuilderWrapper {
self
}

/// Prevent the window from overflowing the working area (e.g. monitor size - taskbar size) on creation
#[must_use]
fn prevent_overflow(mut self, margin: Option<Size>) -> Self {
self.prevent_overflow = margin;
self
}

fn resizable(mut self, resizable: bool) -> Self {
self.inner = self.inner.with_resizable(resizable);
self
Expand Down Expand Up @@ -2702,7 +2737,7 @@ fn handle_user_message<T: UserEvent>(
window_size.height = (rect.bottom - rect.top) as u32;
}
}
window.set_outer_position(calculate_window_center_position(window_size, monitor));
window.set_outer_position(calculate_window_center_position(window_size, &monitor));
}

#[cfg(target_os = "macos")]
Expand Down Expand Up @@ -3542,7 +3577,7 @@ fn create_window<T: UserEvent, F: Fn(RawWindow) + Send + 'static>(
}

#[cfg(desktop)]
if window_builder.center {
if window_builder.prevent_overflow.is_some() || window_builder.center {
let monitor = if let Some(window_position) = &window_builder.inner.window.position {
event_loop.available_monitors().find(|m| {
let monitor_pos = m.position();
Expand All @@ -3559,39 +3594,67 @@ fn create_window<T: UserEvent, F: Fn(RawWindow) + Send + 'static>(
} else {
event_loop.primary_monitor()
};

if let Some(monitor) = monitor {
let scale_factor = monitor.scale_factor();
let desired_size = window_builder
.inner
.window
.inner_size
.unwrap_or_else(|| TaoPhysicalSize::new(800, 600).into());
let scale_factor = monitor.scale_factor();
#[allow(unused_mut)]
let mut window_size = window_builder
let mut inner_size = window_builder
.inner
.window
.inner_size_constraints
.clamp(desired_size, scale_factor)
.to_physical::<u32>(scale_factor);
let mut window_size = inner_size;
#[allow(unused_mut)]
// Left and right window shadow counts as part of the window on Windows
// We need to include it when calculating positions, but not size
let mut shadow_width = 0;
#[cfg(windows)]
{
if window_builder.inner.window.decorations {
use windows::Win32::UI::WindowsAndMessaging::{AdjustWindowRect, WS_OVERLAPPEDWINDOW};
let mut rect = windows::Win32::Foundation::RECT::default();
let result = unsafe { AdjustWindowRect(&mut rect, WS_OVERLAPPEDWINDOW, false) };
if result.is_ok() {
window_size.width += (rect.right - rect.left) as u32;
// rect.bottom is made out of shadow, and we don't care about it
window_size.height += -rect.top as u32;
if window_builder.inner.window.decorations {
use windows::Win32::UI::WindowsAndMessaging::{AdjustWindowRect, WS_OVERLAPPEDWINDOW};
let mut rect = windows::Win32::Foundation::RECT::default();
let result = unsafe { AdjustWindowRect(&mut rect, WS_OVERLAPPEDWINDOW, false) };
if result.is_ok() {
shadow_width = (rect.right - rect.left) as u32;
// rect.bottom is made out of shadow, and we don't care about it
window_size.height += -rect.top as u32;
}
}

if let Some(margin) = window_builder.prevent_overflow {
let size = get_work_area_size(&monitor);
let margin = margin.to_physical::<u32>(scale_factor);
let constraint = PhysicalSize::new(size.width - margin.width, size.height - margin.height);
if window_size.width > constraint.width || window_size.height > constraint.height {
if window_size.width > constraint.width {
inner_size.width = inner_size
.width
.checked_sub(window_size.width - constraint.width)
.unwrap_or_default();
window_size.width = constraint.width;
}
if window_size.height > constraint.height {
inner_size.height = inner_size
.height
.checked_sub(window_size.height - constraint.height)
.unwrap_or_default();
window_size.height = constraint.height;
}
window_builder.inner.window.inner_size = Some(inner_size.into());
}
}
let position = calculate_window_center_position(window_size, monitor);
let logical_position = position.to_logical::<f64>(scale_factor);
window_builder = window_builder.position(logical_position.x, logical_position.y);

if window_builder.center {
window_size.width += shadow_width;
let position = calculate_window_center_position(window_size, &monitor);
let logical_position = position.to_logical::<f64>(scale_factor);
window_builder = window_builder.position(logical_position.x, logical_position.y);
}
}
}
};

let window = window_builder.inner.build(event_loop).unwrap();

Expand Down Expand Up @@ -4128,9 +4191,39 @@ fn inner_size(
window.inner_size()
}

fn get_work_area_size(target_monitor: &MonitorHandle) -> TaoPhysicalSize<u32> {
#[cfg(windows)]
{
use tao::platform::windows::MonitorHandleExtWindows;
use windows::Win32::Graphics::Gdi::{GetMonitorInfoW, HMONITOR, MONITORINFO};
let mut monitor_info = MONITORINFO {
cbSize: std::mem::size_of::<MONITORINFO>() as u32,
..Default::default()
};
let status = unsafe { GetMonitorInfoW(HMONITOR(target_monitor.hmonitor()), &mut monitor_info) };
if status.into() {
return TaoPhysicalSize::new(
(monitor_info.rcWork.right - monitor_info.rcWork.left) as u32,
(monitor_info.rcWork.bottom - monitor_info.rcWork.top) as u32,
);
}
}
#[cfg(target_os = "macos")]
{
use cocoa::{appkit::NSScreen, base::id};
use tao::platform::windows::MonitorHandleExtMacOS;
Legend-Master marked this conversation as resolved.
Show resolved Hide resolved
if let Some(ns_screen) = target_monitor.ns_screen() {
let ns_screen = ns_screen as id;
let rect = unsafe { ns_screen.visibleFrame() };
return TaoPhysicalSize::new(rect.size.width as u32, rect.size.height as u32);
Legend-Master marked this conversation as resolved.
Show resolved Hide resolved
}
}
target_monitor.size()
}

fn calculate_window_center_position(
window_size: TaoPhysicalSize<u32>,
target_monitor: MonitorHandle,
target_monitor: &MonitorHandle,
) -> TaoPhysicalPosition<i32> {
#[cfg(windows)]
{
Expand Down
4 changes: 4 additions & 0 deletions core/tauri-runtime/src/window.rs
Original file line number Diff line number Diff line change
Expand Up @@ -237,6 +237,10 @@ pub trait WindowBuilder: WindowBuilderBase {
#[must_use]
fn max_inner_size(self, max_width: f64, max_height: f64) -> Self;

/// Window max inner size.
#[must_use]
fn prevent_overflow(self, margin: Option<dpi::Size>) -> Self;

/// Whether the window is resizable or not.
/// When resizable is set to false, native window's maximize button is automatically disabled.
#[must_use]
Expand Down
Loading