34

我已经开始开发一个应用程序。我昨天构建了菜单,但是 onClick 方法不起作用!我创建了一个扩展 View 的类并将其命名为 MainMenuObject - 该类适用于主菜单中的任何对象(按钮、徽标等)。我为他们创建了一个特殊的类,因为我在菜单启动时制作动画。在我构建了 MainMenuObject 类之后,我构建了另一个类 (OpeningTimesView),它扩展了 View,其中包含主菜单的所有按钮,并将用作主要活动的布局。

一切都很好,动画效果很好,我想在我的按钮上放置监听器,所以我在 OpeningTimesView 类中添加了 onClickListener 的实现,并覆盖了 onClick 方法。然后我使用 setOnClickListener(this) 和 setClickable(true) 将侦听器添加到按钮中,但它不起作用!我什么都试过了!请帮我弄清楚我做错了什么。我在 onClick 方法中添加了一个 toast,它不依赖于任何“if”,但它也不会显示。

(顺便说一句,有没有办法将屏幕宽度和高度定义为所有类都可以访问的变量?它不能是静态的,因为您从显示对象获取高度和宽度,但必须有另一种方式)

这是代码:

public class OpeningTimesView extends View implements OnClickListener{
    private MainMenuObjectView searchButton;
    private MainMenuObjectView supportButton;
    private MainMenuObjectView aboutButton;
    private int screenWidth;
    private int screenHeight;
    public OpeningTimesView(Context context, Display dis) {
        super(context);

        this.screenWidth = dis.getWidth();
        this.screenHeight = dis.getHeight();

        searchButton = new MainMenuObjectView(context, 200, MovingMode.RIGHT, R.drawable.search, dis);
        supportButton = new MainMenuObjectView(context, 400, MovingMode.LEFT, R.drawable.support, dis);
        aboutButton = new MainMenuObjectView(context, 600, MovingMode.RIGHT, R.drawable.about, dis);

        searchButton.setClickable(true);
        supportButton.setClickable(true);
        aboutButton.setClickable(true);

        searchButton.setOnClickListener(this);
        supportButton.setOnClickListener(this);
        aboutButton.setOnClickListener(this);
    }

    @Override
    public void onClick(View view){
        Toast.makeText(getContext(), "Search button pressed", Toast.LENGTH_SHORT).show();
        if(view == searchButton){
            Toast.makeText(getContext(), "Search button pressed", Toast.LENGTH_SHORT).show();
        }
        else if(view == supportButton){
            Toast.makeText(getContext(), "Support button pressed", Toast.LENGTH_SHORT).show();
        }
        else Toast.makeText(getContext(), "About button pressed", Toast.LENGTH_SHORT).show();
    }
    @Override
    public void onDraw(Canvas canvas)
    {
        // Drawing the buttons
        this.searchButton.onDraw(canvas);
        this.aboutButton.onDraw(canvas);
        this.supportButton.onDraw(canvas);
    }

提前感谢,埃拉德!

4

11 回答 11

23

我刚刚遇到了同样的问题 - 我创建了一个自定义视图,当我通过调用监听器在活动中为它注册一个新的监听器时,v.setOnClickListener(new OnClickListener() {...});并没有被调用。

在我的自定义视图中,我还覆盖了该public boolean onTouchEvent(MotionEvent event) {...}方法。问题是我没有调用 View 类的方法 - super.onTouchEvent(event)。这解决了问题。因此,如果您想知道为什么没有调用您的侦听器,您可能忘记调用超类的onTouchEvent方法

这是一个简单的例子:

private static class CustomView extends View implements View.OnClickListener {
    public CustomView(Context context) {
        super(context);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        super.onTouchEvent(event);   // this super call is important !!!
        // YOUR LOGIC HERE
        return true;
    }

    @Override
    public void onClick(View v) {
        // DO SOMETHING HERE
    }
}
于 2012-01-28T15:33:26.653 回答
17

如果您对 UI 框架的操作方式不满意,在 Android 中创建自定义控件可能会很棘手。如果您还没有,我建议您阅读以下内容:

http://developer.android.com/guide/topics/ui/declaring-layout.html

http://developer.android.com/guide/topics/ui/custom-components.html

http://developer.android.com/guide/topics/ui/layout-objects.html

请注意,当在 XML 中声明布局时,元素是嵌套的。这将创建一个布局层次结构,您必须在仅使用 Java 代码自定义组件时自行创建该层次结构。

您很可能陷入了 Android 的触摸层次结构。与其他一些流行的移动平台不同,Android 从 View 层次结构的顶部开始提供触摸事件,然后向下工作。传统上占据较高层次结构(活动和布局)的类在它们中具有转发它们自己不消耗的触摸的逻辑。

因此,我建议您将您的按钮更改OpeningTimesView为扩展ViewGroup(所有 Android 布局的超类)或特定布局(LinearLayoutRelativeLayout等)并将您的按钮添加为子级。现在,似乎没有定义的层次结构(按钮并不是真正“包含”在容器中,它们只是成员),这可能会混淆事件真正去向的问题。

  1. 触摸应该更自然地流向按钮,允许您的点击事件触发
  2. 您可以利用 Android 的布局机制来绘制视图,而不是依赖绘制代码来完成所有这些工作。

选择一个布局类开始,它将帮助您将按钮放置在它们的最终位置。您可以使用 Android 中的动画框架或自定义绘图代码(就像您现在拥有的那样)以任何您喜欢的方式为它们设置动画。如果需要,按钮的位置和当前绘制该按钮的位置允许有很大不同,这就是当前动画框架在 Android(3.0 之前)中的工作方式......但这是一个单独的问题。你也有AbsoluteLayout,它允许你在任何你喜欢的地方放置和替换对象......但要注意你的应用在所有 Android 设备上的外观(考虑到不同的屏幕尺寸)。

至于你关于显示信息的第二点。 最简单的方法可能只是Context.getResources().getDisplayMetrics()在需要的地方使用。 Activity继承自Context,所以他们可以直接调用这个方法。视图总是有一个Context你可以访问的getContext()。任何其他类,您都可以Context在构造中将其作为参数传递(这是 Android 中的常见模式,您会看到许多对象需要 a Context,主要是为了访问Resources)。

这是一个快速启动的框架示例。这只是将三个水平排列一次作为最终位置:

Public class OpeningTimesView extends LinearLayout implements OnClickListener {
    private MainMenuObjectView searchButton;
    private MainMenuObjectView supportButton;
    private MainMenuObjectView aboutButton;
    private int screenWidth;
    private int screenHeight;

    public OpeningTimesView(Context context) {
        this(context, null);
    }

    //Thus constructor gets used if you ever instantiate your component from XML
    public OpeningTimesView(Context context, AttributeSet attrs) {
        super(context, attrs);

        /* This is a better way to obtain your screen info
        DisplayMetrics display = context.getResources().getDisplayMetrics();
        screenWidth = display.widthPixels;
        screenHeight = display.heightPixels;
        */
        //This way works also, without needing to customize the constructor
        WindowManager wm = (WindowManager)context.getSystemService(Context.WINDOW_SERVICE);
        Display dis = wm.getDefaultDisplay();
        screenWidth = dis.getWidth();
        screenHeight = dis.getHeight();

        searchButton = new MainMenuObjectView(context, 200, MovingMode.RIGHT, R.drawable.search, dis);
        supportButton = new MainMenuObjectView(context, 400, MovingMode.LEFT, R.drawable.support, dis);
        aboutButton = new MainMenuObjectView(context, 600, MovingMode.RIGHT, R.drawable.about, dis);

        //Even if they don't extend button, if MainMenuObjectView is always clickable
        // this should probably be brought into that class's constructor
        searchButton.setClickable(true);
        supportButton.setClickable(true);
        aboutButton.setClickable(true);

        searchButton.setOnClickListener(this);
        supportButton.setOnClickListener(this);
        aboutButton.setOnClickListener(this);

        //Add the buttons to the layout (the buttons are now children of the container)
        setOrientation(LinearLayout.HORIZONTAL);
        LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
        addView(searchButton, params);
        addView(supportButton, params);
        addView(aboutButton, params);       
    }

    @Override
    public void onClick(View view){
        Toast.makeText(getContext(), "Search button pressed", Toast.LENGTH_SHORT).show();
        if(view == searchButton){
            Toast.makeText(getContext(), "Search button pressed", Toast.LENGTH_SHORT).show();
        }
        else if(view == supportButton){
            Toast.makeText(getContext(), "Support button pressed", Toast.LENGTH_SHORT).show();
        }
        else Toast.makeText(getContext(), "About button pressed", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onDraw(Canvas canvas)
    {
        //Drawing the buttons
        // This may only be necessary until they are in place, then just call super.onDraw(canvas)
        this.searchButton.onDraw(canvas);
        this.aboutButton.onDraw(canvas);
        this.supportButton.onDraw(canvas);
    }    
}

您可以从那里自定义它。可能会启动可见性设置为 View.INVISIBLE 的按钮,直到您使用绘图代码或自定义动画对象对它们进行动画处理,然后使它们在最终的静止位置可见。

然而,这里的关键是布局足够聪明,知道当它接收到一个触摸事件时,它应该将它转发给相应的孩子。您可以在没有此功能的情况下创建自定义视图,但您必须拦截容器上的所有触摸并进行数学运算以确定手动将事件转发到哪个子视图。如果你真的不能让任何布局管理器工作,这是你的追索权。

希望有帮助!

于 2011-04-18T13:43:24.860 回答
11

您可以在自定义视图的 onTouchEvent 中调用 performClick()。

在您的自定义视图中使用它:

    @Override
    public boolean onTouchEvent(final MotionEvent event) {
        if(event.getAction() == MotionEvent.ACTION_UP){
            return performClick();
        }
        return true;
    }
于 2013-10-02T23:46:16.230 回答
9

我这样做:

public class YourView extends LinearLayout implements OnClickListener {
    OnClickListener listener;

    //... constructors

    public void setOnClickListener(OnClickListener listener) {
        this.listener = listener;
    }

    @Override
    public void onClick(View v) {
        if (listener != null)
             listener.onClick(v);
    }
}
于 2016-09-01T11:53:58.053 回答
9

您必须调用setOnClickListener(this)构造函数并View.OnClickListener自行实现。

这样:

public class MyView extends View implements View.OnClickListener {

    public MyView(Context context) {
        super(context);
        setOnClickListener(this);
    }

    public MyView(Context context, AttributeSet attrs) {
        super(context, attrs);
        setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        Toast.makeText(getContext(), "On click.", Toast.LENGTH_SHORT).show();
    }
}
于 2017-02-23T22:10:03.400 回答
6

我有同样的问题。在我的例子中,我有一个 LinearLayout 作为我的自定义视图的根元素,并将clickableandfocusable设置为 true,并且自定义视图标签本身(在片段的布局中使用)也设置为clickableand focusable。事实证明,要让它工作,我唯一要做的就是从 XML 中删除所有clickablefocusable属性 :) 违反直觉,但它起作用了。

于 2018-11-21T16:12:38.763 回答
3

onClickListener在类中实现MainMenuObjectView,因为这些是响应点击的对象。
另一种选择是扩展Button而不是View,因为您在那里只使用按钮


更新:完整示例


这是将其直接实现到可点击视图中的想法。有一个TestView类可以根据需要扩展View和覆盖onDraw,并且还可以响应点击。ClickListener我省略了任何动画实现,因为你有那部分,它与讨论无关。
我在 Eclair 模拟器中对其进行了测试,它按预期工作(单击后显示 Toast 消息)。

文件:Test.java

package com.aleadam.test;

import android.app.Activity;
import android.os.Bundle;
import android.widget.LinearLayout;
import android.widget.TextView;

public class Test extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        LinearLayout ll = new LinearLayout(this);
        ll.setOrientation(LinearLayout.VERTICAL);
        TextView label = new TextView(this);
        LinearLayout.LayoutParams layoutParams = new LinearLayout.LayoutParams(
             LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT);
        label.setText("Click the circle!");
        TestView testView = new TestView(this);
        ll.addView(label, layoutParams);
        ll.addView(testView, layoutParams);
        setContentView(ll);
    }
}

文件:TestView.java

package com.aleadam.test;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Toast;

public class TestView extends View implements OnClickListener {
    Context context;

    public TestView(Context context) {
        super(context);
        this.context = context;
        setOnClickListener(this);
    }

    public void onClick(View arg0) {
        Toast.makeText(context, "View clicked.", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onDraw (Canvas canvas) {
        super.onDraw(canvas);
        this.setBackgroundColor(Color.LTGRAY);
        Paint paint = new Paint (Paint.ANTI_ALIAS_FLAG);
        paint.setColor(Color.RED);
        canvas.drawCircle(20, 20, 20, paint);
    }
}

如果您需要一些可点击和一些不可点击,您可以添加一个带有布尔参数的构造函数来确定 ClickListener 是否附加到视图:

public TestView(Context context, boolean clickable) {
    super(context);
    this.context = context;
    if (clickable)
        setOnClickListener(this);
}
于 2011-04-18T13:44:23.520 回答
3

我有办法!这并不是针对这个特定问题的真正解决方案,而是一种全新的方法。我将此线程发送给我认识的人,他告诉我使用 android 具有的动画 SDK(如提到的无线设计),所以我没有用 4 个类来做主菜单页面,我只用一个扩展的类来做Activity 和 Animation 类提供了许多动画选项。我要感谢你们俩对我的帮助,你们很棒。如果有人会遇到同样的问题或其他问题,我将添加代码:

package elad.openapp;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.HapticFeedbackConstants;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.Window;
import android.view.animation.Animation;
import android.view.animation.AnimationSet;
import android.view.animation.ScaleAnimation;
import android.view.animation.TranslateAnimation;
import android.widget.ImageView;
import android.widget.Toast;

public class OpeningTimes extends Activity implements OnClickListener{
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    // Disabling the title bar..
    requestWindowFeature(Window.FEATURE_NO_TITLE);
    setContentView(R.layout.main);

    // Create the buttons and title objects
    ImageView title = (ImageView)findViewById(R.id.title_main);
    ImageView search = (ImageView)findViewById(R.id.search_button_main);
    ImageView support = (ImageView)findViewById(R.id.support_button_main);
    ImageView about = (ImageView)findViewById(R.id.about_button_main);

    // Setting the onClick listeners
    search.setOnClickListener(this);
    support.setOnClickListener(this);
    about.setOnClickListener(this);

    setButtonsAnimation(title, search, support, about);

}

@Override
public void onClick(View v) {
    if(v.getId()==R.id.search_button_main){
        v.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS);
        startActivity(new Intent(this,SearchPage.class));
    }
    else if(v.getId()==R.id.support_button_main){
        v.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS);
        Toast.makeText(this, "Coming soon...", Toast.LENGTH_LONG).show();
    }
    else if(v.getId()==R.id.about_button_main){

        v.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS);
        Toast.makeText(this, "Coming soon...", Toast.LENGTH_LONG).show();
    }
}

// Setting the animation on the buttons
public void setButtonsAnimation(ImageView title, ImageView search, ImageView support, ImageView about){
    // Title animation (two animations - scale and translate)
    AnimationSet animSet = new AnimationSet(true);

    Animation anim = new ScaleAnimation(0.0f, 1.0f, 0.0f, 1.0f);
    anim.setDuration(750);
    animSet.addAnimation(anim);

    anim = new TranslateAnimation(Animation.RELATIVE_TO_SELF, 0.0f, Animation.RELATIVE_TO_SELF, 0.0f,
            Animation.RELATIVE_TO_SELF, -1.0f, Animation.RELATIVE_TO_SELF, 0.0f);
    anim.setDuration(750);
    animSet.addAnimation(anim);

    title.startAnimation(animSet);

    // Search button animation
    anim = new TranslateAnimation(Animation.RELATIVE_TO_SELF, -1.5f, Animation.RELATIVE_TO_SELF, 0.0f,
            Animation.RELATIVE_TO_SELF, 0.0f, Animation.RELATIVE_TO_SELF, 0.0f);
    anim.setDuration(750);

    search.startAnimation(anim);

    // Support button animation
    anim = new TranslateAnimation(Animation.RELATIVE_TO_SELF, 1.5f, Animation.RELATIVE_TO_SELF, 0.0f,
            Animation.RELATIVE_TO_SELF, 0.0f, Animation.RELATIVE_TO_SELF, 0.0f);
    anim.setDuration(750);

    support.startAnimation(anim);

    // About button animation
    anim = new TranslateAnimation(Animation.RELATIVE_TO_SELF, 0.0f, Animation.RELATIVE_TO_SELF, 0.0f,
            Animation.RELATIVE_TO_SELF, 3f, Animation.RELATIVE_TO_SELF, 0.0f);
    anim.setDuration(750);

    about.startAnimation(anim);
}
}
于 2011-04-19T21:10:56.543 回答
3

在我的情况下,我在自定义视图中有一个 RelativeLayout 作为父级,使其工作的唯一方法是在 RelativeLayout 和自定义视图的构造函数中设置focusable和设置,在clickable扩展布局后,添加以下内容:true

View view = inflate(getContext(), R.layout.layout_my_custom_view, this);

view.findViewById(R.id.theparent).setOnClickListener(new OnClickListener() {
        @Override
        public void onClick(View view) {
            performClick();
        }
    });
于 2019-04-30T11:23:46.120 回答
1

这很容易:

public class FancyButton
 extends FrameLayout
 implements View.OnClickListener { ..

    void yourSetupFunction(Context context, @Nullable AttributeSet attrs) {
        ..
        super.setOnClickListener(this); // NOTE THE SUPER
    }

    OnClickListener consumerListener = null;
    @Override
    public void setOnClickListener(@Nullable OnClickListener l) {
        consumerListener = l;
        // DO NOT CALL SUPER HERE
    }

    @Override
    public void onClick(View v) {
        Log.i("dev","perform my custom functions, and then ...");
        if (consumerListener != null) { consumerListener.onClick(v); }
    }
  1. 我们实现了 View.OnClickListener,因此有一个
  2. onClick(查看 v)
  3. 在setOnClickListener中,记住从外界设置的监听器
  4. 将实际的侦听器设置为我们(使用“super.” ...)
  5. 在onClick你自定义的东西,然后调用外界的onClick
于 2021-03-21T13:58:26.273 回答
0

只需添加callOnClick()到您的onTouchEvent()方法 中

public boolean onTouchEvent(MotionEvent event) {
        .....YOURCODE.....
        callOnClick();
        return boolean;
    }
于 2018-12-26T03:27:00.687 回答