Android Widget onClick 不起作用 - 服务无法启动

Posted

技术标签:

【中文标题】Android Widget onClick 不起作用 - 服务无法启动【英文标题】:Android Widget onClick not working - Service won't start 【发布时间】:2015-06-13 16:55:31 【问题描述】:

更新:已解决,解决方法如下

我正在尝试编写一个启动服务的小部件,然后它将执行一些尚未实现的东西。到目前为止,我的服务只是:

public class SmartWifiService extends Service 

private static final String WIDGET_CLICK = "de.regenhardt.smartwifiwidget.WIDGET_CLICK";

@Override
public int onStartCommand(Intent intent, int flags, int startId) 
    super.onStartCommand(intent, flags, startId);
    Log.e("DEBUG", "Service started");
    Toast.makeText(getApplicationContext(), "Widget clicked", Toast.LENGTH_SHORT).show();
    stopSelf();
    return START_NOT_STICKY;


@Override
public IBinder onBind(Intent intent) 
    return null;


所以它所做的一切都是发送一个 Toast 并在此之后自行停止。

不幸的是,事实并非如此。我的 Provider 看起来像这样:

public class SmartWifiWidgetProvider extends AppWidgetProvider 
@Override
public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) 
    Log.e("DEBUG", "onUpdate called");
    super.onUpdate(context, appWidgetManager, appWidgetIds);
    Intent clickIntent = new Intent(context, SmartWifiWidgetProvider.class);
    clickIntent.setAction("de.regenhardt.smartwifiwidget.WIDGET_CLICK");
    PendingIntent pendingIntent = PendingIntent.getBroadcast(context.getApplicationContext(),
            0, clickIntent, 0);
    RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.widget_layout);
    views.setOnClickPendingIntent(R.id.layout, pendingIntent);
    //for (int ID:appWidgetIds) 
    //    views.setOnClickPendingIntent(ID, pendingIntent);
        appWidgetManager.updateAppWidget(appWidgetIds, views);
    //
    @Override
public void onReceive(Context context, Intent intent) 
    Log.e("DEBUG", "received");
    super.onReceive(context, intent);
    if(intent.getAction().equals("de.regenhardt.smartwifiwidget.WIDGET_CLICK"))
        Log.e("DEBUG", "Click action fits");
        Intent i = new Intent(context.getApplicationContext(), SmartWifiService.class);
        context.startService(i);
    


我在这里回答了几个问题,更改了内容,添加了内容,但到目前为止没有任何效果,我仍然不知道为什么。

当我点击小部件时没有动画,但我很确定我的小部件本身是可点击的:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
          android:layout_
          android:layout_
          android:id="@+id/layout"
          android:clickable="true">
<ImageView
    android:layout_
    android:layout_
    android:src="@drawable/wifi"
    android:clickable="true"
    android:id="+id/widgetImage"/>
</LinearLayout>

用ImageButton也试过了,效果没有变化。

我希望你们能帮助我,我已经坐在这个小东西上好几天了,我的头在旋转(不是字面意思)。

您好,

马龙

编辑:这是我的清单:

<application android:allowBackup="true"
             android:label="@string/app_name"
             android:icon="@drawable/wifi"
             android:theme="@style/AppTheme">
    <receiver android:name=".SmartWifiWidgetProvider">
        <intent-filter>
            <action android:name="android.appwidget.action.APPWIDGET_UPDATE"/>
            <action android:name="de.regenhardt.smartwifiwidget.WIDGET_CLICK"/>
        </intent-filter>
        <meta-data android:name="android.appwidget.provider"
                   android:resource="@xml/smart_wifi_widget_info"/>
    </receiver>
    <service android:name="de.regenhardt.smartwifiwidget.SmartWifiService"/>

</application>

编辑 2:更新;更新到我的代码的当前状态,适应 Y.S.'回答。

LogCat 添加小部件后,点击它仍然没有做任何事情:

04-08 20:12:30.985  14867-14867/de.regenhardt.smartwifiwidget E/DEBUG﹕ received
04-08 20:12:30.998  14867-14867/de.regenhardt.smartwifiwidget E/DEBUG﹕ received
04-08 20:12:30.998  14867-14867/de.regenhardt.smartwifiwidget E/DEBUG﹕ onUpdate called
04-08 20:12:31.155  14867-14867/de.regenhardt.smartwifiwidget E/DEBUG﹕ received

解决方案:

吹线是views.setOnClickPendingIntent(R.id.widgetImage, pendingIntent);,我在那里有 R.id.layout 而不是 widgetImage。如果没有得到处理,小部件似乎不会将点击传递到下面的视图。

【问题讨论】:

你看到我的回答了吗? 是的,我有,正在吃东西,然后需要一些时间来尝试一下;-) 【参考方案1】:

问题:

要以这种方式启动Service,您需要使用PendingIntent.getBroadcast(),而不是PendingIntent.getService()。并且WIDGET_CLICK 操作需要在应用清单中的receiver 标签下指定,而不是service 标签。

第 1 步:

替换

Intent clickIntent = new Intent("de.regenhardt.smartwifiwidget.WIDGET_CLICK");

Intent clickIntent = new Intent(context, SmartWifiWidgetProvider.class);
clickIntent.setAction("de.regenhardt.smartwifiwidget.WIDGET_CLICK");

第 2 步:

替换

PendingIntent pendingIntent = PendingIntent.getService(context.getApplicationContext(),
        0, clickIntent, PendingIntent.FLAG_UPDATE_CURRENT);

PendingIntent pendingIntent = PendingIntent.getBroadcast(context.getApplicationContext(),
        0, clickIntent, 0);

第 3 步:

在清单中,将 action 添加到 receiver 标记并从 service 标记中删除:

<application android:allowBackup="true"
    android:label="@string/app_name"
    android:icon="@drawable/wifi"
    android:theme="@style/AppTheme">

    <receiver android:name=".SmartWifiWidgetProvider">
        <intent-filter>
            <action android:name="android.appwidget.action.APPWIDGET_UPDATE"/>
            <action android:name="de.regenhardt.smartwifiwidget.WIDGET_CLICK"/>
        </intent-filter>
        <meta-data android:name="android.appwidget.provider"
                   android:resource="@xml/smart_wifi_widget_info"/>
    </receiver>

    <service android:name="de.regenhardt.smartwifiwidget.SmartWifiService"></service>

</application>

第 4 步:

在小部件的RemoteViews 上设置PendingIntent

RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.widget_layout);
views.setOnClickPendingIntent(R.id.widgetImage, pendingIntent);

第 5 步:

覆盖SmartWifiWidgetProvider类的onReceive()方法:

@Override
public void onReceive(Context context, Intent intent) 
    super.onReceive(context, intent);
    if (intent.getAction().equals("de.regenhardt.smartwifiwidget.WIDGET_CLICK")) 
        Intent i = new Intent(context.getApplicationContext(), SmartWifiService.class)
        startService(i);
    

试试这个。这应该会正确启动Service

【讨论】:

感谢您的详细说明,希望我能得到这样的东西 :) 不幸的是,到目前为止它还不能正常工作,但我还必须进行一些更改:R.id.layout 是我的 LinearLayout 的 ID围绕ImageView,我希望我扣对了。 startService 不会被识别,所以我将其设为 context.startService。最后我得到了appWidgetManager.updateAppWidget(appWidgetIds, views);,而不是for循环 我们在某个地方犯了一个或两个小错误......继续寻找它,让我知道。这不是火箭科学,只是某个地方的一个简单的逻辑错误...... :) 我到处都放了日志命令,也许它有助于看到一些东西,更新了我的问题^^ 是的,我注意到您昨天更新了问题...嗯,我无法猜测为什么它仍然无法正常工作:/ 我知道了 确实是一个小词:setOnClickPendingIntent我必须把它放在 R.id.widgetImage 上,而不是 R.id.layout 上。似乎水龙头没有穿过 LinearLayout o.O 内的 ImageView【参考方案2】:

您应该将您的服务添加到清单中:

<service android:name="SmartWifiService"></service>

你没有说你是如何调用服务的,所以我可以说需要从某个 Activity 调用 startService(Intent) - 没有它就不会启动。

【讨论】:

添加了我的清单,服务已经在那里。我以为views.setOnClickPendingIntent(ID, pendingIntent); 会让我的小部件在我每次点击时启动服务? 我没有在你的代码中看到一些启动服务的意图。它应该看起来像 new Intent(getApplicationContext(), SmartWifiService.class)。 在阅读和尝试其他答案之前,我有过这个问题,但它也不起作用。再次尝试确定,将其添加到我的问题中。

以上是关于Android Widget onClick 不起作用 - 服务无法启动的主要内容,如果未能解决你的问题,请参考以下文章

xml onClick 中的 Kotlin 不起作用

Android工作室:按钮onClick()不起作用

小部件onClick中的Android ListView不起作用

以编程方式更改Chip Widget样式不起作用 - Android

Android数据绑定:无法找到属性android:onClick的setter

也许还需要onClick之外的东西