Android 自定义View之展开收起的Layout

简介: Android 自定义View之展开收起的Layout

效果

image.png


分析

效果图来看,点击事件触发view的展开收起,并在收起状态下保留了第一个子view显示,这个展开收起其实就是view的高度变化,所以只要控制好高度,就能很简单的实现这个效果。


步骤

1.初始化参数 设置方向等

2.根据动画执行进度计算高度

初始化

class ExpandLinearLayout : LinearLayout {
    //是否展开,默认展开
    private var isOpen = true
    //第一个子view的高度,即收起保留高度
    private var firstChildHeight = 0
    //所有子view高度,即总高度
    private var allChildHeight = 0
    /**
     * 动画值改变的时候 请求重新布局
     */
    private var animPercent: Float = 0f
    constructor(context: Context) : super(context) {
        initView()
    }
    constructor(context: Context, attributeSet: AttributeSet) : super(context, attributeSet) {
        initView()
    }
    constructor(context: Context, attributeSet: AttributeSet, defStyleAttr: Int) : super(
        context,
        attributeSet,
        defStyleAttr
    ) {
        initView()
    }
    private fun initView() {
        //横向的话 稍加修改计算宽度即可
        orientation = VERTICAL
        animPercent = 1f
        isOpen = true
    }
}

定义一个类ExpandLinearLayout ,继承自LinearLayout,当然也可以是其他的view。


然后重写构造方法,并在构造方法里面调用initView方法。


在initView方法中,我们对一些参数进行初始化操作,比如方向、默认展开。


计算高度

ok,这个就是重点了。


因为只是view本身高度的变化,我们只需要重写onMeasure去计算高度即可。


来看onMeasure:


 

override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec)
        //重置高度
        allChildHeight = 0
        firstChildHeight = 0
        if (childCount > 0) {
            //遍历计算高度
            for (index in 0 until childCount) {
                //这个地方实际使用中除了measuredHeight,以及margin等,也要计算在内
                if (index == 0) {
                    firstChildHeight = getChildAt(index).measuredHeight
                    +getChildAt(index).marginTop + getChildAt(index).marginBottom
                    +this.paddingTop + this.paddingBottom
                }
                //实际使用时或包括padding等
                allChildHeight += getChildAt(index).measuredHeight + getChildAt(index).marginTop + getChildAt(index).marginBottom
                //最后一条的时候 加上当前view自身的padding
                if (index == childCount - 1) {
                    allChildHeight += this.paddingTop + this.paddingBottom
                }
            }
            // 根据是否展开设置高度
            if (isOpen) {
                setMeasuredDimension(
                    widthMeasureSpec,
                    firstChildHeight + ((allChildHeight - firstChildHeight) * animPercent).toInt()
                )
            } else {
                setMeasuredDimension(
                    widthMeasureSpec,
                    allChildHeight - ((allChildHeight - firstChildHeight) * animPercent).toInt()
                )
            }
        }
    }

onMeasure里面也是分了两个步骤的:


遍历计算高度

//遍历计算高度
for (index in 0 until childCount) {
    //这个地方实际使用中除了measuredHeight,以及margin等,也要计算在内
    if (index == 0) {
        firstChildHeight = getChildAt(index).measuredHeight
        +getChildAt(index).marginTop + getChildAt(index).marginBottom
        +this.paddingTop + this.paddingBottom
    }
    //实际使用时或包括padding等
    allChildHeight += getChildAt(index).measuredHeight + getChildAt(index).marginTop + getChildAt(index).marginBottom
    //最后一条的时候 加上当前view自身的padding
    if (index == childCount - 1) {
        allChildHeight += this.paddingTop + this.paddingBottom
    }
}

来看第一个if判断,记录了第一个子view的高度,这里需要注意,除了measuredHeight,margin也要算上,而且父view的内边距padding也要加上,因为如果父view的padding很大的话,收起时view可能会显示不出来的。


然后就是总高度的计算,道理同上。


来看最后一个if判断,同样总高度计算完之后也要加上父view的上下padding,才是完整的高度。


第一个判断可以理解为收起状态的高度,第二个判断可以理解为展开状态的高度。


展开收起逻辑

// 根据是否展开设置高度
if (isOpen) {
    setMeasuredDimension(
        widthMeasureSpec,
        firstChildHeight + ((allChildHeight - firstChildHeight) * animPercent).toInt()
    )
} else {
    setMeasuredDimension(
        widthMeasureSpec,
        allChildHeight - ((allChildHeight - firstChildHeight) * animPercent).toInt()
    )
}

因为第一个子view是保留显示的,所以在计算的时候都需要减去第一个子view的高度,就是剩余高度。


剩余高度可以很简单的计算出来,但是如何在显示的时候不突兀呢。


这里加一个动画,根据动画的执行进度来计算。


展开:第一个子view的高度 + 剩余高度 × 0到1的Float动画值

收起:总高度 - 剩余高度 × 0到1的Float动画值


author:yechaoa


动画

写一个方法控制展开收起,并在展开收起的时候执行动画。


fun toggle(): Boolean {
        isOpen = !isOpen
        startAnim()
        return isOpen
    }
    /**
     * 执行动画的时候 更改 animPercent 属性的值 即从0-1
     */
    @SuppressLint("AnimatorKeep")
    private fun startAnim() {
        //ofFloat,of xxxX 根据参数类型来确定
        //1,动画对象,即当前view。2.动画属性名。3,起始值。4,目标值。
        val animator = ObjectAnimator.ofFloat(this, "animPercent", 0f, 1f)
        animator.duration = 500
        animator.start()
    }

并修改我们的动画参数:


 

/**
     * 动画值改变的时候 请求重新布局
     */
    private var animPercent: Float = 0f
        set(value) {
            field = value
            requestLayout()
        }

set value的时候调用requestLayout(),重新执行onMeasure。


调用

xml

<com.yechaoa.customviews.expand.ExpandLinearLayout
        android:id="@+id/ell"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="#f5f5f5"
        android:orientation="vertical"
        android:padding="10dp">
        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:padding="10dp"
            android:text="@string/app_name"
            android:textColor="@android:color/holo_red_dark"
            android:textSize="20sp" />
        ...
    </com.yechaoa.customviews.expand.ExpandLinearLayout>

代码

 

ll_btn.setOnClickListener {
            val toggle = ell.toggle()
            tv_tip.text = if (toggle) "收起" else "展开"
        }

扩展

横向:计算高度变成计算宽度即可

高度:可以根据xml自定义属性来控制保留高度

总结

总的来说,效果还是比较实用的,难度系数也不高,可以根据扩展自己去进一步完善。


如果对你有一点点帮助,点个赞呗 ^ _ ^


Github

https://github.com/yechaoa/CustomViews

目录
相关文章
|
2月前
|
缓存 前端开发 Android开发
安卓开发中的自定义视图:从零到英雄
【10月更文挑战第42天】 在安卓的世界里,自定义视图是一块画布,让开发者能够绘制出独一无二的界面体验。本文将带你走进自定义视图的大门,通过深入浅出的方式,让你从零基础到能够独立设计并实现复杂的自定义组件。我们将探索自定义视图的核心概念、实现步骤,以及如何优化你的视图以提高性能和兼容性。准备好了吗?让我们开始这段创造性的旅程吧!
43 1
|
3月前
|
数据可视化 Android开发 开发者
安卓应用开发中的自定义View组件
【10月更文挑战第5天】在安卓应用开发中,自定义View组件是提升用户交互体验的利器。本篇将深入探讨如何从零开始创建自定义View,包括设计理念、实现步骤以及性能优化技巧,帮助开发者打造流畅且富有创意的用户界面。
133 0
|
2月前
|
XML 前端开发 Android开发
Android:UI:Drawable:View/ImageView与Drawable
通过本文的介绍,我们详细探讨了Android中Drawable、View和ImageView的使用方法及其相互关系。Drawable作为图像和图形的抽象表示,提供了丰富的子类和自定义能力,使得开发者能够灵活地实现各种UI效果。View和ImageView则通过使用Drawable实现了各种图像和图形的显示需求。希望本文能为您在Android开发中使用Drawable提供有价值的参考和指导。
51 2
|
2月前
|
搜索推荐 前端开发 Android开发
安卓应用开发中的自定义视图实现
【10月更文挑战第30天】在安卓开发的海洋中,自定义视图是那抹不可或缺的亮色,它为应用界面的个性化和交互体验的提升提供了无限可能。本文将深入探讨如何在安卓平台创建自定义视图,并展示如何通过代码实现这一过程。我们将从基础出发,逐步引导你理解自定义视图的核心概念,然后通过一个实际的代码示例,详细讲解如何将理论应用于实践,最终实现一个美观且具有良好用户体验的自定义控件。无论你是想提高自己的开发技能,还是仅仅出于对安卓开发的兴趣,这篇文章都将为你提供价值。
|
2月前
|
Android开发 开发者 UED
安卓开发中自定义View的实现与性能优化
【10月更文挑战第28天】在安卓开发领域,自定义View是提升应用界面独特性和用户体验的重要手段。本文将深入探讨如何高效地创建和管理自定义View,以及如何通过代码和性能调优来确保流畅的交互体验。我们将一起学习自定义View的生命周期、绘图基础和事件处理,进而探索内存和布局优化技巧,最终实现既美观又高效的安卓界面。
49 5
|
3月前
|
缓存 数据处理 Android开发
在 Android 中使用 RxJava 更新 View
【10月更文挑战第20天】使用 RxJava 来更新 View 可以提供更优雅、更高效的解决方案。通过合理地运用操作符和订阅机制,我们能够轻松地处理异步数据并在主线程中进行 View 的更新。在实际应用中,需要根据具体情况进行灵活运用,并注意相关的注意事项和性能优化,以确保应用的稳定性和流畅性。可以通过不断的实践和探索,进一步掌握在 Android 中使用 RxJava 更新 View 的技巧和方法,为开发高质量的 Android 应用提供有力支持。
|
3月前
|
缓存 调度 Android开发
Android 在子线程更新 View
【10月更文挑战第21天】在 Android 开发中,虽然不能直接在子线程更新 View,但通过使用 Handler、AsyncTask 或 RxJava 等方法,可以实现子线程操作并在主线程更新 View 的目的。在实际应用中,需要根据具体情况选择合适的方法,并注意相关的注意事项和性能优化,以确保应用的稳定性和流畅性。可以通过不断的实践和探索,进一步掌握在子线程更新 View 的技巧和方法,为开发高质量的 Android 应用提供支持。
58 2
|
3月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
|
3月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
32 2
|
Android开发
自定义android 4.0以上的对话框风格
做个笔记,这里是Dialog的风格,如果是用AlertDialog创建的,不能直接用。在styles.xml的写法: 22sp @color/font_green 1 true @st...
712 0