-
Notifications
You must be signed in to change notification settings - Fork 4
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
Staying in normal mode after loosing focus #155
Comments
thank you! took a while to reach a proper stage. but it was fun using kV while building kV in Xcode 😅️
yeah if i remember well Xcode announced Vim some months before i released kV 😅️ but they had less motions than kV. was hoping they would keep pushing but seems they're dropping it or what?
yes. i think there's an easier way to describe this: every time the current UI Element loses focus, kV goes back to Insert Mode. see below for more information about why.
the problem is there's no macOS API to track a UI Element. you can track apps, you could track windows, but you can't track UI Elements. so it's not possible for kV to know when you are back to a previous element.
yes, so there's some issues with that too. the first one is that you can't detect through the AX API that you will leave a UI Element. you can be notified (like with the API that you're pointing below) ONCE the UI Element changed. again, because you can't track what was the previous one, you can't go back and discard the selection. and same, because you're notified ONLY after you've already left the UI Element, you can't make the change before leaving. a second reason is more deliberate. many users want to run programs (like Alfred's Universal Action) on selected text. this is where the passthrough come from. that allows the users to get back to Insert Mode/macOS and run stuff on the selected text. if you'd remove the block cursor/selection, then you'll not be able to apply anything on your kV Visual Mode selection.
i'm actually using those APIs and something related in order to deliberately switch back to Insert Mode 😂️ like for example when you click somewhere. there's a possibility to do stuff, like when you click to a new UI Element (Text only?), then you restart the kV Engine and you're in Normal Mode directly in the new UI Element. (although you would lose all data between UI Elements. to do its job kV needs to grab A LOT of data from a UI Element, most of which are not transferrable.) but would you always want that? i remember playing with this at the beginning, but i've found the experience completely inconsistent. what worked best, with my experience, is to go back to Insert Mode as soon as you leave the current UI Element. maybe i just got used to it? personally it feels very natural to me. (almost maybe i have my escape key mapped to caps lock, and the whole thing is just smooth and natural.)
my pleasure. feel free to share more, ping me, send questions, argue. i'm here everyday. and it helps a lot, making me think, review mindsets and ideas, and ultimately can make kV better. so thank you! |
Seems like they dropped a barebone implementation in Xcode 13 and then forgot about it. No
That sucks. For the sake of argument (as you seem open to it!), would it be possible to use a heuristic to identify a previous UI element? For example the combination of:
This wouldn't work for a lot of UI elements, but for large text fields (such as Xcode's editor) I guess it could be helpful?
I see. If the above heuristic is not possible to identify a previous UI element, how about having a special case when focusing an element having a single-character selection (the one emulating the Normal mode caret), to either trigger the Normal mode, or discard the single-character selection?
I barely scratched the surface of kindaVim, which I'm using mostly with Xcode at the moment. I guess when you use it to navigate through complex user interfaces then it makes sense to revert to Insert (which becomes really the default/normal mode in UI). Maybe choosing the default mode per app could be helpful? Or to have the default mode being Normal only for text fields? |
AppCode is dead??? 😱️ hmm then i don't get how come kV is not bringing me millions every
definitely. haven't reach the bottom of it yet.
i've discovered after, working on an Alfred Workflow to bring windows to the foreground for Wooshy that there is a private API to grab the exact window ID with the AX APIs: _AXUIElementGetWindow i guess there could be some heuristic yeah. maybe even window ID and UI Element position + size + role. could create a hash from those data, may work quite well. although if window is moves or UI Element resized (Text Area), that would coz issues. i think i thought about the value also, but it may be modified by other programs, so that would be an issue too. jfi, i'm definitely not a genius lol, and i'm not the best programmer by far, but i'm very clean, in the sense that 1) all that can be tested is tested (6k+ automated tests on kV) 2) every decision is not taken lightly, so there's (almost guaranteed) a reason why kV ends up the way it is. but still, we haven't reached the bottom of this yet i feel, so let's keep going.
but again, would you always want that? you click on an Input Field, if there's only one character selected, you always want kV to enter Normal Mode? you tab through the Input, same? some other apps like Wooshy sends you there, same? you close a window and macOS reverts to focus on another Input, same? i'm still not comfortable with the possible inconsistency.
yeah personally i use it in Xcode, GitHub comments, iA Writer, etc. but also A LOT in UI. like choosing Alfred's results, going through menus etc. but we could separate Text from Non Text Elements (although not sure it's easy with the current structure of the Engines).
wouldn't work on a per app basis. an app will have different UI Elements. would need to be automatic, from the Element role. i think.
so, is the summary like: go directly to Normal Mode when you enter a UI Element that is a Text one? 😅️😂️ |
Sadly yeah 😞. You might get a bunch of new subscribers once Apple releases a new Xcode version breaking the compatibility with AppCode!
IMHO that's worth testing with this hashing method. Reverting to Insert mode because the window was resized or the text field value changed separately from kindaVim sounds like a fine default to me as we refocus an element that is effectively in a different state than where it was left with kindaVim. I think that would be the cleanest and more consistent way to handle this, compared to the single-character selection hack.
Well that's how Vim works and I think I would actually prefer that (maybe an opt-in option?). More power in my text fields 😉 But maybe I'll get used to the way kindaVim works. I'm going back and forth between Xcode, Android Studio (with IDEAVim) and Neovim so right now it's a brainfuck every time I switch. |
seems it's gonna hold until Dec 2023. plenty of time for me to die first
alright then. there's probably gonna be some issues when kV can't detect whether a UI Element is Text or Non Text. that does happen, and some apps even need to be put in the
true. although yeah, Vim owns its own thing (buffer and all). kV doesn't, which is where it gets hard. especially dealing with when kV is in Insert Mode—normal macOS—because there's many things that can happen and are out of kV's control.
well i'll research on my side, but it'd be very useful if you keep me updated on your usage yeah. curious to know if this is a deal breaker or not. no one mentioned about that before. and where adding motions is pretty straightforward (except for Le Dot and handling Insert Mode), making modifications on the Engines that deal with the whole macOS system is rather tricky. (usually quite a lot of side-effects, and mostly hard to get those things automatedly tested.) |
Thank you for considering it, I'm looking forward to it! I'll keep you posted how it goes using kindaVim. I guess I need to get used to hitting Caps-Lock every time I focus a text field. |
I might have stumbled onto something useful. I was doing some tests and noticed that the Screen.Recording.2023-01-08.at.20.32.13.mov |
oh noice. thank you! do you want to have access to kV's source? 😂️ but actually does it matter now that the Focused Element is the same, if we enter automatically into Normal Mode once you reach an Element that is of Text role? tracking which Element is which would be useful if kV was tracking all the data needed to do its work EVEN when leaving an Element. which is definitely not the case. as soon as you leave the whole world is reset. |
actually the |
I guess not, but if you want to keep close to Vim's behavior the mode should be persisted per element. That would make it more cohesive with other types of elements too. But I think that's not a deal breaker at all.
I actually asked for help to ChatGPT. 😁 Most of it was garbage but the
I would love to, but full disclosure: I'm working on something like SketchyVim but written in Swift (most likely open source eventually). Since kindaVim is proprietary, I understand if you're uncomfortable sharing the secret sauce. I'm all for discussing techniques and ideas though. Eventually I'd like to use real Vim in Xcode and kindaVim for other UIs. |
but the analogy would be like running lots of Vim all over macOS 😅️ i mean, in terms of kV, is that really useful to keep track of each Element's world? that'd be way more CPU and RAM intensive. not sure it's worth it at the end? how often/critical is this needed?
wow LOL
i was making a joke 😂️ but why not. there's already some people on it. also kV was open-source for a while, but had the License for people who didn't want to bother. closed it because people couldn't behave. anyways ultimately i'm not sure you'll be able to "understand" the code. not that i'm judging your intelligence 😅️, not at all, but it's spread over multiple packages, the code is very personal, and mostly a reaction to my understanding of Vim and of the macOS APIs, and the complexity of trying to recreate Vim with those APIs. lots of the code ends up being a solution of a problem that i couldn't foresee, because i lacked knowledge or on Vim, or on the macOS APIs, or of a bug in those. but if you're very familiar with Vim and with the macOS APIs—especially the AX ones—then you may actually understand quite a big chunk, who knows. lemme digest the idea.
but mostly the approach is gonna be VERY different, no? kV started as SketchyVim, with a Vim backend. but i've found it was way more painful to try to make the bridging work, even more when it comes to moves that deal with Screen Lines. when it's text, it's fine (more or less, as long as the text is within the visible area). when it's about the screen (moves like |
@mickael-menu ok i've invited you to a couple of repos. probably the ones where you may find the most info regarding the AX, etc. see if you find something useful. at the end hopefully the people who will end up using your app wouldn't be kV's customers anyways. (thoughts and prayers.) if you find stuff that could be done better or have some other ideas you can share for sure but big chance i'll not understand and will have to stick with my own code sorry 😂️ |
Yeah probably not worth it in the context of kindaVim where we switch between a lot of elements.
I feel you, I have a couple open source projects and it can be overwhelming at times 😄 Thank you for sharing!
Oh yeah, I might loose the few remaining hair I have with this! I really miss full Vim power with Xcode ( |
SketchyVim doesn't have this? with the Vim backend it should work no? only issue might be with the visible area. i think SV works super well but for small bouts of text. as soon as you're out of the visible area it's a little more tricky. regarding your original question, i'm having more thoughts. for example, i personally use the passthroughs a lot. like i'm in Xcode in Normal Mode, then i call Alfred with or adding a new Setting where you could drop apps where you want kV to enter Normal Mode automatically? |
SketchyVim updates the whole xcode.mov
You're right, I can see how that could be an issue... I'm not sure we can be consistent without saving the mode for the element. How about:
|
ah yes, true. hence SV being very good for small texts, not big fat Text Areas.
yes. you can see in the AXEngine repo, by the end. this is how the AXUIElement's value is updated.
thinking out loud here. so yeah, i guess we're back to tracking Elements. and yep you're right, to be able to clean properly we're gonna have to track apps/windows open. another issue is that it's not just about entering Normal Mode again. actually just doing this would already be an issue as entering Normal Mode brings back the block cursor one character to the left, if not at the beginning of a sentence. so this would have to be adjusted and calculated. but you most probably want to save Visual Mode info too. like if you're selecting something, then run Alfred through the passthrough, when you're back in your Element, you will probably want to be back in Visual Mode, with the same selection. may want to keep lots of other info like count, search patterns, etc.? currently there's only one of those per kindaVim Engine, which means, well, one. but that would need to be tracked PER ELEMENT. big changes. also, again, i need to make sure this is worth it, because it's a lot of changes actually. if you're navigating between a lot of different Elements, are you actually keeping track in your head of which one is in which state? like when i'm back to Xcode i'll be in Visual Mode with that specific count, but in Safari GitHub comment i'm in Normal Mode, and in iA Writer i'm also in Normal Mode with a |
i swear i'm not trying to ask questions until you give up 😂️ |
just out of the blue, a case where detecting the Focused Element is not gonna work: ScreenFlow.mp4the Focused Element never changes here. |
To be honest you might want to postpone until I figure things out with my version for Xcode text areas. Then maybe I will be fine with the way kindaVim works in other elements.
And maybe that would be fine for me too, if the single-character selection was discarded when focusing the element. This block caret signals to my brain from years of using Vim that we are in Normal mode. Is it something that could be done now that we have a way to identify the
I think that's correct here? After hitting Cmd-0 you can still type in the editor area so it still has focus. |
Also when we have a passthrough keyboard shortcut (e.g. Cmd-S) cancelling the Normal mode. |
:D
but i think we're back to the issues i've mentioned above. as far as i know, you can detected when the Focused Element changed, but you don't have access to it then. so you can't make changes. also i don't think you can grab an AXUIElement by their hash. although we could get it through their position: https://developer.apple.com/documentation/applicationservices/1462077-axuielementcopyelementatposition#
yeah that's correct. my point is kV enter Insert Mode, because of the passthrough. but Focused Element didn't change. so that would be a case that would have to be taken into consideration too. |
i think again two issues here. not sure we still have access to the Element, but more importantly that breaks the reason why people wanted the passthrough in the first time: act on the selected text. although, maybe an idea is to retain the selection only if kV was in Visual Mode? but again to me, that's a lot of things i would have to remember in my head. i don't want this. i think in your case you're having a strong Vim habit and you may expect kV to work the same. but kV is mixed with macOS and a lot of things are not controllable, which makes it very hard to stay/be consistent. the lack of consistency would then have to be balanced by remembering deliberately in which states are each Focused Element, etc. rather than just remembering than once you leave an Element, you're out of Normal Mode. |
I meant that you could discard the block caret when focusing the element again, not when it looses focus. You check that last time it was focused the mode was Normal (not Insert or Visual), and in this case only you discard the block caret selection.
We can think of it this way: you simulate the Normal mode in an This doesn't apply to the Visual mode for the reason you mentioned (acting on the selected text). |
😅️😅️😅️ ha yeah, makes way more sense.
ok, i get what you mean, but you still need some cognitive work to REMEMBER IN YOUR BRAIN which Element was in Normal Mode and which wasn't. like we would discard the block cursor if kV was in Normal Mode, but not if you selected manually a character. so in a case you go back to an Element and kV removes the block cursor, but in another no. what's the logic? the logic is that you have to remember that you were in Normal Mode in on Element, but not in the other. probably after having played around macOS. maybe it's just me but i still find this tiring. also, why not removing the selection in Visual Mode too, then? the passthrough allows for sending the selection to other programs, but once you're back to the Element, we could discard the whole selection like in Normal Mode. would be a least a little bit more consistent? else you even now need to remember WHICH ELEMENT was in WHICH KV MODE. i can't believe that one will keep track of all this around using macOS.
well, as far as i'm concerned, not really. because you're dealing with macOS, and not with your own Vim self contained app and buffer. you can select one character completely out of kV's system. that's where it's gonna be confusing/have some cognitive load, i think. but again, maybe it's just me. and i'm someone who doesn't use a lot of apps at the same time, but i would already feel the drain. having everything behaving the same (when you're out of an Element, you're out of Normal Mode) is easier for me. may not be perfect, but that's the issue of dealing with a part that is not totally controllable: macOS (and other apps, and Electron apps, and...).
from my comment above, i think it should apply for Visual Mode. anyways, let's digest this. do as you mentioned above, maybe see within your own app. on my side i'll make some tests and see how it feels overall. thanks. |
You know that you lost the focus of the element in Normal mode, so any manual selection would have to be done outside of kindaVim. That seems a bit far fetched for a single character selection, and you could also store the selectedRange to make sure you really discard the last one that was used in Normal mode.
Discarding a "real" selection would be too destructive IMHO, e.g. you select something, then you switch to another window to check something, coming back you loose the selection. Instead when coming back the element could be in Insert mode but keeping the selection (as an Insert mode macOS selection, not a Visual one). That's not confusing because you can see that something is selected (more than one character), and you know that when focusing an element, you're immediately in Insert mode.
Agreed, I think we explored the subject enough. Good luck with your tests! |
still feels wrong to me. if you were in Normal Mode, leave, and come back, the selection is gone and you're in Insert Mode. if you were in Visual Mode, leave, and come back, the selection is still there and you're in Insert Mode. going back to Normal Mode will remove the selection. i guess yeah, i will have to hack something in order to have a real live playground because the whole thing still feels absolutely convoluted to me.
thanks! will keep you posted. |
Hi @godbout, I just released my work integrating Neovim with Xcode: https://github.com/mickael-menu/ShadowVim As you are well aware, AX APIs are very messy across apps, so this integration focuses only on Xcode. I used kindaVim until ShadowVim was good enough to be used in Xcode, and I still struggled with the Normal mode being lost. At this point I think I'm comfortable with a more integrated approach like ShadowVim for coding and something a bit more transient like kindaVim for other apps. Thank you for entertaining this discussion! I'm closing this issue but feel free to reopen if you think this is still something you'd like to change in kindaVim. |
Oh btw, if you want to check out the AX relevant code, it's there: https://github.com/mickael-menu/ShadowVim/tree/main/Sources/AX A part that might interest you or not is the Combine publisher adapter for AX notifications. |
wow this is GREAT! congratulations. i guess then that ShadowVim is the way to go for anyone who wants Vim in Xcode. i wouldn't think anyone would want anything else. i'll reopen the issue tho, coz i haven't explored yet enough, and dropped the idea. thanks for the update! |
First of all, thank you for this outstanding piece of software. This is the most promising system-wide Vim implementation I've tried so far. I'm using it mostly to replace Xcode 13's subpar Vim implementation now that they killed XVim.
There's one issue that keep tripping me up though: the Normal mode is lost regularly and I don't realise it until after I typed a few characters. For example:
Here's a live action movie of the issue:
lost-normal.mov
Ideally, in all these scenarii I would like kindaVim to stay (or at least restore) the Normal mode. But if at least the current selection (selected character simulating the block caret of Vim normal mode) was discarded when switching to Insert mode implicitly, I think the visual feedback would help.
Not sure if that would be useful, but the
kAXFocusedUIElementChangedNotification
andAXObserverAddNotification
seem relevant for the issue when switching to a different window / UI control.Thank you!
The text was updated successfully, but these errors were encountered: