实现TextView的onClick的xml配置
实现TextView的onClick的xml配置
亓斌哥哥 发表于3年前
实现TextView的onClick的xml配置
  • 发表于 3年前
  • 阅读 56
  • 收藏 0
  • 点赞 0
  • 评论 0

腾讯云 技术升级10大核心产品年终让利>>>   

摘要: 看到button的android:onClick=""很方便,现在我们也来做一个TextView的 onClick

看到button的android:onClick=""很方便,试了试TextView的,虽然可以设置onClick属性,但是点击的时候总是不能相应,onClick属性的配置是在View级别设置的,难道是TextView给屏蔽了?等看到TextView的源码再来解决这个疑问。

既然不能用, 那么咱们就仿照着View的源码自己实现个onClick

既然是增强版的TextView,那么首先建立一个类让它继承自TextView

public class MyTextView extends TextView {
public MyTextView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public MyTextView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
TypedArray ta = context.obtainStyledAttributes(attrs, 
R.styleable.MyTextView, defStyle, 0);
final int COUNT = ta.getIndexCount();
for (int i = 0; i < COUNT; i++) {
int attr = ta.getIndex(i);
switch (attr) {
case R.styleable.MyTextView_onClick:
final String handlerName = ta.getString(attr);
if(null != handlerName) {
setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Method handler = null;
try {
handler = getContext().getClass()
.getMethod(handlerName, View.class);
handler.invoke(getContext(), MyTextView.this);
} catch (Exception e) {
e.printStackTrace();
}
}
});
}
break;
}
}
ta.recycle();
}
}

短短几行代码,我们需要功能已经实现了,可以看到我们使用了反射机制,不用想也知道,android源码肯定也是使用了反射机制。

下面一点点的来分析吧:

首先是加载attr,因为我们要实现onClick的xml配置,所以必须要自定义属性:

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

这里我们定义了一个名称为onClick的属性,格式为string类型, 当你配置到xml布局的时候,我们还需要在自定义TextView中获取它的值,首先我们获取TypedArray对象:

TypedArray ta = context.obtainStyledAttributes(attrs,R.styleable.MyTextView, defStyle, 0);

接下来是获取ta的总个数:

final int COUNT = ta.getIndexCount();

接着一个熟悉的for循环,我们来取所有的我们自定义的attr,使用一个swith...case...语句来判断是不是我们需要的那个attr,当然我们只有一个attr,所以这么写有点多次一举,但是有利于以后扩展吧。

注意:

int attr = ta.getIndex(i);

这句话是获取当前attr的索引,即:R.styleable.xxx。

然后我们通过:

final String handlerName = ta.getString(attr);

获取方法名称(xxx:onClick="method")

接下来就是设置onClick事件了,因为当前我的是在一个TextView中,所以直接:

setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
}
}

就ok了。关键的步骤来了,我们需要根据获取的方法名通过反射机制来调用该方法,在使用button的onClick时,我们映射的方法有一个参数是View,我们这也来一个。

通过参数名获取方法:

Method handler = getContext().getClass().getMethod(handlerName, View.class);

这里不难理解,因为该Method是在使用该xml布局的Activity中定义的,所以需要getContext()来获取上下文,getMethod的第一个参数是方法名,第二个参数是方法的参数类型。

最后一步,就是调用这个方法了:

handler.invoke(getContext(), MyTextView.this);

现在我们扩展的TextView已经写完了,接下来就是使用了:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:click="http://schemas.android.com/apk/res/com.example.onclick"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity" >
    <com.example.onclick.MyTextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="50dip"
        android:background="@android:color/darker_gray"
        android:text="I am Custom TextView..."
        click:onClick="click" />
    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="50dip"
        android:background="@android:color/darker_gray"
        android:onClick="click2"
        android:text="I am TextView..." />
</LinearLayout>

我们定义了两个TextView,第一个是我们自定义的,第二个是一个普通的TextView,两个我们都给了onClick配置,需要注意的是:自定义TextView的onClick配置我们使用的是:

click:onClick="click"

我们使用了一个click命名空间,所以要声明该命名空间:

xmlns:click="http://schemas.android.com/apk/res/com.example.onclick"

我们的activity代码很简单:

public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
public void click(View view) {
Toast.makeText(this, "hello android", Toast.LENGTH_SHORT).show();
}
public void click2(View view) {
Toast.makeText(this, "hello world", Toast.LENGTH_SHORT).show();
}
}




共有 人打赏支持
亓斌哥哥
粉丝 29
博文 34
码字总数 12346
作品 13
×
亓斌哥哥
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
* 金额(元)
¥1 ¥5 ¥10 ¥20 其他金额
打赏人
留言
* 支付类型
微信扫码支付
打赏金额:
已支付成功
打赏金额: