像 anko layout 一样写布局

2018-08-19  本文已影响258人  lguipeng

anko

如果不了解anko 的 可以去上面的链接看下anko 如何写布局的。

简单实现

首先,Kotlin的扩展函数功能是我们实现anko 的基础条件。
以下实现一个 ViewGroup (LinearLayout) 和 一个 View (TextView)
View 的最简单创建需要 Context,可能是 Activity 的 或者是 View Parent 的。
所以我们需要对 Context ViewGroup 进行扩展。(以下代码都是 在 Ui.kt 下面建的)

private inline fun <reified T : View> Context.handleView(view: T, init: T.() -> Unit): T{
    view.init()
    return view
}

private inline fun <reified T : View> ViewGroup.handleView(view: T, init: T.() -> Unit): T{
    view.init()
    this.addView(view)
    return view
}

如果是从ViewGroup 来的话 还需要把控件添加上去。
接着写一个 LinearLayout 和 TextView 的扩展

fun Context.linearLayout(init : LinearLayout.() -> Unit): LinearLayout {
    return this.handleView(LinearLayout(this), init)
}

fun ViewGroup.linearLayout(init : LinearLayout.() -> Unit): LinearLayout {
    return this.handleView(LinearLayout(this.context), init)
}

fun Context.text(init : TextView.() -> Unit): TextView{
    return this.handleView(TextView(this), init)
}

fun ViewGroup.text(init : TextView.() -> Unit): TextView{
    return this.handleView(TextView(this.context), init)
}

然后写一个布局参数

fun <T: View> T.lparams(
        width: Int = android.view.ViewGroup.LayoutParams.WRAP_CONTENT,
        height: Int = android.view.ViewGroup.LayoutParams.WRAP_CONTENT,
        marginTop: Int = 0,
        marginLeft: Int = 0,
        marginRight: Int = 0,
        marginBottom: Int = 0
): T {
    val layoutParams = LinearLayout.LayoutParams(width, height)
    layoutParams.setMargins(marginLeft, marginTop, marginRight, marginBottom)
    this.layoutParams = layoutParams
    return this
}

写 onClick

fun android.view.View.onClick(l: (v: android.view.View?) -> Unit) {
    setOnClickListener(l)
}

写 padding

fun noGetter(): Nothing = throw IllegalAccessException("Property does not have a getter")

var android.view.View.padding : Int
    set(value) {
        setPadding(value, value, value, value)
    }
    @Deprecated("Property does not have a getter", level = DeprecationLevel.ERROR)
    get() = noGetter()

现在就可以使用了

val layout = linearLayout {
            orientation = LinearLayout.VERTICAL
            padding = 40
            text {
                text = "H0"
                textSize = 16f
                onClick {
                }
            }
            linearLayout {
                lparams(marginTop = 30)
                orientation = LinearLayout.HORIZONTAL
                text {
                    text = "H1"
                }
                text {
                    lparams(marginLeft = 20)
                    text = "H2"
                }
            }
        }
container.addView(layout)

当然也可以扩展 Activity 然后直接添加到 content view 上面,我这里没写。

上一篇下一篇

猜你喜欢

热点阅读