編輯:關於android開發
一.概述
* 鬧鐘功能概述:添加鬧鐘,刪除鬧鐘
* 思路:
* 1.給一個button添加點擊監聽,用於添加鬧鐘
* 2.提供一個窗口進行鬧鐘時間的選擇
* 3.數據保存:對鬧鐘的數據進行保存
* 4.數據讀取:打開app的時候對鬧鐘的數據進行讀取,以便保留以前設置的鬧鐘
* 5.對鬧鐘進行刪除操作
* 6.鬧鐘響的時候的操作:鈴聲響,顯示一個文字界面
* 你將了解到:
* 1.SharedPreferences的使用
* 2.onFinishInflate方法
* 3.日期的基本操作
* 4.ListView
* 5.自定義視圖
* 6.廣播和消息處理
* 7.如何啟動一個activity
* 8.TabHost視圖的使用
效果圖

二.代碼
AlarmClock
public class AlarmClock extends LinearLayout{
private ListView alarmList;
private Button btn_addAlarm;
private static final String KEY_ALARM_LIST = "alarmlist";
private ArrayAdapter<AlarmData> adapter;
private AlarmManager alarmManager;
//系統會調用兩個構造器
public AlarmClock(Context context, AttributeSet attrs) {
super(context, attrs);
// TODO Auto-generated constructor stub
init();
}
public AlarmClock(Context context) {
super(context);
// TODO Auto-generated constructor stub
init();
}
/*
* getContext的左右是返回當前正在運行中的view中的context,
*以進行主題,資源的訪問
*/
private void init(){
alarmManager = (AlarmManager) getContext().getSystemService(Context.ALARM_SERVICE);
}
@Override
protected void onFinishInflate() {
// TODO Auto-generated method stub
super.onFinishInflate();
alarmList = (ListView) findViewById(R.id.alarmList);
btn_addAlarm = (Button) findViewById(R.id.btn_addAlarm);
adapter = new ArrayAdapter<AlarmClock.AlarmData>(getContext(), android.R.layout.simple_list_item_1);
alarmList.setAdapter(adapter);
readSaveAlarm();
btn_addAlarm.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
addAlarm();
}
});
//給鬧鐘列表設置長按監聽,彈出窗口可以對鬧鐘進行刪除操作
alarmList.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() {
@Override
public boolean onItemLongClick(AdapterView<?> parent, View view,
final int position, long id) {
// TODO Auto-generated method stub
new AlertDialog.Builder(getContext()).setTitle("操作選項").setItems(new CharSequence[]{"刪除"}, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
//which指的是,彈出來的窗口中item的編號,0代表第一個,這裡我們只有一個刪除按鍵所有就用
switch (which) {
case 0:
deleteAlarm(position);
break;
default:
break;
}
}
}).setNeutralButton("取消",null).show();
return true;
}
});
}
//添加鬧鐘
private void addAlarm(){
final Calendar c = Calendar.getInstance();
new TimePickerDialog(getContext(), new OnTimeSetListener() {
@Override
public void onTimeSet(TimePicker view, int hourOfDay, int minute) {
// TODO Auto-generated method stub
Calendar calendar = Calendar.getInstance();
//這裡是把你在dialog中設定的時間傳進calendar這個對象裡面
calendar.set(Calendar.HOUR_OF_DAY, hourOfDay);
calendar.set(Calendar.MINUTE, minute);
calendar.set(Calendar.SECOND, 0);
calendar.set(calendar.MILLISECOND, 0);
//這裡比較你設定的時間和系統的時間,如果是小於系統的時間就延遲24小時執行
if (calendar.getTimeInMillis() <= c.getTimeInMillis()) {
calendar.setTimeInMillis( c.getTimeInMillis() + 24*60*60*1000);
}
AlarmData ad = new AlarmData(calendar.getTimeInMillis());
adapter.add(ad);
alarmManager.set(AlarmManager.RTC_WAKEUP,ad.getTime(), PendingIntent.getBroadcast(getContext(), ad.getId(), new Intent(getContext(),AlarmReceiver.class), 0));
saveAlarmList();
}
}, c.get(Calendar.HOUR_OF_DAY), c.get(Calendar.MINUTE), true).show();
}
//保存添加的鬧鐘數據
private void saveAlarmList(){
Editor editor = getContext().getSharedPreferences("Alarm", Context.MODE_PRIVATE).edit();
StringBuffer sBuffer = new StringBuffer();
//getCount表示這個adapter裡面有多少item,就是有多少鬧鐘
//在這裡遍歷所有的鬧鐘,並把所有的AlarmData裡面的time值作為字符串放進sBuffer裡面
//用逗號隔開不同的鬧鐘time值---讀取數值的可以用sqlite(",")獲得不同的鬧鐘值
for(int i = 0; i < adapter.getCount(); i ++ ){
sBuffer.append(adapter.getItem(i).getTime()).append(",");
}
if(sBuffer.length() > 1){
String content = sBuffer.toString();
editor.putString(KEY_ALARM_LIST, content);
}else {
editor.putString(KEY_ALARM_LIST, null);
}
editor.commit();
}
//讀取已經保存的鬧鐘
private void readSaveAlarm(){
SharedPreferences sp = getContext().getSharedPreferences("Alarm", Context.MODE_PRIVATE);
String content = sp.getString(KEY_ALARM_LIST, null);
//這裡需要判斷,不然沒鬧鐘數據的時候會有空指針異常
if (content != null) {
//這裡取得每一個鬧鐘的time添加到數組裡
String[] time = content.split(",");
//這是增強for循環,這裡就和addAlarm相似,把數據添加到適配器adapter中就可以顯示了
for(String string : time){
adapter.add(new AlarmData(Long.parseLong(string)));
}
}
}
//刪除鬧鐘
public void deleteAlarm(int position){
AlarmData ad = adapter.getItem(position);
adapter.remove(ad);
saveAlarmList();
alarmManager.cancel(PendingIntent.getBroadcast(getContext(), ad.getId(), new Intent(getContext(), AlarmReceiver.class), 0));
}
//鬧鐘的數據,用一個類要保存,這是常用的做法
private static class AlarmData{
private long time = 0;
private String timeLable = "";
private Calendar date;
public AlarmData(long time){
this.time = time;
date = Calendar.getInstance();
date.setTimeInMillis(time);
timeLable = String.format("%d月%d日 %d:%d",date.get(Calendar.MONTH)+1,date.get(Calendar.DAY_OF_MONTH),
date.get(Calendar.HOUR_OF_DAY),date.get(Calendar.MINUTE));
}
public long getTime(){
return time;
}
public String getTimeLable(){
return timeLable;
}
public String toString(){
return getTimeLable();
}
//為了給每一個鬧鐘設定一個標識,方便取消鬧鐘的使用能知道是哪一個鬧鐘
public int getId(){
return (int)(getTime()/1000/60);
}
}
}
1.onFinishInflate是布局文件加載完回調的方法 就是執行完getLayoutInflater().inflate(R.layout.activity_main, null)後會調用onFinishInflate方法 這裡為什麼沒有呢,可以查看setContentView(R.layout.activity_main)源代碼,這裡就包括了上面的那個步驟*第一個參數ELAPSED_REALTIME和RTC的區別是前者是從開機時為0開始算的時間,後者是從1970年那個時間開始算的 * 突然想到前一段時間不是說蘋果有一個bug,把時間設到1970年以後手機會變磚頭,其實蘋果這是采用這個時間標准的(UTC) * 第二個參數是鬧鐘響的時間 * 第三個參數告訴鬧鐘時間到了去做什麼(叫AlarmReceiver去用PlayMusic播放音樂和顯示文字),它有一個身份證ad.getId()(不然怎麼知道按返回鍵的時候取消哪一個鬧鐘)
2.關於alarmManager的set方法
看安卓開發文檔的時候可以看到Beginning with API 19 (KITKAT) alarm delivery is inexact: 就是說為了減少喚醒次數還有提高電池的利用
補充:有個setRepeat的方法,可以設置間隔多久重復鬧鐘的,不過安卓5.0後就不支持了,
MainActivity
public class MainActivity extends Activity {
private TabHost tabHost;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tabHost = (TabHost) findViewById(android.R.id.tabhost);
//addTab之前要setup();
tabHost.setup();
tabHost.addTab(tabHost.newTabSpec("tab_time").setIndicator("時鐘").setContent(R.id.tab_time));
tabHost.addTab(tabHost.newTabSpec("tab_alarm").setIndicator("鬧鐘").setContent(R.id.tab_alarm));
tabHost.addTab(tabHost.newTabSpec("tab_timer").setIndicator("計時器").setContent(R.id.tab_timer));
tabHost.addTab(tabHost.newTabSpec("tab_stopwatch").setIndicator("秒表").setContent(R.id.tab_stopwatch));
}
}
AlarmReceiver
public class AlarmReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
// TODO Auto-generated method stub
AlarmManager alarmManager = (AlarmManager) context.getSystemService(context.ALARM_SERVICE);
alarmManager.cancel(PendingIntent.getBroadcast(context, getResultCode(), new Intent(context , AlarmReceiver.class), 0));
//啟動響鬧鐘的界面
Intent i = new Intent(context, PlayMusic.class);
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(i);
}
}
PlayMusic
public class PlayMusic extends Activity{
private MediaPlayer mp;
@Override
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
setContentView(R.layout.music_play);
mp = MediaPlayer.create(this, R.raw.music);
mp.start();
}
@Override
protected void onPause() {
// TODO Auto-generated method stub
super.onPause();
finish();
}
@Override
protected void onDestroy() {
// TODO Auto-generated method stub
super.onDestroy();
mp.stop();
mp.release();
}
}
在res文件夾下新建了一個文件夾raw,裡面放置了一個名字為muusic的音頻
三.補充
想深入了解PendingIntent的
想深入了解為什麼使用構造器(Context context, AttributeSet attrs)
這是基於極客安卓鬧鐘項目的,不過視頻只是講邏輯過程,看完有點難理解
下載整個項目(只實現顯示時間和鬧鐘功能)
還有這個在安卓4.3運行會有重復添加鬧鐘的bug,onTimeSet方法調用了兩次,5.0以上就沒有
android 5.X Toolbar+DrawerLayout實現抽屜菜單
android 5.X Toolbar+DrawerLayout實現抽屜菜單 前言 ?android5.X新增的一個控件Toolbar,這個控件比ActionBar更
硅谷新聞5--頂部新聞輪播圖事件處理,硅谷5--
硅谷新聞5--頂部新聞輪播圖事件處理,硅谷5--重寫dispatchTouchEvent,並且要在按下的時候 getParent().requestDisallowInt
Android 100多個Styles快速開發布局XML,一行搞定View屬性,一鍵統一配置UI...,androidui..
Android 100多個Styles快速開發布局XML,一行搞定View屬性,一鍵統一配置UI...,androidui.. Android開發中大量使用X
Android應用開發教程之二十七:使用Sencha完成APK局部更新
今天找時間來更新下這段時間學到的,也是我個人覺得不錯的一個東西“Android APK