-
-
Save mildmojo/48e9025070a2ba40795c to your computer and use it in GitHub Desktop.
#!/bin/bash | |
# | |
# rotate_desktop.sh | |
# | |
# Rotates modern Linux desktop screen and input devices to match. Handy for | |
# convertible notebooks. Call this script from panel launchers, keyboard | |
# shortcuts, or touch gesture bindings (xSwipe, touchegg, etc.). | |
# | |
# Using transformation matrix bits taken from: | |
# https://wiki.ubuntu.com/X/InputCoordinateTransformation | |
# | |
# Configure these to match your hardware (names taken from `xinput` output). | |
TOUCHPAD='SynPS/2 Synaptics TouchPad' | |
TOUCHSCREEN='Atmel Atmel maXTouch Digitizer' | |
if [ -z "$1" ]; then | |
echo "Missing orientation." | |
echo "Usage: $0 [normal|inverted|left|right] [revert_seconds]" | |
echo | |
exit 1 | |
fi | |
function do_rotate | |
{ | |
xrandr --output $1 --rotate $2 | |
TRANSFORM='Coordinate Transformation Matrix' | |
case "$2" in | |
normal) | |
[ ! -z "$TOUCHPAD" ] && xinput set-prop "$TOUCHPAD" "$TRANSFORM" 1 0 0 0 1 0 0 0 1 | |
[ ! -z "$TOUCHSCREEN" ] && xinput set-prop "$TOUCHSCREEN" "$TRANSFORM" 1 0 0 0 1 0 0 0 1 | |
;; | |
inverted) | |
[ ! -z "$TOUCHPAD" ] && xinput set-prop "$TOUCHPAD" "$TRANSFORM" -1 0 1 0 -1 1 0 0 1 | |
[ ! -z "$TOUCHSCREEN" ] && xinput set-prop "$TOUCHSCREEN" "$TRANSFORM" -1 0 1 0 -1 1 0 0 1 | |
;; | |
left) | |
[ ! -z "$TOUCHPAD" ] && xinput set-prop "$TOUCHPAD" "$TRANSFORM" 0 -1 1 1 0 0 0 0 1 | |
[ ! -z "$TOUCHSCREEN" ] && xinput set-prop "$TOUCHSCREEN" "$TRANSFORM" 0 -1 1 1 0 0 0 0 1 | |
;; | |
right) | |
[ ! -z "$TOUCHPAD" ] && xinput set-prop "$TOUCHPAD" "$TRANSFORM" 0 1 0 -1 0 1 0 0 1 | |
[ ! -z "$TOUCHSCREEN" ] && xinput set-prop "$TOUCHSCREEN" "$TRANSFORM" 0 1 0 -1 0 1 0 0 1 | |
;; | |
esac | |
} | |
XDISPLAY=`xrandr --current | grep primary | sed -e 's/ .*//g'` | |
XROT=`xrandr --current --verbose | grep primary | egrep -o ' (normal|left|inverted|right) '` | |
do_rotate $XDISPLAY $1 | |
if [ ! -z "$2" ]; then | |
sleep $2 | |
do_rotate $XDISPLAY $XROT | |
exit 0 | |
fi | |
And which script are you using? There are so many here …
If you use this script https://gist.github.com/mildmojo/48e9025070a2ba40795c#gistcomment-3709782, please update it to the current version (there was a typo in there) and post the output of these commands called in the terminal:
xrandr --current --verbose | grep primary xrandr --current --verbose | grep primary | cut --delimiter=" " -f1
Thats the one i am using and the last release.
~$ xrandr --current --verbose | grep primary eDP-1 connected primary 1920x1080+0+0 (0x45) normal (normal left inverted right x axis y axis) 294mm x 165mm
and
xrandr --current --verbose | grep primary | cut --delimiter=" " -f1 eDP-1
Thanks @javiercviegas! That shows us, that the script finds the primary monitor (it's eDP-1
, by the way @frgomes uses a very similar approach).
Are you happy to debug it further? Then please issue the following three (rather lengthy) commands and post the results:
dbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method org.freedesktop.DBus.Properties.Get net.hadess.SensorProxy HasAccelerometer
gdbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method org.freedesktop.DBus.Properties.Get net.hadess.SensorProxy AccelerometerOrientation
gdbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method net.hadess.SensorProxy.ReleaseAccelerometer
That's also where there is a major difference to @frgomes's script: Mine gets the orientation from the orientation from the built-in accelerometer and @frgomes's script from the current orientation of the primary screen. For @frgomes to work, you have to enable the automatic screen rotation in your desktop's settings (and then the following can happen: The screen switches automatically to a different orientation, but the input device does not). For my script to work, you should disable automatic screen rotation and then hold the device in the orientation you like and call my script. It's your choice, what you prefer.
I just noticed another difference: My script rotates all active input devices (so you need to use your pen just before rotation, if it is a bluetooth pen), @frgomes's script only rotates the pen, but not the touchpad (if I understand his script correctly.
Thanks @javiercviegas! That shows us, that the script finds the primary monitor (it's
eDP-1
, by the way @frgomes uses a very similar approach).Are you happy to debug it further? Then please issue the following three (rather lengthy) commands and post the results:
dbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method org.freedesktop.DBus.Properties.Get net.hadess.SensorProxy HasAccelerometer gdbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method org.freedesktop.DBus.Properties.Get net.hadess.SensorProxy AccelerometerOrientation gdbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method net.hadess.SensorProxy.ReleaseAccelerometerThat's also where there is a major difference to @frgomes's script: Mine gets the orientation from the orientation from the built-in accelerometer and @frgomes's script from the current orientation of the primary screen. For @frgomes to work, you have to enable the automatic screen rotation in your desktop's settings (and then the following can happen: The screen switches automatically to a different orientation, but the input device does not). For my script to work, you should disable automatic screen rotation and then hold the device in the orientation you like and call my script. It's your choice, what you prefer.
I just noticed another difference: My script rotates all active input devices (so you need to use your pen just before rotation, if it is a bluetooth pen), @frgomes's script only rotates the pen, but not the touchpad (if I understand his script correctly.
@sphh Sure i am glad to help for other's to use this scripts.
1.- dbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method org.freedesktop.DBus.Properties.Get net.hadess.SensorProxy HasAccelerometer
(,)
2.- gdbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method org.freedesktop.DBus.Properties.Get net.hadess.SensorProxy AccelerometerOrientation
(<'undefined'>,)
3.- gdbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method net.hadess.SensorProxy.ReleaseAccelerometer
()
My distro (Deepin 20.2) does not have an auto rotate function you can only do it manually.
Aha, that explains a bit. It cannot find the accelerometer. I was hoping, that this is caught by my script, but apparently is not.
On my system I had to install the iio-sensor-proxy
. It looks either that this is not installed, not started or that your computer does not have an accelerometer.
Please check, that iio-sensor-proxy
is installed. Then check with this command, if it is started (assuming that your system is using systemd
, which is very likely):
systemctl status iio-sensor-proxy.service
Aha, that explains a bit. It cannot find the accelerometer. I was hoping, that this is caught by my script, but apparently is not.
On my system I had to install the
iio-sensor-proxy
. It looks either that this is not installed, not started or that your computer does not have an accelerometer.Please check, that
iio-sensor-proxy
is installed. Then check with this command, if it is started (assuming that your system is usingsystemd
, which is very likely):systemctl status iio-sensor-proxy.service
`
● iio-sensor-proxy.service - IIO Sensor Proxy service
Loaded: loaded (/lib/systemd/system/iio-sensor-proxy.service; static; vendor preset: enabled)
Active: active (running) since Tue 2021-05-18 16:01:16 EDT; 2h 48min left
Main PID: 664 (iio-sensor-prox)
Tasks: 3 (limit: 4915)
Memory: 1.1M
CGroup: /system.slice/iio-sensor-proxy.service
└─664 /usr/sbin/iio-sensor-proxy
Warning: Journal has been rotated since unit was started. Log output is incomplete or unavailable.
`
@javiercviegas: You are living in the future! 16 o'clock EDT is actually 3 hours away.
Anyway, if you are certain, that iio-sensor-proxy
was started before you issued those three long commands, than you computer does not have an accelerometer which is recognized by iio-sensor-proxy
…
(Still there is a bug in the script: It does not display the proper error message and exit, if it cannot find the accelerometer. Because of this, it calls the commands to set the rotation and you see the error message related to this.)
@javiercviegas: You are living in the future! 16 o'clock EDT is actually 3 hours away.
Anyway, if you are certain, that
iio-sensor-proxy
was started before you issued those three long commands, than you computer does not have an accelerometer which is recognized byiio-sensor-proxy
…(Still there is a bug in the script: It does not display the proper error message and exit, if it cannot find the accelerometer. Because of this, it calls the commands to set the rotation and you see the error message related to this.)
@javiercviegas: You are living in the future! 16 o'clock EDT is actually 3 hours away.
Anyway, if you are certain, that
iio-sensor-proxy
was started before you issued those three long commands, than you computer does not have an accelerometer which is recognized byiio-sensor-proxy
…(Still there is a bug in the script: It does not display the proper error message and exit, if it cannot find the accelerometer. Because of this, it calls the commands to set the rotation and you see the error message related to this.)
Lol i am Marty Mcfly! Hmmm i am not sure that the iio-sensor-proxy was started before. This laptop is new and i wouldn't dare to affirm that is was already up before issuing that command.
If this laptop does not have an acelerometer (quite probably not because it's a chinese low budget BMAX Y13 Power) does this mean i cannot use the script?
Check, that iio-sensor-proxy
is up and running and try to run the first of those lenghty commands again. If you get an output of (true,)
, your computer does have a accelerometer. You could also try to run monitor-sensor
from the command line and investigate its output. If you don't have an accelerometer, you will not have any luck with autodetecting its orientation. Then neither my script nor @frgomes's script will work.
Without an accelerometer, your best bet is this script: https://gist.github.com/mildmojo/48e9025070a2ba40795c#gistcomment-2694429. You have to start it with one of these command line options: normal
, inverted
, left
or right
, e.g. rotate-screen left
.
No more hacking guys.
Let's do it properly. I asked KDE to add a new feature. We just need to wait about 2 years.
https://bugs.kde.org/show_bug.cgi?id=437326
This is the bug for Linux Mint (from 2017): linuxmint/cinnamon-settings-daemon#188
@marian556: Wouldn't it be better to use the d-bus net.hadess.SensorProxy
instead of running another command (which btw returns a lot of data related to the brightness)?
@javiercviegas: Have a look at the updated script https://gist.github.com/mildmojo/48e9025070a2ba40795c#gistcomment-3709782. It can now be used without an accelerometer. Start by calling it with rotate-script --help
to familiarize yourself with the possibilities.
@javiercviegas: Have a look at the updated script https://gist.github.com/mildmojo/48e9025070a2ba40795c#gistcomment-3709782. It can now be used without an accelerometer. Start by calling it with
rotate-script --help
to familiarize yourself with the possibilities.
Confirmed it does have an accelerometer.
… so everything should work!
But if
dbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method org.freedesktop.DBus.Properties.Get net.hadess.SensorProxy HasAccelerometer
does not return (<true>,)
, there are only five reasons I can think of:
- There is no accelerometer available.
- The
iio-sensor-proxy
is not installed. - The
iio-sensor-proxy
is installed but its service not activated. - The accelerometer is not supported by
iio-sensor-proxy
. - The D-Bus is not working.
You can also test, if everything was installed correctly, by running monitor-sensor
(and move your device). If that does not show === Has accelerometer
, you have to sort it first or use the script in "manual" mode (rotate-screen next|previous
or rotate-screen normal|left|inverted|right
). If the screen rotates automatically, you could also use rotate-screen screen
.
… so everything should work!
But if
dbus call --system --dest net.hadess.SensorProxy --object-path /net/hadess/SensorProxy --method org.freedesktop.DBus.Properties.Get net.hadess.SensorProxy HasAccelerometerdoes not return
(<true>,)
, there are only five reasons I can think of:
- There is no accelerometer available.
- The
iio-sensor-proxy
is not installed.- The
iio-sensor-proxy
is installed but its service not activated.- The accelerometer is not supported by
iio-sensor-proxy
.- The D-Bus is not working.
You can also test, if everything was installed correctly, by running
monitor-sensor
(and move your device). If that does not show=== Has accelerometer
, you have to sort it first or use the script in "manual" mode (rotate-screen next|previous
orrotate-screen normal|left|inverted|right
). If the screen rotates automatically, you could also userotate-screen screen
.
It works. But now i am thinking how can i use the accelerometer to trigger the script and at the same time use the accelerometer to determine if the mode should be left right inverted or normal
while true; do
rotate-screen
sleep 1
done
!
while true; do rotate-screen sleep 1 done!
Bare with me i am just starting to study scripts. Where exactly should i add this snippet on the original code?
You could start with a terminal and enter these commands. You can also use the following one-liner:
while true; do rotate-screen; sleep 1; done
If that is what you want, write a tiny script with this contents
#!/bin/sh
while true; do
rotate-screen
sleep 1
done
make it executable (chmod +x rotate-automatically
) and call this script.
See also: https://www.shellscript.sh/ (or any other tutorial on shell scripts).
You could start with a terminal and enter these commands. You can also use the following one-liner:
while true; do rotate-screen; sleep 1; doneIf that is what you want, write a tiny script with this contents
#!/bin/sh true; do rotate-screen sleep 1 donemake it executable (
chmod +x rotate-automatically
) and call this script.See also: https://www.shellscript.sh/ (or any other tutorial on shell scripts).
Great awesome! Thanks
thanks a lot, everything worked, I just substituted my devices
Check this script. Basically automatically rotates the screen based on the orientation
https://github.com/marcomarinho/auto-rotate-tablets-linux/tree/main
@marcomarinho: It looks, like your script uses
monitor-sensor >> sensor.log 2>&1 &
while inotifywait -e modify sensor.log; do
[...]
done
If I understand that correctly, monitor-sensor
writes the output to a file on the (hard-)disk and acts, whenever this file changes. When I start monitor-sensor
, it writes a line roughly ever 0.5s. If you do not restart your script, the sensor.log
file gets longer and longer using up more and more hard-disk space. That would trouble me, if you only send the laptop to sleep and do not restart it.
I can see another problem: SSDs do not like these regular writing and could wear out quickly.
Wouldn't it be better to write a script, which monitors the DBus (as I believe monitor-sensor
does) and act on changes of the orientation? Actually I shortly thought about adding this approach to my script, but did not do it in the end, because I personally do not like automatic screen orientation (it always flips the screen when you do not want it to do). What do you think?
As an aside, the other day my 8-year-old Yoga 11s started auto-rotating the screen when I physically change the laptop's orientation. Maybe I updated a system package somewhere and it suddenly gained support for my hardware? Wild.
Now that it's finally supported, I'd kind of like to turn it off. 😝 I now prefer setting the rotation manually with global hotkeys set to call the script in this gist. I can see how it would be useful in tablet mode, though.
Thank you so much! This works flawlessly on my XPS 13 2-in-1, even if external monitor connected, which remains unaffected. Touch events on the screen, external mouse and digitizer work on spot. Finally video conferences with a digital and streamable paper at hand.
Thank you for the script.
On yoga 11s with my custom kernel-5.15.31 on Slackware-15 I changed only one line
TOUCHSCREEN='Atmel Atmel maXTouch Digitizer'
vs
TOUCHSCREEN='Atmel Atmel maXTouch Digitizer Touchscreen'
I got similar problem, I write this simple utility app that reads sensors and rotate screen + digitizer.
How do i use this when i rotate my screen? Please help I'm new to Linux mint
@V-VR You can assign custom keyboard shortcuts in Mint's Keyboard settings. I have four of them set up, using Ctrl + Alt + Super + to manually set the screen and touchpad orientation by calling this script.
For instance, one of my custom shortcuts is called "Rotate desktop left", assigned to Ctrl + Alt + Super + Right (because the bottom of the screen will be to the right after "rotating" to the left), and it calls /home/mildmojo/bin/rotate_desktop.sh left
.
@V-VR: You can also head to the rotate script at linux-surface for a semi-automatic rotation script: If you follow the Installation there you get an icon in your panel. Whenever you rotated the Surface, you click on that icon and the screen adjusts automatically to the new rotation.
@frgomes: Interesting script. Is the
--listmonitors
parameter forxrandr
documented somewhere (man xrandr
does not show it)? Does it always return the primary monitor in the first line?