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

Mouse AnyEvent tracking (1003) -- work in progress #3538

Merged
merged 185 commits into from
Jan 14, 2025

Conversation

AutumnMeowMeow
Copy link
Contributor

@AutumnMeowMeow AutumnMeowMeow commented Aug 3, 2024

PR for more detailed discussion, for #1679

Current bit compiles, passes tests, fails clippy (not my fault, other parts of zellij-utils doing that).

Things to do:

  • Fix broken mouse Release events
  • Decide how Motion events will be exposed to zellij code, i.e. which Action:: items will it have?
  • Implement Motion events
  • Fix broken mouse text selection / copy-paste
  • Add CSI ? 1003 support inside the terminal pane
  • Test Normal tracking in vttest
  • Test ButtonEvent tracking in vttest
  • Test AnyEvent tracking in vttest
  • Fixup other code style issues (I still suck at Rust)

@AutumnMeowMeow
Copy link
Contributor Author

AutumnMeowMeow commented Aug 3, 2024

@imsnif

You can see my TODOs on top. I will try to minimize commits to this PR so it doesn't constantly spin up the tests.

Initial questions:

  • Does it make sense what I have here so far? Moving away from a "single button down" --> "held" --> "release" model, instead to "button(s) down" --> "button(s) maybe released or dragged, no buttons, etc." I would like to carry this simpler model down to the terminal/grid layer, replacing the six {left/right/middle}_button_{signal/release_signal} functions with three functions {press/release/motion}. On the plugins layer all the existing Actions could stay as-is so as not to break existing code.
  • What Action's would you like to see for Motion events? (My vote would be a Action::MouseMotion(point, buttons).)

@imsnif
Copy link
Member

imsnif commented Aug 5, 2024

Hey @AutumnMeowMeow !

The direction looks very good. We do not enforce clippy anywhere, so all good.

* Does it make sense what I have here so far?  Moving away from a "single button down" --> "held" --> "release" model, instead to "button(s) down" --> "button(s) maybe released or dragged, no buttons, etc."  I would like to carry this simpler model down to the terminal/grid layer, replacing the six {left/right/middle}_button_{signal/release_signal} functions with three functions {press/release/motion}.  On the plugins layer all the existing Actions could stay as-is so as not to break existing code.

Sounds perfect. From my side, as long as the plugin API (and especially the protobuf layer) doesn't change we're not breaking anything, so should be good. It sounds like this structure is much more in-tune with the way these events are encoded as seen in your changes, so let's go with it.

* What Action's would you like to see for Motion events?  (My vote would be a Action::MouseMotion(point, buttons).)

Sounds good to me! Note that when you get to serializing these actions to protobuffs, you can just ignore them as unsupported actions here: https://github.com/zellij-org/zellij/blob/main/zellij-utils/src/plugin_api/action.rs#L1295

I'm excited about this feature! Already thinking of how to use it in the UI.

// unfortunate side effect of the pre-SGR-encoded X10 mouse
// protocol design in which release events don't carry
// information about WHICH button(s) were released, so we have
// to maintain a wee bit of state in between events.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Honestly, even this is much better than what we've been doing up until now with the HeldMouse stuff.

side terminal(s) in, including protobuf.

Removed "held" mouse actions.

Currently commented out calls to left/middle/right-click/release --
need to fix this though, as selection/copy-paste are broken too.

cargo build/test/run works OK.

cargo xtask build/test/run fails, unable to find crate input::mouse.
@AutumnMeowMeow
Copy link
Contributor Author

@imsnif OK most of the core new function is here. :) Some things are broken though, and I could use your pointers. (Also converted this to draft in the futile hope it would not spin up the whole CI stuff on every commit...sigh.)

  • cargo build/run/test work fine. However cargo xtask build/run/test do not. They cannot see the input::mouse crate, no idea why. Any ideas on how to fix?
  • I temporarily commented out the other left/right/middle-click functions, which now breaks text selection of course. But along the way I also removed the MouseHold stuff, and see a few decently large functions that got axed in zellij-server-src/tab/mod.rs. Do we actually need those? Or would those functions still work via only LeftClick / LeftRelease / etc.?
  • Somewhere along the way the mouse position is getting a couple off-by-ones, i.e. not 0-based. I suspect that is an artifact of the find_active_tab stuff I cargo culted from the left_click code, e.g. failing to properly handle the conversion from absolute coordinates returned by termwiz to the relative-pane coordinates needed before sending to the terminal.

Thank you for your help! :-)

self.mouse_buttons_value_sgr(event),
// AZL: Why is event.position not staying 0-based on
// both axes?
event.position.column(),
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see that I missed the relative coordinate conversion in handle_mouse_event(). I'll fix that.

}
Ok(false) // we shouldn't even get here, but might as well not needlessly render if we do
}

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The three functions above are decently large. Is there logic anywhere else to handle changing the selection buffer size as the mouse is dragged around? If not, then we will need to put that in handle_mouse_event().

pub shift: bool,
pub alt: bool,
pub ctrl: bool,

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also FYI that Swing and WIN32 both expose keyboard modifiers with mouse events. Soooo when zellij is running natively on Windows we'll have the infrastructure to do things like Ctrl-Click / Alt-Drag / etc. :-)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Honestly, even Alt + mouse_click would be amazing. I'm thinking of using it for multiple selection of panes.

@imsnif
Copy link
Member

imsnif commented Aug 8, 2024

Hey @AutumnMeowMeow - sorry for the delay in getting to this - I'd like to give this a priority but I had a bit of a hectic day yesterday.

cargo build/run/test work fine. However cargo xtask build/run/test do not. They cannot see the input::mouse crate, no idea why. Any ideas on how to fix?

Brief explanation: this is happening because the Action struct is shared across the webassembly boundary to plugins. Termwiz has some dependency that doesn't play well with wasm (right now I don't fully remember which one) and since the mouse module depends on termwiz, we stub it out here: https://github.com/zellij-org/zellij/blob/main/zellij-utils/src/input/mod.rs#L12

Suggested solution: I think the cleanest way to get around this is to remove the mouse module's dependency on termwiz. I think we can do that if we make the termwiz_mouse_convert and from_termwiz functions independent of the mouse module. We could for example (just from a brief look, maybe you have an idea that makes more sense!) change their signatures to:

fn termwiz_mouse_convert(original_event: &mut MouseEvent, event: &termwiz::input::MouseEvent)
fn from_termwiz(old_event: &mut MouseEvent, event: termwiz::input::MouseEvent) -> MouseEvent

And then place them in the input_handler.rs file where they are used as bare functions. What do you think?

I temporarily commented out the other left/right/middle-click functions, which now breaks text selection of course. But along the way I also removed the MouseHold stuff, and see a few decently large functions that got axed in zellij-server-src/tab/mod.rs. Do we actually need those? Or would those functions still work via only LeftClick / LeftRelease / etc.?

I think we should be able to use these functions as is (maybe even removing the whole is_repeated and last_position logic) with the new motions. Here we control the movement of floating panes and text selection (basically, they detect whether we're holding the mouse on a the frame of a floating pane or inside a terminal pane). Does that answer your question?

Somewhere along the way the mouse position is getting a couple off-by-ones, i.e. not 0-based. I suspect that is an artifact of the find_active_tab stuff I cargo culted from the left_click code, e.g. failing to properly handle the conversion from absolute coordinates returned by termwiz to the relative-pane coordinates needed before sending to the terminal.

I think you found this in your other comment, right?

@AutumnMeowMeow
Copy link
Contributor Author

AutumnMeowMeow commented Aug 29, 2024

Refactoring termwiz_mouse_convert and from_termwiz got the builds all working (oops, test is broken), thank you @imsnif ! :-)

A few more glitchy things fixed, and now viola, actual proof of concept:

simplescreenrecorder-2024-08-29_15.46.26.mp4

@imsnif
Copy link
Member

imsnif commented Aug 30, 2024

Very cool @AutumnMeowMeow !! Just a heads up: I'll be on vacation the next two weeks. Will be sure to answer any questions that come up when I get back.

@imsnif
Copy link
Member

imsnif commented Dec 2, 2024

Hey @AutumnMeowMeow - I hope you are well and don't mind the mention. I want to get this merged and released in the next version, as it will complement some of the planned new features very nicely. Is this something you'd like to keep working on? (no pressure - there is no release date for now)

@darrenburns
Copy link
Contributor

Very cool stuff here! Looking forward to this 🙂

* fix(plugins): various cwd fixes

* fix tests
* feat(plugins): rerun_command_pane API

* fix tests
* separate saved/runtime structure, kind of working

* serializing config

* work

* work

* save config through the configuration screen

* work

* startup wizard

* style(code): cleanups

* fix(session): reload config from disk when switching sessions

* style(fmt): rustfmt

* fix(config): propagate cli config options to screen

* style(fmt): rustfmt
* feat(ux): reload config at runtime

* style(fmt): rustfmt
imsnif and others added 7 commits December 31, 2024 12:36
* initial draft

* working with floating panes as well

* use the same method for applying an initial layout to tiled panes

* some refactoring

* all code paths working with logical positioning fallback!

* get tests to compile

* get e2e tests to pass

* fix e2e remote runner

* breadth-first layout sorting

* fix some bugs

* style(fmt): rustfmt

* style(fmt): remove comments
The response to the 2026 mode query was missing a `?` character.

The response should be of the format `CSI ? 2026 ; N $ y` where N can
be any value in the range 0-4 inclusive.

References:

https://gist.github.com/christianparpart/d8a62cc1ab659194337d73e399004036
https://vt100.net/docs/vt510-rm/DECRPM.html
@imsnif
Copy link
Member

imsnif commented Dec 31, 2024

After receiving permission from @AutumnMeowMeow - I'm continuing the work on this. Just rebased from main.

@imsnif
Copy link
Member

imsnif commented Jan 9, 2025

I spent some time with this and at this point almost everything is working:

  1. Copying text with the mouse is working
  2. Dragging floating panes with the mouse is working
  3. Scrolling focused/unfocused panes with the mouse is working
  4. Pinning panes with the mouse is working
  5. Focusing panes with left/right click is working
  6. When Zellij does not intercept the event for one of the above, we serialize it to the pane properly (thanks to @AutumnMeowMeow 's fantastic work)

Left to do functionality-wise (mostly minor quirks):

  • do not clear the "copied to clipboard" message when the mouse just moves without a button press
  • text selection doesn't release properly when release event happens outside of pane (probably need to turn self.selecting into an Option to solve this)
  • if we focus a pane with a right click or middle click, we shouldn't also send the pane the right click
  • marking text under and left to a pinned pane doesn't work

Otherwise I'd like to get the tests/formatting working, write some more tests for this, do more robust testing for forwarding these events to the underlying panes (maybe vttest as @AutumnMeowMeow suggested) since I might have inadvertently broken something, and then we can merge so we can start adding really cool UI/UX stuff with these crazy good bits of functionality.

@imsnif
Copy link
Member

imsnif commented Jan 13, 2025

Right, so - worked on this some more. Everything is mostly fixed and working!

More things to do:
1.Fix one e2e test (EDIT: my bad, it's not broken)
2. Adjust the default plugins for the change in API (I decided to issue a breaking behavior change here rather than trying to adapt to the old events) so that stuff like clicking on tab names to focus them will work again
3. Conditionally send a notification to plugins about the state change when clicking a mouse action changed something (we used to do this on every click but this would be quite spammy now!)
4. Write some integration tests for AnyEvent

All in all - not a lot more to do. I'm hoping to have this merged soon.

@imsnif imsnif marked this pull request as ready for review January 14, 2025 14:13
@imsnif
Copy link
Member

imsnif commented Jan 14, 2025

Everything is done and working great! 🎉

Once the CI passes I'm going to merge this. If @AutumnMeowMeow has the time and feels like testing this, it would be most welcome (even after the merge - three's still some time before the release).

Thanks again for all your hard work!

@imsnif imsnif merged commit 1ca7477 into zellij-org:main Jan 14, 2025
6 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.