blob: 0e512c6d97dd287800204fbc3ea7700452b3e660 [file] [log] [blame]
/*
* Copyright (C) 2008 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.launcher2;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import com.android.launcher.R;
import android.content.ComponentName;
import android.content.Context;
import android.content.res.Resources;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.PixelFormat;
import android.graphics.Rect;
import android.renderscript.*;
import android.util.AttributeSet;
import android.util.DisplayMetrics;
import android.util.Log;
import android.view.KeyEvent;
import android.view.MotionEvent;
import android.view.SoundEffectConstants;
import android.view.SurfaceHolder;
import android.view.VelocityTracker;
import android.view.View;
import android.view.ViewConfiguration;
import android.view.accessibility.AccessibilityEvent;
public class AllApps3D extends RSSurfaceView
implements AllAppsView, View.OnClickListener, View.OnLongClickListener, DragSource {
private static final String TAG = "Launcher.AllApps3D";
/** Bit for mLocks for when there are icons being loaded. */
private static final int LOCK_ICONS_PENDING = 1;
private static final int TRACKING_NONE = 0;
private static final int TRACKING_FLING = 1;
private static final int TRACKING_HOME = 2;
private static final int SELECTED_NONE = 0;
private static final int SELECTED_FOCUSED = 1;
private static final int SELECTED_PRESSED = 2;
private static final int SELECTION_NONE = 0;
private static final int SELECTION_ICONS = 1;
private static final int SELECTION_HOME = 2;
private Launcher mLauncher;
private DragController mDragController;
/** When this is 0, modifications are allowed, when it's not, they're not.
* TODO: What about scrolling? */
private int mLocks = LOCK_ICONS_PENDING;
private int mSlop;
private int mMaxFlingVelocity;
private Defines mDefines = new Defines();
private ArrayList<ApplicationInfo> mAllAppsList;
private static RenderScriptGL sRS;
private static RolloRS sRollo;
private static boolean sZoomDirty = false;
private static boolean sAnimateNextZoom;
private static float sNextZoom;
/**
* True when we are using arrow keys or trackball to drive navigation
*/
private boolean mArrowNavigation = false;
private boolean mStartedScrolling;
/**
* Used to keep track of the selection when AllAppsView loses window focus.
* One of the SELECTION_ constants.
*/
private int mLastSelection;
/**
* Used to keep track of the selection when AllAppsView loses window focus
*/
private int mLastSelectedIcon;
private VelocityTracker mVelocityTracker;
private int mTouchTracking;
private int mMotionDownRawX;
private int mMotionDownRawY;
private int mDownIconIndex = -1;
private int mCurrentIconIndex = -1;
private int[] mTouchYBorders;
private int[] mTouchXBorders;
private boolean mShouldGainFocus;
private boolean mHaveSurface = false;
private float mZoom;
private float mVelocity;
private AAMessage mMessageProc;
private int mColumnsPerPage;
private int mRowsPerPage;
private boolean mSurrendered;
private int mRestoreFocusIndex = -1;
@SuppressWarnings({"UnusedDeclaration"})
static class Defines {
public static final int COLUMNS_PER_PAGE_PORTRAIT = 4;
public static final int ROWS_PER_PAGE_PORTRAIT = 4;
public static final int COLUMNS_PER_PAGE_LANDSCAPE = 6;
public static final int ROWS_PER_PAGE_LANDSCAPE = 3;
public static final int SELECTION_TEXTURE_WIDTH_PX = 74 + 20;
public static final int SELECTION_TEXTURE_HEIGHT_PX = 74 + 20;
}
public AllApps3D(Context context, AttributeSet attrs) {
super(context, attrs);
setFocusable(true);
setSoundEffectsEnabled(false);
final ViewConfiguration config = ViewConfiguration.get(context);
mSlop = config.getScaledTouchSlop();
mMaxFlingVelocity = config.getScaledMaximumFlingVelocity();
setOnClickListener(this);
setOnLongClickListener(this);
setZOrderOnTop(true);
getHolder().setFormat(PixelFormat.TRANSLUCENT);
if (sRS == null) {
RenderScriptGL.SurfaceConfig sc = new RenderScriptGL.SurfaceConfig();
sc.setDepth(16, 16);
sc.setAlpha(8, 8);
sRS = createRenderScriptGL(sc);
} else {
// Is this even possible?
setRenderScriptGL(sRS);
}
if (sRollo != null) {
sRollo.mAllApps = this;
sRollo.mRes = getResources();
sRollo.mInitialize = true;
}
}
@SuppressWarnings({"UnusedDeclaration"})
public AllApps3D(Context context, AttributeSet attrs, int defStyle) {
this(context, attrs);
}
public void surrender() {
if (sRS != null) {
sRS.setSurface(null, 0, 0);
sRS.setMessageHandler(null);
}
mSurrendered = true;
}
/**
* Note that this implementation prohibits this view from ever being reattached.
*/
@Override
protected void onDetachedFromWindow() {
sRS.setMessageHandler(null);
if (!mSurrendered) {
Log.i(TAG, "onDetachedFromWindow");
destroyRenderScriptGL();
sRS = null;
sRollo = null;
super.onDetachedFromWindow();
}
}
/**
* If you have an attached click listener, View always plays the click sound!?!?
* Deal with sound effects by hand.
*/
public void reallyPlaySoundEffect(int sound) {
boolean old = isSoundEffectsEnabled();
setSoundEffectsEnabled(true);
playSoundEffect(sound);
setSoundEffectsEnabled(old);
}
@Override
public void setup(Launcher launcher, DragController dragController) {
mLauncher = launcher;
mDragController = dragController;
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
super.surfaceDestroyed(holder);
// Without this, we leak mMessageCallback which leaks the context.
if (!mSurrendered) {
sRS.setMessageHandler(null);
}
// We may lose any callbacks that are pending, so make sure that we re-sync that
// on the next surfaceChanged.
sZoomDirty = true;
mHaveSurface = false;
}
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) {
//long startTime = SystemClock.uptimeMillis();
super.surfaceChanged(holder, format, w, h);
final boolean isPortrait = w < h;
mColumnsPerPage = isPortrait ? Defines.COLUMNS_PER_PAGE_PORTRAIT :
Defines.COLUMNS_PER_PAGE_LANDSCAPE;
mRowsPerPage = isPortrait ? Defines.ROWS_PER_PAGE_PORTRAIT :
Defines.ROWS_PER_PAGE_LANDSCAPE;
if (mSurrendered) return;
mHaveSurface = true;
if (sRollo == null) {
sRollo = new RolloRS(this);
sRollo.init(getResources(), w, h);
if (mAllAppsList != null) {
sRollo.setApps(mAllAppsList);
}
if (mShouldGainFocus) {
gainFocus();
mShouldGainFocus = false;
}
} else if (sRollo.mInitialize) {
sRollo.initGl();
sRollo.mInitialize = false;
}
initTouchState(w, h);
sRollo.dirtyCheck();
sRollo.resize(w, h);
Log.d(TAG, "sc " + sRS);
if (sRS != null) {
mMessageProc = new AAMessage();
sRS.setMessageHandler(mMessageProc);
}
//long endTime = SystemClock.uptimeMillis();
//Log.d(TAG, "surfaceChanged took " + (endTime-startTime) + "ms");
}
@Override
public void onWindowFocusChanged(boolean hasWindowFocus) {
super.onWindowFocusChanged(hasWindowFocus);
if (mSurrendered) return;
if (mArrowNavigation) {
if (!hasWindowFocus) {
// Clear selection when we lose window focus
mLastSelectedIcon = sRollo.mScript.get_gSelectedIconIndex();
sRollo.setHomeSelected(SELECTED_NONE);
sRollo.clearSelectedIcon();
} else {
if (sRollo.mScript.get_gIconCount() > 0) {
if (mLastSelection == SELECTION_ICONS) {
int selection = mLastSelectedIcon;
final int firstIcon = Math.round(sRollo.mScrollPos) * mColumnsPerPage;
if (selection < 0 || // No selection
selection < firstIcon || // off the top of the screen
selection >= sRollo.mScript.get_gIconCount() || // past last icon
selection >= firstIcon + // past last icon on screen
(mColumnsPerPage * mRowsPerPage)) {
selection = firstIcon;
}
// Select the first icon when we gain window focus
sRollo.selectIcon(selection, SELECTED_FOCUSED);
} else if (mLastSelection == SELECTION_HOME) {
sRollo.setHomeSelected(SELECTED_FOCUSED);
}
}
}
}
}
@Override
protected void onFocusChanged(boolean gainFocus, int direction, Rect previouslyFocusedRect) {
super.onFocusChanged(gainFocus, direction, previouslyFocusedRect);
if (!isVisible() || mSurrendered) {
return;
}
if (gainFocus) {
if (sRollo != null) {
gainFocus();
} else {
mShouldGainFocus = true;
}
} else {
if (sRollo != null) {
if (mArrowNavigation) {
// Clear selection when we lose focus
sRollo.clearSelectedIcon();
sRollo.setHomeSelected(SELECTED_NONE);
mArrowNavigation = false;
}
} else {
mShouldGainFocus = false;
}
}
}
private void gainFocus() {
if (!mArrowNavigation && sRollo.mScript.get_gIconCount() > 0) {
// Select the first icon when we gain keyboard focus
mArrowNavigation = true;
sRollo.selectIcon(Math.round(sRollo.mScrollPos) * mColumnsPerPage, SELECTED_FOCUSED);
}
}
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
boolean handled = false;
if (!isVisible()) {
return false;
}
final int iconCount = sRollo.mScript.get_gIconCount();
if (keyCode == KeyEvent.KEYCODE_DPAD_CENTER || keyCode == KeyEvent.KEYCODE_ENTER) {
if (mArrowNavigation) {
if (mLastSelection == SELECTION_HOME) {
reallyPlaySoundEffect(SoundEffectConstants.CLICK);
mLauncher.showWorkspace(true);
} else {
int whichApp = sRollo.mScript.get_gSelectedIconIndex();
if (whichApp >= 0) {
ApplicationInfo app = mAllAppsList.get(whichApp);
mLauncher.startActivitySafely(app.intent, app);
handled = true;
}
}
}
}
if (iconCount > 0) {
final boolean isPortrait = getWidth() < getHeight();
mArrowNavigation = true;
int currentSelection = sRollo.mScript.get_gSelectedIconIndex();
int currentTopRow = Math.round(sRollo.mScrollPos);
// The column of the current selection, in the range 0..COLUMNS_PER_PAGE_PORTRAIT-1
final int currentPageCol = currentSelection % mColumnsPerPage;
// The row of the current selection, in the range 0..ROWS_PER_PAGE_PORTRAIT-1
final int currentPageRow = (currentSelection - (currentTopRow * mColumnsPerPage))
/ mRowsPerPage;
int newSelection = currentSelection;
switch (keyCode) {
case KeyEvent.KEYCODE_DPAD_UP:
if (mLastSelection == SELECTION_HOME) {
if (isPortrait) {
sRollo.setHomeSelected(SELECTED_NONE);
int lastRowCount = iconCount % mColumnsPerPage;
if (lastRowCount == 0) {
lastRowCount = mColumnsPerPage;
}
newSelection = iconCount - lastRowCount + (mColumnsPerPage / 2);
if (newSelection >= iconCount) {
newSelection = iconCount-1;
}
int target = (newSelection / mColumnsPerPage) - (mRowsPerPage - 1);
if (target < 0) {
target = 0;
}
if (currentTopRow != target) {
sRollo.moveTo(target);
}
}
} else {
if (currentPageRow > 0) {
newSelection = currentSelection - mColumnsPerPage;
if (currentTopRow > newSelection / mColumnsPerPage) {
sRollo.moveTo(newSelection / mColumnsPerPage);
}
} else if (currentTopRow > 0) {
newSelection = currentSelection - mColumnsPerPage;
sRollo.moveTo(newSelection / mColumnsPerPage);
} else if (currentPageRow != 0) {
newSelection = currentTopRow * mRowsPerPage;
}
}
handled = true;
break;
case KeyEvent.KEYCODE_DPAD_DOWN: {
final int rowCount = iconCount / mColumnsPerPage
+ (iconCount % mColumnsPerPage == 0 ? 0 : 1);
final int currentRow = currentSelection / mColumnsPerPage;
if (mLastSelection != SELECTION_HOME) {
if (currentRow < rowCount-1) {
sRollo.setHomeSelected(SELECTED_NONE);
if (currentSelection < 0) {
newSelection = 0;
} else {
newSelection = currentSelection + mColumnsPerPage;
}
if (newSelection >= iconCount) {
// Go from D to G in this arrangement:
// A B C D
// E F G
newSelection = iconCount - 1;
}
if (currentPageRow >= mRowsPerPage - 1) {
sRollo.moveTo((newSelection / mColumnsPerPage) - mRowsPerPage + 1);
}
} else if (isPortrait) {
newSelection = -1;
sRollo.setHomeSelected(SELECTED_FOCUSED);
}
}
handled = true;
break;
}
case KeyEvent.KEYCODE_DPAD_LEFT:
if (mLastSelection != SELECTION_HOME) {
if (currentPageCol > 0) {
newSelection = currentSelection - 1;
}
} else if (!isPortrait) {
newSelection = ((int) (sRollo.mScrollPos) * mColumnsPerPage) +
(mRowsPerPage / 2 * mColumnsPerPage) + mColumnsPerPage - 1;
sRollo.setHomeSelected(SELECTED_NONE);
}
handled = true;
break;
case KeyEvent.KEYCODE_DPAD_RIGHT:
if (mLastSelection != SELECTION_HOME) {
if (!isPortrait && (currentPageCol == mColumnsPerPage - 1 ||
currentSelection == iconCount - 1)) {
newSelection = -1;
sRollo.setHomeSelected(SELECTED_FOCUSED);
} else if ((currentPageCol < mColumnsPerPage - 1) &&
(currentSelection < iconCount - 1)) {
newSelection = currentSelection + 1;
}
}
handled = true;
break;
}
if (newSelection != currentSelection) {
sRollo.selectIcon(newSelection, SELECTED_FOCUSED);
}
}
return handled;
}
void initTouchState(int width, int height) {
boolean isPortrait = width < height;
int[] viewPos = new int[2];
getLocationOnScreen(viewPos);
mTouchXBorders = new int[mColumnsPerPage + 1];
mTouchYBorders = new int[mRowsPerPage + 1];
// TODO: Put this in a config file/define
int cellHeight = 145;//iconsSize / Defines.ROWS_PER_PAGE_PORTRAIT;
if (!isPortrait) cellHeight -= 12;
int centerY = (int) (height * (isPortrait ? 0.5f : 0.47f));
if (!isPortrait) centerY += cellHeight / 2;
int half = (int) Math.floor((mRowsPerPage + 1) / 2);
int end = mTouchYBorders.length - (half + 1);
for (int i = -half; i <= end; i++) {
mTouchYBorders[i + half] = centerY + (i * cellHeight) - viewPos[1];
}
int x = 0;
// TODO: Put this in a config file/define
int columnWidth = 120;
for (int i = 0; i < mColumnsPerPage + 1; i++) {
mTouchXBorders[i] = x - viewPos[0];
x += columnWidth;
}
}
int chooseTappedIcon(int x, int y) {
float pos = sRollo != null ? sRollo.mScrollPos : 0;
int oldY = y;
// Adjust for scroll position if not zero.
y += (pos - ((int)pos)) * (mTouchYBorders[1] - mTouchYBorders[0]);
int col = -1;
int row = -1;
final int columnsCount = mColumnsPerPage;
for (int i=0; i< columnsCount; i++) {
if (x >= mTouchXBorders[i] && x < mTouchXBorders[i+1]) {
col = i;
break;
}
}
final int rowsCount = mRowsPerPage;
for (int i=0; i< rowsCount; i++) {
if (y >= mTouchYBorders[i] && y < mTouchYBorders[i+1]) {
row = i;
break;
}
}
if (row < 0 || col < 0) {
return -1;
}
int index = (((int) pos) * columnsCount) + (row * columnsCount) + col;
if (index >= mAllAppsList.size()) {
return -1;
} else {
return index;
}
}
@Override
public boolean onTouchEvent(MotionEvent ev)
{
mArrowNavigation = false;
if (!isVisible()) {
return true;
}
if (mLocks != 0) {
return true;
}
super.onTouchEvent(ev);
int x = (int)ev.getX();
int y = (int)ev.getY();
final boolean isPortrait = getWidth() < getHeight();
int action = ev.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
if ((isPortrait && y > mTouchYBorders[mTouchYBorders.length-1]) ||
(!isPortrait && x > mTouchXBorders[mTouchXBorders.length-1])) {
mTouchTracking = TRACKING_HOME;
sRollo.setHomeSelected(SELECTED_PRESSED);
mCurrentIconIndex = -1;
} else {
mTouchTracking = TRACKING_FLING;
mMotionDownRawX = (int)ev.getRawX();
mMotionDownRawY = (int)ev.getRawY();
if (!sRollo.checkClickOK()) {
sRollo.clearSelectedIcon();
} else {
mDownIconIndex = mCurrentIconIndex
= sRollo.selectIcon(x, y, SELECTED_PRESSED);
if (mDownIconIndex < 0) {
// if nothing was selected, no long press.
cancelLongPress();
}
}
sRollo.move(ev.getRawY() / getHeight());
mVelocityTracker = VelocityTracker.obtain();
mVelocityTracker.addMovement(ev);
mStartedScrolling = false;
}
break;
case MotionEvent.ACTION_MOVE:
case MotionEvent.ACTION_OUTSIDE:
if (mTouchTracking == TRACKING_HOME) {
sRollo.setHomeSelected((isPortrait &&
y > mTouchYBorders[mTouchYBorders.length-1]) || (!isPortrait
&& x > mTouchXBorders[mTouchXBorders.length-1])
? SELECTED_PRESSED : SELECTED_NONE);
} else if (mTouchTracking == TRACKING_FLING) {
int rawY = (int)ev.getRawY();
int slop;
slop = Math.abs(rawY - mMotionDownRawY);
if (!mStartedScrolling && slop < mSlop) {
// don't update anything so when we do start scrolling
// below, we get the right delta.
mCurrentIconIndex = chooseTappedIcon(x, y);
if (mDownIconIndex != mCurrentIconIndex) {
// If a different icon is selected, don't allow it to be picked up.
// This handles off-axis dragging.
cancelLongPress();
mCurrentIconIndex = -1;
}
} else {
if (!mStartedScrolling) {
cancelLongPress();
mCurrentIconIndex = -1;
}
sRollo.move(ev.getRawY() / getHeight());
mStartedScrolling = true;
sRollo.clearSelectedIcon();
mVelocityTracker.addMovement(ev);
}
}
break;
case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL:
if (mTouchTracking == TRACKING_HOME) {
if (action == MotionEvent.ACTION_UP) {
if ((isPortrait && y > mTouchYBorders[mTouchYBorders.length-1]) ||
(!isPortrait && x > mTouchXBorders[mTouchXBorders.length-1])) {
reallyPlaySoundEffect(SoundEffectConstants.CLICK);
mLauncher.showWorkspace(true);
}
sRollo.setHomeSelected(SELECTED_NONE);
}
mCurrentIconIndex = -1;
} else if (mTouchTracking == TRACKING_FLING) {
mVelocityTracker.computeCurrentVelocity(1000 /* px/sec */, mMaxFlingVelocity);
sRollo.clearSelectedIcon();
sRollo.fling(ev.getRawY() / getHeight(),
mVelocityTracker.getYVelocity() / getHeight());
if (mVelocityTracker != null) {
mVelocityTracker.recycle();
mVelocityTracker = null;
}
}
mTouchTracking = TRACKING_NONE;
break;
}
return true;
}
public void onClick(View v) {
if (mLocks != 0 || !isVisible()) {
return;
}
if (sRollo.checkClickOK() && mCurrentIconIndex == mDownIconIndex
&& mCurrentIconIndex >= 0 && mCurrentIconIndex < mAllAppsList.size()) {
reallyPlaySoundEffect(SoundEffectConstants.CLICK);
ApplicationInfo app = mAllAppsList.get(mCurrentIconIndex);
mLauncher.startActivitySafely(app.intent, app);
}
}
public boolean onLongClick(View v) {
// We don't accept long click events in these cases
// - If the workspace isn't ready to accept a drop
// - If we're not done loading (because we might be confused about which item
// to pick up
// - If we're not visible
if (!isVisible() || mLauncher.isWorkspaceLocked() || mLocks != 0) {
return true;
}
if (sRollo.checkClickOK() && mCurrentIconIndex == mDownIconIndex
&& mCurrentIconIndex >= 0 && mCurrentIconIndex < mAllAppsList.size()) {
ApplicationInfo app = mAllAppsList.get(mCurrentIconIndex);
final Bitmap bmp = app.iconBitmap;
// We don't really have an accurate location to use. This will do.
int screenX = mMotionDownRawX - (bmp.getWidth() / 2);
int screenY = mMotionDownRawY - bmp.getHeight();
mLauncher.lockScreenOrientation();
mLauncher.getWorkspace().onDragStartedWithItemSpans(1, 1, bmp);
mDragController.startDrag(
bmp, screenX, screenY, this, app, DragController.DRAG_ACTION_COPY);
mLauncher.showWorkspace(true);
}
return true;
}
@Override
public boolean dispatchPopulateAccessibilityEvent(AccessibilityEvent event) {
if (event.getEventType() == AccessibilityEvent.TYPE_VIEW_SELECTED) {
if (!isVisible()) {
return false;
}
String text = null;
int index;
int count = mAllAppsList.size() + 1; // +1 is home
int pos = -1;
switch (mLastSelection) {
case SELECTION_ICONS:
index = sRollo.mScript.get_gSelectedIconIndex();
if (index >= 0) {
ApplicationInfo info = mAllAppsList.get(index);
if (info.title != null) {
text = info.title.toString();
pos = index;
}
}
break;
case SELECTION_HOME:
text = getContext().getString(R.string.all_apps_home_button_label);
pos = count;
break;
}
if (text != null) {
event.setEnabled(true);
event.getText().add(text);
//event.setContentDescription(text);
event.setItemCount(count);
event.setCurrentItemIndex(pos);
}
}
return false;
}
@Override
public void onDragViewVisible() {
}
@Override
public void onDropCompleted(View target, Object dragInfo, boolean success) {
mLauncher.getWorkspace().onDragStopped(success);
mLauncher.unlockScreenOrientation();
}
/**
* Zoom to the specifed level.
*
* @param zoom [0..1] 0 is hidden, 1 is open
*/
public void zoom(float zoom, boolean animate) {
cancelLongPress();
sNextZoom = zoom;
sAnimateNextZoom = animate;
// if we do setZoom while we don't have a surface, we won't
// get the callbacks that actually set mZoom.
if (sRollo == null || !mHaveSurface) {
sZoomDirty = true;
mZoom = zoom;
} else {
sRollo.setZoom(zoom, animate);
}
}
/**
* If sRollo is null, then we're not visible. This is also used to guard against
* sRollo being null.
*/
public boolean isVisible() {
return sRollo != null && mZoom > 0.001f;
}
public boolean isAnimating() {
return isVisible() && mZoom <= 0.999f;
}
public void setApps(ArrayList<ApplicationInfo> list) {
if (sRS == null) {
// We've been removed from the window. Don't bother with all this.
return;
}
if (list != null) {
Collections.sort(list, LauncherModel.APP_NAME_COMPARATOR);
}
boolean reload = false;
if (mAllAppsList == null) {
reload = true;
} else if (list.size() != mAllAppsList.size()) {
reload = true;
} else {
final int count = list.size();
for (int i = 0; i < count; i++) {
if (list.get(i) != mAllAppsList.get(i)) {
reload = true;
break;
}
}
}
mAllAppsList = list;
if (sRollo != null && reload) {
sRollo.setApps(list);
}
if (hasFocus() && mRestoreFocusIndex != -1) {
sRollo.selectIcon(mRestoreFocusIndex, SELECTED_FOCUSED);
mRestoreFocusIndex = -1;
}
mLocks &= ~LOCK_ICONS_PENDING;
}
public void addApps(ArrayList<ApplicationInfo> list) {
if (mAllAppsList == null) {
// Not done loading yet. We'll find out about it later.
return;
}
if (sRS == null) {
// We've been removed from the window. Don't bother with all this.
return;
}
final int N = list.size();
if (sRollo != null) {
sRollo.pause();
sRollo.reallocAppsList(sRollo.mScript.get_gIconCount() + N);
}
for (int i=0; i<N; i++) {
final ApplicationInfo item = list.get(i);
int index = Collections.binarySearch(mAllAppsList, item,
LauncherModel.APP_NAME_COMPARATOR);
if (index < 0) {
index = -(index+1);
}
mAllAppsList.add(index, item);
if (sRollo != null) {
sRollo.addApp(index, item);
}
}
if (sRollo != null) {
sRollo.saveAppsList();
sRollo.resume();
}
}
public void removeApps(ArrayList<ApplicationInfo> list) {
if (mAllAppsList == null) {
// Not done loading yet. We'll find out about it later.
return;
}
if (sRollo != null) {
sRollo.pause();
}
final int N = list.size();
for (int i=0; i<N; i++) {
final ApplicationInfo item = list.get(i);
int index = findAppByComponent(mAllAppsList, item);
if (index >= 0) {
mAllAppsList.remove(index);
if (sRollo != null) {
sRollo.removeApp(index);
}
} else {
Log.w(TAG, "couldn't find a match for item \"" + item + "\"");
// Try to recover. This should keep us from crashing for now.
}
}
if (sRollo != null) {
sRollo.saveAppsList();
sRollo.resume();
}
}
public void updateApps(ArrayList<ApplicationInfo> list) {
// Just remove and add, because they may need to be re-sorted.
removeApps(list);
addApps(list);
}
private static int findAppByComponent(ArrayList<ApplicationInfo> list, ApplicationInfo item) {
ComponentName component = item.intent.getComponent();
final int N = list.size();
for (int i=0; i<N; i++) {
ApplicationInfo x = list.get(i);
if (x.intent.getComponent().equals(component)) {
return i;
}
}
return -1;
}
class AAMessage extends RenderScript.RSMessageHandler {
public void run() {
sRollo.mScrollPos = ((float)mData[0]) / (1 << 16);
mVelocity = ((float)mData[1]) / (1 << 16);
boolean lastVisible = isVisible();
mZoom = ((float)mData[2]) / (1 << 16);
final boolean visible = isVisible();
if (visible != lastVisible) {
post(new Runnable() {
public void run() {
if (visible) {
showSurface();
} else {
hideSurface();
}
}
});
}
sZoomDirty = false;
}
}
public static class RolloRS {
// Allocations ======
private int mWidth;
private int mHeight;
private Resources mRes;
ScriptC_allapps mScript;
private Mesh mMesh;
private ProgramVertexFixedFunction.Constants mPVA;
private ScriptField_VpConsts mUniformAlloc;
private Allocation mHomeButtonNormal;
private Allocation mHomeButtonFocused;
private Allocation mHomeButtonPressed;
private Allocation[] mIcons;
private Allocation mAllocIcons;
private Allocation[] mLabels;
private Allocation mAllocLabels;
private Bitmap mSelectionBitmap;
private Canvas mSelectionCanvas;
private float mScrollPos;
AllApps3D mAllApps;
boolean mInitialize;
private boolean checkClickOK() {
return (Math.abs(mAllApps.mVelocity) < 0.4f) &&
(Math.abs(mScrollPos - Math.round(mScrollPos)) < 0.4f);
}
void pause() {
if (sRS != null) {
sRS.bindRootScript(null);
}
}
void resume() {
if (sRS != null) {
sRS.bindRootScript(mScript);
}
}
public RolloRS(AllApps3D allApps) {
mAllApps = allApps;
}
public void init(Resources res, int width, int height) {
mRes = res;
mWidth = width;
mHeight = height;
mScript = new ScriptC_allapps(sRS, mRes, R.raw.allapps);
initProgramVertex();
initProgramFragment();
initProgramStore();
initGl();
initData();
mScript.bind_gIcons(mAllocIcons);
mScript.bind_gLabels(mAllocLabels);
sRS.bindRootScript(mScript);
}
public void initMesh() {
Mesh.TriangleMeshBuilder tm = new Mesh.TriangleMeshBuilder(sRS, 2, 0);
for (int ct=0; ct < 16; ct++) {
float pos = (1.f / (16.f - 1)) * ct;
tm.addVertex(0.0f, pos);
tm.addVertex(1.0f, pos);
}
for (int ct=0; ct < (16 * 2 - 2); ct+= 2) {
tm.addTriangle(ct, ct+1, ct+2);
tm.addTriangle(ct+1, ct+3, ct+2);
}
mMesh = tm.create(true);
mScript.set_gSMCell(mMesh);
}
Matrix4f getProjectionMatrix(int w, int h) {
// range -1,1 in the narrow axis at z = 0.
Matrix4f m1 = new Matrix4f();
Matrix4f m2 = new Matrix4f();
if(w > h) {
float aspect = ((float)w) / h;
m1.loadFrustum(-aspect,aspect, -1,1, 1,100);
} else {
float aspect = ((float)h) / w;
m1.loadFrustum(-1,1, -aspect,aspect, 1,100);
}
m2.loadRotate(180, 0, 1, 0);
m1.loadMultiply(m1, m2);
m2.loadScale(-2, 2, 1);
m1.loadMultiply(m1, m2);
m2.loadTranslate(0, 0, 2);
m1.loadMultiply(m1, m2);
return m1;
}
void resize(int w, int h) {
Matrix4f proj = getProjectionMatrix(w, h);
mPVA.setProjection(proj);
if (mUniformAlloc != null) {
ScriptField_VpConsts.Item i = new ScriptField_VpConsts.Item();
i.Proj = proj;
i.ScaleOffset.x = (2.f / 480.f);
i.ScaleOffset.y = 0;
i.ScaleOffset.z = -((float)w / 2) - 0.25f;
if (w < h) {
// portrait
i.ScaleOffset.w = -380.25f;
i.BendPos.x = 120.f; // bottom of screen
i.BendPos.y = h - 82.f; // top of screen
} else {
// landscape
i.ScaleOffset.w = -206.25f;
i.BendPos.x = 50.f;
i.BendPos.y = h - 30.f;
}
mUniformAlloc.set(i, 0, true);
}
mWidth = w;
mHeight = h;
}
private void initProgramVertex() {
mPVA = new ProgramVertexFixedFunction.Constants(sRS);
resize(mWidth, mHeight);
ProgramVertexFixedFunction.Builder pvb = new ProgramVertexFixedFunction.Builder(sRS);
pvb.setTextureMatrixEnable(true);
ProgramVertex pv = pvb.create();
((ProgramVertexFixedFunction)pv).bindConstants(mPVA);
sRS.bindProgramVertex(pv);
mUniformAlloc = new ScriptField_VpConsts(sRS, 1);
mScript.bind_vpConstants(mUniformAlloc);
initMesh();
ProgramVertex.Builder sb = new ProgramVertex.Builder(sRS);
String t = "varying vec4 varColor;\n" +
"varying vec2 varTex0;\n" +
"void main() {\n" +
// Animation
" float ani = UNI_Position.z;\n" +
" float bendY1 = UNI_BendPos.x;\n" +
" float bendY2 = UNI_BendPos.y;\n" +
" float bendAngle = 47.0 * (3.14 / 180.0);\n" +
" float bendDistance = bendY1 * 0.4;\n" +
" float distanceDimLevel = 0.6;\n" +
" float bendStep = (bendAngle / bendDistance) * (bendAngle * 0.5);\n" +
" float aDy = cos(bendAngle);\n" +
" float aDz = sin(bendAngle);\n" +
" float scale = (2.0 / " + mWidth + ".0);\n" +
" float x = UNI_Position.x + UNI_ImgSize.x * (1.0 - ani) * (ATTRIB_position.x - 0.5);\n" +
" float ys= UNI_Position.y + UNI_ImgSize.y * (1.0 - ani) * ATTRIB_position.y;\n" +
" float y = 0.0;\n" +
" float z = 0.0;\n" +
" float lum = 1.0;\n" +
" float cv = min(ys, bendY1 - bendDistance) - (bendY1 - bendDistance);\n" +
" y += cv * aDy;\n" +
" z += -cv * aDz;\n" +
" cv = clamp(ys, bendY1 - bendDistance, bendY1) - bendY1;\n" + // curve range
" lum += cv / bendDistance * distanceDimLevel;\n" +
" y += cv * cos(cv * bendStep);\n" +
" z += cv * sin(cv * bendStep);\n" +
" cv = max(ys, bendY2 + bendDistance) - (bendY2 + bendDistance);\n" +
" y += cv * aDy;\n" +
" z += cv * aDz;\n" +
" cv = clamp(ys, bendY2, bendY2 + bendDistance) - bendY2;\n" +
" lum -= cv / bendDistance * distanceDimLevel;\n" +
" y += cv * cos(cv * bendStep);\n" +
" z += cv * sin(cv * bendStep);\n" +
" y += clamp(ys, bendY1, bendY2);\n" +
" vec4 pos;\n" +
" pos.x = (x + UNI_ScaleOffset.z) * UNI_ScaleOffset.x;\n" +
" pos.y = (y + UNI_ScaleOffset.w) * UNI_ScaleOffset.x;\n" +
" pos.z = z * UNI_ScaleOffset.x;\n" +
" pos.w = 1.0;\n" +
" pos.x *= 1.0 + ani * 4.0;\n" +
" pos.y *= 1.0 + ani * 4.0;\n" +
" pos.z -= ani * 1.5;\n" +
" lum *= 1.0 - ani;\n" +
" gl_Position = UNI_Proj * pos;\n" +
" varColor.rgba = vec4(lum, lum, lum, 1.0);\n" +
" varTex0.xy = ATTRIB_position;\n" +
" varTex0.y = 1.0 - varTex0.y;\n" +
"}\n";
sb.setShader(t);
sb.addConstant(mUniformAlloc.getType());
sb.addInput(mMesh.getVertexAllocation(0).getType().getElement());
ProgramVertex pvc = sb.create();
pvc.bindConstants(mUniformAlloc.getAllocation(), 0);
mScript.set_gPVCurve(pvc);
}
private void initProgramFragment() {
Sampler.Builder sb = new Sampler.Builder(sRS);
sb.setMinification(Sampler.Value.LINEAR_MIP_LINEAR);
sb.setMagnification(Sampler.Value.NEAREST);
sb.setWrapS(Sampler.Value.CLAMP);
sb.setWrapT(Sampler.Value.CLAMP);
Sampler linear = sb.create();
sb.setMinification(Sampler.Value.NEAREST);
sb.setMagnification(Sampler.Value.NEAREST);
Sampler nearest = sb.create();
ProgramFragmentFixedFunction.Builder bf = new ProgramFragmentFixedFunction.Builder(sRS);
bf.setTexture(ProgramFragmentFixedFunction.Builder.EnvMode.MODULATE,
ProgramFragmentFixedFunction.Builder.Format.RGBA, 0);
bf.setVaryingColor(true);
ProgramFragment pfTexMip = bf.create();
pfTexMip.bindSampler(linear, 0);
bf.setVaryingColor(false);
ProgramFragment pfTexNearest = bf.create();
pfTexNearest.bindSampler(nearest, 0);
bf.setTexture(ProgramFragmentFixedFunction.Builder.EnvMode.MODULATE,
ProgramFragmentFixedFunction.Builder.Format.ALPHA, 0);
bf.setVaryingColor(true);
ProgramFragment pfTexMipAlpha = bf.create();
pfTexMipAlpha.bindSampler(linear, 0);
mScript.set_gPFTexNearest(pfTexNearest);
mScript.set_gPFTexMip(pfTexMip);
mScript.set_gPFTexMipAlpha(pfTexMipAlpha);
}
private void initProgramStore() {
ProgramStore.Builder bs = new ProgramStore.Builder(sRS);
bs.setDepthFunc(ProgramStore.DepthFunc.ALWAYS);
bs.setColorMaskEnabled(true,true,true,false);
bs.setDitherEnabled(true);
bs.setBlendFunc(ProgramStore.BlendSrcFunc.SRC_ALPHA,
ProgramStore.BlendDstFunc.ONE_MINUS_SRC_ALPHA);
mScript.set_gPS(bs.create());
}
private void initGl() {
}
private void initData() {
mScript.set_COLUMNS_PER_PAGE_PORTRAIT(Defines.COLUMNS_PER_PAGE_PORTRAIT);
mScript.set_ROWS_PER_PAGE_PORTRAIT(Defines.ROWS_PER_PAGE_PORTRAIT);
mScript.set_COLUMNS_PER_PAGE_LANDSCAPE(Defines.COLUMNS_PER_PAGE_LANDSCAPE);
mScript.set_ROWS_PER_PAGE_LANDSCAPE(Defines.ROWS_PER_PAGE_LANDSCAPE);
mHomeButtonNormal = Allocation.createFromBitmapResource(sRS, mRes,
R.drawable.home_button_normal);
mHomeButtonFocused = Allocation.createFromBitmapResource(sRS, mRes,
R.drawable.home_button_focused);
mHomeButtonPressed = Allocation.createFromBitmapResource(sRS, mRes,
R.drawable.home_button_pressed);
mScript.set_gHomeButton(mHomeButtonNormal);
mSelectionBitmap = Bitmap.createBitmap(Defines.SELECTION_TEXTURE_WIDTH_PX,
Defines.SELECTION_TEXTURE_HEIGHT_PX, Bitmap.Config.ARGB_8888);
mSelectionCanvas = new Canvas(mSelectionBitmap);
setApps(null);
}
void dirtyCheck() {
if (sZoomDirty) {
setZoom(mAllApps.sNextZoom, mAllApps.sAnimateNextZoom);
}
}
@SuppressWarnings({"ConstantConditions"})
private void setApps(ArrayList<ApplicationInfo> list) {
sRollo.pause();
final int count = list != null ? list.size() : 0;
int allocCount = count;
if (allocCount < 1) {
allocCount = 1;
}
mIcons = new Allocation[count];
mAllocIcons = Allocation.createSized(sRS, Element.ALLOCATION(sRS), allocCount);
mLabels = new Allocation[count];
mAllocLabels = Allocation.createSized(sRS, Element.ALLOCATION(sRS), allocCount);
mScript.set_gIconCount(count);
for (int i=0; i < count; i++) {
createAppIconAllocations(i, list.get(i));
}
saveAppsList();
android.util.Log.e("rs", "setApps");
sRollo.resume();
}
private void setZoom(float zoom, boolean animate) {
if (animate) {
sRollo.clearSelectedIcon();
sRollo.setHomeSelected(SELECTED_NONE);
}
sRollo.mScript.invoke_setZoom(zoom, animate ? 1 : 0);
}
private void createAppIconAllocations(int index, ApplicationInfo item) {
mIcons[index] = Allocation.createFromBitmap(sRS, item.iconBitmap,
Allocation.MipmapControl.MIPMAP_ON_SYNC_TO_TEXTURE,
Allocation.USAGE_GRAPHICS_TEXTURE);
mLabels[index] = Allocation.createFromBitmap(sRS, item.titleBitmap,
Allocation.MipmapControl.MIPMAP_ON_SYNC_TO_TEXTURE,
Allocation.USAGE_GRAPHICS_TEXTURE);
}
/**
* Puts the empty spaces at the end. Updates mState.iconCount. You must
* fill in the values and call saveAppsList().
*/
private void reallocAppsList(int count) {
Allocation[] icons = new Allocation[count];
mAllocIcons = Allocation.createSized(sRS, Element.ALLOCATION(sRS), count);
Allocation[] labels = new Allocation[count];
mAllocLabels = Allocation.createSized(sRS, Element.ALLOCATION(sRS), count);
final int oldCount = sRollo.mScript.get_gIconCount();
System.arraycopy(mIcons, 0, icons, 0, oldCount);
System.arraycopy(mLabels, 0, labels, 0, oldCount);
mIcons = icons;
mLabels = labels;
}
/**
* Handle the allocations for the new app. Make sure you call saveAppsList when done.
*/
private void addApp(int index, ApplicationInfo item) {
final int count = mScript.get_gIconCount() - index;
final int dest = index + 1;
System.arraycopy(mIcons, index, mIcons, dest, count);
System.arraycopy(mLabels, index, mLabels, dest, count);
createAppIconAllocations(index, item);
mScript.set_gIconCount(mScript.get_gIconCount() + 1);
}
/**
* Handle the allocations for the removed app. Make sure you call saveAppsList when done.
*/
private void removeApp(int index) {
final int count = mScript.get_gIconCount() - index - 1;
final int src = index + 1;
System.arraycopy(mIcons, src, mIcons, index, count);
System.arraycopy(mLabels, src, mLabels, index, count);
mScript.set_gIconCount(mScript.get_gIconCount() - 1);
final int last = mScript.get_gIconCount();
mIcons[last] = null;
mLabels[last] = null;
}
/**
* Send the apps list structures to RS.
*/
private void saveAppsList() {
// WTF: how could mScript be not null but mAllocIconIds null b/2460740.
if (mScript != null && mAllocIcons != null) {
if (mIcons.length > 0) {
mAllocIcons.copyFrom(mIcons);
mAllocLabels.copyFrom(mLabels);
}
mScript.bind_gIcons(mAllocIcons);
mScript.bind_gLabels(mAllocLabels);
}
}
void fling(float pos, float v) {
mScript.invoke_fling(pos, v);
}
void move(float pos) {
mScript.invoke_move(pos);
}
void moveTo(float row) {
mScript.invoke_moveTo(row);
}
/**
* You need to call save() on mState on your own after calling this.
*
* @return the index of the icon that was selected.
*/
int selectIcon(int x, int y, int pressed) {
if (mAllApps != null) {
final int index = mAllApps.chooseTappedIcon(x, y);
selectIcon(index, pressed);
return index;
} else {
return -1;
}
}
/**
* Select the icon at the given index.
*
* @param index The index.
* @param pressed one of SELECTED_PRESSED or SELECTED_FOCUSED
*/
void selectIcon(int index, int pressed) {
final ArrayList<ApplicationInfo> appsList = mAllApps.mAllAppsList;
if (appsList == null || index < 0 || index >= appsList.size()) {
if (mAllApps != null) {
mAllApps.mRestoreFocusIndex = index;
}
mScript.set_gSelectedIconIndex(-1);
if (mAllApps.mLastSelection == SELECTION_ICONS) {
mAllApps.mLastSelection = SELECTION_NONE;
}
} else {
if (pressed == SELECTED_FOCUSED) {
mAllApps.mLastSelection = SELECTION_ICONS;
}
int prev = mScript.get_gSelectedIconIndex();
mScript.set_gSelectedIconIndex(index);
ApplicationInfo info = appsList.get(index);
Bitmap selectionBitmap = mSelectionBitmap;
Utilities.drawSelectedAllAppsBitmap(mSelectionCanvas,
selectionBitmap.getWidth(), selectionBitmap.getHeight(),
pressed == SELECTED_PRESSED, info.iconBitmap);
Allocation si = Allocation.createFromBitmap(sRS, selectionBitmap);
mScript.set_gSelectedIconTexture(si);
if (prev != index) {
if (info.title != null && info.title.length() > 0) {
//setContentDescription(info.title);
mAllApps.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_SELECTED);
}
}
}
}
/**
* You need to call save() on mState on your own after calling this.
*/
void clearSelectedIcon() {
mScript.set_gSelectedIconIndex(-1);
}
void setHomeSelected(int mode) {
final int prev = mAllApps.mLastSelection;
switch (mode) {
case SELECTED_NONE:
mScript.set_gHomeButton(mHomeButtonNormal);
break;
case SELECTED_FOCUSED:
mAllApps.mLastSelection = SELECTION_HOME;
mScript.set_gHomeButton(mHomeButtonFocused);
if (prev != SELECTION_HOME) {
mAllApps.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_SELECTED);
}
break;
case SELECTED_PRESSED:
mScript.set_gHomeButton(mHomeButtonPressed);
break;
}
}
public void dumpState() {
Log.d(TAG, "sRollo.mWidth=" + mWidth);
Log.d(TAG, "sRollo.mHeight=" + mHeight);
Log.d(TAG, "sRollo.mIcons=" + Arrays.toString(mIcons));
if (mIcons != null) {
Log.d(TAG, "sRollo.mIcons.length=" + mIcons.length);
}
//Log.d(TAG, "sRollo.mState.newPositionX=" + mState.newPositionX);
//Log.d(TAG, "sRollo.mState.newTouchDown=" + mState.newTouchDown);
//Log.d(TAG, "sRollo.mState.flingVelocity=" + mState.flingVelocity);
//Log.d(TAG, "sRollo.mState.iconCount=" + mState.iconCount);
//Log.d(TAG, "sRollo.mState.selectedIconIndex=" + mState.selectedIconIndex);
//Log.d(TAG, "sRollo.mState.selectedIconTexture=" + mState.selectedIconTexture);
//Log.d(TAG, "sRollo.mState.zoomTarget=" + mState.zoomTarget);
//Log.d(TAG, "sRollo.mState.homeButtonId=" + mState.homeButtonId);
//Log.d(TAG, "sRollo.mState.targetPos=" + mState.targetPos);
}
}
public void dumpState() {
Log.d(TAG, "sRS=" + sRS);
Log.d(TAG, "sRollo=" + sRollo);
ApplicationInfo.dumpApplicationInfoList(TAG, "mAllAppsList", mAllAppsList);
Log.d(TAG, "mTouchXBorders=" + Arrays.toString(mTouchXBorders));
Log.d(TAG, "mTouchYBorders=" + Arrays.toString(mTouchYBorders));
Log.d(TAG, "mArrowNavigation=" + mArrowNavigation);
Log.d(TAG, "mStartedScrolling=" + mStartedScrolling);
Log.d(TAG, "mLastSelection=" + mLastSelection);
Log.d(TAG, "mLastSelectedIcon=" + mLastSelectedIcon);
Log.d(TAG, "mVelocityTracker=" + mVelocityTracker);
Log.d(TAG, "mTouchTracking=" + mTouchTracking);
Log.d(TAG, "mShouldGainFocus=" + mShouldGainFocus);
Log.d(TAG, "sZoomDirty=" + sZoomDirty);
Log.d(TAG, "sAnimateNextZoom=" + sAnimateNextZoom);
Log.d(TAG, "mZoom=" + mZoom);
Log.d(TAG, "mScrollPos=" + sRollo.mScrollPos);
Log.d(TAG, "mVelocity=" + mVelocity);
Log.d(TAG, "mMessageProc=" + mMessageProc);
if (sRollo != null) {
sRollo.dumpState();
}
if (sRS != null) {
sRS.contextDump();
}
}
public void reset() {
// Do nothing
}
}