views:

404

answers:

2

I am currently learning about widgets in Android.

I want to create a WIFI widget that will display the SSID, the RSSI (Signal) level.

But I also want to be able to send it data from a service I am running that calculates the Quality of Sound over wifi.

Here is what I have after some reading and a quick tutorial:


public class WlanWidget extends AppWidgetProvider{

RemoteViews remoteViews;
AppWidgetManager appWidgetManager;
ComponentName thisWidget;
WifiManager wifiManager;

public void onUpdate(Context context, AppWidgetManager appWidgetManager,
        int[] appWidgetIds) {
        Timer timer = new Timer();
        timer.scheduleAtFixedRate(new WlanTimer(context, appWidgetManager), 1, 10000);

}


private class WlanTimer extends TimerTask{

        RemoteViews remoteViews;
        AppWidgetManager appWidgetManager;
        ComponentName thisWidget;


public WlanTimer(Context context, AppWidgetManager appWidgetManager) {

        this.appWidgetManager = appWidgetManager;
        remoteViews = new RemoteViews(context.getPackageName(), R.layout.widget);
        thisWidget = new ComponentName(context, WlanWidget.class);
        wifiManager = (WifiManager)context.getSystemService(Context.WIFI_SERVICE);


}

@Override
public void run() {

        remoteViews.setTextViewText(R.id.widget_textview,
        wifiManager.getConnectionInfo().getSSID());
        appWidgetManager.updateAppWidget(thisWidget, remoteViews);
}

}

The above seems to work ok, it updates the SSID on the widget every 10 seconds.

However what is the most efficent way to get the information from my service that will be already running to update periodically on my widget?

Also is there a better approach to updating the the widget rather than using a timer and timertask? (Avoid polling)

UPDATE

As per Karan's suggestion I have added the following code in my Service:


RemoteViews remoteViews = new RemoteViews(context.getPackageName(), R.layout.widget);
ComponentName thisWidget = new ComponentName( context, WlanWidget.class );
remoteViews.setTextViewText(R.id.widget_QCLevel, " " + qcPercentage);
AppWidgetManager.getInstance( context ).updateAppWidget( thisWidget, remoteViews );

This gets run everytime the RSSI level changes but it still never updates the TextView on my widget, any ideas why?

EDIT

Got it working, thanks Karan

A: 

If you look at the documentation about widgets, you are supposed to be able to set a time between 2 updates :

<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
android:minWidth="294dp"
android:minHeight="72dp"
android:updatePeriodMillis="86400000"
android:initialLayout="@layout/example_appwidget"
android:configure="com.example.android.ExampleAppWidgetConfigure" >

With this, you should be able to put any period you want ? (It mentions that the update might not occur directly after your call for it, and henceforce to do it as often as possible but you have to take care of the battery)

Sephy
From 1.6 the minimum update the OS will allow and perform is 30 minutes appart to save battery life.I need to be able to update the Widget every time the RSSI level changes significantly, the Wifi access point changes and when the calculated QoS from my service falls below or climbs above a certain level.All of which need dynamic updating on the Widget.
Donal Rafferty
+2  A: 

You can use the following code to update the data of widget :

ComponentName thisWidget = new ComponentName( getContext(), <ProviderClass> );
AppWidgetManager.getInstance( getContext() ).updateAppWidget( thisWidget, rempoteViews );
Karan
I can put that in my service and it will update the Widget dynamically?
Donal Rafferty