1
1
mirror of https://github.com/danbee/persephone synced 2025-03-04 08:39:11 +00:00

Add dynamic dock menu with "Now Playing" section

Inspired pretty heavily by iTunes.
This commit is contained in:
Daniel Barber 2019-04-29 22:44:34 -04:00
parent ef8971f2fe
commit 17e592ef86
Signed by: danbarber
GPG Key ID: 931D8112E0103DD8
4 changed files with 76 additions and 64 deletions

View File

@ -662,11 +662,11 @@
E408D3C1220E134F0006D9BE /* AlbumViewController.swift */,
E47E2FD4222071FD00F747E6 /* AlbumViewItem.swift */,
E47E2FD022205C4600F747E6 /* MainSplitViewController.swift */,
E4405191227644340090CD6F /* MPDServerController.swift */,
E4E8CC932206097F0024217A /* NotificationsController.swift */,
E4E8CC912204F4B80024217A /* QueueViewController.swift */,
E465049921E94DF500A70F4C /* WindowController.swift */,
E4B11BB52275374B0075461B /* UserNotificationsController.swift */,
E4405191227644340090CD6F /* MPDServerController.swift */,
E465049921E94DF500A70F4C /* WindowController.swift */,
);
path = Controllers;
sourceTree = "<group>";

View File

@ -33,7 +33,7 @@ class AppDelegate: NSObject,
mediaKeyTap?.start()
AppDelegate.store.subscribe(self) {
$0.select { $0.playerState.databaseUpdating }
$0.select { $0.playerState }
}
}
@ -41,6 +41,57 @@ class AppDelegate: NSObject,
mpdServerController.disconnect()
}
func applicationDockMenu(_ sender: NSApplication) -> NSMenu? {
let dockMenu = NSMenu()
dockMenu.autoenablesItems = false
guard let state = AppDelegate.store.state.playerState.state else { return nil }
if let currentSong = AppDelegate.store.state.playerState.currentSong,
state.isOneOf([.playing, .paused]) {
let nowPlayingItem = NSMenuItem(title: "Now Playing", action: nil, keyEquivalent: "")
let songItem = NSMenuItem(title: currentSong.title, action: nil, keyEquivalent: "")
let albumItem = NSMenuItem(
title: "\(currentSong.artist)\(currentSong.album.title)",
action: nil,
keyEquivalent: ""
)
nowPlayingItem.isEnabled = false
songItem.indentationLevel = 1
songItem.isEnabled = false
albumItem.indentationLevel = 1
albumItem.isEnabled = false
dockMenu.addItem(nowPlayingItem)
dockMenu.addItem(songItem)
dockMenu.addItem(albumItem)
dockMenu.addItem(NSMenuItem.separator())
}
let playPauseMenuItem = NSMenuItem(
title: state == .playing ? "Pause" : "Play",
action: #selector(playPauseMenuAction),
keyEquivalent: ""
)
let stopMenuItem = NSMenuItem(title: "Stop", action: #selector(stopMenuAction), keyEquivalent: "")
let nextTrackMenuItem = NSMenuItem(title: "Next", action: #selector(nextTrackMenuAction), keyEquivalent: "")
let prevTrackMenuItem = NSMenuItem(title: "Previous", action: #selector(prevTrackMenuAction), keyEquivalent: "")
playPauseMenuItem.isEnabled = state.isOneOf([.playing, .paused, .stopped])
stopMenuItem.isEnabled = state.isOneOf([.playing, .paused])
nextTrackMenuItem.isEnabled = state.isOneOf([.playing, .paused])
prevTrackMenuItem.isEnabled = state.isOneOf([.playing, .paused])
dockMenu.addItem(playPauseMenuItem)
dockMenu.addItem(stopMenuItem)
dockMenu.addItem(nextTrackMenuItem)
dockMenu.addItem(prevTrackMenuItem)
return dockMenu
}
func handle(mediaKey: MediaKey, event: KeyEvent) {
switch mediaKey {
case .playPause:
@ -52,18 +103,18 @@ class AppDelegate: NSObject,
}
}
func setDockTransportControlState(_ state: MPDClient.MPDStatus.State) {
playPauseMenuItem.isEnabled = state.isOneOf([.playing, .paused, .stopped])
stopMenuItem.isEnabled = state.isOneOf([.playing, .paused])
nextTrackMenuItem.isEnabled = state.isOneOf([.playing, .paused])
prevTrackMenuItem.isEnabled = state.isOneOf([.playing, .paused])
if state.isOneOf([.paused, .stopped, .unknown]) {
playPauseMenuItem.title = "Play"
} else {
playPauseMenuItem.title = "Pause"
}
}
// func setDockTransportControlState(_ state: MPDClient.MPDStatus.State) {
// playPauseMenuItem.isEnabled = state.isOneOf([.playing, .paused, .stopped])
// stopMenuItem.isEnabled = state.isOneOf([.playing, .paused])
// nextTrackMenuItem.isEnabled = state.isOneOf([.playing, .paused])
// prevTrackMenuItem.isEnabled = state.isOneOf([.playing, .paused])
//
// if state.isOneOf([.paused, .stopped, .unknown]) {
// playPauseMenuItem.title = "Play"
// } else {
// playPauseMenuItem.title = "Pause"
// }
// }
@objc func enableUpdateDatabaseMenuItem() {
updateDatabaseMenuItem?.isEnabled = true
@ -87,17 +138,15 @@ class AppDelegate: NSObject,
}
@IBOutlet weak var updateDatabaseMenuItem: NSMenuItem!
@IBOutlet weak var playPauseMenuItem: NSMenuItem!
@IBOutlet weak var stopMenuItem: NSMenuItem!
@IBOutlet weak var nextTrackMenuItem: NSMenuItem!
@IBOutlet weak var prevTrackMenuItem: NSMenuItem!
}
extension AppDelegate: StoreSubscriber {
typealias StoreSubscriberStateType = Bool
typealias StoreSubscriberStateType = PlayerState
func newState(state: Bool) {
updateDatabaseMenuItem.isEnabled = !state
func newState(state: PlayerState) {
updateDatabaseMenuItem.isEnabled = !state.databaseUpdating
guard let state = state.state else { return }
// setDockTransportControlState(state)
}
}

View File

@ -20,6 +20,7 @@ class WindowController: NSWindowController {
override func windowDidLoad() {
super.windowDidLoad()
window?.titleVisibility = .hidden
window?.isExcludedFromWindowsMenu = true
AppDelegate.store.subscribe(self) {
$0.select { $0.playerState }
@ -53,12 +54,6 @@ class WindowController: NSWindowController {
}
}
func setWindowTitle() {
guard let status = AppDelegate.mpdClient.status else { return }
self.window?.title = status.
}
func setTrackProgressControls(_ playerState: PlayerState) {
guard let state = playerState.state,
let totalTime = playerState.totalTime,

View File

@ -90,6 +90,8 @@
<action selector="performZoom:" target="Ady-hI-5gd" id="DIl-cC-cCs"/>
</connections>
</menuItem>
<menuItem isSeparatorItem="YES" id="QhP-iz-rRa"/>
<menuItem title="Persephone" state="on" keyEquivalent="0" id="1Sq-L7-znT"/>
<menuItem isSeparatorItem="YES" id="eu3-7i-yIM"/>
<menuItem title="Bring All to Front" id="LE2-aR-0XJ">
<modifierMask key="keyEquivalentModifierMask"/>
@ -116,49 +118,15 @@
</menu>
<connections>
<outlet property="delegate" destination="Voe-Tx-rLC" id="PrD-fu-P6m"/>
<outlet property="dockMenu" destination="HMZ-6U-dkI" id="z8r-8r-l9q"/>
</connections>
</application>
<customObject id="Voe-Tx-rLC" customClass="AppDelegate" customModule="Persephone" customModuleProvider="target">
<connections>
<outlet property="nextTrackMenuItem" destination="InD-dy-4Xk" id="vTQ-Gf-E8k"/>
<outlet property="playPauseMenu" destination="Ldt-58-fDl" id="p2r-nI-tf4"/>
<outlet property="playPauseMenuItem" destination="Ldt-58-fDl" id="KGN-OQ-uHK"/>
<outlet property="prevTrackMenuItem" destination="90s-qF-dI2" id="Zba-8D-UEg"/>
<outlet property="stopMenuItem" destination="F1T-fC-2rX" id="doK-EU-w6M"/>
<outlet property="updateDatabaseMenuItem" destination="EJg-93-1F6" id="gMf-SQ-lyI"/>
</connections>
</customObject>
<customObject id="YLy-65-1bz" customClass="NSFontManager"/>
<customObject id="Ady-hI-5gd" userLabel="First Responder" customClass="NSResponder" sceneMemberID="firstResponder"/>
<menu autoenablesItems="NO" id="HMZ-6U-dkI">
<items>
<menuItem title="Play" enabled="NO" id="Ldt-58-fDl">
<modifierMask key="keyEquivalentModifierMask"/>
<connections>
<action selector="playPauseMenuAction:" target="Voe-Tx-rLC" id="J7X-Fe-jyM"/>
</connections>
</menuItem>
<menuItem title="Stop" enabled="NO" id="F1T-fC-2rX">
<modifierMask key="keyEquivalentModifierMask"/>
<connections>
<action selector="stopMenuAction:" target="Voe-Tx-rLC" id="CnR-Hf-mDH"/>
</connections>
</menuItem>
<menuItem title="Next" enabled="NO" id="InD-dy-4Xk">
<modifierMask key="keyEquivalentModifierMask"/>
<connections>
<action selector="nextMenuAction:" target="Voe-Tx-rLC" id="7QQ-ce-PyS"/>
</connections>
</menuItem>
<menuItem title="Previous" enabled="NO" id="90s-qF-dI2">
<modifierMask key="keyEquivalentModifierMask"/>
<connections>
<action selector="prevMenuAction:" target="Voe-Tx-rLC" id="iw3-oE-E2Y"/>
</connections>
</menuItem>
</items>
</menu>
</objects>
<point key="canvasLocation" x="81" y="-332"/>
</scene>
@ -166,7 +134,7 @@
<scene sceneID="R2V-B0-nI4">
<objects>
<windowController id="B8D-0N-5wS" customClass="WindowController" customModule="Persephone" customModuleProvider="target" sceneMemberID="viewController">
<window key="window" title="Window" allowsToolTipsWhenApplicationIsInactive="NO" autorecalculatesKeyViewLoop="NO" releasedWhenClosed="NO" visibleAtLaunch="NO" animationBehavior="default" id="IQv-IB-iLA" customClass="MainWindow" customModule="Persephone" customModuleProvider="target">
<window key="window" title="Persephone" allowsToolTipsWhenApplicationIsInactive="NO" autorecalculatesKeyViewLoop="NO" releasedWhenClosed="NO" visibleAtLaunch="NO" animationBehavior="default" id="IQv-IB-iLA" customClass="MainWindow" customModule="Persephone" customModuleProvider="target">
<windowStyleMask key="styleMask" titled="YES" closable="YES" miniaturizable="YES" resizable="YES"/>
<windowPositionMask key="initialPositionMask" leftStrut="YES" rightStrut="YES" topStrut="YES" bottomStrut="YES"/>
<rect key="contentRect" x="207" y="570" width="960" height="560"/>