-
Notifications
You must be signed in to change notification settings - Fork 592
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
window:focus() focuses different window of app on same screen #370
Comments
Might be related to #304 |
@tmandry what happens if you call |
hmm, never mind that question, |
Sorry, that was just a placeholder; I tried it with apps other than Chrome and still had the same issue. |
@tmandry could you paste a code snippet to repro? |
I call hs.hints.windowHints() and experiencing the same issue. |
Running into the same issue, it seems to focus the application rather than the window. I have this snippet which lets me choose a specific window to focus, but it isn't able to focus windows on other screens: function createWindowChooser()
choseWindow = function(w)
local window = hs.window.get(w["id"])
hs.alert.show("Switching to" .. window:title())
window:focus()
end
local chooser = hs.chooser.new(choseWindow)
hs.hotkey.bind({"alt"}, "tab", function()
local windows = {}
local wf = hs.window.filter.new()
for _, w in pairs(wf:getWindows()) do
table.insert(windows, {
["text"] = w:title(),
["subText"] = w:application():name(),
["id"] = w:id(),
})
end
chooser:choices(windows)
chooser:show()
end)
end
createWindowChooser() |
FWIW I worked around this issue by calling window:raise() then window:focus(), and the application I was having issue with was iTerm2. |
I wasn't able to get |
I also have this problem. I tried doing For example, say I have two screens, S1 and S2, and three windows, Chrome1, Chrome2, and Term. S1 has Chrome1 and Term. S2 just has Chrome2. If I am focusing Term on Chrome1 and attempt to I'm using 0.9.48 on Sierra. |
I think this is an OS thing - it's activating the application and even though we've said we want to focus another window, it's flipping to the "nearest" window instead, possibly the last window that was the key window. My workaround is to get the application, call |
@cmsj I tried activating the application first and then focusing the window, but I'm getting the same result, it's not activating the window I want, but another focused window of the same application on a different screen. Could we re-open this issue and try to find a solution? My issue affects Firefox. Here is the code I'm running:
|
Update: I'm in High Sierra |
I can confirm that calling activate on the application, and then calling focus after that with the specific window I want, fixes the issue for me (Mojave 10.14.4) |
None of these solution seems to work for me on (Mojave 10.14.5)... :-( My code is finding a particular Chrome window it wants to show to the user, but it fails to reliably show it. It either ends up showing different Chrome window, or showing it but not focusing on it. Here is the best one (showing correct window but not focusing on it):
I have two particular usecases in mind. Let's say I have two chrome windows (A and B) on each screen (1 and 2): A1 and B1 on one screen and A2 and B2 on another. I also have some other app C2 open on screen 2. Expected outcome in both cases to open and focus A2. Use case 1:
Use case 2:
|
I've solved this issue in yabai, by reverse engineering some of the event-handling in the WindowServer. Relevant issue: koekeishiya/yabai#102 The solution relies on some private functions implemented in the SkyLight.framework. I'd be happy to explain the solution if Hammerspoon finds the usage of private APIs acceptable. There is probably a hard limit to which macOS versions are supported. I'm not familiar with when these functions were introduced - I've only tested this on High Sierra 10.13.6 and newer. |
@koekeishiya I would definitely be interested to learn about that! We do use private APIs where necessary, and our current minimum supported version is 10.12, but it's fine if we have new Hammerspoon API that only works on 10.13+. |
Basically what I discovered is that there is a certain category of events that are passed by the system to applications depending on how it gains focus. I'm not exactly sure what this event category is, but I'd refer to them as either system or control events. Anyway, we can then synthesize such an event and send it directly to the target process using its process serial number. The background information that helped me discover this was that I was trying to implement focus follows mouse (autofocus) by looking at how macOS was able to focus a window without raising it when clicking inside the window belonging to an unfocused application while holding the ctrl + alt modifiers. There were multiple such system events being triggered in this scenario. It has been quite some time since I implemented this, so I don't remember all the nitty gritty details of all the events, but the solution for this particular issue boils down to combining the following steps: First just some definitions that are necessary
Actual change in focus:
This method requires the caller to know the psn of the target process, the CGWindowId, and the corresponding AXUIElementRef to perform the operation successfully. Assuming you have the AXUIElementRef, the window id can be retrieved using
which you probably knew already. The remaining information can be retrieved as follows:
|
I found that But I cannot reproduce this issue consistently (I could reproduce this issue yesterday with my work setup, but I cannot reproduce this issue today at home). So I'm not sure if this really works. Maybe we can try the private API listed above? |
These are the steps I used to reliably reproduce this problem:
|
Adding my two cents as a user dealing with this issue for a long time. Given some application
Here is what has worked:
Despite the very short nominal delay of 0.001 seconds, the actual lag is longer but tolerable on my machine. Lowering the delay further does not affect it, it must be caused by the overhead of |
@smackesey a question: does I ask because this will tell us whether it's an issue of giving the app time to activate, or whether its an issue of requiring the Hammerspoon application event loop to advance. (The More detail as to why I'm asking, if you're curious -- you don't need to read this, but I would appreciate an answer to the above, if it's not too much trouble. As we see more complex examples that people come up with, we've found that some combined actions are timing dependent, and we can insert delays at specific points to make them more reliable, while others require the main thread of Hammerspoon to be idle, if only for a few nanoseconds-to-microseconds, in between actions... its one of the reasons I've been working to get coroutines supported and will be introducing a couple of new modules soon which may allow us to rewrite some of these more common actions in a way that gives the application loop more idle time to do the macOS maintenance and upkeep that is expected between such actions. |
@asmagill Just tried |
I encountered this problem while using multiple windows in Kitty, and the issue was resolved by turning off the "Displays have separate Spaces" option for Mission Control. Note that you have to logout for this change to take effect. You can observe a similar effect when using cmd-tab. If you have two windows of an application open on different displays, you will notice that when switching to that application with cmd-tab it will always focus on one of the windows, regardless of which window was focused when the app was last active. Turning off this mission control option also fixes this issue. My guess is that Mission Control is interfering with window focus, such that when you focus either window of a non-active application, it activates that application and then focuses the "favored" window. |
I tried turning off this option and it seemed to improve the behavior of the Hammerspoon. Will test it more. |
I also noticed that this setting changes the user experience significantly so it may not be an appropriate solution to the problem. For example, "Entering Full Screen" action on a window, hides windows on all other displays. Also, the menu bar has to be fixed in a single display and the ordering of windows changes. I used to have 0, 1, 2 ordering for displays (used in |
My comment will be somewhat unrelated, but where is the definition of Closest thing: https://docs.rs/MacTypes-sys/latest/MacTypes_sys/struct.ProcessSerialNumber.html I have some code that I think should be working but I don't get any focusing happening. My PID is coming from /// Type for unique process identifier.
#[repr(C)]
#[derive(Clone, Copy, Debug, Default, Eq, Hash, PartialEq)]
pub struct ProcessSerialNumber {
padding: u32,
val: u32,
}
pub type ProcessSerialNumberPtr = *mut ProcessSerialNumber;
#[test]
pub fn focus_window() {
let pid = 7020;
let wid = 55440;
let mut psn = ProcessSerialNumber {
padding: 0,
val: pid,
};
let ptr = &mut psn as *mut ProcessSerialNumber;
let r = unsafe { _SLPSSetFrontProcessWithOptions(ptr, wid, 0x100) };
println!("{:?}", r);
}
#[link(name = "SkyLight", kind = "framework")]
extern "C" {
fn _SLPSSetFrontProcessWithOptions(
psn: *mut ProcessSerialNumber,
wid: mach_port_t,
mode: mach_port_t,
) -> CFErrorRef;
} Edit: it looks like I have a PID but need to get a PSN. Aren't PSNs deprecated/removed in 12.3.1? How does _SLPSSetFrontProcessWithOptions still work? |
I'm not sure if this answers your question or not, but to get ProcessSerialNumber psn;
psn.highLongOfPSN = 0;
psn.lowLongOfPSN = kCurrentProcess; Header: |
Edit: derp, I misread. Processes.h does indeed contain a definition that can be used to get the process serial number from a PID: |
How do I get kCurrentProcess of another process? From the linked header:
I don't want to focus my PSN, I want to focus the PSN of other apps (that I've scraped CGWindowListCopyWindowInfo). Maybe I'm barking up the wrong tree here because I really just want to focus a space - but I can't seem to find any way to switch the current space to another one. Ideally I'd focus a space with a given window in it, hence why I'm taking this route with _SLPSSetFrontProcessWithOptions. |
If you already have the PID, use |
Holy cow that works! However, this particular method seems to be setting the front process but not actually moving to that particular space. I see the app take over the menubar but the space doesn't navigate to open it. Is there a different method for navigating spaces that I'm missing? #[test]
pub fn focus_window() {
let wid: u32 = 55440;
let pid = 7020;
let mut psn = ProcessSerialNumber { padding: 0, val: 0 };
unsafe { GetProcessForPID(pid, &mut psn) };
let mut bytes1 = [0; 0xf8];
bytes1[0x04] = 0xF8;
bytes1[0x08] = 0x01;
bytes1[0x3a] = 0x10;
let mut bytes2 = [0; 0xf8];
bytes2[0x04] = 0xF8;
bytes2[0x08] = 0x02;
bytes2[0x3a] = 0x10;
bytes1[0x3c..0x3c + 4].copy_from_slice(&wid.to_le_bytes());
bytes1[0x20..(0x20 + 0x10)].fill(0xFF);
bytes2[0x3c..0x3c + 4].copy_from_slice(&wid.to_le_bytes());
bytes2[0x20..(0x20 + 0x10)].fill(0xFF);
unsafe {
_SLPSSetFrontProcessWithOptions(&mut psn, wid, 0x400);
let e1 = SLPSPostEventRecordTo(&mut psn, &bytes1);
let e2 = SLPSPostEventRecordTo(&mut psn, &bytes2);
// TODO: need to get the AXUIElementRef for the window we want to focus
// AXUIElementPerformAction(e1, kAXRaiseAction);
}
} Edit: I came across the need to get the Do you know of any api to get the AXUIElementRef given a window id? |
There is no API for this. However one possible solution would be to retrieve a handle to the application and iterating over its windows AXUIElementRef, request the window id from the AXUIElementRef and see if it matches your target. Something like:
|
Thanks for the help. Is it intended that this method only returns a list of windows currently visible on screen?
I'm trying to get the AXUIElementRef for off-screen windows - only having them being on screen is almost useless. |
Yes, this is an API limitation. You can retrieve the AXUIElementRefs and cache them for later, and you can perform actions on them regardless of which space is active afterwards, but the actual act of retrieving these references must be done in the space that the window is currently in. You need to get creative to work around this issue -- if it is even possible in the latest version of macOS. I'm not sure what exactly you are trying to build, but if it is anything resembling a remotely sophisticated window / spaces tool, prepare to have to spend a lot of time as you wrangle with weird macOS quirks. |
I'm trying to build an updated TotalSpaces app. I have their TotalSpaces3 beta and am trying to reverse engineer their reverse engineering. The primary function I'm trying to attain is programmatically switching to a space (ideally without sending keyboard shortcuts as this will conflict with the keyboard shortcut used to trigger the change). TS3 has this figured out, but looking through the various apps (hammerspoon, ts3, yabai, alt-tab), it seems like this AXUIElementRef dance is the way everyone goes to switch spaces. I don't quite get how TS3 figured it out to make it work without visiting each space once (something alt-tab requires and seems to be somewhat broken around). I've gotten to the part where I get a list of all spaces / their active windows, I just need to somehow switch to the space.
I've already spent many hours navigating this, so I've got that "sunk cost" thing going on... |
I am not sure what TotalSpaces3 is doing if they are able to focus arbitrary windows across spaces without first doing the "discovery" phase. Yabai is capable of doing what you want using only the window id, but that solution requires disabling SIP and injecting code into Dock.app. A workaround I have observed in the wild is that your application can create hidden windows (one on every space), then focus your own window to trigger the space switch, and then return focus to the window that macOS claims should be focused on that space, but this is sort of janky as it messes with focus history and what not. |
Yep - that's the solution. I use contexts (another cmd-tab alternative) and they fill my window search tool with hidden windows.
I don't think totalspaces does that (maybe they throw their window from space to space) but that's exactly what contexts is doing (with multiple windows per space too). I think the code to do that is roughly contained here:
Does this sound about right? Thank you for all the help so far - you're a wealth of information :) Edit: I got the invisible window trick + switching to app working! Just need to somehow throw windows onto all the available spaces. |
Time to fire up Hopper? |
I've never used TotalSpaces2, but their install docs seem to say that you have to disable SIP because they are also injecting code into Dock.app. |
Following a tip in this Hammerspoon issue thread: Hammerspoon/hammerspoon#370 Activate the app a window belongs to, wait a small amount, then try to focus the window on a new Space. Do this before switching to the new Space so a window is focused and PaperWM.spoon is ready to navigate via keyboard shortcuts after switching.
If I have a Google Chrome window on each screen, and I try to
:focus()
the window that's on the screen other than the one currently focused, it will instead focus the Google Chrome window on the current screen.Reproduceable on master using the console on OSX 10.10.1.
I know this didn't used to happen, and haven't run into it until today (hadn't updated in awhile), so I'm curious if anyone else can reproduce it.
The text was updated successfully, but these errors were encountered: