AppWidgetHostView无法正确处理onClick / onLongClick

时间:2014-06-06 04:49:30

标签: android android-appwidget

我正在开发一个启动器,现在我正在使用app小部件,我从这里按照教程: AppWigetHost tutrial - Leonardo Fischer 一切顺利,直到我尝试在AppWidgetHostView

上添加onLongCick监听器
private void createAppWidget(Intent data) {
    Bundle extras = data.getExtras();
    final int appWidgetId = extras.getInt(AppWidgetManager.EXTRA_APPWIDGET_ID);
    AppWidgetProviderInfo appWidgetInfo = mAppWidgetManager.getAppWidgetInfo(appWidgetId);
    AppWidgetHostView hostView = mAppWidgetHost.createView(getApplicationContext(), appWidgetId, appWidgetInfo);
    hostView.setAppWidget(appWidgetId, appWidgetInfo);
    LinearLayout.LayoutParams params = createLayoutParams(appWidgetInfo); 
    rootLayout.addView(hostView, params);
    hostView.setOnLongClickListener(new OnLongClickListener() {
        @Override
        public boolean onLongClick(View v) {
            Toast.makeText(getApplicationContext(), "Long click " + appWidgetId, Toast.LENGTH_LONG).show();
            return true;
        }
    });
}

当我在主机上添加一个小部件(默认的谷歌模拟时钟) - 并试图长时间点击它,它只是不起作用!长按后,它只会打开时钟应用程序(单击默认操作)。

谁能告诉我如何修复它?这是我的完整代码: http://pastebin.com/61TkuLvx

添加的 我刚刚检查过:
  ----如果我长时间点击Clock小部件的边界,则会触发longClick事件!
  ----我添加了我的自定义appwidget(没有onclick处理程序):所以longclick事件总是被正确触发。
所以我想,它必须与touchevent处理程序/调度程序有关。

我刚刚尝试在主机视图的所有子项上设置onLongClick,如下所示:

private void createAppWidget(Intent data) {
...
... 
    setChildrenViewLongClick(hostView, new OnLongClickListener() {
        @Override
        public boolean onLongClick(View v) {
            Toast.makeText(getApplicationContext(), "Long click " + appWidgetId, Toast.LENGTH_LONG).show();
            return true;
        }
    });
}

protected void setChildrenViewLongClick(View view, OnLongClickListener listener) {
    view.setOnLongClickListener(listener);
    String name = view.getClass().getName();
    Logger.logInfo("Classname: " + name);
    if ( view instanceof ViewGroup ) {
        ViewGroup vg = (ViewGroup) view;
        for(int i = 0 ; i < vg.getChildCount() ; i++ ) {
            setViewLongClick(vg.getChildAt(i), listener);
        }
    }
}

它不起作用:(

奇怪的是:我在另一个活动上做了另一个实验,它有一个Linearlayout LL(有onLongClick处理程序)包含2个按钮bt1,bt2 - (每个按钮都有onClick处理程序)并应用方法setChildrenViewLongClick(LL,onLongClickListener) - 然后长按bt2 - Woop,它工作,调用onLongClickListener里面的动作!
那么问题出在哪? AppWidgetHostView或事件/处理程序调度程序?或者我(我的代码)

我很困惑,请帮助我!

韩国社交协会

3 个答案:

答案 0 :(得分:3)

几天后没有得到任何回答,我试着阅读Trebuchet-launcher的源代码 结果很简单:扩展AppWidgetHostView并重写方法onInterceptTouchEvent(),如this source code - 我还没有尝试过,但我想它会起作用:)。

希望这有助于像我这样的人:)

答案 1 :(得分:3)

Hungson175的答案很棒,但它并没有让我一路走来。由于我使用AppWidgetHost创建AppWidgetHostView,我需要扩展AppWidgetHost和AppWidgetHostView。幸运的是,这很简单,并且不需要过多地覆盖默认的android方法。

WidgetHost

public class WidgetHost extends AppWidgetHost {

    public WidgetHost(Context context, int hostId) {
        super(context, hostId);
    }

    @Override
    protected AppWidgetHostView onCreateView(Context context, int appWidgetId, AppWidgetProviderInfo appWidget) {
        // pass back our custom AppWidgetHostView
        return new WidgetView(context);
    }
}

WidgetView

public class WidgetView extends AppWidgetHostView {

    private OnLongClickListener longClick;
    private long down;

    public WidgetView(Context context) {
        super(context);
    }

    public WidgetView(Context context, int animationIn, int animationOut) {
        super(context, animationIn, animationOut);
    }

    @Override
    public void setOnLongClickListener(OnLongClickListener l) {
        this.longClick = l;
    }

    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        switch(MotionEventCompat.getActionMasked( ev ) ) {
            case MotionEvent.ACTION_DOWN:
                down = System.currentTimeMillis();
                break;
            case MotionEvent.ACTION_MOVE:
                boolean upVal = System.currentTimeMillis() - down > 300L;
                if( upVal ) {
                    longClick.onLongClick( WidgetView.this );
                }
                break;
        }

        return false;
    }
}

希望它对某人有所帮助,因为处理AppWidget很难。

答案 2 :(得分:1)

我遇到了同样的问题,我通过覆盖方法

解决了这个问题
ViewGroup#onInterceptTouchEvent(MotionEvent ev).

有关详细信息,请参阅"Managing Touch Events in a ViewGroup"