AllInOneActivity.java revision a7c0390d9c5dd4ff730de505682687fae5f5ced0
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.EVENT_BEGIN_TIME; 20import static android.provider.CalendarContract.EVENT_END_TIME; 21import static android.provider.CalendarContract.Attendees.ATTENDEE_STATUS; 22import static com.android.calendar.CalendarController.EVENT_ATTENDEE_RESPONSE; 23 24import com.android.calendar.CalendarController.EventHandler; 25import com.android.calendar.CalendarController.EventInfo; 26import com.android.calendar.CalendarController.EventType; 27import com.android.calendar.CalendarController.ViewType; 28import com.android.calendar.agenda.AgendaFragment; 29import com.android.calendar.month.MonthByWeekFragment; 30import com.android.calendar.selectcalendars.SelectCalendarsFragment; 31 32import android.animation.Animator; 33import android.animation.Animator.AnimatorListener; 34import android.animation.ObjectAnimator; 35import android.app.ActionBar; 36import android.app.ActionBar.Tab; 37import android.app.Activity; 38import android.app.Fragment; 39import android.app.FragmentManager; 40import android.app.FragmentTransaction; 41import android.content.ContentResolver; 42import android.content.ContentUris; 43import android.content.Context; 44import android.content.Intent; 45import android.content.SharedPreferences; 46import android.content.SharedPreferences.OnSharedPreferenceChangeListener; 47import android.content.res.Configuration; 48import android.content.res.Resources; 49import android.database.ContentObserver; 50import android.net.Uri; 51import android.os.Bundle; 52import android.os.Handler; 53import android.provider.CalendarContract; 54import android.provider.CalendarContract.Events; 55import android.text.TextUtils; 56import android.text.format.DateFormat; 57import android.text.format.DateUtils; 58import android.text.format.Time; 59import android.util.Log; 60import android.view.Menu; 61import android.view.MenuItem; 62import android.view.View; 63import android.view.accessibility.AccessibilityEvent; 64import android.widget.RelativeLayout; 65import android.widget.RelativeLayout.LayoutParams; 66import android.widget.SearchView; 67import android.widget.TextView; 68 69import java.util.List; 70import java.util.Locale; 71import java.util.TimeZone; 72 73public class AllInOneActivity extends Activity implements EventHandler, 74 OnSharedPreferenceChangeListener, SearchView.OnQueryTextListener, 75 ActionBar.TabListener { 76 private static final String TAG = "AllInOneActivity"; 77 private static final boolean DEBUG = false; 78 private static final String EVENT_INFO_FRAGMENT_TAG = "EventInfoFragment"; 79 private static final String BUNDLE_KEY_RESTORE_TIME = "key_restore_time"; 80 private static final String BUNDLE_KEY_EVENT_ID = "key_event_id"; 81 private static final String BUNDLE_KEY_RESTORE_VIEW = "key_restore_view"; 82 private static final int HANDLER_KEY = 0; 83 private static final long CONTROLS_ANIMATE_DURATION = 400; 84 private static int CONTROLS_ANIMATE_WIDTH = 267; 85 private static float mScale = 0; 86 87 private static CalendarController mController; 88 private static boolean mIsMultipane; 89 private static boolean mShowAgendaWithMonth; 90 private static boolean mShowEventDetailsWithAgenda; 91 private boolean mOnSaveInstanceStateCalled = false; 92 private ContentResolver mContentResolver; 93 private int mPreviousView; 94 private int mCurrentView; 95 private boolean mPaused = true; 96 private boolean mUpdateOnResume = false; 97 private boolean mHideControls = false; 98 private boolean mShowSideViews = true; 99 private TextView mHomeTime; 100 private TextView mDateRange; 101 private View mMiniMonth; 102 private View mCalendarsList; 103 private View mMiniMonthContainer; 104 private View mSecondaryPane; 105 private String mTimeZone; 106 private boolean mShowCalendarControls; 107 private boolean mShowEventInfoFullScreen; 108 109 private long mViewEventId = -1; 110 private long mIntentEventStartMillis = -1; 111 private long mIntentEventEndMillis = -1; 112 private int mIntentAttendeeResponse = CalendarController.ATTENDEE_NO_RESPONSE; 113 114 // Action bar and Navigation bar (left side of Action bar) 115 private ActionBar mActionBar; 116 private ActionBar.Tab mDayTab; 117 private ActionBar.Tab mWeekTab; 118 private ActionBar.Tab mMonthTab; 119 private ActionBar.Tab mAgendaTab; 120 private SearchView mSearchView; 121 private MenuItem mControlsMenu; 122 private Menu mOptionsMenu; 123 124 private String mHideString; 125 private String mShowString; 126 127 // Params for animating the controls on the right 128 private LayoutParams mControlsParams = new LayoutParams(CONTROLS_ANIMATE_WIDTH, 0); 129 130 private AnimatorListener mSlideAnimationDoneListener = new AnimatorListener() { 131 132 @Override 133 public void onAnimationCancel(Animator animation) { 134 } 135 136 @Override 137 public void onAnimationEnd(android.animation.Animator animation) { 138 int visibility = mShowSideViews ? View.VISIBLE : View.GONE; 139 mMiniMonth.setVisibility(visibility); 140 mCalendarsList.setVisibility(visibility); 141 mMiniMonthContainer.setVisibility(visibility); 142 } 143 144 @Override 145 public void onAnimationRepeat(android.animation.Animator animation) { 146 } 147 148 @Override 149 public void onAnimationStart(android.animation.Animator animation) { 150 } 151 }; 152 153 private Runnable mHomeTimeUpdater = new Runnable() { 154 @Override 155 public void run() { 156 updateHomeClock(); 157 } 158 }; 159 160 // Create an observer so that we can update the views whenever a 161 // Calendar event changes. 162 private ContentObserver mObserver = new ContentObserver(new Handler()) { 163 @Override 164 public boolean deliverSelfNotifications() { 165 return true; 166 } 167 168 @Override 169 public void onChange(boolean selfChange) { 170 eventsChanged(); 171 } 172 }; 173 174 @Override 175 protected void onNewIntent(Intent intent) { 176 String action = intent.getAction(); 177 if (Intent.ACTION_VIEW.equals(action)) { 178 parseViewAction(intent); 179 } 180 } 181 182 @Override 183 protected void onCreate(Bundle icicle) { 184 if (Utils.getSharedPreference(this, OtherPreferences.KEY_OTHER_1, false)) { 185 setTheme(R.style.CalendarTheme_WithActionBarWallpaper); 186 } 187 super.onCreate(icicle); 188 189 // This needs to be created before setContentView 190 mController = CalendarController.getInstance(this); 191 // Get time from intent or icicle 192 long timeMillis = -1; 193 int viewType = -1; 194 final Intent intent = getIntent(); 195 if (icicle != null) { 196 timeMillis = icicle.getLong(BUNDLE_KEY_RESTORE_TIME); 197 viewType = icicle.getInt(BUNDLE_KEY_RESTORE_VIEW, -1); 198 } else { 199 String action = intent.getAction(); 200 if (Intent.ACTION_VIEW.equals(action)) { 201 // Open EventInfo later 202 timeMillis = parseViewAction(intent); 203 } 204 205 if (timeMillis == -1) { 206 timeMillis = Utils.timeFromIntentInMillis(intent); 207 } 208 } 209 210 if (viewType == -1) { 211 viewType = Utils.getViewTypeFromIntentAndSharedPref(this); 212 } 213 mTimeZone = Utils.getTimeZone(this, mHomeTimeUpdater); 214 Time t = new Time(mTimeZone); 215 t.set(timeMillis); 216 217 if (icicle != null && intent != null) { 218 Log.d(TAG, "both, icicle:" + icicle.toString() + " intent:" + intent.toString()); 219 } else { 220 Log.d(TAG, "not both, icicle:" + icicle + " intent:" + intent); 221 } 222 223 Resources res = getResources(); 224 if (mScale == 0) { 225 mScale = res.getDisplayMetrics().density; 226 CONTROLS_ANIMATE_WIDTH *= mScale; 227 } 228 mHideString = res.getString(R.string.hide_controls); 229 mShowString = res.getString(R.string.show_controls); 230 mControlsParams.addRule(RelativeLayout.ALIGN_PARENT_RIGHT); 231 232 mIsMultipane = Utils.isMultiPaneConfiguration(this); 233 mShowAgendaWithMonth = Utils.getConfigBool(this, R.bool.show_agenda_with_month); 234 mShowCalendarControls = Utils.getConfigBool(this, R.bool.show_calendar_controls); 235 mShowEventDetailsWithAgenda = 236 Utils.getConfigBool(this, R.bool.show_event_details_with_agenda); 237 mShowEventInfoFullScreen = 238 Utils.getConfigBool(this, R.bool.show_event_info_full_screen); 239 240 Utils.setAllowWeekForDetailView(mIsMultipane); 241 242 mDateRange = (TextView) getLayoutInflater().inflate(R.layout.date_range_title, null); 243 244 // setContentView must be called before configureActionBar 245 setContentView(R.layout.all_in_one); 246 // configureActionBar auto-selects the first tab you add, so we need to 247 // call it before we set up our own fragments to make sure it doesn't 248 // overwrite us 249 configureActionBar(); 250 251 // Must be the first to register because this activity can modify the 252 // list of event handlers in it's handle method. This affects who the 253 // rest of the handlers the controller dispatches to are. 254 mController.registerEventHandler(HANDLER_KEY, this); 255 256 mHomeTime = (TextView) findViewById(R.id.home_time); 257 mMiniMonth = findViewById(R.id.mini_month); 258 mCalendarsList = findViewById(R.id.calendar_list); 259 mMiniMonthContainer = findViewById(R.id.mini_month_container); 260 mSecondaryPane = findViewById(R.id.secondary_pane); 261 262 initFragments(timeMillis, viewType, icicle); 263 264 // Listen for changes that would require this to be refreshed 265 SharedPreferences prefs = GeneralPreferences.getSharedPreferences(this); 266 prefs.registerOnSharedPreferenceChangeListener(this); 267 268 mContentResolver = getContentResolver(); 269 } 270 271 private long parseViewAction(final Intent intent) { 272 long timeMillis = -1; 273 Uri data = intent.getData(); 274 if (data != null && data.isHierarchical()) { 275 List<String> path = data.getPathSegments(); 276 if (path.size() == 2 && path.get(0).equals("events")) { 277 try { 278 mViewEventId = Long.valueOf(data.getLastPathSegment()); 279 if (mViewEventId != -1) { 280 mIntentEventStartMillis = intent.getLongExtra(EVENT_BEGIN_TIME, 0); 281 mIntentEventEndMillis = intent.getLongExtra(EVENT_END_TIME, 0); 282 mIntentAttendeeResponse = intent.getIntExtra( 283 ATTENDEE_STATUS, CalendarController.ATTENDEE_NO_RESPONSE); 284 timeMillis = mIntentEventStartMillis; 285 } 286 } catch (NumberFormatException e) { 287 // Ignore if mViewEventId can't be parsed 288 } 289 } 290 } 291 return timeMillis; 292 } 293 294 private void configureActionBar() { 295 mActionBar = getActionBar(); 296 mActionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS); 297 if (mActionBar == null) { 298 Log.w(TAG, "ActionBar is null."); 299 } else { 300 mDayTab = mActionBar.newTab(); 301 mDayTab.setText(getString(R.string.day_view)); 302 mDayTab.setTabListener(this); 303 mActionBar.addTab(mDayTab); 304 mWeekTab = mActionBar.newTab(); 305 mWeekTab.setText(getString(R.string.week_view)); 306 mWeekTab.setTabListener(this); 307 mActionBar.addTab(mWeekTab); 308 mMonthTab = mActionBar.newTab(); 309 mMonthTab.setText(getString(R.string.month_view)); 310 mMonthTab.setTabListener(this); 311 mActionBar.addTab(mMonthTab); 312 mAgendaTab = mActionBar.newTab(); 313 mAgendaTab.setText(getString(R.string.agenda_view)); 314 mAgendaTab.setTabListener(this); 315 mActionBar.addTab(mAgendaTab); 316 mActionBar.setCustomView(mDateRange); 317 if (mIsMultipane) { 318 mActionBar.setDisplayOptions( 319 ActionBar.DISPLAY_SHOW_CUSTOM | ActionBar.DISPLAY_SHOW_HOME); 320 } else { 321 mActionBar.setDisplayOptions(ActionBar.DISPLAY_SHOW_CUSTOM); 322 } 323 } 324 } 325 326 // Clear buttons used in the agenda view 327 private void clearOptionsMenu() { 328 if (mOptionsMenu == null) { 329 return; 330 } 331 MenuItem cancelItem = mOptionsMenu.findItem(R.id.action_cancel); 332 MenuItem deleteItem = mOptionsMenu.findItem(R.id.action_delete); 333 MenuItem editItem = mOptionsMenu.findItem(R.id.action_edit); 334 if (cancelItem != null) { 335 cancelItem.setVisible(false); 336 } 337 if (deleteItem != null) { 338 deleteItem.setVisible(false); 339 } 340 if (editItem != null) { 341 editItem.setVisible(false); 342 } 343 } 344 345 @Override 346 protected void onResume() { 347 super.onResume(); 348 mContentResolver.registerContentObserver(CalendarContract.Events.CONTENT_URI, true, mObserver); 349 if (mUpdateOnResume) { 350 initFragments(mController.getTime(), mController.getViewType(), null); 351 mUpdateOnResume = false; 352 } 353 updateHomeClock(); 354 if (mControlsMenu != null) { 355 mControlsMenu.setTitle(mHideControls ? mShowString : mHideString); 356 } 357 mPaused = false; 358 mOnSaveInstanceStateCalled = false; 359 360 if (mViewEventId != -1 && mIntentEventStartMillis != -1 && mIntentEventEndMillis != -1) { 361 long currentMillis = System.currentTimeMillis(); 362 long selectedTime = -1; 363 if (currentMillis > mIntentEventStartMillis && currentMillis < mIntentEventEndMillis) { 364 selectedTime = currentMillis; 365 } 366 mController.sendEventRelatedEventWithResponse(this, EventType.VIEW_EVENT, mViewEventId, 367 mIntentEventStartMillis, mIntentEventEndMillis, -1, -1, 368 mIntentAttendeeResponse, selectedTime); 369 mViewEventId = -1; 370 mIntentEventStartMillis = -1; 371 mIntentEventEndMillis = -1; 372 } 373 } 374 375 @Override 376 protected void onPause() { 377 super.onPause(); 378 mPaused = true; 379 mHomeTime.removeCallbacks(mHomeTimeUpdater); 380 mContentResolver.unregisterContentObserver(mObserver); 381 if (isFinishing()) { 382 // Stop listening for changes that would require this to be refreshed 383 SharedPreferences prefs = GeneralPreferences.getSharedPreferences(this); 384 prefs.unregisterOnSharedPreferenceChangeListener(this); 385 } 386 // FRAG_TODO save highlighted days of the week; 387 if (mController.getViewType() != ViewType.EDIT) { 388 Utils.setDefaultView(this, mController.getViewType()); 389 } 390 } 391 392 @Override 393 protected void onUserLeaveHint() { 394 mController.sendEvent(this, EventType.USER_HOME, null, null, -1, ViewType.CURRENT); 395 super.onUserLeaveHint(); 396 } 397 398 @Override 399 public void onSaveInstanceState(Bundle outState) { 400 mOnSaveInstanceStateCalled = true; 401 super.onSaveInstanceState(outState); 402 403 outState.putLong(BUNDLE_KEY_RESTORE_TIME, mController.getTime()); 404 if (mCurrentView == ViewType.EDIT) { 405 outState.putInt(BUNDLE_KEY_RESTORE_VIEW, mCurrentView); 406 outState.putLong(BUNDLE_KEY_EVENT_ID, mController.getEventId()); 407 } 408 } 409 410 @Override 411 protected void onDestroy() { 412 super.onDestroy(); 413 414 SharedPreferences prefs = GeneralPreferences.getSharedPreferences(this); 415 prefs.unregisterOnSharedPreferenceChangeListener(this); 416 CalendarController.removeInstance(this); 417 } 418 419 private void initFragments(long timeMillis, int viewType, Bundle icicle) { 420 FragmentTransaction ft = getFragmentManager().beginTransaction(); 421 422 if (mShowCalendarControls) { 423 Fragment miniMonthFrag = new MonthByWeekFragment(timeMillis, true); 424 ft.replace(R.id.mini_month, miniMonthFrag); 425 mController.registerEventHandler(R.id.mini_month, (EventHandler) miniMonthFrag); 426 427 Fragment selectCalendarsFrag = new SelectCalendarsFragment(); 428 ft.replace(R.id.calendar_list, selectCalendarsFrag); 429 mController.registerEventHandler( 430 R.id.calendar_list, (EventHandler) selectCalendarsFrag); 431 } 432 if (!mShowCalendarControls || viewType == ViewType.EDIT) { 433 mMiniMonth.setVisibility(View.GONE); 434 mCalendarsList.setVisibility(View.GONE); 435 } 436 437 EventInfo info = null; 438 if (viewType == ViewType.EDIT) { 439 mPreviousView = GeneralPreferences.getSharedPreferences(this).getInt( 440 GeneralPreferences.KEY_START_VIEW, GeneralPreferences.DEFAULT_START_VIEW); 441 442 long eventId = -1; 443 Intent intent = getIntent(); 444 Uri data = intent.getData(); 445 if (data != null) { 446 try { 447 eventId = Long.parseLong(data.getLastPathSegment()); 448 } catch (NumberFormatException e) { 449 if (DEBUG) { 450 Log.d(TAG, "Create new event"); 451 } 452 } 453 } else if (icicle != null && icicle.containsKey(BUNDLE_KEY_EVENT_ID)) { 454 eventId = icicle.getLong(BUNDLE_KEY_EVENT_ID); 455 } 456 457 long begin = intent.getLongExtra(EVENT_BEGIN_TIME, -1); 458 long end = intent.getLongExtra(EVENT_END_TIME, -1); 459 info = new EventInfo(); 460 if (end != -1) { 461 info.endTime = new Time(); 462 info.endTime.set(end); 463 } 464 if (begin != -1) { 465 info.startTime = new Time(); 466 info.startTime.set(begin); 467 } 468 info.id = eventId; 469 // We set the viewtype so if the user presses back when they are 470 // done editing the controller knows we were in the Edit Event 471 // screen. Likewise for eventId 472 mController.setViewType(viewType); 473 mController.setEventId(eventId); 474 } else { 475 mPreviousView = viewType; 476 } 477 478 setMainPane(ft, R.id.main_pane, viewType, timeMillis, true); 479 ft.commit(); // this needs to be after setMainPane() 480 481 Time t = new Time(mTimeZone); 482 t.set(timeMillis); 483 if (viewType != ViewType.EDIT) { 484 mController.sendEvent(this, EventType.GO_TO, t, null, -1, viewType); 485 } 486 } 487 488 @Override 489 public void onBackPressed() { 490 if (mCurrentView == ViewType.EDIT || mCurrentView == ViewType.DETAIL) { 491 mController.sendEvent(this, EventType.GO_TO, null, null, -1, mPreviousView); 492 } else { 493 super.onBackPressed(); 494 } 495 } 496 497 @Override 498 public boolean onCreateOptionsMenu(Menu menu) { 499 super.onCreateOptionsMenu(menu); 500 mOptionsMenu = menu; 501 getMenuInflater().inflate(R.menu.all_in_one_title_bar, menu); 502 503 mSearchView = (SearchView) menu.findItem(R.id.action_search).getActionView(); 504 if (mSearchView != null) { 505 mSearchView.setIconifiedByDefault(true); 506 mSearchView.setOnQueryTextListener(this); 507 mSearchView.setSubmitButtonEnabled(true); 508 if (!mIsMultipane) 509 mSearchView.setVisibility(View.GONE); 510 } 511 512 // Hide the "show/hide controls" button if this is a phone 513 // or the view type is "Month". 514 515 mControlsMenu = menu.findItem(R.id.action_hide_controls); 516 if (!mShowCalendarControls) { 517 if (mControlsMenu != null) { 518 mControlsMenu.setVisible(false); 519 mControlsMenu.setEnabled(false); 520 } 521 } else if (mControlsMenu != null && mController != null 522 && mController.getViewType() == ViewType.MONTH) { 523 mControlsMenu.setVisible(false); 524 mControlsMenu.setEnabled(false); 525 } 526 return true; 527 } 528 529 @Override 530 public boolean onOptionsItemSelected(MenuItem item) { 531 Time t = null; 532 int viewType = ViewType.CURRENT; 533 switch (item.getItemId()) { 534 case R.id.action_refresh: 535 mController.refreshCalendars(); 536 return true; 537 case R.id.action_today: 538 viewType = ViewType.CURRENT; 539 t = new Time(mTimeZone); 540 t.setToNow(); 541 break; 542 case R.id.action_create_event: 543 t = new Time(); 544 t.set(mController.getTime()); 545 if (t.minute >= 30) { 546 t.hour++; 547 t.minute = 0; 548 } else { 549 t.minute = 30; 550 } 551 mController.sendEventRelatedEvent( 552 this, EventType.CREATE_EVENT, -1, t.toMillis(true), 0, 0, 0, -1); 553 return true; 554 case R.id.action_settings: 555 mController.sendEvent(this, EventType.LAUNCH_SETTINGS, null, null, 0, 0); 556 return true; 557 case R.id.action_hide_controls: 558 mHideControls = !mHideControls; 559 item.setTitle(mHideControls ? mShowString : mHideString); 560 final ObjectAnimator slideAnimation = ObjectAnimator.ofInt(this, "controlsOffset", 561 mHideControls ? 0 : CONTROLS_ANIMATE_WIDTH, 562 mHideControls ? CONTROLS_ANIMATE_WIDTH : 0); 563 slideAnimation.setDuration(CONTROLS_ANIMATE_DURATION); 564 ObjectAnimator.setFrameDelay(0); 565 slideAnimation.start(); 566 return true; 567 default: 568 return false; 569 } 570 mController.sendEvent(this, EventType.GO_TO, t, null, -1, viewType); 571 return true; 572 } 573 574 /** 575 * Sets the offset of the controls on the right for animating them off/on 576 * screen. ProGuard strips this if it's not in proguard.flags 577 * 578 * @param controlsOffset The current offset in pixels 579 */ 580 public void setControlsOffset(int controlsOffset) { 581 mMiniMonth.setTranslationX(controlsOffset); 582 mCalendarsList.setTranslationX(controlsOffset); 583 mHomeTime.setTranslationX(controlsOffset); 584 mControlsParams.width = Math.max(0, CONTROLS_ANIMATE_WIDTH - controlsOffset); 585 mMiniMonthContainer.setLayoutParams(mControlsParams); 586 } 587 588 @Override 589 public void onSharedPreferenceChanged(SharedPreferences prefs, String key) { 590 if (key.equals(GeneralPreferences.KEY_WEEK_START_DAY)) { 591 if (mPaused) { 592 mUpdateOnResume = true; 593 } else { 594 initFragments(mController.getTime(), mController.getViewType(), null); 595 } 596 } 597 } 598 599 private void setMainPane( 600 FragmentTransaction ft, int viewId, int viewType, long timeMillis, boolean force) { 601 if (mOnSaveInstanceStateCalled) { 602 return; 603 } 604 if (!force && mCurrentView == viewType) { 605 return; 606 } 607 608 // Remove this when transition to and from month view looks fine. 609 boolean doTransition = viewType != ViewType.MONTH && mCurrentView != ViewType.MONTH; 610 611 if (viewType != mCurrentView) { 612 // The rules for this previous view are different than the 613 // controller's and are used for intercepting the back button. 614 if (mCurrentView != ViewType.EDIT && mCurrentView > 0) { 615 mPreviousView = mCurrentView; 616 } 617 mCurrentView = viewType; 618 } 619 // Create new fragment 620 Fragment frag = null; 621 Fragment secFrag = null; 622 switch (viewType) { 623 case ViewType.AGENDA: 624 if (mActionBar != null && (mActionBar.getSelectedTab() != mAgendaTab)) { 625 mActionBar.selectTab(mAgendaTab); 626 } 627 frag = new AgendaFragment(timeMillis, false); 628 break; 629 case ViewType.DAY: 630 if (mActionBar != null && (mActionBar.getSelectedTab() != mDayTab)) { 631 mActionBar.selectTab(mDayTab); 632 } 633 frag = new DayFragment(timeMillis, 1); 634 break; 635 case ViewType.WEEK: 636 if (mActionBar != null && (mActionBar.getSelectedTab() != mWeekTab)) { 637 mActionBar.selectTab(mWeekTab); 638 } 639 frag = new DayFragment(timeMillis, 7); 640 break; 641 case ViewType.MONTH: 642 if (mActionBar != null && (mActionBar.getSelectedTab() != mMonthTab)) { 643 mActionBar.selectTab(mMonthTab); 644 } 645 frag = new MonthByWeekFragment(timeMillis, false); 646 if (mShowAgendaWithMonth) { 647 secFrag = new AgendaFragment(timeMillis, false); 648 } 649 break; 650 default: 651 throw new IllegalArgumentException( 652 "Must be Agenda, Day, Week, or Month ViewType, not " + viewType); 653 } 654 655 // Clear unnecessary buttons from the option menu when switching from the agenda view 656 if (viewType != ViewType.AGENDA) { 657 clearOptionsMenu(); 658 } 659 660 boolean doCommit = false; 661 if (ft == null) { 662 doCommit = true; 663 ft = getFragmentManager().beginTransaction(); 664 } 665 666 if (doTransition) { 667 ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE); 668 } 669 670 ft.replace(viewId, frag); 671 if (mShowAgendaWithMonth) { 672 673 // Show/hide secondary fragment 674 675 if (secFrag != null) { 676 ft.replace(R.id.secondary_pane, secFrag); 677 mSecondaryPane.setVisibility(View.VISIBLE); 678 } else { 679 mSecondaryPane.setVisibility(View.GONE); 680 Fragment f = getFragmentManager().findFragmentById(R.id.secondary_pane); 681 if (f != null) { 682 ft.remove(f); 683 } 684 mController.deregisterEventHandler(R.id.secondary_pane); 685 } 686 } 687 if (DEBUG) { 688 Log.d(TAG, "Adding handler with viewId " + viewId + " and type " + viewType); 689 } 690 // If the key is already registered this will replace it 691 mController.registerEventHandler(viewId, (EventHandler) frag); 692 if (secFrag != null) { 693 mController.registerEventHandler(viewId, (EventHandler) secFrag); 694 } 695 696 if (doCommit) { 697 Log.d(TAG, "setMainPane AllInOne=" + this + " finishing:" + this.isFinishing()); 698 ft.commit(); 699 } 700 } 701 702 private void setTitleInActionBar(EventInfo event) { 703 if (event.eventType != EventType.UPDATE_TITLE || mActionBar == null) { 704 return; 705 } 706 707 final long start = event.startTime.toMillis(false /* use isDst */); 708 final long end; 709 if (event.endTime != null) { 710 end = event.endTime.toMillis(false /* use isDst */); 711 } else { 712 end = start; 713 } 714 715 final String msg = Utils.formatDateRange(this, start, end, (int) event.extraLong); 716 CharSequence oldDate = mDateRange.getText(); 717 mDateRange.setText(msg); 718 if (!TextUtils.equals(oldDate, msg)) { 719 mDateRange.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_FOCUSED); 720 } 721 } 722 723 private void updateHomeClock() { 724 mTimeZone = Utils.getTimeZone(this, mHomeTimeUpdater); 725 if (mIsMultipane && (mCurrentView == ViewType.DAY || mCurrentView == ViewType.WEEK) 726 && !TextUtils.equals(mTimeZone, Time.getCurrentTimezone())) { 727 Time time = new Time(mTimeZone); 728 time.setToNow(); 729 long millis = time.toMillis(true); 730 boolean isDST = time.isDst != 0; 731 int flags = DateUtils.FORMAT_SHOW_TIME; 732 if (DateFormat.is24HourFormat(this)) { 733 flags |= DateUtils.FORMAT_24HOUR; 734 } 735 // Formats the time as 736 String timeString = (new StringBuilder( 737 Utils.formatDateRange(this, millis, millis, flags))).append(" ").append( 738 TimeZone.getTimeZone(mTimeZone).getDisplayName( 739 isDST, TimeZone.SHORT, Locale.getDefault())).toString(); 740 mHomeTime.setText(timeString); 741 mHomeTime.setVisibility(View.VISIBLE); 742 // Update when the minute changes 743 mHomeTime.postDelayed( 744 mHomeTimeUpdater, 745 DateUtils.MINUTE_IN_MILLIS - (millis % DateUtils.MINUTE_IN_MILLIS)); 746 } else { 747 mHomeTime.setVisibility(View.GONE); 748 } 749 } 750 751 @Override 752 public long getSupportedEventTypes() { 753 return EventType.GO_TO | EventType.VIEW_EVENT | EventType.UPDATE_TITLE; 754 } 755 756 @Override 757 public void handleEvent(EventInfo event) { 758 Log.d(TAG, "handleEvent AllInOne=" + this); 759 if (event.eventType == EventType.GO_TO) { 760 setMainPane( 761 null, R.id.main_pane, event.viewType, event.startTime.toMillis(false), false); 762 if (mSearchView != null) { 763 mSearchView.clearFocus(); 764 } 765 766 if (mShowCalendarControls) { 767 if (event.viewType == ViewType.MONTH || event.viewType == ViewType.AGENDA) { 768 // hide minimonth and calendar frag 769 mShowSideViews = false; 770 if (mControlsMenu != null) { 771 mControlsMenu.setVisible(false); 772 mControlsMenu.setEnabled(false); 773 774 if (!mHideControls) { 775 final ObjectAnimator slideAnimation = ObjectAnimator.ofInt(this, 776 "controlsOffset", 0, CONTROLS_ANIMATE_WIDTH); 777 slideAnimation.addListener(mSlideAnimationDoneListener); 778 slideAnimation.setDuration(220); 779 ObjectAnimator.setFrameDelay(0); 780 slideAnimation.start(); 781 } 782 } else { 783 mMiniMonth.setVisibility(View.GONE); 784 mCalendarsList.setVisibility(View.GONE); 785 mMiniMonthContainer.setVisibility(View.GONE); 786 } 787 } else { 788 // show minimonth and calendar frag 789 mShowSideViews = true; 790 mMiniMonth.setVisibility(View.VISIBLE); 791 mCalendarsList.setVisibility(View.VISIBLE); 792 mMiniMonthContainer.setVisibility(View.VISIBLE); 793 if (mControlsMenu != null) { 794 mControlsMenu.setVisible(true); 795 mControlsMenu.setEnabled(true); 796 if (!mHideControls && 797 (mController.getPreviousViewType() == ViewType.MONTH || 798 mController.getPreviousViewType() == ViewType.AGENDA)) { 799 final ObjectAnimator slideAnimation = ObjectAnimator.ofInt(this, 800 "controlsOffset", CONTROLS_ANIMATE_WIDTH, 0); 801 slideAnimation.setDuration(220); 802 ObjectAnimator.setFrameDelay(0); 803 slideAnimation.start(); 804 } 805 } 806 } 807 } 808 } else if (event.eventType == EventType.VIEW_EVENT) { 809 810 // If in Agenda view and "show_event_details_with_agenda" is "true", 811 // do not create the event info fragment here, it will be created by the Agenda 812 // fragment 813 814 if (mCurrentView == ViewType.AGENDA && mShowEventDetailsWithAgenda) { 815 if (event.selectedTime != null) { 816 mController.sendEvent(this, EventType.GO_TO, event.selectedTime, 817 event.selectedTime, event.id, ViewType.AGENDA); 818 } 819 } else { 820 if (mShowEventInfoFullScreen) { 821 // start event info as activity 822 Intent intent = new Intent(Intent.ACTION_VIEW); 823 Uri eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, event.id); 824 intent.setData(eventUri); 825 intent.setClassName(this, EventInfoActivity.class.getName()); 826 intent.setFlags(Intent.FLAG_ACTIVITY_REORDER_TO_FRONT | 827 Intent.FLAG_ACTIVITY_SINGLE_TOP); 828 intent.putExtra(EVENT_BEGIN_TIME, event.startTime.toMillis(false)); 829 intent.putExtra(EVENT_END_TIME, event.endTime.toMillis(false)); 830 intent.putExtra(EVENT_ATTENDEE_RESPONSE, (int)event.extraLong); 831 startActivity(intent); 832 } else { 833 // start event info as a dialog 834 EventInfoFragment fragment = new EventInfoFragment(this, 835 event.id, event.startTime.toMillis(false), 836 event.endTime.toMillis(false), (int) event.extraLong, true); 837 // TODO Fix the temp hack below: && mCurrentView != 838 // ViewType.AGENDA 839 if (event.selectedTime != null && mCurrentView != ViewType.AGENDA) { 840 mController.sendEvent(this, EventType.GO_TO, event.selectedTime, 841 event.selectedTime, -1, ViewType.DETAIL); 842 } 843 fragment.setDialogParams(event.x, event.y); 844 FragmentManager fm = getFragmentManager(); 845 FragmentTransaction ft = fm.beginTransaction(); 846 // if we have an old popup close it 847 Fragment fOld = fm.findFragmentByTag(EVENT_INFO_FRAGMENT_TAG); 848 if (fOld != null && fOld.isAdded()) { 849 ft.remove(fOld); 850 } 851 ft.add(fragment, EVENT_INFO_FRAGMENT_TAG); 852 ft.commit(); 853 } 854 } 855 } else if (event.eventType == EventType.UPDATE_TITLE) { 856 setTitleInActionBar(event); 857 } 858 updateHomeClock(); 859 } 860 861 @Override 862 public void eventsChanged() { 863 mController.sendEvent(this, EventType.EVENTS_CHANGED, null, null, -1, ViewType.CURRENT); 864 } 865 866 @Override 867 public boolean onQueryTextChange(String newText) { 868 return false; 869 } 870 871 @Override 872 public boolean onQueryTextSubmit(String query) { 873 if ("TARDIS".equalsIgnoreCase(query)) { 874 Utils.tardis(); 875 } 876 mSearchView.clearFocus(); 877 mController.sendEvent(this, EventType.SEARCH, null, null, -1, ViewType.CURRENT, -1, query, 878 getComponentName()); 879 return false; 880 } 881 882 @Override 883 public void onTabSelected(Tab tab, FragmentTransaction ft) { 884 Log.w(TAG, "TabSelected AllInOne=" + this + " finishing:" + this.isFinishing()); 885 if (tab == mDayTab && mCurrentView != ViewType.DAY) { 886 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.DAY); 887 } else if (tab == mWeekTab && mCurrentView != ViewType.WEEK) { 888 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.WEEK); 889 } else if (tab == mMonthTab && mCurrentView != ViewType.MONTH) { 890 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.MONTH); 891 } else if (tab == mAgendaTab && mCurrentView != ViewType.AGENDA) { 892 mController.sendEvent(this, EventType.GO_TO, null, null, -1, ViewType.AGENDA); 893 } else { 894 Log.w(TAG, "TabSelected event from unknown tab: " 895 + (tab == null ? "null" : tab.getText())); 896 Log.w(TAG, "CurrentView:" + mCurrentView + " Tab:" + tab.toString() + " Day:" + mDayTab 897 + " Week:" + mWeekTab + " Month:" + mMonthTab + " Agenda:" + mAgendaTab); 898 } 899 } 900 901 @Override 902 public void onTabReselected(Tab tab, FragmentTransaction ft) { 903 } 904 905 @Override 906 public void onTabUnselected(Tab tab, FragmentTransaction ft) { 907 } 908} 909