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