2016-02-15 02:15:38 -06:00
# -*- coding: utf-8 -*-
2019-04-10 17:36:02 -05:00
""" Audio player module for socializer.
As this player does not have ( still ) an associated GUI , I have decided to place here the code for the interactor , which connects a bunch of pubsub events , and the presenter itself .
"""
2019-01-01 20:04:30 -06:00
import sys
2017-03-13 02:16:34 -06:00
import random
2016-09-25 14:24:40 -05:00
import logging
2019-04-10 17:36:02 -05:00
import sound_lib
import output
2019-01-15 11:43:32 -06:00
import config
from sound_lib . config import BassConfig
2016-02-15 02:15:38 -06:00
from sound_lib . stream import URLStream
2016-09-19 17:13:40 -05:00
from sound_lib . main import BassError
2016-03-26 09:08:41 -06:00
from pubsub import pub
2019-04-10 17:36:02 -05:00
from mysc . repeating_timer import RepeatingTimer
2019-04-11 17:43:52 -05:00
from mysc . thread_utils import call_threaded
2019-04-16 15:45:25 -05:00
from sessionmanager import utils
2016-02-15 02:15:38 -06:00
player = None
2016-09-25 14:24:40 -05:00
log = logging . getLogger ( " player " )
2016-02-15 02:15:38 -06:00
2019-04-10 17:36:02 -05:00
# This function will be deprecated when the player works with pubsub events, as will no longer be needed to instantiate and import the player directly.
2016-02-15 02:15:38 -06:00
def setup ( ) :
global player
if player == None :
player = audioPlayer ( )
class audioPlayer ( object ) :
2019-04-10 17:36:02 -05:00
""" A media player which will play all passed URLS. """
2016-02-15 02:15:38 -06:00
def __init__ ( self ) :
2019-04-10 17:36:02 -05:00
# control variable for checking if another file has been sent to the player before,
# thus avoiding double file playback and other oddities happening in sound_lib from time to time.
2016-02-15 02:15:38 -06:00
self . is_playing = False
2019-04-10 17:36:02 -05:00
# This will be the URLStream handler
2016-02-15 02:15:38 -06:00
self . stream = None
2019-01-22 17:49:18 -06:00
self . vol = config . app [ " sound " ] [ " volume " ]
2019-04-10 17:36:02 -05:00
# this variable is set to true when the URLPlayer is decoding something, thus it will block other calls to the play method.
2016-02-19 17:30:11 -06:00
self . is_working = False
2019-04-10 17:36:02 -05:00
# Playback queue.
2016-03-23 08:17:45 -06:00
self . queue = [ ]
2019-04-10 17:36:02 -05:00
# Index of the currently playing track.
2019-01-22 17:41:39 -06:00
self . playing_track = 0
2019-04-10 17:36:02 -05:00
# Status of the player.
2016-03-23 08:17:45 -06:00
self . stopped = True
2019-01-15 11:43:32 -06:00
# Modify some default settings present in Bass so it will increase timeout connection, thus causing less "connection timed out" errors when playing.
bassconfig = BassConfig ( )
# Set timeout connection to 30 seconds.
bassconfig [ " net_timeout " ] = 30000
2019-04-10 17:49:23 -05:00
pub . subscribe ( self . play , " play " )
2019-04-11 17:15:45 -05:00
pub . subscribe ( self . play_all , " play-all " )
2019-04-10 17:49:23 -05:00
pub . subscribe ( self . pause , " pause " )
pub . subscribe ( self . stop , " stop " )
2019-04-11 17:43:52 -05:00
pub . subscribe ( self . play_next , " play-next " )
pub . subscribe ( self . play_previous , " play-previous " )
2019-04-17 11:46:26 -05:00
pub . subscribe ( self . seek , " seek " )
2016-02-15 02:15:38 -06:00
2019-04-10 17:36:02 -05:00
def play ( self , object , set_info = True , fresh = False ) :
""" Play an URl Stream.
@object dict : typically an audio object as returned by VK , with a " url " component which must be a valid URL to a media file .
@set_info bool : If true , will set information about the currently playing audio in the application status bar .
@fresh bool : If True , will remove everything playing in the queue and start this file only . otherwise it will play the new file but not remove the current queue . """
2019-04-11 17:15:45 -05:00
if " url " in object and object [ " url " ] == " " :
pub . sendMessage ( " notify " , message = _ ( " This file could not be played because it is not allowed in your country " ) )
return
2016-02-15 02:15:38 -06:00
if self . stream != None and self . stream . is_playing == True :
2016-09-19 17:13:40 -05:00
try :
self . stream . stop ( )
except BassError :
log . exception ( " error when stopping the file " )
self . stream = None
2016-03-23 08:17:45 -06:00
self . stopped = True
2019-01-22 17:41:39 -06:00
if fresh == True and hasattr ( self , " worker " ) and self . worker != None :
2016-03-25 11:06:02 -06:00
self . worker . cancel ( )
self . worker = None
self . queue = [ ]
2016-02-19 17:30:11 -06:00
# Make sure that there are no other sounds trying to be played.
if self . is_working == False :
self . is_working = True
2019-01-01 20:04:30 -06:00
# Let's encode the URL as bytes if on Python 3
2019-04-16 15:45:25 -05:00
url_ = utils . transform_audio_url ( object [ " url " ] )
url_ = bytes ( url_ , " utf-8 " )
2016-09-19 17:13:40 -05:00
try :
2019-01-01 20:04:30 -06:00
self . stream = URLStream ( url = url_ )
2019-04-16 15:45:25 -05:00
except :
2019-04-10 17:36:02 -05:00
log . error ( " Unable to play URL %s " % ( url_ ) )
2016-09-19 17:13:40 -05:00
return
2016-06-29 10:56:41 -05:00
# Translators: {0} will be replaced with a song's title and {1} with the artist.
2016-09-25 14:24:40 -05:00
if set_info :
2019-04-10 17:36:02 -05:00
msg = _ ( " Playing {0} by {1} " ) . format ( object [ " title " ] , object [ " artist " ] )
2016-09-25 14:24:40 -05:00
pub . sendMessage ( " update-status-bar " , status = msg )
2016-02-19 17:30:11 -06:00
self . stream . volume = self . vol / 100.0
self . stream . play ( )
2016-03-23 08:17:45 -06:00
self . stopped = False
2016-02-19 17:30:11 -06:00
self . is_working = False
2016-02-15 02:15:38 -06:00
def stop ( self ) :
2019-04-10 17:36:02 -05:00
""" Stop audio playback. """
2016-02-15 02:15:38 -06:00
if self . stream != None and self . stream . is_playing == True :
self . stream . stop ( )
2016-03-23 08:17:45 -06:00
self . stopped = True
if hasattr ( self , " worker " ) and self . worker != None :
self . worker . cancel ( )
self . worker = None
self . queue = [ ]
2016-02-15 02:15:38 -06:00
def pause ( self ) :
2019-04-10 17:36:02 -05:00
""" pause the current playback, without destroying the queue or the current stream. If the stream is already paused this function will resume the playback. """
2016-02-19 17:36:19 -06:00
if self . stream != None :
if self . stream . is_playing == True :
self . stream . pause ( )
2016-03-23 08:17:45 -06:00
self . stopped = True
2016-02-19 17:36:19 -06:00
else :
2016-09-19 17:13:40 -05:00
try :
self . stream . play ( )
self . stopped = False
except BassError :
pass
2016-02-15 02:15:38 -06:00
@property
def volume ( self ) :
2019-01-22 16:40:00 -06:00
return self . vol
2016-02-15 02:15:38 -06:00
@volume.setter
def volume ( self , vol ) :
2016-02-15 05:42:50 -06:00
if vol < = 100 and vol > = 0 :
self . vol = vol
2019-03-25 16:35:28 -06:00
elif vol < 0 :
self . vol = 0
elif vol > 100 :
self . vol = 100
2016-02-15 02:15:38 -06:00
if self . stream != None :
2016-02-15 05:42:50 -06:00
self . stream . volume = self . vol / 100.0
2016-02-15 02:15:38 -06:00
2019-04-10 17:36:02 -05:00
def play_all ( self , list_of_songs , shuffle = False ) :
""" Play all passed songs and adds all of those to the queue.
@list_of_songs list : A list of audio objects returned by VK .
@shuffle bool : If True , the files will be played randomly . """
2019-04-11 17:43:52 -05:00
if self . is_working :
return
2019-01-22 17:41:39 -06:00
self . playing_track = 0
2016-05-14 20:33:16 -05:00
self . stop ( )
2018-12-16 00:53:02 -06:00
# Skip all country restricted tracks as they are not playable here.
2019-04-10 17:36:02 -05:00
self . queue = [ i for i in list_of_songs if i [ " url " ] != " " ]
2017-03-13 02:16:34 -06:00
if shuffle :
random . shuffle ( self . queue )
2019-04-11 17:43:52 -05:00
call_threaded ( self . play , self . queue [ self . playing_track ] )
2016-03-23 08:17:45 -06:00
self . worker = RepeatingTimer ( 5 , self . player_function )
self . worker . start ( )
def player_function ( self ) :
2019-04-10 17:36:02 -05:00
""" Check if the stream has reached the end of the file so it will play the next song. """
2016-07-10 22:59:59 -05:00
if self . stream != None and self . stream . is_playing == False and self . stopped == False and len ( self . stream ) == self . stream . position :
2019-01-22 17:41:39 -06:00
if len ( self . queue ) == 0 or self . playing_track > = len ( self . queue ) :
2016-03-23 08:17:45 -06:00
self . worker . cancel ( )
return
2019-01-22 17:41:39 -06:00
if self . playing_track < len ( self . queue ) :
self . playing_track + = 1
self . play ( self . queue [ self . playing_track ] )
def play_next ( self ) :
2019-04-10 17:36:02 -05:00
""" Play the next song in the queue. """
2019-01-22 17:41:39 -06:00
if len ( self . queue ) == 0 :
return
2019-04-11 17:43:52 -05:00
if self . is_working :
return
2019-03-05 13:44:35 -06:00
if self . playing_track < len ( self . queue ) - 1 :
2019-01-22 17:41:39 -06:00
self . playing_track + = 1
else :
self . playing_track = 0
2019-04-11 17:43:52 -05:00
call_threaded ( self . play , self . queue [ self . playing_track ] )
2019-01-22 17:41:39 -06:00
def play_previous ( self ) :
2019-04-10 17:36:02 -05:00
""" Play the previous song in the queue. """
2019-01-22 17:41:39 -06:00
if len ( self . queue ) == 0 :
return
2019-04-11 17:43:52 -05:00
if self . is_working :
return
2019-01-22 17:41:39 -06:00
if self . playing_track < = 0 :
self . playing_track = len ( self . queue ) - 1
else :
self . playing_track - = 1
2019-04-11 17:43:52 -05:00
call_threaded ( self . play , self . queue [ self . playing_track ] )
2016-07-18 17:34:37 -05:00
2019-04-17 11:46:26 -05:00
def seek ( self , ms = 0 ) :
if self . check_is_playing ( ) :
if self . stream . position < 500000 and ms < 0 :
self . stream . position = 0
else :
try :
self . stream . position = self . stream . position + ms
except :
pass
2016-07-18 17:34:37 -05:00
def check_is_playing ( self ) :
2019-04-10 17:36:02 -05:00
""" check if the player is already playing a stream. """
2016-07-18 17:34:37 -05:00
if self . stream == None :
return False
if self . stream != None and self . stream . is_playing == False :
return False
else :
2019-04-16 15:45:25 -05:00
return True