FragmentMenu侧滑菜单界面实现
1.界面实现
- 界面显示比较简单,因为用的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了
比较复杂的是更换城市和音量改变
-
更换城市
- 在用户输入的时候给用户动态提示
- 配置数据库
首先配置好数据库,在第一次启动应用的时候将数据库拷贝到本地
- 配置数据库
- 在用户输入的时候给用户动态提示
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" />