Предупреждение onTouchListener: onTouch должен вызывать View # performClick при обнаружении щелчка

110

Я создал onTouchListener. К сожалению, метод onTouch () throwsполучил предупреждение:

com/calculator/activitys/Calculator$1#onTouch should call View#performClick when a click is detected

Что это значит? Я не нашел информации об этом предупреждении. Вот полный код:

LinearLayout llCalculatorContent = (LinearLayout) fragmentView.findViewById(R.id.calculator_content);

llCalculatorContent.setOnTouchListener(new View.OnTouchListener() {

    @Override
    public boolean onTouch(View v, MotionEvent event) {
        Tools.hideKeyboard(getActivity(), getView());
        getView().clearFocus();
        return false;
    }   
});
Trzy Gracje
источник
17
вы хотите избавиться от предупреждения, просто позвоните v.performClick(). Реализация будет воспроизводить небольшой звук (если он включен на вашем устройстве) и вызывать onClickListener, который вы, вероятно, не переопределили
Blackbelt
Ваш ответ был правильным. Спасибо
Trzy Gracje

Ответы:

131

Ну вот:

public boolean onTouch(View v, MotionEvent event) {
    switch (event.getAction()) {
    case MotionEvent.ACTION_DOWN:
        //some code....
        break;
    case MotionEvent.ACTION_UP:
        v.performClick();
        break;
    default:
        break;
    }
    return true;
}
Секо
источник
14
не следует ли его вызывать только тогда, когда событие == MotionEvent.Action_UP или что-то в этом роде? Кроме того, не будет ли возврат «false» на самом деле вызывать исходный метод щелчка, поэтому вместо этого вы должны вернуть «true»?
разработчик Android
@longilong Ну, если хотите, вы также можете настроить его на использование switch-case, но логически это то же самое ...
разработчик Android
3
У меня аналогичное предупреждение. Но мой сценарий немного отличается, так как я устанавливаю анонимный OnTouchListener при настройке RecyclerView (вызывая myRecyclerView.setOnTouchListener. Android Studio отмечает весь анонимный класс аналогичным предупреждением, и даже если я добавлю v.performClick, предупреждение останется.
fr4gus
8
если мы вернем false из этого метода, нам не нужно будет вызывать performClick, верно? В этом случае я не понимаю, почему предупреждение о ворсинках все еще существует
Цзечао Ван
9
Ничего не делает для устранения предупреждения.
TheLibrarian 03
2

Вы можете подавить ворс

@SuppressLint("ClickableViewAccessibility")

Вам следует зайти performClick()внутрь onTouchEvent().

@Override
public boolean onTouchEvent(MotionEvent event) {
    //A logic 

    performClick();
    return super.onTouchEvent(event);
}

или

findViewById(R.id.view1).setOnTouchListener(new View.OnTouchListener() {
    @Override
    public boolean onTouch(View v, MotionEvent event) {

        v.performClick();

        return v.onTouchEvent(event);
    }
});

OnTouch течь

Подробнее здесь

yoAlex5
источник
1
Вам не нужно отменять performClick().
Поделитесь,
1

Если вы не используете, Custom Viewкоторый явно переопределяет onPerformClick, предупреждение не будет удалено, просто следуя ответу Секко.

В дополнение к его ответу, чтобы сделать то же самое с такими классами, как android.widget.Buttonили, Buttonвам нужно создать простое настраиваемое представление, которое расширяет целевое представление.

Пример :

Пользовательский класс просмотра:

public class UselessButton extends AppCompatButton {
    public UselessButton(Context context) {
        super(context);
    }

    public UselessButton(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public UselessButton(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    @Override
    public boolean performClick() {
        return super.performClick();
    }
}

XML :

<stackoverflow.onEarth.UselessButton
    android:id="@+id/left"
    android:layout_width="60dp"
    android:layout_height="60dp"
    android:layout_marginStart="16dp"
    android:layout_marginTop="16dp"
    android:layout_marginEnd="8dp"
    android:layout_marginBottom="8dp"
    android:background="@drawable/left"
    app:layout_constraintBottom_toBottomOf="parent"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintHorizontal_bias="0.16"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintTop_toTopOf="parent"
    app:layout_constraintBaseline_toBaselineOf="@+id/right"
    app:layout_constraintVertical_bias="0.5" />

Java :

    left.setOnTouchListener((v, event) -> {
        if (event.getAction() == MotionEvent.ACTION_DOWN) {
            enLeft = 1;
            enRight = 0;
            return true;
        } else if (event.getAction() == MotionEvent.ACTION_UP) {
            enLeft = 0;
            v.performClick();
            return false;
        } else {
            return false;
        }
    });

Текущие проблемы: предупреждение устраняется в среде IDE, но на реальном устройстве Android оно практически не выполняет щелчок.

РЕДАКТИРОВАТЬ : исправлено получение события щелчка: используйтеView.setPressed(boolean)

down.setOnTouchListener((v, event) -> {
    if (event.getAction() == MotionEvent.ACTION_DOWN) {
        enFront = 0;
        enBack = 1;
        left.setPressed(true);
        return true;
    } else if (event.getAction() == MotionEvent.ACTION_UP) {
        enBack = 0;
        v.performClick();
        v.setPressed(false);
        return false;
    } else {
        return false;
    }
APEXBoi
источник
0

просто вызовите метод performClick, например:

@Override
public boolean onTouch(View v, MotionEvent event) {
    v.performClick();
    Tools.hideKeyboard(getActivity(), getView());
    getView().clearFocus();
    return false;
}   
Клэртон Луз
источник
0

У меня была аналогичная проблема с a, MultiTouchListenerи я решил, что она реализует GestureDetectorи прослушивает SingleTap(это не удаляет предупреждение, но начинает запускать onClickсобытия в моем представлении)

class TouchListener(context: Context) : MultiTouchListener() {

    private var tochedView: View? = null
    private var mGestureDetector = CustomGestureDetector()
    private var gestureDetector: GestureDetector = GestureDetector(context, mGestureDetector)

    @SuppressLint("ClickableViewAccessibility")
    override fun onTouch(view: View?, event: MotionEvent?): Boolean {
        val aux = super.onTouch(view, event)

        tochedView = view
        gestureDetector.onTouchEvent(event)

        return aux
    }

    private inner class CustomGestureDetector: GestureDetector.SimpleOnGestureListener() {

        override fun onSingleTapUp(e: MotionEvent?): Boolean {
            // this will be called even when a double tap is
            tochedView?.performClick()
            return super.onSingleTapUp(e)
        }

        override fun onSingleTapConfirmed(e: MotionEvent?): Boolean {
            // this will only be called after the detector is confident that the
            // user's first tap is not followed by a second tap leading to a double-tap gesture.
            tochedView?.performClick()
            return super.onSingleTapConfirmed(e)
        }

    }

}
Освел Альварес Джакомино
источник