|
|
/*
|
|
|
* Copyright (c) 2010-2011, The MiCode Open Source Community (www.micode.net)
|
|
|
*
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
* You may obtain a copy of the License at
|
|
|
*
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
*
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
* See the License for the specific language governing permissions and
|
|
|
* limitations under the License.
|
|
|
*/
|
|
|
|
|
|
// 包声明,表明该类所在的包名,此处在net.micode.notes.ui包下
|
|
|
package net.micode.notes.ui;
|
|
|
|
|
|
import android.app.Activity;
|
|
|
import android.app.AlertDialog;
|
|
|
import android.content.Context;
|
|
|
import android.content.DialogInterface;
|
|
|
import android.content.DialogInterface.OnClickListener;
|
|
|
import android.content.DialogInterface.OnDismissListener;
|
|
|
import android.content.Intent;
|
|
|
import android.media.AudioManager;
|
|
|
import android.media.MediaPlayer;
|
|
|
import android.media.RingtoneManager;
|
|
|
import android.net.Uri;
|
|
|
import android.os.Bundle;
|
|
|
import android.os.PowerManager;
|
|
|
import android.provider.Settings;
|
|
|
import android.view.Window;
|
|
|
import android.view.WindowManager;
|
|
|
|
|
|
import net.micode.notes.R;
|
|
|
import net.micode.notes.data.Notes;
|
|
|
import net.micode.notes.tool.DataUtils;
|
|
|
|
|
|
import java.io.IOException;
|
|
|
|
|
|
// AlarmAlertActivity类继承自Activity类,实现了OnClickListener和OnDismissListener接口,用于处理闹钟提醒相关的界面展示、声音播放等功能
|
|
|
public class AlarmAlertActivity extends Activity implements OnClickListener, OnDismissListener {
|
|
|
// 记录笔记的ID,用于后续操作中标识具体的笔记
|
|
|
private long mNoteId;
|
|
|
// 笔记的摘要内容,用于在提醒对话框中展示部分笔记内容
|
|
|
private String mSnippet;
|
|
|
// 定义摘要预览的最大长度为60个字符
|
|
|
private static final int SNIPPET_PREW_MAX_LEN = 60;
|
|
|
// 用于播放闹钟提醒声音的MediaPlayer对象
|
|
|
MediaPlayer mPlayer;
|
|
|
|
|
|
// 重写Activity的onCreate方法,在Activity创建时被调用,进行一些初始化操作
|
|
|
@Override
|
|
|
protected void onCreate(Bundle savedInstanceState) {
|
|
|
// 调用父类的onCreate方法,完成Activity创建的基本初始化流程,这是必须的操作
|
|
|
super.onCreate(savedInstanceState);
|
|
|
// 请求去除Activity的标题栏,使界面更加简洁,通常用于全屏等特定显示需求
|
|
|
requestWindowFeature(Window.FEATURE_NO_TITLE);
|
|
|
|
|
|
// 获取当前Activity的窗口对象,后续用于设置窗口相关的属性
|
|
|
final Window win = getWindow();
|
|
|
// 给窗口添加标志位,使得在屏幕锁定时也能显示该Activity,方便用户看到闹钟提醒
|
|
|
win.addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED);
|
|
|
|
|
|
// 判断屏幕是否处于开启状态
|
|
|
if (!isScreenOn()) {
|
|
|
// 如果屏幕未开启,添加一系列窗口标志位,用于控制屏幕相关的行为,比如保持屏幕常亮、点亮屏幕等,确保闹钟提醒能正常展示给用户
|
|
|
win.addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON
|
|
|
| WindowManager.LayoutParams.FLAG_TURN_SCREEN_ON
|
|
|
| WindowManager.LayoutParams.FLAG_ALLOW_LOCK_WHILE_SCREEN_ON
|
|
|
| WindowManager.LayoutParams.FLAG_LAYOUT_INSET_DECOR);
|
|
|
}
|
|
|
|
|
|
// 获取启动该Activity的Intent对象,用于获取传递过来的数据等信息
|
|
|
Intent intent = getIntent();
|
|
|
|
|
|
try {
|
|
|
// 从Intent携带的数据中获取笔记的ID,具体是从数据的路径片段中的第二个元素获取(假设数据格式符合特定规范)
|
|
|
mNoteId = Long.valueOf(intent.getData().getPathSegments().get(1));
|
|
|
// 通过工具类DataUtils根据笔记ID获取笔记的摘要内容
|
|
|
mSnippet = DataUtils.getSnippetById(this.getContentResolver(), mNoteId);
|
|
|
// 如果获取到的摘要内容长度大于预设的最大长度,进行截断处理,并添加提示信息,以合适的长度展示摘要
|
|
|
mSnippet = mSnippet.length() > SNIPPET_PREW_MAX_LEN? mSnippet.substring(0,
|
|
|
SNIPPET_PREW_MAX_LEN) + getResources().getString(R.string.notelist_string_info)
|
|
|
: mSnippet;
|
|
|
} catch (IllegalArgumentException e) {
|
|
|
// 如果在获取笔记ID或处理摘要过程中出现参数异常,打印异常堆栈信息,并且直接返回,不再继续执行后续可能出错的操作
|
|
|
e.printStackTrace();
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// 创建一个MediaPlayer对象,用于播放闹钟提醒的声音
|
|
|
mPlayer = new MediaPlayer();
|
|
|
// 判断该笔记是否在笔记数据库中可见(通过DataUtils工具类的方法进行判断)
|
|
|
if (DataUtils.visibleInNoteDatabase(getContentResolver(), mNoteId, Notes.TYPE_NOTE)) {
|
|
|
// 如果可见,展示操作对话框,给用户提供相关操作选项
|
|
|
showActionDialog();
|
|
|
// 调用方法播放闹钟提醒声音
|
|
|
playAlarmSound();
|
|
|
} else {
|
|
|
// 如果笔记不可见,直接结束该Activity,不需要进行后续的提醒相关操作
|
|
|
finish();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 方法用于判断屏幕是否处于开启状态,通过获取PowerManager服务并调用其isScreenOn方法来判断
|
|
|
private boolean isScreenOn() {
|
|
|
PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
|
|
|
return pm.isScreenOn();
|
|
|
}
|
|
|
|
|
|
// 方法用于播放闹钟提醒声音,进行一系列设置并启动声音播放
|
|
|
private void playAlarmSound() {
|
|
|
// 获取系统默认的闹钟铃声的Uri,通过RingtoneManager来获取实际使用的默认铃声的Uri
|
|
|
Uri url = RingtoneManager.getActualDefaultRingtoneUri(this, RingtoneManager.TYPE_ALARM);
|
|
|
|
|
|
// 获取系统设置中受静音模式影响的音频流相关设置,用于后续判断闹钟音频流的设置情况
|
|
|
int silentModeStreams = Settings.System.getInt(getContentResolver(),
|
|
|
Settings.System.MODE_RINGER_STREAMS_AFFECTED, 0);
|
|
|
|
|
|
// 判断静音模式下是否影响闹钟音频流,如果受影响,则按照相应设置设置MediaPlayer的音频流类型
|
|
|
if ((silentModeStreams & (1 << AudioManager.STREAM_ALARM))!= 0) {
|
|
|
mPlayer.setAudioStreamType(silentModeStreams);
|
|
|
} else {
|
|
|
// 如果不受影响,将MediaPlayer的音频流类型设置为普通的闹钟音频流类型
|
|
|
mPlayer.setAudioStreamType(AudioManager.STREAM_ALARM);
|
|
|
}
|
|
|
try {
|
|
|
// 设置MediaPlayer的数据源为获取到的闹钟铃声的Uri,指定从哪里获取音频数据来播放
|
|
|
mPlayer.setDataSource(this, url);
|
|
|
// 准备MediaPlayer,使其进入可播放状态,加载音频数据等资源
|
|
|
mPlayer.prepare();
|
|
|
// 设置MediaPlayer循环播放,使得闹钟声音一直响,直到被用户操作停止
|
|
|
mPlayer.setLooping(true);
|
|
|
// 启动MediaPlayer开始播放闹钟声音
|
|
|
mPlayer.start();
|
|
|
} catch (IllegalArgumentException e) {
|
|
|
// 如果在设置数据源等操作中出现参数异常,打印异常堆栈信息
|
|
|
// TODO Auto-generated catch block
|
|
|
e.printStackTrace();
|
|
|
} catch (SecurityException e) {
|
|
|
// 如果出现安全相关异常,比如没有权限访问音频资源等情况,打印异常堆栈信息
|
|
|
// TODO Auto-generated catch block
|
|
|
e.printStackTrace();
|
|
|
} catch (IllegalStateException e) {
|
|
|
// 如果MediaPlayer处于不正确的状态(比如已经在播放等情况)导致操作失败,打印异常堆栈信息
|
|
|
// TODO Auto-generated catch block
|
|
|
e.printStackTrace();
|
|
|
} catch (IOException e) {
|
|
|
// 如果在读取音频数据等操作中出现IO异常,比如文件不存在等情况,打印异常堆栈信息
|
|
|
// TODO Auto-generated catch block
|
|
|
e.printStackTrace();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 方法用于展示操作对话框,给用户提供针对闹钟提醒的操作选项,比如确认、进入编辑等
|
|
|
private void showActionDialog() {
|
|
|
// 创建一个AlertDialog的构建器,用于构建自定义的对话框
|
|
|
AlertDialog.Builder dialog = new AlertDialog.Builder(this);
|
|
|
// 设置对话框的标题为应用的名称,通过资源文件中的字符串资源来设置
|
|
|
dialog.setTitle(R.string.app_name);
|
|
|
// 设置对话框的消息内容为前面获取和处理后的笔记摘要内容,展示给用户提醒相关的笔记信息
|
|
|
dialog.setMessage(mSnippet);
|
|
|
// 设置对话框的确认按钮(通常是正向操作按钮)的文本及点击监听器,点击时会触发OnClickListener中的相应逻辑
|
|
|
dialog.setPositiveButton(R.string.notealert_ok, this);
|
|
|
// 判断屏幕是否开启,如果开启,设置对话框的取消按钮(通常是进入相关操作按钮)的文本及点击监听器
|
|
|
if (isScreenOn()) {
|
|
|
dialog.setNegativeButton(R.string.notealert_enter, this);
|
|
|
}
|
|
|
// 构建并显示对话框,同时给对话框设置关闭监听器,当对话框关闭时会触发相应逻辑
|
|
|
dialog.show().setOnDismissListener(this);
|
|
|
}
|
|
|
|
|
|
// 实现OnClickListener接口的方法,处理对话框按钮点击事件,根据点击的按钮执行相应的操作
|
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
|
switch (which) {
|
|
|
case DialogInterface.BUTTON_NEGATIVE:
|
|
|
// 如果点击的是取消按钮(通常是进入相关操作按钮),创建一个Intent用于启动笔记编辑Activity
|
|
|
Intent intent = new Intent(this, NoteEditActivity.class);
|
|
|
// 设置Intent的动作,这里设置为查看动作,表明是查看笔记详情的意图
|
|
|
intent.setAction(Intent.ACTION_VIEW);
|
|
|
// 将笔记的ID作为额外数据添加到Intent中,传递给NoteEditActivity,以便其能识别要编辑的具体笔记
|
|
|
intent.putExtra(Intent.EXTRA_UID, mNoteId);
|
|
|
// 启动NoteEditActivity,切换到笔记编辑界面
|
|
|
startActivity(intent);
|
|
|
break;
|
|
|
default:
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 实现OnDismissListener接口的方法,当对话框关闭时被调用,用于停止闹钟声音并结束当前Activity
|
|
|
public void onDismiss(DialogInterface dialog) {
|
|
|
// 调用方法停止闹钟声音,释放相关资源
|
|
|
stopAlarmSound();
|
|
|
// 结束当前Activity,关闭闹钟提醒界面
|
|
|
finish();
|
|
|
}
|
|
|
|
|
|
// 方法用于停止闹钟声音,释放MediaPlayer相关的资源,将其设置为null
|
|
|
private void stopAlarmSound() {
|
|
|
if (mPlayer!= null) {
|
|
|
// 停止MediaPlayer的播放
|
|
|
mPlayer.stop();
|
|
|
// 释放MediaPlayer占用的资源,如音频缓冲区等
|
|
|
mPlayer.release();
|
|
|
// 将MediaPlayer对象设置为null,避免后续误操作
|
|
|
mPlayer = null;
|
|
|
}
|
|
|
}
|
|
|
} |