• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2011 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 
17 package com.android.launcher2;
18 
19 import android.animation.Animator;
20 import android.animation.AnimatorListenerAdapter;
21 import android.animation.AnimatorSet;
22 import android.animation.ObjectAnimator;
23 import android.content.Context;
24 import android.content.res.Resources;
25 import android.util.AttributeSet;
26 import android.util.Log;
27 import android.view.LayoutInflater;
28 import android.view.MotionEvent;
29 import android.view.View;
30 import android.view.ViewGroup;
31 import android.widget.FrameLayout;
32 import android.widget.LinearLayout;
33 import android.widget.TabHost;
34 import android.widget.TabWidget;
35 import android.widget.TextView;
36 
37 import com.android.launcher.R;
38 
39 import java.util.ArrayList;
40 
41 public class AppsCustomizeTabHost extends TabHost implements LauncherTransitionable,
42         TabHost.OnTabChangeListener  {
43     static final String LOG_TAG = "AppsCustomizeTabHost";
44 
45     private static final String APPS_TAB_TAG = "APPS";
46     private static final String WIDGETS_TAB_TAG = "WIDGETS";
47 
48     private final LayoutInflater mLayoutInflater;
49     private ViewGroup mTabs;
50     private ViewGroup mTabsContainer;
51     private AppsCustomizePagedView mAppsCustomizePane;
52     private boolean mSuppressContentCallback = false;
53     private FrameLayout mAnimationBuffer;
54     private LinearLayout mContent;
55 
56     private boolean mInTransition;
57     private boolean mTransitioningToWorkspace;
58     private boolean mResetAfterTransition;
59     private Runnable mRelayoutAndMakeVisible;
60 
61     private Launcher mLauncher;
62 
AppsCustomizeTabHost(Context context, AttributeSet attrs)63     public AppsCustomizeTabHost(Context context, AttributeSet attrs) {
64         super(context, attrs);
65         mLayoutInflater = LayoutInflater.from(context);
66         mRelayoutAndMakeVisible = new Runnable() {
67                 public void run() {
68                     mTabs.requestLayout();
69                     mTabsContainer.setAlpha(1f);
70                 }
71             };
72     }
73 
setup(Launcher launcher)74     public void setup(Launcher launcher) {
75         mLauncher = launcher;
76     }
77 
78     /**
79      * Convenience methods to select specific tabs.  We want to set the content type immediately
80      * in these cases, but we note that we still call setCurrentTabByTag() so that the tab view
81      * reflects the new content (but doesn't do the animation and logic associated with changing
82      * tabs manually).
83      */
setContentTypeImmediate(AppsCustomizePagedView.ContentType type)84     private void setContentTypeImmediate(AppsCustomizePagedView.ContentType type) {
85         onTabChangedStart();
86         onTabChangedEnd(type);
87     }
selectAppsTab()88     void selectAppsTab() {
89         setContentTypeImmediate(AppsCustomizePagedView.ContentType.Applications);
90         setCurrentTabByTag(APPS_TAB_TAG);
91     }
selectWidgetsTab()92     void selectWidgetsTab() {
93         setContentTypeImmediate(AppsCustomizePagedView.ContentType.Widgets);
94         setCurrentTabByTag(WIDGETS_TAB_TAG);
95     }
96 
97     /**
98      * Setup the tab host and create all necessary tabs.
99      */
100     @Override
onFinishInflate()101     protected void onFinishInflate() {
102         // Setup the tab host
103         setup();
104 
105         final ViewGroup tabsContainer = (ViewGroup) findViewById(R.id.tabs_container);
106         final TabWidget tabs = getTabWidget();
107         final AppsCustomizePagedView appsCustomizePane = (AppsCustomizePagedView)
108                 findViewById(R.id.apps_customize_pane_content);
109         mTabs = tabs;
110         mTabsContainer = tabsContainer;
111         mAppsCustomizePane = appsCustomizePane;
112         mAnimationBuffer = (FrameLayout) findViewById(R.id.animation_buffer);
113         mContent = (LinearLayout) findViewById(R.id.apps_customize_content);
114         if (tabs == null || mAppsCustomizePane == null) throw new Resources.NotFoundException();
115 
116         // Configure the tabs content factory to return the same paged view (that we change the
117         // content filter on)
118         TabContentFactory contentFactory = new TabContentFactory() {
119             public View createTabContent(String tag) {
120                 return appsCustomizePane;
121             }
122         };
123 
124         // Create the tabs
125         TextView tabView;
126         String label;
127         label = getContext().getString(R.string.all_apps_button_label);
128         tabView = (TextView) mLayoutInflater.inflate(R.layout.tab_widget_indicator, tabs, false);
129         tabView.setText(label);
130         tabView.setContentDescription(label);
131         addTab(newTabSpec(APPS_TAB_TAG).setIndicator(tabView).setContent(contentFactory));
132         label = getContext().getString(R.string.widgets_tab_label);
133         tabView = (TextView) mLayoutInflater.inflate(R.layout.tab_widget_indicator, tabs, false);
134         tabView.setText(label);
135         tabView.setContentDescription(label);
136         addTab(newTabSpec(WIDGETS_TAB_TAG).setIndicator(tabView).setContent(contentFactory));
137         setOnTabChangedListener(this);
138 
139         // Setup the key listener to jump between the last tab view and the market icon
140         AppsCustomizeTabKeyEventListener keyListener = new AppsCustomizeTabKeyEventListener();
141         View lastTab = tabs.getChildTabViewAt(tabs.getTabCount() - 1);
142         lastTab.setOnKeyListener(keyListener);
143         View shopButton = findViewById(R.id.market_button);
144         shopButton.setOnKeyListener(keyListener);
145 
146         // Hide the tab bar until we measure
147         mTabsContainer.setAlpha(0f);
148     }
149 
150     @Override
onMeasure(int widthMeasureSpec, int heightMeasureSpec)151     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
152         boolean remeasureTabWidth = (mTabs.getLayoutParams().width <= 0);
153         super.onMeasure(widthMeasureSpec, heightMeasureSpec);
154 
155         // Set the width of the tab list to the content width
156         if (remeasureTabWidth) {
157             int contentWidth = mAppsCustomizePane.getPageContentWidth();
158             if (contentWidth > 0 && mTabs.getLayoutParams().width != contentWidth) {
159                 // Set the width and show the tab bar
160                 mTabs.getLayoutParams().width = contentWidth;
161                 post(mRelayoutAndMakeVisible);
162             }
163         }
164         super.onMeasure(widthMeasureSpec, heightMeasureSpec);
165     }
166 
onInterceptTouchEvent(MotionEvent ev)167      public boolean onInterceptTouchEvent(MotionEvent ev) {
168          // If we are mid transitioning to the workspace, then intercept touch events here so we
169          // can ignore them, otherwise we just let all apps handle the touch events.
170          if (mInTransition && mTransitioningToWorkspace) {
171              return true;
172          }
173          return super.onInterceptTouchEvent(ev);
174      };
175 
176     @Override
onTouchEvent(MotionEvent event)177     public boolean onTouchEvent(MotionEvent event) {
178         // Allow touch events to fall through to the workspace if we are transitioning there
179         if (mInTransition && mTransitioningToWorkspace) {
180             return super.onTouchEvent(event);
181         }
182 
183         // Intercept all touch events up to the bottom of the AppsCustomizePane so they do not fall
184         // through to the workspace and trigger showWorkspace()
185         if (event.getY() < mAppsCustomizePane.getBottom()) {
186             return true;
187         }
188         return super.onTouchEvent(event);
189     }
190 
onTabChangedStart()191     private void onTabChangedStart() {
192         mAppsCustomizePane.hideScrollingIndicator(false);
193     }
194 
reloadCurrentPage()195     private void reloadCurrentPage() {
196         if (!LauncherApplication.isScreenLarge()) {
197             mAppsCustomizePane.flashScrollingIndicator(true);
198         }
199         mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage());
200         mAppsCustomizePane.requestFocus();
201     }
202 
onTabChangedEnd(AppsCustomizePagedView.ContentType type)203     private void onTabChangedEnd(AppsCustomizePagedView.ContentType type) {
204         mAppsCustomizePane.setContentType(type);
205     }
206 
207     @Override
onTabChanged(String tabId)208     public void onTabChanged(String tabId) {
209         final AppsCustomizePagedView.ContentType type = getContentTypeForTabTag(tabId);
210         if (mSuppressContentCallback) {
211             mSuppressContentCallback = false;
212             return;
213         }
214 
215         // Animate the changing of the tab content by fading pages in and out
216         final Resources res = getResources();
217         final int duration = res.getInteger(R.integer.config_tabTransitionDuration);
218 
219         // We post a runnable here because there is a delay while the first page is loading and
220         // the feedback from having changed the tab almost feels better than having it stick
221         post(new Runnable() {
222             @Override
223             public void run() {
224                 if (mAppsCustomizePane.getMeasuredWidth() <= 0 ||
225                         mAppsCustomizePane.getMeasuredHeight() <= 0) {
226                     reloadCurrentPage();
227                     return;
228                 }
229 
230                 // Take the visible pages and re-parent them temporarily to mAnimatorBuffer
231                 // and then cross fade to the new pages
232                 int[] visiblePageRange = new int[2];
233                 mAppsCustomizePane.getVisiblePages(visiblePageRange);
234                 if (visiblePageRange[0] == -1 && visiblePageRange[1] == -1) {
235                     // If we can't get the visible page ranges, then just skip the animation
236                     reloadCurrentPage();
237                     return;
238                 }
239                 ArrayList<View> visiblePages = new ArrayList<View>();
240                 for (int i = visiblePageRange[0]; i <= visiblePageRange[1]; i++) {
241                     visiblePages.add(mAppsCustomizePane.getPageAt(i));
242                 }
243 
244                 // We want the pages to be rendered in exactly the same way as they were when
245                 // their parent was mAppsCustomizePane -- so set the scroll on mAnimationBuffer
246                 // to be exactly the same as mAppsCustomizePane, and below, set the left/top
247                 // parameters to be correct for each of the pages
248                 mAnimationBuffer.scrollTo(mAppsCustomizePane.getScrollX(), 0);
249 
250                 // mAppsCustomizePane renders its children in reverse order, so
251                 // add the pages to mAnimationBuffer in reverse order to match that behavior
252                 for (int i = visiblePages.size() - 1; i >= 0; i--) {
253                     View child = visiblePages.get(i);
254                     if (child instanceof PagedViewCellLayout) {
255                         ((PagedViewCellLayout) child).resetChildrenOnKeyListeners();
256                     } else if (child instanceof PagedViewGridLayout) {
257                         ((PagedViewGridLayout) child).resetChildrenOnKeyListeners();
258                     }
259                     PagedViewWidget.setDeletePreviewsWhenDetachedFromWindow(false);
260                     mAppsCustomizePane.removeView(child);
261                     PagedViewWidget.setDeletePreviewsWhenDetachedFromWindow(true);
262                     mAnimationBuffer.setAlpha(1f);
263                     mAnimationBuffer.setVisibility(View.VISIBLE);
264                     LayoutParams p = new FrameLayout.LayoutParams(child.getMeasuredWidth(),
265                             child.getMeasuredHeight());
266                     p.setMargins((int) child.getLeft(), (int) child.getTop(), 0, 0);
267                     mAnimationBuffer.addView(child, p);
268                 }
269 
270                 // Toggle the new content
271                 onTabChangedStart();
272                 onTabChangedEnd(type);
273 
274                 // Animate the transition
275                 ObjectAnimator outAnim = ObjectAnimator.ofFloat(mAnimationBuffer, "alpha", 0f);
276                 outAnim.addListener(new AnimatorListenerAdapter() {
277                     @Override
278                     public void onAnimationEnd(Animator animation) {
279                         mAnimationBuffer.setVisibility(View.GONE);
280                         mAnimationBuffer.removeAllViews();
281                     }
282                     @Override
283                     public void onAnimationCancel(Animator animation) {
284                         mAnimationBuffer.setVisibility(View.GONE);
285                         mAnimationBuffer.removeAllViews();
286                     }
287                 });
288                 ObjectAnimator inAnim = ObjectAnimator.ofFloat(mAppsCustomizePane, "alpha", 1f);
289                 inAnim.addListener(new AnimatorListenerAdapter() {
290                     @Override
291                     public void onAnimationEnd(Animator animation) {
292                         reloadCurrentPage();
293                     }
294                 });
295                 AnimatorSet animSet = new AnimatorSet();
296                 animSet.playTogether(outAnim, inAnim);
297                 animSet.setDuration(duration);
298                 animSet.start();
299             }
300         });
301     }
302 
setCurrentTabFromContent(AppsCustomizePagedView.ContentType type)303     public void setCurrentTabFromContent(AppsCustomizePagedView.ContentType type) {
304         mSuppressContentCallback = true;
305         setCurrentTabByTag(getTabTagForContentType(type));
306     }
307 
308     /**
309      * Returns the content type for the specified tab tag.
310      */
getContentTypeForTabTag(String tag)311     public AppsCustomizePagedView.ContentType getContentTypeForTabTag(String tag) {
312         if (tag.equals(APPS_TAB_TAG)) {
313             return AppsCustomizePagedView.ContentType.Applications;
314         } else if (tag.equals(WIDGETS_TAB_TAG)) {
315             return AppsCustomizePagedView.ContentType.Widgets;
316         }
317         return AppsCustomizePagedView.ContentType.Applications;
318     }
319 
320     /**
321      * Returns the tab tag for a given content type.
322      */
getTabTagForContentType(AppsCustomizePagedView.ContentType type)323     public String getTabTagForContentType(AppsCustomizePagedView.ContentType type) {
324         if (type == AppsCustomizePagedView.ContentType.Applications) {
325             return APPS_TAB_TAG;
326         } else if (type == AppsCustomizePagedView.ContentType.Widgets) {
327             return WIDGETS_TAB_TAG;
328         }
329         return APPS_TAB_TAG;
330     }
331 
332     /**
333      * Disable focus on anything under this view in the hierarchy if we are not visible.
334      */
335     @Override
getDescendantFocusability()336     public int getDescendantFocusability() {
337         if (getVisibility() != View.VISIBLE) {
338             return ViewGroup.FOCUS_BLOCK_DESCENDANTS;
339         }
340         return super.getDescendantFocusability();
341     }
342 
reset()343     void reset() {
344         if (mInTransition) {
345             // Defer to after the transition to reset
346             mResetAfterTransition = true;
347         } else {
348             // Reset immediately
349             mAppsCustomizePane.reset();
350         }
351     }
352 
enableAndBuildHardwareLayer()353     private void enableAndBuildHardwareLayer() {
354         // isHardwareAccelerated() checks if we're attached to a window and if that
355         // window is HW accelerated-- we were sometimes not attached to a window
356         // and buildLayer was throwing an IllegalStateException
357         if (isHardwareAccelerated()) {
358             // Turn on hardware layers for performance
359             setLayerType(LAYER_TYPE_HARDWARE, null);
360 
361             // force building the layer, so you don't get a blip early in an animation
362             // when the layer is created layer
363             buildLayer();
364 
365             // Let the GC system know that now is a good time to do any garbage
366             // collection; makes it less likely we'll get a GC during the all apps
367             // to workspace animation
368             System.gc();
369         }
370     }
371 
372     @Override
getContent()373     public View getContent() {
374         return mContent;
375     }
376 
377     /* LauncherTransitionable overrides */
378     @Override
onLauncherTransitionPrepare(Launcher l, boolean animated, boolean toWorkspace)379     public void onLauncherTransitionPrepare(Launcher l, boolean animated, boolean toWorkspace) {
380         mAppsCustomizePane.onLauncherTransitionPrepare(l, animated, toWorkspace);
381         mInTransition = true;
382         mTransitioningToWorkspace = toWorkspace;
383 
384         if (toWorkspace) {
385             // Going from All Apps -> Workspace
386             setVisibilityOfSiblingsWithLowerZOrder(VISIBLE);
387             // Stop the scrolling indicator - we don't want All Apps to be invalidating itself
388             // during the transition, especially since it has a hardware layer set on it
389             mAppsCustomizePane.cancelScrollingIndicatorAnimations();
390         } else {
391             // Going from Workspace -> All Apps
392             mContent.setVisibility(VISIBLE);
393 
394             // Make sure the current page is loaded (we start loading the side pages after the
395             // transition to prevent slowing down the animation)
396             mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage(), true);
397 
398             if (!LauncherApplication.isScreenLarge()) {
399                 mAppsCustomizePane.showScrollingIndicator(true);
400             }
401         }
402 
403         if (mResetAfterTransition) {
404             mAppsCustomizePane.reset();
405             mResetAfterTransition = false;
406         }
407     }
408 
409     @Override
onLauncherTransitionStart(Launcher l, boolean animated, boolean toWorkspace)410     public void onLauncherTransitionStart(Launcher l, boolean animated, boolean toWorkspace) {
411         if (animated) {
412             enableAndBuildHardwareLayer();
413         }
414     }
415 
416     @Override
onLauncherTransitionStep(Launcher l, float t)417     public void onLauncherTransitionStep(Launcher l, float t) {
418         // Do nothing
419     }
420 
421     @Override
onLauncherTransitionEnd(Launcher l, boolean animated, boolean toWorkspace)422     public void onLauncherTransitionEnd(Launcher l, boolean animated, boolean toWorkspace) {
423         mAppsCustomizePane.onLauncherTransitionEnd(l, animated, toWorkspace);
424         mInTransition = false;
425         if (animated) {
426             setLayerType(LAYER_TYPE_NONE, null);
427         }
428 
429         if (!toWorkspace) {
430             // Going from Workspace -> All Apps
431             setVisibilityOfSiblingsWithLowerZOrder(INVISIBLE);
432 
433             // Dismiss the workspace cling and show the all apps cling (if not already shown)
434             l.dismissWorkspaceCling(null);
435             mAppsCustomizePane.showAllAppsCling();
436             // Make sure adjacent pages are loaded (we wait until after the transition to
437             // prevent slowing down the animation)
438             mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage());
439 
440             if (!LauncherApplication.isScreenLarge()) {
441                 mAppsCustomizePane.hideScrollingIndicator(false);
442             }
443         }
444     }
445 
setVisibilityOfSiblingsWithLowerZOrder(int visibility)446     private void setVisibilityOfSiblingsWithLowerZOrder(int visibility) {
447         ViewGroup parent = (ViewGroup) getParent();
448         if (parent == null) return;
449 
450         final int count = parent.getChildCount();
451         if (!isChildrenDrawingOrderEnabled()) {
452             for (int i = 0; i < count; i++) {
453                 final View child = parent.getChildAt(i);
454                 if (child == this) {
455                     break;
456                 } else {
457                     if (child.getVisibility() == GONE) {
458                         continue;
459                     }
460                     child.setVisibility(visibility);
461                 }
462             }
463         } else {
464             throw new RuntimeException("Failed; can't get z-order of views");
465         }
466     }
467 
onWindowVisible()468     public void onWindowVisible() {
469         if (getVisibility() == VISIBLE) {
470             mContent.setVisibility(VISIBLE);
471             // We unload the widget previews when the UI is hidden, so need to reload pages
472             // Load the current page synchronously, and the neighboring pages asynchronously
473             mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage(), true);
474             mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage());
475         }
476     }
477 
onTrimMemory()478     public void onTrimMemory() {
479         mContent.setVisibility(GONE);
480         // Clear the widget pages of all their subviews - this will trigger the widget previews
481         // to delete their bitmaps
482         mAppsCustomizePane.clearAllWidgetPages();
483     }
484 
isTransitioning()485     boolean isTransitioning() {
486         return mInTransition;
487     }
488 }
489