您的位置:首页 > 移动开发 > Android开发

android开发系列之使用xml自定义控件

2016-03-23 00:04 316 查看
在android开发的过程中,有的时候面对多个Activity里面一些相同的布局,我们需要写多次相同的代码,同时这种方法给我们的项目维护也带来了很大不便。那么有没有一种可行的办法能够将Activity里面相同的布局拆分的很清楚呢?当然是有的,这个时候就轮到自定义控件闪亮登场了。

其实在android里面有多种方法去实现自定义控件,但是今天这篇博客里面只介绍使用xml的方式进行自定义控件的创建。请看下面的这种场景,不管在哪个页面里面都有个标题,包括一条居中的文本信息和在左边的返回按钮。这个时候我们就可以将上面的标题拆分成一个控件,然后在该控件里面暴露出一个text属性和一个按钮的点击事件。

首先让我们先来创建一个前台的xml文件用于放置布局,代码如下所示:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/test"
android:gravity="center_vertical">

<TextView
android:id="@+id/tvBack"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="<"
android:layout_alignParentLeft="true" />

<TextView
android:id="@+id/tvTitle"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="登录"
android:layout_centerHorizontal="true"/>

</RelativeLayout>


可以看到在上面的布局里面,我们只是放置了两个TextView,一个表示返回,另一个则表示title。

看到这里聪明的你也许就能猜到了,会不会在自定义控件的时候,我们也同样需要采用某种方法加载上面的布局文件,然后在加载的时候同时关联自己定义好的属性和事件到上面返回的TextView点击事件和标题的TextView的text属性呢?是的,请看比较重要的后台关联代码类定义:

public class TitleControl extends RelativeLayout {
}


可以发现TitleControl我们是直接从RelativeLayout继承而来的,这样的话TitleControl就属于一种View控件了。接下来所要做的事情,就是定义属性和事件了。现在假设我们需要定义一个MyText属性,那么应该怎么做呢?请看如下代码:

public class TitleControl extends RelativeLayout {

private TextView tvBack, tvTitle;
private String title;

public TitleControl(Context context) {
super(context);
}

public TitleControl(Context context, AttributeSet attrs) {
super(context, attrs);

View view = View.inflate(context, R.layout.activity_title, this);
tvTitle = (TextView) view.findViewById(R.id.tvTitle);

tvBack = (TextView) view.findViewById(R.id.tvBack);
tvBack.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {

}
});

TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.title);
title = a.getString(R.styleable.title_MyText);
a.recycle();

tvTitle.setText(title, TextView.BufferType.SPANNABLE);

}

public String getTitle() {
return this.title;
}

public void setTitle(String title) {
this.title = title;
}
}


在上面的代码里面我们通过View的inflate方法加载前台页面,然后通过findViewById方法就找到了前台需要关联的控件了。但是我们自己定义的属性放在哪里呢?一种比较容易维护的做法就是将属性放置在xml文件里面。下面就让我们来看看属性的xml代码:

<?xml version="1.0" encoding="utf-8"?>
<resources>
<declare-styleable name="title">
<attr name="MyText" format="string"/>
</declare-styleable>
</resources>


当我们定义好属性xml之后,就可以通过下面的这段代码,将我们自己定义的MyText属性关联到TextView的Text属性上面了。这样就相当于实现了自定义属性。

TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.title);
title = a.getString(R.styleable.title_MyText);
a.recycle();

tvTitle.setText(title, TextView.BufferType.SPANNABLE);


做到这里我们就差一个自定义事件了,那么我们怎样去定义一个事件关联到TextView的点击上面呢?答案是通过回调方法的方式实现的,请看回调方法的定义:

private ITitleCallback iTitleCallback;

public interface ITitleCallback {
void OnBackClickLinear();
}

public void setTitleClickLinear(ITitleCallback iTitleCallback) {
this.iTitleCallback = iTitleCallback;
}


然后在TextView点击的时候执行如下代码就可以了:

tvBack = (TextView) view.findViewById(R.id.tvBack);
tvBack.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if(iTitleCallback!=null){
iTitleCallback.OnBackClickLinear();
}
}
});


好了,今天就到这里吧!如有不对,欢迎拍砖。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: