fix/prevent_button_spam_on_scroll_to_replied_message - and VisibleMessageViews in general (#983)

* WIP

* Minor tidyup

* Removed some blank lines

* Fix typo

* Tweaks

---------

Co-authored-by: alansley <aclansley@gmail.com>
Co-authored-by: ThomasSession <thomas.r@getsession.org>
pull/1712/head
AL-Session 3 weeks ago committed by GitHub
parent 58e142c5d2
commit 635cee1585
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -155,6 +155,5 @@ class QuoteView @JvmOverloads constructor(context: Context, attrs: AttributeSet?
}
interface QuoteViewDelegate {
fun cancelQuoteDraft()
}

@ -8,6 +8,7 @@ import android.graphics.Rect
import android.graphics.drawable.ColorDrawable
import android.os.Handler
import android.os.Looper
import android.os.SystemClock
import android.util.AttributeSet
import android.view.Gravity
import android.view.HapticFeedbackConstants
@ -123,6 +124,10 @@ class VisibleMessageView : FrameLayout {
var onLongPress: (() -> Unit)? = null
val messageContentView: VisibleMessageContentView get() = binding.messageContentView.root
// Prevent button spam
val MINIMUM_DURATION_BETWEEN_CLICKS_ON_SAME_VIEW_MS = 500L
var lastClickTimestampMS = 0L
companion object {
const val swipeToReplyThreshold = 64.0f // dp
const val longPressMovementThreshold = 10.0f // dp
@ -613,15 +618,22 @@ class VisibleMessageView : FrameLayout {
onLongPress?.invoke()
}
fun onContentClick(event: MotionEvent) {
binding.messageContentView.root.onContentClick(event)
}
private fun clickedTooFast() = (SystemClock.elapsedRealtime() - lastClickTimestampMS < MINIMUM_DURATION_BETWEEN_CLICKS_ON_SAME_VIEW_MS)
// Note: `onPress` is called BEFORE `onContentClick` is called, so we only filter here rather than
// in both places otherwise `onContentClick` will instantly fail the button spam test.
private fun onPress(event: MotionEvent) {
// Don't process the press if it's too soon after the last one..
if (clickedTooFast()) return
// ..otherwise take note of the time and process the event.
lastClickTimestampMS = SystemClock.elapsedRealtime()
onPress?.invoke(event)
pressCallback = null
}
fun onContentClick(event: MotionEvent) = binding.messageContentView.root.onContentClick(event)
private fun maybeShowUserDetails(publicKey: String, threadID: Long) {
UserDetailsBottomSheet().apply {
arguments = bundleOf(

@ -160,8 +160,13 @@ class MediaSendActivity : ScreenLockActionBarActivity(), MediaPickerFolderFragme
}
override fun onMediaSelected(media: Media) {
viewModel.onSingleMediaSelected(this, media)
navigateToMediaSend(recipient!!)
try {
viewModel.onSingleMediaSelected(this, media)
navigateToMediaSend(recipient!!)
} catch (e: Exception){
Log.e(TAG, "Error selecting media", e)
Toast.makeText(this, R.string.errorUnknown, Toast.LENGTH_LONG).show()
}
}
override fun onAddMediaClicked(bucketId: String) {

@ -0,0 +1,21 @@
package org.thoughtcrime.securesms.util
import android.os.SystemClock
import android.view.View
// Listener class that only accepts clicks at a given interval to prevent button spam.
// Note: While this cannot be used on conversation views without interfering with motion events it may still be useful.
class SafeClickListener(
private var minimumClickIntervalMS: Long = 500L,
private val onSafeClick: (View) -> Unit
) : View.OnClickListener {
private var lastClickTimestampMS: Long = 0L
override fun onClick(v: View) {
// Ignore any follow-up clicks if the minimum interval has not passed
if (SystemClock.elapsedRealtime() - lastClickTimestampMS < minimumClickIntervalMS) return
lastClickTimestampMS = SystemClock.elapsedRealtime()
onSafeClick(v)
}
}

@ -121,3 +121,13 @@ fun EditText.addTextChangedListener(listener: (String) -> Unit) {
}
})
}
// Listener class that only accepts clicks at given interval to prevent button spam - can be used instead
// of a standard `onClickListener` in many places. A separate mechanism exists for VisibleMessageViews to
// prevent interfering with gestures.
fun View.setSafeOnClickListener(clickIntervalMS: Long = 1000L, onSafeClick: (View) -> Unit) {
val safeClickListener = SafeClickListener(minimumClickIntervalMS = clickIntervalMS) {
onSafeClick(it)
}
setOnClickListener(safeClickListener)
}

@ -17,6 +17,7 @@ abstract class Message {
var recipient: String? = null
var sender: String? = null
var isSenderSelf: Boolean = false
var groupPublicKey: String? = null
var openGroupServerMessageID: Long? = null
var serverHash: String? = null

Loading…
Cancel
Save