UI: show chapter list and allow skip to chapter
This commit is contained in:
@ -23,7 +23,6 @@ These features are not available, some of them are TBD and others are not planne
|
|||||||
- Limited system integration:
|
- Limited system integration:
|
||||||
- No [SMTC](https://learn.microsoft.com/en-us/uwp/api/windows.media.systemmediatransportcontrols) support under Windows for now
|
- No [SMTC](https://learn.microsoft.com/en-us/uwp/api/windows.media.systemmediatransportcontrols) support under Windows for now
|
||||||
- No [MPRIS](https://www.freedesktop.org/wiki/Specifications/mpris-spec/) support under Linux desktop for now
|
- No [MPRIS](https://www.freedesktop.org/wiki/Specifications/mpris-spec/) support under Linux desktop for now
|
||||||
- No "playback progress on taskbar icon" and "taskbar thumbnail buttons" support whatever on Windows or Linux desktop for now
|
|
||||||
- Limited lyrics (`.lrc`) loading support:
|
- Limited lyrics (`.lrc`) loading support:
|
||||||
- Currently no `.tlrc` (for translated lyrics) or `.rlrc` (for romanized lyrics) support.
|
- Currently no `.tlrc` (for translated lyrics) or `.rlrc` (for romanized lyrics) support.
|
||||||
- Multi-line lyrics and duplicated timestamps are not supported
|
- Multi-line lyrics and duplicated timestamps are not supported
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// SPDX-FileCopyrightText: 2024 Gary Wang <git@blumia.net>
|
// SPDX-FileCopyrightText: 2025 Gary Wang <opensource@blumia.net>
|
||||||
//
|
//
|
||||||
// SPDX-License-Identifier: MIT
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
@ -61,6 +61,9 @@ MainWindow::MainWindow(QWidget *parent)
|
|||||||
m_mediaPlayer->setLoops(QMediaPlayer::Infinite);
|
m_mediaPlayer->setLoops(QMediaPlayer::Infinite);
|
||||||
ui->playlistView->setModel(m_playlistManager->model());
|
ui->playlistView->setModel(m_playlistManager->model());
|
||||||
|
|
||||||
|
ui->chapterlistView->setModel(ui->playbackProgressIndicator->chapterModel());
|
||||||
|
ui->chapterlistView->setRootIsDecorated(false);
|
||||||
|
|
||||||
ui->actionHelp->setShortcut(QKeySequence::HelpContents);
|
ui->actionHelp->setShortcut(QKeySequence::HelpContents);
|
||||||
addAction(ui->actionHelp);
|
addAction(ui->actionHelp);
|
||||||
ui->actionOpen->setShortcut(QKeySequence::Open);
|
ui->actionOpen->setShortcut(QKeySequence::Open);
|
||||||
@ -368,16 +371,6 @@ void MainWindow::on_playBtn_clicked()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
QString MainWindow::ms2str(qint64 ms)
|
|
||||||
{
|
|
||||||
QTime duaTime(QTime::fromMSecsSinceStartOfDay(ms));
|
|
||||||
if (duaTime.hour() > 0) {
|
|
||||||
return duaTime.toString("h:mm:ss");
|
|
||||||
} else {
|
|
||||||
return duaTime.toString("m:ss");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
QList<QUrl> MainWindow::strlst2urllst(QStringList strlst)
|
QList<QUrl> MainWindow::strlst2urllst(QStringList strlst)
|
||||||
{
|
{
|
||||||
QList<QUrl> urlList;
|
QList<QUrl> urlList;
|
||||||
@ -520,12 +513,24 @@ void MainWindow::initConnections()
|
|||||||
});
|
});
|
||||||
|
|
||||||
connect(m_mediaPlayer, &QMediaPlayer::positionChanged, this, [=](qint64 pos) {
|
connect(m_mediaPlayer, &QMediaPlayer::positionChanged, this, [=](qint64 pos) {
|
||||||
ui->nowTimeLabel->setText(ms2str(pos));
|
ui->nowTimeLabel->setText(PlaybackProgressIndicator::formatTime(pos));
|
||||||
if (m_mediaPlayer->duration() != 0) {
|
if (m_mediaPlayer->duration() != 0) {
|
||||||
ui->playbackProgressIndicator->setPosition(pos);
|
ui->playbackProgressIndicator->setPosition(pos);
|
||||||
m_taskbarManager->setProgressValue(pos);
|
m_taskbarManager->setProgressValue(pos);
|
||||||
}
|
}
|
||||||
m_lrcbar->playbackPositionChanged(pos, m_mediaPlayer->duration());
|
m_lrcbar->playbackPositionChanged(pos, m_mediaPlayer->duration());
|
||||||
|
|
||||||
|
static QString lastChapterName;
|
||||||
|
if (ui->playbackProgressIndicator->chapterModel()->rowCount() > 0) {
|
||||||
|
QString currentChapterName = ui->playbackProgressIndicator->currentChapterName();
|
||||||
|
if (currentChapterName != lastChapterName) {
|
||||||
|
ui->chapterNameBtn->setText(currentChapterName);
|
||||||
|
lastChapterName = currentChapterName;
|
||||||
|
}
|
||||||
|
} else if (!lastChapterName.isEmpty()) {
|
||||||
|
ui->chapterNameBtn->setText("");
|
||||||
|
lastChapterName.clear();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
connect(m_audioOutput, &QAudioOutput::mutedChanged, this, [=](bool muted) {
|
connect(m_audioOutput, &QAudioOutput::mutedChanged, this, [=](bool muted) {
|
||||||
@ -539,7 +544,7 @@ void MainWindow::initConnections()
|
|||||||
connect(m_mediaPlayer, &QMediaPlayer::durationChanged, this, [=](qint64 dua) {
|
connect(m_mediaPlayer, &QMediaPlayer::durationChanged, this, [=](qint64 dua) {
|
||||||
ui->playbackProgressIndicator->setDuration(dua);
|
ui->playbackProgressIndicator->setDuration(dua);
|
||||||
m_taskbarManager->setProgressMaximum(dua);
|
m_taskbarManager->setProgressMaximum(dua);
|
||||||
ui->totalTimeLabel->setText(ms2str(dua));
|
ui->totalTimeLabel->setText(PlaybackProgressIndicator::formatTime(dua));
|
||||||
});
|
});
|
||||||
|
|
||||||
connect(m_mediaPlayer, &QMediaPlayer::playbackStateChanged, this, [=](QMediaPlayer::PlaybackState newState) {
|
connect(m_mediaPlayer, &QMediaPlayer::playbackStateChanged, this, [=](QMediaPlayer::PlaybackState newState) {
|
||||||
@ -687,7 +692,16 @@ void MainWindow::on_setSkinBtn_clicked()
|
|||||||
|
|
||||||
void MainWindow::on_playListBtn_clicked()
|
void MainWindow::on_playListBtn_clicked()
|
||||||
{
|
{
|
||||||
setFixedSize(size().height() < 200 ? fullSize : miniSize);
|
if (size().height() < 200) {
|
||||||
|
setFixedSize(fullSize);
|
||||||
|
ui->pluginStackedWidget->setCurrentWidget(ui->playlistViewPage);
|
||||||
|
} else {
|
||||||
|
if (ui->pluginStackedWidget->currentWidget() == ui->playlistViewPage) {
|
||||||
|
setFixedSize(miniSize);
|
||||||
|
} else {
|
||||||
|
ui->pluginStackedWidget->setCurrentWidget(ui->playlistViewPage);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::on_playlistView_activated(const QModelIndex &index)
|
void MainWindow::on_playlistView_activated(const QModelIndex &index)
|
||||||
@ -706,6 +720,33 @@ void MainWindow::on_lrcBtn_clicked()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MainWindow::on_chapterlistView_activated(const QModelIndex &index)
|
||||||
|
{
|
||||||
|
if (!index.isValid()) return;
|
||||||
|
|
||||||
|
QModelIndex timeColumnIndex = index.sibling(index.row(), 0);
|
||||||
|
QStandardItem* timeItem = ui->playbackProgressIndicator->chapterModel()->itemFromIndex(timeColumnIndex);
|
||||||
|
if (!timeItem) return;
|
||||||
|
|
||||||
|
qint64 chapterStartTime = timeItem->data(PlaybackProgressIndicator::StartTimeMsRole).toLongLong();
|
||||||
|
m_mediaPlayer->setPosition(chapterStartTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MainWindow::on_chapterNameBtn_clicked()
|
||||||
|
{
|
||||||
|
if (size().height() < 200) {
|
||||||
|
setFixedSize(fullSize);
|
||||||
|
}
|
||||||
|
ui->pluginStackedWidget->setCurrentWidget(ui->chaptersViewPage);
|
||||||
|
if (ui->playbackProgressIndicator->chapterModel()->rowCount() > 0) {
|
||||||
|
const QModelIndex & curChapterItem = ui->playbackProgressIndicator->currentChapterItem();
|
||||||
|
if (curChapterItem.isValid()) {
|
||||||
|
ui->chapterlistView->setCurrentIndex(curChapterItem);
|
||||||
|
ui->chapterlistView->scrollTo(curChapterItem, QAbstractItemView::EnsureVisible);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void MainWindow::on_actionOpen_triggered()
|
void MainWindow::on_actionOpen_triggered()
|
||||||
{
|
{
|
||||||
loadFile();
|
loadFile();
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// SPDX-FileCopyrightText: 2024 Gary Wang <git@blumia.net>
|
// SPDX-FileCopyrightText: 2025 Gary Wang <opensource@blumia.net>
|
||||||
//
|
//
|
||||||
// SPDX-License-Identifier: MIT
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
@ -80,6 +80,8 @@ private slots:
|
|||||||
void on_playListBtn_clicked();
|
void on_playListBtn_clicked();
|
||||||
void on_playlistView_activated(const QModelIndex &index);
|
void on_playlistView_activated(const QModelIndex &index);
|
||||||
void on_lrcBtn_clicked();
|
void on_lrcBtn_clicked();
|
||||||
|
void on_chapterlistView_activated(const QModelIndex &index);
|
||||||
|
void on_chapterNameBtn_clicked();
|
||||||
void on_actionOpen_triggered();
|
void on_actionOpen_triggered();
|
||||||
void on_actionHelp_triggered();
|
void on_actionHelp_triggered();
|
||||||
|
|
||||||
|
@ -109,6 +109,18 @@ QLabel#coverLabel {
|
|||||||
QListView {
|
QListView {
|
||||||
color: white;
|
color: white;
|
||||||
background: rgba(0, 0, 0, 50);
|
background: rgba(0, 0, 0, 50);
|
||||||
|
}
|
||||||
|
|
||||||
|
/****** TreeView ******/
|
||||||
|
|
||||||
|
QTreeView {
|
||||||
|
color: white;
|
||||||
|
background: rgba(0, 0, 0, 50);
|
||||||
|
}
|
||||||
|
|
||||||
|
QHeaderView {
|
||||||
|
color: white;
|
||||||
|
background-color: rgba(200, 200, 200, 50);
|
||||||
}</string>
|
}</string>
|
||||||
</property>
|
</property>
|
||||||
<property name="locale">
|
<property name="locale">
|
||||||
@ -358,6 +370,19 @@ QListView {
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="chapterNameBtn">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Expanding" vsizetype="Fixed">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<property name="text">
|
||||||
|
<string/>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QLabel" name="totalTimeLabel">
|
<widget class="QLabel" name="totalTimeLabel">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
@ -626,7 +651,10 @@ QListView {
|
|||||||
<height>0</height>
|
<height>0</height>
|
||||||
</size>
|
</size>
|
||||||
</property>
|
</property>
|
||||||
<widget class="QWidget" name="pluginStackedWidgetPage1">
|
<property name="currentIndex">
|
||||||
|
<number>1</number>
|
||||||
|
</property>
|
||||||
|
<widget class="QWidget" name="playlistViewPage">
|
||||||
<layout class="QVBoxLayout" name="verticalLayout_2">
|
<layout class="QVBoxLayout" name="verticalLayout_2">
|
||||||
<property name="spacing">
|
<property name="spacing">
|
||||||
<number>0</number>
|
<number>0</number>
|
||||||
@ -648,6 +676,29 @@ QListView {
|
|||||||
</item>
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
|
<widget class="QWidget" name="chaptersViewPage">
|
||||||
|
<layout class="QVBoxLayout" name="verticalLayout_3">
|
||||||
|
<property name="leftMargin">
|
||||||
|
<number>4</number>
|
||||||
|
</property>
|
||||||
|
<property name="topMargin">
|
||||||
|
<number>0</number>
|
||||||
|
</property>
|
||||||
|
<property name="rightMargin">
|
||||||
|
<number>4</number>
|
||||||
|
</property>
|
||||||
|
<property name="bottomMargin">
|
||||||
|
<number>4</number>
|
||||||
|
</property>
|
||||||
|
<item>
|
||||||
|
<widget class="QTreeView" name="chapterlistView">
|
||||||
|
<property name="editTriggers">
|
||||||
|
<set>QAbstractItemView::NoEditTriggers</set>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</widget>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
|
@ -25,6 +25,37 @@ PlaybackProgressIndicator::PlaybackProgressIndicator(QWidget *parent) :
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QModelIndex PlaybackProgressIndicator::currentChapterItem() const
|
||||||
|
{
|
||||||
|
int currentChapterIndex = -1;
|
||||||
|
|
||||||
|
for (int i = 0; i < m_chapterModel.rowCount(); i++) {
|
||||||
|
QStandardItem* timeItem = m_chapterModel.item(i, 0);
|
||||||
|
qint64 chapterStartTime = timeItem->data(PlaybackProgressIndicator::StartTimeMsRole).toLongLong();
|
||||||
|
|
||||||
|
if (m_position >= chapterStartTime) {
|
||||||
|
currentChapterIndex = i;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (currentChapterIndex >= 0) {
|
||||||
|
return m_chapterModel.index(currentChapterIndex, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
QString PlaybackProgressIndicator::currentChapterName() const
|
||||||
|
{
|
||||||
|
const QModelIndex timeIndex(currentChapterItem());
|
||||||
|
if (timeIndex.isValid()) {
|
||||||
|
return m_chapterModel.item(timeIndex.row(), 1)->text();
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
void PlaybackProgressIndicator::setPosition(qint64 pos)
|
void PlaybackProgressIndicator::setPosition(qint64 pos)
|
||||||
{
|
{
|
||||||
m_position = pos;
|
m_position = pos;
|
||||||
@ -37,14 +68,34 @@ void PlaybackProgressIndicator::setDuration(qint64 dur)
|
|||||||
emit durationChanged(m_duration);
|
emit durationChanged(m_duration);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString PlaybackProgressIndicator::formatTime(qint64 milliseconds)
|
||||||
|
{
|
||||||
|
QTime duaTime(QTime::fromMSecsSinceStartOfDay(milliseconds));
|
||||||
|
if (duaTime.hour() > 0) {
|
||||||
|
return duaTime.toString("h:mm:ss");
|
||||||
|
} else {
|
||||||
|
return duaTime.toString("m:ss");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void PlaybackProgressIndicator::setChapters(QList<std::pair<qint64, QString> > chapters)
|
void PlaybackProgressIndicator::setChapters(QList<std::pair<qint64, QString> > chapters)
|
||||||
{
|
{
|
||||||
qDebug() << chapters;
|
|
||||||
m_chapterModel.clear();
|
m_chapterModel.clear();
|
||||||
|
|
||||||
|
m_chapterModel.setHorizontalHeaderLabels(QStringList() << tr("Time") << tr("Chapter Name"));
|
||||||
|
|
||||||
for (const std::pair<qint64, QString> & chapter : chapters) {
|
for (const std::pair<qint64, QString> & chapter : chapters) {
|
||||||
|
QList<QStandardItem*> row;
|
||||||
|
|
||||||
|
QStandardItem * timeItem = new QStandardItem(formatTime(chapter.first));
|
||||||
|
timeItem->setData(chapter.first, StartTimeMsRole);
|
||||||
|
row.append(timeItem);
|
||||||
|
|
||||||
QStandardItem * chapterItem = new QStandardItem(chapter.second);
|
QStandardItem * chapterItem = new QStandardItem(chapter.second);
|
||||||
chapterItem->setData(chapter.first, StartTimeMsRole);
|
chapterItem->setData(chapter.first, StartTimeMsRole);
|
||||||
m_chapterModel.appendRow(chapterItem);
|
row.append(chapterItem);
|
||||||
|
|
||||||
|
m_chapterModel.appendRow(row);
|
||||||
}
|
}
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
@ -24,10 +24,15 @@ public:
|
|||||||
explicit PlaybackProgressIndicator(QWidget *parent = nullptr);
|
explicit PlaybackProgressIndicator(QWidget *parent = nullptr);
|
||||||
~PlaybackProgressIndicator() = default;
|
~PlaybackProgressIndicator() = default;
|
||||||
|
|
||||||
|
QStandardItemModel* chapterModel() { return &m_chapterModel; }
|
||||||
|
QModelIndex currentChapterItem() const;
|
||||||
|
QString currentChapterName() const;
|
||||||
|
|
||||||
void setPosition(qint64 pos);
|
void setPosition(qint64 pos);
|
||||||
void setDuration(qint64 dur);
|
void setDuration(qint64 dur);
|
||||||
void setChapters(QList<std::pair<qint64, QString>> chapters);
|
void setChapters(QList<std::pair<qint64, QString>> chapters);
|
||||||
|
|
||||||
|
static QString formatTime(qint64 milliseconds);
|
||||||
static QList<std::pair<qint64, QString>> tryLoadChapters(const QString & filePath);
|
static QList<std::pair<qint64, QString>> tryLoadChapters(const QString & filePath);
|
||||||
static QList<std::pair<qint64, QString>> tryLoadSidecarChapterFile(const QString & filePath);
|
static QList<std::pair<qint64, QString>> tryLoadSidecarChapterFile(const QString & filePath);
|
||||||
static QList<std::pair<qint64, QString>> tryLoadChaptersFromMetadata(const QString & filePath);
|
static QList<std::pair<qint64, QString>> tryLoadChaptersFromMetadata(const QString & filePath);
|
||||||
|
Reference in New Issue
Block a user