Подключить сервис к активности?
Когда я впервые открываю Activity, он создает MediaBrowserService. Когда я пытаюсь открыть его снова, я хотел снова подключиться к нему. Я попробовал несколько способов:
override fun onResume(){
mMediaBrowser.connect()
buildTransportControls()
}
Когда я пытаюсь сделать это, он всегда выдает сообщение об ошибке:
connect() called while not disconnected (state=CONNECT_STATE_CONNECTING)
Как это может быть даже через я звоню в onStop()
функция mMediaBrowser.disconnect()
? Или это просто неправильный путь? Все это происходит, когда я возвращаюсь через BackButton.
Мой полный код:
class MusicPlayer : AppCompatActivity() {
private var songAdapter : RecyclerView.Adapter<MusicRecyclerSongAdapter.ViewHolder>? = null
private var albumAdapter: RecyclerView.Adapter<MusicRecyclerAlbumAdapter.ViewHolder>? = null
val context:Context = this
//Media Browser
private lateinit var mMediaBrowser: MediaBrowserCompat
//prepare variables for late init
lateinit var songUri:ArrayList<String>
lateinit var album_name:ArrayList<String>
lateinit var mediaController : MediaControllerCompat
//------------------------------------------------------------------------------
fun buildTransportControls(){
mMediaBrowser.subscribe(mMediaBrowser.root,object: MediaBrowserCompat.SubscriptionCallback(){})
mediaController = MediaControllerCompat.getMediaController(this@MusicPlayer)
//Show Init state
var metadat = mediaController.metadata
var pbState = mediaController.playbackState
//Register Callback to stay synced
mediaController.registerCallback(controllerCallback)
}
//------------------------------------------------------------------------------
val controllerCallback = object : MediaControllerCompat.Callback() {
override fun onMetadataChanged(metadat: MediaMetadataCompat){
Log.i("Musik",metadat.toString())
}
override fun onPlaybackStateChanged(state :PlaybackStateCompat){
}
}
//------------------------------------------------------------------------------
private val mConnectionCallbacks = object : MediaBrowserCompat.ConnectionCallback() {
override fun onConnected() {
Log.i("Connection","Connecting")
// Get the token for the MediaSession
val token = mMediaBrowser.sessionToken
// Create a MediaControllerCompat
val mediaController = MediaControllerCompat(this@MusicPlayer,token)
// Save the controller
MediaControllerCompat.setMediaController(this@MusicPlayer, mediaController)
// Finish building the UI
buildTransportControls()
}
override fun onConnectionSuspended() {
// The Service has crashed. Disable transport controls until it automatically reconnects
Log.i("ERROR","Connection Suspended")
}
override fun onConnectionFailed() {
// The Service has refused our connection
Log.i("ERROR","Connection Failed")
}
}
//------------------------------------------------------------------------------
fun startSong(position: Int){
if(songAdapter == null) {
songAdapter = MusicRecyclerSongAdapter(context, album_name[position])
//Getting the Location of the Songs
songUri = GetMusic.uris
recyclerViewMusic.adapter = songAdapter
}else{
var pbSate = MediaControllerCompat.getMediaController(this@MusicPlayer).playbackState.playbackState
MediaControllerCompat.getMediaController(this@MusicPlayer).transportControls.playFromUri(Uri.parse(songUri[position]), null)
//TODO Testing if no song is playing already
}
}
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_music_player)
//Initialize MediaBrowserServiceCompat
if(ContextCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED){
//Create MediaBrowser Service
createMediaBrowser()
}else {
permissionRequest()
}
//Create Recycler View and get Adapter
//val recyclerViewMusic: RecyclerView = findViewById(R.id.recyclerViewMusic)
recyclerViewMusic.layoutManager = LinearLayoutManager(this, LinearLayoutManager.HORIZONTAL,false)
albumAdapter = MusicRecyclerAlbumAdapter(context)
//Getting values from Companion Object
album_name = GetAlbum.albums
//Apply Adapter to Recyclerview and change Backgroundcolor
recyclerViewMusic.adapter = albumAdapter
recyclerViewMusic.setBackgroundColor(Color.BLUE)
recyclerViewMusic.addOnItemTouchListener(
RecyclerItemClickListener(context, recyclerViewMusic, object : RecyclerItemClickListener.OnItemClickListener {
override fun onItemClick(view: View, position: Int) {
//Change to Song selection
startSong(position)
}
override fun onLongItemClick(view: View, position: Int) {
//Change to Song selection
startSong(position)
}
}
)
)
musicprogressbar.setOnSeekBarChangeListener(object: SeekBar.OnSeekBarChangeListener{
override fun onProgressChanged(seekBar: SeekBar?, progress: Int, fromUser: Boolean) {
if(fromUser) {
//mplayer.seekTo(progress)
}
}
override fun onStartTrackingTouch(seekBar: SeekBar?) {
//if(::mplayer.isInitialized) {
// mplayer.pause()
//}
}
override fun onStopTrackingTouch(seekBar: SeekBar?) {
//if(::mplayer.isInitialized) {
// mplayer.start() }
}
}
)
}
override fun onStart() {
super.onStart()
mMediaBrowser.connect()
}
override fun onResume() {
super.onResume()
Log.i("I","RE OPEN")
Log.i("MediaBrowserStatus",mMediaBrowser.isConnected.toString())
if(!mMediaBrowser.isConnected) {
mMediaBrowser.connect()
}
buildTransportControls()
volumeControlStream = AudioManager.STREAM_MUSIC
}
override fun onStop() {
super.onStop()
Log.i("STOP","STOP")
if(MediaControllerCompat.getMediaController(this@MusicPlayer) != null){
MediaControllerCompat.getMediaController(this@MusicPlayer).unregisterCallback(controllerCallback)
}
mMediaBrowser.disconnect()
}
fun permissionRequest(){
val permissions = ContextCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE)
if(permissions != PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(this, arrayOf(
Manifest.permission.READ_EXTERNAL_STORAGE
),101)
}
}
override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<out String>, grantResults: IntArray) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
createMediaBrowser()
}
fun createMediaBrowser(){
Log.i("HI","HI")
mMediaBrowser = MediaBrowserCompat(this,
ComponentName(this, MediaPlaybackService::class.java),
mConnectionCallbacks,
null)
}
fun musicControls(view: View) {
//applying functions to buttons
if (mMediaBrowser.isConnected) {
when (view.id) {
//Pause or Start Music
R.id.musicStart -> if (mediaController.playbackState.playbackState != null) {
mediaController.transportControls.play()
Log.i("TEST",mediaController.playbackState.playbackState.toString())}
R.id.musicPause -> if (mediaController.playbackState.playbackState != null) {
mediaController.transportControls.pause()
Log.i("TEST",mediaController.playbackState.playbackState.toString())}
/* //Forward/Backward Music
R.id.musicbackward -> {
mplayer.pause()
mplayer.seekTo(mplayer.currentPosition - 5000)
mplayer.start()
}
R.id.musicforward -> {
mplayer.pause()
mplayer.seekTo(mplayer.currentPosition + 5000)
mplayer.start()
}*/
}
}
}
override fun onBackPressed() {
super.onBackPressed()
}
}
// РЕДАКТИРОВАТЬ, как просили код службы:
class MediaPlaybackService: MediaBrowserServiceCompat(){
//private val MY_MEDIA_ROOT_ID = "MediaStore.Audio.Media.EXTERNAL_CONTENT_URI"
private val MY_MEDIA_ROOT_ID ="root"
private lateinit var mMediaSession: MediaSessionCompat
private lateinit var mStateBuilder : PlaybackStateCompat.Builder
private lateinit var MySessionCallback : MediaSessionCompat.Callback
private lateinit var mMediaPlayer: MediaPlayer
override fun onCreate() {
super.onCreate()
initMediaPlayer()
//Create the MediaSession
mMediaSession = MediaSessionCompat(this,"PLAYER")
//Setting the necessary Flags (Media Buttons)
mMediaSession.setFlags(
MediaSessionCompat.FLAG_HANDLES_MEDIA_BUTTONS or MediaSessionCompat.FLAG_HANDLES_TRANSPORT_CONTROLS)
//Set an inital PlaybackStatewith ACTION_BUTTONS, so Media buttons can start the player
mStateBuilder = PlaybackStateCompat.Builder().setActions(
PlaybackStateCompat.ACTION_PLAY or PlaybackStateCompat.ACTION_PAUSE)
//Set our PLaybackState for the MediaSession
mMediaSession.setPlaybackState(mStateBuilder.build())
MySessionCallback = object : MediaSessionCompat.Callback() {
override fun onPlayFromUri(uri: Uri?, extras: Bundle?) {
super.onPlayFromUri(uri, extras)
if(!mMediaPlayer.isPlaying) {
mMediaPlayer.setDataSource(application.applicationContext, uri)
mMediaPlayer.setOnPreparedListener {
mMediaPlayer.start()
Log.i("DURATION",mMediaPlayer.duration.toString())
}
mMediaPlayer.prepareAsync()
}
}
override fun onPause() {
super.onPause()
if(mMediaPlayer.isPlaying){
mMediaPlayer.pause()}
}
override fun onPlay() {
super.onPlay()
mMediaPlayer.start()
}
}
//Handles callbacks from Media Controller MySessionCalback is a PlaeHolder
mMediaSession.setCallback(MySessionCallback)
mMediaSession.isActive = true
//Set SessionToken so Activites can communicate with it
setSessionToken(mMediaSession.getSessionToken());
}
override fun onLoadChildren(parentId: String, result: Result<MutableList<MediaBrowserCompat.MediaItem>>) {
var mediaItems:ArrayList<MediaBrowserCompat.MediaItem> = ArrayList()
/*
var albums = MusicLibrary(this@MediaPlaybackService).getMusic()
for(item in MusicLibrary.MusicFiles){
val songList = MediaBrowserCompat.MediaItem(item,
MediaBrowserCompat.MediaItem.FLAG_PLAYABLE)
mediaItems.add(songList)
}
Log.i("MEDIA_ITEMS",mediaItems.toString())
*/
result.sendResult(mediaItems)
}
override fun onGetRoot(clientPackageName: String, clientUid: Int, rootHints: Bundle?): BrowserRoot? {
return BrowserRoot(MY_MEDIA_ROOT_ID,null)
}
fun initMediaPlayer(){
mMediaPlayer = MediaPlayer()
}
}
1 ответ
Проблема, которую вы видите, связана с тем, что вызовы для подключения и отключения являются асинхронными. Когда вы звоните connect()
, соединение не установлено сразу. Вы просто "запрашиваете соединение". У вас есть соединение только после обратного вызова onConnected()
называется.
Так что, когда пользователь уходит обратно из другого Activity
сначала позвони connect()
в onStart()
и почти сразу onResume()
называется. В onResume()
когда ты звонишь isConnected()
, это вернется false
потому что соединение еще не установлено (т.е. обратный вызов onConnected()
еще не назывался). Вы тогда звоните connect()
который терпит неудачу с данным исключением. Вы даже можете увидеть из сообщения об исключении:
connect() called while not disconnected (state=CONNECT_STATE_CONNECTING)
Что текущее состояние не "СОЕДИНЕНО", а "СОЕДИНЕНО".
Вам нужно подождать, пока поступят асинхронные обратные вызовы, прежде чем взаимодействовать с MediaBrowser
,