亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Android自定義view實現滑動解鎖效果

 更新時間:2021年05月31日 14:54:54   作者:左郁  
這篇文章主要為大家詳細介紹了Android自定義view實現滑動解鎖效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下

本文實例為大家分享了Android自定義view實現滑動解鎖的具體代碼,供大家參考,具體內容如下

1. 需求如下:

近期需要做一個類似屏幕滑動解鎖的功能,右劃開始,左劃暫停。

2. 需求效果圖如下

3. 實現效果展示

4. 自定義view如下

/**
 * Desc 自定義滑動解鎖View
 * Author ZY
 * Mail sunnyfor98@gmail.com
 * Date 2021/5/17 11:52
 */
@SuppressLint("ClickableViewAccessibility")
class SlideSwitchButton : ViewGroup {

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

    constructor(
        context: Context?,
        attrs: AttributeSet?,
        defStyleAttr: Int,
        defStyleRes: Int
    ) : super(context, attrs, defStyleAttr, defStyleRes)


    var duration = 300

    var isOpen = false

    var scrollView: ScrollView? = null

    var onSwitchListener: ((isOpen: Boolean) -> Unit)? = null

    private var itemHeight = 0
    private var itemPadding = 0
    private var parentWidth = 0

    private val stopImgView: ImageView by lazy {
        ImageView(context).apply {
            setImageResource(R.drawable.f1_svg_btn_stop)
        }
    }

    private val startImgView: ImageView by lazy {
        ImageView(context).apply {
            setImageResource(R.drawable.f1_svg_btn_start)
        }
    }

    private val hintView: TextView by lazy {
        TextView(context).apply {
            setTextSize(TypedValue.COMPLEX_UNIT_PX, resources.getDimension(R.dimen.dp_14))
            compoundDrawablePadding = resources.getDimension(R.dimen.dp_5).toInt()
            setTextColor(Color.parseColor("#727b9f"))
        }
    }

    init {
        setBackgroundResource(R.drawable.f1_sel_bg_slide_btn)
        addView(hintView)
        updateHint()

        addView(stopImgView)
        addView(startImgView)

        var x = 0
        startImgView.setOnTouchListener { v, event ->

            when (event.action) {
                MotionEvent.ACTION_DOWN -> {
                    scrollView?.requestDisallowInterceptTouchEvent(true)
                    x = event.x.toInt()
                }

                MotionEvent.ACTION_UP -> {

                    if (startImgView.x < (parentWidth - startImgView.width) / 2) {
                        play(false)
                    } else {
                        play(true)
                    }

                    scrollView?.requestDisallowInterceptTouchEvent(false)
                }
                MotionEvent.ACTION_MOVE -> {
                    val lastX = event.x - x
                    if (startImgView.x + lastX > parentWidth - itemPadding - startImgView.width) {
                        return@setOnTouchListener true
                    }

                    if (startImgView.x + lastX < itemPadding) {
                        return@setOnTouchListener true
                    }
                    startImgView.x += lastX
                }
            }

            return@setOnTouchListener true
        }
    }


    override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec)
        setMeasuredDimension(widthMeasureSpec, resources.getDimension(R.dimen.dp_90).toInt())
        itemPadding = resources.getDimension(R.dimen.dp_5).toInt()
        itemHeight = resources.getDimension(R.dimen.dp_80).toInt()
        parentWidth = MeasureSpec.getSize(widthMeasureSpec)
    }


    override fun onLayout(changed: Boolean, l: Int, t: Int, r: Int, b: Int) {
        stopImgView.layout(
            itemPadding,
            itemPadding,
            itemPadding + itemHeight,
            itemPadding + itemHeight
        )

        startImgView.layout(
            itemPadding,
            itemPadding,
            itemPadding + itemHeight,
            itemPadding + itemHeight
        )

        val len =
            hintView.paint.measureText(hintView.text.toString()) + resources.getDimension(R.dimen.dp_24)
        val let = (r - len) / 2
        hintView.layout(
            let.toInt(),
            resources.getDimension(R.dimen.dp_35).toInt(),
            (let + len).toInt(),
            resources.getDimension(R.dimen.dp_55).toInt()
        )
    }


    /**
     * flag tue為開始 false為停止
     */
    private fun play(flag: Boolean) {
        val mStart = startImgView.x
        val mEnd = if (flag) {
            parentWidth - itemPadding * 2 - startImgView.width.toFloat()
        } else {
            stopImgView.x - itemPadding
        }

        val animatorOBJ =
            ObjectAnimator.ofFloat(startImgView, "translationX", mStart, mEnd)
        animatorOBJ.duration = duration.toLong()
        animatorOBJ.addListener(object : Animator.AnimatorListener {
            override fun onAnimationRepeat(animation: Animator?) {

            }

            override fun onAnimationEnd(animation: Animator?) {
                updateHint(flag)
                if (flag != isOpen) {
                    isOpen = flag
                    onSwitchListener?.invoke(flag)
                }
            }

            override fun onAnimationCancel(animation: Animator?) {

            }

            override fun onAnimationStart(animation: Animator?) {

            }
        })
        animatorOBJ.start()
    }

    private fun updateHint(lock: Boolean = false) {
        val icon = if (lock) {
            hintView.text = "滑動停止"
            ResourcesCompat.getDrawable(resources, R.drawable.f1_svg_left_arrow, null)
        } else {
            hintView.text = "滑動開始"
            ResourcesCompat.getDrawable(resources, R.drawable.f1_svg_right_arrow, null)
        }
        icon?.setBounds(
            0,
            0,
            resources.getDimension(R.dimen.dp_14).toInt(),
            resources.getDimension(R.dimen.dp_12).toInt()
        )
        if (lock) {
            hintView.setCompoundDrawables(icon, null, null, null)
        } else {
            hintView.setCompoundDrawables(null, null, icon, null)
        }
    }


    fun stop() {
        play(false)
    }


    fun start() {
        play(true)
    }
}

這里需要注意一點:頁面過長時,ScrollView和SlideSwitchButton滑動事件會沖突,所以需要吧scrollView傳進來

5. 調用方式如下

/**
 * Desc 自定義滑動解鎖View
 * Author ZY
 * Mail sunnyfor98@gmail.com
 * Date 2021/5/28 17:48
 */
class SlideSwitchButtonActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.f1_act_main)

        btn_start.scrollView = scrollView

        btn_start.onSwitchListener = {
            if (it) {
                Toast.makeText(this,"開始操作",Toast.LENGTH_LONG).show()
                btn_start.start()
            } else {
                Toast.makeText(this,"停止操作",Toast.LENGTH_LONG).show()
                btn_start.stop()
            }
        }
    }

}

之前封裝了一版ZyFrame框架,集工具類、自定義組件、網絡請求框架一體,感覺用起來有些厚重,接下來會抽時間做拆分,ZyFrame保留網絡請求功能,ZyUI專做自定義組件,ZyTool專做工具類,大概就這樣。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • Flutter自定義下拉刷新時的loading樣式的方法詳解

    Flutter自定義下拉刷新時的loading樣式的方法詳解

    Flutter中的下拉刷新,我們通常RefreshIndicator,可以通過color或strokeWidth設置下拉刷新的顏色粗細等樣式,但如果要自定義自己的widget,RefreshIndicator并沒有暴露出對應的屬性,那如何修改呢,文中給大家介紹的非常詳細,需要的朋友可以參考下
    2024-01-01
  • Android開發(fā)之文件操作模式深入理解

    Android開發(fā)之文件操作模式深入理解

    本文將介紹Android開發(fā)之文件操作模式,需要了解的朋友可以參考下
    2012-12-12
  • 不依賴于Activity的Android全局懸浮窗的實現

    不依賴于Activity的Android全局懸浮窗的實現

    在Android應用開發(fā)中,經常要遇到做全局懸浮窗的效果,本文的內容主要是如何不依賴于Activity的全局懸浮窗的實現及原理,有需要的可以參考。
    2016-07-07
  • Android上傳多張圖片的實例代碼(RxJava異步分發(fā))

    Android上傳多張圖片的實例代碼(RxJava異步分發(fā))

    本篇文章主要介紹了Android上傳多張圖片的實例代碼(RxJava異步分發(fā)),具有一定的參考價值,有興趣的可以了解一下
    2017-08-08
  • Android布局之LinearLayout線性布局

    Android布局之LinearLayout線性布局

    LinearLayout是線性布局控件:要么橫向排布,要么豎向排布,下面通過本篇文章給大家介紹Android布局之LinearLayout線性布局,涉及到android linearlayout 布局相關知識,對本文感興趣的朋友一起學習吧
    2015-12-12
  • Android實現相機拍攝、選擇、圖片裁剪功能

    Android實現相機拍攝、選擇、圖片裁剪功能

    自定義控件,重寫ImageView 功能實現:點擊圓形頭像之后可以實現相冊上傳或者開啟相機,然后把得到的圖片經過剪裁,把剪裁過的圖片設置為頭像的背景圖,需要的朋友可以參考下
    2016-09-09
  • Kotlin中?和!!的區(qū)別詳細對比

    Kotlin中?和!!的區(qū)別詳細對比

    這篇文章主要給大家介紹了關于Kotlin中?和!!區(qū)別的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-05-05
  • Android 仿淘寶、京東商品詳情頁向上拖動查看圖文詳情控件DEMO詳解

    Android 仿淘寶、京東商品詳情頁向上拖動查看圖文詳情控件DEMO詳解

    本文給大家介紹android 仿淘寶、京東商品詳情頁向上拖動查看圖文詳情控件DEMO詳解,使用兩個scrollView,兩個scrollView 豎直排列,通過自定義viewGroup來控制兩個scrollView的豎直排列,以及滑動事件的處理。對android 拖動查看圖文詳情知識感興趣的朋友一起學習吧
    2016-09-09
  • Android學習之BottomSheetDialog組件的使用

    Android學習之BottomSheetDialog組件的使用

    BottomSheetDialog是底部操作控件,可在屏幕底部創(chuàng)建一個支持滑動關閉視圖。本文將通過示例詳細講解它的使用,感興趣的小伙伴可以了解一下
    2022-06-06
  • Android AsyncTask實現異步處理任務的方法詳解

    Android AsyncTask實現異步處理任務的方法詳解

    這篇文章主要介紹了Android AsyncTask實現異步處理任務的方法詳解的相關資料,需要的朋友可以參考下
    2017-04-04

最新評論