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

FragmentMenu侧滑菜单界面实现

1.界面实现


侧滑菜单.jpg
  • 界面显示比较简单,因为用的SlidingMenu的第三方库所以实现起来比较简单
    github地址:https://github.com/jfeinstein10/SlidingMenu
    -在新建一个侧滑菜单的布局文件,布局写成什么样菜单就长什么样
    显示菜单的Activity需要继承SlidingFragmentActivity
    setBehindContentView(R.layout.menu_home);
    //获取到slidingMenu对象SlidingMenu slidingMenu = getSlidingMenu();
    slidingMenu.setMode(SlidingMenu.RIGHT);//设置菜单在哪一边
    slidingMenu.setShadowWidthRes(R.dimen.shadow_width);
    //设置阴影图片
    slidingMenu.setShadowDrawable(R.drawable.shadow);
    //设置为全屏拉出菜单
    slidingMenu.setTouchModeAbove(SlidingMenu.TOUCHMODE_FULLSCREEN);
    //设置菜单宽度
    slidingMenu.setBehindOffset(200);
    侧滑菜单就可以愉快的开始工作了

2.功能实现

天气和震动比较简单,用SP记录一下用户的选择就OK了
比较复杂的是更换城市和音量改变

  • 更换城市


    changecity.jpg
    • 在用户输入的时候给用户动态提示
      • 配置数据库
        首先配置好数据库,在第一次启动应用的时候将数据库拷贝到本地
        in=getResources().getAssets().open(dbName);
        out=new FileOutputStream(file);
        byte[] buffer=new byte[1024];
        int len=0;
        while((len=in.read(buffer))!=-1){
            out.write(buffer, 0, len);
        }
        定义一个工具类进行查找数据库,这里需要进行模糊查找
        "select 字段名 from 表名 where 字段名like '%" + 用户输入的字符+ "%'"
        将包含该字符的数据全部找出,返回String列表
        public ArrayList<String> find(String input){
            String path="data/data/com.joe.lazyalarm/files/china_Province_city_zone.db";
            SQLiteDatabase sql=SQLiteDatabase.openDatabase(path, null, SQLiteDatabase.OPEN_READONLY);
            //模糊查询
            ArrayList<String> cityList=new ArrayList<String>();
            Cursor cursor=sql.rawQuery("select CityName from T_City where CityName like '%" + input + "%'", null);
            while (cursor.moveToNext()){
                String cityName=cursor.getString(cursor.getColumnIndex("CityName"));
                cityList.add(cityName);
            }
            sql.close();
            cursor.close();
            return cityList;
        }
      • 界面实现
        Android自己有一个控件叫AutoCompleteTextView-自动补全文本编辑框,用这个控件可以实现这个效果,与数据库建立连接就可以了。
        我没有用AutoCompleteTextView,而是自己实现这个效果。分析一下这个控件的效果不难得出就是一个EditText和一个ListView就可以实现
        • 布局文件:
          <RelativeLayout 
          xmlns:android="http://schemas.android.com/apk/res/android"    
          android:layout_width="match_parent"    
          android:layout_height="match_parent"    >
            <EditText
                android:id="@id/et_change_city"
                android:hint="例如:重庆"
                android:layout_width="match_parent"
                android:layout_height="wrap_content" />
            <ListView
                android:visibility="gone"
                android:layout_below="@id/et_change_city"
                android:id="@id/lv_change_city"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"/>
          </RelativeLayout>
          说明:我是通过setView()将布局添加到对话框中,为了省事并没有让ListView悬浮在对话框上面,如果需要悬浮的话改一下布局文件就可以了
          因为ListView没有悬浮,必然会占用dialog的宽高,所以在初始化的时候将ListView的状态设置为gone,不设置也没有关系,因为代码中会设置输入一个字符后才开启提示
      • 代码实现
        首先将布局塞给对话框
        dao = new CityDao();//数据库读取工具
        cityList = new ArrayList<String>();//查找到的城市列表
        final View autoLayout = View.inflate(mActivity, R.layout.auto_edit_view, null);
        final EditText autoText = (EditText) autoLayout.findViewById(R.id.et_change_city);
        ListView listHint = (ListView) autoLayout.findViewById(R.id.lv_change_city);
        builder.setView(autoLayout);
        然后为ListView设置一个Adapter
        Adapter的getView()方法
        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            ViewHolder holder;
            if(convertView==null){
                convertView=View.inflate(mActivity,R.layout.item_change_city,null);
                holder=new ViewHolder();
                holder.hint= (TextView) convertView.findViewById(R.id.tv_hint_city);
                convertView.setTag(holder);
            }else{
                holder= (ViewHolder) convertView.getTag();
            }
            holder.hint.setText(cityList.get(position));
            return convertView;
        }
        适配器设置好了以后,就需要监听文本框的输入情况来动态改变ListView
        EditText有一个方法addTextChangedListener(TextWather tw);用于监听文本框的文字改变,在onTextChanged中实现显示提示
        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {
            if (s.length() > 0) {
                Log.d("changecity", "显示listview");
                //如果字符大于0,显示listview
                listHint.setVisibility(View.VISIBLE);
                cityList = dao.find(s.toString());
                adapter.notifyDataSetChanged();
            } else {
                listHint.setVisibility(View.GONE);
            }
        }
        每当文字发生改变时获取到用户输入的字符,查找数据库更新cityList并刷新UI。
        还有一个问题,由于每次查询返回的数据条数不一致,导致有时候数据条数太多ListView太长被键盘盖住了。这里还需要限制一下ListView的高度
        int itemCount=adapter.getCount()
        获取到当前的数据条数,如果条数大于四条就将ListView限制死,如果小于四条就以ListView本身的高度为准,在onTextChanged中添加
        int itemCount = adapter.getCount();
        Log.d("changecity", "count" + itemCount);
        RelativeLayout.LayoutParams params = (RelativeLayout.LayoutParams) listHint.getLayoutParams();
        if (itemCount > 4) {
            params.height = 500;
        } else {
            params.height = RelativeLayout.LayoutParams.WRAP_CONTENT;
        }
        listHint.setLayoutParams(params);
        最后一步,监听ListView的Item点击事件,实现用户点击提示补全编辑框
        点击后提示消失
        listHint.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                TextView cityName= (TextView) view.findViewById(R.id.tv_hint_city);
                autoText.setText(cityName.getText());
                listHint.setVisibility(View.GONE);
            }
        });
        大功告成!
  • 音量控制
    利用SeekBar来改变音量,首先在用户滑动了SeekBar的时候要播放铃声(不然调屁的音量啊!)
    设置SeekBar的监听,在onProgressChanged()中去实现逻辑
    setOnSeekBarChangeListener(OnSeekBarChangeListener listener);

    @Override
    public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
            playAlarmMusic();//播放音乐
            setSystemVolume(progress);//调节音量
            PrefUtils.putInt(mActivity, ConsUtils.ALARM_VOLUME, progress);//记录音量
    }

    现在依次来写出这两个方法吧,首先是播放音乐

    //播放音乐
        private void playAlarmMusic() {
            if(mPlayer==null){
                try {
                    mPlayer=new MediaPlayer();
                    AssetFileDescriptor assetFileDescriptor=mActivity.getAssets().openFd("everybody.mp3");
                    mPlayer.reset();
                    mPlayer.setAudioStreamType(AudioManager.STREAM_ALARM);
                    mPlayer.setDataSource(assetFileDescriptor.getFileDescriptor(), assetFileDescriptor.getStartOffset(), assetFileDescriptor.getLength());
                    mPlayer.setVolume(1f, 1f);
                    mPlayer.prepare();
                    mPlayer.start();
                    isMusicOn=true;
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }

    必须将mPlayer的AudioStreamType设置为某一个类型,否则是没有什么卵用的,因为是闹钟所以就设置为STREAM_ALARM就好
    另外因为每次滑动都会调用该方法,所以先判断一下当前是否已经在播放了,在播放了就不有再播放了。
    isMusicOn是做一个标记为后面关掉音乐用

    • AudioManager来设置音量
      AudioManager是系统服务通过getSystem(AUDIO_SERVICE)得到实例
      private void setSystemVolume(int progress) {
          //获取到音量调节管理器
          int maxVolume= mAudioManager.getStreamMaxVolume(AudioManager.STREAM_ALARM);
          int setVolume=(maxVolume*progress)/100;
          mAudioManager.setStreamVolume(AudioManager.STREAM_ALARM, setVolume, 0);
      }
      setStreamVolume()中的第二个参数的解释是传入音量的绝对值,我的理解是音量绝对值可能不一定是百分数,所以先获取到手机的最大音量值,然后通过百分比计算应该传入的绝对值(一定要先乘再除啊,否则int类型你懂得,误差会比较大),具体绝对值是什么我也没去研究,不过这样做应该没有太大的问题。
      获取和修改系统音量需要用到的权限
      <uses-permission android:name="android.permission.CAPTURE_AUDIO_OUTPUT" />
      <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" />
      写到这儿,这个项目基本上比较重要的地方都总结完了,另外为了防止应用被无情的杀死,我翻看了很多大神的博客,目前比较有效的应该是开启一个独立进程相互监视,互相扶持~,后面有空的话再补一篇吧!

完结!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值