onTouchListener警告:onTouchはクリックが検出されたときにView#performClickを呼び出す必要があります


108

を作成しました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;
    }   
});

17
警告を取り除きたい場合は、を呼び出してくださいv.performClick()。実装は少しサウンドを再生し(デバイスで有効になっている場合)、おそらくオーバーライドされていないonClickListenerを呼び出します
Blackbelt

正解でした。ありがとう
Trzy Gracje 2014

回答:


128

どうぞ:

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
event == MotionEvent.Action_UPまたはそのような場合にのみ呼び出されるべきではありませんか?また、「false」を返すと実際には元のクリックメソッドが呼び出されないので、代わりに「true」を返す必要がありますか?
Android開発者

@longilongさて、あなたはまた、スイッチケースを使用するように設定することができたい場合、それは論理的に同じ...だ
Androidデベロッパー

3
同様の警告があります。ただし、RecyclerViewを設定するときに匿名のOnTouchListenerを設定しているため(myRecyclerView.setOnTouchListenerを呼び出しているため、私のシナリオは少し異なります。AndroidStudioは、匿名クラス全体に同様の警告をマークし、v.performClickを追加しても、警告は残ります。
fr4gus 2017

8
このメソッドからfalseを返す場合、を呼び出す必要はありませんperformClickよね?この場合、糸くずの警告がまだ表示される理由がわかりません
Jiechao Wang 2017

9
警告を修正するために何もしません。
TheLibrarian 2018年

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 フロー

詳細はこちら


1
オーバーライドする必要はありませんperformClick()。コードを共有してください
yoAlex5

1

Custom Viewを明示的に上書きするを使用していない場合onPerformClick、Seckoの回答に従うだけでは警告は削除されません。

彼の答えに加えて、クラスで同じことを行うために好き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;
    }

0

次のように、performClickメソッドを呼び出すだけです。

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

0

私はaで同様の問題がありMultiTouchListener、それを実装してa 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)
        }

    }

}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.