How is it possible to know which View has clicked

2020-06-19 18:30发布

问题:

I have a layout which contains some views. I want to set some actions when a user clicks anywhere in the layout, so I have set an OnClickListener for the layout which works clear. But how can I know which view was clicked? I want to assign different actions depends on the view which was clicked; or maybe no view was clicked and it was only the layout itself. So the problem is if I set OnClickListener for one of the views, both OnCLickListeners related to the layout and the view will get activated, but I want only the view action.Thanks in advance.

回答1:

Other answers are quite abstract and some are incorrect. You can't Override onClick method for an Activity as it doesn't have one (unless of course you implement an OnClickListener).

Furthermore, if you set the listener for the layout the View argument received by the onClick method will always be the layout.


You can either create your own custom layout that intercepts the clicks and check if you clicked on a view, or you can set listeners for all the views.

A somewhat cleaner solution is using a single listener and checking if it's a view that was clicked.

private final String TAG = "MainActivity";
List<View> views = new ArrayList<View>();

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    RelativeLayout rLayout = (RelativeLayout) findViewById(R.id.relativeLayout);
    TextView tv1 = (TextView) findViewById(R.id.tv1);
    TextView tv2 = (TextView) findViewById(R.id.tv2);
    TextView tv3 = (TextView) findViewById(R.id.tv3);

    views.add(tv1);
    views.add(tv2);
    views.add(tv3);

    View.OnClickListener clickListener = new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            boolean isView = false;
            for (View view : views) {
                if (v.equals(view)) {
                    isView = true;
                    break;
                }
            }

            if (isView) {
                Log.e(TAG, "Click on view");
            } else {
                Log.e(TAG, "Click on layout");
            }
        }
    };

    tv1.setOnClickListener(clickListener);
    tv2.setOnClickListener(clickListener);
    tv3.setOnClickListener(clickListener);
    rLayout.setOnClickListener(clickListener);
}

If you want different actions for different views, just create a listener for each of them.



回答2:

For example Layout has two views View1 and View2. I assume that you have already inflated them. So you have to set OnClickListiner for each of them

Layout.setOnClickListener(this);
View1.setOnClickListener(this);
View2.setOnClickListener(this);

Then you have to override method:

public void onClick(View v) 
    {
        switch(v.getId())
        {
            case R.id.id_for_layout:
                // do what you want when user click layout
                break;

            case R.id.id_for_first_view:
                // do what you want when user click first view
                break;

            case R.id.id_for_second_view:
                // do what you want when user click second view
                break;
          }
}


回答3:

Remember that the method OnClick have one parameter that indicates which View is clicked!

public void onClick(View v) {
      switch(v.getId()) {
        case R.id.btn1:
          // first button
          break;
        case R.id.btn2:
          // second button
          break;
      }
  }


回答4:

Assign the id to each of your views by android:id="@+id/v1" then in

onClick(View v)

check for the view like

if(v == findViewById(R.id.v1))
{
 // v1 specific action
}
else if(v == findViewById(R.id.v2))
    {
     // v2 specific action
    }


回答5:

Ok, it's easy.
You have to add an OnClickListener to each view (button, textView and so on), and to the whole layout as well.
In this case, the onClickListeners that will be launched if you click are the view's onClickListener and the layout's onClickListener.
So no need for a whole bunch of classes.
One will be enough if you try this (do actions depending on the id and don't forget the layout ;) ):

public void onClick(View v) {
  switch(v.getId()) {
    case R.id.layout:
      layout actions here
      break;
    case R.id.textview:
      textview actions here
      break;

this onClickListener is for all of your views ;)



回答6:

There is View v parameter in onClickListner(View v) method : 


@Override
protected void onCreate(Bundle savedInstanceState) {
    LinearLayout layout=(LinearLayout)findViewById(R.id.linearLayout);
    layout.setOnclickListener(this);

}

@Override
public void onClick(View v) {
    if (v.getId() == R.id.linearLayout){
        //your logic
    } 
}


回答7:

for anyone interested you can also log the currently clicked view:

 getActivity()
      .getWindow()
      .getDecorView()
      .findViewById(android.R.id.content)
      .setOnTouchListener((v, event) -> {

                if (event.getAction() == MotionEvent.ACTION_UP) {
                    Log.v("[onClick]", v.getClass().getCanonicalName());
                }

                return false;
            });