สคริปต์เพื่อปิดเสียงแอปพลิเคชัน


14

เป้าหมายของฉันคือสามารถปิดแอปพลิเคชั่น Spotify ไม่ใช่ระบบทั้งหมด การใช้คำสั่ง: ps -C spotify -o pid=ผมสามารถที่จะหากระบวนการ ID ของ Spotify ในกรณีนี้ ID "22981"คือ ด้วยกระบวนการ ID pacmd list-sink-inputsที่ฉันต้องการจะค้นหาจากรายการนี้: คำสั่งนั้นส่งคืนรายการเช่นนี้:

eric@eric-desktop:~$ pacmd list-sink-inputs
Welcome to PulseAudio! Use "help" for usage information.
>>> 1 sink input(s) available.
    index: 0
    driver: <protocol-native.c>
    flags: START_CORKED 
    state: RUNNING
    sink: 1 <alsa_output.pci-0000_00_1b.0.analog-stereo>
    volume: 0: 100% 1: 100%
            0: -0.00 dB 1: -0.00 dB
            balance 0.00
    muted: no
    current latency: 1019.80 ms
    requested latency: 371.52 ms
    sample spec: s16le 2ch 44100Hz
    channel map: front-left,front-right
                 Stereo
    resample method: (null)
    module: 8
    client: 10 <Spotify>
    properties:
        media.role = "music"
        media.name = "Spotify"
        application.name = "Spotify"
        native-protocol.peer = "UNIX socket client"
        native-protocol.version = "26"
        application.process.id = "22981"
        application.process.user = "eric"
        application.process.host = "eric-desktop"
        application.process.binary = "spotify"
        window.x11.display = ":0"
        application.language = "en_US.UTF-8"
        application.process.machine_id = "058c89ad77c15e1ce0dd5a7800000012"
        application.process.session_id = "058c89ad77c15e1ce0dd5a7800000012-1345692739.486413-85297109"
        application.icon_name = "spotify-linux-512x512"
        module-stream-restore.id = "sink-input-by-media-role:music"

ตอนนี้ฉันต้องการให้สัมพันธ์กับการจัดทำดัชนีการป้อนข้อมูลอ่างซึ่งในกรณีนี้คือapplication.process.id = "22981" index: 0ตอนนี้ด้วยหมายเลขดัชนีฉันจะต้องเรียกใช้คำสั่งนี้pacmd set-sink-input-mute 0 1เพื่อปิด Spotify และpacmd set-sink-input-mute 0 0เปิดเสียง Spotify สำหรับคำสั่งเหล่านั้นหมายเลขแรกจะต้องถูกแทนที่ด้วยหมายเลขดัชนีที่พบก่อนหน้านี้และหมายเลขถัดไปคือบูลีนเพื่อเปิดหรือปิดเสียง ฉันจะรวมสิ่งนี้เข้ากับสคริปต์ได้อย่างไรเพื่อให้ได้รับคำสั่งให้ปิด / เปิดเสียงแอปพลิเคชัน Spotify

แก้ไข: สคริปต์ด้านล่างทั้งสองทำงานตามที่คาดไว้ใครสามารถเพิ่มสลับที่จะตรวจสอบmuted: yesหรือmuted: noจากนั้นปิดเสียงหรือเปิดเสียงตาม?

แก้ไข: ฉันสามารถแก้ไขสคริปต์ของ glenn jackman เพื่อเพิ่มสลับ:

#!/bin/bash

main() {
    local action=toggle
    while getopts :mu option; do 
        case "$option" in 
            m) action=mute ;;
            u) action=unmute ;;
            ?) usage 1 "invalid option: -$OPTARG" ;;
        esac
    done
    shift $((OPTIND - 1))
    local pid=$(pidof "$1")
    if [[ -z "$pid" ]]; then
        echo "error: no running processes for: $1" >&2
    elif [[ "$1" ]]; then
        $action "$1"
    else
        usage 1 "specify an application name" 
    fi
}

usage() {
    [[ "$2" ]] && echo "error: $2"
    echo "Usage: $0 [-m | -u] appname"
    echo "Default: toggle mute"
    echo "Arguments:"
    echo "-m = mute application"
    echo "-u = unmute application"
    exit $1
}

toggle() {
    local status=$(get_status "$1")
    if [[ "$status" == "yes" ]]; then
      unmute "$1"
    elif [[ "$status" == "no" ]]; then
      mute "$1"
    fi
}

mute()   { adjust_muteness "$1" 1; }
unmute() { adjust_muteness "$1" 0; }

adjust_muteness() { 
    local index=$(get_index "$1")
    local status=$(get_status "$1")
    [[ "$index" ]] && pacmd set-sink-input-mute "$index" $2 >/dev/null 
}

get_index() {
    local pid=$(pidof "$1")
    pacmd list-sink-inputs | 
    awk -v pid=$pid '
    $1 == "index:" {idx = $2} 
    $1 == "application.process.id" && $3 == "\"" pid "\"" {print idx; exit}
    '
}

get_status() {
   local pid=$(pidof "$1")
   pacmd list-sink-inputs | 
   awk -v pid=$pid '
   $1 == "muted:" {idx = $2} 
   $1 == "application.process.id" && $3 == "\"" pid "\"" {print idx; exit}
   '
}

main "$@"

ทำไมไม่ใช้pactl list sink-inputs? จากนั้นมันจะทำงานผ่านเครือข่าย
Janus Troelsen

คำตอบ:


13

นี่คือความท้าทายที่คุณสนใจ:

#!/bin/bash

main() {
    local action=mute
    while getopts :hu option; do 
        case "$option" in 
            h) usage 0 ;;
            u) action=unmute ;;
            ?) usage 1 "invalid option: -$OPTARG" ;;
        esac
    done
    shift $((OPTIND - 1))

    if [[ "$1" ]]; then
        $action "$1"
    else
        usage 1 "specify an application name" 
    fi
}

usage() {
    [[ "$2" ]] && echo "error: $2"
    echo "usage: $0 [-h] [-u] appname"
    echo "where: -u = ummute application (default action is to mute)"
    exit $1
}

mute()   { adjust_muteness "$1" 1; }
unmute() { adjust_muteness "$1" 0; }

adjust_muteness() { 
    local index=$(get_index "$1")
    [[ "$index" ]] && pacmd set-sink-input-mute "$index" $2 >/dev/null 
}

get_index() {
    local pid=$(pidof "$1")
    if [[ -z "$pid" ]]; then
        echo "error: no running processes for: $1" >&2
    else
        pacmd list-sink-inputs | 
        awk -v pid=$pid '
            $1 == "index:" {idx = $2} 
            $1 == "application.process.id" && $3 == "\"" pid "\"" {print idx; exit}
        '
    fi
}

main "$@"

มันทำงานได้อย่างสมบูรณ์เช่นกัน
era878

@ era878 ฉันชอบความคิดในการสลับเป็นการกระทำเริ่มต้น อย่างไรก็ตามget_statusฟังก์ชั่นของคุณจะค้นหาบรรทัด "ปิดเสียง" โดยไม่ต้องตรวจสอบข้ามว่าสถานะเป็นของแอปพลิเคชันที่เหมาะสม อ่านget_indexฟังก์ชั่นของฉันอีกครั้งเพื่อดูรายละเอียด
เกล็นแจ็

3
ทักษะ awk ที่ดี :)
hytromo

@glennjackman, ยุพฉันคิดออกหลังจากนั้นไม่นาน ฉันเชื่อว่าสคริปต์ที่ฉันเพิ่งโพสต์นั้นทำงานได้อย่างถูกต้องแล้ว
era878

1
รายละเอียด: awk -v var=val. Awk วนรอบบรรทัด 1 ต่อ 1 พยายามจับคู่$1 == ...คำสั่งใด ๆเรียกใช้งานโค้ดในวงเล็บเหลี่ยมในการจับคู่และดำเนินการต่อ คำสั่งแรกตรงกับบรรทัดที่มีคำแรกคือindex:และเก็บคำที่สอง (SINK INDEX) ในidxตัวแปร ดังนั้นidxได้รับการเขียนทับโดยindex: <SINK INDEX>บรรทัดถัดไปจนกว่า awk ตรงกับคำสั่งที่สอง ( $1= application.process.id, $2= =, $3= <expected pid val>) เมื่อคำสั่งที่ 2 ตรงกับ awk จะพิมพ์idx(ซึ่งเป็นบรรทัดสุดท้ายที่ตรงกับคำสั่งแรกindex:) และออก
KrisWebDev

7

ขอบคุณสำหรับการแก้ปัญหา! ฉันจัดการเพื่อใช้สคริปต์ที่มีให้ที่นี่เพื่อแก้ไขปัญหาของฉัน เนื่องจากฉันต้องแก้ไขพวกเขาเล็กน้อยที่นี่ฉันเข้าร่วมรุ่นที่ปรับปรุงแล้ว

เหตุผลที่สคริปต์ต้นฉบับไม่ทำงานสำหรับฉันเพราะบางแอปพลิเคชันอาจมีหลายกรณีเช่น PID หลายตัว แต่อาจมีเพียงหนึ่งรายการเท่านั้นที่สร้างเสียงและจริง ๆ แล้วเชื่อมต่อกับ Pulseaudio เนื่องจากสคริปต์ใช้ PID แรกที่พบเท่านั้นโดยทั่วไปจะเป็น / ไม่ / ปิดเสียงแอปพลิเคชันที่ต้องการ

ดังนั้นนี่คือรุ่นที่อาร์กิวเมนต์เป็นชื่อแอปพลิเคชันที่ลงทะเบียนใน PulseAudio คุณสามารถค้นหาชื่อนี้ได้โดยเรียกใช้pacmd list-sink-inputsคำสั่งและค้นหาapplication.nameฟิลด์

โซลูชันทางเลือกคือปิด / เปิดเสียง PID ทั้งหมดที่มีชื่อแอปพลิเคชันเดียวกัน

#!/bin/bash

# Adapter from glenn jackman on http://askubuntu.com/questions/180612/script-to-mute-an-application
# to depend directly on the name of the PulseAudio client
# rather than the application name (several instances of one application could
# run while only one is connected to PulseAudio)

# Possible further improvement: it could be useful to also mute all clients having
# the specified name. Here, only the first one is muted.

#!/bin/bash

main() {
    local action=mute
    while getopts :hu option; do
        case "$option" in
            h) usage 0 ;;
            u) action=unmute ;;
            ?) usage 1 "invalid option: -$OPTARG" ;;
        esac
    done
    shift $((OPTIND - 1))

    if [[ "$1" ]]; then
        $action "$1"
    else
        usage 1 "specify the name of a PulseAudio client"
    fi
}

usage() {
    [[ "$2" ]] && echo "error: $2"
    echo "usage: $0 [-h] [-u] appname"
    echo "where: -u = ummute application (default action is to mute)"
    exit $1
}

mute()   { adjust_muteness "$1" 1; }
unmute() { adjust_muteness "$1" 0; }

adjust_muteness() {
    local index=$(get_index "$1")
    if [[ -z "$index" ]]; then
        echo "error: no PulseAudio sink named $1 was found" >&2
    else
        [[ "$index" ]] && pacmd set-sink-input-mute "$index" $2 >/dev/null
    fi
}

get_index() {
#    local pid=$(pidof "$1")
#    if [[ -z "$pid" ]]; then
#        echo "error: no running processes for: $1" >&2
#    else
        pacmd list-sink-inputs |
        awk -v name=$1 '
            $1 == "index:" {idx = $2}
            $1 == "application.name" && $3 == "\"" name "\"" {print idx; exit}
        '
#    fi
}

main "$@"

6

แม้ว่าคำถามจะขอสคริปต์ แต่ฉันต้องการออกจากที่นี่

ฉันได้เขียนแอปพลิเคชัน C ที่ทำสิ่งนี้บน Ubuntu ยิ่งไปกว่านั้นมันตั้งอยู่บนถาดตัวบ่งชี้ (โดยใช้libappindicator) และตรวจสอบสิ่งที่ Spotify กำลังเล่นในช่วงเวลาสั้น ๆ หากกำลังเล่นโฆษณา (ตรวจสอบกับบัญชีดำ) จะปิดเสียง Spotify หากมีการเล่นโฆษณาใหม่คุณเพียงคลิกปิดเสียงในเมนูตัวบ่งชี้และเพิ่มลงในบัญชีดำ

สิ่งที่มันไม่เป็นรูปลักษณ์สำหรับหน้าต่าง X ซึ่งผลตอบแทนXFetchName Spotify - Linux Previewจากนั้นจะเรียกXGetWindowPropertyเพื่อสอบถาม_NET_WM_ICON_NAMEคุณสมบัติของหน้าต่างนั้นซึ่งส่งคืนสตริงใน"Spotify – <Artist> – <Song>"รูปแบบ เมื่อเล่นโฆษณามันจะแสดงผลดังนี้:

"Spotify – Spotify – Premium Free Trial Cancel Any Time"

มันรักษาโครงสร้างการค้นหา Ternaryของรายการโฆษณาเพื่อตรวจสอบอย่างมีประสิทธิภาพว่าชื่อปัจจุบันอยู่ในรายการ

นอกจากนี้ยังใช้PulseAudio Asynchronous APIเพื่อสอบถามsink-inputsและset-mute:

pa_context_get_sink_input_info_list()
pa_context_set_sink_input_mute()

เนื่องจากมันเป็นรหัส C ง่าย ๆ มันมีน้ำหนักเบา ตรวจสอบรหัสที่มาและ Ubuntu .debแพคเกจที่: บ่งชี้ muteads มันอาจเอาชนะเชลล์สคริปต์ได้ขนาด 2-3 ออเดอร์


ไม่ทำงานกับรุ่น 1.0.11
Janus Troelsen

4

ก่อนอื่นวิธีที่ "ถูกต้องมากกว่า" ในการค้นหา PID ของแอปพลิเคชันเช่น spotify คือการใช้:

pidof spotify

ฉันได้สร้างสคริปต์ที่ทำงานได้แล้วฉันไม่รู้ว่ามันเป็นวิธีที่ดีที่สุดในการทำงาน แต่มันทำงานได้อย่างสมบูรณ์แบบ:

#!/bin/bash
# Script to mute an application using PulseAudio, depending solely on
# process name, constructed as answer on askubuntu.com: 
# http://askubuntu.com/questions/180612/script-to-mute-an-application

#It works as: mute_application.sh vlc mute OR mute_application.sh vlc unmute

if [ -z "$1" ]; then
   echo "Please provide me with an application name"
   exit 1
fi

if [ -z "$2" ]; then
   echo "Please provide me with an action mute/unmute after the application name"
   exit 1
fi

if ! [[ "$2" == "mute" || "$2" == "unmute" ]]; then
   echo "The 2nd argument must be mute/unmute"
   exit 1
fi

process_id=$(pidof "$1")

if [ $? -ne 0 ]; then
   echo "There is no such process as "$1""
   exit 1
fi

temp=$(mktemp)

pacmd list-sink-inputs > $temp

inputs_found=0;
current_index=-1;

while read line; do
   if [ $inputs_found -eq 0 ]; then
      inputs=$(echo -ne "$line" | awk '{print $2}')
      if [[ "$inputs" == "to" ]]; then
         continue
      fi
      inputs_found=1
   else
      if [[ "${line:0:6}" == "index:" ]]; then
         current_index="${line:7}"
      elif [[ "${line:0:25}" == "application.process.id = " ]]; then
         if [[ "${line:25}" == "\"$process_id\"" ]]; then
            #index found...
            break;
         fi
      fi
   fi
done < $temp

rm -f $temp

if [ $current_index -eq -1 ]; then
   echo "Could not find "$1" in the processes that output sound."
   exit 1
fi

#muting...
if [[ "$2" == "mute" ]]; then
   pacmd set-sink-input-mute "$current_index" 1 > /dev/null 2>&1
else
   pacmd set-sink-input-mute "$current_index" 0 > /dev/null 2>&1
fi

exit 0

คุณสามารถทำงานกับเป็นดังนี้:

./mute_application.sh spotify mute

หรือ

./mute_application.sh spotify unmute

ทดสอบกับทั้ง Audacious และ Vlc ที่ใช้งานอยู่และปิด / ปิดเสียงเพียงหนึ่งเดียว


สคริปต์ที่สมบูรณ์แบบทำงานได้ตามที่คาดไว้
era878

1

ฉันไม่สามารถเขียนสคริปต์ได้ แต่ฉันได้แก้ไขสคริปต์ของ hakermania เพื่อสร้างสคริปต์ใหม่

อันนี้จะเพิ่มหรือลดปริมาณของแอปพลิเคชันเฉพาะที่เพิ่มขึ้น 5%:

แก้ไข: อันที่จริงมันทำงานในการเปลี่ยนแปลงแอปที่เปิดล่าสุดเสมอ ไอเดีย?

#!/bin/bash
# Script to increase or decrease an individual application's volume using PulseAudio, depending solely on
# process name, based on another script by hakermania, constructed as answer on askubuntu.com: 
# http://askubuntu.com/questions/180612/script-to-mute-an-application

# It works as: change_app_volume.sh vlc increase OR change_app_volume.sh vlc decrease
# Set desired increments in lines #66 and #68

if [ -z "$1" ]; then
   echo "Please provide me with an application name"
   exit 1
fi

if [ -z "$2" ]; then
   echo "Please provide me with an action increase/decrease after the application name"
   exit 1
fi

if ! [[ "$2" == "increase" || "$2" == "decrease" ]]; then
   echo "The 2nd argument must be increase/decrease"
   exit 1
fi

process_id=$(pidof "$1")

if [ $? -ne 0 ]; then
   echo "There is no such process as "$1""
   exit 1
fi

temp=$(mktemp)

pacmd list-sink-inputs > $temp

inputs_found=0;
current_index=-1;

while read line; do
   if [ $inputs_found -eq 0 ]; then
      inputs=$(echo -ne "$line" | awk '{print $2}')
      if [[ "$inputs" == "to" ]]; then
         continue
      fi
      inputs_found=1
   else
      if [[ "${line:0:6}" == "index:" ]]; then
         current_index="${line:7}"
      elif [[ "${line:0:25}" == "application.process.id = " ]]; then
         if [[ "${line:25}" == "\"$process_id\"" ]]; then
            #index found...
            break;
         fi
      fi
   fi
done < $temp

rm -f $temp

if [ $current_index -eq -1 ]; then
   echo "Could not find "$1" in the processes that output sound."
   exit 1
fi

#increase/decrease...
if [[ "$2" == "increase" ]]; then
   pactl set-sink-input-volume "$current_index" +5% > /dev/null 2>&1
else
   pactl set-sink-input-volume "$current_index" -5% > /dev/null 2>&1
fi

exit 0

0

สคริปต์ที่แก้ไขเพื่อปิดเสียงอินพุตทั้งหมดของแอพ (หลายกระบวนการ) และค่าเริ่มต้นเป็นสลับ:

#!/bin/bash

main() {
    local action=toggle
    while getopts :hu option; do
        case "$option" in
            h) usage 0 ;;
            m) action=mute ;;
            u) action=unmute ;;
            ?) usage 1 "invalid option: -$OPTARG" ;;
        esac
    done
    shift $((OPTIND - 1))

    if [[ "$1" ]]; then
        $action "$1"
    else
        usage 1 "specify an application name"
    fi
}

usage() {
    [[ "$2" ]] && echo "error: $2"
    echo "usage: $0 [-h] [-u] appname"
    echo "where: -u = ummute , -m = mute (default action is to toggle)"
    exit $1
}

mute()   { adjust_muteness "$1" 1; }
unmute() { adjust_muteness "$1" 0; }
toggle() { adjust_muteness "$1" toggle; }

adjust_muteness() {
    clients=$(pactl list clients short | awk '/[0-9]+.*'$1'.*/{print $1}')
    inputs=$(pactl list sink-inputs short)
    for c in $clients; do
        for i in $(printf '%s' "$inputs" | awk '/[0-9]+\s[0-9]+\s'$c'/{print $1}'); do
            pactl set-sink-input-mute $i $2 &
        done
    done
}

main "$@"
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.