package com.android.deskclock;
import android.app.Activity;
+import android.app.AlarmManager;
import android.app.AlertDialog;
+import android.app.PendingIntent;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.DialogInterface;
import android.os.PowerManager;
import android.provider.Settings;
import android.text.TextUtils;
+import android.text.format.DateFormat;
import android.util.DisplayMetrics;
import android.util.Log;
import android.view.ContextMenu.ContextMenuInfo;
import android.view.ContextMenu;
import android.view.LayoutInflater;
import android.view.Menu;
+import android.view.MenuInflater;
import android.view.MenuItem;
import android.view.View.OnClickListener;
import android.view.View.OnCreateContextMenuListener;
import java.io.IOException;
import java.io.InputStream;
-import java.text.DateFormat;
+import java.util.Calendar;
import java.util.Date;
import java.util.Locale;
import java.util.Random;
private static final String LOG_TAG = "DeskClock";
- // Intent used to start the music player.
- private static final String MUSIC_NOW_PLAYING = "com.android.music.PLAYBACK_VIEWER";
+ // Package ID of the music player.
+ private static final String MUSIC_PACKAGE_ID = "com.android.music";
+
+ // Alarm action for midnight (so we can update the date display).
+ private static final String ACTION_MIDNIGHT = "com.android.deskclock.MIDNIGHT";
// Interval between polls of the weather widget. Its refresh period is
// likely to be much longer (~3h), but we want to pick up any changes
// within 5 minutes.
- private final long FETCH_WEATHER_DELAY = 5 * 60 * 1000; // 5 min
+ private final long QUERY_WEATHER_DELAY = 5 * 60 * 1000; // 5 min
// Delay before engaging the burn-in protection mode (green-on-black).
- private final long SCREEN_SAVER_TIMEOUT = 10 * 60 * 1000; // 10 min
+ private final long SCREEN_SAVER_TIMEOUT = 5* 60 * 1000; // 10 min
// Repositioning delay in screen saver.
private final long SCREEN_SAVER_MOVE_DELAY = 60 * 1000; // 1 min
private final int SCREEN_SAVER_COLOR = 0xFF308030;
private final int SCREEN_SAVER_COLOR_DIM = 0xFF183018;
+ // Opacity of black layer between clock display and wallpaper.
+ private final float DIM_BEHIND_AMOUNT_NORMAL = 0.4f;
+ private final float DIM_BEHIND_AMOUNT_DIMMED = 0.7f; // higher contrast when display dimmed
+
// Internal message IDs.
- private final int FETCH_WEATHER_DATA_MSG = 0x1000;
+ private final int QUERY_WEATHER_DATA_MSG = 0x1000;
private final int UPDATE_WEATHER_DISPLAY_MSG = 0x1001;
private final int SCREEN_SAVER_TIMEOUT_MSG = 0x2000;
private final int SCREEN_SAVER_MOVE_MSG = 0x2001;
private static final String[] WEATHER_CONTENT_COLUMNS = new String[] {
"location",
"timestamp",
+ "temperature",
"highTemperature",
"lowTemperature",
"iconUrl",
"description",
};
+ private static final String ACTION_GENIE_REFRESH = "com.google.android.apps.genie.REFRESH";
+
// State variables follow.
private DigitalClock mTime;
private TextView mDate;
private TextView mNextAlarm = null;
private TextView mBatteryDisplay;
+ private TextView mWeatherCurrentTemperature;
private TextView mWeatherHighTemperature;
private TextView mWeatherLowTemperature;
private TextView mWeatherLocation;
private ImageView mWeatherIcon;
+ private String mWeatherCurrentTemperatureString;
private String mWeatherHighTemperatureString;
private String mWeatherLowTemperatureString;
private String mWeatherLocationString;
private boolean mDimmed = false;
private boolean mScreenSaverMode = false;
- private DateFormat mDateFormat;
+ private String mDateFormat;
private int mBatteryLevel = -1;
private boolean mPluggedIn = false;
- private int mIdleTimeoutEpoch = 0;
+ private boolean mInDock = false;
- private boolean mWeatherFetchScheduled = false;
+ private int mIdleTimeoutEpoch = 0;
private Random mRNG;
+ private PendingIntent mMidnightIntent;
+
private final BroadcastReceiver mIntentReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
private final Handler mHandy = new Handler() {
@Override
public void handleMessage(Message m) {
- if (m.what == FETCH_WEATHER_DATA_MSG) {
- if (!mWeatherFetchScheduled) return;
- mWeatherFetchScheduled = false;
- new Thread() { public void run() { fetchWeatherData(); } }.start();
- scheduleWeatherFetchDelayed(FETCH_WEATHER_DELAY);
+ if (m.what == QUERY_WEATHER_DATA_MSG) {
+ new Thread() { public void run() { queryWeatherData(); } }.start();
+ scheduleWeatherQueryDelayed(QUERY_WEATHER_DELAY);
} else if (m.what == UPDATE_WEATHER_DISPLAY_MSG) {
updateWeatherDisplay();
} else if (m.what == SCREEN_SAVER_TIMEOUT_MSG) {
private void moveScreenSaverTo(int x, int y) {
if (!mScreenSaverMode) return;
- final View time_date = findViewById(R.id.time_date);
+ final View saver_view = findViewById(R.id.saver_view);
DisplayMetrics metrics = new DisplayMetrics();
getWindowManager().getDefaultDisplay().getMetrics(metrics);
if (x < 0 || y < 0) {
- int myWidth = time_date.getMeasuredWidth();
- int myHeight = time_date.getMeasuredHeight();
+ int myWidth = saver_view.getMeasuredWidth();
+ int myHeight = saver_view.getMeasuredHeight();
x = (int)(mRNG.nextFloat()*(metrics.widthPixels - myWidth));
y = (int)(mRNG.nextFloat()*(metrics.heightPixels - myHeight));
}
if (DEBUG) Log.d(LOG_TAG, String.format("screen saver: %d: jumping to (%d,%d)",
System.currentTimeMillis(), x, y));
- time_date.setLayoutParams(new AbsoluteLayout.LayoutParams(
+ saver_view.setLayoutParams(new AbsoluteLayout.LayoutParams(
ViewGroup.LayoutParams.WRAP_CONTENT,
ViewGroup.LayoutParams.WRAP_CONTENT,
x,
mScreenSaverMode = false;
initViews();
doDim(false); // restores previous dim mode
+ // policy: update weather info when returning from screen saver
+ if (mPluggedIn) requestWeatherDataFetch();
refreshAll();
}
mTime = (DigitalClock) findViewById(R.id.time);
mDate = (TextView) findViewById(R.id.date);
+ mNextAlarm = (TextView) findViewById(R.id.nextAlarm);
final int color = mDimmed ? SCREEN_SAVER_COLOR_DIM : SCREEN_SAVER_COLOR;
((TextView)findViewById(R.id.timeDisplay)).setTextColor(color);
((TextView)findViewById(R.id.am_pm)).setTextColor(color);
mDate.setTextColor(color);
+ mNextAlarm.setTextColor(color);
+ mNextAlarm.setCompoundDrawablesWithIntrinsicBounds(
+ getResources().getDrawable(mDimmed
+ ? R.drawable.ic_lock_idle_alarm_saver_dim
+ : R.drawable.ic_lock_idle_alarm_saver),
+ null, null, null);
mBatteryDisplay =
- mNextAlarm =
+ mWeatherCurrentTemperature =
mWeatherHighTemperature =
mWeatherLowTemperature =
mWeatherLocation = null;
mWeatherIcon = null;
refreshDate();
+ refreshAlarm();
moveScreenSaverTo(oldLoc[0], oldLoc[1]);
}
restoreScreen();
}
+ // Tell the Genie widget to load new data from the network.
+ private void requestWeatherDataFetch() {
+ if (DEBUG) Log.d(LOG_TAG, "forcing the Genie widget to update weather now...");
+ sendBroadcast(new Intent(ACTION_GENIE_REFRESH).putExtra("requestWeather", true));
+ // update the display with any new data
+ scheduleWeatherQueryDelayed(5000);
+ }
+
private boolean supportsWeather() {
return (mGenieResources != null);
}
- private void scheduleWeatherFetchDelayed(long delay) {
- if (mWeatherFetchScheduled) return;
+ private void scheduleWeatherQueryDelayed(long delay) {
+ // cancel any existing scheduled queries
+ unscheduleWeatherQuery();
if (DEBUG) Log.d(LOG_TAG, "scheduling weather fetch message for " + delay + "ms from now");
- mWeatherFetchScheduled = true;
-
- mHandy.sendEmptyMessageDelayed(FETCH_WEATHER_DATA_MSG, delay);
+ mHandy.sendEmptyMessageDelayed(QUERY_WEATHER_DATA_MSG, delay);
}
- private void unscheduleWeatherFetch() {
- mWeatherFetchScheduled = false;
+ private void unscheduleWeatherQuery() {
+ mHandy.removeMessages(QUERY_WEATHER_DATA_MSG);
}
- private static final boolean sCelsius;
- static {
- String cc = Locale.getDefault().getCountry().toLowerCase();
- sCelsius = !("us".equals(cc) || "bz".equals(cc) || "jm".equals(cc));
- }
-
- private static int celsiusToLocal(int tempC) {
- return sCelsius ? tempC : (int) Math.round(tempC * 1.8f + 32);
- }
-
- private void fetchWeatherData() {
+ private void queryWeatherData() {
// if we couldn't load the weather widget's resources, we simply
// assume it's not present on the device.
if (mGenieResources == null) return;
mWeatherIconDrawable = mGenieResources.getDrawable(cur.getInt(
cur.getColumnIndexOrThrow("iconResId")));
+ mWeatherCurrentTemperatureString = String.format("%d\u00b0",
+ (cur.getInt(cur.getColumnIndexOrThrow("temperature"))));
mWeatherHighTemperatureString = String.format("%d\u00b0",
- celsiusToLocal(cur.getInt(cur.getColumnIndexOrThrow("highTemperature"))));
+ (cur.getInt(cur.getColumnIndexOrThrow("highTemperature"))));
mWeatherLowTemperatureString = String.format("%d\u00b0",
- celsiusToLocal(cur.getInt(cur.getColumnIndexOrThrow("lowTemperature"))));
+ (cur.getInt(cur.getColumnIndexOrThrow("lowTemperature"))));
mWeatherLocationString = cur.getString(
cur.getColumnIndexOrThrow("location"));
} else {
mWeatherIconDrawable = null;
mWeatherHighTemperatureString = "";
mWeatherLowTemperatureString = "";
- mWeatherLocationString = "Weather data unavailable."; // TODO: internationalize
+ mWeatherLocationString = getString(R.string.weather_fetch_failure);
}
mHandy.sendEmptyMessage(UPDATE_WEATHER_DISPLAY_MSG);
private void refreshWeather() {
if (supportsWeather())
- scheduleWeatherFetchDelayed(0);
+ scheduleWeatherQueryDelayed(0);
updateWeatherDisplay(); // in case we have it cached
}
private void updateWeatherDisplay() {
- if (mWeatherHighTemperature == null) return;
+ if (mWeatherCurrentTemperature == null) return;
+ mWeatherCurrentTemperature.setText(mWeatherCurrentTemperatureString);
mWeatherHighTemperature.setText(mWeatherHighTemperatureString);
mWeatherLowTemperature.setText(mWeatherLowTemperatureString);
mWeatherLocation.setText(mWeatherLocationString);
final boolean pluggedIn = (plugStatus == BATTERY_STATUS_CHARGING || plugStatus == BATTERY_STATUS_FULL);
if (pluggedIn != mPluggedIn) {
setWakeLock(pluggedIn);
+
+ if (pluggedIn) {
+ // policy: update weather info when attaching to power
+ requestWeatherDataFetch();
+ }
}
if (pluggedIn != mPluggedIn || batteryLevel != mBatteryLevel) {
mBatteryLevel = batteryLevel;
}
private void refreshDate() {
- mDate.setText(mDateFormat.format(new Date()));
+ final Date now = new Date();
+ if (DEBUG) Log.d(LOG_TAG, "refreshing date..." + now);
+ mDate.setText(DateFormat.format(mDateFormat, now));
}
private void refreshAlarm() {
Window win = getWindow();
WindowManager.LayoutParams winParams = win.getAttributes();
- // secret!
winParams.flags |= (WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN);
winParams.flags |= (WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS);
if (mDimmed) {
winParams.flags |= WindowManager.LayoutParams.FLAG_FULLSCREEN;
- winParams.dimAmount = 0.67f; // pump up contrast in dim mode
+ winParams.dimAmount = DIM_BEHIND_AMOUNT_DIMMED;
// show the window tint
tintView.startAnimation(AnimationUtils.loadAnimation(this,
: R.anim.dim_instant));
} else {
winParams.flags &= (~WindowManager.LayoutParams.FLAG_FULLSCREEN);
- winParams.dimAmount = 0.5f; // lower contrast in normal mode
+ winParams.dimAmount = DIM_BEHIND_AMOUNT_NORMAL;
// hide the window tint
tintView.startAnimation(AnimationUtils.loadAnimation(this,
// reload the date format in case the user has changed settings
// recently
- mDateFormat = java.text.DateFormat.getDateInstance(java.text.DateFormat.FULL);
+ mDateFormat = getString(com.android.internal.R.string.full_wday_month_day_no_year);
IntentFilter filter = new IntentFilter();
filter.addAction(Intent.ACTION_DATE_CHANGED);
filter.addAction(Intent.ACTION_BATTERY_CHANGED);
+ filter.addAction(ACTION_MIDNIGHT);
+
+ Calendar today = Calendar.getInstance();
+ today.add(Calendar.DATE, 1);
+ mMidnightIntent = PendingIntent.getBroadcast(this, 0, new Intent(ACTION_MIDNIGHT), 0);
+ AlarmManager am = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
+ am.setRepeating(AlarmManager.RTC, today.getTimeInMillis(), AlarmManager.INTERVAL_DAY, mMidnightIntent);
registerReceiver(mIntentReceiver, filter);
+ // un-dim when resuming
+ mDimmed = false;
doDim(false);
- restoreScreen();
+
+ restoreScreen(); // disable screen saver
refreshAll(); // will schedule periodic weather fetch
setWakeLock(mPluggedIn);
mHandy.sendMessageDelayed(
Message.obtain(mHandy, SCREEN_SAVER_TIMEOUT_MSG, mIdleTimeoutEpoch, 0),
SCREEN_SAVER_TIMEOUT);
+
+ final boolean launchedFromDock
+ = getIntent().hasCategory(Intent.CATEGORY_DESK_DOCK);
+
+ if (supportsWeather() && launchedFromDock && !mInDock) {
+ // policy: fetch weather if launched via dock connection
+ if (DEBUG) Log.d(LOG_TAG, "Device now docked; forcing weather to refresh right now");
+ requestWeatherDataFetch();
+ }
+
+ mInDock = launchedFromDock;
}
@Override
public void onPause() {
if (DEBUG) Log.d(LOG_TAG, "onPause");
- // Turn off the screen saver.
+ // Turn off the screen saver. (But don't un-dim.)
restoreScreen();
- // Avoid situations where the user launches Alarm Clock and is
- // surprised to find it in dim mode (because it was last used in dim
- // mode, but that last use is long in the past).
- mDimmed = false;
-
// Other things we don't want to be doing in the background.
unregisterReceiver(mIntentReceiver);
- unscheduleWeatherFetch();
+ AlarmManager am = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
+ am.cancel(mMidnightIntent);
+ unscheduleWeatherQuery();
super.onPause();
}
+ @Override
+ public void onStop() {
+ if (DEBUG) Log.d(LOG_TAG, "onStop");
+
+ // Avoid situations where the user launches Alarm Clock and is
+ // surprised to find it in dim mode (because it was last used in dim
+ // mode, but that last use is long in the past).
+ mDimmed = false;
+
+ super.onStop();
+ }
private void initViews() {
// give up any internal focus before we switch layouts
mTime.getRootView().requestFocus();
+ mWeatherCurrentTemperature = (TextView) findViewById(R.id.weather_temperature);
mWeatherHighTemperature = (TextView) findViewById(R.id.weather_high_temperature);
mWeatherLowTemperature = (TextView) findViewById(R.id.weather_low_temperature);
mWeatherLocation = (TextView) findViewById(R.id.weather_location);
mWeatherIcon = (ImageView) findViewById(R.id.weather_icon);
- mNextAlarm = (TextView) findViewById(R.id.nextAlarm);
-
- final ImageButton alarmButton = (ImageButton) findViewById(R.id.alarm_button);
- alarmButton.setOnClickListener(new View.OnClickListener() {
+ final View.OnClickListener alarmClickListener = new View.OnClickListener() {
public void onClick(View v) {
startActivity(new Intent(DeskClock.this, AlarmClock.class));
}
- });
+ };
+
+ mNextAlarm = (TextView) findViewById(R.id.nextAlarm);
+ mNextAlarm.setOnClickListener(alarmClickListener);
+
+ final ImageButton alarmButton = (ImageButton) findViewById(R.id.alarm_button);
+ alarmButton.setOnClickListener(alarmClickListener);
final ImageButton galleryButton = (ImageButton) findViewById(R.id.gallery_button);
galleryButton.setOnClickListener(new View.OnClickListener() {
startActivity(new Intent(
Intent.ACTION_VIEW,
android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI)
- .putExtra("slideshow", true));
+ .putExtra("slideshow", true)
+ .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK|Intent.FLAG_ACTIVITY_CLEAR_TOP));
} catch (android.content.ActivityNotFoundException e) {
Log.e(LOG_TAG, "Couldn't launch image browser", e);
}
musicButton.setOnClickListener(new View.OnClickListener() {
public void onClick(View v) {
try {
- startActivity(new Intent(MUSIC_NOW_PLAYING));
+ Intent musicAppQuery = getPackageManager()
+ .getLaunchIntentForPackage(MUSIC_PACKAGE_ID)
+ .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK|Intent.FLAG_ACTIVITY_CLEAR_TOP);
+ if (musicAppQuery != null) {
+ startActivity(musicAppQuery);
+ }
} catch (android.content.ActivityNotFoundException e) {
Log.e(LOG_TAG, "Couldn't launch music browser", e);
}
public void onClick(View v) {
startActivity(
new Intent(Intent.ACTION_MAIN)
+ .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK|Intent.FLAG_ACTIVITY_CLEAR_TOP)
.addCategory(Intent.CATEGORY_HOME));
}
});
public void onClick(View v) {
if (!supportsWeather()) return;
- Intent genieAppQuery = getPackageManager().getLaunchIntentForPackage(GENIE_PACKAGE_ID);
+ Intent genieAppQuery = getPackageManager()
+ .getLaunchIntentForPackage(GENIE_PACKAGE_ID)
+ .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK|Intent.FLAG_ACTIVITY_CLEAR_TOP);
if (genieAppQuery != null) {
startActivity(genieAppQuery);
}
}
@Override
+ public boolean onOptionsItemSelected(MenuItem item) {
+ if (item.getItemId() == R.id.menu_item_alarms) {
+ startActivity(new Intent(DeskClock.this, AlarmClock.class));
+ return true;
+ } else if (item.getItemId() == R.id.menu_item_add_alarm) {
+ AlarmClock.addNewAlarm(this);
+ return true;
+ }
+ return false;
+ }
+
+ @Override
+ public boolean onCreateOptionsMenu(Menu menu) {
+ MenuInflater inflater = getMenuInflater();
+ inflater.inflate(R.menu.desk_clock_menu, menu);
+ return true;
+ }
+
+ @Override
protected void onCreate(Bundle icicle) {
super.onCreate(icicle);