Пользовательское представление Android: делегирование атрибутов элементам управления внутри представления - PullRequest
0 голосов
/ 03 сентября 2018

У меня есть настраиваемое представление с тремя элементами управления внутри (ImageView, TextInputLayout и TextInputEditText), и я хочу делегировать атрибуты тегов xml для моих пользовательских представлений соответствующим представлениям внутри моего настраиваемого Посмотреть. Можно ли сделать так, чтобы мой пользовательский вид принял xml-тег android: text = "Some string", а затем заставил его показать строку в соответствующем TextInputEditText внутри моего пользовательского представления?

enter image description here

Мой пользовательский вид (InputView.kt):

class InputView(context: Context, attrs: AttributeSet?, @AttrRes defStyleAttr: Int) : ConstraintLayout(context, attrs, defStyleAttr) {

    private lateinit var textInputLayout: TextInputLayout
    private lateinit var textView: TextInputEditText
    private lateinit var imageView: ImageView

//    var drawableRes = imageView.drawable ?: 0
    var text = textView.text
    var hint = textInputLayout.hint
    var error = textInputLayout.error
    var helperText = textInputLayout.helperText
    var ellipsize = textView.ellipsize ?: null

    constructor(context: Context) : this(context, null)
    constructor(context: Context, attrs: AttributeSet?) : this(context, attrs, 0)

    init {
        val view = LayoutInflater.from(context).inflate(R.layout.custom_inputview, this, true)

        textInputLayout = view.custom_inputview_text_input_layout
        textView = view.custom_inputview_text_input_edit_text
        imageView = view.custom_inputview_image_view

        attrs.let {
            context.theme.obtainStyledAttributes(
                attrs,
                R.styleable.InputView,
                0, 0).apply {

                try {
//                    mShowText = getString(R.styleable.InputView_text)
                } finally {
                    recycle()
                }
            }
        }
    }

}

Файл макета для представления (custom_inputview.xml):

<merge
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="0dp"
    android:layout_height="wrap_content"
    tools:parentTag="com.lucasurbas.masterdetail.ui.persondetails.InputView"
    tools:ignore="ContentDescription">

    <ImageView
        android:id="@+id/custom_inputview_image_view"
        android:layout_width="40dp"
        android:layout_height="40dp"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        android:tint="@color/bg_action_mode"
        tools:src="@drawable/ic_vd_hospital_24dp"/>

    <com.google.android.material.textfield.TextInputLayout
        android:id="@+id/custom_inputview_text_input_layout"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_marginStart="24dp"
        android:layout_marginEnd="24dp"
        app:layout_constraintBottom_toBottomOf="@+id/custom_inputview_image_view"
        app:layout_constraintStart_toEndOf="@+id/custom_inputview_image_view"
        app:layout_constraintTop_toTopOf="@+id/custom_inputview_image_view">

        <com.google.android.material.textfield.TextInputEditText
            android:id="@+id/custom_inputview_text_input_edit_text"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            tools:hint="Department"/>

    </com.google.android.material.textfield.TextInputLayout>
</merge>

Ответы [ 2 ]

0 голосов
/ 06 сентября 2018

Чтобы помочь другим, вот мое рабочее решение.

class InputView(context: Context, attrs: AttributeSet?, @AttrRes defStyleAttr: Int) : ConstraintLayout(context, attrs, defStyleAttr) {

    private var textInputLayout: TextInputLayout
    private var textView: TextInputEditText
    private var imageView: ImageView

    constructor(context: Context) : this(context, null)
    constructor(context: Context, attrs: AttributeSet?) : this(context, attrs, 0)

    init {
        val view = LayoutInflater.from(context).inflate(R.layout.custom_inputview, this, true)

        textInputLayout = view.custom_inputview_text_input_layout
        textView = view.custom_inputview_text_input_edit_text
        imageView = view.custom_inputview_image_view

        attrs.let {
            context.theme.obtainStyledAttributes(
                it,
                R.styleable.InputView,
                defStyleAttr, 0).apply {

                try {
                    textView.setText(getText(R.styleable.InputView_android_text))
                    textInputLayout.hint = getText(R.styleable.InputView_android_hint)
                    Log.d("InputView", "attrs " + textView.text)
                } finally {
                    recycle()
                }
            }
        }
    }

}

С атрибутами, определенными в attr.xml:

<declare-styleable name="InputView">
        <attr name="android:text" format="string" />
        <attr name="android:hint" format="string"/>
</declare-styleable>
0 голосов
/ 03 сентября 2018

Если вы хотите просто использовать атрибуты стиля Android, вы можете сделать что-то вроде следующего.

В этом примере он должен подобрать атрибут text, если вы определили его в xml при создании InputView.

init {
    attrs?.let {
        val typedArray = context.obtainStyledAttributes(attrs, STYLE_ATTRIBUTES)
        val text = typedArray.getText(0)
        typedArray.recycle()
    }
}

companion object {
    private val STYLE_ATTRIBUTES = intArrayOf(
        android.R.attr.text
    )
}
...