2016年2月28日 星期日

show some thing when list view no data

we should show “no weather information available on the screen”,not only blank screen



Step1,create a sibling to the List view ,show the string:no weather information


<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
   xmlns:tools="http://schemas.android.com/tools"
   android:layout_width="match_parent"
   android:layout_height="match_parent"
   tools:context="com.example.android.sunshine.app.ForecastFragment">
   <ListView
       style="@style/ForecastListStyle"
       android:id="@+id/listview_forecast"
       android:layout_width="match_parent"
       android:layout_height="match_parent"
       android:divider="@null" />

   <TextView
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"
       android:text="@string/empty_string"
       android:id="@+id/tvEmpty"
       android:textSize="30sp"/>
</FrameLayout>


step2:setEmptyView for the list view


mListView = (ListView) rootView.findViewById(R.id.listview_forecast);
       mListView.setEmptyView(rootView.findViewById(R.id.tvEmpty));
       mListView.setAdapter(mForecastAdapter);

now,when the array list is empty, it will show the string.


More improve

no we can show no information when the ArrayList is empty,however,can we show more information?No information can show reason:
1.no connection
2.server side problem,down,so give us bad input

basic information,how to check the connection state:

1.get the permission:
<!--Permission to check the connection state-->
   <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>

2.get the ConnectivityManager
Context.getSystemService(Context.CONNECTIVITY_SERVICE).

3.check
ConnectivityManager cm =
       (ConnectivityManager)context.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo activeNetwork = cm.getActiveNetworkInfo();
boolean isConnected = activeNetwork != null &&
                     activeNetwork.isConnectedOrConnecting();

Now,we start to check is the network connection is ok,we solve 1.no cinnection first


answer:
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
       mForecastAdapter.swapCursor(data);
       if (data.getCount() == 0) {
           ConnectivityManager cm = (ConnectivityManager) getActivity().getSystemService(Context.CONNECTIVITY_SERVICE);

           NetworkInfo activeNetwork = cm.getActiveNetworkInfo();
           boolean isConnected = activeNetwork != null && activeNetwork.isConnectedOrConnecting();
           if(!isConnected){
               String show=getString(R.string.empty_string)+" The Networ is not available";
               tvEmpty.setText(show);
           }
       }
       if (mPosition != ListView.INVALID_POSITION) {
           // If we don't need to restart the loader, and there's a desired position to restore
           // to, do so now.
           mListView.smoothScrollToPosition(mPosition);
       }
   }

Some basic information for enum and IntDef/StringDef: Typedef Annotations:


use enum in adnroid is not efficient ,so we use IntDef/StringDef to replace it

How to use IntDef/StringDef:
Annotation explain:

Enumerated Annotations explain:

Now ,show how to use IntDef replace the Enum

if we have a function
input(int i)

it only can accpet the below number

public static final int LOCATION_STATUS_OK = 0;
public static final int LOCATION_STATUS_SERVER_DOWN = 1;
public static final int LOCATION_STATUS_SERVER_INVALID = 2;
public static final int LOCATION_STATUS_UNKNOWN = 3;

how can we limite the function only accept the above:
Method1:,we can use enum

enum State {
       LOCATION_STATUS_OK, LOCATION_STATUS_SERVER_DOWN, LOCATION_STATUS_SERVER_INVALID,LOCATION_STATUS_UNKNOWN
   }
void input(State s){
//do something
}

However, enum in android is not efficient ,we will not use it

Method2:use int

public static final int LOCATION_STATUS_OK = 0;
public static final int LOCATION_STATUS_SERVER_DOWN = 1;
public static final int LOCATION_STATUS_SERVER_INVALID = 2;
public static final int LOCATION_STATUS_UNKNOWN = 3;

void input(int state){
//do something
}

however, we can’t not limited user only input the above number,we need to write more function to prevent the use input other number

Method3:

use IntDef

@IntDef(flag = true, value = {LOCATION_STATUS_OK, LOCATION_STATUS_SERVER_DOWN,LOCATION_STATUS_SERVER_INVALID,LOCATION_STATUS_UNKNOWN})
   @Retention(RetentionPolicy.SOURCE)
   public @interface State {}
public static final int LOCATION_STATUS_OK = 0;
public static final int LOCATION_STATUS_SERVER_DOWN = 1;
public static final int LOCATION_STATUS_SERVER_INVALID = 2;
public static final int LOCATION_STATUS_UNKNOWN = 3;

void input(@Stateint state){
//do something
}

Some information of RetentionPolicy

RetentionPolicy.SOURCE: Discard during the compile. These annotations don't make any sense after the compile has completed, so they aren't written to the bytecode.
Example: @Override, @SuppressWarnings

RetentionPolicy.CLASS: Discard during class load. Useful when doing bytecode-level post-processing. Somewhat surprisingly, this is the default.

RetentionPolicy.RUNTIME: Do not discard. The annotation should be available for reflection at runtime. Example: @Deprecated

now ,solve reason 2,server give us the bad information


step1:define the state code
@IntDef(flag = true, value = {LOCATION_STATUS_OK, LOCATION_STATUS_SERVER_DOWN, LOCATION_STATUS_SERVER_INVALID, LOCATION_STATUS_UNKNOWN})
   @Retention(RetentionPolicy.SOURCE)
   public @interface LocationStates {
   }
   public static final int LOCATION_STATUS_OK = 0;
   public static final int LOCATION_STATUS_SERVER_DOWN = 1;
   public static final int LOCATION_STATUS_SERVER_INVALID = 2;
   public static final int LOCATION_STATUS_UNKNOWN = 3;

step2,define the function to save the state by sharedPreference

private void setLocationState(@LocationStates int locationstate){
       SharedPreferences sp = mContext.getSharedPreferences(mContext.getString(R.string.pref_location_state_key), Context.MODE_PRIVATE);
       SharedPreferences.Editor editor = sp.edit();
       editor.putInt(mContext.getString(R.string.pref_location_state_key), locationstate);
       editor.commit();
   }

step3 call the function on different location in the app
principle:
can get the data and from json to the database,the LOCATION_STATUS_OK

can get data but the json string can’t cahnge to the object.that mean the josn string is not correct,

so LOCATION_STATUS_SERVER_INVALID

can’t connect to theserver but connect to the internet
or
can’’t get any data(return zero length string) but can connect to the server,

so LOCATION_STATUS_SERVER_DOWN

we can use the below path to try

FORECAST_BASE_URL that will return an invalid result:http://google.com/?
th is LOCATION_STATUS_SERVER_INVALID
FORECAST_BASE_URL that will return an empty result:http://google.com/ping?
this is LOCATION_STATUS_SERVER_DOWN

full answeer
package com.example.android.sunshine.app.sync;

import android.accounts.Account;
import android.accounts.AccountManager;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.AbstractThreadedSyncAdapter;
import android.content.ContentProviderClient;
import android.content.ContentResolver;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.Context;
import android.content.Intent;
import android.content.SharedPreferences;
import android.content.SyncRequest;
import android.content.SyncResult;
import android.content.res.Resources;
import android.database.Cursor;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.preference.PreferenceManager;
import android.support.annotation.IntDef;
import android.support.v4.app.NotificationCompat;
import android.support.v4.app.TaskStackBuilder;
import android.text.format.Time;
import android.util.Log;

import com.example.android.sunshine.app.BuildConfig;
import com.example.android.sunshine.app.MainActivity;
import com.example.android.sunshine.app.R;
import com.example.android.sunshine.app.Utility;
import com.example.android.sunshine.app.data.WeatherContract;

import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.Vector;

public class SunshineSyncAdapter extends AbstractThreadedSyncAdapter {

   @IntDef(flag = true, value = {LOCATION_STATUS_OK, LOCATION_STATUS_SERVER_DOWN, LOCATION_STATUS_SERVER_INVALID, LOCATION_STATUS_UNKNOWN})
   @Retention(RetentionPolicy.SOURCE)
   public @interface LocationStates {
   }
   public static final int LOCATION_STATUS_OK = 0;
   public static final int LOCATION_STATUS_SERVER_DOWN = 1;
   public static final int LOCATION_STATUS_SERVER_INVALID = 2;
   public static final int LOCATION_STATUS_UNKNOWN = 3;
   // Interval at which to sync with the weather, in seconds.
   // 60 seconds (1 minute) * 180 = 3 hours
   public static final int SYNC_INTERVAL = 60 * 180;
   public static final int SYNC_FLEXTIME = SYNC_INTERVAL / 3;
   private static final long DAY_IN_MILLIS = 1000 * 60 * 60 * 24;
   private static final int WEATHER_NOTIFICATION_ID = 3004;
   private static final String[] NOTIFY_WEATHER_PROJECTION = new String[]{
           WeatherContract.WeatherEntry.COLUMN_WEATHER_ID,
           WeatherContract.WeatherEntry.COLUMN_MAX_TEMP,
           WeatherContract.WeatherEntry.COLUMN_MIN_TEMP,
           WeatherContract.WeatherEntry.COLUMN_SHORT_DESC
   };
   // these indices must match the projection
   private static final int INDEX_WEATHER_ID = 0;
   private static final int INDEX_MAX_TEMP = 1;
   private static final int INDEX_MIN_TEMP = 2;
   private static final int INDEX_SHORT_DESC = 3;
   public final String LOG_TAG = SunshineSyncAdapter.class.getSimpleName();
   private final Context mContext;

   public SunshineSyncAdapter(Context context, boolean autoInitialize) {
       super(context, autoInitialize);
       mContext=context;
   }

   /**
    * Helper method to schedule the sync adapter periodic execution
    */
   private void setLocationState(@LocationStates int locationstate){
       SharedPreferences sp = mContext.getSharedPreferences(mContext.getString(R.string.pref_location_state_key), Context.MODE_PRIVATE);
       SharedPreferences.Editor editor = sp.edit();
       editor.putInt(mContext.getString(R.string.pref_location_state_key), locationstate);
       editor.commit();
   }
   public static void configurePeriodicSync(Context context, int syncInterval, int flexTime) {
       Account account = getSyncAccount(context);
       String authority = context.getString(R.string.content_authority);
       if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
           // we can enable inexact timers in our periodic sync
           SyncRequest request = new SyncRequest.Builder().
                   syncPeriodic(syncInterval, flexTime).
                   setSyncAdapter(account, authority).
                   setExtras(new Bundle()).build();
           ContentResolver.requestSync(request);
       } else {
           ContentResolver.addPeriodicSync(account,
                   authority, new Bundle(), syncInterval);
       }
   }

   /**
    * Helper method to have the sync adapter sync immediately
    *
    * @param context The context used to access the account service
    */
   public static void syncImmediately(Context context) {
       Bundle bundle = new Bundle();
       bundle.putBoolean(ContentResolver.SYNC_EXTRAS_EXPEDITED, true);
       bundle.putBoolean(ContentResolver.SYNC_EXTRAS_MANUAL, true);
       ContentResolver.requestSync(getSyncAccount(context),
               context.getString(R.string.content_authority), bundle);
   }

   /**
    * Helper method to get the fake account to be used with SyncAdapter, or make a new one
    * if the fake account doesn't exist yet.  If we make a new account, we call the
    * onAccountCreated method so we can initialize things.
    *
    * @param context The context used to access the account service
    * @return a fake account.
    */
   public static Account getSyncAccount(Context context) {
       // Get an instance of the Android account manager
       AccountManager accountManager =
               (AccountManager) context.getSystemService(Context.ACCOUNT_SERVICE);

       // Create the account type and default account
       Account newAccount = new Account(
               context.getString(R.string.app_name), context.getString(R.string.sync_account_type));

       // If the password doesn't exist, the account doesn't exist
       if (null == accountManager.getPassword(newAccount)) {

       /*
        * Add the account and account type, no password or user data
        * If successful, return the Account object, otherwise report an error.
        */
           if (!accountManager.addAccountExplicitly(newAccount, "", null)) {
               return null;
           }
           /*
            * If you don't set android:syncable="true" in
            * in your <provider> element in the manifest,
            * then call ContentResolver.setIsSyncable(account, AUTHORITY, 1)
            * here.
            */

           onAccountCreated(newAccount, context);
       }
       return newAccount;
   }

   private static void onAccountCreated(Account newAccount, Context context) {
       /*
        * Since we've created an account
        */
       SunshineSyncAdapter.configurePeriodicSync(context, SYNC_INTERVAL, SYNC_FLEXTIME);

       /*
        * Without calling setSyncAutomatically, our periodic sync will not be enabled.
        */
       ContentResolver.setSyncAutomatically(newAccount, context.getString(R.string.content_authority), true);

       /*
        * Finally, let's do a sync to get things started
        */
       syncImmediately(context);
   }

   public static void initializeSyncAdapter(Context context) {
       getSyncAccount(context);
   }

   @Override
   public void onPerformSync(Account account, Bundle extras, String authority, ContentProviderClient provider, SyncResult syncResult) {
       Log.d(LOG_TAG, "Starting sync");
       String locationQuery = Utility.getPreferredLocation(getContext());

       // These two need to be declared outside the try/catch
       // so that they can be closed in the finally block.
       HttpURLConnection urlConnection = null;
       BufferedReader reader = null;

       // Will contain the raw JSON response as a string.
       String forecastJsonStr = null;

       String format = "json";
       String units = "metric";
       int numDays = 14;

       try {
           // Construct the URL for the OpenWeatherMap query
           // Possible parameters are avaiable at OWM's forecast API page, at
           // http://openweathermap.org/API#forecast
           final String FORECAST_BASE_URL =
                   "http://api.openweathermap.org/data/2.5/forecast/daily?";
           final String QUERY_PARAM = "q";
           final String FORMAT_PARAM = "mode";
           final String UNITS_PARAM = "units";
           final String DAYS_PARAM = "cnt";
           final String APPID_PARAM = "APPID";

           Uri builtUri = Uri.parse(FORECAST_BASE_URL).buildUpon()
                   .appendQueryParameter(QUERY_PARAM, locationQuery)
                   .appendQueryParameter(FORMAT_PARAM, format)
                   .appendQueryParameter(UNITS_PARAM, units)
                   .appendQueryParameter(DAYS_PARAM, Integer.toString(numDays))
                   .appendQueryParameter(APPID_PARAM, BuildConfig.OPEN_WEATHER_MAP_API_KEY)
                   .build();

           URL url = new URL(builtUri.toString());

           // Create the request to OpenWeatherMap, and open the connection
           urlConnection = (HttpURLConnection) url.openConnection();
           urlConnection.setRequestMethod("GET");
           urlConnection.connect();

           // Read the input stream into a String
           InputStream inputStream = urlConnection.getInputStream();
           StringBuffer buffer = new StringBuffer();
           if (inputStream == null) {
               // Nothing to do.
               return;
           }
           reader = new BufferedReader(new InputStreamReader(inputStream));

           String line;
           while ((line = reader.readLine()) != null) {
               // Since it's JSON, adding a newline isn't necessary (it won't affect parsing)
               // But it does make debugging a *lot* easier if you print out the completed
               // buffer for debugging.
               buffer.append(line + "\n");
           }

           if (buffer.length() == 0) {
               // Stream was empty.  No point in parsing.
               setLocationState(LOCATION_STATUS_SERVER_DOWN);
               return;
           }
           forecastJsonStr = buffer.toString();
           getWeatherDataFromJson(forecastJsonStr, locationQuery);
       } catch (IOException e) {
           Log.e(LOG_TAG, "Error ", e);
           // If the code didn't successfully get the weather data, there's no point in attempting
           // to parse it.
           setLocationState(LOCATION_STATUS_SERVER_DOWN);
       } catch (JSONException e) {
           Log.e(LOG_TAG, e.getMessage(), e);
           e.printStackTrace();
           setLocationState(LOCATION_STATUS_SERVER_INVALID);
       } finally {
           if (urlConnection != null) {
               urlConnection.disconnect();
           }
           if (reader != null) {
               try {
                   reader.close();
               } catch (final IOException e) {
                   Log.e(LOG_TAG, "Error closing stream", e);
               }
           }
       }

       return;
   }

   /**
    * Take the String representing the complete forecast in JSON Format and
    * pull out the data we need to construct the Strings needed for the wireframes.
    * <p/>
    * Fortunately parsing is easy:  constructor takes the JSON string and converts it
    * into an Object hierarchy for us.
    */
   private void getWeatherDataFromJson(String forecastJsonStr,
                                       String locationSetting)
           throws JSONException {

       // Now we have a String representing the complete forecast in JSON Format.
       // Fortunately parsing is easy:  constructor takes the JSON string and converts it
       // into an Object hierarchy for us.

       // These are the names of the JSON objects that need to be extracted.

       // Location information
       final String OWM_CITY = "city";
       final String OWM_CITY_NAME = "name";
       final String OWM_COORD = "coord";

       // Location coordinate
       final String OWM_LATITUDE = "lat";
       final String OWM_LONGITUDE = "lon";

       // Weather information.  Each day's forecast info is an element of the "list" array.
       final String OWM_LIST = "list";

       final String OWM_PRESSURE = "pressure";
       final String OWM_HUMIDITY = "humidity";
       final String OWM_WINDSPEED = "speed";
       final String OWM_WIND_DIRECTION = "deg";

       // All temperatures are children of the "temp" object.
       final String OWM_TEMPERATURE = "temp";
       final String OWM_MAX = "max";
       final String OWM_MIN = "min";

       final String OWM_WEATHER = "weather";
       final String OWM_DESCRIPTION = "main";
       final String OWM_WEATHER_ID = "id";

       try {
           JSONObject forecastJson = new JSONObject(forecastJsonStr);
           JSONArray weatherArray = forecastJson.getJSONArray(OWM_LIST);

           JSONObject cityJson = forecastJson.getJSONObject(OWM_CITY);
           String cityName = cityJson.getString(OWM_CITY_NAME);

           JSONObject cityCoord = cityJson.getJSONObject(OWM_COORD);
           double cityLatitude = cityCoord.getDouble(OWM_LATITUDE);
           double cityLongitude = cityCoord.getDouble(OWM_LONGITUDE);

           long locationId = addLocation(locationSetting, cityName, cityLatitude, cityLongitude);

           // Insert the new weather information into the database
           Vector<ContentValues> cVVector = new Vector<ContentValues>(weatherArray.length());

           // OWM returns daily forecasts based upon the local time of the city that is being
           // asked for, which means that we need to know the GMT offset to translate this data
           // properly.

           // Since this data is also sent in-order and the first day is always the
           // current day, we're going to take advantage of that to get a nice
           // normalized UTC date for all of our weather.

           Time dayTime = new Time();
           dayTime.setToNow();

           // we start at the day returned by local time. Otherwise this is a mess.
           int julianStartDay = Time.getJulianDay(System.currentTimeMillis(), dayTime.gmtoff);

           // now we work exclusively in UTC
           dayTime = new Time();

           for (int i = 0; i < weatherArray.length(); i++) {
               // These are the values that will be collected.
               long dateTime;
               double pressure;
               int humidity;
               double windSpeed;
               double windDirection;

               double high;
               double low;

               String description;
               int weatherId;

               // Get the JSON object representing the day
               JSONObject dayForecast = weatherArray.getJSONObject(i);

               // Cheating to convert this to UTC time, which is what we want anyhow
               dateTime = dayTime.setJulianDay(julianStartDay + i);

               pressure = dayForecast.getDouble(OWM_PRESSURE);
               humidity = dayForecast.getInt(OWM_HUMIDITY);
               windSpeed = dayForecast.getDouble(OWM_WINDSPEED);
               windDirection = dayForecast.getDouble(OWM_WIND_DIRECTION);

               // Description is in a child array called "weather", which is 1 element long.
               // That element also contains a weather code.
               JSONObject weatherObject =
                       dayForecast.getJSONArray(OWM_WEATHER).getJSONObject(0);
               description = weatherObject.getString(OWM_DESCRIPTION);
               weatherId = weatherObject.getInt(OWM_WEATHER_ID);

               // Temperatures are in a child object called "temp".  Try not to name variables
               // "temp" when working with temperature.  It confuses everybody.
               JSONObject temperatureObject = dayForecast.getJSONObject(OWM_TEMPERATURE);
               high = temperatureObject.getDouble(OWM_MAX);
               low = temperatureObject.getDouble(OWM_MIN);

               ContentValues weatherValues = new ContentValues();

               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_LOC_KEY, locationId);
               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_DATE, dateTime);
               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_HUMIDITY, humidity);
               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_PRESSURE, pressure);
               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_WIND_SPEED, windSpeed);
               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_DEGREES, windDirection);
               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_MAX_TEMP, high);
               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_MIN_TEMP, low);
               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_SHORT_DESC, description);
               weatherValues.put(WeatherContract.WeatherEntry.COLUMN_WEATHER_ID, weatherId);

               cVVector.add(weatherValues);
           }

           int inserted = 0;
           // add to database
           if (cVVector.size() > 0) {
               ContentValues[] cvArray = new ContentValues[cVVector.size()];
               cVVector.toArray(cvArray);
               getContext().getContentResolver().bulkInsert(WeatherContract.WeatherEntry.CONTENT_URI, cvArray);

               // delete old data so we don't build up an endless history
               getContext().getContentResolver().delete(WeatherContract.WeatherEntry.CONTENT_URI,
                       WeatherContract.WeatherEntry.COLUMN_DATE + " <= ?",
                       new String[]{Long.toString(dayTime.setJulianDay(julianStartDay - 1))});

               notifyWeather();
               setLocationState(LOCATION_STATUS_OK);
           }

           Log.d(LOG_TAG, "Sync Complete. " + cVVector.size() + " Inserted");

       } catch (JSONException e) {
           Log.e(LOG_TAG, e.getMessage(), e);
           e.printStackTrace();
       }
   }

   private void notifyWeather() {
       Context context = getContext();
       //checking the last update and notify if it' the first of the day
       SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
       String displayNotificationsKey = context.getString(R.string.pref_enable_notifications_key);
       boolean displayNotifications = prefs.getBoolean(displayNotificationsKey,
               Boolean.parseBoolean(context.getString(R.string.pref_enable_notifications_default)));

       if (displayNotifications) {

           String lastNotificationKey = context.getString(R.string.pref_last_notification);
           long lastSync = prefs.getLong(lastNotificationKey, 0);

           if (System.currentTimeMillis() - lastSync >= DAY_IN_MILLIS) {
               // Last sync was more than 1 day ago, let's send a notification with the weather.
               String locationQuery = Utility.getPreferredLocation(context);

               Uri weatherUri = WeatherContract.WeatherEntry.buildWeatherLocationWithDate(locationQuery, System.currentTimeMillis());

               // we'll query our contentProvider, as always
               Cursor cursor = context.getContentResolver().query(weatherUri, NOTIFY_WEATHER_PROJECTION, null, null, null);

               if (cursor.moveToFirst()) {
                   int weatherId = cursor.getInt(INDEX_WEATHER_ID);
                   double high = cursor.getDouble(INDEX_MAX_TEMP);
                   double low = cursor.getDouble(INDEX_MIN_TEMP);
                   String desc = cursor.getString(INDEX_SHORT_DESC);

                   int iconId = Utility.getIconResourceForWeatherCondition(weatherId);
                   Resources resources = context.getResources();
                   Bitmap largeIcon = BitmapFactory.decodeResource(resources,
                           Utility.getArtResourceForWeatherCondition(weatherId));
                   String title = context.getString(R.string.app_name);

                   // Define the text of the forecast.
                   String contentText = String.format(context.getString(R.string.format_notification),
                           desc,
                           Utility.formatTemperature(context, high),
                           Utility.formatTemperature(context, low));

                   // NotificationCompatBuilder is a very convenient way to build backward-compatible
                   // notifications.  Just throw in some data.
                   NotificationCompat.Builder mBuilder =
                           new NotificationCompat.Builder(getContext())
                                   .setColor(resources.getColor(R.color.sunshine_light_blue))
                                   .setSmallIcon(iconId)
                                   .setLargeIcon(largeIcon)
                                   .setContentTitle(title)
                                   .setContentText(contentText);

                   // Make something interesting happen when the user clicks on the notification.
                   // In this case, opening the app is sufficient.
                   Intent resultIntent = new Intent(context, MainActivity.class);

                   // The stack builder object will contain an artificial back stack for the
                   // started Activity.
                   // This ensures that navigating backward from the Activity leads out of
                   // your application to the Home screen.
                   TaskStackBuilder stackBuilder = TaskStackBuilder.create(context);
                   stackBuilder.addNextIntent(resultIntent);
                   PendingIntent resultPendingIntent =
                           stackBuilder.getPendingIntent(
                                   0,
                                   PendingIntent.FLAG_UPDATE_CURRENT
                           );
                   mBuilder.setContentIntent(resultPendingIntent);

                   NotificationManager mNotificationManager =
                           (NotificationManager) getContext().getSystemService(Context.NOTIFICATION_SERVICE);
                   // WEATHER_NOTIFICATION_ID allows you to update the notification later on.
                   mNotificationManager.notify(WEATHER_NOTIFICATION_ID, mBuilder.build());

                   //refreshing last sync
                   SharedPreferences.Editor editor = prefs.edit();
                   editor.putLong(lastNotificationKey, System.currentTimeMillis());
                   editor.commit();
               }
               cursor.close();
           }
       }
   }

   /**
    * Helper method to handle insertion of a new location in the weather database.
    *
    * @param locationSetting The location string used to request updates from the server.
    * @param cityName        A human-readable city name, e.g "Mountain View"
    * @param lat             the latitude of the city
    * @param lon             the longitude of the city
    * @return the row ID of the added location.
    */
   long addLocation(String locationSetting, String cityName, double lat, double lon) {
       long locationId;

       // First, check if the location with this city name exists in the db
       Cursor locationCursor = getContext().getContentResolver().query(
               WeatherContract.LocationEntry.CONTENT_URI,
               new String[]{WeatherContract.LocationEntry._ID},
               WeatherContract.LocationEntry.COLUMN_LOCATION_SETTING + " = ?",
               new String[]{locationSetting},
               null);

       if (locationCursor.moveToFirst()) {
           int locationIdIndex = locationCursor.getColumnIndex(WeatherContract.LocationEntry._ID);
           locationId = locationCursor.getLong(locationIdIndex);
       } else {
           // Now that the content provider is set up, inserting rows of data is pretty simple.
           // First create a ContentValues object to hold the data you want to insert.
           ContentValues locationValues = new ContentValues();

           // Then add the data, along with the corresponding name of the data type,
           // so the content provider knows what kind of value is being inserted.
           locationValues.put(WeatherContract.LocationEntry.COLUMN_CITY_NAME, cityName);
           locationValues.put(WeatherContract.LocationEntry.COLUMN_LOCATION_SETTING, locationSetting);
           locationValues.put(WeatherContract.LocationEntry.COLUMN_COORD_LAT, lat);
           locationValues.put(WeatherContract.LocationEntry.COLUMN_COORD_LONG, lon);

           // Finally, insert location data into the database.
           Uri insertedUri = getContext().getContentResolver().insert(
                   WeatherContract.LocationEntry.CONTENT_URI,
                   locationValues
           );

           // The resulting URI contains the ID for the row.  Extract the locationId from the Uri.
           locationId = ContentUris.parseId(insertedUri);
       }

       locationCursor.close();
       // Wait, that worked?  Yes!
       return locationId;
   }

}

now,we know the server state,but don’t show to the UI

Now show the state to UI

1.Make a function to read the location state.
public static int getLocationState(Context context) {
  return context.getSharedPreferences(context.getString(R.string.pref_location_state_key), Context.MODE_PRIVATE).
          getInt((context.getString(R.string.pref_location_state_key)), SunshineSyncAdapter.LOCATION_STATUS_UNKNOWN);
}

2when start the app .show the location statue to the UI
private void updatetheTextViewshow(){
  int i=Utility.getLocationState(getContext());
  switch (i){
      case SunshineSyncAdapter.LOCATION_STATUS_SERVER_DOWN:
          tvEmpty.setText(getContext().getString(R.string.empty_forecast_list_server_down));
          break;
      case SunshineSyncAdapter.LOCATION_STATUS_SERVER_INVALID:
          tvEmpty.setText(getContext().getString(R.string.empty_forecast_list_server_error));
          break;
  }

}

call at onCreate View of ForecastFragment

3.Next time when the asyncadapter get information,the location state maybe change,show the state show on th ui need to change.

so, we need to Register the registerOnSharedPreferenceChangeListener at onResume of ForestFragmnt

Context context=getContext();
       context.getSharedPreferences(context.getString(R.string.pref_location_state_key),Context.MODE_PRIVATE).registerOnSharedPreferenceChangeListener(this);

and

unregisterOnSharedPreferenceChangeListener at onPauseof ForestFragmnt

Context context=getContext();
       context.getSharedPreferences(context.getString(R.string.pref_location_state_key),Context.MODE_PRIVATE).unregisterOnSharedPreferenceChangeListener(this);

4.ForestFragmnt implement SharedPreferences.OnSharedPreferenceChangeListener and override onSharedPreferenceChanged

in onSharedPreferenceChanged,
updatetheTextViewshow();





Now, when the SharedPreference ,the UI will update again.

Now,we solve the nothing to show problem