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