python实现音视频下载器

news2025/4/19 3:33:22

一、环境准备

确保当前系统已安装了wxPython 、 yt-dlp 和FFmpeg。当前主要支持下载youtube音视频

1、安装wxPython

pip install wxPython

2、安装yt-dp

pip install wxPython yt-dlp

3、安装FFmpeg

在Windows 10上通过命令行安装FFmpeg,最简便的方式是使用包管理器,比如Scoop或Chocolatey。以下是使用这两种工具的步骤:

① 使用Scoop安装FFmpeg

如果还没有安装Scoop,首先需要按照Scoop官网上的说明进行安装。打开PowerShell并运行以下命令来安装Scoop(如果尚未安装):

Set-ExecutionPolicy RemoteSigned -scope CurrentUser
irm get.scoop.sh | iex

安装完Scoop后,你可以通过以下命令安装FFmpeg:

scoop install ffmpeg

② 使用Chocolatey安装FFmpeg

如果更倾向于使用Chocolatey,首先确保已经安装了Chocolatey。可以通过Chocolatey官网获取安装指南。
在这里插入图片描述

通常,可以在管理员模式下的PowerShell中运行以下命令来安装Chocolatey:

Set-ExecutionPolicy Bypass -Scope Process -Force; [System.Net.ServicePointManager]::SecurityProtocol = [System.Net.ServicePointManager]::SecurityProtocol -bor 3072; iex ((New-Object System.Net.WebClient).DownloadString('https://community.chocolatey.org/install.ps1'))

在这里插入图片描述

安装完Chocolatey后,可以通过以下命令安装FFmpeg:

choco install ffmpeg

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
无论你选择哪种方式安装FFmpeg,完成后都可以通过命令行输入以下命令来验证是否安装成功:

ffmpeg -version

在这里插入图片描述

如果你看到FFmpeg的版本信息以及其他相关细节,那么说明FFmpeg已经正确安装,并且可以通过命令行使用了。

请注意,为了使这些命令在任何目录下都能生效,可能需要重启计算机或者重新打开一个新的命令提示符窗口。此外,确保你的环境变量Path包含了FFmpeg的安装路径,这样系统才能识别ffmpeg命令。通过上述方法安装时,通常会自动处理好这个问题。

二、代码编写

编写名字为downloader.py的python代码文件

import wx
import yt_dlp
import threading

class DownloaderFrame(wx.Frame):
    def __init__(self):
        super().__init__(None, title="Downloader", size=(400, 350))

        self.panel = wx.Panel(self)

        self.source_label = wx.StaticText(self.panel, label="Audio or video URL:")
        self.source_text = wx.TextCtrl(self.panel)
        self.source_label.SetBackgroundColour(wx.Colour(173, 216, 230))  # 设置背景为浅蓝色
        self.source_label.SetForegroundColour(wx.Colour(0, 0, 128))  # 设置文本颜色为黑色,确保对比度良好

        self.folder_label = wx.StaticText(self.panel, label="Save Folder:")
        self.folder_text = wx.TextCtrl(self.panel)
        self.folder_label.SetBackgroundColour(wx.Colour(173, 216, 230))  # 设置背景为浅蓝色
        self.folder_label.SetForegroundColour(wx.Colour(0, 0, 128))  # 设置文本颜色为黑色,确保对比度良好
        self.folder_button = wx.Button(self.panel, label="Browse...")
        self.folder_button.Bind(wx.EVT_BUTTON, self.browse_folder)

        self.quality_label = wx.StaticText(self.panel, label="Quality:")
        self.quality_choice = wx.ComboBox(self.panel, choices=[], style=wx.CB_READONLY)
        self.quality_label.SetBackgroundColour(wx.Colour(173, 216, 230))  # 设置背景为浅蓝色
        self.quality_label.SetForegroundColour(wx.Colour(0, 0, 128))  # 设置文本颜色为黑色,确保对比度良好

        self.start_button = wx.Button(self.panel, label="Start Download")
        self.start_button.Bind(wx.EVT_BUTTON, self.start)

        self.process_label = wx.StaticText(self.panel, label="Percentage:")
        self.process_bar = wx.Gauge(self.panel, range=100)
        self.process_label.SetBackgroundColour(wx.Colour(173, 216, 230))  # 设置背景为浅蓝色
        self.process_label.SetForegroundColour(wx.Colour(0, 0, 128))  # 设置文本颜色为黑色,确保对比度良好

        self.sizer = wx.BoxSizer(wx.VERTICAL)
        self.sizer.Add(self.source_label, 0, wx.ALL, 5)
        self.sizer.Add(self.source_text, 0, wx.EXPAND | wx.ALL, 5)
        self.sizer.Add(self.folder_label, 0, wx.ALL, 5)
        self.sizer.Add(self.folder_text, 0, wx.EXPAND | wx.ALL, 5)
        self.sizer.Add(self.folder_button, 0, wx.ALL, 5)
        self.sizer.Add(self.quality_label, 0, wx.ALL, 5)
        self.sizer.Add(self.quality_choice, 0, wx.EXPAND | wx.ALL, 5)
        self.sizer.Add(self.start_button, 0, wx.ALL, 5)
        self.sizer.Add(self.process_label, 0, wx.ALL, 5)
        self.sizer.Add(self.process_bar, 0, wx.EXPAND | wx.ALL, 5)

        self.panel.SetSizer(self.sizer)
        self.Show()

    def browse_folder(self, event):
        dlg = wx.DirDialog(self, "Choose a directory", style=wx.DD_DEFAULT_STYLE)
        if dlg.ShowModal() == wx.ID_OK:
            self.folder_text.SetValue(dlg.GetPath())
        dlg.Destroy()

    def get_video_formats(self, url):
        with yt_dlp.YoutubeDL({}) as ydl:
            info_dict = ydl.extract_info(url, download=False)
            #print(info_dict)  # 添加此行来查看所有可用格式
            formats = info_dict.get('formats', [])
            #print("Available formats:", formats)  # 打印所有可用格式以供调试

            self.quality_choice.Clear()
            best_format_id = None  # 初始化为None
            has_1080p = False

            # 首先尝试找到1080p或更高的视频格式
            for f in formats:
                if 'height' in f and 'format_id' in f and f['vcodec'] != 'none' and f['acodec'] != 'none':
                    display_str = f"{f['format_id']} - {f['height']}p"
                    if f['height'] >= 1080:  # 检查是否为1080p或更高
                        if not has_1080p or f['height'] > int(best_format_id.split(" - ")[1][:-1]):
                            best_format_id = display_str
                            has_1080p = True

            # 如果没有找到1080p或更高的格式,则选择其他可用的最高分辨率
            if not has_1080p:
                for f in formats:
                    if 'height' in f and 'format_id' in f and f['vcodec'] != 'none' and f['acodec'] != 'none':
                        display_str = f"{f['format_id']} - {f['height']}p"
                        if best_format_id is None or (
                                'height' in f and f['height'] > int(best_format_id.split(" - ")[1][:-1])):
                            best_format_id = display_str

            # 添加所有满足条件的格式到下拉列表中供用户选择
            for f in formats:
                if 'height' in f and 'format_id' in f and f['vcodec'] != 'none' and f['acodec'] != 'none':
                    self.quality_choice.Append(f"{f['format_id']} - {f['height']}p")

            if self.quality_choice.GetCount() > 0:
                self.quality_choice.SetSelection(0)
            return best_format_id

    def start(self, event):
        url = self.source_text.GetValue()
        save_folder = self.folder_text.GetValue()

        # 获取并填充清晰度选项
        best_format = self.get_video_formats(url)
        #print(best_format)

        if not best_format:  # 如果没有找到任何格式
            wx.MessageBox("No suitable video format found.", "Error", wx.OK | wx.ICON_ERROR)
            return

        download_thread = threading.Thread(target=self.download_Audio_or_video, args=(url, save_folder, best_format))
        download_thread.start()

    def download_Audio_or_video(self, url, save_folder, best_format):
        selected_format = self.quality_choice.GetStringSelection()
        format_id = selected_format.split(" - ")[0] if selected_format else best_format.split(" - ")[0]  # 提取format id

        options = {
            'outtmpl': f'{save_folder}/%(title)s.%(ext)s',
            #'format': 'best',  # 使用最佳质量
            'format': format_id,
            'progress_hooks': [self.update_progress],
        }
        try:
            with yt_dlp.YoutubeDL(options) as ydl:
                ydl.download([url])
            wx.CallAfter(wx.MessageBox, 'Download completed!', 'Info', wx.OK | wx.ICON_INFORMATION)
        except Exception as e:
            wx.CallAfter(wx.MessageBox, f'Download error occurred: {str(e)}', 'Error', wx.OK | wx.ICON_ERROR)

    def update_progress(self, progress ):
        if 'total_bytes' in progress and 'downloaded_bytes' in progress:
            percentage = int(progress['downloaded_bytes'] * 100 / progress['total_bytes'])
            wx.CallAfter(self.process_bar.SetValue, percentage)


app = wx.App(False)
frame = DownloaderFrame()
app.MainLoop()

三、代码运行测试

1.命令行运行测试

1.python downloader.py
在这里插入图片描述

三、其他方式

或者直接利用exe下载工具来进行命令行下载
在这里插入图片描述
参考文件:
https://blog.csdn.net/qq_31339083/article/details/132195733
https://blog.csdn.net/winniezhang/article/details/132127382

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2337684.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

三、小白如何用Pygame制作一款跑酷类游戏(按键图片和距离的计算)

三、小白如何用Pygame制作一款跑酷类游戏(实现移动距离的计算,以及按键指引的添加) 文章目录 三、小白如何用Pygame制作一款跑酷类游戏(实现移动距离的计算,以及按键指引的添加)前言一、创建字体文件夹1.可…

H5:实现安卓和苹果点击下载App自动跳转到对应的应用市场

一、需求场景 手机扫描下载App,需要根据不同手机自动跳转到对应的应用市场(商店)里,苹果手机直接打开App Store里指定的app页面,安卓手机如果是海外用户则打开GooglePlay 商店里指定的app页面,国内直接下载…

【Linux】文件传输归档与压缩

目录 配置实验环境 文件传输方法--scp,rsync scp rsync 归档与压缩--tar,gz,bz2,xz,zip 归档---tar 压缩 zip gzip bzip2 xz 归档并压缩 gz bz2 xz 拓展du 配置实验环境 在多个linux系统进行系统传输…

3D人脸扫描技术如何让真人“进入“虚拟,虚拟数字人反向“激活“现实?

随着虚拟人技术的飞速发展,超写实数字人已经成为数字娱乐、广告营销和虚拟互动领域的核心趋势。无论是企业家、知名主持人还是明星,数字分身正在以高度还原的形象替代真人参与各类活动,甚至成为品牌代言、直播互动的新宠。 3D人脸扫描&#…

Git标签的认识

Git标签完全指南:从基础到企业级发布策略 前言 在软件发布领域,Git标签是版本管理的基石。根据2023年GitHub年度报告显示,85%的开源项目使用标签进行版本控制。然而,许多开发者仅停留在git tag的基础使用层面,未能充分…

【Rust基础】使用Rocket构建基于SSE的流式回复

背景 我们正在使用Rust开发基于RAG的知识库系统,其中对于模型的回复使用了常用的SSE,Web框架使用Rocket,Rocket提供了一个简单的方式支持SSE,但没有会话保持、会话恢复等功能,因此我们自己简单实现这两个功能。 使用R…

大前端基础学习

一、cs架构和bs架构 c:客户端, b:浏览器(无需安装,无需更新,可跨平台)√ s:server服务端,帮我们保 存信息,传递信息 二、 altshift向下键向下复制一行 …

Axios 的 POST 请求:QS 处理数据的奥秘与使用场景解析

在现代前端开发中,Axios 已经成为了进行 HTTP 请求的首选库之一,它的简洁易用和强大功能深受开发者喜爱。当使用 Axios 进行 POST 请求时,我们常常会遇到一个问题:是否需要使用 QS 库来处理请求数据?什么时候又可以不用…

Linux 防火墙( iptables )

目录 一、 Linux 防火墙基础 1. 防火墙基础概念 (1)防火墙的概述与作用 (2)防火墙的结构与匹配流程 (3)防火墙的类别与各个防火墙的区别 2. iptables 的表、链结构 (1)规则表 …

【redis进阶三】分布式系统之主从复制结构(1)

目录 一 为什么要有分布式系统? 二 分布式系统涉及到的非常关键的问题:单点问题 三 学习部署主从结构的redis (1)创建一个目录 (2)进入目录拷贝两份原有redis (3)使用vim修改几个选项 (4)启动两个从节点服务器 (5)建立复制,要想配…

EM储能网关ZWS智慧储能云应用(9) — 远程OTA升级

ZWS智慧储能云平台支持远程OTA固件升级,可以针对具体的储能设备进行升级,升级储能网关、EMS主控软件、PCS、BMS等。 简介 储能系统通常高度集成化,一体化设计,将EMS、BMS(电池管理系统)、PCS&#xff08…

ubuntu24.04LTS安装向日葵解决方案

去向日葵官方下载ubuntu使用的deb包 向日葵 输入如下命令安装,将具体版本修改成自己下载的版本 andrew in ~/下载 λ sudo dpkg -i SunloginClient_15.2.0.63064_amd64.deb 正在选中未选择的软件包 sunloginclient。 (正在读取数据库 ... 系统当前共安装有 290947…

达梦官方管理工具SQLark:自动识别外键约束、check约束与虚拟列,助力高效生成测试数据

在数据库管理和应用开发过程中,高质量的测试数据对于系统调试和POC测试至关重要。达梦官方推出的新一代管理工具 SQLark百灵连接,其数据生成功能,可以为应用开发者、DBA 以及测试人员带来极大便利,能够轻松应对各类复杂的测试场景…

不关“猫”如何改变外网IP?3种免重启切换IP方案

每次更换外网IP都要重启路由器?太麻烦了!那么,不关猫怎么改变外网IP?无论是为了网络调试、爬虫需求,还是解决IP限制问题,频繁重启设备既耗时又影响效率。其实,更换外网IP并不一定要依赖“重启大…

C#进阶学习(五)单向链表和双向链表,循环链表(中)双向链表

目录 一、双向链表的声明 1. 节点类声明 2. 链表类框架声明 3、实现其中的每一个函数 增删操作(核心方法组) 删除操作(核心方法组) 查询操作(辅助方法) 维护方法(内部逻辑) …

重学Redis:Redis常用数据类型+存储结构(源码篇)

一、SDS 1,SDS源码解读 sds (Simple Dynamic String),Simple的意思是简单,Dynamic即动态,意味着其具有动态增加空间的能力,扩容不需要使用者关心。String是字符串的意思。说白了就是用C语言自己封装了一个字符串类型&a…

js原型和原型链

js原型: 1、原型诞生的目的是什么呢? js原型的产生是为了解决在js对象实例之间共享属性和方法,并把他们很好聚集在一起(原型对象上)。每个函数都会创建一个prototype属性,这个属性指向的就是原型对象。 …

OpenHarmony - 小型系统内核(LiteOS-A)(五)

OpenHarmony - 小型系统内核(LiteOS-A)(五) 六、文件系统 虚拟文件系统 基本概念 VFS(Virtual File System)是文件系统的虚拟层,它不是一个实际的文件系统,而是一个异构文件系统之…

PyTorch进阶学习笔记[长期更新]

第一章 PyTorch简介和安装 PyTorch是一个很强大的深度学习库,在学术中使用占比很大。 我这里是Mac系统的安装,相比起教程中的win/linux安装感觉还是简单不少(之前就已经安好啦),有需要指导的小伙伴可以评论。 第二章…

proteus8.17 环境配置

Proteus介绍 Proteus 8.17 是一款功能强大的电子设计自动化(EDA)软件,广泛应用于电子电路设计、仿真和分析。以下是其主要特点和新功能: ### 主要功能 - **电路仿真**:支持数字和模拟电路的仿真,包括静态…