浅析Android Dialog中setContentView()方法

概述

Dialog在Android中是一个很优秀的工具。在使用Dialog时,我们一般都会自定义要显示的内容布局。Dialog自带了三个方法来支持自定义内容布局。

 public void setContentView (int layoutResID);
 public void setContentView (View view);
 public void setContentView (View view, ViewGroup.LayoutParams params);

这三个方法内部的实现原理都是一样的,只是其封装深度不同而已。三个方法可以说分别照顾了不同定制深度的开发者。

setContentView()流程

直接查看Dialog的源代码,如下图1所示。

【图1】

上图中mWindow在Dialog类中的定义如下:

 import android.view.Window;
 Window mWindow;

那么,它从何而来呢?如下图2所示。

【图2】

由上图2可知,在构造Dialog对象时,这个mWindow的值也被确定。它由PolicyManager提供。再往下跟系统代码。

makeNewWindow(Context)方法的实现如下:

  // The static methods to spawn new policy-specific objects
  public static Window makeNewWindow(Context context) {
   return sPolicy.makeNewWindow(context);
  }

还得继续往下跟。

import com.andorid.policy.internal.policy.impl.Policy;
/**
 * {@hide}
 */
public class Policy implements IPolicy {
 //...

 public Window makeNewWindow(Context context) {
  return new PhoneWindow(context);
 }
}

到这,貌似就差不多看到尽头了,原来我们调用的setContentView就是在这个PhoneWindow类中被实现的。继续跟进。

import com.android.internal.policy.impl.PhoneWindow;
/**
 * Android-specific Window.
 * <p>
 * todo: need to pull the generic functionality out into a base class
 * in android.widget.
 */
public class PhoneWindow extends Window implements MenuBuilder.Callback {
 //...
}

setContentView(int)

这个方法的代码实现如下图3所示。

【图3】

整个的实现流程乍一看还算简单明了。我们传入的布局参数最后就是被加载到上图所示的那个 mContentParent 中的。这个mContentParent是一个ViewGroup类对象。

在上图所示的代码中第367行作了空判断,可见这个对象的实例的创建与installDecor()方法有关系。这个方法的实现较为复杂,这里我们只看mContentParent的实例化过程。

【图4】

这个generateLayout()方法的实现过程很繁杂。我们没有必要去把每一行的代码都看懂。只需要知道在它内部是这样创建mContentParent对象的就好了

最后会把这个contentParent作为结果返回即可。然后再回到图3,在图中所示代码处第378行完成了将我们传入的布局加载进系统容器中的操作。

setContentView(View)与setContentView(View, ViewGroup.LayoutParams)

这种方式设置内容布局比较灵活。一般用于布局中有需要在Java代码中做特殊操作的布局。如设置监听等。其具体实现代码如下图5所示。

【图5】

这个代码,并没什么特别的,它的目的都已经明明白白的表现在代码上了,就不再赘述了。

setContentView(View)无法设置布局尺寸的问题

使用setContentView(int)的方式时,可以直接通过在layout的根容器中指定宽、高来设置布局的尺寸。这里得注意,我指的是在根视图中直接指定宽度多少像素,高度多少像素这种直白的写法才可以控制布局的尺寸。若直接设为MATCH_PARENT,那么它的效果等同于WRAP_CONTENT。为什么会是这样的结果呢?本人并没有深究它的原因,但本人猜测(且后续并未去证实)这与mContentParent加载了布局后重新确定整个视图的尺寸的过程脱不了干系。我们来翻翻ViewGroup的代码,在ViewGroup中,有如下图6所示的一段代码。

【图6】

对于一个ViewGroup及其子类来说,它的MeasureSpec要么是EXACTLY要么是AT_MOST,我记不清这里头的具体关系了。但上图6所示的代码也已经非常直白了。对于MATCH_PARENT,它确实是按照WRAP_CONTENT的方式来处理的。这也就解释了上面所说的“令人费解”的情况了。虽然这个只是本人的猜测,但我估计也是八九不离十了。

而使用setContentView(View)的方式时,无论layout中根容器的宽高是什么,都按照WRAP_CONTENT的方式来走。这是为什么?我们先回去看看图5所示代码中这个方法的实现。可以发现,PhoneWindow给了一个默认的ViewGroup.LayoutParams对象。并且宽、高的值不偏不倚,正好是MATCH_PARENT。因此,当使用这种方式时,无论layout中根容器的宽高如何设置,它都表现成按内容的尺寸来适配布局的效果。因此,想要能控制对话框布局的尺寸,还是老老实实自己建一个有指定宽高值的LayoutParams对象给PhoneWindow对象吧,不要指望人家帮你擦屁股,擦不干净的~~

那么,到了这里,我们再来简单探究一下,setContentView(int)又是如何做到可以直接设置尺寸的。我们回去图3,看看这个方法的实现代码中,第378行。它在映射xml时,第二个参数传的直接是mContentParent。比较一下我们平时使用映射布局函数时,讲道理,直接传一个null的比较多吧,或者说,或许我们平时都很少注意到这个参数。我们去LayoutInflater中转转。

 import android.view.LayoutInflater;

inflate()方法的代码还是挺长的,这里就不详细贴了,我们只挑有代表性的来看。

public View inflate(XmlPullParser parser, ViewGroup root, boolean attachToRoot)

// ...

final AttributeSet attrs = Xml.asAttributeSet(parser);

// ...

// Create layout params that match root, if supplied
params = root.generateLayoutParams(attrs);

// ...

// Temp is the root view that was found in the xml
final View temp = createViewFromTag(root, name, attrs, false);

// ...

if (root != null && attachToRoot) {
 root.addView(temp, params);
}
// ...

够清晰了吧。在这里,LayoutInflater在映射xml布局时主动去解析了所有的属性。当然会包括外层容器的属性。然后根据解析的结果生成一个LayoutParams对象,最后,再将要内容布局联合这个即时创建的LayoutParams对象一同添加到mContentParent容器中去,其实就相当于调用setContentView(View, LayoutParams)方法。所以在文章开头我才说到Dialog的三个设置内容布局的方法本质是一样的,只是其封装深度不同而已。

设置Dialog的背景为完全透明

Dialog默认有一个灰色的背景,首先这个背景巨丑,其次背景的存在还影响我们对对话框UI的定制。

   

在Activity中,可以通过在创建Dialog时传入一个无背景对话框的风格样式给构造器,以构造出无灰色背景的对话框出来。也可以通过Java代码控制对话框的背景色为透明色。还可以先show()对话框,然后再给它setContentView()来达到无背景色的对话框的目的。

1、 通过风格样式

 <style name="transBg" parent="@android:Theme.Dialog">
  <item name="android:windowBackground">@android:color/transparent"</item>
 </style> 

 AlertDialog dialog = new AlertDialog.Builder(mContext, R.style.transBg).create();
 //或
 Dialog dlg = new Dialog(mContext, R.style.transBg);
 //等

2、通过Java代码控制

所谓背景,其实就是PhoneWindow的背景。我们只需要设置PhoneWindow的背景为透明,就能达到我们想要的结果了。

// ...
AlertDialog dialog = builder.create();
dialog.show();
dialog.setContentView(view);
//方式1,使用透明的ColorDrawable对象。
dialog.getWindow().setBackgroundDrawable(new ColorDrawable(0));
//方式2,使用一张透明的Drawable图片。
dialog.getWindow().setBackgroundDrawableResource(R.drawable.transparent);

3、先显示对话框再设置布局

这种方式只在Activity中有效果。

 AlertDialog dialog = builder.create();
 dialog.show();
 dialog.setContentView(view);

至于Service中为什么没有效果,本人怀疑是由于在Service中要想弹出对话框,只能将它设为系统级对话框,需要加多的一段代码导致的。但其具体原理还没有去研究过。

dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);

在Service中。只能通过上述第1、第2种方式来实现背景透明的目的。

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持我们!

(0)

相关推荐

  • 深入解析Android中的setContentView加载布局原理

    前言 对于Android的开发者来说,setContentView大家再熟悉不过了,在我们的Activity中首先就是要用它加载我们的布局,但是应该有一部分人是不知道加载布局的原理,也包括我,今天就从源码的角度分析setContentView加载布局原理. 准备工作 由于我们使用的Android API部分源码是隐藏的,当我们在AndroidStudio中是不能找到源码的,我们可以去官网下载相应源码去查看,当然在GitHub下载相应版本的API替换我们sdk下platforms相应api的and

  • Android开发中setContentView和inflate的区别分析

    本文实例讲述了Android开发中setContentView和inflate的区别.分享给大家供大家参考,具体如下: 一般用LayoutInflater做一件事:inflate inflate这个方法总共有四种形式(见下面),目的都是把xml表述的layout转化为View对象. 其中有一个比较常用,View inflate(int resource, ViewGroup root),另三个,其实目的和这个差不多. int resource,也就是resource/layout文件在R文件中对

  • 浅析Android Dialog中setContentView()方法

    概述 Dialog在Android中是一个很优秀的工具.在使用Dialog时,我们一般都会自定义要显示的内容布局.Dialog自带了三个方法来支持自定义内容布局. public void setContentView (int layoutResID); public void setContentView (View view); public void setContentView (View view, ViewGroup.LayoutParams params); 这三个方法内部的实现原

  • 浅析Android Service中实现弹出对话框的坑

    一.手机版本问题,大多数文章没有涉及这个点,导致他们的代码并无法正常使用 M版本以上需要使用的Type--> TYPE_APPLICATION_OVERLAY AlertDialog.Builder builder=new AlertDialog.Builder(getApplicationContext()); builder.setTitle("提示"); builder.setMessage("service弹框"); builder.setNegati

  • 浅析Android系统中HTTPS通信的实现

    前言 最近有一个跟HTTPS相关的问题需要解决,因此花时间学习了一下Android平台HTTPS的使用,同时也看了一些HTTPS的原理,这里分享一下学习心得. HTTPS原理 HTTPS(Hyper Text Transfer Protocol Secure),是一种基于SSL/TLS的HTTP,所有的HTTP数据都是在SSL/TLS协议封装之上进行传输的.HTTPS协议是在HTTP协议的基础上,添加了SSL/TLS握手以及数据加密传输,也属于应用层协议.所以,研究HTTPS协议原理,最终就是研

  • Android Dialog中软键盘的显示与隐藏的示例

    1.写在前面 本篇的主要内容是关于在Dialog中软键盘的显示与隐藏问题,需求是在Dialog中有一个密码输入框,弹出Dialog显示软键盘,关闭Dialog隐藏软键盘. 嗯,是不是有点简单,不过在实现的过程中还是遇到了一些问题,在试过了网上大部分的方法之后,最终找到了一个还不错的方法,分享给大家. 看下效果图: 2.实现过程 先说说最开始的实现方法: // 显示Dialog dialog.show(); // 显示软键盘 SoftInputUtils.showSoftInput(activit

  • Android应用中clearFocus方法调用无效的问题解决

    clearFocus 无效? EditText在focus与非focus的时候,显示效果是不同的:focus的时候光标是闪的,而且我们通常也会给它设置selector,focus的时候给它加上边框之类的. 通常当我们触摸EditText之外的View时,需要清除EditText的焦点.很自然的就会想到EditText.clearFocus(),然而常常并没有用.(EditText.isFocus()依然是true,光标也依然在跳跃...) clearFocus的实现 clearFocus的调用栈

  • Spinner在Dialog中的使用效果实例代码详解

    背景: 记得很久以前,碰到一个需求场景,需要在Android Dialog中显示Spinner,用来进行选择操作.那个时候还很困惑,不知道是否可以这么搞.抱着试试看的心态,做起了实验,看起来效果还可行,不过最终还是选用了一个开源项目,效果看起来更棒. 代码演示: Spinner在Dialog中的使用,Dialog中关于view的xml布局. <?xml version="1.0" encoding="utf-8"?> <LinearLayout x

  • Android 开发之Dialog中隐藏键盘的正确使用方法

    Android 开发之Dialog中隐藏键盘的正确使用方法 场景:弹出一个Dialog,里面有一个EditText,用来输入内容,因为输入时,需要弹出键盘,所以当Dialog消失时,键盘要一起隐藏. 现在我们做一个自定义的Dialog MyDialog extends Dialog 一开始认为这个功能很容易实现,于是写了下面的代码 //Dialog的构造函数中写 this.setOnDismissListener(new OnDismissListener() { @Override publi

  • 浅析SVN在Android Studio中的安装和配置方法

    在AndroidStudio中开发版本控制,除了Git就是SVN,和Eclipse不同Android Studio没有提供单独的插件,只能和SVN客户端关联使用,和Eclipse安装有很大区别,下面介绍个在AndroidStudio中SVN的安装和配置方法. 一.SVN的安装 Eclipse都是直接安装插件就可以了,AndroidStudio不行,只能通过关联SVN客户端,需要特别注意的是安装SVN时必须安装command line 功能,只有安装带有 command line 功能的 SVN

  • 浅析Android中常见三种弹框在项目中的应用

    一丶概述 弹框在Android项目中经常出现,常见的实现方法有三种:Dialog 弹框,Window弹框,Activity伪弹框.本文就说一说三种弹框的实现及在项目中的运用. 二丶演示图         图一为常见的三种弹框(文末上链接),图二为项目中用到的Activity伪弹框 三丶正文 1.Dialog弹框 先看一篇一篇文章: android 8种对话框(Dialog)使用方法汇总 Dialog是系统自带的弹框,然而常常因为UI不好看而遭嫌弃,常需要自定义 public class MyDi

随机推荐