Mouse scroll wheel acceleration, implemented in user space
Background: Mouse scroll wheel acceleration
What is that?
It's the same as mouse cursor movement acceleration, but for the mouse scroll wheel.
This makes esp sense for devices where scrolling is continuous, such as the trackpad (but even if the scrolling is discrete, like other mouse, it makes sense). In any case you want to be able to scroll slowly (e.g. pixel by pixel, or maybe max only a couple of lines). If you are in a very long document / webpage, you also want to be able to scroll very fastly. It is not possible to have both slow and fast scrolling without acceleration. With acceleration, this becomes very natural.
If you have used MacOSX, you have that enabled. You might not even have noticed, as this feels very natural.
Non-MacOSX support
Unfortunately, this is not supported yet in other desktop operating systems (Linux or Windows) (it is supported on iOS and Android though). When you switch from MacOSX to Linux or Windows, you will probably greatly miss this. (If you have never used MacOSX much, you might not have gotten used to it, and might not even know what you miss.)
As this is not implemented in the OS, some applications and frameworks slowly start to add their own support for it. This is really a bad solution, as it means that the behavior will be inconsistent from app to app. E.g. GTK has some support (here), Firefox has some support (I think only trackpad, here).
If you want to do it right, it's still not so clear where exactly this should be implemented. Long time ago (2010), I implemented a patch for xf86-input-mouse (here). This code used a similar acceleration logic as the mouse cursor movement acceleration. In Xorg/X11, you get discrete button press events for scroll events, which made it a bit ugly. The discussion was mostly about whether this is useful at all, and also where to actually implement it, where the conclusion was mostly either in libinput, or in xf86-input-libinput. A new proposal for libinput mouse wheel acceleration was opened here. As continuous scrolling and high resolution scrolling becomes more widely used, corresponding support in libinput for high-resolution scroll wheel support was merged now (2021). This was blocking any further development on the scroll acceleration. Which probably makes sense, as a clean high resolution API makes any implementation of scroll acceleration much cleaner. However, it is also slightly problematic, as applications which do not support the new high-res scroll API will use the old API.
For reference, in MacOSX, this is deeply implemented in the kernel (see here), specifically in IOHIDFamily (e.g. see here).
User space implementation
I just want to have that support now, on my desktop.
How?
We can just send extra scroll events, and basically replicate the logic of my original xf86-input-mouse patch.
This uses pynput both to listen to scroll events, and also to send out further scroll events.
Pynput supports all the major desktop platforms like X11, Wayland, MacOSX and Windows. It even works on MacOSX in addition to the OS scroll acceleration, such that you can further increase the acceleration.
Dependencies
pip install -r requirements.txt
Usage
You can customize the behavior with two numeric values. Example:
./main.py -v --exp 0.4 --multiplier 1.2
Settings
exp
: the exponential factormultiplier
: additional multiplier. if this is >1, it means that every single scroll event will always get multiplied by this factor
The formula is:
m = user_scroll_speed ** exp
target_scroll_speed = user_scroll_speed * m * multiplier
Installation
If you found values that work for you, you can install the script as a systemd user unit (only on Linux):
./install-daemon.py
This will create a configuration file in
~/.config/mouse-scroll-wheel-accelerator/config.py
.
Enter your preferred values there.
The systemd unit can then be controlled like this:
systemctl enable --now --user mouse-scroll-wheel-accelerator
systemctl status --user mouse-scroll-wheel-accelerator
systemctl restart --user mouse-scroll-wheel-accelerator