使用anko 抛弃XML

anko

anko 是一款结合kotlin语言抛弃布局xml的工具,使用他就可以不用写布局xml代码了

举个栗子

如果我们需要实现一个功能 点击按钮toast弹出输入框输入的字符(如下效果图)

效果图

普通方式实现这样一个功能,可能我们会这么做

  • 创建布局xml文件
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context="com.github.jokar.ankotest.MainActivity">

    <EditText
        android:id="@+id/edit"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:text="showEdit"
        android:onClick="showEdit"/>
</LinearLayout>

  • 在activity里初始化输入框和 写点击事件
class MainActivity : AppCompatActivity() {
    var edit: EditText? = null

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        edit = findViewById(R.id.edit) as EditText
    }


    fun showEdit(v: View) {
        var string = edit?.text.toString()
        Toast.makeText(applicationContext, string, Toast.LENGTH_SHORT).show()
    }
}
但是如果我们使用anko就不同了,我们不需要XML文件了,然后修改activity如下
class MainActivity : AppCompatActivity() {
    var edit: EditText? = null

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        createView()
    }


    fun showEdit() {
        var string = edit?.text.toString()
        Toast.makeText(applicationContext, string, Toast.LENGTH_SHORT).show()
    }

    fun createView() {
        linearLayout {
            lparams(width = matchParent, height = matchParent)
            orientation = LinearLayout.VERTICAL

            edit = editText {
                textSize = 15f
            }.lparams(width = matchParent, height = wrapContent)

            button {
                text = "ShowText"
                textSize = 15f
                onClick {
                    showEdit()
                }
            }.lparams(width = matchParent, height = wrapContent) {
                topMargin = dip(10)
            }
        }
    }
}

这就是anko的基本用法了,跟写xml布局没有多大区别;但在实际使用过程中会发现如果不是再activity类里想直接使用anko控件是不行的;;下面介绍开发中遇到的基本场景

在fragment里使用anko

在fragment里使用还是比较容易的;只需要在最外层加上UI {}.view就可以使用了;样例如下:


class Fragment1 : Fragment() {

    var swipeRefreshLayout: SwipeRefreshLayout? = null
    var recyclerView: RecyclerView? = null

    override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        return super.onCreateView(inflater, container, savedInstanceState)
    }

    /**
     * createView
     */
    fun createView(): View {
        return UI {
            linearLayout {
                //swipeRefreshLayout
                swipeRefreshLayout = swipeRefreshLayout {
                    setColorSchemeResources(android.R.color.holo_blue_bright)
                    setOnRefreshListener {
                        getData()
                    }
                    //recyclerView
                    recyclerView = recyclerView {
                        layoutManager = LinearLayoutManager(context)
                        backgroundColor = Color.parseColor("#f3f3f3")
                    }
                }.lparams(width = matchParent, height = matchParent)

            }
        }.view
    }

    fun getData() {

    }
}

这样就行了,使用很简单


在Adapter里使用

这里介绍如何在非activity,fragment里使用anko,也很简单使用到了with字符,例如这样:

 fun createItemView(context: Context): View {
        return with(context) {
            linearLayout {
                lparams(width = matchParent, height = wrapContent)
                cardView {
                    linearLayout {
                        id = R.id.percentFrameLayout
                        backgroundResource = selectableItemBackground(context)
                        lparams(width = matchParent, height = dip(80))
                        orientation = LinearLayout.VERTICAL

                        textView {
                            id = R.id.text
                            gravity = Gravity.LEFT
                            textSize = 17f
                            textColor = Color.BLACK
                        }.lparams(width = matchParent, height = matchParent) {
                            setPadding(dip(10), dip(10), dip(10), dip(10))
                        }
                    }
                }.lparams(width = matchParent, height = wrapContent) {
                    margin = dip(5)
                }
            }
        }
    }

    fun selectableItemBackground(context: Context): Int {
        var outValue = TypedValue()
        context.theme.resolveAttribute(android.R.attr.selectableItemBackground, outValue, true)
        return outValue.resourceId
    }
  • 注意这里给控件设置了id,所以你需要在res/value文件夹下创建一个名为ids.xml文件,然后添加你所需要的id
<?xml version="1.0" encoding="utf-8"?>
<resources>
    <item name="percentFrameLayout" type="id"/>
    <item name="text" type="id"/>
</resources>

然后在adapter里使用就是这样:

 override fun onCreateViewHolder(parent: ViewGroup?, viewType: Int): ViewHolder {
        return ViewHolder(createItemView(context))
    }

 inner class ViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
        var tvTitle = itemView.findViewById(R.id.text) as TextView
    }

引用xml布局文件

当然anko也不是那么死板的工具,他也可以引用xml布局文件很简单;使用include字符就行

include<View>(R.layout.common_toolbar)

就像这样就可以了

高级用法 - 如何使用自定义view

在实际开发中anko给我们提供的控件肯定是不能满足我们的需求的,那我们怎么把自己的自定义控件使用到anko中呢,其实很简单

  • 首先创建一个Views.kt文件
  • 在该文件中添加下面代码
inline fun ViewManager.customizeView(theme: Int = 0) = customizeView(theme) {}

inline fun ViewManager.customizeView(theme: Int = 0, init: CustomizeView.() -> Unit) = ankoView({ CustomizeView(it) }, theme, init)
      *  这里的```CustomizeView```是你的自定义View
      *  这里的```customizeView```是你用在anko里的名称,一般就是自定义view的名称,首字母小写这样的写法
  • 然后就可以在anko里这样使用了customizeView{}

anko的基本使用就到这了,后续若遇到其他问题会补充上的;若 想看anko具体在项目里使用;可以看看我这个项目,一个使用 kotlin编写的知乎日报 ;;若觉得对你有用欢迎点赞和star

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容