如何使用 python 在 Windows 10 中获取当前正在播放的媒体的标题

Ale*_*der 5 python audio windows-10

每当在 Windows 10 中播放音频时,无论是来自 Spotify、Firefox 还是游戏。当你调大音量时,windows 的角落里会有一个东西,上面写着歌曲艺术家、标题和正在播放的应用程序,如下图所示(有时它只会说如果游戏正在播放声音,它就会显示哪个应用程序正在播放声音)

在此处输入图片说明

我想以某种方式用 python 获取这些数据。我的最终目标是将正在播放我不喜欢的内容(例如广告)的应用程序静音。

tam*_*TNT 10

事实证明,在没有解决方法的情况,通过使用Windows 运行时 API ( winrt)直接访问此信息,这是可能的。

显示的所有代码都使用 Python 3 和winrt通过 pip 安装的库

收集媒体/“正在播放”信息

以下代码允许您使用 Windows 运行时 API 的 winrt 包装器收集 Windows 可用的媒体信息字典。它不依赖于窗口标题/应用程序名称的更改,就像这里的其他答案一样。

import asyncio

from winrt.windows.media.control import \
    GlobalSystemMediaTransportControlsSessionManager as MediaManager


async def get_media_info():
    sessions = await MediaManager.request_async()

    # This source_app_user_model_id check and if statement is optional
    # Use it if you want to only get a certain player/program's media
    # (e.g. only chrome.exe's media not any other program's).

    # To get the ID, use a breakpoint() to run sessions.get_current_session()
    # while the media you want to get is playing.
    # Then set TARGET_ID to the string this call returns.

    current_session = sessions.get_current_session()
    if current_session:  # there needs to be a media session running
        if current_session.source_app_user_model_id == TARGET_ID:
            info = await current_session.try_get_media_properties_async()

            # song_attr[0] != '_' ignores system attributes
            info_dict = {song_attr: info.__getattribute__(song_attr) for song_attr in dir(info) if song_attr[0] != '_'}

            # converts winrt vector to list
            info_dict['genres'] = list(info_dict['genres'])

            return info_dict

    # It could be possible to select a program from a list of current
    # available ones. I just haven't implemented this here for my use case.
    # See references for more information.
    raise Exception('TARGET_PROGRAM is not the current media session')


if __name__ == '__main__':
    current_media_info = asyncio.run(get_media_info())
Run Code Online (Sandbox Code Playgroud)

current_media_info 将是以下格式的字典,然后可以根据需要在程序中访问信息:

{
    'album_artist': str,
    'album_title': str,
    'album_track_count': int, 
    'artist': str,
    'genres': list,
    'playback_type': int,
    'subtitle': str, 
    'thumbnail': 
        <_winrt_Windows_Storage_Streams.IRandomAccessStreamReference object at ?>, 
    'title': str,
    'track_number': int,
}
Run Code Online (Sandbox Code Playgroud)

控制媒体

正如 OP 所说,他们的最终目标是控制媒体,这应该可以通过相同的库实现。请参阅此处以获取更多信息(在我的情况下我不需要这个):

(获取媒体缩略图)

事实上,也可以“刮取”当前正在播放的媒体的专辑封面/媒体缩略图(显示在 OP 屏幕截图的右侧)(尽管 OP 没有要求这样做,但有人可能想要这样做):

from winrt.windows.storage.streams import \
    DataReader, Buffer, InputStreamOptions


async def read_stream_into_buffer(stream_ref, buffer):
    readable_stream = await stream_ref.open_read_async()
    readable_stream.read_async(buffer, buffer.capacity, InputStreamOptions.READ_AHEAD)


# create the current_media_info dict with the earlier code first
thumb_stream_ref = current_media_info['thumbnail']

# 5MB (5 million byte) buffer - thumbnail unlikely to be larger
thumb_read_buffer = Buffer(5000000)

# copies data from data stream reference into buffer created above
asyncio.run(read_stream_into_buffer(thumb_stream_ref, thumb_read_buffer))

# reads data (as bytes) from buffer
buffer_reader = DataReader.from_buffer(thumb_read_buffer)
byte_buffer = buffer_reader.read_bytes(thumb_read_buffer.length)

with open('media_thumb.jpg', 'wb+') as fobj:
    fobj.write(bytearray(byte_buffer))
Run Code Online (Sandbox Code Playgroud)

这会将 a 保存media_thumb.jpg到当前工作目录 (cwd),然后可以在其他地方使用它。

文档和参考:

可能从多个可用媒体流中进行选择?

请注意,我还没有测试或尝试过这个,这只是给可能想要尝试的任何人的一个指针:

与目前使用的相反

  • 根据此处的答案,自 python 3.10.0 起,您必须将“winrt”模块替换为“winsdk”:/sf/ask/4872716201/。此外,应该选择这个答案。受访者正在回答原来的问题,使用Windows RunTime API,而不是赤裸裸地使用Win32 API,来捕获正在运行的播放器的窗口标题。 (3认同)

Ale*_*der 3

我正在获取窗口的标题来获取歌曲信息。通常,应用程序名称显示在标题中,但当播放歌曲时,会显示歌曲名称。这是一个返回所有窗口标题列表的函数。

from __future__ import print_function
import ctypes
def get_titles(): 
    EnumWindows = ctypes.windll.user32.EnumWindows
    EnumWindowsProc = ctypes.WINFUNCTYPE(ctypes.c_bool, ctypes.POINTER(ctypes.c_int), ctypes.POINTER(ctypes.c_int))
    GetWindowText = ctypes.windll.user32.GetWindowTextW
    GetWindowTextLength = ctypes.windll.user32.GetWindowTextLengthW
    IsWindowVisible = ctypes.windll.user32.IsWindowVisible
    
    titles = []
    def foreach_window(hwnd, lParam):
        if IsWindowVisible(hwnd):
            length = GetWindowTextLength(hwnd)
            buff = ctypes.create_unicode_buffer(length + 1)
            GetWindowText(hwnd, buff, length + 1)
            titles.append(buff.value)
        return True
    EnumWindows(EnumWindowsProc(foreach_window), 0)
    return titles
Run Code Online (Sandbox Code Playgroud)