详解Android中提示对话框(ProgressDialog和DatePickerDialog和TimePickerDialog&PopupWindow)
progressdialog(精度条对话框):
1.直接调用progressdialog提供的静态方法show()显示
2.创建progressdialog,再设置对话框的参数,最后show()出来
package com.example.test3; import android.app.activity; import android.app.progressdialog; import android.content.context; import android.os.bundle; import android.os.handler; import android.os.message; import android.view.view; import android.widget.button; public class mainactivity extends activity implements view.onclicklistener{ private button btn_one; private button btn_two; private button btn_three; private progressdialog pd1 = null; private progressdialog pd2 = null; private final static int maxvalue = 100; private int progressstart = 0; private int add = 0; private context mcontext = null; //定义一个用于更新进度的handler,因为只能由主线程更新界面,所以要用handler传递信息 final handler hand = new handler() { @override public void handlemessage(message msg) { //这里的话如果接受到信息码是123 if(msg.what == 123) { //设置进度条的当前值 pd2.setprogress(progressstart); } //如果当前大于或等于进度条的最大值,调用dismiss()方法关闭对话框 if(progressstart >= maxvalue) { pd2.dismiss(); } } }; @override protected void oncreate(bundle savedinstancestate) { super.oncreate(savedinstancestate); setcontentview(r.layout.activity_main); mcontext = mainactivity.this; bindviews(); } private void bindviews() { btn_one = (button) findviewbyid(r.id.btn1); btn_two = (button) findviewbyid(r.id.btn2); btn_three = (button) findviewbyid(r.id.btn3); btn_one.setonclicklistener(this); btn_two.setonclicklistener(this); btn_three.setonclicklistener(this); } @override public void onclick(view v) { switch (v.getid()){ case r.id.btn1: //这里的话参数依次为,上下文,标题,内容,是否显示进度,是否可以用取消按钮关闭 progressdialog.show(mainactivity.this, "资源加载中", "资源加载中,请稍后...",false,true); break; case r.id.btn2: pd1 = new progressdialog(mcontext); //依次设置标题,内容,是否用取消按钮关闭,是否显示进度 pd1.settitle("软件更新中"); pd1.setmessage("软件正在更新中,请稍后..."); pd1.setcancelable(true); //这里是设置进度条的风格,horizontal是水平进度条,spinner是圆形进度条 pd1.setprogressstyle(progressdialog.style_horizontal); pd1.setindeterminate(true); //调用show()方法将progressdialog显示出来 pd1.show(); break; case r.id.btn3: //初始化属性 progressstart = 0; add = 0; //依次设置一些属性 pd2 = new progressdialog(mainactivity.this); pd2.setmax(maxvalue); pd2.settitle("文件读取中"); pd2.setmessage("文件加载中,请稍后..."); //这里设置为不可以通过按取消按钮关闭进度条 pd2.setcancelable(false); pd2.setprogressstyle(progressdialog.style_horizontal); //这里设置的是是否显示进度,设为false才是显示的哦! pd2.setindeterminate(false); pd2.show(); //这里的话新建一个线程,重写run()方法, new thread() { public void run() { while(progressstart < maxvalue) { //这里的算法是决定进度条变化的,可以按需要写 progressstart = 2 * usetime() ; //把信息码发送给handle让更新界面 hand.sendemptymessage(123); } } }.start(); break; } } //这里设置一个耗时的方法: private int usetime() { add++; try{ thread.sleep(100); }catch (interruptedexception e) { e.printstacktrace(); } return add; } }
2.date/timepickerdialog只是供用户来选择日期时间,对于android系统的系统时间, 日期没有任何影响
他们两个的构造方法非常相似: datepickerdialog(上下文;datepickerdialog.ondatesetlistener()监听器;年;月;日)
timepickerdialog(上下文;timepickerdialog.ontimesetlistener()监听器;小时,分钟,是否采用24小时制)
public class mainactivity extends appcompatactivity implements view.onclicklistener{ private button btn_date; private button btn_time; private string result = ""; @override protected void oncreate(bundle savedinstancestate) { super.oncreate(savedinstancestate); setcontentview(r.layout.activity_main); bindviews(); } private void bindviews() { btn_date = (button) findviewbyid(r.id.btn_date); btn_time = (button) findviewbyid(r.id.btn_time); btn_date.setonclicklistener(this); btn_time.setonclicklistener(this); } @override public void onclick(view v) { result = ""; switch (v.getid()){ case r.id.btn_date: calendar cale1 = calendar.getinstance(); new datepickerdialog(mainactivity.this,new datepickerdialog.ondatesetlistener() { @override public void ondateset(datepicker view, int year, int monthofyear, int dayofmonth) { //这里获取到的月份需要加上1哦~ result += "你选择的是"+year+"年"+(monthofyear+1)+"月"+dayofmonth+"日"; toast.maketext(getapplicationcontext(), result, toast.length_short).show(); } } ,cale1.get(calendar.year) ,cale1.get(calendar.month) ,cale1.get(calendar.day_of_month)).show(); break; case r.id.btn_time: calendar cale2 = calendar.getinstance(); new timepickerdialog(mainactivity.this, new timepickerdialog.ontimesetlistener() { @override public void ontimeset(timepicker view, int hourofday, int minute) { result = ""; result += "您选择的时间是:"+hourofday+"时"+minute+"分"; toast.maketext(getapplicationcontext(), result, toast.length_short).show(); } }, cale2.get(calendar.hour_of_day), cale2.get(calendar.minute), true).show(); break; } } }
最后一个用于显示信息的ui控件——popupwindow(悬浮框),如果你想知道 他长什么样子,你可以打开你手机的qq,长按列表中的某项,这个时候后弹出一个黑色的小 对话框,这种就是popupwindow了,和alertdialog对话框不同的是,他的位置可以是随意的; 另外alertdialog是非堵塞线程的,而popupwindow则是堵塞线程的
1)几个常用的构造方法
我们在文档中可以看到,提供给我们的popupwindow的构造方法有九种之多,这里只贴实际 开发中用得较多的几个构造方法:
public popupwindow (context context) public popupwindow(view contentview, int width, int height) public popupwindow(view contentview) public popupwindow(view contentview, int width, int height, boolean focusable)
参数就不用多解释了吧,contentview是popupwindow显示的view,focusable是否显示焦点
2)常用的一些方法
下面介绍几个用得较多的一些方法,其他的可自行查阅文档:
setcontentview(view contentview):设置popupwindow显示的view
getcontentview():获得popupwindow显示的view
showasdropdown(view anchor):相对某个控件的位置(正左下方),无偏移
showasdropdown(view anchor, int xoff, int yoff):相对某个控件的位置,有偏移
showatlocation(view parent, int gravity, int x, int y): 相对于父控件的位置(例如正*gravity.center,下方gravity.bottom等),可以设置偏移或无偏移 ps:parent这个参数只要是activity中的view就可以了!
setwidth/setheight:设置宽高,也可以在构造方法那里指定好宽高, 除了可以写具体的值,还可以用wrap_content或match_parent, popupwindow的width和height属性直接和第一层view相对应。
setfocusable(true):设置焦点,popupwindow弹出后,所有的触屏和物理按键都由popupwindows 处理。其他任何事件的响应都必须发生在popupwindow消失之后,(home 等系统层面的事件除外)。 比如这样一个popupwindow出现的时候,按back键首先是让popupwindow消失,第二次按才是退出 activity,准确的说是想退出activity你得首先让popupwindow消失,因为不并是任何情况下按back popupwindow都会消失,必须在popupwindow设置了背景的情况下 。
setanimationstyle(int):设置动画效果
public class mainactivity extends activity { private button btn_show; private context mcontext; @override protected void oncreate(bundle savedinstancestate) { super.oncreate(savedinstancestate); setcontentview(r.layout.activity_main); mcontext = mainactivity.this; btn_show = (button) findviewbyid(r.id.btn_show); btn_show.setonclicklistener(new view.onclicklistener() { @override public void onclick(view v) { initpopwindow(v); } }); } private void initpopwindow(view v) { view view = layoutinflater.from(mcontext).inflate(r.layout.item_popup, null, false); button btn_xixi = (button) view.findviewbyid(r.id.btn_xixi); button btn_hehe = (button) view.findviewbyid(r.id.btn_hehe); //1.构造一个popupwindow,参数依次是加载的view,宽高 final popupwindow popwindow = new popupwindow(view, viewgroup.layoutparams.wrap_content, viewgroup.layoutparams.wrap_content, true); popwindow.setanimationstyle(r.anim.anim_pop); //设置加载动画 //这些为了点击非popupwindow区域,popupwindow会消失的,如果没有下面的 //代码的话,你会发现,当你把popupwindow显示出来了,无论你按多少次后退键 //popupwindow并不会关闭,而且退不出程序,加上下述代码可以解决这个问题 popwindow.settouchable(true); popwindow.settouchinterceptor(new view.ontouchlistener() { @override public boolean ontouch(view v, motionevent event) { return false; // 这里如果返回true的话,touch事件将被拦截 // 拦截后 popupwindow的ontouchevent不被调用,这样点击外部区域无法dismiss } }); popwindow.setbackgrounddrawable(new colordrawable(0x00000000)); //要为popwindow设置一个背景才有效 //设置popupwindow显示的位置,参数依次是参照view,x轴的偏移量,y轴的偏移量 popwindow.showasdropdown(v, 50, 0); //设置popupwindow里的按钮的事件 btn_xixi.setonclicklistener(new view.onclicklistener() { @override public void onclick(view v) { toast.maketext(mainactivity.this, "你点击了嘻嘻~", toast.length_short).show(); } }); btn_hehe.setonclicklistener(new view.onclicklistener() { @override public void onclick(view v) { toast.maketext(mainactivity.this, "你点击了呵呵~", toast.length_short).show(); popwindow.dismiss(); } }); } }