Android:Moring-早安闹钟开发过程记录(一)

该文章用于记录整理开发早安闹钟过程的思路及知识点

1.程序介绍

想做这样一个APP源于我经常在睡梦中轻易的把闹钟关掉,导致起不来,各种迟到,并且醒了以后完全不知道闹钟竟然响过,而且经常早上时间来不及了,穿好衣服出了门才发现不是太冷就是太热。

2. 界面布局

为了让HomeActivity的逻辑不会过于复杂,添加了三个Fragment,独立完成各自的逻辑。

  • FragmentAlarm

    用于显示闹钟的相关信息,并处理相关逻辑。核心控件为一个ListView,
    在Item中使用RelativeLayout实现点击出现编辑菜单。

    FragmentAlarm.jpg


    附上菜单动画的代码

    RelativeLayout rl_main = (RelativeLayout) view.findViewById(R.id.rl_main_item);LinearLayout ll_button = (LinearLayout) view.findViewById(R.id.ll_button_item);
    final Button bt_delete = (Button) view.findViewById(R.id.bt_delete_item);final Button bt_update = (Button) view.findViewById(R.id.bt_update_item);
    final SwitchButton sb= (SwitchButton) view.findViewById(R.id.bt_turn_item);
    float back = ll_button.getWidth();
    float front = rl_main.getWidth();float width = back / front;
    ll_button.setMinimumHeight(rl_main.getHeight());
    Log.d("alarm", width + "宽度");TranslateAnimation ta;
    if(isMenuOn){    
        //开着的就关    
          ta= new TranslateAnimation(Animation.RELATIVE_TO_SELF, width, Animation.RELATIVE_TO_SELF, 0f,Animation.RELATIVE_TO_SELF, 0f, Animation.RELATIVE_TO_SELF, 0f);    
          onMenu=null;    
          isMenuOn=false;
    }else{    
          ta= new TranslateAnimation(Animation.RELATIVE_TO_SELF, 0f, Animation.RELATIVE_TO_SELF, width, Animation.RELATIVE_TO_SELF, 0f, Animation.RELATIVE_TO_SELF, 0f); 
          onMenu=view;    
          isMenuOn=true;
    }
    ta.setDuration(200);
    ta.setFillAfter(true);

    View onMenu用来接收当前有菜单开启的item,便于点击下一个Item的时候关闭该菜单,Boolean isMenuOn用于判断当前是否有菜单开启

  • FragmentWeather

     用于显示天气的相关信息,并处理相关逻辑。
     布局较为简单不进行赘述。

    *FragmentMenu

      用于显示侧滑菜单,侧滑菜单核心控件为ListView

    3.功能分析

  • 闹钟功能模块
    -添加闹钟
        核心逻辑:将用户添加的闹钟的各种信息记录到数据库中,同时在点
        击完成的时候将数据传给FragmentAlarm,刷新UI。
        用到的类AlarmInfo.class(闹钟信息对象),AlarmInfoDao(数据库读取工具类)。
    -修改闹钟
        与添加闹钟相同,返回数据时通过不同的RequestCode来与添加闹钟区分
    -启动闹钟定时任务
    AlarmManager
    通过getSystemService(ALARM_SERVICE)获取到AlarmManager对象
    AlarmManger中有set()和setRepeating()方法用于一次性定时任务和重复定时任务
    但在实际测试中发现会有不同程度的delay,(PS.小米真的很严重)
    通过查阅官方文档可知
    在API 19以后这两个方法将不再精准,如果需要使用精确定时任务,应该使用setExact();(API 19以前没有问题)
    -因此在设置定时任务时加一个判断(由于该方法在多处进行调用,我把它单独封装成了一个类)
    public void startAlarm(AlarmManager mAlamManager, PendingIntent pi){   
          //设置定时任务触发的时间 
          Calendar c=Calendar.getInstance();                     
          c.set(Calendar.HOUR_OF_DAY,alarmInfo.getHour());    
          c.set(Calendar.MINUTE,alarmInfo.getMinute());    
          c.set(Calendar.SECOND,0);    
          c.set(Calendar.MILLISECOND, 0);    
      //  Log.d("alarm", "当前系统版本" + Build.VERSION.SDK_INT);    
          if(c.getTimeInMillis()<System.currentTimeMillis()){        
              if(Build.VERSION.SDK_INT>=19) {            
                  mAlamManager.setExact(AlarmManager.RTC_WAKEUP, c.getTimeInMillis() + 24 * 60 * 60 * 1000, pi);        
              }else{            
                  mAlamManager.set(AlarmManager.RTC_WAKEUP,     c.getTimeInMillis() + 24 * 60 * 60 * 1000, pi);
              }    
          }else{
               if(Build.VERSION.SDK_INT>=19) {                        
                   mAlamManager.setExact(AlarmManager.RTC_WAKEUP, c.getTimeInMillis(), pi);
          }else{
                   mAlamManager.set(AlarmManager.RTC_WAKEUP, c.getTimeInMillis(), pi);
          } 
       }
    }
    -问题来了:setExact();只能执行一次性定时任务如何实现重复闹钟?
    这里我最后采取得方法是在AlarmReceiver中再定一次相同的定时任务,这样在每次收到定时广播后,又重新设定一个相同的定时任务,就能达到重复精确定时的效果。
        另外,在判断闹钟应该在那几天重复,我写的逻辑是,除了一次性闹钟外,所有的重复闹钟都会在每天执行定时广播
        在AlarmReceiver中去判断今天的dayOfWeek和闹钟信息的dayOfWeek是否吻合
        吻合就启动闹钟service,否则重新定时
    用于判断当前dayOfweek的代码
    Calendar calendar=Calendar.getInstance();
    calendar.setTimeInMillis(System.currentTimeMillis());
    int currentDay=calendar.get(Calendar.DAY_OF_WEEK)-1;
    for(int i=0;i<dayOfWeek.length;i++){
      if(currentDay==dayOfWeek[i]){
          wakePhoneAndUnlock();//点亮屏幕并解锁
          ringAlarm();
      }
    这里用到了点亮屏幕和解锁,也贴一下代码,这段代码是我在CSDN上学到的不知道哪位大大的代码(那天脑袋晕晕的真的忘了T_T,如果原作者看到并且希望我删除或者怎么样的请联系我,我会照做的)
      //点亮屏幕并解锁    private void wakePhoneAndUnlock() {
          PowerManager pm = (PowerManager)context.getSystemService(Context.POWER_SERVICE);
          PowerManager.WakeLock mWakelock = pm.newWakeLock(PowerManager.ACQUIRE_CAUSES_WAKEUP | PowerManager.FULL_WAKE_LOCK, "WakeLock");
          mWakelock.acquire();//唤醒屏幕//......
          KeyguardManager mManager = (KeyguardManager)context.getSystemService(Context.KEYGUARD_SERVICE);
          KeyguardManager.KeyguardLock mKeyguardLock = mManager.newKeyguardLock("Lock");
          mKeyguardLock.disableKeyguard();
          mWakelock.release();
    //释放
      }
    到这里闹钟的基本定时功能就完成了,另外在AlarmReceiver中加入了开机广播的监听,来开启应该要开启的闹钟
    //开机时检查是否有闹钟需要开启
    private void checkAndStartAlarm() {
      Log.d("alarm","开始检查是否有闹钟");
      AlarmInfoDao dao=new AlarmInfoDao(context);
      ArrayList<AlarmInfo> list= (ArrayList<AlarmInfo>) dao.getAllInfo();
      AlarmClock clock=new AlarmClock(context);
      for (AlarmInfo alarmInfo:list) {
          if(PrefUtils.getBoolean(context,alarmInfo.getId(),true)){
              Log.d("alarm","有闹钟,开启");
              clock.turnAlarm(alarmInfo,true);
          }
      }
    }
    //请自动忽略掉Log信息

    结尾

    到这里闹钟的定时任务就全部完成,在下一篇文章中分析对话框和天气早知道功能的实现
    未完待续...
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值