Android Widget onClick无法正常工作 - 服务无法启动

时间:2015-04-08 16:00:08

标签: android android-intent android-widget android-service android-broadcast

更新:解决了问题,解决方案

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

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并在此之后停止。

不幸的是,它没有达到这个目的。我的提供商看起来像这样:

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);
    }
}
}

我在这里经历了几个已回答的问题,更改了内容,添加了内容,到目前为止还没有真正有效,我仍然不知道为什么。

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

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
          android:layout_width="wrap_content"
          android:layout_height="wrap_content"
          android:id="@+id/layout"
          android:clickable="true">
<ImageView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    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。如果没有处理,小部件似乎不会将点击移动到下面的视图。

2 个答案:

答案 0 :(得分:4)

问题:

要以这种方式开始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步:

在清单中,将操作添加到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步:

在小部件的PendingIntent上设置RemoteViews

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

第5步:

覆盖onReceive()类的SmartWifiWidgetProvider方法:

@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

答案 1 :(得分:0)

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

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

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