スクリプトでグラフィックタブレット(たとえば、wacomのグラフィックタブレット)をシミュレートしたいと思います。進行する「良い」方法は、uinputの上にある抽象化レイヤであるlibeventを使用するようです。そのため、Pythonライブラリを使用してorなどのいくつかのイベントを送信するスクリプトを作成しようとしていますEV_ABS.ABS_X
。残念ながら、Krita / Gimp / ...を使用してテストすると、描画線は圧力に応じて外観を変えず、イベントも表示されません。理由をご存知ですか?EV_ABS.ABS_PRESSURE
libevent
xinput test-xi2
pressure
ありがとうございます!
再現段階
次のコードを実行します。
sudo pip3 install libevdev
chmod +x ./simulate_graphics_tablet.py
sudo ./simulate_graphics_tablet.py
これは30秒を与えます:
xinput list
次に実行します。xinput test-xi2 <number of Tablet alone>
- または、Gimpを開き、「編集/入力デバイス」に移動して、「タブレットのみ」デバイスを「画面」に設定し、ポップアップを保存して閉じてから、新しいファイルを作成し(Ctrl-N)、ズームインして「Tab」を押します。上書きします。ほとんどの画面の描画面。 「p」キーを使ってブラシに切り替え、ブラシがに設定されていることを確認してください
Pressure size
。
私が得るものは次のとおりです。均一な線、xinputからの圧力への参照はありません。次のようなものがあります。
EVENT type 17 (RawMotion)
device: 11 (11)
detail: 0
flags:
valuators:
0: 29897.54 (29897.54)
1: 29897.54 (29897.54)
私が期待するのは、一定ではないサイズの行(スクリプトが直線的に圧力を上げる)またはxinputでいくつかの圧力関連イベントを見ることです。
#!/usr/bin/env python3
import sys
import libevdev
import time
def main(args):
dev = libevdev.Device()
dev.name = "Tablet alone"
dev.enable(libevdev.EV_ABS.ABS_X,
libevdev.InputAbsInfo(minimum=0, maximum=32767))
dev.enable(libevdev.EV_ABS.ABS_Y,
libevdev.InputAbsInfo(minimum=0, maximum=32767))
dev.enable(libevdev.EV_ABS.ABS_Z,
libevdev.InputAbsInfo(minimum=0, maximum=8191))
# dev.enable(libevdev.EV_ABS.ABS_0B,
# libevdev.InputAbsInfo(minimum=0, maximum=8191))
# dev.enable(libevdev.EV_ABS.ABS_DISTANCE,
# libevdev.InputAbsInfo(minimum=0, maximum=8191))
dev.enable(libevdev.EV_ABS.ABS_PRESSURE,
libevdev.InputAbsInfo(minimum=0, maximum=8191))
dev.enable(libevdev.EV_MSC.MSC_SCAN)
dev.enable(libevdev.EV_KEY.KEY_P)
dev.enable(libevdev.EV_KEY.BTN_LEFT)
dev.enable(libevdev.EV_KEY.BTN_RIGHT)
dev.enable(libevdev.EV_KEY.BTN_MIDDLE)
dev.enable(libevdev.EV_KEY.BTN_TOUCH)
dev.enable(libevdev.EV_SYN.SYN_REPORT)
dev.enable(libevdev.EV_SYN.SYN_CONFIG)
dev.enable(libevdev.EV_SYN.SYN_MT_REPORT)
dev.enable(libevdev.EV_SYN.SYN_DROPPED)
dev.enable(libevdev.EV_SYN.SYN_04)
dev.enable(libevdev.EV_SYN.SYN_05)
dev.enable(libevdev.EV_SYN.SYN_06)
dev.enable(libevdev.EV_SYN.SYN_07)
dev.enable(libevdev.EV_SYN.SYN_08)
dev.enable(libevdev.EV_SYN.SYN_09)
dev.enable(libevdev.EV_SYN.SYN_0A)
dev.enable(libevdev.EV_SYN.SYN_0B)
dev.enable(libevdev.EV_SYN.SYN_0C)
dev.enable(libevdev.EV_SYN.SYN_0D)
dev.enable(libevdev.EV_SYN.SYN_0E)
dev.enable(libevdev.EV_SYN.SYN_MAX)
try:
uinput = dev.create_uinput_device()
print("New device at {} ({})".format(uinput.devnode, uinput.syspath))
# Sleep for a bit so udev, libinput, Xorg, Wayland, ...
# all have had a chance to see the device and initialize
# it. Otherwise the event will be sent by the kernel but
# nothing is ready to listen to the device yet.
print("Waiting 30s to let you:")
print("1) open Gimp")
print("2) Go to 'Edit/Input device' and configure the device 'Tablet alone' to 'Screen'.")
print("3) Save and close the pop up")
print("4) Create a new file (Ctrl-N)")
print("5) Zoom and press 'tab' to have a drawing surface coverint most of the screen.")
print("6) Switch to brush using 'p' key.")
time.sleep(30)
pc = 0
direc = +1
already_pressed_one = False
# uinput.send_events([
# libevdev.InputEvent(libevdev.EV_KEY.KEY_P, 1),
# libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT, 0),
# ])
# time.sleep(0.1)
# uinput.send_events([
# libevdev.InputEvent(libevdev.EV_KEY.KEY_P, 0),
# libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT, 0),
# ])
for i in range(250):
pc_ = pc/100
val_x = int(pc_*10000 + (1-pc_)*17767)
val_y = int(pc_*5000 + (1-pc_)*22767)
val_pres = int(pc_*10 + (1-pc_)*6000)
print("Will send: x={}, y={}, press={} (pc={})".format(
val_x,
val_y,
val_pres,
pc))
uinput.send_events([
libevdev.InputEvent(libevdev.EV_ABS.ABS_PRESSURE, val_pres),
libevdev.InputEvent(libevdev.EV_ABS.ABS_X, val_y),
libevdev.InputEvent(libevdev.EV_ABS.ABS_Y, val_y),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT, 0),
])
pc += direc
if not already_pressed_one:
print("Press!")
uinput.send_events([
libevdev.InputEvent(libevdev.EV_KEY.BTN_LEFT, 1),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT, 0),
])
already_pressed_one = True
if pc >= 100 or pc <=0 :
print("Release click.")
uinput.send_events([
libevdev.InputEvent(libevdev.EV_KEY.BTN_LEFT, 0),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT, 0),
])
if pc >= 100:
pc = 100
direc = -1
if pc <= 0:
pc = 0
direc = +1
time.sleep(10)
print("Press!")
uinput.send_events([
libevdev.InputEvent(libevdev.EV_KEY.BTN_LEFT, 1),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT, 0),
])
already_pressed_one = True
time.sleep(0.1)
except KeyboardInterrupt:
pass
except OSError as e:
print(e)
if __name__ == "__main__":
if len(sys.argv) > 2:
print("Usage: {}")
sys.exit(1)
main(sys.argv)
編集:説明どおりにABS_KEY.BTN_TOOL_PENを試しました。ここただし、有効にすると検出されなくなった理由がわかりませんxinput list
。
dev.enable(libevdev.EV_KEY.BTN_TOOL_PEN)
答え1
一定時間が経過した後文書/パスワード読んでまた読んだ後、ついに解決策を見つけました。
(この回答の終わりにある)スクリプトをテストしたい場合は、rootとして実行してください。
$ chmod +x completely_fake_tablet.py
$ sudo pip3 install libevdev
$ sudo ./completely_fake_tablet.py
次に、GIMPを開き、デバイスを入力デバイスとして追加し、ブラシダイナミクスを選択してPencil Generic
お楽しみください。詳細を知りたい場合は、以下をお読みください。
全体として、システムは非常に厳しいので、次の点を確認する必要があります。
libevdev.INPUT_PROP_DIRECT
タブレットに似たデバイスがあると言えます(上記のドキュメントリンクを参照)。次のタブレットに似たすべてのツールを有効にできます。
libevdev.EV_KEY.BTN_TOOL_PEN
これは、ペンがタブレットの近くにあるときにクリックしたlibevdev.EV_KEY.BTN_TOUCH
ときを示すために使用されます。libevdev.EV_KEY.BTN_STYLUS
/はlibevdev.EV_KEY.BTN_STYLUS2
ペンのボタンに対応します。libevdev.EV_ABS.ABS_{X,Y}
場所の場合(最小、最大、そして解像度:解像度がないとデバイスは検出されません! )libevdev.EV_ABS.ABS_PRESSURE
圧力のためlibevdev.EV_SYN.SYN_REPORT
この情報は、情報ブロックが送信されるたびに送信されなければなりません。これを送信しないと、カーネルはイベントを処理しないか、または非常に遅い速度(たとえば1 / s)で処理します。
また確認したいデバイスを有効にしてから1秒以上待ちます。イベントを送信する前にそうでなければ、今後のタブレットは認識されません。また、xinput
最初のイベントを送信するまでマウスのペン部分がリストされていないことも確認しました。 2つのデバイスが一覧表示されますxinput
。 1つはキーボードであると推測されるボタン用で、もう1つはペン用です(名前Tablet alone Pen (0)
とTablet alone
:
$ xinput list
⎡ Virtual core pointer id=2 [master pointer (3)]
⎜ ↳ Virtual core XTEST pointer id=4 [slave pointer (2)]
⎜ ↳ ETPS/2 Elantech Touchpad id=17 [slave pointer (2)]
⎜ ↳ lircd-uinput id=18 [slave pointer (2)]
⎜ ↳ Tablet alone Pen (0) id=12 [slave pointer (2)]
⎣ Virtual core keyboard id=3 [master keyboard (2)]
↳ Virtual core XTEST keyboard id=5 [slave keyboard (3)]
↳ Power Button id=6 [slave keyboard (3)]
↳ Asus Wireless Radio Control id=7 [slave keyboard (3)]
↳ Video Bus id=8 [slave keyboard (3)]
↳ Video Bus id=9 [slave keyboard (3)]
↳ Sleep Button id=10 [slave keyboard (3)]
↳ USB2.0 HD UVC WebCam: USB2.0 HD id=14 [slave keyboard (3)]
↳ Asus WMI hotkeys id=15 [slave keyboard (3)]
↳ AT Translated Set 2 keyboard id=16 [slave keyboard (3)]
↳ lircd-uinput id=19 [slave keyboard (3)]
↳ Tablet alone id=11 [slave keyboard (3)]
テストにGimpを使用している場合は、ソフトウェアを開く必要があります。後ろに xinput
リストできる必要があります。それ以外の場合は、入力デバイスに表示されず、GIMPを再起動する必要があります(GIMPを再起動せずにスクリプトを再起動できることに注意してください)。また、Edit/input device
デバイスで設定し、Tablet alone
スクリーンショットとScreen
同様の動的設定を持つブラシを選択する必要があります。Pencil Generic
スクリプトが正しく機能するには、を押してTab
より広い描画領域を取得し(Tab
再び通常のウィンドウに戻る)、すべての領域が含まれるまでズームする必要があります。
スクリプト:
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
import sys
import libevdev
import time
## Some doc needed for this project
# http://www.infradead.org/~mchehab/kernel_docs_pdf/linux-input.pdf
## Some code to get inspiration from
# https://github.com/linuxwacom/input-wacom/blob/master/4.5/wacom_w8001.c
## Some doc to read at some point in my life:
# https://lwn.net/Kernel/LDD3/
# https://www.kernel.org/doc/html/v4.11/driver-api/index.html
def main(args):
dev = libevdev.Device()
dev.name = "Tablet alone"
### NB: all the following information needs to be enabled
### in order to recognize the device as a tablet.
# Say that the device will send "absolute" values
dev.enable(libevdev.INPUT_PROP_DIRECT)
# Say that we are using the pen (not the erasor), and should be set to 1 when we are at proximity to the device.
# See http://www.infradead.org/~mchehab/kernel_docs_pdf/linux-input.pdf page 9 (=13) and guidelines page 12 (=16), or the https://github.com/linuxwacom/input-wacom/blob/master/4.5/wacom_w8001.c (rdy=proximity)
dev.enable(libevdev.EV_KEY.BTN_TOOL_PEN)
dev.enable(libevdev.EV_KEY.BTN_TOOL_RUBBER)
# Click
dev.enable(libevdev.EV_KEY.BTN_TOUCH)
# Press button 1 on pen
dev.enable(libevdev.EV_KEY.BTN_STYLUS)
# Press button 2 on pen, see great doc
dev.enable(libevdev.EV_KEY.BTN_STYLUS2)
# Send absolute X coordinate
dev.enable(libevdev.EV_ABS.ABS_X,
libevdev.InputAbsInfo(minimum=0, maximum=32767, resolution=100))
# Send absolute Y coordinate
dev.enable(libevdev.EV_ABS.ABS_Y,
libevdev.InputAbsInfo(minimum=0, maximum=32767, resolution=100))
# Send absolute pressure
dev.enable(libevdev.EV_ABS.ABS_PRESSURE,
libevdev.InputAbsInfo(minimum=0, maximum=8191))
# Use to confirm that we finished to send the informations
# (to be sent after every burst of information, otherwise
# the kernel does not proceed the information)
dev.enable(libevdev.EV_SYN.SYN_REPORT)
# Report buffer overflow
dev.enable(libevdev.EV_SYN.SYN_DROPPED)
try:
uinput = dev.create_uinput_device()
print("New device at {} ({})".format(uinput.devnode, uinput.syspath))
# Sleep for a bit so udev, libinput, Xorg, Wayland, ...
# all have had a chance to see the device and initialize
# it. Otherwise the event will be sent by the kernel but
# nothing is ready to listen to the device yet. And it
# will never be detected in the futur ;-)
time.sleep(1)
# Reports that the PEN is close to the surface
# Important to make sure xinput can detect (and list)
# the pen. Otherwise, it won't write anything in gimp.
uinput.send_events([
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOUCH,
value=0),
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOOL_PEN,
value=1),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT,
value=0),
])
# Says that the pen it out of range of the tablet. Useful
# to make sure you can move your mouse, and to avoid
# strange things during the first draw.
uinput.send_events([
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOUCH,
value=0),
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOOL_PEN,
value=0),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT,
value=0),
])
print("Waiting 30s to let you:")
print("1) open Gimp")
print("2) Go to 'Edit/Input device' and configure the device 'Tablet alone' to 'Screen'.")
print("3) Save and close the pop up")
print("4) Create a new file (Ctrl-N)")
print("5) Zoom and press 'tab' to have a drawing surface coverint most of the screen.")
print("6) Switch to brush using 'p' key.")
time.sleep(25)
pc = 0
direc = +1
already_pressed_one = False
for i in range(250):
pc_ = pc/100
val_x = int(pc_*10000 + (1-pc_)*17767)
val_y = int(pc_*5000 + (1-pc_)*22767)
val_pres = int(pc_*10 + (1-pc_)*6000)
print("Will send: x={}, y={}, press={} (pc={})".format(
val_x,
val_y,
val_pres,
pc))
uinput.send_events([
libevdev.InputEvent(libevdev.EV_ABS.ABS_X,
value=val_y),
libevdev.InputEvent(libevdev.EV_ABS.ABS_Y,
value=val_y),
libevdev.InputEvent(libevdev.EV_ABS.ABS_PRESSURE,
value=val_pres),
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOUCH,
value=1),
libevdev.InputEvent(libevdev.EV_KEY.BTN_STYLUS,
value=0),
libevdev.InputEvent(libevdev.EV_KEY.BTN_STYLUS2,
value=0),
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOOL_PEN,
value=1),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT,
value=0),
])
pc += direc
if not already_pressed_one:
print("Press!")
uinput.send_events([
# Pen close to device
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOOL_PEN,
value=1),
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOUCH,
value=1),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT,
value=0),
])
already_pressed_one = True
if pc >= 100 or pc <=0 :
print("Release click.")
uinput.send_events([
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOUCH,
value=0),
# Pen outside of the position
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOOL_PEN,
value=0),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT,
value=0),
])
if pc >= 100:
pc = 100
direc = -1
if pc <= 0:
pc = 0
direc = +1
time.sleep(5)
print("Press!")
uinput.send_events([
# Pen close to device
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOOL_PEN,
value=1),
libevdev.InputEvent(libevdev.EV_KEY.BTN_TOUCH,
value=1),
libevdev.InputEvent(libevdev.EV_SYN.SYN_REPORT,
value=0),
])
already_pressed_one = True
time.sleep(0.1)
except KeyboardInterrupt:
pass
except OSError as e:
print(e)
if __name__ == "__main__":
if len(sys.argv) > 2:
print("Usage: {}")
sys.exit(1)
main(sys.argv)
これで圧力xinput test <id you get with xinput list>
も表示されます。
$ xinput test 12
motion a[0]=4151295 a[1]=4151295 a[2]=241
motion a[0]=4060671 a[1]=4060671 a[2]=226
motion a[0]=3969535 a[1]=3969535 a[2]=211
motion a[0]=3878399 a[1]=3878399 a[2]=196
motion a[0]=3787775 a[1]=3787775 a[2]=181
motion a[0]=3696639 a[1]=3696639 a[2]=166
motion a[0]=3605503 a[1]=3605503 a[2]=151
motion a[0]=3514879 a[1]=3514879 a[2]=137
motion a[0]=3423743 a[1]=3423743 a[2]=122
motion a[0]=3332607 a[1]=3332607 a[2]=107
motion a[0]=3241983 a[1]=3241983 a[2]=92
motion a[0]=3150847 a[1]=3150847 a[2]=77
motion a[0]=3059711 a[1]=3059711 a[2]=62
motion a[0]=2969087 a[1]=2969087 a[2]=47
motion a[0]=2877951 a[1]=2877951 a[2]=32
motion a[0]=2650623 a[1]=2650623 a[2]=17
button release 1