Продолжайте вызывать сервис / трансляцию друг от друга - PullRequest
0 голосов
/ 30 мая 2018

Я создаю простое приложение, которое постоянно отслеживает уровень мультимедиа и настраивает его на 20% от максимального уровня все время, если пользователь увеличился, оно должно снова вернуться к 20%.

Концепция, которой я придерживался, заключается в том, чтобы осуществлять процесс мониторинга с помощью службы, когда эта служба уничтожается, она вызывает широковещательный приемник, который, в свою очередь, снова вызывает получателя и т. Д., Как бесконечный цикл, но выглядит в коде что-то не так.ниже, он не работает должным образом, и служба / широковещание не продолжают звонить друг другу!

Я запустил mainActivity как:

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        val audio = this.getSystemService(Context.AUDIO_SERVICE) as AudioManager
        val level = audio.getStreamVolume(AudioManager.STREAM_MUSIC)

        val maxVolume = audio.getStreamMaxVolume(AudioManager.STREAM_MUSIC)
        val percent = 0.2f
        val twintyVolume = (maxVolume * percent).toInt()

        if ( level > twintyVolume) {
            Toast.makeText(this,"audio level is $level", Toast.LENGTH_LONG).show()
            audio.setStreamVolume(AudioManager.STREAM_MUSIC,twintyVolume,0)
        }

        this.startService(Intent(this, VolumeCheck::class.java))
    }
}

Выше сделайте начальную проверку и уменьшитеОбъем носителя до 20% от максимального объема, затем запустите службу, которая делает то же самое с кодом ниже:

class VolumeCheck : Service() {

    private lateinit var context: Context

    override fun onCreate() {
        super.onCreate()
        context = this
        Toast.makeText(this, "service created", Toast.LENGTH_SHORT).show();
    }

    override fun onStartCommand(intent: Intent, flags: Int, startId: Int): Int {
        super.onStartCommand(intent, flags, startId)
        Toast.makeText(this, "service starting", Toast.LENGTH_SHORT).show();

        val audio = context.getSystemService(Context.AUDIO_SERVICE) as AudioManager
        val mediaPlayer = MediaPlayer()


     //   Thread().run {
            val maxVolume = audio.getStreamMaxVolume(AudioManager.STREAM_MUSIC)
            val percent = 0.2f
            val twintyVolume = (maxVolume * percent).toInt()
            if (mediaPlayer.isPlaying) {
                val level = audio.getStreamVolume(AudioManager.STREAM_MUSIC)
                if ( level > twintyVolume) {
                    Toast.makeText(context,"audio level is $level", Toast.LENGTH_LONG).show()
                    audio.setStreamVolume(AudioManager.STREAM_MUSIC,twintyVolume,0)
                }
            }
      //      Thread.sleep(3000)
     //   }
        stopSelf()
        return Service.START_STICKY
    }

    override fun onBind(intent: Intent): IBinder? {
        //TODO for communication return IBinder implementation
        return null
    }

    override fun onDestroy() {
        super.onDestroy()
        Toast.makeText(this, "service done", Toast.LENGTH_SHORT).show()
        val intent = Intent("com.kortex.mediafix.BootUpReceiver")
        sendBroadcast(intent)
    }
}

Как только служба уничтожена, она вызывает загрузочный приемник широковещания, который в своемПовторите вызов службы снова:

class BootUpReceiver: BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        context.startService(Intent(context, VolumeCheck::class.java))
    }
}

Manifest является:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.kortex.mediafix">

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <receiver android:name=".BootUpReceiver"
            android:permission="android.permission.RECEIVE_BOOT_COMPLETED">
            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED" />
                <action android:name="android.intent.action.QUICKBOOT_POWERON" />

                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
        </receiver>
        <service android:name=".VolumeCheck" />
    </application>

</manifest>

Ответы [ 3 ]

0 голосов
/ 01 июня 2018

Я решил это, используя ContentObserver, который вызывается из service, поэтому мой код теперь:

  1. MainActivity, чтобы запустить приложение и сделать первый разНастройка и запуск службы:

    class MainActivity : AppCompatActivity() {
    
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
    
        val audio = this.getSystemService(Context.AUDIO_SERVICE) as AudioManager
        val level = audio.getStreamVolume(AudioManager.STREAM_MUSIC)
    
        val maxVolume = audio.getStreamMaxVolume(AudioManager.STREAM_MUSIC)
        val percent = 0.2f
        val twintyVolume = (maxVolume * percent).toInt()
    
        if ( level > twintyVolume) audio.setStreamVolume(AudioManager.STREAM_MUSIC,twintyVolume,0)
    
        Toast.makeText(this,"Audio level adjusted to 20% instead of $level", Toast.LENGTH_LONG).show()
    
        this.startService(Intent(this, VolumeCheck::class.java))
    
        finish()
       }
    }
    
  2. Broadcast для запуска приложения и службы при каждой перезагрузке устройства:

    class BootUpReceiver: BroadcastReceiver() {
    
    override fun onReceive(context: Context, intent: Intent) {
        context.startService(Intent(context, VolumeCheck::class.java))
       }
    }
    
  3. Service для регистрации и вызова наблюдателя:

    class VolumeCheck : Service() {
    
    private lateinit var context: Context
    private lateinit var myObserver: VolumeOnserver
    
    
    override fun onCreate() {
        super.onCreate()
        context = this
        // Define the VolumeOnserver
        myObserver = VolumeOnserver(context, Handler())
        Toast.makeText(this, "service created", Toast.LENGTH_SHORT).show();
    }
    
    override fun onStartCommand(intent: Intent, flags: Int, startId: Int): Int {
        super.onStartCommand(intent, flags, startId)
        Toast.makeText(this, "service starting", Toast.LENGTH_SHORT).show();
    
        // Register the VolumeOnserver for setting changes
        contentResolver.registerContentObserver(android.provider.Settings.System.CONTENT_URI ,true, myObserver);
    
        return Service.START_STICKY
    }
    
    override fun onBind(intent: Intent): IBinder? {
        return null
    }
    
    override fun onDestroy() {
        super.onDestroy()
    
        // Unregister the VolumeOnserver
        contentResolver.unregisterContentObserver(myObserver);
       }
    }
    
  4. Observer, который наблюдает любые изменения в settings, проверяет громкость носителя и регулирует егоесли требуется:

    class VolumeOnserver (context: Context, h: Handler?): ContentObserver(h) {
    
    private val context = context
    
    override fun onChange(selfChange: Boolean) {
        onChange(selfChange, uri = null)
    }
    
    // Implement the onChange(boolean, Uri) method to take advantage of the new Uri argument.
    override fun onChange(selfChange: Boolean, uri: Uri?) {
        // Handle change.
        val audio = context.getSystemService(Context.AUDIO_SERVICE) as AudioManager
        val level = audio.getStreamVolume(AudioManager.STREAM_MUSIC)
    
        val maxVolume = audio.getStreamMaxVolume(AudioManager.STREAM_MUSIC)
        val percent = 0.2f
        val twintyVolume = (maxVolume * percent).toInt()
    
        if ( level > twintyVolume) audio.setStreamVolume(AudioManager.STREAM_MUSIC,twintyVolume,0)
    
        // Toast.makeText(context,"audio level is $level", Toast.LENGTH_LONG).show()
       }
    }
    

Файл Manifest:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.kortex.mediafix">

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <receiver android:name=".BootUpReceiver"
            android:permission="android.permission.RECEIVE_BOOT_COMPLETED">
            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED" />
                <action android:name="android.intent.action.QUICKBOOT_POWERON" />
               <action android:name="com.kortex.mediafix.BootUpReceiver" />

                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
        </receiver>
        <service android:name=".VolumeCheck" />
    </application>

</manifest>

И мое работающее приложение здесь .

Но есть проблема, которая не является стабильной, я настаиваю на том, чтобы изменить громкость, после многих испытаний служба и / или наблюдатель больше не работают, пока приложение не будет перезапущено.

0 голосов
/ 01 июня 2018

Другой способ сделать это без Service - это вызвать ContentObserver из BroadcastReceiver

  1. BroadcastReceiver:

    class BootUpReceiver: BroadcastReceiver () {

    override fun onReceive(context: Context, intent: Intent) {
        val myObserver = VolumeOnserver(context, Handler())
    
        // Register the VolumeOnserver for setting changes
        context.contentResolver.registerContentObserver(
                android.provider.Settings.System.CONTENT_URI ,true,
                myObserver)
       }
    }
    
  2. ContentObserver:

    class VolumeOnserver (контекст: Context, h: Обработчик?): ContentObserver (h) {

    private val context = context
    
    override fun onChange(selfChange: Boolean) {
        onChange(selfChange, uri = null)
    }
    
    // Implement the onChange(boolean, Uri) method to take advantage of the new Uri argument.
    override fun onChange(selfChange: Boolean, uri: Uri?) {
        // Handle change.
        val audio = context.getSystemService(Context.AUDIO_SERVICE) as AudioManager
        val level = audio.getStreamVolume(AudioManager.STREAM_MUSIC)
    
        val maxVolume = audio.getStreamMaxVolume(AudioManager.STREAM_MUSIC)
        val percent = 0.2f
        val twintyVolume = (maxVolume * percent).toInt()
    
        if ( level > twintyVolume) audio.setStreamVolume(AudioManager.STREAM_MUSIC,twintyVolume,0)
        }
    }
    
  3. MainActivity:

    class MainActivity: AppCompatActivity () {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
    
        val audio = this.getSystemService(Context.AUDIO_SERVICE) as AudioManager
        val level = audio.getStreamVolume(AudioManager.STREAM_MUSIC)
    
        val maxVolume = audio.getStreamMaxVolume(AudioManager.STREAM_MUSIC)
        val percent = 0.2f
        val twintyVolume = (maxVolume * percent).toInt()
    
        if ( level > twintyVolume) audio.setStreamVolume(AudioManager.STREAM_MUSIC,twintyVolume,0)
    
        finish()
       }
    }
    
0 голосов
/ 30 мая 2018
val intent = Intent("com.kortex.mediafix.BootUpReceiver")

Эта строка не вызывает вашего широковещательного получателя, а скорее создает намерение с действием намерения как "com.kortex.mediafix.BootUpReceiver"

Измените запись вашего BootUpReceiver в манифесте, чтобы получить это действие

<receiver android:name=".BootUpReceiver"
        android:permission="android.permission.RECEIVE_BOOT_COMPLETED">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED" />
            <action android:name="android.intent.action.QUICKBOOT_POWERON" />
            <action android:name="com.kortex.mediafix.BootUpReceiver" />

            <category android:name="android.intent.category.DEFAULT" />
        </intent-filter>
    </receiver>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...