Aviso onTouchListener: onTouch deve chamar View # performClick quando um clique for detectado

108

Eu criei um onTouchListener. Infelizmente, o método onTouch () throwsme avisou :

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

O que isso significa? Não encontrei nenhuma informação sobre este aviso. Aqui está o código completo:

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
fonte
17
você deseja se livrar do aviso, basta ligar v.performClick(). A implementação irá tocar um pouco de som (se estiver habilitado em seu dispositivo) e chamar o onClickListener, que provavelmente você não substituiu
Blackbelt
Sua resposta estava correta. Obrigado
Trzy Gracje

Respostas:

128

Aqui está:

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;
}
Secko
fonte
14
não deveria ser chamado apenas quando o evento == MotionEvent.Action_UP, ou algo parecido? Além disso, retornar "false" não chamaria o método de clique original, então você deveria retornar "true" em vez disso?
desenvolvedor Android de
@longilong Bem, se desejar, você também pode configurá-lo para usar caixa de comutação, mas é logicamente o mesmo ...
desenvolvedor do Android
3
Eu tenho um aviso semelhante. Mas meu cenário é um pouco diferente, pois estou definindo um OnTouchListener anônimo ao configurar o RecyclerView (chamando myRecyclerView.setOnTouchListener. O Android Studio está marcando toda a classe anônima com um aviso semelhante e mesmo se eu adicionar v.performClick o aviso permanece.
fr4gus
8
se retornarmos false a partir deste método, não devemos ser obrigados a chamar performClick, certo? Neste caso, não entendo por que o aviso de fiapo ainda está lá
Jiechao Wang
9
Não faz nada para corrigir o aviso.
TheLibrarian
2

Você pode suprimir o fiapo

@SuppressLint("ClickableViewAccessibility")

Você deve ligar para performClick()dentro onTouchEvent().

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

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

ou

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

        v.performClick();

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

OnTouch fluxo

Leia mais aqui

yoAlex5
fonte
1
Você não tem que substituir performClick(). Compartilhe seu código
yoAlex5
1

Caso você não esteja usando um Custom Viewque se sobrepõe explicitamente onPerformClick, o aviso não será removido apenas seguindo a resposta de Secko.

Além de sua resposta, para fazer o mesmo em classes como android.widget.Buttonou Buttonvocê precisa fazer uma visualização customizada simples que amplie a visualização de destino.

Exemplo:

A classe de visualização personalizada:

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;
        }
    });

Problemas atuais: o aviso é resolvido pelo IDE, mas não consigo ver isso praticamente realizando uma ação de clique em um dispositivo Android real.

EDIT : Corrigido o recebimento do evento click: UseView.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
fonte
0

basta chamar o método performClick, como este:

@Override
public boolean onTouch(View v, MotionEvent event) {
    v.performClick();
    Tools.hideKeyboard(getActivity(), getView());
    getView().clearFocus();
    return false;
}   
Clairton Luz
fonte
0

Tive um problema semelhante com um MultiTouchListenere resolvi implementando um GestureDetectore ouvindo um SingleTap(isso não remove o aviso, mas começa a disparar onClickeventos na minha visualização)

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)
        }

    }

}
Osvel Alvarez Jacomino
fonte