Last active
February 17, 2017 09:09
-
-
Save jimitjaishwal/492bc5abe677a3a826f984eaf3aee95c to your computer and use it in GitHub Desktop.
When Launch First time App Loader Manager Does not querying data..But when I Close the app and run again app now see list full forecast data.please Explain me
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| package com.example.android.sunshine.app; | |
| import android.content.ContentUris; | |
| import android.content.ContentValues; | |
| import android.content.Context; | |
| import android.database.Cursor; | |
| import android.net.Uri; | |
| import android.os.AsyncTask; | |
| import android.text.format.Time; | |
| import android.util.Log; | |
| import com.example.android.sunshine.app.data.WeatherContract; | |
| import com.example.android.sunshine.app.data.WeatherContract.WeatherEntry; | |
| 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.net.HttpURLConnection; | |
| import java.net.URL; | |
| import java.util.Vector; | |
| /** | |
| * Created by JIMIT JAISHWAL on 16-02-2017. | |
| */ | |
| public class FetchWeatherTask extends AsyncTask<String, Void, Void> { | |
| private final String LOG_TAG = FetchWeatherTask.class.getSimpleName(); | |
| private final Context mContext; | |
| public FetchWeatherTask(Context context) { | |
| mContext = context; | |
| } | |
| private boolean DEBUG = true; | |
| /** | |
| * 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. | |
| */ | |
| private long addLocation(String locationSetting, String cityName, double lat, double lon) { | |
| long locationId; | |
| Cursor cursor = mContext.getContentResolver().query( | |
| WeatherContract.LocationEntry.LOCATION_URI, | |
| new String[]{WeatherContract.LocationEntry._ID}, | |
| WeatherContract.LocationEntry.COLUMN_LOCATION_SETTING + "=?", | |
| new String[]{locationSetting}, | |
| null | |
| ); | |
| if (cursor != null && cursor.moveToFirst()) { | |
| int locationIndex = cursor.getColumnIndex(WeatherContract.LocationEntry._ID); | |
| locationId = cursor.getLong(locationIndex); | |
| } else { | |
| ContentValues contentValues = new ContentValues(); | |
| contentValues.put(WeatherContract.LocationEntry.COLUMN_LOCATION_SETTING, locationSetting); | |
| contentValues.put(WeatherContract.LocationEntry.COLUMN_CITY_NAME, cityName); | |
| contentValues.put(WeatherContract.LocationEntry.COLUMN_COORD_LAT, lat); | |
| contentValues.put(WeatherContract.LocationEntry.COLUMN_COORD_LONG, lon); | |
| Uri buildUri = mContext.getContentResolver().insert(WeatherContract.LocationEntry.LOCATION_URI, contentValues); | |
| locationId = ContentUris.parseId(buildUri); | |
| } | |
| // First, check if the location with this city name exists in the db | |
| if (cursor != null) { | |
| cursor.close(); | |
| } | |
| return locationId; | |
| } | |
| private Void getWeatherDataFromJson(String forecastJsonStr, | |
| String locationSetting) | |
| throws JSONException { | |
| 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()); | |
| 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(WeatherEntry.COLUMN_LOC_KEY, locationId); | |
| weatherValues.put(WeatherEntry.COLUMN_DATE, dateTime); | |
| weatherValues.put(WeatherEntry.COLUMN_HUMIDITY, humidity); | |
| weatherValues.put(WeatherEntry.COLUMN_PRESSURE, pressure); | |
| weatherValues.put(WeatherEntry.COLUMN_WIND_SPEED, windSpeed); | |
| weatherValues.put(WeatherEntry.COLUMN_DEGREES, windDirection); | |
| weatherValues.put(WeatherEntry.COLUMN_MAX_TEMP, high); | |
| weatherValues.put(WeatherEntry.COLUMN_MIN_TEMP, low); | |
| weatherValues.put(WeatherEntry.COLUMN_SHORT_DESC, description); | |
| weatherValues.put(WeatherEntry.COLUMN_WEATHER_ID, weatherId); | |
| cVVector.add(weatherValues); | |
| } | |
| int inserted = 0; | |
| // add to database | |
| if (cVVector.size() > 0) { | |
| // Student: call bulkInsert to add the weatherEntries to the database here | |
| ContentValues[] contentValues = new ContentValues[cVVector.size()]; | |
| cVVector.toArray(contentValues); | |
| inserted = mContext.getContentResolver().bulkInsert(WeatherEntry.WEATHER_URI, contentValues); | |
| Log.d("FetchWeatherTask :", inserted + " Rows Inserted"); | |
| } | |
| return null; | |
| } catch (JSONException e) { | |
| Log.e(LOG_TAG, e.getMessage(), e); | |
| e.printStackTrace(); | |
| } | |
| return null; | |
| } | |
| @Override | |
| protected Void doInBackground(String... params) { | |
| // If there's no zip code, there's nothing to look up. Verify size of params. | |
| if (params.length == 0) { | |
| return null; | |
| } | |
| String locationQuery = params[0]; | |
| // 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, params[0]) | |
| .appendQueryParameter(FORMAT_PARAM, format) | |
| .appendQueryParameter(UNITS_PARAM, units) | |
| .appendQueryParameter(DAYS_PARAM, Integer.toString(numDays)) | |
| .appendQueryParameter(APPID_PARAM, "b7feebba97df4bcc5885e36ba67ce966") | |
| .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 null; | |
| } | |
| 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. | |
| return null; | |
| } | |
| 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 attemping | |
| // to parse it. | |
| return null; | |
| } catch (JSONException e) { | |
| e.printStackTrace(); | |
| Log.e(LOG_TAG, e.getMessage(), e); | |
| } 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 null; | |
| } | |
| } |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| package com.example.android.sunshine.app; | |
| import android.database.Cursor; | |
| import android.net.Uri; | |
| import android.os.Bundle; | |
| import android.support.annotation.Nullable; | |
| import android.support.v4.app.Fragment; | |
| import android.support.v4.app.LoaderManager; | |
| import android.support.v4.content.CursorLoader; | |
| import android.support.v4.content.Loader; | |
| import android.util.Log; | |
| import android.view.LayoutInflater; | |
| import android.view.Menu; | |
| import android.view.MenuInflater; | |
| import android.view.MenuItem; | |
| import android.view.View; | |
| import android.view.ViewGroup; | |
| import android.widget.ListView; | |
| import com.example.android.sunshine.app.data.WeatherContract; | |
| /** | |
| * Encapsulates fetching the forecast and displaying it as a {@link ListView} layout. | |
| */ | |
| public class ForecastFragment extends Fragment implements LoaderManager.LoaderCallbacks<Cursor> { | |
| private ForecastAdapter mForecastAdapter; | |
| public static final int MY_LOADER_ID = 0; | |
| public ForecastFragment() { | |
| } | |
| @Override | |
| public void onCreate(Bundle savedInstanceState) { | |
| super.onCreate(savedInstanceState); | |
| // Add this line in order for this fragment to handle menu events. | |
| setHasOptionsMenu(true); | |
| } | |
| @Override | |
| public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { | |
| inflater.inflate(R.menu.forecastfragment, menu); | |
| } | |
| @Override | |
| public boolean onOptionsItemSelected(MenuItem item) { | |
| // Handle action bar item clicks here. The action bar will | |
| // automatically handle clicks on the Home/Up button, so long | |
| // as you specify a parent activity in AndroidManifest.xml. | |
| int id = item.getItemId(); | |
| if (id == R.id.action_refresh) { | |
| updateWeather(); | |
| getLoaderManager().restartLoader(MY_LOADER_ID, null, this); | |
| return true; | |
| } | |
| return super.onOptionsItemSelected(item); | |
| } | |
| @Override | |
| public void onStart() { | |
| super.onStart(); | |
| updateWeather(); | |
| Log.e("ForecastFragment", "onStart() "); | |
| } | |
| public void updateWeather() { | |
| String locationSetting = Utility.getPreferredLocation(getActivity()); | |
| FetchWeatherTask weatherTask = new FetchWeatherTask(getActivity()); | |
| weatherTask.execute(locationSetting); | |
| } | |
| @Override | |
| public View onCreateView(LayoutInflater inflater, ViewGroup container, | |
| Bundle savedInstanceState) { | |
| mForecastAdapter = new ForecastAdapter(getActivity(), null, 0); | |
| View rootView = inflater.inflate(R.layout.fragment_main, container, false); | |
| // Get a reference to the ListView, and attach this adapter to it. | |
| ListView listView = (ListView) rootView.findViewById(R.id.listview_forecast); | |
| listView.setAdapter(mForecastAdapter); | |
| return rootView; | |
| } | |
| @Override | |
| public void onActivityCreated(@Nullable Bundle savedInstanceState) { | |
| getLoaderManager().initLoader(MY_LOADER_ID, null, this).forceLoad(); | |
| super.onActivityCreated(savedInstanceState); | |
| } | |
| @Override | |
| public Loader<Cursor> onCreateLoader(int id, Bundle args) { | |
| String sortOrder = WeatherContract.WeatherEntry.COLUMN_DATE + " ASC"; | |
| String locationSetting = Utility.getPreferredLocation(getActivity()); | |
| Uri weatherLocationWithStartDate = | |
| WeatherContract.WeatherEntry.buildWeatherLocationWithStartDate(locationSetting, System.currentTimeMillis()); | |
| return new CursorLoader( | |
| getActivity(), | |
| weatherLocationWithStartDate, | |
| null, | |
| null, | |
| null, | |
| sortOrder); | |
| } | |
| @Override | |
| public void onLoadFinished(Loader<Cursor> loader, Cursor data) { | |
| mForecastAdapter.swapCursor(data); | |
| mForecastAdapter.notifyDataSetChanged(); | |
| } | |
| @Override | |
| public void onLoaderReset(Loader<Cursor> loader) { | |
| mForecastAdapter.swapCursor(null); | |
| } | |
| } |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment