39

How to override performClick in Kotlin to avoid warning?

next.setOnTouchListener(View.OnTouchListener { view, motionEvent ->
        when (motionEvent.action){
            MotionEvent.ACTION_DOWN -> {
                val icon: Drawable = ContextCompat.getDrawable(activity.applicationContext, R.drawable.layer_bt_next)
                icon.setColorFilter(Color.GRAY, PorterDuff.Mode.MULTIPLY)
                next.setImageDrawable(icon)
            }
            MotionEvent.ACTION_UP -> {
                //view.performClick()
                next.setImageResource(R.drawable.layer_bt_next)
            }
        }
        return@OnTouchListener true
    })

view.performClick does not work.

lambda
  • 990
  • 1
  • 10
  • 29

7 Answers7

55

Try this way :

 next.setOnTouchListener(object : View.OnTouchListener {
        override fun onTouch(v: View?, event: MotionEvent?): Boolean {
            when (event?.action) {
                MotionEvent.ACTION_DOWN -> //Do Something
            }

            return v?.onTouchEvent(event) ?: true
        }
    })
Manohar
  • 22,116
  • 9
  • 108
  • 144
15

Okay, I solved my own problem by overriding the OnTouch listener.

override fun onTouch(view: View, motionEvent: MotionEvent): Boolean {
    when (view) {
        next -> {
            Log.d("next", "yeyy")
            when (motionEvent.action){
                MotionEvent.ACTION_DOWN -> {
                    val icon: Drawable = ContextCompat.getDrawable(activity.applicationContext, R.drawable.layer_bt_next)
                    icon.setColorFilter(Color.GRAY, PorterDuff.Mode.MULTIPLY)
                    next.setImageDrawable(icon)
                }
                MotionEvent.ACTION_UP -> {
                    view.performClick()
                    next.setImageResource(R.drawable.layer_bt_next)
                }
            }
        }
        previous -> {
            //ingredients here XD
        }
    }
    return true
}

And in that way, I can call single onTouch and implement it to many button and also can use the onClick by :

view.performClick()

Don't forget to implement :

View.OnTouchListener

And set the listener :

next.setOnTouchListener(this)
previous.setOnTouchListener(this)
lambda
  • 990
  • 1
  • 10
  • 29
8

I don't think your solution will actually solve them problem presented by the warning. The warning states that certain accessibility functions use performClick() to activate buttons. If you look in the View class, the performClick() funtions calls the onClickListener directly, meaning the code in the onTouchListener will not be executed (next.setImageResource(R.drawable.layer_bt_next)) for these accessibility functions, since the view will never be physically touched, and thus your onTouch code won't run. You have to do one of either:

  1. Subclass the view you are setting the onTouchListener on, and override performClick to execute the code, or
  2. Set an onClickListener on the view that executes the code.

You could just implement onClickListener in your onTouchListener class and manually call onClick() from your onTouchListener (where you have view.performClick() now), and then move your executable code to the onClick override. You would also have to set BOTH onTouchListener and onClickListener on your views.

Tony Danilov
  • 430
  • 3
  • 11
  • 22
mhswtf
  • 165
  • 1
  • 7
  • Can you just give example in terms of code ? for better understanding. Thanks – Ketan Ramani Mar 07 '18 at 10:08
  • Doesn't make any sense to me. I have code that must not be executed onClick(). It's kind of progress indication for a long touch. Accessibility doesn't matter. If you can't touch you can't use it. Maybe I really have to ignore the warning. – The incredible Jan Feb 22 '21 at 15:03
  • @TheincredibleJan - The warning is for accessibility purposes, so if that is not important or relevant to you, then yes, you should ignore it. – mhswtf Apr 22 '21 at 10:45
2

I'm not sure this is the same issue you saw, but since I found this page searching for my issue, I thought I'd add my experience to help others :)

In my case the warning was being generated because the nullable view could have been of type Void. Calling the following:

nullableView?.setOnTouchListener(this)

produced the error:

Custom view Void has setOnTouchListener called on it but does not override performClick

Performing a null check and casting to a View before setting the listener solved for me in this case, since View will override performClick:

if (nullableView != null) (nullableView as View).setOnTouchListener(this)
Taslim Oseni
  • 6,086
  • 10
  • 44
  • 69
head in the codes
  • 1,159
  • 12
  • 24
2

After a ton of digging, and not being able to fix my variation of this issue with anything in this thread, I finally found a fix. Maybe it will work for some of you. I had this widget listener setter in my MainActivity onCreate function:

findViewById<TextView>(R.id.tvAnimalList).setOnTouchListener { v, event ->
    mGestureDetector.onTouchEvent(event)
}

Which results in the warnings:

  • 'onTouch' lambda should call 'View#performClick' when a click is detected
  • Custom view "TextView" has 'setOnTouchListener' called on it but does not override 'performClick'

First, I added a call to v.performClick(), which got rid of the first warning. Like this:

findViewById<TextView>(R.id.tvAnimalList).setOnTouchListener { v, event ->
    v.performClick()
    mGestureDetector.onTouchEvent(event)
}

I got rid of the second warning by changing the findViewById cast from <TextView> to <View>. Here's my warning-free result:

findViewById<View>(R.id.tvAnimalList).setOnTouchListener { v, event ->
    v.performClick()
    mGestureDetector.onTouchEvent(event)
}
Carmen DiMichele
  • 555
  • 4
  • 18
0
  private fun closeKeyboard(binding: ContollerMeterBinding) {
    binding.scrollView.apply {
        setOnTouchListener(OnTouchListener { v, event ->
            if (event != null && event.action == MotionEvent.ACTION_MOVE) {
                val imm =
                    activity?.getSystemService(Context.INPUT_METHOD_SERVICE) as InputMethodManager
                val isKeyboardUp = imm.isAcceptingText
                if (isKeyboardUp) {
                    imm.hideSoftInputFromWindow(v.windowToken, 0)
                }
            }
            performClick()
            false
        })
    }
}

This works for me: (not directly related to onTouch event but yields the same warning, might be helpful to someone)

Samir Ramic
  • 29
  • 1
  • 6
-1
 takePhotoButton.setOnTouchListener { _, motionEvent ->
            when (motionEvent.action) {
                MotionEvent.ACTION_DOWN -> {
                    //when user touch down
                }
                MotionEvent.ACTION_UP -> {
                    //when user touch release
                }
            }
            true
        }
Mohamed Ben Romdhane
  • 1,005
  • 3
  • 11
  • 22