Window及WindowManager(三)Window的创建过程

微信小程序 自定义Toast实例代码

Android中的Toast是一种简易的消息提示框,toast提示框不能被用户点击,toast会根据用户设置的显示时间后自动消失。
 
 
创建Toast
 两个方法创建Toast
Java代码 
makeText(Context context, int resId, int duration) 
 
  参数:context是toast显示在哪个上下文,通常是当前Activity;resId指显示内容引用Resouce那条数据,就是从R类中去指定显示的消息内容;duration指定显示时间,Toast默认有LENGTH_SHORT和LENGTH_LONG两常量,分别表示短时间显示和长时间显示。
 
Java代码 
makeText(Context context, CharSequence text, int duration) 
 
  参数context和duration与第一个方法相同,参数text可以自己写消息内容。
 
 用上面任意方法创建Toast对象之后调用方法show()即可显示。
 
Java代码 
Toast toast = Toast.makeText(ToastDemoActivity.this,
“这是一个普通的Toast!”, Toast.LENGTH_SHORT); 
toast.show(); 
 
 
 
 
设置Toast显示位置
 两个方法可以设置显示位置:
 方法一:setGravity(int gravity, int xOffset, int
yOffset)三个参数分别表示(起点位置,水平向右位移,垂直向下位移)
 方法二:setMargin(float horizontalMargin, float verticalMargin)
 以横向和纵向的百分比设置显示位置,参数均为float类型(水平位移正右负左,竖直位移正上负下)
 
Java代码 
//设置Toast显示位置(起点位置,水平向右位移,垂直向下位移) 
toast.setGravity(Gravity.TOP | Gravity.LEFT, 0, 200); 
//Toast显示位置,以横向和纵向的百分比计算,参数均为float类型(水平位移正右负左,竖直位移正上负下) 
toast.setMargin(-0.5f, 0f); 
 
 
自定义Toast
 下面这段代码是从iteye抄过来的,显示一个带图片的Toast效果:
Java代码 
// 带图片的Toast 
        Button btn2 = (Button) findViewById(R.id.toast2); 
        btn2.setOnClickListener(new OnClickListener() { 
            public void onClick(View v) { 
                // 定义一个Toast 
Toast toast = Toast.makeText(ToastDemoActivity.this,
“这是一个代图片的Toast!”, Toast.LENGTH_LONG); 
                // 定义一个ImageView 
                ImageView imageView = new
ImageView(ToastDemoActivity.this); 
                imageView.setImageResource(R.drawable.icon); 
                // 获得Toast的View 
                View toastView = toast.getView(); 
                // 定义一个Layout,这里是Layout 
                LinearLayout linearLayout = new
LinearLayout(ToastDemoActivity.this); 
                linearLayout.setOrientation(LinearLayout.HORIZONTAL); 
                // 将ImageView和ToastView合并到Layout中 
                linearLayout.addView(imageView); 
                linearLayout.addView(toastView); 
                // 替换掉原有的ToastView 
                toast.setView(linearLayout); 
                toast.show(); 
            } 
        }); 

Toast的Window创建过程

由于Toast具有定时取消这一功能,所以系统采用了Handler。在Toast内部有两类IPC过程,第一类是Toast访问NotificationManagerService,第二类是NotificationManagerService回调Toast里的TN接口。下面将NotificationManagerService简称为NMS。
Toast内部的视图既可以是系统自定义也可以用setView方法来进行自定义View。

Toast的显示和隐藏都需要通过NMS来实现,由于NMS运行在系统的进程中,所以只能通过远程调用的方式来进行显示和隐藏Toast。TN这个类,它是一个Binder类,NMS处理Toast的显示或隐藏请求时会跨进程回调TN中的方法,这个时候由于TN运行在Binder线程池中,所以需要使用Handler来切换到当前线程(发起Toast请求的线程),由此可以Toast无法运行在没有Looper的线程中。

  1. Toast的显示过程
    1. 通过enqueueToast方法,将Toast请求封装为ToastRecord对象并将其添加到一个名为mToastQueue的队列中。对于非系统的应用来说mToastQueue中最多可以同时存在0个ToastRecord(防止Dos攻击)。
    2. 当ToastRecord被添加到mToastQueue中或,NM就会通过showNextToastLocked方法来显示当前的Toast,Toast的显示是由ToastRecord的callback来完成的,这个callback其实就是TN对象的远程BInder,最终该方法会运行在Binder线程池中。
    3. Toast显示以后,NMS还会通过scheduleTimeoutLocked方法来发送一个延时消息,来决定Toast的显示时长。
  2. Toast的隐藏过程
    1. Toast的隐藏过程也是通过callback方法来完成的,这同样是一次IPC过程。

Toast样式可以根据需求自定义,本例中是圆形

 

Dialog的Window创建过程

Dialog和Activity类似

  1. 创建Window 同样是通过PolicyManager的makeNewWindow方法来完成的
  2. 初始化DecorView并将Dialog的视图添加到DecorView中去
  3. DecorView添加到Window中并显示。(通过dialog的show方法),移除通过dismiss(WindowManager的remoteViewImmediate)

注意普通Dialog有一个特殊之处,那就是必须采用Activity的Context,如果采用ApplicationContext则会报错,错误在于没用应用token导致的,而应用token一般只有Activity拥有。另外系统Window(通过type参数进行设定)比较特殊,它可以不需要token。