package uk.me.njae.sunshine;
import android.content.Intent;
-import android.content.SharedPreferences;
+import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
-import android.preference.PreferenceManager;
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.support.v4.view.MenuItemCompat;
import android.support.v7.app.ActionBarActivity;
import android.support.v7.widget.ShareActionProvider;
import android.view.ViewGroup;
import android.widget.TextView;
-import java.io.UnsupportedEncodingException;
-import java.net.URLEncoder;
+import uk.me.njae.sunshine.data.WeatherContract;
+import uk.me.njae.sunshine.data.WeatherContract.WeatherEntry;
// import android.widget.ShareActionProvider;
private final String LOG_TAG = DetailActivity.class.getSimpleName();
private ShareActionProvider mShareActionProvider;
+ public static final String DATE_KEY = "forecast_date";
+ private static final String LOCATION_KEY = "location";
+
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
/**
* A placeholder fragment containing a simple view.
*/
- public static class DetailFragment extends Fragment {
+ public static class DetailFragment extends Fragment implements LoaderManager.LoaderCallbacks<Cursor> {
private static final String LOG_TAG = DetailFragment.class.getSimpleName();
private static final String FORECAST_SHARE_HASHTAG = " #SunshineApp";
- private String mForecastStr;
+
+ public static final String DATE_KEY = "forecast_date";
+
+ private ShareActionProvider mShareActionProvider;
+ private String mLocation;
+ private String mForecast;
+
+ private static final int DETAIL_LOADER = 0;
+
+ private static final String[] FORECAST_COLUMNS = {
+ WeatherEntry.TABLE_NAME + "." + WeatherEntry._ID,
+ WeatherEntry.COLUMN_DATETEXT,
+ WeatherEntry.COLUMN_SHORT_DESC,
+ WeatherEntry.COLUMN_MAX_TEMP,
+ WeatherEntry.COLUMN_MIN_TEMP,
+ };
public DetailFragment() {
setHasOptionsMenu(true);
}
+ @Override
+ public void onSaveInstanceState(Bundle outState) {
+ outState.putString(LOCATION_KEY, mLocation);
+ super.onSaveInstanceState(outState);
+ }
+
+ @Override
+ public void onResume() {
+ super.onResume();
+ if (mLocation != null &&
+ !mLocation.equals(Utility.getPreferredLocation(getActivity()))) {
+ getLoaderManager().restartLoader(DETAIL_LOADER, null, this);
+ }
+ }
+
+
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
- Intent intent = getActivity().getIntent();
- View rootView = inflater.inflate(R.layout.fragment_detail, container, false);
- if (intent != null && intent.hasExtra(Intent.EXTRA_TEXT)) {
- mForecastStr = intent.getStringExtra(Intent.EXTRA_TEXT);
- ((TextView) rootView.findViewById(R.id.detail_text))
- .setText(mForecastStr);
- }
- return rootView;
+ return inflater.inflate(R.layout.fragment_detail, container, false);
}
@Override
Intent shareIntent = new Intent(Intent.ACTION_SEND);
shareIntent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET);
shareIntent.setType("text/plain");
- shareIntent.putExtra(Intent.EXTRA_TEXT,
- mForecastStr + FORECAST_SHARE_HASHTAG);
+ shareIntent.putExtra(Intent.EXTRA_TEXT, mForecast + FORECAST_SHARE_HASHTAG);
return shareIntent;
}
+
@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_show_location) {
- SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(getActivity());
- String location = preferences.getString(getString(R.string.pref_location_key),
- getString(R.string.pref_location_default));
- Intent intent = new Intent(Intent.ACTION_VIEW);
- Uri geoLocation;
- try {
- geoLocation = Uri.parse("geo:0,0?q=" + URLEncoder.encode(location, "UTF-8"));
- intent.setData(geoLocation);
- } catch (UnsupportedEncodingException e) {
- Log.e(LOG_TAG, "Error ", e);
- }
- if (intent.resolveActivity(getActivity().getPackageManager()) != null) {
- startActivity(intent);
- }
+ public void onActivityCreated(Bundle savedInstanceState) {
+ getLoaderManager().initLoader(DETAIL_LOADER, null, this);
+ if (savedInstanceState != null) {
+ mLocation = savedInstanceState.getString(LOCATION_KEY);
+ }
+ super.onActivityCreated(savedInstanceState);
+ }
+ @Override
+ public Loader<Cursor> onCreateLoader(int id, Bundle args) {
+ Log.v(LOG_TAG, "In onCreateLoader");
+ Intent intent = getActivity().getIntent();
+ if (intent == null || !intent.hasExtra(DATE_KEY)) {
+ return null;
+ }
+ String forecastDate = intent.getStringExtra(DATE_KEY);
+
+ // Sort order: Ascending, by date.
+ String sortOrder = WeatherContract.WeatherEntry.COLUMN_DATETEXT + " ASC";
+
+ mLocation = Utility.getPreferredLocation(getActivity());
+ Uri weatherForLocationUri = WeatherContract.WeatherEntry.buildWeatherLocationWithDate(
+ mLocation, forecastDate);
+ Log.v(LOG_TAG, weatherForLocationUri.toString());
+
+ // Now create and return a CursorLoader that will take care of
+ // creating a Cursor for the data being displayed.
+ return new CursorLoader(
+ getActivity(),
+ weatherForLocationUri,
+ FORECAST_COLUMNS,
+ null,
+ null,
+ sortOrder
+ );
+ }
+
+ @Override
+ public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
+ Log.v(LOG_TAG, "In onLoadFinished");
+ if (!data.moveToFirst()) { return; }
+
+ String dateString = Utility.formatDate(
+ data.getString(data.getColumnIndex(WeatherEntry.COLUMN_DATETEXT)));
+ ((TextView) getView().findViewById(R.id.detail_date_textview))
+ .setText(dateString);
+
+ String weatherDescription =
+ data.getString(data.getColumnIndex(WeatherEntry.COLUMN_SHORT_DESC));
+ ((TextView) getView().findViewById(R.id.detail_forecast_textview))
+ .setText(weatherDescription);
+
+ boolean isMetric = Utility.isMetric(getActivity());
+
+ String high = Utility.formatTemperature(
+ data.getDouble(data.getColumnIndex(WeatherEntry.COLUMN_MAX_TEMP)), isMetric);
+ ((TextView) getView().findViewById(R.id.detail_high_textview)).setText(high);
+
+ String low = Utility.formatTemperature(
+ data.getDouble(data.getColumnIndex(WeatherEntry.COLUMN_MIN_TEMP)), isMetric);
+ ((TextView) getView().findViewById(R.id.detail_low_textview)).setText(low);
+
+ // We still need this for the share intent
+ mForecast = String.format("%s - %s - %s/%s", dateString, weatherDescription, high, low);
+
+ Log.v(LOG_TAG, "Forecast String: " + mForecast);
+
+ // If onCreateOptionsMenu has already happened, we need to update the share intent now.
+ if (mShareActionProvider != null) {
+ mShareActionProvider.setShareIntent(createShareForecastIntent());
}
- return super.onOptionsItemSelected(item);
}
+
+ @Override
+ public void onLoaderReset(Loader<Cursor> loader) { }
+
+
+
+// @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_show_location) {
+// SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(getActivity());
+// String location = preferences.getString(getString(R.string.pref_location_key),
+// getString(R.string.pref_location_default));
+// Intent intent = new Intent(Intent.ACTION_VIEW);
+// Uri geoLocation;
+// try {
+// geoLocation = Uri.parse("geo:0,0?q=" + URLEncoder.encode(location, "UTF-8"));
+// intent.setData(geoLocation);
+// } catch (UnsupportedEncodingException e) {
+// Log.e(LOG_TAG, "Error ", e);
+// }
+// if (intent.resolveActivity(getActivity().getPackageManager()) != null) {
+// startActivity(intent);
+// }
+//
+// }
+// return super.onOptionsItemSelected(item);
+// }
}
}
import android.support.v4.content.CursorLoader;
import android.support.v4.content.Loader;
import android.support.v4.widget.SimpleCursorAdapter;
-import android.util.Log;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.ListView;
+import android.widget.TextView;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import uk.me.njae.sunshine.data.WeatherContract.LocationEntry;
import uk.me.njae.sunshine.data.WeatherContract.WeatherEntry;
+import static android.util.Log.e;
+
/**
* A placeholder fragment containing a simple view.
*/
geoLocation = Uri.parse("geo:0,0?q=" + URLEncoder.encode(location, "UTF-8"));
intent.setData(geoLocation);
} catch (UnsupportedEncodingException e) {
- Log.e(LOG_TAG, "Error ", e);
+ e(LOG_TAG, "Error ", e);
}
if (intent.resolveActivity(getActivity().getPackageManager()) != null) {
startActivity(intent);
0
);
+ mForecastAdapter.setViewBinder(new SimpleCursorAdapter.ViewBinder() {
+ @Override
+ public boolean setViewValue(View view, Cursor cursor, int columnIndex) {
+ boolean isMetric = Utility.isMetric(getActivity());
+ switch (columnIndex) {
+ case COL_WEATHER_MAX_TEMP:
+ case COL_WEATHER_MIN_TEMP: {
+ // we have to do some formatting and possibly a conversion
+ ((TextView) view).setText(Utility.formatTemperature(
+ cursor.getDouble(columnIndex), isMetric));
+ return true;
+ }
+ case COL_WEATHER_DATE: {
+ String dateString = cursor.getString(columnIndex);
+ TextView dateView = (TextView) view;
+ dateView.setText(Utility.formatDate(dateString));
+ return true;
+ }
+ }
+ return false;
+ }
+ });
+
View rootView = inflater.inflate(R.layout.fragment_main, container, false);
ListView listView = (ListView) rootView.findViewById(R.id.listview_forecast);
listView.setAdapter(mForecastAdapter);
+
listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> adapterView, View view, int position, long l) {
- // String forecast = mForecastAdapter.getItem(position);
- Intent intent = new Intent(getActivity(), DetailActivity.class)
- .putExtra(Intent.EXTRA_TEXT, "placeholder");
- startActivity(intent);
+ Cursor cursor = mForecastAdapter.getCursor();
+ if (cursor != null && cursor.moveToPosition(position)) {
+ Intent intent = new Intent(getActivity(), DetailActivity.class)
+ .putExtra(DetailActivity.DATE_KEY, cursor.getString(COL_WEATHER_DATE));
+ startActivity(intent);
+ }
}
});
return rootView;
}
- public void onStart() {
- super.onStart();
- updateWeather();
+ @Override
+ public void onResume() {
+ super.onResume();
+ if (mLocation != null && !mLocation.equals(Utility.getPreferredLocation(getActivity()))) {
+ getLoaderManager().restartLoader(FORECAST_LOADER, null, this);
+ }
}
@Override
import android.preference.PreferenceActivity;
import android.preference.PreferenceManager;
+import uk.me.njae.sunshine.data.WeatherContract;
+
/**
* A {@link PreferenceActivity} that presents a set of application settings.
* <p>
public class SettingsActivity extends PreferenceActivity
implements Preference.OnPreferenceChangeListener {
+ // since we use the preference change initially to populate the summary
+ // field, we'll ignore that change at start of the activity
+ boolean mBindingPreference;
+
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
* is changed.)
*/
private void bindPreferenceSummaryToValue(Preference preference) {
+ mBindingPreference = true;
+
// Set the listener to watch for value changes.
preference.setOnPreferenceChangeListener(this);
PreferenceManager
.getDefaultSharedPreferences(preference.getContext())
.getString(preference.getKey(), ""));
+
+ mBindingPreference = false;
}
@Override
public boolean onPreferenceChange(Preference preference, Object value) {
String stringValue = value.toString().trim();
+ // are we starting the preference activity?
+ if ( !mBindingPreference ) {
+ if (preference.getKey().equals(getString(R.string.pref_location_key))) {
+ FetchWeatherTask weatherTask = new FetchWeatherTask(this);
+ String location = value.toString();
+ weatherTask.execute(location);
+ } else {
+ // notify code that weather may be impacted
+ getContentResolver().notifyChange(WeatherContract.WeatherEntry.CONTENT_URI, null);
+ }
+ }
+
if (preference instanceof ListPreference) {
// For list preferences, look up the correct display value in
// the preference's 'entries' list (since they have separate labels/values).
-<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
+<?xml version="1.0" encoding="utf-8"?>
+
+<LinearLayout
+ android:orientation="horizontal"
android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:paddingLeft="@dimen/activity_horizontal_margin"
- android:paddingRight="@dimen/activity_horizontal_margin"
- android:paddingTop="@dimen/activity_vertical_margin"
- android:paddingBottom="@dimen/activity_vertical_margin"
- tools:context="uk.me.njae.sunshine.DetailActivity$PlaceholderFragment">
-
- <TextView
- android:id="@+id/detail_text"
+ android:layout_height="wrap_content"
+ android:padding="16dp"
+ xmlns:android="http://schemas.android.com/apk/res/android">
+
+ <TextView xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="wrap_content"
+ android:layout_height="match_parent"
+ android:id="@+id/detail_date_textview" />
+
+ <TextView xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="wrap_content"
+ android:layout_height="match_parent"
+ android:text=" - " />
+
+ <TextView xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="wrap_content"
+ android:layout_height="match_parent"
+ android:id="@+id/detail_forecast_textview" />
+
+ <TextView xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="wrap_content"
+ android:layout_height="match_parent"
+ android:text=" - " />
+
+ <TextView xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="wrap_content"
+ android:layout_height="match_parent"
+ android:id="@+id/detail_high_textview" />
+
+ <TextView xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="wrap_content"
+ android:layout_height="match_parent"
+ android:text="/" />
+
+ <TextView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
- android:layout_height="wrap_content" />
+ android:layout_height="match_parent"
+ android:id="@+id/detail_low_textview" />
+
+</LinearLayout>
+
+
+
-</RelativeLayout>