时间:2011-01-06 19:28:18

标签: android event-bubbling

8 个答案:

答案 0 :(得分:5)

这有点脏,但是如果你需要多个监听器我会这样做的方法是注册一个知道另一个的人。第一个(实际注册的那个)将需要知道何时根据事件的条件委托给另一个监听器。实际上,实际上,没有必要有两个OnClickListener类。第二个类可以实现您想要的任何接口。此外,您无需为所需内容创建特殊界面。

public class MyClickListener implements OnClickListener{
  private SomeCustomClass mSecondListener = new SomeCustomClass();
  public void onClick(View v){
    if (needToForward){
      mSecondListener.handleClick(v);
    }else{
      //handle the click
    }
  }
}

然后,在您的活动代码中,您将执行此操作

MyClickListener lstn = new MyClickListener();
mCheckBox.setOnClickListener(lstn);

这有什么理由不适合你吗?

或者,如果您愿意,第二个类也可以实现OnClickListener接口。

此外,如果您需要真正的冒泡,您可以定义自己的接口,该接口支持将多个单击侦听器添加到恰好实现OnClickListener接口的中间类。从那里,在该类的onClick()方法中,您将遍历已注册的侦听器,并调用适当的方法。

答案 1 :(得分:3)

更简洁的方法是使用CompositeListener模式。

取自: how can I set up multiple listeners for one event?

您必须在项目中添加此类:

/**
 * Aux class to collect multiple click listeners.
 */
class CompositeListener implements OnClickListener {
    private List<OnClickListener> registeredListeners = new ArrayList<OnClickListener>();

    public void registerListener (OnClickListener listener) {
        registeredListeners.add(listener);
    }

    @Override
    public void onClick(View v) {

        for(OnClickListener listener:registeredListeners) {
            listener.onClick(View v);
        }
    }
}

然后在MyCheckButton

上添加此内容
private CompositeListener clickListener = new CompositeListener();

public MyCheckButton()
{
    super.setOnClickListener(clickListener); //multi event listener initialization
}

@Override
public void setOnClickListener(OnClickListener l) {
    clickListener.registerListener(l);
}

setOnClickListener的调用都将通过此覆盖,添加到列表中并在触发事件时调用。希望它有所帮助。

答案 2 :(得分:1)

因为看起来每个View只能有一个onClickListener。我认为我必须做的是定义一个接口:

public interface MyOnClickListener {
    public void onMyClick(View v);
}

从我的活动中实现它并覆盖onMyClick函数以执行我想要的任何操作,在MyCheckButton类中,我需要在构造函数中传递MyOnClickListener并保存它并在onClick处理程序内调用listener.onMyClick。

如果有更好的方法,请告诉我。我考虑在activity或MyCheckButton类中使用onTouch处理程序,但稍后如果我将onTouch或onClick添加到任何一个,我将会收到一个很难注意到的错误。

我的想法不起作用,因为我不知道如何从我的构造函数中获取对该活动的引用:

public class TVCheckButton extends TextView {

    private MyOnClickListener mListener;

    public TVCheckButton(Context context, AttributeSet attrs) {
        super(context, attrs);

        mListener = ???;
    }
}

答案 3 :(得分:1)

由于只有一个OnclickListener在Android 2.1上工作[我不知道更高版本]使视图变为私有和静态,并创建一个可以改变它的静态函数,例如。

公共类ExampleActivity扩展了Activity {

private SomeOtherclass someOtherClass;
private static Button b_replay;

 public void onCreate(Bundle savedInstanceState){
     someOtherClass = new SomeOtherclass();

   b_replay = (Button) findViewById(R.id.b_replay);
   b_replay.setOnClickListener(someOtherClass);
 }

 public static void changeReplayText(String text){
   b_replay.setText(text);
 }

}

答案 4 :(得分:0)

一个很好的通用方法是使用一个监听器列表,例如ListenerList中的WeakListenerListBeryl library

答案 5 :(得分:0)

出于某种原因,我无法使用上面的答案,所以这里有一个替代方案: //我在一个方法中拥有所有这一切,其中我有两个按钮,并且基于外部因素,一个将是可见的,而另一个不会被称为“消失”。所以,我已经检查了!希望它可以帮助别人!!

Button b = (Button) findViewById(R.id.b_reset);
Button breakk = (Button) findViewById(R.id.b_break);

    if ((findViewById(R.id.b_reset)).getVisibility() == View.VISIBLE) {
        b.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {

//一些代码和方法......

            }
        });
    } else if ((findViewById(R.id.b_break)).getVisibility() == View.VISIBLE) {

        breakk.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {

//一些代码和方法......

            }
        });
    }

答案 6 :(得分:0)

在第一节课中,定义一个虚拟按钮:

private static Button myVirtualButton = new Button(context);

...以及注册它的公共方法:

public static void registerMyVirtualButton(Button b) { myVirtualButton = b;}

OnClickListener做任何需要的操作,最后点击虚拟按钮:

if (myVirtualButton!=null) { myVirtualButton.callOnClick(); }

在第二节课中,定义一个按钮及其OnClickListener,其中包含额外需要的任何动作。

通过registerMyVirtualButton将按钮传送到第一堂课。 单击第一个类的对象后,将执行这两个操作。

答案 7 :(得分:-2)