Merge 345a20a740 into a4cb73db84
commit
b68bffef77
|
|
@ -548,6 +548,14 @@ extension Ghostty {
|
|||
return v;
|
||||
}
|
||||
|
||||
var mouseHideAfter: UInt {
|
||||
guard let config = self.config else { return 0 }
|
||||
var v: UInt = 0
|
||||
let key = "mouse-hide-after"
|
||||
_ = ghostty_config_get(config, &v, key, UInt(key.lengthOfBytes(using: .utf8)))
|
||||
return v;
|
||||
}
|
||||
|
||||
var undoTimeout: Duration {
|
||||
guard let config = self.config else { return .seconds(5) }
|
||||
var v: UInt = 0
|
||||
|
|
|
|||
|
|
@ -133,6 +133,10 @@ extension Ghostty {
|
|||
// then the view is moved to a new window.
|
||||
var initialSize: NSSize? = nil
|
||||
|
||||
// Timer used for `mouse-hide-after` behavior on macOS. This hides the mouse
|
||||
// cursor after a period of no mouse movement.
|
||||
private var mouseHideAfterTimer: Timer? = nil
|
||||
|
||||
// A content size received through sizeDidChange that may in some cases
|
||||
// be different from the frame size.
|
||||
private var contentSizeBacking: NSSize?
|
||||
|
|
@ -778,6 +782,30 @@ extension Ghostty {
|
|||
userInfo: nil))
|
||||
}
|
||||
|
||||
/// Reset the `mouse-hide-after` timer on mouse movement, if configured.
|
||||
private func resetMouseHideAfterTimer() {
|
||||
mouseHideAfterTimer?.invalidate()
|
||||
mouseHideAfterTimer = nil
|
||||
|
||||
// Note: DerivedConfig does not expose the full Ghostty.Config, so we
|
||||
// read directly from the global Ghostty config instead.
|
||||
guard let appDelegate = NSApplication.shared.delegate as? AppDelegate else { return }
|
||||
let ms = appDelegate.ghostty.config.mouseHideAfter
|
||||
guard ms > 0 else { return }
|
||||
|
||||
let interval = TimeInterval(ms) / 1000.0
|
||||
mouseHideAfterTimer = Timer.scheduledTimer(
|
||||
withTimeInterval: interval,
|
||||
repeats: false,
|
||||
block: { [weak self] _ in
|
||||
guard let self else { return }
|
||||
// Hide the cursor. This will remain hidden until the mouse
|
||||
// moves again (or other platform behavior such as new window).
|
||||
NSCursor.setHiddenUntilMouseMoves(true)
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
override func viewDidChangeBackingProperties() {
|
||||
super.viewDidChangeBackingProperties()
|
||||
|
||||
|
|
@ -901,6 +929,8 @@ extension Ghostty {
|
|||
mods: .init(nsFlags: event.modifierFlags)
|
||||
)
|
||||
surfaceModel.sendMousePos(mouseEvent)
|
||||
|
||||
resetMouseHideAfterTimer()
|
||||
}
|
||||
|
||||
override func mouseExited(with event: NSEvent) {
|
||||
|
|
@ -934,6 +964,8 @@ extension Ghostty {
|
|||
)
|
||||
surfaceModel.sendMousePos(mouseEvent)
|
||||
|
||||
resetMouseHideAfterTimer()
|
||||
|
||||
// Handle focus-follows-mouse
|
||||
if let window,
|
||||
let controller = window.windowController as? BaseTerminalController,
|
||||
|
|
|
|||
|
|
@ -587,6 +587,14 @@ pub const Surface = extern struct {
|
|||
// Progress bar
|
||||
progress_bar_timer: ?c_uint = null,
|
||||
|
||||
/// Timer source id for `mouse-hide-after` idle hiding. When non-null,
|
||||
/// a one-shot GLib timeout is scheduled to hide the mouse after the
|
||||
/// configured idle duration.
|
||||
mouse_hide_after_timer: ?c_uint = null,
|
||||
|
||||
/// The last time we observed mouse movement, used for `mouse-hide-after`.
|
||||
last_mouse_move_time: ?std.time.Instant = null,
|
||||
|
||||
// True while the bell is ringing. This will be set to false (after
|
||||
// true) under various scenarios, but can also manually be set to
|
||||
// false by a parent widget.
|
||||
|
|
@ -911,6 +919,24 @@ pub const Surface = extern struct {
|
|||
return @intFromBool(glib.SOURCE_REMOVE);
|
||||
}
|
||||
|
||||
/// Timer callback for `mouse-hide-after`. Hides the mouse after a period
|
||||
/// of no mouse movement.
|
||||
fn mouseHideAfterTimer(ud: ?*anyopaque) callconv(.c) c_int {
|
||||
const self: *Self = @ptrCast(@alignCast(ud orelse
|
||||
return @intFromBool(glib.SOURCE_REMOVE)));
|
||||
const priv = self.private();
|
||||
|
||||
// Clear our timer handle first to avoid reusing it.
|
||||
priv.mouse_hide_after_timer = null;
|
||||
|
||||
// Hide the mouse cursor directly.
|
||||
if (!priv.mouse_hidden) {
|
||||
self.setMouseHidden(true);
|
||||
}
|
||||
|
||||
return @intFromBool(glib.SOURCE_REMOVE);
|
||||
}
|
||||
|
||||
/// Request that this terminal come to the front and become focused.
|
||||
/// It is up to the embedding widget to react to this.
|
||||
pub fn present(self: *Self) void {
|
||||
|
|
@ -1716,6 +1742,13 @@ pub const Surface = extern struct {
|
|||
priv.progress_bar_timer = null;
|
||||
}
|
||||
|
||||
if (priv.mouse_hide_after_timer) |timer| {
|
||||
if (glib.Source.remove(timer) == 0) {
|
||||
log.warn("unable to remove mouse hide-after timer", .{});
|
||||
}
|
||||
priv.mouse_hide_after_timer = null;
|
||||
}
|
||||
|
||||
if (priv.idle_rechild) |v| {
|
||||
if (glib.Source.remove(v) == 0) {
|
||||
log.warn("unable to remove idle source", .{});
|
||||
|
|
@ -2641,6 +2674,12 @@ pub const Surface = extern struct {
|
|||
@abs(priv.cursor_pos.y - pos.y) < 1;
|
||||
if (is_cursor_still) return;
|
||||
|
||||
// If the mouse is currently hidden (for example due to `mouse-hide-after`),
|
||||
// show it again on real mouse movement.
|
||||
if (priv.mouse_hidden) {
|
||||
self.setMouseHidden(false);
|
||||
}
|
||||
|
||||
// If we don't have focus, and we want it, grab it.
|
||||
if (priv.config) |config| {
|
||||
const gl_area_widget = priv.gl_area.as(gtk.Widget);
|
||||
|
|
@ -2654,6 +2693,27 @@ pub const Surface = extern struct {
|
|||
// Our pos changed, update
|
||||
priv.cursor_pos = pos;
|
||||
|
||||
// Track mouse movement time for `mouse-hide-after`.
|
||||
priv.last_mouse_move_time = std.time.Instant.now() catch null;
|
||||
|
||||
// Reset `mouse-hide-after` idle timer if configured.
|
||||
if (priv.mouse_hide_after_timer) |timer| {
|
||||
if (glib.Source.remove(timer) == 0) {
|
||||
log.warn("unable to remove mouse hide-after timer", .{});
|
||||
}
|
||||
priv.mouse_hide_after_timer = null;
|
||||
}
|
||||
if (priv.config) |config| {
|
||||
const ms = config.get().@"mouse-hide-after".asMilliseconds();
|
||||
if (ms > 0) {
|
||||
priv.mouse_hide_after_timer = glib.timeoutAdd(
|
||||
ms,
|
||||
mouseHideAfterTimer,
|
||||
self,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// Notify the callback
|
||||
if (priv.core_surface) |surface| {
|
||||
const gtk_mods = event.getModifierState();
|
||||
|
|
|
|||
|
|
@ -824,6 +824,22 @@ palette: Palette = .{},
|
|||
/// the mouse is shown again when a new window, tab, or split is created.
|
||||
@"mouse-hide-while-typing": bool = false,
|
||||
|
||||
/// Hide the mouse after a period of mouse inactivity (no mouse movement).
|
||||
/// When set, the mouse is hidden once it has been idle for at least this long
|
||||
/// and becomes visible again when the mouse is used (button, movement, etc.).
|
||||
/// Typing does not affect the idle timer for this configuration; once the
|
||||
/// mouse has been idle long enough to hide, it will remain hidden while
|
||||
/// typing until the mouse is moved again.
|
||||
///
|
||||
/// Set this to a duration such as `5s` to hide the mouse 5 seconds after the
|
||||
/// last mouse movement. The default value of `0` disables this behavior.
|
||||
///
|
||||
/// This is not mutually exclusive with `mouse-hide-while-typing`; both can be
|
||||
/// enabled at the same time.
|
||||
///
|
||||
/// Available since 1.3.0
|
||||
@"mouse-hide-after": Duration = .{ .duration = 0 },
|
||||
|
||||
/// When to scroll the surface to the bottom. The format of this is a list of
|
||||
/// options to enable separated by commas. If you prefix an option with `no-`
|
||||
/// then it is disabled. If you omit an option, its default value is used.
|
||||
|
|
|
|||
Loading…
Reference in New Issue