设置连接/断开时自动输出 HDMI 声音

设置连接/断开时自动输出 HDMI 声音

我的笔记本电脑(使用 12.04 LTS)上有一个双屏设置,使用 HDMI 连接显示器。一切正常,但每次我连接/断开电缆时,我都必须转到“声音”首选项并手动更改声音输出设备。

有没有办法在连接/断开电缆时更改声音输出设备,所以当我连接显示器时,声音输出设置为 HDMI,而当我断开连接时,声音会回到笔记本电脑扬声器?

答案1

对于那些偶然发现这个问题的人来说 - Salem 的解决方案在 13.04 中几乎对我有用,我最终从网上收集了一些信息,我认为对我来说最关键的是缺少环境变量 PULSE_SERVER

这是我的完整解决方案,基本上就是重复 Salem 的解决方案,但缺少了一些部分。我还将其重写为 shell 脚本(尽管我喜欢 Python),因为我起初担心我的 Python 脚本会遇到导入路径问题:


(与 Salem 的答案相同)/etc/udev/rules.d/hdmi_sound.rules以 root 身份创建一个文件,内容如下:

SUBSYSTEM=="drm", ACTION=="change", RUN+="/usr/local/bin/hdmi_sound_toggle"

以 root 身份创建一个文件,/usr/local/bin/hdmi_sound_toggle内容如下:

#!/bin/sh
USER_NAME=`who | grep "(:0)" | cut -f 1 -d ' '`
USER_ID=`id -u $USER_NAME`
HDMI_STATUS=`cat /sys/class/drm/card0/*HDMI*/status`

export PULSE_SERVER="unix:/run/user/"$USER_ID"/pulse/native"

if [ $HDMI_STATUS = "connected" ]
then
    sudo -u $USER_NAME pactl --server $PULSE_SERVER set-card-profile 0 output:hdmi-stereo+input:analog-stereo
else
    sudo -u $USER_NAME pactl --server $PULSE_SERVER set-card-profile 0 output:analog-stereo+input:analog-stereo
fi

然后使其可执行chmod 0755 /usr/local/bin/hdmi_sound_toggle

我尝试让这个脚本尽可能通用,但你仍然可能需要更改一些行,例如 HDMI_STATUS 文件路径或使用的配置文件。你可以运行pactl list cards并在配置文件下查看配置文件列表。

请注意,当我在设置 PULSE_SERVER 时删除关键字“export”时,脚本失败了,我认为 pactl 正在寻找环境变量

不要忘记重新加载你的 udev 规则:sudo udevadm control --reload-rules

更新此脚本已针对 14.04 进行了更新。在此之前,您需要在任何地方使用 USER_NAME 而不是 USER_ID

答案2

Ubuntu 16.04 - 20.04 答案

这适用于 Ubuntu 16.04 - 20.04,该系统引入了 Pulse Audio 8 的一个错误。创建文件hotplugtv(或者hotplug-hdmi如果您愿意)并复制以下几行:

#!/bin/bash

# NAME: hotplugtv
# PATH: /home/$USER/bin
# DESC: Update pulseaudio output device when HDMI TV plugged / unplugged
# CALL: called from /etc/udev/rules.d/99-hotplugtv.rules 
#       and /home/$USER/bin/lock-screen-timer
# DATE: Created Nov 26, 2016.
# NOTE: logs output using log-file
# UPDT: Dec 14, 2016 - Sometimes /sys/class/drm/card0 & sometimes /sys/class/drm/card1
#       so use /sys/class/dmcard* instead.
#       Dec 21, 2016 - Relocated to /home/$USER/bin for calling by lock-screen-timer
#       Aug 06, 2017 - Convert from home grown log-file to universal logger command.

if [[ $(cat /sys/class/drm/card*-HDMI-A-1/status | grep -Ec "^connected") -eq 1 ]]; then
        logger -t /home/rick/bin/log-hotplugtv "HDMI TV connected"
        /bin/sleep 2;
        export PULSE_RUNTIME_PATH="/run/user/1000/pulse/";
        sudo -u rick -E pacmd set-card-profile 0 output:hdmi-stereo;
else
        logger -t /home/rick/bin/log-hotplugtv "HDMI TV disconnected"
        export PULSE_RUNTIME_PATH="/run/user/1000/pulse/";
        sudo -u rick -E pacmd set-card-profile 0 output:analog-stereo;
fi

exit 0

重要的:将用户名“rick”更改为您的用户名。

为了在udev热插拔事件期间调用此脚本,请创建/etc/udev/rules.d/99-hotplugtv.rules包含以下内容的文件:

ACTION=="change", SUBSYSTEM=="drm", ENV{HOTPLUG}=="1", RUN+="/home/rick/bin/hotplugtv"

更改/home/rick/bin/为放置脚本的路径hotplugtv

答案3

我最终设法使用 使它工作udev。因此,如果有人想要相同的行为,请按照以下步骤操作:

首先我们需要创建一个/etc/udev/rules.d/hdmi_sound.rules包含以下内容的文件:

    SUBSYSTEM=="drm", ACTION=="change", RUN+="/usr/local/bin/hdmi_sound_toggle"

这将使得每次 HDMI 连接发生变化时udev执行该脚本hdmi_sound_toggle。该脚本必须具有执行权限,内容如下:

#!/usr/bin/env python

import subprocess
from syslog import syslog

def output(cmd):
    return subprocess.check_output(cmd, shell=True)

# the following variables may need some modification.
user = "my_username"
card = "/sys/class/drm/card0"
dev_speaker = "output:analog-stereo+input:analog-stereo"
dev_hdmi = "output:hdmi-stereo+input:analog-stereo"
#

interfaces = output("ls {0}".format(card), ).split("\n")

vga = filter(lambda x: "VGA" in x, interfaces)[0]
hdmi = filter(lambda x: "HDMI" in x, interfaces)[0]

syslog("HDMI connection was changed!")

hdmi_connected = output("cat {0}/{1}/status".format(card,hdmi)).startswith("connected")
title = "HDMI was {0}".format("connected" if hdmi_connected else "disconnected")
message = "Audio output has changed to {opt}.".format(opt = "HDMI" if hdmi_connected else "built-in speakers")

cmd = "sudo -u " + user + " /usr/bin/pactl set-card-profile 0 " + (dev_hdmi if hdmi_connected else dev_speaker)

syslog("HDMI was connected." if hdmi_connected else "HDMI was disconnected.")
try:
    a = output(cmd)
    output("sudo -u {0} notify-send \"{1}\" \"{2}\"".format(user, title, message))
    syslog("Audio output changed.")
except Exception as ex:
    syslog("Error changing output device: " + str(ex))

可能这可以在 bash 中轻松实现,但由于我的主要语言是 python,所以我使用了它。除了通知之外,一切都正常:它没有显示,我真的不知道为什么。如果有人知道如何修复它,请说点什么。

注意:脚本/udev规则的名称可以更改,但需要使用完整路径。

答案4

有一个更简单的方法:

在 中定义 default-sink 和 default-source /etc/pulse/default.pa,如下所述:https://rastating.github.io/setting-default-audio-device-in-ubuntu-18-04/

grep default- /etc/pulse/default.pa
load-module module-default-device-restore
set-default-sink alsa_output.pci-0000_00_1f.3.analog-stereo
set-default-source alsa_input.pci-0000_00_1f.3.analog-stereo

重试之前不要忘记重新启动 pulseaudio:

pulseaudio -k

相关内容