Не удалось проверить, установлен ли будильник AlarmManager
Я проверяю, был ли будильник уже установлен AlarmManager, используя этот ответ.
Ниже приведен фрагмент кода.
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);
Однако alarmUp
всегда устанавливается как истина. То есть, я устанавливаю будильник или нет, всякий раз, когда я перезапускаю приложение, он говорит мне, что alarmUp
является истинным (я проверяю его, создавая Toast
).
Пожалуйста, помогите, где я ошибаюсь.
Ответы
Ответ 1
Чтобы эта проверка работала, вы должны быть абсолютно уверены, что PendingIntent
существует только в том случае, если установлен будильник. Есть две вещи, которые вы можете сделать, чтобы убедиться в этом:
1) При тестировании кода убедитесь, что вы удалили приложение, а затем повторно установите приложение перед его тестированием. Удаление вашего приложения приведет к удалению любого PendingIntent
, который может создать ваше приложение, которое все еще ожидается.
2) Когда вы отмените будильник, убедитесь, что вы также отменили PendingIntent
. Вы можете сделать это с помощью
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();
}
Ответ 2
Так как API 21 вы можете использовать
public AlarmManager.AlarmClockInfo getNextAlarmClock ()
http://developer.android.com/reference/android/app/AlarmManager.html#getNextAlarmClock()
Теперь, код, который вы пытаетесь использовать:
(PendingIntent.getBroadcast(MainActivity.this, 0,
new Intent(MainActivity.this, AlarmReceiver.class), PendingIntent.FLAG_NO_CREATE) != null);
Бассивно вы просите о превиуме и существующем намерении под названием AlarmReceiver. Но его AlarmReceiver ссылается на ваш собственный BroadcastReceiver.
Как вы можете видеть в этой строке ответа, который вы отправили ранее:
boolean alarmUp = (PendingIntent.getBroadcast(context, 0,
new Intent("com.my.package.MY_UNIQUE_ACTION"),
PendingIntent.FLAG_NO_CREATE) != null);
Они используют "MY_UNIQUE_ACTION", чтобы узнать, существует ли намерение.
Также на этом сайте вы можете увидеть учебное пособие, используя это:
http://justcallmebrian.com/2010/04/27/using-alarmmanager-to-schedule-activities-on-android/
Если вы не можете получить доступ к системному сигналу AlarmReceiver и посмотреть, существует ли намерение, вы не сможете запросить "общий" будильник. Это то, что вы пытаетесь сделать? Если это так, вы уверены, что в случае, если у вас нет какой-либо запланированной тревоги, в любом случае система AlarmReceiver не создавалась системой?
Кажется, у нас нет контроля над теми компонентами, которые нам нравятся.
Рабочий пример:
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
}
}
}
И 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);
}
}