/* * Copyright (C) 2011 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.launcher3; import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.animation.AnimatorSet; import android.animation.ObjectAnimator; import android.content.Context; import android.content.res.Resources; import android.graphics.Color; import android.graphics.Rect; import android.os.Trace; import android.util.AttributeSet; import android.view.LayoutInflater; import android.view.MotionEvent; import android.view.View; import android.view.ViewGroup; import android.widget.FrameLayout; import android.widget.LinearLayout; import android.widget.TabHost; import android.widget.TabWidget; import android.widget.TextView; import java.util.ArrayList; import com.mediatek.launcher3.ext.LauncherLog; import com.mediatek.common.featureoption.FeatureOption; //add by messi for HOSIN_CUST_LAUNCHER_MESSI start import java.io.ByteArrayOutputStream; import android.graphics.Bitmap.CompressFormat; import android.net.Uri; import android.content.ContentValues; import android.database.Cursor; import java.io.InputStream; import java.io.ByteArrayInputStream; import android.graphics.drawable.Drawable; import android.graphics.drawable.BitmapDrawable; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.util.Log; import android.graphics.PorterDuff; //add by messi for HOSIN_CUST_LAUNCHER_MESSI end public class AppsCustomizeTabHost extends TabHost implements LauncherTransitionable, TabHost.OnTabChangeListener, Insettable { static final String LOG_TAG = "AppsCustomizeTabHost"; private static final String APPS_TAB_TAG = "APPS"; private static final String WIDGETS_TAB_TAG = "WIDGETS"; private final LayoutInflater mLayoutInflater; private ViewGroup mTabs; private ViewGroup mTabsContainer; private AppsCustomizePagedView mAppsCustomizePane; private FrameLayout mAnimationBuffer; private LinearLayout mContent; private boolean mInTransition; private boolean mTransitioningToWorkspace; private boolean mResetAfterTransition; private Runnable mRelayoutAndMakeVisible; private final Rect mInsets = new Rect(); private Context mContext; public AppsCustomizeTabHost(Context context, AttributeSet attrs) { super(context, attrs); mContext = context; mLayoutInflater = LayoutInflater.from(context); mRelayoutAndMakeVisible = new Runnable() { public void run() { mTabs.requestLayout(); mTabsContainer.setAlpha(1f); } }; } /** * Convenience methods to select specific tabs. We want to set the content type immediately * in these cases, but we note that we still call setCurrentTabByTag() so that the tab view * reflects the new content (but doesn't do the animation and logic associated with changing * tabs manually). */ void setContentTypeImmediate(AppsCustomizePagedView.ContentType type) { setOnTabChangedListener(null); onTabChangedStart(); onTabChangedEnd(type); setCurrentTabByTag(getTabTagForContentType(type)); setOnTabChangedListener(this); } @Override public void setInsets(Rect insets) { mInsets.set(insets); FrameLayout.LayoutParams flp = (LayoutParams) mContent.getLayoutParams(); flp.topMargin = insets.top; flp.bottomMargin = insets.bottom; flp.leftMargin = insets.left; flp.rightMargin = insets.right; mContent.setLayoutParams(flp); } /** * Setup the tab host and create all necessary tabs. */ @Override protected void onFinishInflate() { // Setup the tab host setup(); final ViewGroup tabsContainer = (ViewGroup) findViewById(R.id.tabs_container); final TabWidget tabs = getTabWidget(); final AppsCustomizePagedView appsCustomizePane = (AppsCustomizePagedView) findViewById(R.id.apps_customize_pane_content); mTabs = tabs; mTabsContainer = tabsContainer; mAppsCustomizePane = appsCustomizePane; mAnimationBuffer = (FrameLayout) findViewById(R.id.animation_buffer); mContent = (LinearLayout) findViewById(R.id.apps_customize_content); if (tabs == null || mAppsCustomizePane == null) throw new Resources.NotFoundException(); // Configure the tabs content factory to return the same paged view (that we change the // content filter on) TabContentFactory contentFactory = new TabContentFactory() { public View createTabContent(String tag) { return appsCustomizePane; } }; // Create the tabs TextView tabView; String label; label = getContext().getString(R.string.all_apps_button_label); tabView = (TextView) mLayoutInflater.inflate(R.layout.tab_widget_indicator, tabs, false); tabView.setText(label); tabView.setContentDescription(label); addTab(newTabSpec(APPS_TAB_TAG).setIndicator(tabView).setContent(contentFactory)); label = getContext().getString(R.string.widgets_tab_label); tabView = (TextView) mLayoutInflater.inflate(R.layout.tab_widget_indicator, tabs, false); tabView.setText(label); tabView.setContentDescription(label); addTab(newTabSpec(WIDGETS_TAB_TAG).setIndicator(tabView).setContent(contentFactory)); setOnTabChangedListener(this); // Setup the key listener to jump between the last tab view and the market icon AppsCustomizeTabKeyEventListener keyListener = new AppsCustomizeTabKeyEventListener(); View lastTab = tabs.getChildTabViewAt(tabs.getTabCount() - 1); lastTab.setOnKeyListener(keyListener); View shopButton = findViewById(R.id.market_button); shopButton.setOnKeyListener(keyListener); // Hide the tab bar until we measure mTabsContainer.setAlpha(0f); } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { boolean remeasureTabWidth = (mTabs.getLayoutParams().width <= 0); super.onMeasure(widthMeasureSpec, heightMeasureSpec); if (LauncherLog.DEBUG_LAYOUT) { LauncherLog.d(LOG_TAG, "onMeasure end: remeasureTabWidth = " + remeasureTabWidth + ", widthMeasureSpec = " + widthMeasureSpec + ", heightMeasureSpec = " + heightMeasureSpec + ", this = " + this); } // Set the width of the tab list to the content width if (remeasureTabWidth) { int contentWidth = mAppsCustomizePane.getPageContentWidth(); if (contentWidth > 0 && mTabs.getLayoutParams().width != contentWidth) { // Set the width and show the tab bar mTabs.getLayoutParams().width = contentWidth; mRelayoutAndMakeVisible.run(); } super.onMeasure(widthMeasureSpec, heightMeasureSpec); } } public boolean onInterceptTouchEvent(MotionEvent ev) { // If we are mid transitioning to the workspace, then intercept touch events here so we // can ignore them, otherwise we just let all apps handle the touch events. if (mInTransition && mTransitioningToWorkspace) { return true; } return super.onInterceptTouchEvent(ev); }; @Override public boolean onTouchEvent(MotionEvent event) { if (LauncherLog.DEBUG) { LauncherLog.d(LOG_TAG, "onTouchEvent: action = " + event.getAction() + ", y = " + event.getY()); } // Allow touch events to fall through to the workspace if we are transitioning there if (mInTransition && mTransitioningToWorkspace) { return super.onTouchEvent(event); } // Intercept all touch events up to the bottom of the AppsCustomizePane so they do not fall // through to the workspace and trigger showWorkspace() if (event.getY() < mAppsCustomizePane.getBottom()) { return true; } return super.onTouchEvent(event); } private void onTabChangedStart() { } private void reloadCurrentPage() { mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage()); mAppsCustomizePane.requestFocus(); } private void onTabChangedEnd(AppsCustomizePagedView.ContentType type) { int bgAlpha = (int) (255 * (getResources().getInteger( R.integer.config_appsCustomizeSpringLoadedBgAlpha) / 100f)); //by qinjian update for remove main menu background change black and no transparent if(FeatureOption.HOSIN_MAIN_MENU_BG){ // setBackgroundColor(Color.argb(bgAlpha, 0, 0, 0)); } else if(FeatureOption.HOSIN_CUST_LAUNCHER_MESSI){ Uri url = Uri.parse("content://"+"com.android.launcher3.settings"+"/"+"wallpaper"); Cursor cursor = null; String[] array = mContext.getResources().getStringArray(R.array.wallpaper_chooser); cursor = mContext.getContentResolver().query(url,null,"_id=?",new String[]{"2"},null); byte[] bytes = null; int flag = 0; if(cursor!=null&&cursor.moveToFirst()){ do{ bytes = cursor.getBlob(cursor.getColumnIndex("wallpaper")); flag = cursor.getInt(cursor.getColumnIndex("flag")); }while(cursor.moveToNext()); } if(bytes!=null&&flag==2){ InputStream is = new ByteArrayInputStream(bytes); Bitmap bmp = BitmapFactory.decodeStream(is); Drawable d = new BitmapDrawable(bmp); d.setColorFilter(Color.LTGRAY,PorterDuff.Mode.MULTIPLY); setBackgroundDrawable(d); } else{ setBackgroundColor(Color.argb(bgAlpha, 0, 0, 0)); } } else{ setBackgroundColor(Color.argb(bgAlpha, 0, 0, 0)); } // by qinjan update for background end mAppsCustomizePane.setContentType(type); } @Override public void onTabChanged(String tabId) { Trace.traceBegin(Trace.TRACE_TAG_INPUT, "onTabChanged"); final AppsCustomizePagedView.ContentType type = getContentTypeForTabTag(tabId); if (LauncherLog.DEBUG) { LauncherLog.d(LOG_TAG, "onTabChanged: tabId = " + tabId + ", type = " + type); } // Animate the changing of the tab content by fading pages in and out final Resources res = getResources(); final int duration = res.getInteger(R.integer.config_tabTransitionDuration); // We post a runnable here because there is a delay while the first page is loading and // the feedback from having changed the tab almost feels better than having it stick post(new Runnable() { @Override public void run() { if (mAppsCustomizePane.getMeasuredWidth() <= 0 || mAppsCustomizePane.getMeasuredHeight() <= 0) { reloadCurrentPage(); return; } // Take the visible pages and re-parent them temporarily to mAnimatorBuffer // and then cross fade to the new pages int[] visiblePageRange = new int[2]; mAppsCustomizePane.getVisiblePages(visiblePageRange); if (visiblePageRange[0] == -1 && visiblePageRange[1] == -1) { // If we can't get the visible page ranges, then just skip the animation reloadCurrentPage(); return; } ArrayList<View> visiblePages = new ArrayList<View>(); for (int i = visiblePageRange[0]; i <= visiblePageRange[1]; i++) { visiblePages.add(mAppsCustomizePane.getPageAt(i)); } // We want the pages to be rendered in exactly the same way as they were when // their parent was mAppsCustomizePane -- so set the scroll on mAnimationBuffer // to be exactly the same as mAppsCustomizePane, and below, set the left/top // parameters to be correct for each of the pages mAnimationBuffer.scrollTo(mAppsCustomizePane.getScrollX(), 0); // mAppsCustomizePane renders its children in reverse order, so // add the pages to mAnimationBuffer in reverse order to match that behavior for (int i = visiblePages.size() - 1; i >= 0; i--) { View child = visiblePages.get(i); if (child instanceof AppsCustomizeCellLayout) { ((AppsCustomizeCellLayout) child).resetChildrenOnKeyListeners(); } else if (child instanceof PagedViewGridLayout) { ((PagedViewGridLayout) child).resetChildrenOnKeyListeners(); } PagedViewWidget.setDeletePreviewsWhenDetachedFromWindow(false); if (LauncherLog.DEBUG) { LauncherLog.d(LOG_TAG, "onTabChanged before remove view: i = " + i + ", child = " + child + ", mAppsCustomizePane = " + mAppsCustomizePane); } mAppsCustomizePane.removeView(child); PagedViewWidget.setDeletePreviewsWhenDetachedFromWindow(true); mAnimationBuffer.setAlpha(1f); mAnimationBuffer.setVisibility(View.VISIBLE); LayoutParams p = new FrameLayout.LayoutParams(child.getMeasuredWidth(), child.getMeasuredHeight()); p.setMargins((int) child.getLeft(), (int) child.getTop(), 0, 0); mAnimationBuffer.addView(child, p); } // Toggle the new content onTabChangedStart(); onTabChangedEnd(type); // Animate the transition ObjectAnimator outAnim = LauncherAnimUtils.ofFloat(mAnimationBuffer, "alpha", 0f); outAnim.addListener(new AnimatorListenerAdapter() { private void clearAnimationBuffer() { mAnimationBuffer.setVisibility(View.GONE); PagedViewWidget.setRecyclePreviewsWhenDetachedFromWindow(false); mAnimationBuffer.removeAllViews(); PagedViewWidget.setRecyclePreviewsWhenDetachedFromWindow(true); } @Override public void onAnimationEnd(Animator animation) { clearAnimationBuffer(); } @Override public void onAnimationCancel(Animator animation) { clearAnimationBuffer(); } }); ObjectAnimator inAnim = LauncherAnimUtils.ofFloat(mAppsCustomizePane, "alpha", 1f); inAnim.addListener(new AnimatorListenerAdapter() { @Override public void onAnimationEnd(Animator animation) { Trace.traceBegin(Trace.TRACE_TAG_INPUT, "onTabChanged_onAnimationEnd"); reloadCurrentPage(); Trace.traceEnd(Trace.TRACE_TAG_INPUT); } }); final AnimatorSet animSet = LauncherAnimUtils.createAnimatorSet(); animSet.playTogether(outAnim, inAnim); animSet.setDuration(duration); animSet.start(); } }); Trace.traceEnd(Trace.TRACE_TAG_INPUT); } public void setCurrentTabFromContent(AppsCustomizePagedView.ContentType type) { setOnTabChangedListener(null); setCurrentTabByTag(getTabTagForContentType(type)); setOnTabChangedListener(this); } /** * Returns the content type for the specified tab tag. */ public AppsCustomizePagedView.ContentType getContentTypeForTabTag(String tag) { if (tag.equals(APPS_TAB_TAG)) { return AppsCustomizePagedView.ContentType.Applications; } else if (tag.equals(WIDGETS_TAB_TAG)) { return AppsCustomizePagedView.ContentType.Widgets; } return AppsCustomizePagedView.ContentType.Applications; } /** * Returns the tab tag for a given content type. */ public String getTabTagForContentType(AppsCustomizePagedView.ContentType type) { if (type == AppsCustomizePagedView.ContentType.Applications) { return APPS_TAB_TAG; } else if (type == AppsCustomizePagedView.ContentType.Widgets) { return WIDGETS_TAB_TAG; } return APPS_TAB_TAG; } /** * Disable focus on anything under this view in the hierarchy if we are not visible. */ @Override public int getDescendantFocusability() { if (getVisibility() != View.VISIBLE) { return ViewGroup.FOCUS_BLOCK_DESCENDANTS; } return super.getDescendantFocusability(); } void reset() { if (mInTransition) { // Defer to after the transition to reset mResetAfterTransition = true; } else { // Reset immediately mAppsCustomizePane.reset(); } } private void enableAndBuildHardwareLayer() { // isHardwareAccelerated() checks if we're attached to a window and if that // window is HW accelerated-- we were sometimes not attached to a window // and buildLayer was throwing an IllegalStateException if (isHardwareAccelerated()) { // Turn on hardware layers for performance setLayerType(LAYER_TYPE_HARDWARE, null); // force building the layer, so you don't get a blip early in an animation // when the layer is created layer buildLayer(); } } @Override public View getContent() { return mContent; } /* LauncherTransitionable overrides */ @Override public void onLauncherTransitionPrepare(Launcher l, boolean animated, boolean toWorkspace) { if (LauncherLog.DEBUG_LAYOUT) { LauncherLog.d(LOG_TAG, "onLauncherTransitionPrepare: toWorkspace = " + toWorkspace + ", animated = " + animated + ", mResetAfterTransition = " + mResetAfterTransition + ", mContent visibility = " + mContent.getVisibility() + ", current page = " + mAppsCustomizePane.getCurrentPage()); } mAppsCustomizePane.onLauncherTransitionPrepare(l, animated, toWorkspace); mInTransition = true; mTransitioningToWorkspace = toWorkspace; if (toWorkspace) { // Going from All Apps -> Workspace setVisibilityOfSiblingsWithLowerZOrder(VISIBLE); } else { // Going from Workspace -> All Apps mContent.setVisibility(VISIBLE); // Make sure the current page is loaded (we start loading the side pages after the // transition to prevent slowing down the animation) mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage(), true); } if (mResetAfterTransition) { mAppsCustomizePane.reset(); mResetAfterTransition = false; } } @Override public void onLauncherTransitionStart(Launcher l, boolean animated, boolean toWorkspace) { if (LauncherLog.DEBUG) { LauncherLog.d(LOG_TAG, "onLauncherTransitionStart: l = " + l + ", animated = " + animated + ", toWorkspace = " + toWorkspace); } if (animated) { enableAndBuildHardwareLayer(); } // Dismiss the workspace cling l.dismissWorkspaceCling(null); } @Override public void onLauncherTransitionStep(Launcher l, float t) { // Do nothing } @Override public void onLauncherTransitionEnd(Launcher l, boolean animated, boolean toWorkspace) { Trace.traceBegin(Trace.TRACE_TAG_INPUT, "AppsCustomizeTabHost.onLauncherTransitionEnd"); if (LauncherLog.DEBUG) { LauncherLog.d(LOG_TAG, "onLauncherTransitionEnd: l = " + l + ", animated = " + animated + ", toWorkspace = " + toWorkspace + ", current page = " + mAppsCustomizePane.getCurrentPage()); } Trace.traceCounter(Trace.TRACE_TAG_PERF, "AppUpdate", 1); mAppsCustomizePane.onLauncherTransitionEnd(l, animated, toWorkspace); mInTransition = false; if (animated) { setLayerType(LAYER_TYPE_NONE, null); } Trace.traceCounter(Trace.TRACE_TAG_PERF, "AppUpdate", 0); LauncherLog.d(LOG_TAG, "onLauncherTransitionEnd: toWorkspace = " + toWorkspace); if (!toWorkspace) { // Show the all apps cling (if not already shown) mAppsCustomizePane.showAllAppsCling(); // Make sure adjacent pages are loaded (we wait until after the transition to // prevent slowing down the animation) mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage()); // Going from Workspace -> All Apps // NOTE: We should do this at the end since we check visibility state in some of the // cling initialization/dismiss code above. setVisibilityOfSiblingsWithLowerZOrder(INVISIBLE); if (LauncherLog.DEBUG) { LauncherLog.d(LOG_TAG, "[All apps launch time][End] onLauncherTransitionEnd."); } } Trace.traceEnd(Trace.TRACE_TAG_INPUT); /// M: Add for op09 Edit and Hide app icons. if (mIsInEditMode) { mAppsCustomizePane.enterEditMode(); } } private void setVisibilityOfSiblingsWithLowerZOrder(int visibility) { ViewGroup parent = (ViewGroup) getParent(); if (parent == null) return; View overviewPanel = ((Launcher) getContext()).getOverviewPanel(); final int count = parent.getChildCount(); if (!isChildrenDrawingOrderEnabled()) { for (int i = 0; i < count; i++) { final View child = parent.getChildAt(i); if (child == this) { break; } else { if (child.getVisibility() == GONE || child == overviewPanel) { continue; } child.setVisibility(visibility); } } } else { throw new RuntimeException("Failed; can't get z-order of views"); } } public void onWindowVisible() { if (getVisibility() == VISIBLE) { mContent.setVisibility(VISIBLE); // We unload the widget previews when the UI is hidden, so need to reload pages // Load the current page synchronously, and the neighboring pages asynchronously mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage(), true); mAppsCustomizePane.loadAssociatedPages(mAppsCustomizePane.getCurrentPage()); } } public void onTrimMemory() { if (LauncherLog.DEBUG) { LauncherLog.d(LOG_TAG, "onTrimMemory."); } mContent.setVisibility(GONE); // Clear the widget pages of all their subviews - this will trigger the widget previews // to delete their bitmaps mAppsCustomizePane.clearAllWidgetPages(); } boolean isTransitioning() { return mInTransition; } /** * M: Set the visibility of host content view. * * @param visibility */ public void setContentVisibility(int visibility) { mContent.setVisibility(visibility); } /** * M: Get the visibility of host content view. */ public int getContentVisibility() { return mContent.getVisibility(); } /** * M: Enter edit mode, allow user to rearrange application icons, add for OP09. */ public void enterEditMode() { // We need to make the tab widget invisible instead of gone, or else the // edit title layout will shrink. //mAppsCustomizePane.enterEditMode(); mIsInEditMode = true; setAlpha(1.0f); } /// M: Add for op09 Edit and Hide app icons. private boolean mIsInEditMode; /** * M: Exit edit mode, add for OP09. */ public void exitEditMode() { mIsInEditMode = false; mAppsCustomizePane.exitEditMode(); } /// M: check whether tab container is visible or not public boolean isTabContainerVisible() { return mTabsContainer != null && mTabsContainer.getVisibility() == View.VISIBLE; } }