详解安卓项目-闹钟

一.概述
* 闹钟功能概述:添加闹钟,删除闹钟

* 思路:

* 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();
    }

}

三.补充

想深入了解PendingIntent的

想深入了解为什么使用构造器(Context context, AttributeSet attrs)

这是基于极客安卓闹钟项目的,不过视频只是讲逻辑过程,看完有点难理解

下载整个项目(只实现显示时间和闹钟功能)

还有这个在安卓4.3运行会有重复添加闹钟的bug,onTimeSet方法调用了两次,5.0以上就没有

时间: 2024-08-29 03:04:13

详解安卓项目-闹钟的相关文章

net core 中间件详解及项目实战

net core 中间件详解及项目实战 前言 在上篇文章主要介绍了DotNetCore项目状况,本篇文章是我们在开发自己的项目中实际使用的,比较贴合实际应用,算是对中间件的一个深入使用了,不是简单的Hello World,如果你觉得本篇文章对你有用的话,不妨点个[推荐]. 目录 中间件(Middleware)的作用 中间件的运行方式 中间件(Middleware)和过滤器(Filter)的区别 什么情况我们需要中间件 怎么样自定义自己的中间件 中间件(Middleware)的作用 我们知道,任何

详解Maven项目利用java service wrapper将Java程序生成Windows服务

在项目的开发中,有时候需要将Java应用程序打包成Windows服务,我们就直接可以通过windows的服务来启动和关闭java程序了. 本博文将通过有两种方法实现该功能,手动创建法和Maven自动打包法. 一. 准备 下载java service wrapper 网址:http://sourceforge.net/projects/wrapper/或http://wrapper.tanukisoftware.com/doc/english/download.jsp 我下载的版本是wrapper

详解安卓Fragment(碎片化)

Fragment从字面意思理解就是碎片的意思,当然是为了解决安卓各类设备碎片化严重的问题,比如同样一个App在手机上显示效果还不错,但是一旦上了16:9的平板立刻就变了味,使用安卓平板的同学可能体(bei)会(keng)更深,为此Google官方从android 3.0(对应API 11)引入Fragment,简单理解就是把界面分割成很多碎片,然后根据实际要求最后选择性的进行拼接,比如在手机竖屏模式下只能显示一本书的目录列表,但是如果是横屏模式(也可以立即为平板模式)下就可以在屏幕的左半边显示目

详解-vue项目中的文件和目录

可以用vue-cli来支持一个项目. 建议使用npm 3+更高效的依赖关系树: $ npm install -g vue-cli $ vue init webpack my-project $ cd my-project $ npm install $ npm run dev 项目结构: . ├── build/ # webpack配置文件 │ └── ... ├── config/ │ ├── index.js # 主要项目配置 │ └── ... ├── src/ │ ├── main.js

Android开发 详解开源项目CircleImageView

之前的项目中在需要显示圆形头像的地方,使用到了Github上的优秀开源项目CircleImageView https://github.com/hdodenhof/CircleImageView 今天仔细地研究了这个项目,学习其原理,收获不少. 源码主要分为下面这几大部分 取图片Bitmap 取自定义属性 创建Paint画笔 计算内外圆半径 Canvas绘制内外圆形 一.取图片Bitmap CircleImageView继承自ImageView 重写了下面几个方法 @Override publi

ASP.NET Core 中间件详解及项目实战

前言 在上篇文章主要介绍了DotNetCore项目状况,本篇文章是我们在开发自己的项目中实际使用的,比较贴合实际应用,算是对中间件的一个深入使用了,不是简单的Hello World,如果你觉得本篇文章对你有用的话,不妨点个[推荐]. 目录 中间件(Middleware)的作用 中间件的运行方式 中间件(Middleware)和过滤器(Filter)的区别 什么情况我们需要中间件 怎么样自定义自己的中间件 中间件(Middleware)的作用 我们知道,任何的一个web框架都是把http请求封装成

.Net工程详解及项目版本管理

前言 写这篇文章的目地是为了让更多的小伙伴对VS生成的工程有一个清晰的认识.在开发过程中,为了赶进度,并不是每个人都有学习的时间. 但如果上层项目管理人员对这些工程目录不熟悉的话,把VS编译的中间文件也提库到SVN,那么其它同事在提代码时,就会发现有N多文件被修改了,其实有些目录及文件是没必要进行版本控制的. 说说为什么最近会写一些C#客户端的文章(包括WinForm和WPF),我所在公司游戏项目的需求,需要一些小工具和游戏微端,我们是在windows下开发,自然就会想到微软大哥的客户端技术,自

docker-compose.yml 配置文件详解及项目发布

摘自:https://blog.csdn.net/qq_36148847/article/details/79427878 docker部署tomcat项目 1.上传war包2.制作镜像 Dockerfile3.调用镜像启动新的容器更新升级1.docker stop item2. docker rm item3.docker-compose -f docker-compose.yml up -d item 一. 前言关于 docker compose 技术可以查看官方文档 Docker Comp

Web服务器项目详解 - 00 项目概述

目录 00 项目概述 01 线程同步机制包装类 02 半同步/半反应堆线程池(上) 03 半同步/半反应堆线程池(下) 04 http连接处理(上) 05 http连接处理(中) 06 http连接处理(下) 07 定时器处理非活动连接(上) 08 定时器处理非活动连接(下) 09 日志系统(上) 10 日志系统(下) 11 数据连接池 12 注册和登录校验 13 服务器测试 14 项目遇到的问题及解决方案 15 项目涉及的常见面试题 功能 Linux下C++轻量级Web服务器基于C/C++语言