不定期更新小部件

Posted

技术标签:

【中文标题】不定期更新小部件【英文标题】:update Widget Non periodiccally 【发布时间】:2013-05-01 06:36:04 【问题描述】:

我有一个包含TextView 的小部件,我只想在调用Onreceive()此接收器不是为小部件实现的那个 方法时更新它。我不想定期更新小部件,但只有在我的 Onreceive() 被调用时。

Recver.java

public class Recver extends BroadcastReceiver 



    @Override
    public void onReceive(Context context, Intent intent) 

                Widgetd.updatewid(context);

            
    

    

Widgetd.java

public class Widgetd extends AppWidgetProvider 


    @Override
    public void onUpdate(Context context, AppWidgetManager appWidgetManager,
            int[] appWidgetIds) 
        // TODO Auto-generated method stub
        super.onUpdate(context, appWidgetManager, appWidgetIds);

    

    @Override
    public void onDeleted(Context context, int[] appWidgetIds) 
        // TODO Auto-generated method stub
        super.onDeleted(context, appWidgetIds);
    
    static void updatewid(Context context)

        RemoteViews rv = new RemoteViews(context.getPackageName(),
                R.layout.widgetlay);
        rv.setTextViewText(R.id.widgetUpdateTv, "1");



    


ma​​nifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="delete.detailduplicate"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="17" />
    <uses-permission android:name="android.permission.RECEIVE_SMS"/>
    <uses-permission android:name="android.permission.READ_SMS"/>

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="delete.detailduplicate.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity android:name="Newmain"></activity>
        <receiver android:name="Recver">
            <intent-filter android:priority="9999999">
        <action android:name="android.provider.Telephony.SMS_RECEIVED"></action>
    </intent-filter>
        </receiver>
        <receiver android:name="Widgetd">
            <intent-filter >
                <action 
                    android:name="android.appwidget.action.APPWIDGET_UPDATE" />
            </intent-filter>

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

</manifest>

widgetsetter.xml

<?xml version="1.0" encoding="utf-8"?>
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
    android:initialLayout="@layout/widgetlay"
    android:minHeight="300dp"
    android:minWidth="72dp"
    android:updatePeriodMillis="0" >

</appwidget-provider>

【问题讨论】:

【参考方案1】:

设置updatePeriodMillis = "0":

<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
android:initialLayout="@layout/widget_initial_layout"
android:minHeight="40dp"
android:minWidth="40dp"
android:updatePeriodMillis="0"
android:widgetCategory="home_screen" 
android:previewImage="@drawable/my_app_logo"/>

在 AndroidManifest 中添加应触发更新的意图(此处以 WIFI_STATE_CHANGED 为例):

        <intent-filter>
            <action android:name="android.net.wifi.WIFI_STATE_CHANGED" />
        </intent-filter>

并在应用小部件代码中调用您的 updateWidget 方法:

    @Override
    public void onReceive(Context context, Intent intent)
    
            Log.d(TAG, "onReceive() " + intent.getAction());
            super.onReceive(context, intent);

            if (intent != null)
            
                    if (WifiManager.WIFI_STATE_CHANGED_ACTION.equals(intent.getAction()))
                    
                            updateWidgets(context);
                    
            ...
            
     

    @Override
    public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds)
    
         Log.d(TAG, "onUpdate()");
         updateWidgets(context);
    


    private void updateWidgets(Context context)
    
        AppWidgetManager m = AppWidgetManager.getInstance(context);
        if (m != null)
        
            int[] appWidgetIds = 
                  m.getAppWidgetIds(new ComponentName(context, getClass()));
            if ((appWidgetIds != null) && (appWidgetIds.length > 0))
            
                 final RemoteViews updateViews = 
                    new RemoteViews(context.getPackageName(),
                    R.layout.mywidget_layout);
                    ...
                    m.updateAppWidget(appWidgetIds, updateViews);
            
        
    

【讨论】:

是否需要注册扩展 AppWidgetProvider 的类才能在 manifest 中注册为接收者 如果我没有遵循注册接收器的常规方法,则小部件不会创建。正如你提到的,我在类中创建了一个扩展 AppWidgetProvider 的方法,并从 Onreceive() 方法调用它,但它不更新小部件。 好的,你应该删除你的Recver 并添加一个OnReceive 方法到Widgetd.java,就像我描述的那样:intent.getAction().equals("android.provider.Telephony.SMS_RECEIVED")。还将 SMS_RECEIVED 的意图过滤器移动到 manifest.xml 中的 Widgetd。在 logcat 中,您应该会看到所有收到的消息。 @ j.holetzeck 先生,我想要的只是在收到短信时提高小部件中 textview 的值,并在 Recver 的 OnReceive() 方法中执行一些其他操作。所以我应该删除 Recver 类并将 OnReceive() 代码转移到 Widgetd.java 中的 OnReceive 作为一个应用程序小部件只是一个广播接收器,我认为这将是一个好方法。

以上是关于不定期更新小部件的主要内容,如果未能解决你的问题,请参考以下文章

定期更新小部件 TextView 的简单方法

每 5 分钟或定期刷新 Flutter Text 小部件内容

小部件更新的 RemoteViews 超出最大位图内存使用错误

通过服务更新 Android 小部件,即执行 http 请求

Today Extension Widget 内容更新

为啥我们必须从一个小部件更新多个小部件?