Android 小部件在模拟器中运行良好,但在手机上它变成了 Google App 小部件

Posted

技术标签:

【中文标题】Android 小部件在模拟器中运行良好,但在手机上它变成了 Google App 小部件【英文标题】:Android widget works well in emulator but on phone it turns into the Google App widget 【发布时间】:2015-10-14 02:51:55 【问题描述】:

我创建了一个 android 小部件。 在运行 Android 4.4.4 的 Genymotion Nexus 4 模拟器中,一切正常。 在运行 Android 4.4.4 的 Nexus 4 设备上,我将小部件放在主屏幕上,它变成了 Google 应用小部件。 然后它变成了我的小部件,然后又变成了 Google App 小部件,依此类推。在我从主屏幕中删除小部件之前,它会执行此操作。 此外,我使用 parse.com 作为在线存储,在我的手机上似乎没有获得数据。我无法确定,因为小部件一直在变化。 我的小部件包含 3 个文件。 扩展应用程序类的一种:

public class MyApp extends Application

    private MyModel model;

     @Override
    public void onCreate() 
        Parse.enableLocalDatastore(this);
        ParseObject.registerSubclass(GdbdData.class);
        Parse.initialize(this, "-", "-");

        if(model == null) 
            Intent intent = new Intent(this,GdbdWidgetProvider.class);
            intent.setAction(WidgetUtils.WIDGET_REFRESH_STORAGE);
            intent.putExtra("userId",123);
            sendBroadcast(intent);
        

        super.onCreate();
    

一个 WidgetProvider:

@Override
public void onReceive(Context context, Intent intent) 
    super.onReceive(context, intent);
    final String action = intent.getAction();
    if (action.equals(WidgetUtils.WIDGET_REFRESH_STORAGE))
    
        MyApp app = ((MyApp)context.getApplicationContext());
        int userId = intent.getIntExtra("userId",0);
        TryGetModelFromRemoteStorage(userId,context);
    




static void updateAppWidget(Context context, AppWidgetManager appWidgetManager, int appWidgetId)

    MyApp app = ((MyApp)context.getApplicationContext());
    MyModel model = app.getModel();

    // Construct the RemoteViews object
    RemoteViews remoteViews = new RemoteViews(context.getPackageName(), R.layout.the_widget);

    PendingIntent clickerPendingIntent = buildButtonPendingIntent(context);

    // I change some textviews and imageviews inside the widget here

    appWidgetManager.updateAppWidget(appWidgetId, remoteViews);


public static PendingIntent buildButtonPendingIntent(Context context) 

    // initiate widget update request
    Intent intent = new Intent();
    intent.setAction(WidgetUtils.WIDGET_UPDATE_ACTION);
    PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0, intent, 0);
    return pendingIntent;

还有一个处理小部件上的按钮的广播接收器:

@Override
public void onReceive(Context context, Intent intent) 
    if (intent.getAction().equals(WidgetUtils.WIDGET_UPDATE_ACTION)) 
        updateWidget(context);
    


private void updateWidget(Context context) 


    MyApp app = ((MyApp)context.getApplicationContext());
    MyModel model = app.getModel();

    //I update some fields in the model based on some business rules at this point

    MyWidgetProvider.SendUpdateMessageToWidgets(context);



有谁知道我做错了什么?

编辑 1:按要求添加清单文件

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.my.app.main" >

    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

    <application
        android:name="MyApp"
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <receiver android:name=".GdbdWidgetProvider" >
            <intent-filter>
                <action android:name="android.appwidget.action.APPWIDGET_ENABLED" />
                <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />

            </intent-filter>

            <meta-data
                android:name="android.appwidget.provider"
                android:resource="@xml/the_widget_info" />
        </receiver>
        <receiver
            android:name=".TapMarkDayIntentReceiver"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="com.my.app.intents.UPDATE_WIDGET" />
            </intent-filter>

            <meta-data
                android:name="android.appwidget.provider"
                android:resource="@xml/the_widget_info" />
        </receiver>
    </application>

</manifest>

编辑2,添加widget xml文件:

  <?xml version="1.0" encoding="utf-8"?>
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
    android:minWidth="140dp" android:minHeight="140dp"
    android:updatePeriodMillis="1000000"
    android:previewImage="@drawable/example_appwidget_preview"
    android:initialLayout="@layout/the_widget" android:widgetCategory="home_screen"
    android:initialKeyguardLayout="@layout/the_widget"></appwidget-provider>

【问题讨论】:

@DavidWasser 现在添加了它 也请发帖the_widget_info.xml @DavidWasser 也添加了一个 请在模拟器上运行但在您的设备上无法运行的截图发布,谢谢(只是好奇症状是什么样的) 你有什么异常吗?特别是在您的更新/布局方法中? 【参考方案1】:

很抱歉没有回答最后的 cmets,但我真的很忙。 感谢大家的帮助。 发生的事情是我在 WidgetProvider 的 onUpdate、onEnabled、onDisabled 和 orReceive 方法中有返回语句,显然这是一件坏事。一旦我删除它们,这不再发生。

【讨论】:

以上是关于Android 小部件在模拟器中运行良好,但在手机上它变成了 Google App 小部件的主要内容,如果未能解决你的问题,请参考以下文章

Nativescript SideDrawer 无法在 iOS 模拟器中运行,但在 android 中运行良好

使用 Android 小部件

React-Native 应用程序在 android 设备上立即崩溃,但在模拟器上运行良好

在flutter中从image_picker包中打开相机会导致真实设备上的应用程序崩溃,但在模拟器(android)中运行良好

应用程序在模拟器上运行良好,但在真实设备上崩溃

如何在 Android 4.2 模拟器中添加小部件?