Android组合控件

Android组合控件

组合控件,顾名思义,多个控件组合成一个控件使用。好比,咱们想要这样一个ImageView,图片的底部覆盖一个浮层,浮层上面显示一行文字,这个控件咱们能够用TextView覆盖在ImageView之上实现,咱们把这个控件命名为“CoverImageView”吧。java

怎样组合

既然是组合,那么就须要一个容器把这些分散的控件装在一块儿,这个容器就是ViewGroup,如:LinearLayout、RelativeLayout等,因此组合控件都是继承于一个ViewGroup的。android

组装的方式有两种:git

  1. xml文件里定义好控件效果,而后经过LayoutInflater布局到ViewGroup里面;
  2. 直接在ViewGroup里面经过代码添加子控件,实现效果。

第一种方式可以比较直观的看到组合效果,可是因为须要解析xml,因此性能上稍微差点;第二种并不能直观的看到效果,可是性能稍好。对于新手来讲,能够先用第一种方式实现,等到对组合控件的实现比较熟悉后可使用第二种方式实现。github

基本效果实现

能实现覆盖效果的ViewGroup有RelativeLayout、FrameLayout,咱们以FrameLayout为例。app

xml布局以下:函数

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:orientation="vertical">

    <!--图片占据整个控件大小-->
    <ImageView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_gravity="center"
        android:scaleType="centerCrop"
        android:src="@drawable/demo"/>

    <!--文字与控件等宽、在整个控件的底部-->
    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="bottom"
        android:background="#64000000"
        android:gravity="center"
        android:padding="6dp"
        android:text="@string/laomengzhu"
        android:textColor="#ffffff"/>
</FrameLayout>

效果以下:工具

base_preview

xml已经定义好了,咱们把它布局到咱们的ViewGroup里面去。咱们的控件“CoverImageView”继承于FrameLayout,咱们实现一个“setupViews”函数,用来初始化View,并在全部构造函数里面调用这个函数。布局

public class CoverImageView extends FrameLayout {

    public CoverImageView(Context context) {
        super(context);
        setupViews(context, null);
    }

    public CoverImageView(Context context, AttributeSet attrs) {
        super(context, attrs);
        setupViews(context, attrs);
    }

    public CoverImageView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        setupViews(context, attrs);
    }

    @TargetApi(Build.VERSION_CODES.LOLLIPOP)
    public CoverImageView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
        super(context, attrs, defStyleAttr, defStyleRes);
        setupViews(context, attrs);
    }

    private void setupViews(Context context, AttributeSet attrs) {
        //将xml布局到ViewGroup里面来
        View.inflate(context, R.layout.layout_civ, this);
    }
}

如今咱们能够把这个控件放到咱们的Activity里面看看效果了:性能

preview_in_activity1

布局优化

前面咱们已经看到咱们想要的效果了,可是咱们在xml里面的根布局是FrameLayout,咱们的CoverImageView也是继承于FrameLayout,那咱们最终实现的效果会不会套了两层FrameLayout呢?优化

咱们可使用ADT里面的Hierarchyviewer工具查看布局层级关系,使用方法也很简单:

  1. 在模拟器上运行你的程序(真机上我试了下不行);
  2. 打开Android Device Monitor,选中你程序的进程;
  3. 点击右上角的树形图标,过一会咱们就能看到咱们当前页面的层级关系了。

咱们当前页面的部分层级关系以下图:

preview_in_activity1

能够看到CoverImage下面的确还有一层FrameLayout,这一层FrameLayout是毫无用处的,这样的层级关系就会致使过分绘制问题(不懂的能够百度一下,这里就不赘述了),影响咱们的控件性能。

减小视图层级,咱们可使用merge标签,它能够删减多余的层级,优化UI。咱们把咱们xml布局里面的FrameLayout换成merge,而后运行,看看视图层级关系。

preview_in_activity1

层级减小了!!!

关于布局优化,你们能够看看这个博客:http://blog.csdn.net/xyz_lmn/article/details/14524567

经过代码添加控件

前面讲到,组装控件的另外一个方式就是经过代码添加控件,原理也很简单,就是在咱们的ViewGroup里面直接建立子控件、而后添加到咱们的ViewGroup里,调整子控件的布局,达到咱们想要的效果。因为少了xml解析这一步(虽然你没有解析,可是系统须要解析的),因此性能是优于第一种的。

private void setupViews(Context context, AttributeSet attrs) {
    //将xml布局到ViewGroup里面来
    /*View.inflate(context, R.layout.layout_civ, this);*/

    //经过代码添加子控件
    LayoutParams lp;

    ImageView imageView = new ImageView(context);
    //设置缩放模式
    imageView.setScaleType(ImageView.ScaleType.CENTER_CROP);
    imageView.setImageResource(R.drawable.demo);
    //设置子控件布局参数
    lp = new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT);
    //将子控件加入ViewGroup里
    addView(imageView, lp);

    TextView textView = new TextView(context);
    //设置内边距
    int padding;
    if (isInEditMode()) {
        padding = 18;
    } else {
        padding = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 6,
                getResources().getDisplayMetrics());
    }
    textView.setPadding(padding, padding, padding, padding);
    textView.setBackgroundColor(Color.parseColor("#64000000"));
    textView.setTextColor(Color.WHITE);
    //文字居中
    textView.setGravity(Gravity.CENTER);
    textView.setText(R.string.laomengzhu);
    lp = new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT);
    lp.gravity = Gravity.BOTTOM;
    addView(textView, lp);
}

效果和上面彻底一致。

自定义属性

前面咱们实现了咱们想要的效果,可是咱们没法修改控件里面的图片、文字大小,咱们须要定义控件的这些属性,固然你还能够定义文字颜色、文字背景色等属性,咱们这里只挑图片、文字大小以做演示。

  1. 在res/values目录下新建attrs.xml文件;
  2. 在attrs.xml里面定义控件属性;
  3. 在布局中使用属性
<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="CoverImageView">
        <!--name:属性名称 format:属性格式-->
        <!--属性支持的格式有:
        1. reference:参考某一资源ID。
        2. color:颜色值。
        3. boolean:布尔值。
        4. dimension:尺寸值。
        5. float:浮点值。
        6. integer:整型值。
        7. string:字符串。
        8. fraction:百分数。
        9. enum:枚举值。
        10. flag:位或运算。-->
        <attr name="imgSrc" format="reference"/>
        <attr name="coverTextSize" format="dimension|reference"/>
    </declare-styleable>
</resources>

而后咱们须要在控件中解析属性:

//解析控件属性
TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.CoverImageView);

//解析图片资源ID
int resId = ta.getResourceId(R.styleable.CoverImageView_imgSrc, -1);
if (resId != -1) {
    imageView.setImageResource(resId);
}

//解析文字大小
resId = ta.getResourceId(R.styleable.CoverImageView_coverTextSize, -1);
int textSize = 0;
if (resId != -1) {
    if (!isInEditMode()) {
        textSize = getResources().getDimensionPixelSize(resId);
    }
} else {
    textSize = ta.getDimensionPixelSize(R.styleable.CoverImageView_coverTextSize, 0);
}
if (textSize > 0) {
    textView.setTextSize(TypedValue.COMPLEX_UNIT_PX, textSize);
}

//释放
ta.recycle();

使用属性:

<com.laomengzhu.civ.CoverImageView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:id="@+id/view"
    app:imgSrc="@drawable/demo1"
    app:coverTextSize="18sp"
    android:layout_centerInParent="true"/>

自定义控件的属性默认是在xmlns:app="http://schemas.android.com/apk/res-auto"命名空间下的。

到此,咱们自定义组合控件就完成了,最终效果以下图:

preview_in_activity1

代码:https://github.com/laomengzhu/CustomWidget/tree/master/CombinationWidget/CoverImageView

相关文章
相关标签/搜索