如何基于布局创建XML友好的控件?

时间:2017-07-20 19:59:16

标签: android android-layout android-custom-view android-attributes

我创建了一个自定义控件,它是LinearLayout的子类。我还创建了一个该控件所基于的布局文件。最后,我定义了在构造函数中解析的属性,以设置我的自定义属性。例如,其中一个属性称为“文本”。

这是我的代码的简化版本(我已经删除了很多其他属性,所以我们可以专注于一个属性'text'):

首先,该类(我们的RadioButton的自定义版本)......

public class RadioButton extends LinearLayout
{
    private TextView textView;

    public RadioButton(Context context, AttributeSet attrs)
    {
        super(context, attrs);
        initAttributes(attrs, 0);
    }

    private void initAttributes(AttributeSet attrs, int defStyle)
    {
        final TypedArray a = getContext().obtainStyledAttributes(attrs, R.styleable.CheckBoxView, defStyle, 0);

        text = a.getString(R.styleable.RadioButton_text);
        if(text == null)
            text = "Not set";

        a.recycle();
    }

    @Override
    protected void onFinishInflate()
    {
        super.onFinishInflate();

        textView = (TextView)findViewById(R.id.textView);
        textView.setText(text);
    }

    private String text;
    public String getText() { return text; }
    public void setText(String newValue)
    {
        text = newValue;

        if(textView != null)
            textView.setText(text);
    }
}

这是attrs.xml文件......

<resources>

    <attr name="text" format="string" />

    <declare-styleable name="RadioButton">
        <attr name="text" />
    </declare-styleable>

</resources>

这里是'reusable_radiobutton.xml'布局文件(注意:内部RadioButtonView是一个自定义渲染的视图,工作正常):

<?xml version="1.0" encoding="utf-8"?>
<com.somedomain.reusable.ui.RadioButton
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="horizontal"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:gravity="center_vertical">

    <com.somedomain.reusable.ui.RadioButtonView
        android:id="@+id/radioButtonView"
        style="@style/DefaultRadioButtonView" />

    <TextView
        android:id="@+id/textView"
        style="@style/DefaultRadioButtonText" />

</com.somedomain.reusable.ui.RadioButton>

通过上述内容,我的控件的用户可以简单地将其包含在他们自己的布局文件中,如此......

<include android:id="@+id/someRadioButton"
    layout="@layout/reusable_radiobutton" />

在他们的代码中,使用以下内容,他们可以获得该实例并按照他们的意愿去做,就像这样......

RadioButton someRadioButton = (RadioButton)findViewById(R.id.someRadioButton);
someRadioButton.text = "Woot!";

这可以按预期工作。

然而,这不是......

<include android:id="@+id/someRadioButton"
    layout="@layout/reusable_radiobutton"
    app:text="Hello World!" />

它给了我一个警告,但忽略了它。

然后我尝试了这个......

<com.somedomain.reusable.ui.RadioButton
    app:text="Hello World!" />

虽然这确实实例化了我的控件并确实传递了“Hello World!”通过属性到我的属性,没有任何实际加载甚至将布局文件关联到我的类,所以屏幕上没有任何内容!

那么如何根据布局创建自定义视图,其他开发人员可以在自己的布局文件中引用,同时还允许他们设置自定义属性?

希望一切都有意义! :)

  

注意:Android文档正好讲述了我正在追求的内容,'复合控件'as referenced here,但他们没有给出使用布局来定义复合控件的示例。但是,我觉得非常接近。

1 个答案:

答案 0 :(得分:0)

在SO上找到它here。我没有意识到你可以给自己充气。那个并使用Merge标签为我照顾它。