av一区二区在线观看_亚洲男人的天堂网站_日韩亚洲视频_在线成人免费_欧美日韩精品免费观看视频_久草视

您的位置:首頁技術(shù)文章
文章詳情頁

Android自定義view實(shí)現(xiàn)滑動(dòng)解鎖效果

瀏覽:12日期:2022-09-18 11:59:20

本文實(shí)例為大家分享了Android自定義view實(shí)現(xiàn)滑動(dòng)解鎖的具體代碼,供大家參考,具體內(nèi)容如下

1. 需求如下:

近期需要做一個(gè)類似屏幕滑動(dòng)解鎖的功能,右劃開始,左劃暫停。

2. 需求效果圖如下

Android自定義view實(shí)現(xiàn)滑動(dòng)解鎖效果

3. 實(shí)現(xiàn)效果展示

Android自定義view實(shí)現(xiàn)滑動(dòng)解鎖效果

4. 自定義view如下

/** * Desc 自定義滑動(dòng)解鎖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 = 0startImgView.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) / 2hintView.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.xval 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 = '滑動(dòng)停止' ResourcesCompat.getDrawable(resources, R.drawable.f1_svg_left_arrow, null)} else { hintView.text = '滑動(dòng)開始' 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) }}

這里需要注意一點(diǎn):頁面過長時(shí),ScrollView和SlideSwitchButton滑動(dòng)事件會(huì)沖突,所以需要吧scrollView傳進(jìn)來

5. 調(diào)用方式如下

/** * Desc 自定義滑動(dòng)解鎖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 = scrollViewbtn_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框架,集工具類、自定義組件、網(wǎng)絡(luò)請(qǐng)求框架一體,感覺用起來有些厚重,接下來會(huì)抽時(shí)間做拆分,ZyFrame保留網(wǎng)絡(luò)請(qǐng)求功能,ZyUI專做自定義組件,ZyTool專做工具類,大概就這樣。

以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Android
相關(guān)文章:
主站蜘蛛池模板: 久久成人免费视频 | 亚洲精品久久久一区二区三区 | 日韩一区二区三区在线 | 青青青伊人| 久久精品视频网站 | 午夜精品久久久久久久久久久久久 | 性国产xxxx乳高跟 | 福利视频二区 | 久久久激情视频 | 91中文字幕在线观看 | 国产成人精品视频在线观看 | 看毛片网站 | 亚洲综合天堂网 | 黄色一级片视频 | 超碰精品在线 | 暖暖成人免费视频 | 日韩视频一区在线观看 | 精品视频在线观看 | 日韩中文字幕在线播放 | 欧美日韩一区精品 | 久久com | 精品av久久久久电影 | 欧美亚洲国产日韩 | 亚洲理论在线观看电影 | 久久狼人天堂 | 黄色一级免费看 | 国产91久久精品一区二区 | 草草视频在线观看 | 日韩精品人成在线播放 | 精品久久久久久久久久久久 | 一区二区三区精品视频 | 99久久婷婷国产综合精品 | 国产精品色av | 91免费看片 | 午夜精品一区二区三区在线播放 | 久久精品国产一区老色匹 | 成人精品网 | 免费的一级视频 | 欧美性视频在线播放 | 五月天婷婷激情 | 人成在线 |