在开发具有提醒功能的笔记应用时,一个常见的挑战是:当用户点击弹出的笔记提醒通知时,如何确保应用能打开并显示正确的笔记详情。直接传递recyclerview的列表位置是不可靠的,因为笔记的增删改查可能导致其在列表中的位置发生变化。最健壮的方法是传递一个唯一标识符(如笔记在数据库中的id),然后由目标activity根据此id加载数据。
整个流程涉及三个主要组件:
关键在于数据传递:从设置闹钟的地方将笔记的唯一ID放入AlarmManager触发的Intent中,然后在BroadcastReceiver中取出该ID,再将其放入启动目标Activity的Intent中。
在设置闹钟的方法(例如setAlarm)中,我们需要将要提醒的笔记的唯一ID(例如noteId)添加到发送给AlarmReceiver的Intent中。
import android.app.AlarmManager; import android.app.PendingIntent; import android.content.Context; import android.content.Intent; import android.widget.Toast; import java.util.Calendar; import androidx.appcompat.app.AppCompatActivity; // 假设在Activity中调用 public class MainActivity extends AppCompatActivity { // 假设在MainActivity中设置闹钟 private AlarmManager alarmManager; private PendingIntent pendingIntent; // 假设这些值在实际应用中会动态获取 private int hour = 10; private int minute = 30; private String noteId = "unique_note_id_123"; // 假设这是笔记的唯一ID private String noteTitle = "我的重要笔记"; // 假设这是笔记的标题,用于通知显示 private void setAlarm() { Calendar calendar = Calendar.getInstance(); calendar.set(Calendar.HOUR_OF_DAY, hour); // 使用HOUR_OF_DAY以避免AM/PM问题 calendar.set(Calendar.MINUTE, minute); calendar.set(Calendar.SECOND, 0); // 清零秒和毫秒,确保精确 alarmManager = (AlarmManager) getSystemService(Context.ALARM_SERVICE); Intent intent = new Intent(this, AlarmReceiver.class); // 将笔记的唯一ID和标题作为额外数据放入Intent intent.putExtra("note_id", noteId); intent.putExtra("note_title", noteTitle); // PendingIntent.getBroadcast的requestCode需要是唯一的,以便区分不同的闹钟 // 这里使用noteId的哈希值作为requestCode,以确保每个笔记的闹钟是独立的 pendingIntent = PendingIntent.getBroadcast( this, noteId.hashCode(), // 使用笔记ID的哈希值作为requestCode intent, PendingIntent.FLAG_UPDATE_CURRENT | PendingIntent.FLAG_IMMUTABLE // 添加FLAG_IMMUTABLE ); // 设置精确闹钟 alarmManager.setExact(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), pendingIntent); Toast.makeText(this, "闹钟已设置", Toast.LENGTH_SHORT).show(); } }
注意事项:
当AlarmManager触发闹钟时,AlarmReceiver的onReceive方法会被调用。在这里,我们需要从接收到的Intent中取出笔记ID和标题,然后用这些数据构建一个新的Intent来启动目标Activity并显示通知。
import android.app.PendingIntent;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import androidx.core.app.NotificationCompat;
import androidx.core.app.NotificationManagerCompat;
public class AlarmReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
// 从接收到的Intent中获取笔记ID和标题
String receivedNoteId = intent.getStringExtra("note_id");
String receivedNoteTitle = intent.getStringExtra("note_title");
// 构建启动目标Activity的Intent
Intent targetActivityIntent = new Intent(context, ChecklistChildActivity.class);
// 设置Activity启动标志,确保新任务栈和清除顶部Activity
targetActivityIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP);
// 将获取到的笔记ID和标题重新放入启动目标Activity的Intent中
targetActivityIntent.putExtra("note_id", receivedNoteId);
targetActivityIntent.putExtra("note_title", receivedNoteTitle); // 可以用于Activity内的显示
// 创建PendingIntent,当用户点击通知时,会启动targetActivityIntent
// 请求码同样使用笔记ID的哈希值,确保唯一性
PendingIntent pendingIntent = PendingIntent.getActivity(
context,
receivedNoteId.hashCode(), // 使用笔记ID的哈希值作为requestCode
targetActivityIntent,
PendingIntent.FLAG_UPDATE_CURRENT | PendingIntent.FLAG_IMMUTABLE // 添加FLAG_IMMUTABLE
);
// 构建通知
// "alarmChannel" 需要在您的应用中预先定义通知渠道
NotificationCompat.Builder builder = new NotificationCompat.Builder(context, "alarmChannel")
.setContentTitle(receivedNoteTitle != null ? receivedNoteTitle : "提醒") // 使用笔记标题作为通知标题
.setContentText("是时候查看您的笔记了。") // 通知内容
.setAutoCancel(true) // 用户点击后自动取消通知
.setDefaults(NotificationCompat.DEFAULT_ALL) // 默认震动、声音、灯光
.setSmallIcon(R.drawable.