用Qt写一个简单的音乐播放器(四):歌曲浏览、上一曲、下一曲

本文详细介绍如何使用Qt创建音乐播放器,涵盖歌曲选择、播放控制、音量调节及上一曲/下一曲功能实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、前言

用Qt写一个简单的音乐播放器(一):使用QMediaPlayer播放音乐中,我们已经知道如何去使用QMediaPlayer播放音乐。
用Qt写一个简单的音乐播放器(二):增加界面(开始和暂停音乐)中,我们增加了播放音乐/暂停音乐的按钮。
用Qt写一个简单的音乐播放器(三):增加界面(播放跳转与音量控制)中,我们加入了播放控制和音量控制。
但是从头到尾,我们一直都只能播放一首歌曲,连选择歌曲都不行,怎么能算是音乐播放器呢?
这一篇就来谈一谈歌曲选择播放的问题。

二、浏览文件夹

1.控件

想要选择歌曲,肯定是要浏览文件夹的,那么首先就要面临的问题就是选择什么控件显示,什么事件触发?
在这里插入图片描述
如上图所示,首先我们选择一个按钮,并将其重命名为btnAddMusic
在这里插入图片描述
如上图所示,这里我选择的是List Widget。将其拖动到设计界面上,然后为了编程方便,重命名为lwMusicList。

2.浏览文件夹

在这里插入图片描述
如上图所示,选择Add按钮,点击右键转到槽,选择Clicked(),点击OK,出现下图:
在这里插入图片描述
在这个槽函数中增加以下代码:

    QString fileName = QFileDialog::getExistingDirectory(NULL,"Select Music Dir",".",NULL);
    QDir dir(fileName);
    QStringList nameFilters;
    nameFilters << "*.mp3";
    QStringList files = dir.entryList(nameFilters, QDir::Files|QDir::Readable, QDir::Name);
    ui->lwMusicList->addItems(files);
    m_playPath = fileName;

这个时候,你如果构建,肯定是失败的,因为头文件和m_playPath还没有声明。所以 在mainwindow.h 中加入以下代码:

#include <QListWidgetItem>
#include <QString>
#include <QFileDialog>
……
QString m_playPath;

这个时候构建就可以添加歌曲文件了。

三、双击播放

在这里插入图片描述
如上图所示,选中控件,右键,转到槽,选择itemDoubleClicked(),点击OK,出现下图:
在这里插入图片描述
如上图所示,在这个槽函数中加入以下代码:

void MainWindow::on_lwMusicList_itemDoubleClicked(QListWidgetItem *item)
{
    m_mediaPlayer.stop();
    m_mediaPlayer.setMedia(QUrl::fromLocalFile(m_playPath+"/"+item->text()));
    m_mediaPlayer.play();
    ui->hSliderPlayProgress->setValue(m_mediaPlayer.position());
    connect(timer, SIGNAL(timeout()), this, SLOT(setSliderValue()))
}

四、Play与Pause

之前已经实现过Play与Pause了,这里做一个小小的改进,将二者合为一个按钮来实现。
删掉与Pause相关的函数以及控件,将play的槽函数修改为以下代码

void MainWindow::on_btnPlay_clicked()
{
    if(QMediaPlayer:: PlayingState == m_mediaPlayer.state()){
        m_mediaPlayer.pause();
        ui->btnPlay->setText("Play");
    }
    else{
        m_mediaPlayer.play();
        ui->btnPlay->setText("Pause");
    }
}

五、上一曲与下一曲

如图所示,首先增加pre和Next两个按钮,分别代码上一曲和下一曲,并且为了编程方便,重命名为btnPreMusic与btnNextMusic。
在这里插入图片描述
分别选中pre和Next按钮,点击右键转到槽,选择Clicked(),点击OK,出现下图所示两个槽函数:
在这里插入图片描述
分别在其中加入以下代码:

void MainWindow::on_btnPreMusic_clicked()
{
    if(m_PlayRow == 0){
        m_PlayRow = ui->lwMusicList->count() - 1;
    }
    else{
        m_PlayRow--;
    }
    QListWidgetItem *item = ui->lwMusicList->item(m_PlayRow);
    item->setSelected(true);
    m_mediaPlayer.stop();
    m_mediaPlayer.setMedia(QUrl::fromLocalFile(m_playPath+"/"+item->text()));
    m_mediaPlayer.play();
}

void MainWindow::on_btnNextMusic_clicked()
{
    if(m_PlayRow + 1 == ui->lwMusicList->count()){
        m_PlayRow = 0;
    }
    else{
        m_PlayRow++;
    }
    QListWidgetItem *item = ui->lwMusicList->item(m_PlayRow);
    item->setSelected(true);
    m_mediaPlayer.stop();
    m_mediaPlayer.setMedia(QUrl::fromLocalFile(m_playPath+"/"+item->text()));
    m_mediaPlayer.play();
}

当然这样还是不行的,还差两行代码,分别是
在播放时初始化m_PlayRow,和在mainwindow.h中声明m_PlayRow。
加上这两行就可以使用了。完整代码,见最后。

 m_PlayRow = ui->lwMusicList->row(item);
 int m_PlayRow;

六、修改音量控制

我们可以看到很多音量控制都是隐藏的,不像我们现在这样一直显示,所以简单的修改一下,增加一个volume按钮,并重命名为btnVolume:
在这里插入图片描述
选择按钮,点击右键转到槽,选择Clicked(),点击OK,出现下图所示槽函数:
在这里插入图片描述
首先我们在初始化的时候,也就是构造函数中加入如下代码,使vSliderVolume默认不可见。

ui->vSliderVolume->setVisible(false);

然后在槽函数中加入以下代码:

void MainWindow::on_btnVolume_clicked()
{
    if(ui->vSliderVolume->isVisible()){
        ui->vSliderVolume->setVisible(false);
    }else{
        ui->vSliderVolume->setVisible(true);
    }
}

再加一个,鼠标离开滑块,自动隐藏的功能:
在这里插入图片描述
在生成的对应槽函数加入以下代码:

void MainWindow::on_vSliderVolume_sliderReleased()
{
     ui->vSliderVolume->setVisible(false);
}

七、效果

打开界面:
在这里插入图片描述
点击Add,添加歌曲:
在这里插入图片描述
显示歌曲播放进度,并且可以跳转播放
点击Pre/Next 选择上一首、下一首。
在这里插入图片描述

八、拓展

是否到这里就结束了呢?当然不!还有很多,比如歌词显示、换肤、播放时间、在线音乐等等。这些功能,敬请期待之后的文章。
夜逐渐地深了,晚安!
用Qt写一个简单的音乐播放器(五):歌曲播放时间显示

九、完整代码

1.mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    timer = new QTimer();
    timer->setInterval(2000);
    timer->start();
    ui->vSliderVolume->setVisible(false);
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::on_btnPlay_clicked()
{
    if(QMediaPlayer:: PlayingState == m_mediaPlayer.state()){
        m_mediaPlayer.pause();
        ui->btnPlay->setText("Play");
    }
    else{
        m_mediaPlayer.play();
        ui->btnPlay->setText("Pause");
    }
}


void MainWindow::on_hSliderPlayProgress_sliderMoved(int position)
{
    m_mediaPlayer.setPosition(m_mediaPlayer.duration()*position/100);
}
void MainWindow::setSliderValue()
{
    ui->hSliderPlayProgress->setValue(m_mediaPlayer.position()*100/m_mediaPlayer.duration());
}

void MainWindow::on_vSliderVolume_sliderMoved(int position)
{
    m_mediaPlayer.setVolume(position);
}

void MainWindow::on_lwMusicList_itemDoubleClicked(QListWidgetItem *item)
{
    m_PlayRow = ui->lwMusicList->row(item);
    m_mediaPlayer.stop();
    m_mediaPlayer.setMedia(QUrl::fromLocalFile(m_playPath+"/"+item->text()));
    m_mediaPlayer.play();
    ui->hSliderPlayProgress->setValue(m_mediaPlayer.position());
    ui->btnPlay->setText("Pause");
    connect(timer, SIGNAL(timeout()), this, SLOT(setSliderValue()));
}

void MainWindow::on_btnAddMusic_clicked()
{
    QString fileName = QFileDialog::getExistingDirectory(NULL,"Select Music Dir",".",NULL);
    QDir dir(fileName);
    QStringList nameFilters;
    nameFilters << "*.mp3";
    QStringList files = dir.entryList(nameFilters, QDir::Files|QDir::Readable, QDir::Name);
    ui->lwMusicList->addItems(files);
    m_playPath = fileName;
}

void MainWindow::on_btnPreMusic_clicked()
{
    if(m_PlayRow == 0){
        m_PlayRow = ui->lwMusicList->count() - 1;
    }
    else{
        m_PlayRow--;
    }
    QListWidgetItem *item = ui->lwMusicList->item(m_PlayRow);
    item->setSelected(true);
    m_mediaPlayer.stop();
    m_mediaPlayer.setMedia(QUrl::fromLocalFile(m_playPath+"/"+item->text()));
    m_mediaPlayer.play();
}

void MainWindow::on_btnNextMusic_clicked()
{
    if(m_PlayRow + 1 == ui->lwMusicList->count()){
        m_PlayRow = 0;
    }
    else{
        m_PlayRow++;
    }
    QListWidgetItem *item = ui->lwMusicList->item(m_PlayRow);
    item->setSelected(true);
    m_mediaPlayer.stop();
    m_mediaPlayer.setMedia(QUrl::fromLocalFile(m_playPath+"/"+item->text()));
    m_mediaPlayer.play();
}

void MainWindow::on_btnVolume_clicked()
{
    if(ui->vSliderVolume->isVisible()){
        ui->vSliderVolume->setVisible(false);
    }else{
        ui->vSliderVolume->setVisible(true);
    }
}

void MainWindow::on_vSliderVolume_sliderReleased()
{
     ui->vSliderVolume->setVisible(false);
}

2.mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QtMultimedia/QMediaPlayer> // 包含头文件
#include <QTimer>
#include <QListWidgetItem>
#include <QString>
#include <QFileDialog>
namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();

private slots:
    void on_btnPlay_clicked();

    void on_hSliderPlayProgress_sliderMoved(int position);

    void setSliderValue();

    void on_vSliderVolume_sliderMoved(int position);

    void on_lwMusicList_itemDoubleClicked(QListWidgetItem *item);

    void on_btnAddMusic_clicked();

    void on_btnPreMusic_clicked();

    void on_btnNextMusic_clicked();

    void on_btnVolume_clicked();

    void on_vSliderVolume_sliderReleased();

private:
    Ui::MainWindow *ui;
    QMediaPlayer m_mediaPlayer; // 实例化对象
    QTimer *timer; //定时器
    QString m_playPath;
    int m_PlayRow;
};

#endif // MAINWINDOW_H

<think>好的,我现在需要帮助用户开发一个基于Qt 5.12.10的音乐播放器,要求包括添加、删除音乐,暂停播放,并且每首音乐都有背景图片。首先,我得理清实现这些功能的关键步骤和可能遇到的问题。 首先,用户界面设计。可能需要一个播放列表视图来显示添加的音乐,以及控制按钮(播放、暂停、停止)。然后,背景图片的需求,每首音乐都要有对应的图片,这需要考虑如何关联音乐文件和图片,可能需要一个数据结构来存储这种关联,比如使用QMap或者自定义的数据结构。 接下来,添加和删除音乐的功能。添加音乐可以通过QFileDialog选择文件,支持常见的音频格式如mp3、wav等。删除则可以从播放列表中移除选中的项。这部分需要用到QMediaPlaylist来管理播放列表,但要注意Qt的多媒体模块在不同版本中的支持情况,尤其是5.12.10版本是否兼容某些功能。 然后,播放控制。使用QMediaPlayer来处理播放、暂停等功能。需要连接相应的信号和槽,比如媒体状态变化时更新按钮状态或界面显示。同时,要处理当前播放曲目的变化,以便切换对应的背景图片。 关于每首音乐的背景图片,用户可能需要为每首音乐指定一个图片路径,或者在添加音乐时让用户选择对应的图片。这需要在数据结构中保存每个音乐文件对应的图片路径。当播放曲目改变时,读取该路径的图片并设置为背景。可以使用QLabel或者QGraphicsView来显示图片,并调整其大小和布局。 另外,需要注意Qt版本5.12.10的多媒体模块是否支持某些功能,比如元数据读取。例如,获取音乐文件的封面图片,如果QMediaPlayer可以自动提取,可能简化实现。但用户需求是每首音乐都要有背景图片,可能需要手动关联,因此最好让用户在添加音乐时选择对应的图片文件。 数据结构方面,可以创建一个结构体或类,包含音乐文件路径和对应的图片路径。播放列表可以使用QList存储这些结构体,同时QMediaPlaylist管理播放顺序。当用户添加音乐时,弹出文件对话框选择音乐文件和图片文件,然后将这对路径添加到数据结构中,并更新播放列表的显示。 界面布局方面,主窗口可以分为几个部分:上方显示背景图片,中间显示播放列表,下方是控制按钮。背景图片可以使用QLabel,并设置为缩放适应区域。播放列表可以使用QListView或QListWidget,显示音乐名称,同时每个项可能需要自定义Widget来显示相关信息。 信号和槽的连接也很重要。比如,当播放列表的当前项改变时,触发背景图片的更新。QMediaPlayer的positionChanged信号可以用于更新进度条,但用户没有提到进度条,可能暂时不需要。另外,处理播放结束自动下一首的逻辑,可能需要连接mediaStatusChanged信号,检查是否到达末尾并切换下一首。 测试时需要注意不同音频格式的支持情况,确保Qt 5.12.10的多媒体后端(如DirectShow在Windows或GStreamer在Linux)已正确安装和配置。如果遇到无法播放的格式,可能需要安装额外的解码器。 总结步骤: 1. 创建Qt项目,配置.pro文件添加multimedia模块。 2. 设计UI:播放列表、控制按钮、背景图片显示区域。 3. 实现添加/删除音乐功能,同时关联图片路径。 4. 使用QMediaPlayer和QMediaPlaylist管理播放。 5. 当切换歌曲时,更新背景图片。 6. 处理播放状态更新按钮(播放/暂停切换)。 7. 测试并调试。 可能遇到的问题: - Qt多媒体模块在某些系统上支持有限,需要确保后端正确。 - 图片路径的管理,如果使用相对路径可能需要处理当前工作目录。 - 播放列表的同步更新,当删除音乐时需要更新数据结构和QMediaPlaylist。 需要查阅Qt 5.12.10的文档,确认QMediaPlayer和QMediaPlaylist的可用方法,特别是关于播放列表管理和元数据读取的部分。如果QMediaPlayer无法直接获取歌曲的元数据中的图片,就必须让用户手动选择图片,并在程序中维护两者的关联。</think>以下是用Qt 5.12.10开发音乐播放器的分步实现指南: --- ### 一、环境准备 1. 安装Qt 5.12.10 2. 在`.pro`文件中添加多媒体模块: ```qmake QT += multimedia multimediawidgets ``` --- ### 二、界面设计(`mainwindow.ui`) ```xml <!-- 垂直布局包含三部分 --> <QVBoxLayout> <!-- 1. 背景图片区域 --> <QLabel id="coverLabel" alignment="Center" minimumHeight="200"/> <!-- 2. 播放列表 --> <QListWidget id="playlistWidget"/> <!-- 3. 控制按钮 --> <QHBoxLayout> <QPushButton id="btnAdd" text="添加音乐"/> <QPushButton id="btnRemove" text="删除音乐"/> <QPushButton id="btnPlay" text="播放"/> <QPushButton id="btnPause" text="暂停"/> </QHBoxLayout> </QVBoxLayout> ``` --- ### 三、数据结构定义(`mainwindow.h`) ```cpp struct MusicItem { QString audioPath; // 音频文件路径 QString coverPath; // 封面图片路径 }; QList<MusicItem> musicList; // 存储所有音乐项 ``` --- ### 、核心功能实现 #### 1. 添加音乐 ```cpp void MainWindow::on_btnAdd_clicked() { // 选择音频文件 QString audioPath = QFileDialog::getOpenFileName(this, "选择音频", "", "音频文件(*.mp3 *.wav)"); if(audioPath.isEmpty()) return; // 选择封面图片 QString coverPath = QFileDialog::getOpenFileName(this, "选择封面", "", "图片(*.png *.jpg)"); if(coverPath.isEmpty()) return; // 添加到数据结构 musicList.append({audioPath, coverPath}); // 更新播放列表显示 QListWidgetItem *item = new QListWidgetItem(QFileInfo(audioPath).fileName()); playlistWidget->addItem(item); } ``` #### 2. 删除音乐 ```cpp void MainWindow::on_btnRemove_clicked() { int row = playlistWidget->currentRow(); if(row >= 0) { musicList.removeAt(row); delete playlistWidget->takeItem(row); } } ``` #### 3. 初始化播放器 ```cpp // 在MainWindow类中声明 QMediaPlayer *player; QMediaPlaylist *playlist; // 构造函数中初始化 player = new QMediaPlayer(this); playlist = new QMediaPlaylist(player); player->setPlaylist(playlist); ``` #### 4. 播放控制 ```cpp void MainWindow::on_btnPlay_clicked() { int index = playlistWidget->currentRow(); if(index >= 0) { playlist->clear(); playlist->addMedia(QUrl::fromLocalFile(musicList[index].audioPath)); player->play(); // 更新封面 QPixmap pix(musicList[index].coverPath); coverLabel->setPixmap(pix.scaled(coverLabel->size(), Qt::KeepAspectRatio)); } } void MainWindow::on_btnPause_clicked() { player->pause(); } ``` --- ### 五、关键问题处理 1. **封面自适应**:监听窗口大小变化事件 ```cpp void MainWindow::resizeEvent(QResizeEvent *event) { if(!coverLabel->pixmap().isNull()) { QPixmap pix = coverLabel->pixmap()->scaled(coverLabel->size(), Qt::KeepAspectRatio); coverLabel->setPixmap(pix); } } ``` 2. **跨平台兼容**:音频后端可能需要安装: - Windows:确保DirectShow支持 - Linux:安装`gstreamer`插件 --- ### 六、扩展建议 1. 增加进度条显示 2. 支持拖拽添加文件 3. 实现自动播放下一曲(通过`QMediaPlayer::mediaStatusChanged`信号) 完整示例代码可通过Qt官方示例扩展实现: https://doc.qt.io/qt-5.12/qtwidgets-multimedia-player-example.html
评论 14
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值