2015-06-05 12 views
11

Sto verificando se l'allarme è già stato impostato da AlarmManager utilizzando this answer.Impossibile controllare se AlarmManager ha impostato l'allarme

Di seguito è riportato lo snippet di codice.

boolean alarmUp = (PendingIntent.getBroadcast(MainActivity.this, 0, 
    new Intent(MainActivity.this, AlarmReceiver.class), PendingIntent.FLAG_NO_CREATE) != null); 
if (alarmUp) { 
    // alarm is set; do some stuff 
} 

Intent alarmIntent = new Intent(MainActivity.this, AlarmReceiver.class); 
final PendingIntent pendingIntent = PendingIntent.getBroadcast(MainActivity.this, 0, alarmIntent, PendingIntent.FLAG_UPDATE_CURRENT); 

AlarmManager manager = (AlarmManager) getSystemService(Context.ALARM_SERVICE); 
manager.setInexactRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis(), 10 * 1000, pendingIntent); 

Tuttavia, alarmUp viene sempre impostato come true. Cioè, sia che io abbia impostato l'allarme o meno, ogni volta che riavvio la mia app, mi viene detto che lo alarmUp è vero (lo sto verificando facendo un Toast).

Si prega di aiutare dove sto andando male.

+1

Non ho trovato la tecnica see-if-the -PendingIntent per essere affidabile. – CommonsWare

+0

Potresti per favore elaborare, o magari fornire un link su come farlo? –

+0

Mi riferisco al codice nella tua domanda. Non ho trovato il trucco "FLAG_NO_CREATE' per essere affidabile. – CommonsWare

risposta

17

Affinché questo controllo funzioni, è necessario essere assolutamente sicuri che lo PendingIntent esista solo quando è impostato l'allarme. Ci sono 2 cose che puoi fare per assicurarti che sia così:

1) Quando collaudi il tuo codice, assicurati di disinstallare la tua applicazione e poi di reinstallare la tua applicazione prima di testarla.La disinstallazione della tua app rimuoverà tutti gli PendingIntent s che la tua app potrebbe aver creato e che sono ancora in sospeso.

2) Quando si annulla l'allarme, assicurarsi di annullare anche lo PendingIntent. È possibile farlo con

Intent alarmIntent = new Intent(MainActivity.this, AlarmReceiver.class); 
final PendingIntent pendingIntent = 
      PendingIntent.getBroadcast(MainActivity.this, 0, alarmIntent, 
      PendingIntent.FLAG_NO_CREATE); 
if (pendingIntent != null) { 
    pendingIntent.cancel(); 
} 
0

Dal API 21 è possibile utilizzare

public AlarmManager.AlarmClockInfo getNextAlarmClock() 

http://developer.android.com/reference/android/app/AlarmManager.html#getNextAlarmClock()


Ora, il codice che si sta tentando di utilizzare:

(PendingIntent.getBroadcast(MainActivity.this, 0, 
    new Intent(MainActivity.this, AlarmReceiver.class), PendingIntent.FLAG_NO_CREATE) != null); 

bassically si sta chiedendo un previus e l'intento esistente chiamato AlarmReceiver. Ma il suo AlarmReceiver si riferisce al tuo BroadcastReceiver.

Come si può vedere in questa linea della risposta che hai postato prima:

boolean alarmUp = (PendingIntent.getBroadcast(context, 0, 
    new Intent("com.my.package.MY_UNIQUE_ACTION"), 
    PendingIntent.FLAG_NO_CREATE) != null); 

Usano "MY_UNIQUE_ACTION" per vedere se esiste l'intenzione.

Anche in questo sito è possibile vedere un tutorial utilizzando tale:

http://justcallmebrian.com/2010/04/27/using-alarmmanager-to-schedule-activities-on-android/


A meno che non si è in grado di accedere al android-sistema di AlarmReceiver e vedere se esiste l'intenzione, non sarà in grado di chiedere un allarme programmato 'generico'. E 'quello che stai cercando di fare? Se è il caso, sei sicuro che nel caso in cui non hai nessun allarme programmato l'intento AlarmReceiver del sistema android non è stato comunque creato dal sistema? Sembra che non abbiamo il controllo di quei componenti come ci piace.


Un esempio di lavoro: Manifest.xml:

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
package="com.blablabla.testa" > 
<uses-permission android:name="android.permission.WAKE_LOCK" /> 

<application 
    android:allowBackup="true" 
    android:icon="@mipmap/ic_launcher" 
    android:label="@string/app_name" 
    android:theme="@style/AppTheme" > 
    <activity 
     android:name=".MainActivity" 
     android:label="@string/app_name" > 
     <intent-filter> 
      <action android:name="android.intent.action.MAIN" /> 

      <category android:name="android.intent.category.LAUNCHER" /> 
     </intent-filter> 

     <receiver 
      android:name=".MySuperReceiver" 
      android:label="MySuperReceiverName" /> 

    </activity> 
</application> 

MySuperReceiver.java

package com.blablabla.testa; 
import android.content.BroadcastReceiver; 
import android.content.Context; 
import android.content.Intent; 
import android.util.Log; 

import java.util.Calendar; 

public class MySuperReceiver extends BroadcastReceiver { 
    public static final String TAG = MySuperReceiver.class.getSimpleName(); 
    public static final String ACTION_ALARM_RECEIVER = "ACTION_ALARM_RECEIVER"; 
    private Calendar c = Calendar.getInstance(); 

    @Override 
    public void onReceive(Context context, Intent intent) { 
     if (intent != null) 
      if (ACTION_ALARM_RECEIVER.equals(intent.getAction())) { 
       Log.d(TAG, new Exception().getStackTrace()[0].getMethodName() + " " + c.getTime()); 
       //do something here 
      } 
    } 
} 

E MainActivity.java:

package com.blablabla.testa; 

import android.app.AlarmManager; 
import android.app.PendingIntent; 
import android.content.Context; 
import android.content.Intent; 
import android.support.v7.app.ActionBarActivity; 
import android.os.Bundle; 
import android.util.Log; 
import android.view.Menu; 
import android.view.MenuItem; 


public class MainActivity extends ActionBarActivity { 

    public static final String TAG = "TEST APP:"; 

    AlarmManager alarmManager; 


    @Override 
    protected void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.activity_main); 

     GetAlarmService(); 
     CreateAlarm(); 
     CheckForAlarm(); 
     CancelAlarms(); 
     CheckForAlarm(); 
    } 

    private void GetAlarmService() 
    { 
     alarmManager = (AlarmManager) getApplicationContext().getSystemService(Context.ALARM_SERVICE); 
     Log.d(TAG, " GET Alarm Service ! "); 
    } 
    private void CreateAlarm() 
    { 
     long aroundInterval = 1*60*1000; 

     Intent intent = new Intent(getApplicationContext(), MySuperReceiver.class); 
     intent.setAction(MySuperReceiver.ACTION_ALARM_RECEIVER);//my custom string action name 

     PendingIntent pendingIntent = PendingIntent.getBroadcast(getApplicationContext(), 1001, intent, PendingIntent.FLAG_CANCEL_CURRENT);//used unique ID as 1001 
     alarmManager.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis(), aroundInterval, pendingIntent);//first start will start asap 

     Log.d(TAG, " CREATE Alarm ! "); 
    } 

    private void CancelAlarms() 
    { 
     Intent intent = new Intent(getApplicationContext(), MySuperReceiver.class);//the same as up 
     intent.setAction(MySuperReceiver.ACTION_ALARM_RECEIVER);//the same as up 
     PendingIntent pendingIntent = PendingIntent.getBroadcast(getApplicationContext(), 1001, intent, PendingIntent.FLAG_CANCEL_CURRENT);//the same as up 
     alarmManager.cancel(pendingIntent);//important 
     pendingIntent.cancel();//important 

     Log.d(TAG, " Cancel Alarm ! "); 
    } 

    private void CheckForAlarm() 
    { 
     //checking if alarm is working with pendingIntent #3 
     Intent intent = new Intent(getApplicationContext() , MySuperReceiver.class);//the same as up 
     intent.setAction(MySuperReceiver.ACTION_ALARM_RECEIVER);//the same as up 
     boolean isWorking = (PendingIntent.getBroadcast(getApplicationContext() , 1001, intent, PendingIntent.FLAG_NO_CREATE) != null);//just changed the flag 
     Log.d("TAG: TEST APP: ", "alarm is " + (isWorking ? "" : "not") + " working..."); 

    } 

    @Override 
    public boolean onCreateOptionsMenu(Menu menu) { 
     // Inflate the menu; this adds items to the action bar if it is present. 
     getMenuInflater().inflate(R.menu.menu_main, menu); 
     return true; 
    } 

    @Override 
    public boolean onOptionsItemSelected(MenuItem item) { 
     // Handle action bar item clicks here. The action bar will 
     // automatically handle clicks on the Home/Up button, so long 
     // as you specify a parent activity in AndroidManifest.xml. 
     int id = item.getItemId(); 

     //noinspection SimplifiableIfStatement 
     if (id == R.id.action_settings) { 
      return true; 
     } 

     return super.onOptionsItemSelected(item); 
    } 
} 
+0

Non c'è modo che funzioni API prima di 21? Tutto quello che voglio sapere è se la mia app ha già impostato un allarme. –

+0

mmm, quindi se hai impostato un allarme usando il tuo BroadcastReceiver devi essere in grado di leggere quel registro. Ecco un altro collegamento con un altro esempio: http://blog.mokrzycki.io/2015/01/working-example-of-setting-alarm-with-repeating-stuff-checking-if-alarm-was-set-with- pendingintent/ – mayo

+0

Ho fatto la stessa cosa menzionata al punto # 2 nel tuo link. E questo è esattamente ciò che non funziona come menzionato nella mia domanda. –

Problemi correlati