Skip to content

Commit

Permalink
move provider bar out of sidebar.py
Browse files Browse the repository at this point in the history
  • Loading branch information
cosven committed Nov 19, 2024
1 parent 32b29e4 commit 8826c11
Show file tree
Hide file tree
Showing 2 changed files with 224 additions and 205 deletions.
216 changes: 216 additions & 0 deletions feeluown/gui/uimain/provider_bar.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,216 @@
from typing import TYPE_CHECKING

from PyQt5.QtCore import Qt
from PyQt5.QtWidgets import (
QFrame, QLabel, QVBoxLayout, QHBoxLayout, QFormLayout,
QDialog, QLineEdit, QDialogButtonBox, QMessageBox, QWidget, QSizePolicy
)

from feeluown.excs import ProviderIOError, NoUserLoggedIn
from feeluown.library import (
SupportsPlaylistDelete,
SupportsPlaylistCreateByName,
SupportsCurrentUser,
)
from feeluown.utils import aio
from feeluown.gui.components import Avatar
from feeluown.gui.widgets import (
DiscoveryButton,
StarButton,
PlusButton,
TriagleButton,
)
from feeluown.gui.widgets.playlists import PlaylistsView
from feeluown.gui.widgets.my_music import MyMusicView

if TYPE_CHECKING:
from feeluown.app.gui_app import GuiApp


class ListViewContainer(QFrame):

def __init__(self, label, view, parent=None):
super().__init__(parent)

self._btn_length = 14
self._label = label
self._view = view
self._toggle_btn = TriagleButton(length=self._btn_length, padding=0.2)
self.create_btn = PlusButton(length=self._btn_length)
# Show this button when needed.
self.create_btn.hide()

self._toggle_btn.clicked.connect(self.toggle_view)
self.setup_ui()

def setup_ui(self):
self._label.setFixedHeight(25)

self._layout = QVBoxLayout(self)
self._layout.setContentsMargins(0, 0, 0, 0)
self._layout.setSpacing(0)

self._t_h_layout = QHBoxLayout()
self._b_h_layout = QHBoxLayout()
self._t_h_layout.addWidget(self._label)
self._t_h_layout.addStretch(0)
self._t_h_layout.addWidget(self.create_btn)
self._t_h_layout.addSpacing(self._btn_length // 2)
self._t_h_layout.addWidget(self._toggle_btn)
self._b_h_layout.addWidget(self._view)

self._layout.addLayout(self._t_h_layout)
self._layout.addLayout(self._b_h_layout)
# XXX: 本意是让 ListViewContainer 下方不要出现多余的空间
self.setSizePolicy(QSizePolicy.Preferred, QSizePolicy.Maximum)

def toggle_view(self):
if self._view.isVisible():
self._toggle_btn.set_direction('down')
self._view.hide()
else:
self._toggle_btn.set_direction('up')
self._view.show()


LVC = ListViewContainer


class ProviderBar(QWidget):
"""
A panel shows provider-specific contents.
"""
def __init__(self, app: 'GuiApp', parent=None):
super().__init__(parent)
self._app = app

self.discovery_btn = DiscoveryButton(height=30, padding=0.2, parent=self)
self.fav_btn = StarButton('我的收藏', height=30, parent=self)
self.fold_top_btn = TriagleButton(length=14, padding=0.2)
self.fold_top_btn.setCheckable(True)

self.playlists_header = QLabel('歌单列表', self)
self.my_music_header = QLabel('我的音乐', self)

self._layout = QVBoxLayout(self)
# Layout to let provider add it's own buttons.
self._btn_layout = QVBoxLayout()
self.playlists_view = PlaylistsView(self)
self.my_music_view = MyMusicView(self)
self.playlists_view.setModel(self._app.pl_uimgr.model)
self.my_music_view.setModel(self._app.mymusic_uimgr.model)

self.playlists_con = LVC(self.playlists_header, self.playlists_view)
self.my_music_con = LVC(self.my_music_header, self.my_music_view)

self.playlists_view.show_playlist.connect(
lambda pl: self._app.browser.goto(model=pl))
self.playlists_view.remove_playlist.connect(self._remove_playlist)
self.playlists_con.create_btn.clicked.connect(self._create_playlist)
self._app.current_pvd_ui_mgr.changed.connect(
self.on_current_pvd_ui_changed)
self.discovery_btn.clicked.connect(
lambda: self._app.browser.goto(page='/rec'))
self.fav_btn.clicked.connect(
lambda: self._app.browser.goto(page='/my_fav'))

self.setup_ui()

def setup_ui(self):
self._layout.setSpacing(0)
self._layout.setContentsMargins(0, 0, 0, 0)

self.playlists_view.setFrameShape(QFrame.NoFrame)
self.my_music_view.setFrameShape(QFrame.NoFrame)

self._avatar_layout = QHBoxLayout()
self._avatar_layout.addWidget(Avatar(self._app, height=48))
self._avatar_layout.addWidget(self.fold_top_btn)

self._layout.addLayout(self._avatar_layout)
self._layout.addWidget(self.discovery_btn)
self._layout.addWidget(self.fav_btn)
self._layout.addLayout(self._btn_layout)
self._layout.addWidget(self.my_music_con)
self._layout.addWidget(self.playlists_con)

# 让各个音乐库来决定是否显示这些组件
self.playlists_con.hide()
self.my_music_con.hide()
self.discovery_btn.setDisabled(True)
self.fav_btn.setDisabled(True)
self.discovery_btn.setToolTip('当前资源提供方未知')
self.fold_top_btn.setToolTip('折叠/打开“主页和本地收藏集”功能')

def add_btn(self, btn):
self._btn_layout.addWidget(btn)

def on_current_pvd_ui_changed(self, pvd_ui, _):
if pvd_ui:
self.discovery_btn.setEnabled(True)
self.fav_btn.setEnabled(True)
self.discovery_btn.setToolTip(f'点击进入 {pvd_ui.provider.name} 推荐页')
else:
self.discovery_btn.setEnabled(False)
self.fav_btn.setEnabled(False)

def _create_playlist(self):
provider_ui = self._app.current_pvd_ui_mgr.get()
if provider_ui is None:
self._app.show_msg('当前的资源提供方未注册其 UI')
return
provider = provider_ui.provider
if not isinstance(provider, SupportsPlaylistCreateByName) \
or not isinstance(provider, SupportsCurrentUser) \
or not provider.has_current_user():
self._app.show_msg('当前的资源提供方不支持创建歌单')
return

dialog = QDialog(self)
# Set WA_DeleteOnClose so that the dialog can be deleted (from self.children).
dialog.setAttribute(Qt.WA_DeleteOnClose)
layout = QFormLayout(dialog)
title_edit = QLineEdit(dialog)
layout.addRow('歌单名', title_edit)
button_box = QDialogButtonBox(QDialogButtonBox.Cancel | QDialogButtonBox.Yes)
layout.addRow('', button_box)
button_box.accepted.connect(dialog.accept)
button_box.rejected.connect(dialog.reject)

def create_playlist_and_reload():
title = title_edit.text()

async def do():
try:
playlist = await aio.run_fn(provider.playlist_create_by_name, title)
except (ProviderIOError, NoUserLoggedIn) as e:
QMessageBox.warning(self._app, '错误', f"创建歌单 '{title}' 失败: {e}")
else:
# Add playlist to pl_uimgr is a workaround, which may cause bug.
# For example, the order of the newly created playlist should be
# in the top for some providers.
# TODO: re-fetch user's playlists and fill the UI.
self._app.pl_uimgr.add(playlist, is_fav=False)
self._app.show_msg(f"创建歌单 '{title}' 成功")

aio.run_afn(do)

dialog.accepted.connect(create_playlist_and_reload)
dialog.open()

def _remove_playlist(self, playlist):

async def do():
provider = self._app.library.get(playlist.source)
if isinstance(provider, SupportsPlaylistDelete):
ok = await aio.run_fn(provider.playlist_delete, playlist.identifier)
self._app.show_msg(f"删除歌单 {playlist.name} {'成功' if ok else '失败'}")
if ok is True:
self._app.pl_uimgr.model.remove(playlist)
else:
self._app.show_msg(f'资源提供方({playlist.source})不支持删除歌单')

box = QMessageBox(QMessageBox.Warning, '提示', f"确认删除歌单 '{playlist.name}' 吗?",
QMessageBox.Yes | QMessageBox.No, self)
box.accepted.connect(lambda: aio.run_afn(do))
box.open()
Loading

0 comments on commit 8826c11

Please sign in to comment.