-
-
Notifications
You must be signed in to change notification settings - Fork 650
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
Handle native tabs properly #68
Comments
I don't think the implementation in Amethyst is complete, or fully robust. How does Amethyst handle minimization and deminization of such grouped windows? What about hiding and restoring an application that has a grouped window, and a "normal" window? I'm also unsure if the same heuristic applied by Amethyst can be used here, because I believe that Amethyst behaves differently in terms of how windows are stored / tracked. IIRC they have some caching mechanism that is flushed and regenerated upon a space activation? I may be mistaken as the last time I looked at Amethyst was maybe 4years ago. |
So I just tried Amethyst to see how it behaves with tabbed windows.
Hiding an application did not work for me at all, regardless of the application having tabbed windows or not, so I could not verify my assumption there. I did take a quick glance at the code that is run upon a space change, and it does indeed drop and regenerate some sort of cache of windows that it consider to be current. This is not something that can be replicated in yabai, as we don't have this form of caching. In yabai, a window is tracked from the moment we learn about its existence until it is destroyed, regardless of which space or display or whatever we transition from/to, or consider active. I'll probably spend some time to observe the behaviour in detail myself, and if I can come up with some solution that can integrate properly with yabai, I will support this. If I cannot do so, I'd rather state that tab integration is not supported. Implementing a partially working, but flawed solution, would be worse than ignoring it completely in my opinion. |
There is an additional case here that Amethyst also does not seem to tackle at this moment in time. When a window is de-tabbed we somehow need to detect this and tile the window. |
I spent a bit of time looking into this, and we can actually query information about the tabs using the accessibility API. That seems a lot better to me than having to try and differentiate based on which order we receive notifications in. |
Tested some more. The accessibility API is not reliable. All it really gives us is the fact that a tabgroup has been created, and the number of tabs. However, if you go from a non-tabbed window to a window with 3 or more tabs (spamming new tab hotkey). The creation of all but the last tab will not report as part of a tabgroup, as the tabgroup have yet to be created. Looks like we may have to do some weird combination, or rely on notifications only, which I'm not sure will be good enough. |
When a window is de-tabbed we only receive moved and resized notifications for said window. |
So when a tab bar is present in an When a window is de-tabbed, this count changes. To associate this with the corresponding window id, there'd need to be a mapping of tab button UI element identifier (does such a thing exist?) to tab window id. |
The problem here is that the detabbed window no longer has that tab-group as an accessibility element, so we are not able to know which group it came from. Tracking windows that belong to a group is not possible, at least I have not managed to do it reliably:
|
Couldn't you created a cache that maps window id -> tab group? Now when a window is moved/resized, you check if the windows id belonged to a different tab group than it does now. |
I have not managed to do this no, as no API return the id of windows that are the inactive windows in a tab group. When trying to get the elements in a tab group, they all return the window id of the active window as well. I also tried to cache the AXUIElementRef itself, and use the appropriate function to compare them, but that also did not work. This starts to become a problem when multiple tabs are opened in a short amount of time, as per my comment above. |
The only way to access the tabs programmatically seems to be by using It really seems like a position based workaround is the only option: By default, macOS will never spawn a new window with the same coordinates as an existing window of the same application, except for when it's a tab. |
Please handle this somehow, many people use tabs and this is a big inconvenience. How does yabai restore proper layout for tabs when switching windows but not when the tab is created in the first place? There's gotta be some way to fix it. |
@koekeishiya @dominiklohmann Sorry for the noise but is there any improvement in the area? |
I agree that this is a big inconvenience, but as it is yabai would need to be restructured internally for some kind of stacking support (#203, so a tree node can hold multiple windows), and with that in place this change would be easier to make, although still based on workarounds (frame x/y stay the same, w/h may change slightly for some apps like Terminal). Some kind of caching may also need to be introduced, because macOS does not include window ids from background tab windows in query results. It's just a tough issue to solve correctly and likely to cause the need for further workarounds in the future. ¯\_(ツ)_/¯ In general I think this should still be approached, especially with 10.15 bringing iPadOS apps to macOS with Catalyst, where all document-based apps are gonna have this problem. |
@dominiklohmann Thanks for your kindly explanation! Sent with GitHawk |
@dominiklohmann One thing I've seen: open a random app and finder. Add a new tab in finder, now yabai thinks there are three window and splits finder. Close one tab and it restores to normal. Now try again, add a new tab, but instead of closing, switch to another desktop and switch back. Magically, Yabai recognizes that there are only two windows, despite multiple tabs being open. It seems like Yabai just dropped all but the last tab from the tree. If you close the last tab, finder becomes a floating window because the first tab is not part of the tree. I'm sure it's just accessibility API fucking up, but how does the tree just drop a node out of nowhere? P.s. can you point me to the part where window tree is being managed? Thanks. |
Every single API that macOS exposes (both public and private) stop reporting the window id of windows that are not the active tab, which is why yabai currently assumes that the window no longer exists, and releases said region.
Mostly from functions called from |
I'm sure most people will probably be displeased with this answer, but I don't consider this issue worth more of my time. I have spent probably more than 40+ hours messing around experimenting and looking for robust ways to fully support this throughout the features that yabai make available, and my conclusion is that it is simply not possible (at the level of quality I'd expect it to have). I also never use this feature, and macOS is not my primary OS anymore. I'll happily remain a part of the discussion If other people are interested in trying to tackle this problem themselves. |
I am curious that which OS is now your primary OS? MacOS's WM is a bit frustrating |
I think the best course of action would be to expose necessary functionality through rules and then let people write these per application. How to implement this would still be a big problem, maybe just adding a |
@leonardopennino I found a similar workaround, which doesn't rearange the windows. The problem with setting the layout to float and back to bsp is that it seams to always put the active window in the left top corner. But if you go to another space and come back, the layout is recalucated, as already mentioned above. What we need is a way to refresh the layout. My temportary "solution" is this. It works pretty ok but of course there is a flicker, as one needs to focus on anoter space and back on the previous one to work. That's also the reason fo the delay. It's not pretty but it get's the job done: # Refresh yabai if a tab in finder or terminal is created or moved to new window
yabai -m signal --add event=window_created app="^Terminal$|^Finder$" \
action="yabai -m space --focus next && sleep 0.01 && yabai -m space --focus recent"
yabai -m signal --add event=window_destroyed app="^Terminal$|^Finder$" \
action="yabai -m space --focus next && sleep 0.01 && yabai -m space --focus recent"
yabai -m signal --add event=window_moved app="^Terminal$|^Finder$" \
action="yabai -m space --focus next && sleep 0.01 && yabai -m space --focus recent"
yabai -m signal --add event=window_resized app="^Terminal$|^Finder$" \
action="yabai -m space --focus next && sleep 0.01 && yabai -m space --focus recent" In Amethyst there is an option "Force windows to be reevaluated". Would be nice if one has a similar option here as well. Like @koekeishiya Do you think it is simple to implement such a comand? You already gave a startingpoint where this is handeled.
|
I found a workaround which works for me by accident. I wanted to have unresizable windows like some settings dialogs to float. Found this issue: #1317 The last comment with the signal fixed the Finder tab behavior as a side effect. EDIT: Just realized the signal lets the Finder window itself float. |
This helped me with PhpStorm tabs. Thanks! |
Unfortunately, |
You can achieve the same thing using
This works on my setup without SIP disabled. |
You probably need a second display for this workaround, right? I only have a single display.
|
The NSDocument tab issue is a pretty annoying issue and is easily surfaced with native / default apps like Finder and Terminal, which are built into the OS. To avoid users creating multiple issues to track the same issue, this lists the caveat on the caveats section and links to the associated issue. See: koekeishiya#68
Looks like Amethyst somehow works around this issue since windows with native tabs stay balanced most of the time. In worst case scenario if I spam tabs too much it swaps places with other window. Other than that it works. |
is it possible to ignore the tabs for particular program like alacritty? |
@aayushsapkota9 correct, basically. |
current workaround for me is after i have created a new tab, switch to another space and back, that seems to fix it |
I just started using |
is it possible to fix now |
I am playing around with this workaround for now while using Ghostty:
Isn't perfected but works most of the time Update 07/10/2024:
moving focus to next space and back to previous seems to work better - since it doesn't reset sizing of windows |
With Ghostty finally open release, I've just stumbled over this issue (don't seem to have it with Finder though?) Thanks for the workaround, but yeah... that flicker... Guess I'll see if Ghostty sticks first, but if so, looks like it's not ever going to work nicely with Yabai, so perhaps time to look over Amethyst or Aerospace :( |
AeroSpace has the same issue: nikitabobko/AeroSpace#68 |
Issue
NSDocument-based applications send window created notification on tab creation.
This is a re-iteration of the same issue with chunkwm (https://github.com/koekeishiya/chunkwm/issues/235).
I have created another radar for this issue now that the Catalina beta is a thing. This time around, I got an answer: Works as expected.
iPad apps brought to the Mac using Catalyst have the same behaviour, so this should probably be fixed before the release of macOS 10.15 Catalina.
Proposed fix
Investigate how Amethyst works around this and replicate the behaviour in yabai.
Here's what they're doing from my understanding: When a
kAXWindowCreated
notification triggers, check whether the window has the same position and same size as another window of the same application. For applications like Terminal, some tolerance is needed because the window size may change ever so slightly when changing tabs.The text was updated successfully, but these errors were encountered: