导航栏左右拖动切换

背景

交互设计了一个导航栏左右交换的功能,左右各设置一个不同的功能区,然后左右可以切换

方案设计

  1. 使用startDragAndDrop方法,进行移动
  2. 通过修改ConstraintLayout.LayoutParams变更位置

实现

布局代码

左侧代码

java 复制代码
            <androidx.constraintlayout.widget.ConstraintLayout
                android:id="@+id/cl_left"
                android:layout_width="1360dp"
                android:layout_height="match_parent"
                android:background="@color/transparent"
                android:focusable="false"
                app:layout_constraintBottom_toBottomOf="parent"
                app:layout_constraintLeft_toLeftOf="parent"
                app:layout_constraintTop_toTopOf="parent">
          </androidx.constraintlayout.widget.ConstraintLayout>

右侧代码

java 复制代码
            <androidx.constraintlayout.widget.ConstraintLayout
                android:id="@+id/cl_right"
                android:layout_width="1360dp"
                android:layout_height="match_parent"
                android:background="@color/transparent"
                android:focusable="false"
                app:layout_constraintBottom_toBottomOf="parent"
                app:layout_constraintRight_toRightOf="parent"
                app:layout_constraintTop_toTopOf="parent">
                   </androidx.constraintlayout.widget.ConstraintLayout>

移动代码

左侧设置长按点击效果,设计移动效果

kotlin 复制代码
        binding.clLeft.setOnLongClickListener {

            val data = ClipData.newPlainText("clLeft", "clLeft move item")
            val shadowBuilder = View.DragShadowBuilder(binding.clLeft)
					//这里可以携带参数,也可以修改移动的效果
            val dragState = Bundle().apply {
                putBoolean("allow_return", false)
                putInt("original_visibility", binding.clLeft.visibility)
            }
            binding.clLeft.startDragAndDrop(data, shadowBuilder, dragState, 0)



            binding.clLeft.visibility = View.INVISIBLE


            return@setOnLongClickListener false
        }

右侧设置长按点击效果,设计移动效果

kotlin 复制代码
        binding.clRight.setOnLongClickListener {
            val data = ClipData.newPlainText("clRight", "clRight move item")
            val shadowBuilder = View.DragShadowBuilder(binding.clRight)

            val dragState = Bundle().apply {
                putBoolean("allow_return", false)
                putInt("original_visibility", binding.clRight.visibility)
            }
            binding.clRight.startDragAndDrop(data, shadowBuilder, dragState, 0)

            binding.clRight.visibility = View.INVISIBLE
            return@setOnLongClickListener true
        }

设置全局移动监听

kotlin 复制代码
    private val dragListener = View.OnDragListener { _, event ->
        Log.d(TAG, "dragListener:   x = ${event.x}   y = ${event.y} ")
        when (event.action) {
            DragEvent.ACTION_DROP -> {
                // 获取拖动的View
                val draggedView = event.localState as? View

                draggedView?.let { view ->
                    // 在新位置显示(不返回原位置)
                    view.visibility = View.INVISIBLE

                }

							//完成切换
                moveChangeSide()
                true
            }

            DragEvent.ACTION_DRAG_ENDED -> {

                binding.clLeft.visibility = View.VISIBLE
                binding.clRight.visibility = View.VISIBLE

                true
            }

            else -> true
        }
    }


    binding.root.setOnDragListener(dragListener)

完成移动的代码

kotlin 复制代码
    fun setLefBar(isLeft: Boolean) {


        val params = binding.clLeft.layoutParams as ConstraintLayout.LayoutParams
        if (isLeft) {
            params.startToStart = ConstraintLayout.LayoutParams.PARENT_ID
            params.endToEnd = ConstraintLayout.LayoutParams.UNSET
        } else {
            params.startToStart = ConstraintLayout.LayoutParams.UNSET
            params.endToEnd = ConstraintLayout.LayoutParams.PARENT_ID
        }


        binding.clLeft.layoutParams = params
        binding.clLeft.requestLayout()

        binding.clLeft.visibility = View.VISIBLE
    }

遇到的问题

设置参数的时候,网上很多资料都是指向使用ConstraintSet来更新参数,但是这个方法不生效。所以这里记录一下

kotlin 复制代码
val constraintSet = ConstraintSet()
constraintSet.clone(constraintLayout = binding.clleft) // parentLayeout是ConstraintLayout的实例
constraintSet.connect( startlD = binding.clleft.id,
startSide = ConstraintSet.END, endlD = ConstraintSet.PARENTT_ID, endSide = ConstraintSet.END, margin = 0
constraintSet.connect( startID= binding.clleft.id, startSide = ConstraintSt.START, endlD = -1, endSide = ConstraintSet.START, margin =
constraintSet.constrainWidth( viewld = binding.clleft.id, vvidth=1360)//设置宽高
constraintSet.constrainHeight( viewld = binding.clleft.id,height = Constraintset.MATCH_CONSTRAINT)
constraintSet.applyTo( constraintLayout = bindiing.clleft)

总结

  1. 使用startDragAndDrop方法,进行移动
  2. 通过修改ConstraintLayout.LayoutParams变更位置
相关推荐
麦兜*1 分钟前
【springboot】图文详解Spring Boot自动配置原理:为什么@SpringBootApplication是核心?
android·java·spring boot·spring·spring cloud·tomcat
le16161611 分钟前
Android 依赖种类及区别:远程仓库依赖、打包依赖、模块依赖、本地仓库依赖
android
lxysbly11 分钟前
psp模拟器安卓版带金手指
android
云上凯歌1 小时前
02 Spring Boot企业级配置详解
android·spring boot·后端
hqiangtai1 小时前
Android 高级专家技术能力图谱
android·职场和发展
aqi001 小时前
FFmpeg开发笔记(九十七)国产的开源视频剪辑工具AndroidVideoEditor
android·ffmpeg·音视频·直播·流媒体
stevenzqzq1 小时前
Android Koin 注入入门教程
android·kotlin
炼金术2 小时前
SkyPlayer v1.1.0 - 在线视频播放功能更新
android·ffmpeg
用户276038157812 小时前
鲲鹏+昇腾:开启 AI for Science 新范式——基于PINN的流体仿真加速实践
android
此去正年少2 小时前
编写adb脚本工具对Android设备上的闪退问题进行监控分析
android·adb·logcat·ndk·日志监控