|
@@ -16,9 +16,9 @@ use audio::{VorbisDecoder, VorbisPacket};
|
|
use metadata::{FileFormat, Track, Metadata};
|
|
use metadata::{FileFormat, Track, Metadata};
|
|
use mixer::AudioFilter;
|
|
use mixer::AudioFilter;
|
|
|
|
|
|
-#[derive(Clone)]
|
|
|
|
pub struct Player {
|
|
pub struct Player {
|
|
- commands: std::sync::mpsc::Sender<PlayerCommand>,
|
|
|
|
|
|
+ commands: Option<std::sync::mpsc::Sender<PlayerCommand>>,
|
|
|
|
+ thread_handle: Option<thread::JoinHandle<()>>,
|
|
}
|
|
}
|
|
|
|
|
|
struct PlayerInternal {
|
|
struct PlayerInternal {
|
|
@@ -47,7 +47,7 @@ impl Player {
|
|
{
|
|
{
|
|
let (cmd_tx, cmd_rx) = std::sync::mpsc::channel();
|
|
let (cmd_tx, cmd_rx) = std::sync::mpsc::channel();
|
|
|
|
|
|
- thread::spawn(move || {
|
|
|
|
|
|
+ let handle = thread::spawn(move || {
|
|
debug!("new Player[{}]", session.session_id());
|
|
debug!("new Player[{}]", session.session_id());
|
|
|
|
|
|
let internal = PlayerInternal {
|
|
let internal = PlayerInternal {
|
|
@@ -64,12 +64,13 @@ impl Player {
|
|
});
|
|
});
|
|
|
|
|
|
Player {
|
|
Player {
|
|
- commands: cmd_tx,
|
|
|
|
|
|
+ commands: Some(cmd_tx),
|
|
|
|
+ thread_handle: Some(handle),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
fn command(&self, cmd: PlayerCommand) {
|
|
fn command(&self, cmd: PlayerCommand) {
|
|
- self.commands.send(cmd).unwrap();
|
|
|
|
|
|
+ self.commands.as_ref().unwrap().send(cmd).unwrap();
|
|
}
|
|
}
|
|
|
|
|
|
pub fn load(&self, track: SpotifyId, start_playing: bool, position_ms: u32)
|
|
pub fn load(&self, track: SpotifyId, start_playing: bool, position_ms: u32)
|
|
@@ -98,6 +99,19 @@ impl Player {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+impl Drop for Player {
|
|
|
|
+ fn drop(&mut self) {
|
|
|
|
+ debug!("Shutting down player thread ...");
|
|
|
|
+ self.commands = None;
|
|
|
|
+ if let Some(handle) = self.thread_handle.take() {
|
|
|
|
+ match handle.join() {
|
|
|
|
+ Ok(_) => (),
|
|
|
|
+ Err(_) => error!("Player thread panicked!")
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
type Decoder = VorbisDecoder<Subfile<AudioDecrypt<AudioFile>>>;
|
|
type Decoder = VorbisDecoder<Subfile<AudioDecrypt<AudioFile>>>;
|
|
enum PlayerState {
|
|
enum PlayerState {
|
|
Stopped,
|
|
Stopped,
|