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