一.概述 * 鬧鐘功能概述:添加鬧鐘,刪除鬧鐘 * 思路: * 1.給一個button添加點擊監聽,用於添加鬧鐘 * 2.提供一個視窗進行鬧鐘時間的選擇 * 3.數據保存:對鬧鐘的數據進行保存 * 4.數據讀取:打開app的時候對鬧鐘的數據進行讀取,以便保留以前設置的鬧鐘 * 5.對鬧鐘進行刪除操作 ...
一.概述
* 鬧鐘功能概述:添加鬧鐘,刪除鬧鐘
* 思路:
* 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,
*以進行主題,資源的訪問(不過還是不懂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)源代碼,這裡就包括了上面的那個步驟
2.關於alarmManager的set方法
*第一個參數ELAPSED_REALTIME和RTC的區別是前者是從開機時為0開始算的時間,後者是從1970年那個時間開始算的
* 突然想到前一段時間不是說蘋果有一個bug,把時間設到1970年以後手機會變磚頭,其實蘋果這是採用這個時間標準的(UTC)
* 第二個參數是鬧鐘響的時間
* 第三個參數告訴鬧鐘時間到了去做什麼(叫AlarmReceiver去用PlayMusic播放音樂和顯示文字),它有一個身份證ad.getId()(不然怎麼知道按返回鍵的時候取消哪一個鬧鐘)
補充:有個setRepeat的方法,可以設置間隔多久重覆鬧鐘的,不過安卓5.0後就不支持了,
看安卓開發文檔的時候可以看到Beginning with API 19 (KITKAT) alarm delivery is inexact:
就是說為了減少喚醒次數還有提高電池的利用
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(); } }
三.補充
想深入瞭解為什麼使用構造器(Context context, AttributeSet attrs)
這是基於極客安卓鬧鐘項目的,不過視頻只是講邏輯過程,看完有點難理解
還有這個在安卓4.3運行會有重覆添加鬧鐘的bug,onTimeSet方法調用了兩次,5.0以上就沒有