安卓小项目实战之--定时提醒备忘录

来源:互联网 发布:php如何写短信接口 编辑:程序博客网 时间:2024/04/27 19:20

写在前面:

      安卓的学习也有半年多了,期间也曾写过博客,但大多都是一些琐碎的笔记,基本没用任何参考价值,这几天闲来无事,便想做个小项目来玩玩,巩固一下基本知识,并且完整的记录下来整个开发的过程,以作留念。

/————————我是华丽的分割线—————————-/
      本次项目选择了可定时提示的备忘录。原理是利用系统每分钟发送一条时间改变的广播,通过接受这条广播来判断是否为用户设置的时间,如果是则与用户交互提醒用户。
首先,在Eclipse中创建一个Android项目,取名为Notification

类:MainActivity.java

MainActivity
此Activity主要用于用户添加备忘录页面的转跳与已添加备忘录的展示。
在其布局文件activity_main.xml中加入一个Button和一个TextView 。目前还没有实现多事件的共同设置,所以暂时使用TextView显示事件。
因为储存的数据比较少,这里采用SharedPreferences进行存储,在Activity的onCreate()方法中进行成员变量的初始化操作:

        protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        init();        sp = getSharedPreferences("UserNote",MODE_PRIVATE);    }
public void init() {        button = (Button) findViewById(R.id.bt);        title = (TextView) findViewById(R.id.main_title);        note = (TextView) findViewById(R.id.main_note);        time = (TextView)findViewById(R.id.main_time);    }

在onStart()中对按钮bt添加点击事件监听,并通过startActivity(intent)启动AddActivity转跳到添加的Activity:

    protected void onStart() {        super.onStart();        button.setOnClickListener(new OnClickListener() {            @Override            public void onClick(View arg0) {                Intent intent = new Intent(MainActivity.this, AddActivity.class);                startActivity(intent);            }        });    }

在onResume()方法中,获取SharedPerfences中的数据并更新TextView的内容:

protected void onResume(){        super.onResume();        strTime = sp.getString("time","null");        strTitle = sp.getString("title","null");        strNote = sp.getString("text","null");        title.setText(strTitle);        note.setText(strNote);        time.setText(strTime);        System.out.println("onResume");    }

MainActivity.java整体代码如下:

package com.example.notification;import android.app.Activity;import android.content.Intent;import android.content.SharedPreferences;import android.os.Bundle;import android.support.v7.app.ActionBarActivity;import android.util.Log;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;import android.widget.TextView;public class MainActivity extends Activity {    private Button button;    private SharedPreferences sp;    private TextView title, note,time;    private String strTitle,strNote,strTime;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        init();        sp = getSharedPreferences("UserNote",MODE_PRIVATE);    }    @Override    protected void onStart() {        super.onStart();        button.setOnClickListener(new OnClickListener() {            @Override            public void onClick(View arg0) {                Intent intent = new Intent(MainActivity.this, AddActivity.class);                startActivity(intent);            }        });    }    protected void onResume(){        super.onResume();        strTime = sp.getString("time","null");        strTitle = sp.getString("title","null");        strNote = sp.getString("text","null");        title.setText(strTitle);        note.setText(strNote);        time.setText(strTime);        System.out.println("onResume");    }    public void init() {        button = (Button) findViewById(R.id.bt);        title = (TextView) findViewById(R.id.main_title);        note = (TextView) findViewById(R.id.main_note);        time = (TextView)findViewById(R.id.main_time);    }}
该Activity布局文件activity_main.xml如下
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent"    android:paddingBottom="@dimen/activity_vertical_margin"    android:paddingLeft="@dimen/activity_horizontal_margin"    android:paddingRight="@dimen/activity_horizontal_margin"    android:paddingTop="@dimen/activity_vertical_margin"    tools:context="com.example.notification.MainActivity"    android:background="#fcf3ea" >    <Button        android:id="@+id/bt"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:text="添加备忘录"        android:textSize="20dp" />    <TextView        android:id="@+id/setting"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:layout_below="@id/bt"        android:textSize="20dp"        android:text="" />    <LinearLayout        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:layout_alignTop="@+id/setting"        android:layout_toRightOf="@+id/setting"        android:orientation="vertical" >        <TextView            android:id="@+id/main_time"            android:layout_width="wrap_content"            android:layout_height="wrap_content"            android:textSize="20dp"            android:text="" />        <TextView            android:id="@+id/main_title"            android:layout_width="wrap_content"            android:layout_height="wrap_content"            android:textSize="20dp"            android:text="" />          <TextView            android:id="@+id/main_note"            android:layout_width="wrap_content"            android:layout_height="wrap_content"            android:textSize="20dp"            android:text="" />    </LinearLayout></RelativeLayout>

类:AddActivity.java

AddActivity
该Activity为用户添加备忘录事件的Activity,用户通过EditText和DatePicker和TimePicker进行数据的输入,单击按钮由SharedPerfences记录用户输入的数据.在onCreate()中进行初始化,在onStart()中对按钮添加监听,并且将时间选择器和用户输入的内容存入SharedPerfences,这里应当注意的是,时间选择器选择的数据如果为一位数如1月 则记录的日期为1 并非01,这点与SimpleDateFormat生成的不一致,所以我们需要自己建立一个类来规范我们的时间,这个类取名为MyTime,在存储完毕用户键入的数据以后,Activity会启动一个服务NotifyService,并且调用finish()关闭自身。

AddActivity代码如下:
package com.example.notification;import java.text.SimpleDateFormat;import java.util.Date;import android.annotation.SuppressLint;import android.app.Activity;import android.content.Intent;import android.content.SharedPreferences;import android.os.Bundle;import android.util.Log;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;import android.widget.DatePicker;import android.widget.EditText;import android.widget.TimePicker;import android.widget.Toast;public class AddActivity extends Activity {    private Button conButton;    private TimePicker tp;    private DatePicker dp;    private EditText etTitle, etNote;    private String title, text;    private SharedPreferences sp;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_add);        init();    }    protected void onStart() {        super.onStart();        conButton.setOnClickListener(new OnClickListener() {            @Override            public void onClick(View arg0) {                saveTime();            }        });    }    @SuppressLint("NewApi")    public void init() {        conButton = (Button) findViewById(R.id.add_bt_confirm);        tp = (TimePicker) findViewById(R.id.timePicker1);        dp = (DatePicker) findViewById(R.id.datePicker1);        etTitle = (EditText) findViewById(R.id.add_et_title);        etNote = (EditText) findViewById(R.id.add_et_note);        tp.setIs24HourView(true);        dp.setCalendarViewShown(false);        sp = getSharedPreferences("UserNote", MODE_PRIVATE);    }    public void saveTime() {        text = etNote.getText().toString();        title = etTitle.getText().toString();        int day = dp.getDayOfMonth();        int mon = dp.getMonth()+1;//getMonth 返回从0开始        System.out.println("setMon is"+mon);        int year = dp.getYear();        int hour = tp.getCurrentHour();        int min = tp.getCurrentMinute();        SimpleDateFormat sdf = new SimpleDateFormat("HHmm");        String time = MyTime.getTime(year,mon,day,hour, min);        System.out.println(time);        SharedPreferences.Editor editor = sp.edit();        editor.putString("text", text);        editor.putString("time", time);        editor.putString("title",title);        editor.commit();        System.out.println(time);        Intent intent = new Intent(AddActivity.this, NotifyService.class);        startService(intent);        Log.i("ADD","intent完毕");        finish();    }}
MyTime.java如下
package com.example.notification;public class MyTime {    public MyTime() {        // TODO 自动生成的构造函数存根    }    public static String getTime(int year,int mon,int day,int hour, int min) {        String mins,mons,days,hours;        if(hour<10)            hours = 0+String.valueOf(hour);        else            hours = String.valueOf(hour);        if(min<10)            mins =0+ String.valueOf(min);        else            mins = String.valueOf(min);        if(mon<10)            mons =0+ String.valueOf(mon);        else            mons = String.valueOf(mon);        if(day<10)            days = 0+String.valueOf(mon);        else            days = String.valueOf(day);        return year+":"+mons+":"+days+":"+hours+":"+mins;    }}

服务NotifyService.java

新建服务需要在AndroidManifest.xml进行注册,在节点下加入以下代码进行注册

  <service android:name=".NotifyService" ></service>

该服务主要功能为判断当前时间与用户储存时间是否一致,并进行相应动作,在onCreate()方法动态注册一个广播接收器TimeRecevier,该广播接收器接收系统时间变化的广播,该广播每分钟由系统发出一次,并在OnDestroy()方法中注销该广播接收器。具体ACTION如下(要接收该广播,必须进行动态注册):

private String ACTION = Intent.ACTION_TIME_TICK;

在onStartCommand()方法中,利用SimpleDateFormat将当前系统时间格式化与本地存储一致的格式,当前时间可以通过new Date()获取(new Date()实质上就是System.currentTimeMillis());然后进行时间的比对,如果一致,则通知栏通知用户,并且启动一个可以在锁屏状态下唤醒屏幕的Activity。这里值得注意的是,在服务中启动一个Activity必须为调用的intent添加一个Flag如下:

ActIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);  

最后调用stopSelf()方法,关闭自身.

NotifyService.java代码如下:
package com.example.notification;import java.text.SimpleDateFormat;import java.util.Date;import android.app.Notification;import android.app.NotificationManager;import android.app.PendingIntent;import android.app.Service;import android.content.Intent;import android.content.IntentFilter;import android.content.SharedPreferences;import android.os.IBinder;import android.util.Log;public class NotifyService extends Service {    private NotificationManager nm;    private boolean isRec = false;    private boolean isFirst = true;    private String ACTION = Intent.ACTION_TIME_TICK;    private SharedPreferences sp;    private IntentFilter ifter;    private TimeReceiver receiver;    @Override    public IBinder onBind(Intent arg0) {        // TODO 自动生成的方法存根        return null;    }    @Override    public void onCreate() {        // TODO 自动生成的方法存根        super.onCreate();        sp = getSharedPreferences("UserNote", MODE_PRIVATE);        nm = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);        ifter = new IntentFilter();        ifter.addAction(ACTION);        receiver = new TimeReceiver();        if (isRec == false) {            registerReceiver(receiver, ifter);            isRec = true;        }        Log.i("Service","onCreate");    }    @Override    public void onDestroy() {        super.onDestroy();        System.out.println("服务拜拜");        unregisterReceiver(receiver);    }    @SuppressWarnings("deprecation")    @Override    public int onStartCommand(Intent intent, int flags, int startId) {        // TODO 自动生成的方法存根        Log.i("Service","onStart");        if (isFirst) {            isFirst = false;        } else {            SimpleDateFormat sdf = new SimpleDateFormat("yyyy:MM:dd:HH:mm");            String curTime = sdf.format(new Date());            System.out.println(curTime);            String time =sp.getString("time", "0000");            System.out.println("curTime"+curTime+" setTime"+time);            if (curTime.equals(time)) {                String text = sp.getString("text", "默认事件");                System.out.println("onStart");                Notification notify = new Notification(R.drawable.ic_launcher,                        "小贴士", System.currentTimeMillis());                PendingIntent pi = PendingIntent.getActivity(this,0,                         new Intent(this,MainActivity.class),0);                notify.setLatestEventInfo(this, "小贴士提醒您", time+"  "+text, pi);                notify.defaults = Notification.DEFAULT_ALL;                nm.notify(1044, notify);                Intent ActIntent = new Intent(this,DialogActivity.class);                ActIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);                  startActivity(ActIntent);                stopSelf();            }        }        return super.onStartCommand(intent, flags, startId);    }}

类,广播接收器:TimeReceiver.java

整个接收器的作用只有一个,就是每次接收到广播,便启动服务NotifyService。

TimeReceiver.java如下
package com.example.notification;import android.content.BroadcastReceiver;import android.content.Context;import android.content.Intent;public class TimeReceiver extends BroadcastReceiver{    @Override    public void onReceive(Context context, Intent intent) {        System.out.println("recevice");        context.startService(new Intent(context,NotifyService.class));    }}

类DialogActivity.java

DialogActivity
该Activity为用户预设的时间到了通知用户的Activity,该Activity有如下特性:
      1.可以点亮屏幕。
      2.可以在锁屏状态下启动。
实现这两个特点并不困难,只需要加入这两句话:

 getWindow().addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED                    | WindowManager.LayoutParams.FLAG_TURN_SCREEN_ON);

另外,值得一提的是,QQ客户端的锁屏聊天窗口就是这样实现的(如图):
QQ对话框
这个窗口看似是在弹窗之上。其实本身只是一个Activity,玄机就在他的背景图片,他是一个以系统壁纸为背景的没有标题栏的Activity。因此为了不让Activity突然弹出看起来那么突兀,我们要自定义一个Theme。在AndroidManifest.xml的节点下加入以下代码

<activity            android:name=".DialogActivity"            android:label="提醒您"             android:theme="@android:style/Theme.Wallpaper.NoTitleBar"            >

在res/values/styles.xml加入自定义的Style

  <style name="FullScreenTheme" parent="@android:style/Theme.Light.NoTitleBar">        <item name="android:windowBackground">@android:color/transparent</item>        <item name="android:colorBackgroundCacheHint">@null</item>        <item name="android:windowNoTitle">true</item>        <item name="android:windowIsTranslucent">true</item>    </style>

完成

感谢你的阅读,在此我们就完成了定时提醒备忘录这个小项目,当然这个项目还有很多不足,只是一个功能的实现,如果有兴趣你可以根据源代码添加想要的功能例如多条事件的添加,还可以加上桌面组件.

源码下载地址:

http://download.csdn.net/detail/wingichoy/8717199

2 0