AllInOneActivity.java revision 132eeb8acba46be5d0a3dc69b3a4dc0876430b08
1/* 2 * Copyright (C) 2010 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17package com.android.calendar; 18 19import static android.provider.CalendarContract.EXTRA_EVENT_ALL_DAY; 20import static android.provider.CalendarContract.EXTRA_EVENT_BEGIN_TIME; 21import static android.provider.CalendarContract.EXTRA_EVENT_END_TIME; 22import static android.provider.CalendarContract.Attendees.ATTENDEE_STATUS; 23import static com.android.calendar.CalendarController.EVENT_ATTENDEE_RESPONSE; 24 25import android.accounts.AccountManager; 26import android.accounts.AccountManagerCallback; 27import android.accounts.AccountManagerFuture; 28import android.accounts.AuthenticatorException; 29import android.accounts.OperationCanceledException; 30import android.animation.Animator; 31import android.animation.Animator.AnimatorListener; 32import android.animation.ObjectAnimator; 33import android.app.ActionBar; 34import android.app.ActionBar.Tab; 35import android.app.Activity; 36import android.app.Fragment; 37import android.app.FragmentManager; 38import android.app.FragmentTransaction; 39import android.content.AsyncQueryHandler; 40import android.content.BroadcastReceiver; 41import android.content.ContentResolver; 42import android.content.ContentUris; 43import android.content.Context; 44import android.content.Intent; 45import android.content.IntentFilter; 46import android.content.SharedPreferences; 47import android.content.SharedPreferences.OnSharedPreferenceChangeListener; 48import android.content.res.Configuration; 49import android.content.res.Resources; 50import android.database.ContentObserver; 51import android.database.Cursor; 52import android.graphics.drawable.LayerDrawable; 53import android.net.Uri; 54import android.os.Bundle; 55import android.os.Handler; 56import android.provider.CalendarContract; 57import android.provider.CalendarContract.Attendees; 58import android.provider.CalendarContract.Calendars; 59import android.provider.CalendarContract.Events; 60import android.text.TextUtils; 61import android.text.format.DateFormat; 62import android.text.format.DateUtils; 63import android.text.format.Time; 64import android.util.Log; 65import android.view.Menu; 66import android.view.MenuItem; 67import android.view.View; 68import android.view.accessibility.AccessibilityEvent; 69import android.widget.LinearLayout; 70import android.widget.RelativeLayout; 71import android.widget.RelativeLayout.LayoutParams; 72import android.widget.SearchView; 73import android.widget.SearchView.OnSuggestionListener; 74import android.widget.TextView; 75 76import com.android.calendar.CalendarController.EventHandler; 77import com.android.calendar.CalendarController.EventInfo; 78import com.android.calendar.CalendarController.EventType; 79import com.android.calendar.CalendarController.ViewType; 80import com.android.calendar.agenda.AgendaFragment; 81import com.android.calendar.month.MonthByWeekFragment; 82import com.android.calendar.selectcalendars.SelectVisibleCalendarsFragment; 83 84import java.io.IOException; 85import java.util.List; 86import java.util.Locale; 87import java.util.TimeZone; 88 89public class AllInOneActivity extends Activity implements EventHandler, 90 OnSharedPreferenceChangeListener, SearchView.OnQueryTextListener, ActionBar.TabListener, 91 ActionBar.OnNavigationListener, OnSuggestionListener { 92 private static final String TAG = "AllInOneActivity"; 93 private static final boolean DEBUG = false; 94 private static final String EVENT_INFO_FRAGMENT_TAG = "EventInfoFragment"; 95 private static final String BUNDLE_KEY_RESTORE_TIME = "key_restore_time"; 96 private static final String BUNDLE_KEY_EVENT_ID = "key_event_id"; 97 private static final String BUNDLE_KEY_RESTORE_VIEW = "key_restore_view"; 98 private static final String BUNDLE_KEY_CHECK_ACCOUNTS = "key_check_for_accounts"; 99 private static final int HANDLER_KEY = 0; 100 private static float mScale = 0; 101 102 // Indices of buttons for the drop down menu (tabs replacement) 103 // Must match the strings in the array buttons_list in arrays.xml and the 104 // OnNavigationListener 105 private static final int BUTTON_DAY_INDEX = 0; 106 private static final int BUTTON_WEEK_INDEX = 1; 107 private static final int BUTTON_MONTH_INDEX = 2; 108 private static final int BUTTON_AGENDA_INDEX = 3; 109 110 private CalendarController mController; 111 private static boolean mIsMultipane; 112 private static boolean mIsTabletConfig; 113 private static boolean mShowAgendaWithMonth; 114 private static boolean mShowEventDetailsWithAgenda; 115 private boolean mOnSaveInstanceStateCalled = false; 116 private boolean mBackToPreviousView = false; 117 private ContentResolver mContentResolver; 118 private int mPreviousView; 119 private int mCurrentView; 120 private boolean mPaused = true; 121 private boolean mUpdateOnResume = false; 122 private boolean mHideControls = false; 123 private boolean mShowSideViews = true; 124 private boolean mShowWeekNum = false; 125 private TextView mHomeTime; 126 private TextView mDateRange; 127 private TextView mWeekTextView; 128 private View mMiniMonth; 129 private View mCalendarsList; 130 private View mMiniMonthContainer; 131 private View mSecondaryPane; 132 private String mTimeZone; 133 private boolean mShowCalendarControls; 134 private boolean mShowEventInfoFullScreenAgenda; 135 private boolean mShowEventInfoFullScreen; 136 private int mWeekNum; 137 private int mCalendarControlsAnimationTime; 138 private int mControlsAnimateWidth; 139 private int mControlsAnimateHeight; 140 141 private long mViewEventId = -1; 142 private long mIntentEventStartMillis = -1; 143 private long mIntentEventEndMillis = -1; 144 private int mIntentAttendeeResponse = Attendees.ATTENDEE_STATUS_NONE; 145 private boolean mIntentAllDay = false; 146 147 // Action bar and Navigation bar (left side of Action bar) 148 private ActionBar mActionBar; 149 private ActionBar.Tab mDayTab; 150 private ActionBar.Tab mWeekTab; 151 private ActionBar.Tab mMonthTab; 152 private ActionBar.Tab mAgendaTab; 153 private SearchView mSearchView; 154 private MenuItem mSearchMenu; 155 private MenuItem mControlsMenu; 156 private Menu mOptionsMenu; 157 private CalendarViewAdapter mActionBarMenuSpinnerAdapter; 158 private QueryHandler mHandler; 159 private boolean mCheckForAccounts = true; 160 161 private String mHideString; 162 private String mShowString; 163 164 DayOfMonthDrawable mDayOfMonthIcon; 165 166 int mOrientation; 167 168 // Params for animating the controls on the right 169 private LayoutParams mControlsParams; 170 private LinearLayout.LayoutParams mVerticalControlsParams; 171 172 private final AnimatorListener mSlideAnimationDoneListener = new AnimatorListener() { 173 174 @Override 175 public void onAnimationCancel(Animator animation) { 176 } 177 178 @Override 179 public void onAnimationEnd(android.animation.Animator animation) { 180 int visibility = mShowSideViews ? View.VISIBLE : View.GONE; 181 mMiniMonth.setVisibility(visibility); 182 mCalendarsList.setVisibility(visibility); 183 mMiniMonthContainer.setVisibility(visibility); 184 } 185 186 @Override 187 public void onAnimationRepeat(android.animation.Animator animation) { 188 } 189 190 @Override 191 public void onAnimationStart(android.animation.Animator animation) { 192 } 193 }; 194 195 private class QueryHandler extends AsyncQueryHandler { 196 public QueryHandler(ContentResolver cr) { 197 super(cr); 198 } 199 200 @Override 201 protected void onQueryComplete(int token, Object cookie, Cursor cursor) { 202 mCheckForAccounts = false; 203 try { 204 // If the query didn't return a cursor for some reason return 205 if (cursor == null || cursor.getCount() > 0 || isFinishing()) { 206 return; 207 } 208 } finally { 209 if (cursor != null) { 210 cursor.close(); 211 } 212 } 213 214 Bundle options = new Bundle(); 215 options.putCharSequence("introMessage", 216 getResources().getString(R.string.create_an_account_desc)); 217 options.putBoolean("allowSkip", true); 218 219 AccountManager am = AccountManager.get(AllInOneActivity.this); 220 am.addAccount("com.google", CalendarContract.AUTHORITY, null, options, 221 AllInOneActivity.this, 222 new AccountManagerCallback<Bundle>() { 223 @Override 224 public void run(AccountManagerFuture<Bundle> future) { 225 if (future.isCancelled()) { 226 return; 227 } 228 try { 229 Bundle result = future.getResult(); 230 boolean setupSkipped = result.getBoolean("setupSkipped"); 231 232 if (setupSkipped) { 233 Utils.setSharedPreference(AllInOneActivity.this, 234 GeneralPreferences.KEY_SKIP_SETUP, true); 235 } 236 237 } catch (OperationCanceledException ignore) { 238 // The account creation process was canceled 239 } catch (IOException ignore) { 240 } catch (AuthenticatorException ignore) { 241 } 242 } 243 }, null); 244 } 245 } 246 247 private final Runnable mHomeTimeUpdater = new Runnable() { 248 @Override 249 public void run() { 250 mTimeZone = Utils.getTimeZone(AllInOneActivity.this, mHomeTimeUpdater); 251 updateSecondaryTitleFields(-1); 252 AllInOneActivity.this.invalidateOptionsMenu(); 253 Utils.setMidnightUpdater(mHandler, mTimeChangesUpdater, mTimeZone); 254 } 255 }; 256 257 // runs every midnight/time changes and refreshes the today icon 258 private final Runnable mTimeChangesUpdater = new Runnable() { 259 @Override 260 public void run() { 261 mTimeZone = Utils.getTimeZone(AllInOneActivity.this, mHomeTimeUpdater); 262 AllInOneActivity.this.invalidateOptionsMenu(); 263 Utils.setMidnightUpdater(mHandler, mTimeChangesUpdater, mTimeZone); 264 } 265 }; 266 267 268 // Create an observer so that we can update the views whenever a 269 // Calendar event changes. 270 private final ContentObserver mObserver = new ContentObserver(new Handler()) { 271 @Override 272 public boolean deliverSelfNotifications() { 273 return true; 274 } 275 276 @Override 277 public void onChange(boolean selfChange) { 278 eventsChanged(); 279 } 280 }; 281 282 BroadcastReceiver mCalIntentReceiver; 283 284 @Override 285 protected void onNewIntent(Intent intent) { 286 String action = intent.getAction(); 287 if (DEBUG) 288 Log.d(TAG, "New intent received " + intent.toString()); 289 // Don't change the date if we're just returning to the app's home 290 if (Intent.ACTION_VIEW.equals(action) 291 && !intent.getBooleanExtra(Utils.INTENT_KEY_HOME, false)) { 292 long millis = parseViewAction(intent); 293 if (millis == -1) { 294 millis = Utils.timeFromIntentInMillis(intent); 295 } 296 if (millis != -1 && mViewEventId == -1 && mController != null) { 297 Time time = new Time(mTimeZone); 298 time.set(millis); 299 time.normalize(true); 300 mController.sendEvent(this, EventType.GO_TO, time, time, -1, ViewType.CURRENT); 301 } 302 } 303 } 304 305 @Override 306 protected void onCreate(Bundle icicle) { 307 if (Utils.getSharedPreference(this, OtherPreferences.KEY_OTHER_1, false)) { 308 setTheme(R.style.CalendarTheme_WithActionBarWallpaper); 309 } 310 super.onCreate(icicle); 311 312 if (icicle != null && icicle.containsKey(BUNDLE_KEY_CHECK_ACCOUNTS)) { 313 mCheckForAccounts = icicle.getBoolean(BUNDLE_KEY_CHECK_ACCOUNTS); 314 } 315 // Launch add google account if this is first time and there are no 316 // accounts yet 317 if (mCheckForAccounts 318 && !Utils.getSharedPreference(this, GeneralPreferences.KEY_SKIP_SETUP, false)) { 319 320 mHandler = new QueryHandler(this.getContentResolver()); 321 mHandler.startQuery(0, null, Calendars.CONTENT_URI, new String[] { 322 Calendars._ID 323 }, null, null /* selection args */, null /* sort order */); 324 } 325 326 // This needs to be created before setContentView 327 mController = CalendarController.getInstance(this); 328 329 330 // Get time from intent or icicle 331 long timeMillis = -1; 332 int viewType = -1; 333 final Intent intent = getIntent(); 334 if (icicle != null) { 335 timeMillis = icicle.getLong(BUNDLE_KEY_RESTORE_TIME); 336 viewType = icicle.getInt(BUNDLE_KEY_RESTORE_VIEW, -1); 337 } else { 338 String action = intent.getAction(); 339 if (Intent.ACTION_VIEW.equals(action)) { 340 // Open EventInfo later 341 timeMillis = parseViewAction(intent); 342 } 343 344 if (timeMillis == -1) { 345 timeMillis = Utils.timeFromIntentInMillis(intent); 346 } 347 } 348 349 if (viewType == -1) { 350 viewType = Utils.getViewTypeFromIntentAndSharedPref(this); 351 } 352 mTimeZone = Utils.getTimeZone(this, mHomeTimeUpdater); 353 Time t = new Time(mTimeZone); 354 t.set(timeMillis); 355 356 if (DEBUG) { 357 if (icicle != null && intent != null) { 358 Log.d(TAG, "both, icicle:" + icicle.toString() + " intent:" + intent.toString()); 359 } else { 360 Log.d(TAG, "not both, icicle:" + icicle + " intent:" + intent); 361 } 362 } 363 364 Resources res = getResources(); 365 mHideString = res.getString(R.string.hide_controls); 366 mShowString = res.getString(R.string.show_controls); 367 mOrientation = res.getConfiguration().orientation; 368 if (mOrientation == Configuration.ORIENTATION_LANDSCAPE) { 369 mControlsAnimateWidth = (int)res.getDimension(R.dimen.calendar_controls_width); 370 if (mControlsParams == null) { 371 mControlsParams = new LayoutParams(mControlsAnimateWidth, 0); 372 } 373 mControlsParams.addRule(RelativeLayout.ALIGN_PARENT_RIGHT); 374 } else { 375 // Make sure width is in between allowed min and max width values 376 mControlsAnimateWidth = Math.max(res.getDisplayMetrics().widthPixels * 45 / 100, 377 (int)res.getDimension(R.dimen.min_portrait_calendar_controls_width)); 378 mControlsAnimateWidth = Math.min(mControlsAnimateWidth, 379 (int)res.getDimension(R.dimen.max_portrait_calendar_controls_width)); 380 } 381 382 mControlsAnimateHeight = (int)res.getDimension(R.dimen.calendar_controls_height); 383 384 mHideControls = !Utils.getSharedPreference( 385 this, GeneralPreferences.KEY_SHOW_CONTROLS, true); 386 mIsMultipane = Utils.getConfigBool(this, R.bool.multiple_pane_config); 387 mIsTabletConfig = Utils.getConfigBool(this, R.bool.tablet_config); 388 mShowAgendaWithMonth = Utils.getConfigBool(this, R.bool.show_agenda_with_month); 389 mShowCalendarControls = 390 Utils.getConfigBool(this, R.bool.show_calendar_controls); 391 mShowEventDetailsWithAgenda = 392 Utils.getConfigBool(this, R.bool.show_event_details_with_agenda); 393 mShowEventInfoFullScreenAgenda = 394 Utils.getConfigBool(this, R.bool.agenda_show_event_info_full_screen); 395 mShowEventInfoFullScreen = 396 Utils.getConfigBool(this, R.bool.show_event_info_full_screen); 397 mCalendarControlsAnimationTime = res.getInteger(R.integer.calendar_controls_animation_time); 398 Utils.setAllowWeekForDetailView(mIsMultipane); 399 400 // setContentView must be called before configureActionBar 401 setContentView(R.layout.all_in_one); 402 403 if (mIsTabletConfig) { 404 mDateRange = (TextView) findViewById(R.id.date_bar); 405 mWeekTextView = (TextView) findViewById(R.id.week_num); 406 } else { 407 mDateRange = (TextView) getLayoutInflater().inflate(R.layout.date_range_title, null); 408 } 409 410 // configureActionBar auto-selects the first tab you add, so we need to 411 // call it before we set up our own fragments to make sure it doesn't 412 // overwrite us 413 configureActionBar(viewType); 414 415 mHomeTime = (TextView) findViewById(R.id.home_time); 416 mMiniMonth = findViewById(R.id.mini_month); 417 if (mIsTabletConfig && mOrientation == Configuration.ORIENTATION_PORTRAIT) { 418 mMiniMonth.setLayoutParams(new LinearLayout.LayoutParams(mControlsAnimateWidth, 419 mControlsAnimateHeight)); 420 } 421 mCalendarsList = findViewById(R.id.calendar_list); 422 mMiniMonthContainer = findViewById(R.id.mini_month_container); 423 mSecondaryPane = findViewById(R.id.secondary_pane); 424 425 // Must register as the first activity because this activity can modify 426 // the list of event handlers in it's handle method. This affects who 427 // the rest of the handlers the controller dispatches to are. 428 mController.registerFirstEventHandler(HANDLER_KEY, this); 429 430 initFragments(timeMillis, viewType, icicle); 431 432 // Listen for changes that would require this to be refreshed 433 SharedPreferences prefs = GeneralPreferences.getSharedPreferences(this); 434 prefs.registerOnSharedPreferenceChangeListener(this); 435 436 mContentResolver = getContentResolver(); 437 } 438 439 private long parseViewAction(final Intent intent) { 440 long timeMillis = -1; 441 Uri data = intent.getData(); 442 if (data != null && data.isHierarchical()) { 443 List<String> path = data.getPathSegments(); 444 if (path.size() == 2 && path.get(0).equals("events")) { 445 try { 446 mViewEventId = Long.valueOf(data.getLastPathSegment()); 447 if (mViewEventId != -1) { 448 mIntentEventStartMillis = intent.getLongExtra(EXTRA_EVENT_BEGIN_TIME, 0); 449 mIntentEventEndMillis = intent.getLongExtra(EXTRA_EVENT_END_TIME, 0); 450 mIntentAttendeeResponse = intent.getIntExtra( 451 ATTENDEE_STATUS, Attendees.ATTENDEE_STATUS_NONE); 452 mIntentAllDay = intent.getBooleanExtra(EXTRA_EVENT_ALL_DAY, false); 453 timeMillis = mIntentEventStartMillis; 454 } 455 } catch (NumberFormatException e) { 456 // Ignore if mViewEventId can't be parsed 457 } 458 } 459 } 460 return timeMillis; 461 } 462 463 private void configureActionBar(int viewType) { 464 createButtonsSpinner(viewType, mIsTabletConfig); 465 if (mIsMultipane) { 466 mActionBar.setDisplayOptions( 467 ActionBar.DISPLAY_SHOW_CUSTOM | ActionBar.DISPLAY_SHOW_HOME); 468 } else { 469 mActionBar.setDisplayOptions(0); 470 } 471 } 472 473 private void createTabs() { 474 mActionBar = getActionBar(); 475 if (mActionBar == null) { 476 Log.w(TAG, "ActionBar is null."); 477 } else { 478 mActionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS); 479 mDayTab = mActionBar.newTab(); 480 mDayTab.setText(getString(R.string.day_view)); 481 mDayTab.setTabListener(this); 482 mActionBar.addTab(mDayTab); 483 mWeekTab = mActionBar.newTab(); 484 mWeekTab.setText(getString(R.string.week_view)); 485 mWeekTab.setTabListener(this); 486 mActionBar.addTab(mWeekTab); 487 mMonthTab = mActionBar.newTab(); 488 mMonthTab.setText(getString(R.string.month_view)); 489 mMonthTab.setTabListener(this); 490 mActionBar.addTab(mMonthTab); 491 mAgendaTab = mActionBar.newTab(); 492 mAgendaTab.setText(getString(R.string.agenda_view)); 493 mAgendaTab.setTabListener(this); 494 mActionBar.addTab(mAgendaTab); 495 } 496 } 497 498 private void createButtonsSpinner(int viewType, boolean tabletConfig) { 499 // If tablet configuration , show spinner with no dates 500 mActionBarMenuSpinnerAdapter = new CalendarViewAdapter (this, viewType, !tabletConfig); 501 mActionBar = getActionBar(); 502 mActionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_LIST); 503 mActionBar.setListNavigationCallbacks(mActionBarMenuSpinnerAdapter, this); 504 switch (viewType) { 505 case ViewType.AGENDA: 506 mActionBar.setSelectedNavigationItem(BUTTON_AGENDA_INDEX); 507 break; 508 case ViewType.DAY: 509 mActionBar.setSelectedNavigationItem(BUTTON_DAY_INDEX); 510 break; 511 case ViewType.WEEK: 512 mActionBar.setSelectedNavigationItem(BUTTON_WEEK_INDEX); 513 break; 514 case ViewType.MONTH: 515 mActionBar.setSelectedNavigationItem(BUTTON_MONTH_INDEX); 516 break; 517 default: 518 mActionBar.setSelectedNavigationItem(BUTTON_DAY_INDEX); 519 break; 520 } 521 } 522 // Clear buttons used in the agenda view 523 private void clearOptionsMenu() { 524 if (mOptionsMenu == null) { 525 return; 526 } 527 MenuItem cancelItem = mOptionsMenu.findItem(R.id.action_cancel); 528 if (cancelItem != null) { 529 cancelItem.setVisible(false); 530 } 531 } 532 533 @Override 534 protected void onResume() { 535 super.onResume(); 536 537 // Must register as the first activity because this activity can modify 538 // the list of event handlers in it's handle method. This affects who 539 // the rest of the handlers the controller dispatches to are. 540 mController.registerFirstEventHandler(HANDLER_KEY, this); 541 542 mOnSaveInstanceStateCalled = false; 543 mContentResolver.registerContentObserver(CalendarContract.Events.CONTENT_URI, 544 true, mObserver); 545 if (mUpdateOnResume) { 546 initFragments(mController.getTime(), mController.getViewType(), null); 547 mUpdateOnResume = false; 548 } 549 Time t = new Time(mTimeZone); 550 t.set(mController.getTime()); 551 mController.sendEvent(this, EventType.UPDATE_TITLE, t, t, -1, ViewType.CURRENT, 552 mController.getDateFlags(), null, null); 553 // Make sure the drop-down menu will get its date updated at midnight 554 if (mActionBarMenuSpinnerAdapter != null) { 555 mActionBarMenuSpinnerAdapter.refresh(this); 556 } 557 558 if (mControlsMenu != null) { 559 mControlsMenu.setTitle(mHideControls ? mShowString : mHideString); 560 } 561 mPaused = false; 562 563 if (mViewEventId != -1 && mIntentEventStartMillis != -1 && mIntentEventEndMillis != -1) { 564 long currentMillis = System.currentTimeMillis(); 565 long selectedTime = -1; 566 if (currentMillis > mIntentEventStartMillis && currentMillis < mIntentEventEndMillis) { 567 selectedTime = currentMillis; 568 } 569 mController.sendEventRelatedEventWithExtra(this, EventType.VIEW_EVENT, mViewEventId, 570 mIntentEventStartMillis, mIntentEventEndMillis, -1, -1, 571 EventInfo.buildViewExtraLong(mIntentAttendeeResponse,mIntentAllDay), 572 selectedTime); 573 mViewEventId = -1; 574 mIntentEventStartMillis = -1; 575 mIntentEventEndMillis = -1; 576 mIntentAllDay = false; 577 } 578 Utils.setMidnightUpdater(mHandler, mTimeChangesUpdater, mTimeZone); 579 // Make sure the today icon is up to date 580 invalidateOptionsMenu(); 581 582 mCalIntentReceiver = Utils.setTimeChangesReceiver(this, mTimeChangesUpdater); 583 } 584 585 @Override 586 protected void onPause() { 587 super.onPause(); 588 589 mController.deregisterEventHandler(HANDLER_KEY); 590 mPaused = true; 591 mHomeTime.removeCallbacks(mHomeTimeUpdater); 592 if (mActionBarMenuSpinnerAdapter != null) { 593 mActionBarMenuSpinnerAdapter.onPause(); 594 } 595 mContentResolver.unregisterContentObserver(mObserver); 596 if (isFinishing()) { 597 // Stop listening for changes that would require this to be refreshed 598 SharedPreferences prefs = GeneralPreferences.getSharedPreferences(this); 599 prefs.unregisterOnSharedPreferenceChangeListener(this); 600 } 601 // FRAG_TODO save highlighted days of the week; 602 if (mController.getViewType() != ViewType.EDIT) { 603 Utils.setDefaultView(this, mController.getViewType()); 604 } 605 Utils.resetMidnightUpdater(mHandler, mTimeChangesUpdater); 606 Utils.clearTimeChangesReceiver(this, mCalIntentReceiver); 607 } 608 609 @Override 610 protected void onUserLeaveHint() { 611 mController.sendEvent(this, EventType.USER_HOME, null, null, -1, ViewType.CURRENT); 612 super.onUserLeaveHint(); 613 } 614 615 @Override 616 public void onSaveInstanceState(Bundle outState) { 617 mOnSaveInstanceStateCalled = true; 618 super.onSaveInstanceState(outState); 619 outState.putLong(BUNDLE_KEY_RESTORE_TIME, mController.getTime()); 620 outState.putInt(BUNDLE_KEY_RESTORE_VIEW, mCurrentView); 621 if (mCurrentView == ViewType.EDIT) { 622 outState.putLong(BUNDLE_KEY_EVENT_ID, mController.getEventId()); 623 } else if (mCurrentView == ViewType.AGENDA) { 624 FragmentManager fm = getFragmentManager(); 625 Fragment f = fm.findFragmentById(R.id.main_pane); 626 if (f instanceof AgendaFragment) { 627 outState.putLong(BUNDLE_KEY_EVENT_ID, ((AgendaFragment)f).getLastShowEventId()); 628 } 629 } 630 outState.putBoolean(BUNDLE_KEY_CHECK_ACCOUNTS, mCheckForAccounts); 631 } 632 633 @Override 634 protected void onDestroy() { 635 super.onDestroy(); 636 637 SharedPreferences prefs = GeneralPreferences.getSharedPreferences(this); 638 prefs.unregisterOnSharedPreferenceChangeListener(this); 639 640 mController.deregisterAllEventHandlers(); 641 642 CalendarController.removeInstance(this); 643 } 644 645 private void initFragments(long timeMillis, int viewType, Bundle icicle) { 646 if (DEBUG) { 647 Log.d(TAG, "Initializing to " + timeMillis + " for view " + viewType); 648 } 649 FragmentTransaction ft = getFragmentManager().beginTransaction(); 650 651 if (mShowCalendarControls) { 652 Fragment miniMonthFrag = new MonthByWeekFragment(timeMillis, true); 653 ft.replace(R.id.mini_month, miniMonthFrag); 654 mController.registerEventHandler(R.id.mini_month, (EventHandler) miniMonthFrag); 655 656 Fragment selectCalendarsFrag = new SelectVisibleCalendarsFragment(); 657 ft.replace(R.id.calendar_list, selectCalendarsFrag); 658 mController.registerEventHandler( 659 R.id.calendar_list, (EventHandler) selectCalendarsFrag); 660 } 661 if (!mShowCalendarControls || viewType == ViewType.EDIT) { 662 mMiniMonth.setVisibility(View.GONE); 663 mCalendarsList.setVisibility(View.GONE); 664 } 665 666 EventInfo info = null; 667 if (viewType == ViewType.EDIT) { 668 mPreviousView = GeneralPreferences.getSharedPreferences(this).getInt( 669 GeneralPreferences.KEY_START_VIEW, GeneralPreferences.DEFAULT_START_VIEW); 670 671 long eventId = -1; 672 Intent intent = getIntent(); 673 Uri data = intent.getData(); 674 if (data != null) { 675 try { 676 eventId = Long.parseLong(data.getLastPathSegment()); 677 } catch (NumberFormatException e) { 678 if (DEBUG) { 679 Log.d(TAG, "Create new event"); 680 } 681 } 682 } else if (icicle != null && icicle.containsKey(BUNDLE_KEY_EVENT_ID)) { 683 eventId = icicle.getLong(BUNDLE_KEY_EVENT_ID); 684 } 685 686 long begin = intent.getLongExtra(EXTRA_EVENT_BEGIN_TIME, -1); 687 long end = intent.getLongExtra(EXTRA_EVENT_END_TIME, -1); 688 info = new EventInfo(); 689 if (end != -1) { 690 info.endTime = new Time(); 691 info.endTime.set(end); 692 } 693 if (begin != -1) { 694 info.startTime = new Time(); 695 info.startTime.set(begin); 696 } 697 info.id = eventId; 698 // We set the viewtype so if the user presses back when they are 699 // done editing the controller knows we were in the Edit Event 700 // screen. Likewise for eventId 701 mController.setViewType(viewType); 702 mController.setEventId(eventId); 703 } else { 704 mPreviousView = viewType; 705 } 706 707 setMainPane(ft, R.id.main_pane, viewType, timeMillis, true); 708 ft.commit(); // this needs to be after setMainPane() 709 710 Time t = new Time(mTimeZone); 711 t.set(timeMillis); 712 if (viewType == ViewType.AGENDA && icicle != null) { 713 mController.sendEvent(this, EventType.GO_TO, t, null, 714 icicle.getLong(BUNDLE_KEY_EVENT_ID, -1), viewType); 715 } else if (viewType != ViewType.EDIT) { 716 mController.sendEvent(this, EventType.GO_TO, t, null, -1, viewType); 717 } 718 } 719 720 @Override 721 public void onBackPressed() { 722 if (mCurrentView == ViewType.EDIT || mBackToPreviousView) { 723 mController.sendEvent(this, EventType.GO_TO, null, null, -1, mPreviousView); 724 } else { 725 super.onBackPressed(); 726 } 727 } 728 729 @Override 730 public boolean onCreateOptionsMenu(Menu menu) { 731 super.onCreateOptionsMenu(menu); 732 mOptionsMenu = menu; 733 getMenuInflater().inflate(R.menu.all_in_one_title_bar, menu); 734 735 mSearchMenu = menu.findItem(R.id.action_search); 736 mSearchView = (SearchView) mSearchMenu.getActionView(); 737 if (mSearchView != null) { 738 Utils.setUpSearchView(mSearchView, this); 739 mSearchView.setOnQueryTextListener(this); 740 mSearchView.setOnSuggestionListener(this); 741 } 742 743 // Hide the "show/hide controls" button if this is a phone 744 // or the view type is "Month" or "Agenda". 745 746 mControlsMenu = menu.findItem(R.id.action_hide_controls); 747 if (!mShowCalendarControls) { 748 if (mControlsMenu != null) { 749 mControlsMenu.setVisible(false); 750 mControlsMenu.setEnabled(false); 751 } 752 } else if (mControlsMenu != null && mController != null 753 && (mController.getViewType() == ViewType.MONTH || 754 mController.getViewType() == ViewType.AGENDA)) { 755 mControlsMenu.setVisible(false); 756 mControlsMenu.setEnabled(false); 757 } else if (mControlsMenu != null){ 758 mControlsMenu.setTitle(mHideControls ? mShowString : mHideString); 759 } 760 761 // replace the default top layer drawable of the today icon with a custom drawable 762 // that shows the day of the month of today 763 LayerDrawable icon = (LayerDrawable)menu.findItem(R.id.action_today).getIcon(); 764 Utils.setTodayIcon(icon, this, mTimeZone); 765 return true; 766 } 767 768 @Override 769 public boolean onOptionsItemSelected(MenuItem item) { 770 Time t = null; 771 int viewType = ViewType.CURRENT; 772 long extras = CalendarController.EXTRA_GOTO_TIME; 773 switch (item.getItemId()) { 774 case R.id.action_refresh: 775 mController.refreshCalendars(); 776 return true; 777 case R.id.action_today: 778 viewType = ViewType.CURRENT; 779 t = new Time(mTimeZone); 780 t.setToNow(); 781 extras |= CalendarController.EXTRA_GOTO_TODAY; 782 break; 783 case R.id.action_create_event: 784 t = new Time(); 785 t.set(mController.getTime()); 786 if (t.minute > 30) { 787 t.hour++; 788 t.minute = 0; 789 } else if (t.minute > 0 && t.minute < 30) { 790 t.minute = 30; 791 } 792 mController.sendEventRelatedEvent( 793 this, EventType.CREATE_EVENT, -1, t.toMillis(true), 0, 0, 0, -1); 794 return true; 795 case R.id.action_select_visible_calendars: 796 mController.sendEvent(this, EventType.LAUNCH_SELECT_VISIBLE_CALENDARS, null, null, 797 0, 0); 798 return true; 799 case R.id.action_settings: 800 mController.sendEvent(this, EventType.LAUNCH_SETTINGS, null, null, 0, 0); 801 return true; 802 case R.id.action_hide_controls: 803 mHideControls = !mHideControls; 804 Utils.setSharedPreference( 805 this, GeneralPreferences.KEY_SHOW_CONTROLS, !mHideControls); 806 item.setTitle(mHideControls ? mShowString : mHideString); 807 if (!mHideControls) { 808 mMiniMonth.setVisibility(View.VISIBLE); 809 mCalendarsList.setVisibility(View.VISIBLE); 810 mMiniMonthContainer.setVisibility(View.VISIBLE); 811 } 812 final ObjectAnimator slideAnimation = ObjectAnimator.ofInt(this, "controlsOffset", 813 mHideControls ? 0 : mControlsAnimateWidth, 814 mHideControls ? mControlsAnimateWidth : 0); 815 slideAnimation.setDuration(mCalendarControlsAnimationTime); 816 ObjectAnimator.setFrameDelay(0); 817 slideAnimation.start(); 818 return true; 819 case R.id.action_search: 820 return false; 821 default: 822 return false; 823 } 824 mController.sendEvent(this, EventType.GO_TO, t, null, t, -1, viewType, extras, null, null); 825 return true; 826 } 827 828 /** 829 * Sets the offset of the controls on the right for animating them off/on 830 * screen. ProGuard strips this if it's not in proguard.flags 831 * 832 * @param controlsOffset The current offset in pixels 833 */ 834 public void setControlsOffset(int controlsOffset) { 835 if (mOrientation == Configuration.ORIENTATION_LANDSCAPE) { 836 mMiniMonth.setTranslationX(controlsOffset); 837 mCalendarsList.setTranslationX(controlsOffset); 838 mControlsParams.width = Math.max(0, mControlsAnimateWidth - controlsOffset); 839 mMiniMonthContainer.setLayoutParams(mControlsParams); 840 } else { 841 mMiniMonth.setTranslationY(controlsOffset); 842 mCalendarsList.setTranslationY(controlsOffset); 843 if (mVerticalControlsParams == null) { 844 mVerticalControlsParams = new LinearLayout.LayoutParams( 845 LinearLayout.LayoutParams.MATCH_PARENT, mControlsAnimateHeight); 846 } 847 mVerticalControlsParams.height = Math.max(0, mControlsAnimateHeight - controlsOffset); 848 mMiniMonthContainer.setLayoutParams(mVerticalControlsParams); 849 } 850 } 851 852 @Override 853 public void onSharedPreferenceChanged(SharedPreferences prefs, String key) { 854 if (key.equals(GeneralPreferences.KEY_WEEK_START_DAY)) { 855 if (mPaused) { 856 mUpdateOnResume = true; 857 } else { 858 initFragments(mController.getTime(), mController.getViewType(), null); 859 } 860 } 861 } 862 863 private void setMainPane( 864 FragmentTransaction ft, int viewId, int viewType, long timeMillis, boolean force) { 865 if (mOnSaveInstanceStateCalled) { 866 return; 867 } 868 if (!force && mCurrentView == viewType) { 869 return; 870 } 871 872 // Remove this when transition to and from month view looks fine. 873 boolean doTransition = viewType != ViewType.MONTH && mCurrentView != ViewType.MONTH; 874 FragmentManager fragmentManager = getFragmentManager(); 875 // Check if our previous view was an Agenda view 876 // TODO remove this if framework ever supports nested fragments 877 if (mCurrentView == ViewType.AGENDA) { 878 // If it was, we need to do some cleanup on it to prevent the 879 // edit/delete buttons from coming back on a rotation. 880 Fragment oldFrag = fragmentManager.findFragmentById(viewId); 881 if (oldFrag instanceof AgendaFragment) { 882 ((AgendaFragment) oldFrag).removeFragments(fragmentManager); 883 } 884 } 885 886 if (viewType != mCurrentView) { 887 // The rules for this previous view are different than the 888 // controller's and are used for intercepting the back button. 889 if (mCurrentView != ViewType.EDIT && mCurrentView > 0) { 890 mPreviousView = mCurrentView; 891 } 892 mCurrentView = viewType; 893 } 894 // Create new fragment 895 Fragment frag = null; 896 Fragment secFrag = null; 897 switch (viewType) { 898 case ViewType.AGENDA: 899 if (mActionBar != null && (mActionBar.getSelectedTab() != mAgendaTab)) { 900 mActionBar.selectTab(mAgendaTab); 901 } 902 if (mActionBarMenuSpinnerAdapter != null) { 903 mActionBar.setSelectedNavigationItem(CalendarViewAdapter.AGENDA_BUTTON_INDEX); 904 } 905 frag = new AgendaFragment(timeMillis, false); 906 break; 907 case ViewType.DAY: 908 if (mActionBar != null && (mActionBar.getSelectedTab() != mDayTab)) { 909 mActionBar.selectTab(mDayTab); 910 } 911 if (mActionBarMenuSpinnerAdapter != null) { 912 mActionBar.setSelectedNavigationItem(CalendarViewAdapter.DAY_BUTTON_INDEX); 913 } 914 frag = new DayFragment(timeMillis, 1); 915 break; 916 case ViewType.WEEK: 917 if (mActionBar != null && (mActionBar.getSelectedTab() != mWeekTab)) { 918 mActionBar.selectTab(mWeekTab); 919 } 920 if (mActionBarMenuSpinnerAdapter != null) { 921 mActionBar.setSelectedNavigationItem(CalendarViewAdapter.WEEK_BUTTON_INDEX); 922 } 923 frag = new DayFragment(timeMillis, 7); 924 break; 925 case ViewType.MONTH: 926 if (mActionBar != null && (mActionBar.getSelectedTab() != mMonthTab)) { 927 mActionBar.selectTab(mMonthTab); 928 } 929 if (mActionBarMenuSpinnerAdapter != null) { 930 mActionBar.setSelectedNavigationItem(CalendarViewAdapter.MONTH_BUTTON_INDEX); 931 } 932 frag = new MonthByWeekFragment(timeMillis, false); 933 if (mShowAgendaWithMonth) { 934 secFrag = new AgendaFragment(timeMillis, false); 935 } 936 break; 937 default: 938 throw new IllegalArgumentException( 939 "Must be Agenda, Day, Week, or Month ViewType, not " + viewType); 940 } 941 942 // Update the current view so that the menu can update its look according to the 943 // current view. 944 if (mActionBarMenuSpinnerAdapter != null) { 945 mActionBarMenuSpinnerAdapter.setMainView(viewType); 946 if (!mIsTabletConfig) { 947 mActionBarMenuSpinnerAdapter.setTime(timeMillis); 948 } 949 } 950 951 952 // Show date only on tablet configurations in views different than Agenda 953 if (!mIsTabletConfig) { 954 mDateRange.setVisibility(View.GONE); 955 } else if (viewType != ViewType.AGENDA) { 956 mDateRange.setVisibility(View.VISIBLE); 957 } else { 958 mDateRange.setVisibility(View.GONE); 959 } 960 961 // Clear unnecessary buttons from the option menu when switching from the agenda view 962 if (viewType != ViewType.AGENDA) { 963 clearOptionsMenu(); 964 } 965 966 boolean doCommit = false; 967 if (ft == null) { 968 doCommit = true; 969 ft = fragmentManager.beginTransaction(); 970 } 971 972 if (doTransition) { 973 ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE); 974 } 975 976 ft.replace(viewId, frag); 977 if (mShowAgendaWithMonth) { 978 979 // Show/hide secondary fragment 980 981 if (secFrag != null) { 982 ft.replace(R.id.secondary_pane, secFrag); 983 mSecondaryPane.setVisibility(View.VISIBLE); 984 } else { 985 mSecondaryPane.setVisibility(View.GONE); 986 Fragment f = fragmentManager.findFragmentById(R.id.secondary_pane); 987 if (f != null) { 988 ft.remove(f); 989 } 990 mController.deregisterEventHandler(R.id.secondary_pane); 991 } 992 } 993 if (DEBUG) { 994 Log.d(TAG, "Adding handler with viewId " + viewId + " and type " + viewType); 995 } 996 // If the key is already registered this will replace it 997 mController.registerEventHandler(viewId, (EventHandler) frag); 998 if (secFrag != null) { 999 mController.registerEventHandler(viewId, (EventHandler) secFrag); 1000 } 1001 1002 if (doCommit) { 1003 if (DEBUG) { 1004 Log.d(TAG, "setMainPane AllInOne=" + this + " finishing:" + this.isFinishing()); 1005 } 1006 ft.commit(); 1007 } 1008 } 1009 1010 private void setTitleInActionBar(EventInfo event) { 1011 if (event.eventType != EventType.UPDATE_TITLE || mActionBar == null) { 1012 return; 1013 } 1014 1015 final long start = event.startTime.toMillis(false /* use isDst */); 1016 final long end; 1017 if (event.endTime != null) { 1018 end = event.endTime.toMillis(false /* use isDst */); 1019 } else { 1020 end = start; 1021 } 1022 1023 final String msg = Utils.formatDateRange(this, start, end, (int) event.extraLong); 1024 CharSequence oldDate = mDateRange.getText(); 1025 mDateRange.setText(msg); 1026 updateSecondaryTitleFields(event.selectedTime != null ? event.selectedTime.toMillis(true) 1027 : start); 1028 if (!TextUtils.equals(oldDate, msg)) { 1029 mDateRange.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_FOCUSED); 1030 if (mShowWeekNum && mWeekTextView != null) { 1031 mWeekTextView.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_FOCUSED); 1032 } 1033 } 1034 } 1035 1036 private void updateSecondaryTitleFields(long visibleMillisSinceEpoch) { 1037 mShowWeekNum = Utils.getShowWeekNumber(this); 1038 mTimeZone = Utils.getTimeZone(this, mHomeTimeUpdater); 1039 if (visibleMillisSinceEpoch != -1) { 1040 int weekNum = Utils.getWeekNumberFromTime(visibleMillisSinceEpoch, this); 1041 mWeekNum = weekNum; 1042 } 1043 1044 if (mShowWeekNum && (mCurrentView == ViewType.WEEK) && mIsTabletConfig 1045 && mWeekTextView != null) { 1046 String weekString = getResources().getQuantityString(R.plurals.weekN, mWeekNum, 1047 mWeekNum); 1048 mWeekTextView.setText(weekString); 1049 mWeekTextView.setVisibility(View.VISIBLE); 1050 } else if (visibleMillisSinceEpoch != -1 && mWeekTextView != null 1051 && mCurrentView == ViewType.DAY && mIsTabletConfig) { 1052 Time time = new Time(mTimeZone); 1053 time.set(visibleMillisSinceEpoch); 1054 int julianDay = Time.getJulianDay(visibleMillisSinceEpoch, time.gmtoff); 1055 time.setToNow(); 1056 int todayJulianDay = Time.getJulianDay(time.toMillis(false), time.gmtoff); 1057 String dayString = Utils.getDayOfWeekString(julianDay, todayJulianDay, 1058 visibleMillisSinceEpoch, this); 1059 mWeekTextView.setText(dayString); 1060 mWeekTextView.setVisibility(View.VISIBLE); 1061 } else if (mWeekTextView != null && (!mIsTabletConfig || mCurrentView != ViewType.DAY)) { 1062 mWeekTextView.setVisibility(View.GONE); 1063 } 1064 1065 if (mHomeTime != null 1066 && (mCurrentView == ViewType.DAY || mCurrentView == ViewType.WEEK 1067 || mCurrentView == ViewType.AGENDA) 1068 && !TextUtils.equals(mTimeZone, Time.getCurrentTimezone())) { 1069 Time time = new Time(mTimeZone); 1070 time.setToNow(); 1071 long millis = time.toMillis(true); 1072 boolean isDST = time.isDst != 0; 1073 int flags = DateUtils.FORMAT_SHOW_TIME; 1074 if (DateFormat.is24HourFormat(this)) { 1075 flags |= DateUtils.FORMAT_24HOUR; 1076 } 1077 // Formats the time as 1078 String timeString = (new StringBuilder( 1079 Utils.formatDateRange(this, millis, millis, flags))).append(" ").append( 1080 TimeZone.getTimeZone(mTimeZone).getDisplayName( 1081 isDST, TimeZone.SHORT, Locale.getDefault())).toString(); 1082 mHomeTime.setText(timeString); 1083 mHomeTime.setVisibility(View.VISIBLE); 1084 // Update when the minute changes 1085 mHomeTime.removeCallbacks(mHomeTimeUpdater); 1086 mHomeTime.postDelayed( 1087 mHomeTimeUpdater, 1088 DateUtils.MINUTE_IN_MILLIS - (millis % DateUtils.MINUTE_IN_MILLIS)); 1089 } else if (mHomeTime != null) { 1090 mHomeTime.setVisibility(View.GONE); 1091 } 1092 } 1093 1094 @Override 1095 public long getSupportedEventTypes() { 1096 return EventType.GO_TO | EventType.VIEW_EVENT | EventType.UPDATE_TITLE; 1097 } 1098 1099 @Override 1100 public void handleEvent(EventInfo event) { 1101 long displayTime = -1; 1102 if (event.eventType == EventType.GO_TO) { 1103 if ((event.extraLong & CalendarController.EXTRA_GOTO_BACK_TO_PREVIOUS) != 0) { 1104 mBackToPreviousView = true; 1105 } else if (event.viewType != mController.getPreviousViewType() 1106 && event.viewType != ViewType.EDIT) { 1107 // Clear the flag is change to a different view type 1108 mBackToPreviousView = false; 1109 } 1110 1111 setMainPane( 1112 null, R.id.main_pane, event.viewType, event.startTime.toMillis(false), false); 1113 if (mSearchView != null) { 1114 mSearchView.clearFocus(); 1115 } 1116 if (mShowCalendarControls) { 1117 int animationSize = (mOrientation == Configuration.ORIENTATION_LANDSCAPE) ? 1118 mControlsAnimateWidth : mControlsAnimateHeight; 1119 boolean noControlsView = event.viewType == ViewType.MONTH || event.viewType == ViewType.AGENDA; 1120 if (mControlsMenu != null) { 1121 mControlsMenu.setVisible(!noControlsView); 1122 mControlsMenu.setEnabled(!noControlsView); 1123 } 1124 if (noControlsView || mHideControls) { 1125 // hide minimonth and calendar frag 1126 mShowSideViews = false; 1127 if (!mHideControls) { 1128 final ObjectAnimator slideAnimation = ObjectAnimator.ofInt(this, 1129 "controlsOffset", 0, animationSize); 1130 slideAnimation.addListener(mSlideAnimationDoneListener); 1131 slideAnimation.setDuration(mCalendarControlsAnimationTime); 1132 ObjectAnimator.setFrameDelay(0); 1133 slideAnimation.start(); 1134 } else { 1135 mMiniMonth.setVisibility(View.GONE); 1136 mCalendarsList.setVisibility(View.GONE); 1137 mMiniMonthContainer.setVisibility(View.GONE); 1138 } 1139 } else { 1140 // show minimonth and calendar frag 1141 mShowSideViews = true; 1142 mMiniMonth.setVisibility(View.VISIBLE); 1143 mCalendarsList.setVisibility(View.VISIBLE); 1144 mMiniMonthContainer.setVisibility(View.VISIBLE); 1145 if (!mHideControls && 1146 (mController.getPreviousViewType() == ViewType.MONTH || 1147 mController.getPreviousViewType() == ViewType.AGENDA)) { 1148 final ObjectAnimator slideAnimation = ObjectAnimator.ofInt(this, 1149 "controlsOffset", animationSize, 0); 1150 slideAnimation.setDuration(mCalendarControlsAnimationTime); 1151 ObjectAnimator.setFrameDelay(0); 1152 slideAnimation.start(); 1153 } 1154 } 1155 } 1156 displayTime = event.selectedTime != null ? event.selectedTime.toMillis(true) 1157 : event.startTime.toMillis(true); 1158 if (!mIsTabletConfig) { 1159 mActionBarMenuSpinnerAdapter.setTime(displayTime); 1160 } 1161 } else if (event.eventType == EventType.VIEW_EVENT) { 1162 1163 // If in Agenda view and "show_event_details_with_agenda" is "true", 1164 // do not create the event info fragment here, it will be created by the Agenda 1165 // fragment 1166 1167 if (mCurrentView == ViewType.AGENDA && mShowEventDetailsWithAgenda) { 1168 if (event.startTime != null && event.endTime != null) { 1169 // Event is all day , adjust the goto time to local time 1170 if (event.isAllDay()) { 1171 Utils.convertAlldayUtcToLocal( 1172 event.startTime, event.startTime.toMillis(false), mTimeZone); 1173 Utils.convertAlldayUtcToLocal( 1174 event.endTime, event.endTime.toMillis(false), mTimeZone); 1175 } 1176 mController.sendEvent(this, EventType.GO_TO, event.startTime, event.endTime, 1177 event.selectedTime, event.id, ViewType.AGENDA, 1178 CalendarController.EXTRA_GOTO_TIME, null, null); 1179 } else if (event.selectedTime != null) { 1180 mController.sendEvent(this, EventType.GO_TO, event.selectedTime, 1181 event.selectedTime, event.id, ViewType.AGENDA); 1182 } 1183 } else { 1184 // TODO Fix the temp hack below: && mCurrentView != 1185 // ViewType.AGENDA 1186 if (event.selectedTime != null && mCurrentView != ViewType.AGENDA) { 1187 mController.sendEvent(this, EventType.GO_TO, event.selectedTime, 1188 event.selectedTime, -1, ViewType.CURRENT); 1189 } 1190 int response = event.getResponse(); 1191 if ((mCurrentView == ViewType.AGENDA && mShowEventInfoFullScreenAgenda) || 1192 ((mCurrentView == ViewType.DAY || (mCurrentView == ViewType.WEEK) || 1193 mCurrentView == ViewType.MONTH) && mShowEventInfoFullScreen)){ 1194 // start event info as activity 1195 Intent intent = new Intent(Intent.ACTION_VIEW); 1196 Uri eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, event.id); 1197 intent.setData(eventUri); 1198 intent.setClass(this, EventInfoActivity.class); 1199 intent.setFlags(Intent.FLAG_ACTIVITY_REORDER_TO_FRONT | 1200 Intent.FLAG_ACTIVITY_SINGLE_TOP); 1201 intent.putExtra(EXTRA_EVENT_BEGIN_TIME, event.startTime.toMillis(false)); 1202 intent.putExtra(EXTRA_EVENT_END_TIME, event.endTime.toMillis(false)); 1203 intent.putExtra(EVENT_ATTENDEE_RESPONSE, response); 1204 startActivity(intent); 1205 } else { 1206 // start event info as a dialog 1207 EventInfoFragment fragment = new EventInfoFragment(this, 1208 event.id, event.startTime.toMillis(false), 1209 event.endTime.toMillis(false), response, true, 1210 EventInfoFragment.DIALOG_WINDOW_STYLE); 1211 fragment.setDialogParams(event.x, event.y, mActionBar.getHeight()); 1212 FragmentManager fm = getFragmentManager(); 1213 FragmentTransaction ft = fm.beginTransaction(); 1214 // if we have an old popup replace it 1215 Fragment fOld = fm.findFragmentByTag(EVENT_INFO_FRAGMENT_TAG); 1216 if (fOld != null && fOld.isAdded()) { 1217 ft.remove(fOld); 1218 } 1219 ft.add(fragment, EVENT_INFO_FRAGMENT_TAG); 1220 ft.commit(); 1221 } 1222 } 1223 displayTime = event.startTime.toMillis(true); 1224 } else if (event.eventType == EventType.UPDATE_TITLE) { 1225 setTitleInActionBar(event); 1226 if (!mIsTabletConfig) { 1227 mActionBarMenuSpinnerAdapter.setTime(mController.getTime()); 1228 } 1229 } 1230 updateSecondaryTitleFields(displayTime); 1231 } 1232 1233 // Needs to be in proguard whitelist 1234 // Specified as listener via android:onClick in a layout xml 1235 public void handleSelectSyncedCalendarsClicked(View v) { 1236 mController.sendEvent(this, EventType.LAUNCH_SETTINGS, null, null, null, 0, 0, 1237 CalendarController.EXTRA_GOTO_TIME, null, 1238 null); 1239 } 1240 1241 @Override 1242 public void eventsChanged() { 1243 mController.sendEvent(this, EventType.EVENTS_CHANGED, null, null, -1, ViewType.CURRENT); 1244 } 1245 1246 @Override 1247 public boolean onQueryTextChange(String newText) { 1248 return false; 1249 } 1250 1251 @Override 1252 public boolean onQueryTextSubmit(String query) { 1253 if ("TARDIS".equalsIgnoreCase(query)) { 1254 Utils.tardis(); 1255 } 1256 mSearchMenu.collapseActionView(); 1257 mController.sendEvent(this, EventType.SEARCH, null, null, -1, ViewType.CURRENT, 0, query, 1258 getComponentName()); 1259 return true; 1260 } 1261 1262 @Override 1263 public void onTabSelected(Tab tab, FragmentTransaction ft) { 1264 Log.w(TAG, "TabSelected AllInOne=" + this + " finishing:" + this.isFinishing()); 1265 if (tab == mDayTab && mCurrentView != ViewType.DAY) { 1266 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.DAY); 1267 } else if (tab == mWeekTab && mCurrentView != ViewType.WEEK) { 1268 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.WEEK); 1269 } else if (tab == mMonthTab && mCurrentView != ViewType.MONTH) { 1270 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.MONTH); 1271 } else if (tab == mAgendaTab && mCurrentView != ViewType.AGENDA) { 1272 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.AGENDA); 1273 } else { 1274 Log.w(TAG, "TabSelected event from unknown tab: " 1275 + (tab == null ? "null" : tab.getText())); 1276 Log.w(TAG, "CurrentView:" + mCurrentView + " Tab:" + tab.toString() + " Day:" + mDayTab 1277 + " Week:" + mWeekTab + " Month:" + mMonthTab + " Agenda:" + mAgendaTab); 1278 } 1279 } 1280 1281 @Override 1282 public void onTabReselected(Tab tab, FragmentTransaction ft) { 1283 } 1284 1285 @Override 1286 public void onTabUnselected(Tab tab, FragmentTransaction ft) { 1287 } 1288 1289 1290 @Override 1291 public boolean onNavigationItemSelected(int itemPosition, long itemId) { 1292 switch (itemPosition) { 1293 case CalendarViewAdapter.DAY_BUTTON_INDEX: 1294 if (mCurrentView != ViewType.DAY) { 1295 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.DAY); 1296 } 1297 break; 1298 case CalendarViewAdapter.WEEK_BUTTON_INDEX: 1299 if (mCurrentView != ViewType.WEEK) { 1300 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.WEEK); 1301 } 1302 break; 1303 case CalendarViewAdapter.MONTH_BUTTON_INDEX: 1304 if (mCurrentView != ViewType.MONTH) { 1305 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.MONTH); 1306 } 1307 break; 1308 case CalendarViewAdapter.AGENDA_BUTTON_INDEX: 1309 if (mCurrentView != ViewType.AGENDA) { 1310 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.AGENDA); 1311 } 1312 break; 1313 default: 1314 Log.w(TAG, "ItemSelected event from unknown button: " + itemPosition); 1315 Log.w(TAG, "CurrentView:" + mCurrentView + " Button:" + itemPosition + 1316 " Day:" + mDayTab + " Week:" + mWeekTab + " Month:" + mMonthTab + 1317 " Agenda:" + mAgendaTab); 1318 break; 1319 } 1320 return false; 1321 } 1322 1323 @Override 1324 public boolean onSuggestionSelect(int position) { 1325 return false; 1326 } 1327 1328 @Override 1329 public boolean onSuggestionClick(int position) { 1330 mSearchMenu.collapseActionView(); 1331 return false; 1332 } 1333 1334 @Override 1335 public boolean onSearchRequested() { 1336 if (mSearchMenu != null) { 1337 mSearchMenu.expandActionView(); 1338 } 1339 return false; 1340 } 1341} 1342