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