Featured image of post 转换遥控器的退出键为返回键

转换遥控器的退出键为返回键

本文为《PVE安装Kodi》系列文章的一部分。

前言

在上文 《像使用盒子/LibreELEC一样使用PVE直装的Kodi,实现遥控器启停》 中,我们把2.4G无线遥控器的电源键支持并转换为开关Kodi了。不过,有些2.4G无线遥控器的返回键不是退格键(backspace键),而是退出键(esc键),导致短按它的效果是直接退出到上一级菜单(PreviousMenu),而不是返回到上一个界面(Back)。这种情况在点的界面比较深的时候会比较明显,一点返回结果返回了很远。如果不习惯这样的操作方式,那么我们可以转换按键。

说明

看到本篇教程这里,我默认你已经按照《PVE 直接安装最新版 Kodi》《像使用盒子/LibreELEC一样使用PVE直装的Kodi,实现遥控器启停》《kodi-send使用相关说明》这三篇教程安装好了必备的软件包,编辑好了必备的脚本,相关的安装过程就不再赘述了,直接进入正题。

2.4G无线遥控器一般被Linux系统识别为键盘设备,所以这种遥控器的按键实际上就是普通键盘上的常规按键,它的按键的作用是通过/usr/share/kodi/system/keymaps/keyboard.xml来处理的(在未被Keymap Editor映射的前提下)。打开该xml文件内容可知(见下),大部分界面中长按esc的效果等同于短按backspace(然而似乎遥控器并不支持长按…普通键盘才支持…),如果你习惯短按是PreviousMenu,也可以不再往下看了。不过对我来说,短按Back才更符合我的习惯,所以我需要将这个遥控器的esc键转换为backspace键。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
<keymap>
  <global>
    <keyboard>
      ...
      <backspace>Back</backspace>
      ...
      <escape>PreviousMenu</escape>
      <escape mod="longpress">Back</escape>
      ...
    </keyboard>
    ...
  </global>
  ...
</keymap>

转换按键有很多方式,一种使用Kodi本身支持的特性,新建~/.kodi/userdata/keymaps/my_remap.xml(文件名可以自己定义,详见Keymaps~/.kodi/userdata/keymaps下的xml覆盖/usr/share/kodi/keymaps下的,同文件夹下的文件名字母序靠后的覆盖靠前的),内容可以形如下面这种:

1
2
3
4
5
6
7
8
<keymap>
  <global>
    <keyboard>
      <escape>Back</escape>
      <escape mod="longpress">PreviousMenu</escape>
    </keyboard>
  </global>
</keymap>

上面的示例就是把短按esc和长按esc的作用交换了一下(尽管遥控器可能并不支持长按)。虽然上述配置并未列全,但基本上可以应付大部分情况了,你也可以自行参考/usr/share/kodi/system/keymaps/keyboard.xml中的escape所在的全部模块进行自定义覆盖。

既然我已经在使用evsieve来胁持电源键了,那么再用它来胁持esc键也是个不错的办法,所以就有了另外第二种办法,流程如下。

流程

  1. 测试遥控器的返回键到底是个什么按键,不出意外,出现一点返回结果返回了很远的一个界面这种情况下,返回键应该就是esc键。注:在测试前请先退出Kodi,保证画面处于控制台界面。

     1
     2
     3
     4
     5
     6
     7
     8
     9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    
    # 运行evtest
    evtest --grab
    
    # evtest会列出所有可用的输入设备,可以看到我的遥控器有event5-event8共4个设备
    # 你可以选择具体的编号来进行测试,找到每个设备分别能控制哪些按键
    # 经过测试,我发现我遥控器的返回键在event5,也就是2.4G Composite Devic这个设备
    No device specified, trying to scan all of /dev/input/event*
    Available devices:
    /dev/input/event0:      Power Button
    /dev/input/event1:      AT Translated Set 2 keyboard
    /dev/input/event2:      VirtualPS/2 VMware VMMouse
    /dev/input/event3:      VirtualPS/2 VMware VMMouse
    /dev/input/event4:      QEMU QEMU USB Tablet
    /dev/input/event5:      2.4G Composite Devic
    /dev/input/event6:      2.4G Composite Devic Mouse
    /dev/input/event7:      2.4G Composite Devic Consumer Control
    /dev/input/event8:      2.4G Composite Devic System Control
    /dev/input/event9:      PC Speaker
    Select the device event number [0-9]:5   # 反馈键一般在遥控器名称最短的那个设备上,也就是2.4G Composite Devic,你也可以把多次测试一个一个试出来
    Input driver version is 1.0.1
    Input device ID: bus 0x3 vendor 0x276d product 0x1101 version 0x111
    Input device name: "2.4G Composite Devic"
    Supported events:
    Event type 0 (EV_SYN)
    Event type 1 (EV_KEY)
        Event code 1 (KEY_ESC)
        Event code 2 (KEY_1)
        Event code 3 (KEY_2)
        Event code 4 (KEY_3)
        Event code 5 (KEY_4)
        Event code 6 (KEY_5)
        ...  ## 中间太长了,略去
    Properties:
    Testing ... (interrupt to exit)  ## 按下返回键,有反应说明设备选择对了,无反应说明返回键不在这个设备上
    Event: time 1691294047.183861, type 4 (EV_MSC), code 4 (MSC_SCAN), value 70029
    Event: time 1691294047.183861, type 1 (EV_KEY), code 1 (KEY_ESC), value 1
    Event: time 1691294047.183861, -------------- SYN_REPORT ------------
    Event: time 1691294047.255862, type 4 (EV_MSC), code 4 (MSC_SCAN), value 70029
    Event: time 1691294047.255862, type 1 (EV_KEY), code 1 (KEY_ESC), value 0
    Event: time 1691294047.255862, -------------- SYN_REPORT ------------
    
    ## Ctrl+C退出
    

    由上述输出内容可知,这款遥控器的返回键是esc键,而另外一些遥控器的返回键是backspace键,这才是返回上一层而非退出的按键。所以我们接下来把这个遥控器的esc键转换为backspace键。

  2. 先确认/dev/input/by-id下已经有/dev/input/event5(对我而言,event5就是我的2.4G Composite Devic这个遥控器设备)的软连接。一般udev程序识别这个设备没有问题,不再需要自己编辑相应的udev规则;然后修改在《像使用盒子/LibreELEC一样使用PVE直装的Kodi,实现遥控器启停》中创建的脚本/usr/local/bin/run-evsieve.sh如下,并注意按照注释修改为自己的信息。

     1
     2
     3
     4
     5
     6
     7
     8
     9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    
    #!/usr/bin/env bash
    
    export PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin
    
    ## 要劫持的目标设备的绝对路径,请注意修改成自己的
    ## 不要直接写/dev/input/eventX,而要写/dev/input/by-id下的设备
    ## target_hijack_input_device1是返回键(esc键)所在的设备
    ## target_hijack_input_device2是电源键(power键)所在的设备
    target_hijack_input_device1="/dev/input/by-id/usb-0627_2.4G_Composite_Devic-event-kbd"
    target_hijack_input_device2="/dev/input/by-id/usb-0627_2.4G_Composite_Devic-System-Control"
    
    ## 要胁持的按键的键名,应该分别就是esc和power了,大概率不用改
    ## target_hijack_button1就是target_hijack_input_device1上的esc键
    ## target_hijack_button2就是target_hijack_input_device2上的power键
    target_hijack_button1="esc"
    target_hijack_button2="power"
    
    ###### 以下无需修改 ######
    ## 先检测 target_hijack_input_device1 和 target_hijack_input_device2 是否已经存在
    while :; do
        if [[ -L "$target_hijack_input_device1" && -L "$target_hijack_input_device2" ]]; then
            break
        else
            echo "The '$target_hijack_input_device1' or '$target_hijack_input_device2' is not inserted, wait 2 seconds..."
            sleep 2
        fi
    done
    
    ## 执行按键胁持,详细用法请见:https://github.com/KarsMulder/evsieve
    ## 当按下退出键时,转换为返回键
    ## 当按下电源键时,转换为运行脚本/usr/local/bin/kodi-power.sh
    exec evsieve \
        --input "$target_hijack_input_device1" grab persist=reopen \
        --map key:$target_hijack_button1 key:backspace \
        --output \
        --input "$target_hijack_input_device2" grab persist=reopen \
        --hook key:$target_hijack_button2 exec-shell="/usr/local/bin/kodi-power.sh" \
        --block key:$target_hijack_button2
    
  3. 然后重启在《像使用盒子/LibreELEC一样使用PVE直装的Kodi,实现遥控器启停》中创建的服务/etc/systemd/system/evsieve.service

    1
    2
    
    systemctl daemon-reload
    systemctl restart evsieve.service
    
  4. 确认服务正常运行起来了,然后你就可以用遥控器打开Kodi并测试退出键是否转换为返回键了。

     1
     2
     3
     4
     5
     6
     7
     8
     9
    10
    11
    12
    13
    14
    15
    
    ## 查看状态
    systemctl status evsieve.service
    
    ## 如下输出内容像下面这样,就表示胁持好了
    ● evsieve.service - Run Evsieve
        Loaded: loaded (/etc/systemd/system/evsieve.service; enabled; preset: enabled)
        Active: active (running) since Sun 2023-08-06 12:05:36 CST; 1h 20min ago
    Main PID: 3612370 (evsieve)
        Tasks: 2 (limit: 75726)
        Memory: 1.3M
            CPU: 317ms
        CGroup: /system.slice/evsieve.service
                └─3612370 evsieve --input /dev/input/by-id/usb-0627_2.4G_Composite_Devic-event-kbd grab persist=reopen --map key:esc key:backspace --output --input /dev/input/by-id/usb-0627_2.4G_Composite_Devic-System-Control grab persist=reopen --hook key:power exec-shell=/usr/local/bin/kodi-power.sh --block key:power
    
    8月 06 12:05:36 pve systemd[1]: Started evsieve.service - Run Evsieve.
    
Built with Hugo
主题 StackJimmy 设计