/*
* 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;
}
}