Android Open Source - iosched2011 Workspace






From Project

Back to project page iosched2011.

License

The source code is released under:

Apache License

If you think the Android project iosched2011 listed in this page is inappropriate, such as containing malicious code/tools or violating the copyright, please email info at java2s dot com, thanks.

Java Source Code

/*
 * Copyright 2011 Google Inc.//from   w  w  w. j  av a 2 s  .com
 *
 * 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.google.android.apps.iosched.ui.widget;

import com.google.android.apps.iosched.util.MotionEventUtils;
import com.google.android.apps.iosched.util.ReflectionUtils;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Rect;
import android.graphics.drawable.Drawable;
import android.os.Parcel;
import android.os.Parcelable;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.VelocityTracker;
import android.view.View;
import android.view.ViewConfiguration;
import android.view.ViewGroup;
import android.view.ViewParent;
import android.widget.Scroller;

import java.util.ArrayList;

/**
 * A {@link android.view.ViewGroup} that shows one child at a time, allowing the user to swipe
 * horizontally to page between other child views. Based on <code>Workspace.java</code> in the
 * <code>Launcher.git</code> AOSP project.
 */
public class Workspace extends ViewGroup {
    private static final String TAG = "Workspace";

    private static final int INVALID_SCREEN = -1;

    /**
     * The velocity at which a fling gesture will cause us to snap to the next screen
     */
    private static final int SNAP_VELOCITY = 500;

    /**
     * The user needs to drag at least this much for it to be considered a fling gesture. This
     * reduces the chance of a random twitch sending the user to the next screen.
     */
    // TODO: refactor
    private static final int MIN_LENGTH_FOR_FLING = 100;

    private int mDefaultScreen;

    private boolean mFirstLayout = true;
    private boolean mHasLaidOut = false;

    private int mCurrentScreen;
    private int mNextScreen = INVALID_SCREEN;
    private Scroller mScroller;
    private VelocityTracker mVelocityTracker;

    /**
     * X position of the active pointer when it was first pressed down.
     */
    private float mDownMotionX;

    /**
     * Y position of the active pointer when it was first pressed down.
     */
    private float mDownMotionY;

    /**
     * This view's X scroll offset when the active pointer was first pressed down.
     */
    private int mDownScrollX;

    private final static int TOUCH_STATE_REST = 0;
    private final static int TOUCH_STATE_SCROLLING = 1;

    private int mTouchState = TOUCH_STATE_REST;

    private OnLongClickListener mLongClickListener;

    private boolean mAllowLongPress = true;

    private int mTouchSlop;
    private int mPagingTouchSlop;
    private int mMaximumVelocity;

    private static final int INVALID_POINTER = -1;

    private int mActivePointerId = INVALID_POINTER;

    private Drawable mSeparatorDrawable;

    private OnScreenChangeListener mOnScreenChangeListener;
    private OnScrollListener mOnScrollListener;

    private boolean mLocked;

    private int mDeferredScreenChange = -1;
    private boolean mDeferredScreenChangeFast = false;
    private boolean mDeferredNotify = false;

    private boolean mIgnoreChildFocusRequests;

    private boolean mIsVerbose = false;

    public interface OnScreenChangeListener {
        void onScreenChanged(View newScreen, int newScreenIndex);
        void onScreenChanging(View newScreen, int newScreenIndex);
    }

    public interface OnScrollListener {
        void onScroll(float screenFraction);
    }

    /**
     * Used to inflate the com.google.android.ext.workspace.Workspace from XML.
     *
     * @param context The application's context.
     * @param attrs   The attributes set containing the com.google.android.ext.workspace.Workspace's
     *                customization values.
     */
    public Workspace(Context context, AttributeSet attrs) {
        super(context, attrs);

        mDefaultScreen = 0;
        mLocked = false;

        setHapticFeedbackEnabled(false);
        initWorkspace();

        mIsVerbose = Log.isLoggable(TAG, Log.VERBOSE);
    }

    /**
     * Initializes various states for this workspace.
     */
    private void initWorkspace() {
        mScroller = new Scroller(getContext());
        mCurrentScreen = mDefaultScreen;

        final ViewConfiguration configuration = ViewConfiguration.get(getContext());
        mTouchSlop = configuration.getScaledTouchSlop();
        mMaximumVelocity = configuration.getScaledMaximumFlingVelocity();

        mPagingTouchSlop = ReflectionUtils.callWithDefault(configuration,
                "getScaledPagingTouchSlop", mTouchSlop * 2);
    }

    /**
     * Returns the index of the currently displayed screen.
     */
    int getCurrentScreen() {
        return mCurrentScreen;
    }

    /**
     * Returns the number of screens currently contained in this Workspace.
     */
    int getScreenCount() {
        int childCount = getChildCount();
        if (mSeparatorDrawable != null) {
            return (childCount + 1) / 2;
        }
        return childCount;
    }

    View getScreenAt(int index) {
        if (mSeparatorDrawable == null) {
            return getChildAt(index);
        }
        return getChildAt(index * 2);
    }

    int getScrollWidth() {
        int w = getWidth();
        if (mSeparatorDrawable != null) {
            w += mSeparatorDrawable.getIntrinsicWidth();
        }
        return w;
    }

    void handleScreenChangeCompletion(int currentScreen) {
        mCurrentScreen = currentScreen;
        View screen = getScreenAt(mCurrentScreen);
        //screen.requestFocus();
        try {
            ReflectionUtils.tryInvoke(screen, "dispatchDisplayHint",
                    new Class[]{int.class}, View.VISIBLE);
            invalidate();
        } catch (NullPointerException e) {
            Log.e(TAG, "Caught NullPointerException", e);
        }
        notifyScreenChangeListener(mCurrentScreen, true);
    }

    void notifyScreenChangeListener(int whichScreen, boolean changeComplete) {
        if (mOnScreenChangeListener != null) {
            if (changeComplete)
                mOnScreenChangeListener.onScreenChanged(getScreenAt(whichScreen), whichScreen);
            else
                mOnScreenChangeListener.onScreenChanging(getScreenAt(whichScreen), whichScreen);
        }
        if (mOnScrollListener != null) {
            mOnScrollListener.onScroll(getCurrentScreenFraction());
        }
    }

    /**
     * Registers the specified listener on each screen contained in this workspace.
     *
     * @param listener The listener used to respond to long clicks.
     */
    @Override
    public void setOnLongClickListener(OnLongClickListener listener) {
        mLongClickListener = listener;
        final int count = getScreenCount();
        for (int i = 0; i < count; i++) {
            getScreenAt(i).setOnLongClickListener(listener);
        }
    }

    @Override
    public void computeScroll() {
        if (mScroller.computeScrollOffset()) {
            scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
            if (mOnScrollListener != null) {
                mOnScrollListener.onScroll(getCurrentScreenFraction());
            }
            postInvalidate();
        } else if (mNextScreen != INVALID_SCREEN) {
            // The scroller has finished.
            handleScreenChangeCompletion(Math.max(0, Math.min(mNextScreen, getScreenCount() - 1)));
            mNextScreen = INVALID_SCREEN;
        }
    }

    @Override
    protected void dispatchDraw(Canvas canvas) {
        boolean restore = false;
        int restoreCount = 0;

        // ViewGroup.dispatchDraw() supports many features we don't need:
        // clip to padding, layout animation, animation listener, disappearing
        // children, etc. The following implementation attempts to fast-track
        // the drawing dispatch by drawing only what we know needs to be drawn.

        boolean fastDraw = mTouchState != TOUCH_STATE_SCROLLING && mNextScreen == INVALID_SCREEN;
        // If we are not scrolling or flinging, draw only the current screen
        if (fastDraw) {
            if (getScreenAt(mCurrentScreen) != null) {
                drawChild(canvas, getScreenAt(mCurrentScreen), getDrawingTime());
            }
        } else {
            final long drawingTime = getDrawingTime();
            // If we are flinging, draw only the current screen and the target screen
            if (mNextScreen >= 0 && mNextScreen < getScreenCount() &&
                    Math.abs(mCurrentScreen - mNextScreen) == 1) {
                drawChild(canvas, getScreenAt(mCurrentScreen), drawingTime);
                drawChild(canvas, getScreenAt(mNextScreen), drawingTime);
            } else {
                // If we are scrolling, draw all of our children
                final int count = getChildCount();
                for (int i = 0; i < count; i++) {
                    drawChild(canvas, getChildAt(i), drawingTime);
                }
            }
        }

        if (restore) {
            canvas.restoreToCount(restoreCount);
        }
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);

        // The children are given the same width and height as the workspace
        final int count = getChildCount();
        for (int i = 0; i < count; i++) {
            if (mSeparatorDrawable != null && (i & 1) == 1) {
                // separator
                getChildAt(i).measure(mSeparatorDrawable.getIntrinsicWidth(), heightMeasureSpec);
            } else {
                getChildAt(i).measure(widthMeasureSpec, heightMeasureSpec);
            }
        }

        if (mFirstLayout) {
            setHorizontalScrollBarEnabled(false);
            int width = MeasureSpec.getSize(widthMeasureSpec);
            if (mSeparatorDrawable != null) {
                width += mSeparatorDrawable.getIntrinsicWidth();
            }
            scrollTo(mCurrentScreen * width, 0);
            setHorizontalScrollBarEnabled(true);
            mFirstLayout = false;
        }
    }

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        int childLeft = 0;

        final int count = getChildCount();
        for (int i = 0; i < count; i++) {
            final View child = getChildAt(i);
            if (child.getVisibility() != View.GONE) {
                final int childWidth = child.getMeasuredWidth();
                child.layout(childLeft, 0, childLeft + childWidth, child.getMeasuredHeight());
                childLeft += childWidth;
            }
        }

        mHasLaidOut = true;
        if (mDeferredScreenChange >= 0) {
            snapToScreen(mDeferredScreenChange, mDeferredScreenChangeFast, mDeferredNotify);
            mDeferredScreenChange = -1;
            mDeferredScreenChangeFast = false;
        }
    }

    @Override
    public boolean requestChildRectangleOnScreen(View child, Rect rectangle, boolean immediate) {
        int screen = indexOfChild(child);
        if (mIgnoreChildFocusRequests && !mScroller.isFinished()) {
            Log.w(TAG, "Ignoring child focus request: request " + mCurrentScreen + " -> " + screen);
            return false;
        }
        if (screen != mCurrentScreen || !mScroller.isFinished()) {
            snapToScreen(screen);
            return true;
        }
        return false;
    }

    @Override
    protected boolean onRequestFocusInDescendants(int direction, Rect previouslyFocusedRect) {
        int focusableScreen;
        if (mNextScreen != INVALID_SCREEN) {
            focusableScreen = mNextScreen;
        } else {
            focusableScreen = mCurrentScreen;
        }
        View v = getScreenAt(focusableScreen);
        if (v != null) {
            return v.requestFocus(direction, previouslyFocusedRect);
        }
        return false;
    }

    @Override
    public boolean dispatchUnhandledMove(View focused, int direction) {
        if (direction == View.FOCUS_LEFT) {
            if (getCurrentScreen() > 0) {
                snapToScreen(getCurrentScreen() - 1);
                return true;
            }
        } else if (direction == View.FOCUS_RIGHT) {
            if (getCurrentScreen() < getScreenCount() - 1) {
                snapToScreen(getCurrentScreen() + 1);
                return true;
            }
        }
        return super.dispatchUnhandledMove(focused, direction);
    }

    @Override
    public void addFocusables(ArrayList<View> views, int direction, int focusableMode) {
        View focusableSourceScreen = null;
        if (mCurrentScreen >= 0 && mCurrentScreen < getScreenCount()) {
            focusableSourceScreen = getScreenAt(mCurrentScreen);
        }
        if (direction == View.FOCUS_LEFT) {
            if (mCurrentScreen > 0) {
                focusableSourceScreen = getScreenAt(mCurrentScreen - 1);
            }
        } else if (direction == View.FOCUS_RIGHT) {
            if (mCurrentScreen < getScreenCount() - 1) {
                focusableSourceScreen = getScreenAt(mCurrentScreen + 1);
            }
        }

        if (focusableSourceScreen != null) {
            focusableSourceScreen.addFocusables(views, direction, focusableMode);
        }
    }

    /**
     * If one of our descendant views decides that it could be focused now, only pass that along if
     * it's on the current screen.
     *
     * This happens when live folders requery, and if they're off screen, they end up calling
     * requestFocus, which pulls it on screen.
     */
    @Override
    public void focusableViewAvailable(View focused) {
        View current = getScreenAt(mCurrentScreen);
        View v = focused;
        ViewParent parent;
        while (true) {
            if (v == current) {
                super.focusableViewAvailable(focused);
                return;
            }
            if (v == this) {
                return;
            }
            parent = v.getParent();
            if (parent instanceof View) {
                v = (View) v.getParent();
            } else {
                return;
            }
        }
    }

    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        /*
         * This method JUST determines whether we want to intercept the motion.
         * If we return true, onTouchEvent will be called and we do the actual
         * scrolling there.
         */

        // Begin tracking velocity even before we have intercepted touch events.
        if (mVelocityTracker == null) {
            mVelocityTracker = VelocityTracker.obtain();
        }
        mVelocityTracker.addMovement(ev);

        /*
         * Shortcut the most recurring case: the user is in the dragging
         * state and he is moving his finger.  We want to intercept this
         * motion.
         */
        final int action = ev.getAction();
        if (mIsVerbose) {
            Log.v(TAG, "onInterceptTouchEvent: " + (ev.getAction() & MotionEventUtils.ACTION_MASK));
        }
        if (((action & MotionEventUtils.ACTION_MASK) == MotionEvent.ACTION_MOVE)
                && (mTouchState == TOUCH_STATE_SCROLLING)) {
            if (mIsVerbose) {
                Log.v(TAG, "Intercepting touch events");
            }
            return true;
        }

        switch (action & MotionEventUtils.ACTION_MASK) {
            case MotionEvent.ACTION_MOVE: {
                if (mLocked) {
                    // we're locked on the current screen, don't allow moving
                    break;
                }

                /*
                 * Locally do absolute value. mDownMotionX is set to the y value
                 * of the down event.
                 */
                final int pointerIndex = MotionEventUtils.findPointerIndex(ev, mActivePointerId);
                final float x = MotionEventUtils.getX(ev, pointerIndex);
                final float y = MotionEventUtils.getY(ev, pointerIndex);
                final int xDiff = (int) Math.abs(x - mDownMotionX);
                final int yDiff = (int) Math.abs(y - mDownMotionY);

                boolean xPaged = xDiff > mPagingTouchSlop;
                boolean xMoved = xDiff > mTouchSlop;
                boolean yMoved = yDiff > mTouchSlop;

                if (xMoved || yMoved) {
                    if (xPaged) {
                        // Scroll if the user moved far enough along the X axis
                        mTouchState = TOUCH_STATE_SCROLLING;
                    }
                    // Either way, cancel any pending longpress
                    if (mAllowLongPress) {
                        mAllowLongPress = false;
                        // Try canceling the long press. It could also have been scheduled
                        // by a distant descendant, so use the mAllowLongPress flag to block
                        // everything
                        final View currentScreen = getScreenAt(mCurrentScreen);
                        if (currentScreen != null) {
                            currentScreen.cancelLongPress();
                        }
                    }
                }
                break;
            }

            case MotionEvent.ACTION_DOWN: {
                final float x = ev.getX();
                final float y = ev.getY();
                // Remember location of down touch
                mDownMotionX = x;
                mDownMotionY = y;
                mDownScrollX = getScrollX();
                mActivePointerId = MotionEventUtils.getPointerId(ev, 0);
                mAllowLongPress = true;

                /*
                 * If being flinged and user touches the screen, initiate drag;
                 * otherwise don't.  mScroller.isFinished should be false when
                 * being flinged.
                 */
                mTouchState = mScroller.isFinished() ? TOUCH_STATE_REST : TOUCH_STATE_SCROLLING;
                break;
            }

            case MotionEvent.ACTION_CANCEL:
            case MotionEvent.ACTION_UP:
                // Release the drag
                mTouchState = TOUCH_STATE_REST;
                mAllowLongPress = false;
                mActivePointerId = INVALID_POINTER;
                if (mVelocityTracker == null) {
                    mVelocityTracker.recycle();
                    mVelocityTracker = null;
                }
                break;

            case MotionEventUtils.ACTION_POINTER_UP:
                onSecondaryPointerUp(ev);
                break;
        }

        /*
         * The only time we want to intercept motion events is if we are in the
         * drag mode.
         */
        boolean intercept = mTouchState != TOUCH_STATE_REST;
        if (mIsVerbose) {
            Log.v(TAG, "Intercepting touch events: " + Boolean.toString(intercept));
        }
        return intercept;
    }

    void onSecondaryPointerUp(MotionEvent ev) {
        final int pointerIndex = (ev.getAction() & MotionEventUtils.ACTION_POINTER_INDEX_MASK) >>
                MotionEventUtils.ACTION_POINTER_INDEX_SHIFT;
        final int pointerId = MotionEventUtils.getPointerId(ev, pointerIndex);
        if (pointerId == mActivePointerId) {
            // This was our active pointer going up. Choose a new
            // active pointer and adjust accordingly.
            // TODO: Make this decision more intelligent.
            final int newPointerIndex = pointerIndex == 0 ? 1 : 0;
            mDownMotionX = MotionEventUtils.getX(ev, newPointerIndex);
            mDownMotionX = MotionEventUtils.getY(ev, newPointerIndex);
            mDownScrollX = getScrollX();
            mActivePointerId = MotionEventUtils.getPointerId(ev, newPointerIndex);
            if (mVelocityTracker != null) {
                mVelocityTracker.clear();
            }
        }
    }

    @Override
    public void requestChildFocus(View child, View focused) {
        super.requestChildFocus(child, focused);
        int screen = indexOfChild(child);
        if (mSeparatorDrawable != null) {
            screen /= 2;
        }
        if (screen >= 0 && !isInTouchMode()) {
            snapToScreen(screen);
        }
    }

    @Override
    public boolean onTouchEvent(MotionEvent ev) {
        if (mIsVerbose) {
            Log.v(TAG, "onTouchEvent: " + (ev.getAction() & MotionEventUtils.ACTION_MASK));
        }

        if (mVelocityTracker == null) {
            mVelocityTracker = VelocityTracker.obtain();
        }
        mVelocityTracker.addMovement(ev);

        final int action = ev.getAction();

        switch (action & MotionEventUtils.ACTION_MASK) {
            case MotionEvent.ACTION_DOWN:
                // If being flinged and user touches, stop the fling. isFinished
                // will be false if being flinged.
                if (!mScroller.isFinished()) {
                    mScroller.abortAnimation();
                }

                // Remember where the motion event started
                mDownMotionX = ev.getX();
                mDownMotionY = ev.getY();
                mDownScrollX = getScrollX();
                mActivePointerId = MotionEventUtils.getPointerId(ev, 0);
                break;

            case MotionEvent.ACTION_MOVE:
                if (mIsVerbose) {
                    Log.v(TAG, "mTouchState=" + mTouchState);
                }

                if (mTouchState == TOUCH_STATE_SCROLLING) {
                    // Scroll to follow the motion event
                    final int pointerIndex = MotionEventUtils
                            .findPointerIndex(ev, mActivePointerId);
                    final float x = MotionEventUtils.getX(ev, pointerIndex);

                    final View lastChild = getChildAt(getChildCount() - 1);
                    final int maxScrollX = lastChild.getRight() - getWidth();
                    scrollTo(Math.max(0, Math.min(maxScrollX,
                            (int)(mDownScrollX + mDownMotionX - x
                            ))), 0);
                    if (mOnScrollListener != null) {
                        mOnScrollListener.onScroll(getCurrentScreenFraction());
                    }

                } else if (mTouchState == TOUCH_STATE_REST) {
                    if (mLocked) {
                        // we're locked on the current screen, don't allow moving
                        break;
                    }

                    /*
                     * Locally do absolute value. mLastMotionX is set to the y value
                     * of the down event.
                     */
                    final int pointerIndex = MotionEventUtils.findPointerIndex(ev, mActivePointerId);
                    final float x = MotionEventUtils.getX(ev, pointerIndex);
                    final float y = MotionEventUtils.getY(ev, pointerIndex);
                    final int xDiff = (int) Math.abs(x - mDownMotionX);
                    final int yDiff = (int) Math.abs(y - mDownMotionY);

                    boolean xPaged = xDiff > mPagingTouchSlop;
                    boolean xMoved = xDiff > mTouchSlop;
                    boolean yMoved = yDiff > mTouchSlop;

                    if (xMoved || yMoved) {
                        if (xPaged) {
                            // Scroll if the user moved far enough along the X axis
                            mTouchState = TOUCH_STATE_SCROLLING;
                        }
                        // Either way, cancel any pending longpress
                        if (mAllowLongPress) {
                            mAllowLongPress = false;
                            // Try canceling the long press. It could also have been scheduled
                            // by a distant descendant, so use the mAllowLongPress flag to block
                            // everything
                            final View currentScreen = getScreenAt(mCurrentScreen);
                            if (currentScreen != null) {
                                currentScreen.cancelLongPress();
                            }
                        }
                    }
                }
                break;

            case MotionEvent.ACTION_UP:
                if (mTouchState == TOUCH_STATE_SCROLLING) {
                    final int activePointerId = mActivePointerId;
                    final int pointerIndex = MotionEventUtils.findPointerIndex(ev, activePointerId);
                    final float x = MotionEventUtils.getX(ev, pointerIndex);
                    final VelocityTracker velocityTracker = mVelocityTracker;
                    velocityTracker.computeCurrentVelocity(1000, mMaximumVelocity);
                    //TODO(minsdk8): int velocityX = (int) MotionEventUtils.getXVelocity(velocityTracker, activePointerId);
                    int velocityX = (int) velocityTracker.getXVelocity();
                    boolean isFling = Math.abs(mDownMotionX - x) > MIN_LENGTH_FOR_FLING;

                    final float scrolledPos = getCurrentScreenFraction();
                    final int whichScreen = Math.round(scrolledPos);

                    if (isFling && mIsVerbose) {
                        Log.v(TAG, "isFling, whichScreen=" + whichScreen
                                + " scrolledPos=" + scrolledPos
                                + " mCurrentScreen=" + mCurrentScreen
                                + " velocityX=" + velocityX);
                    }
                    if (isFling && velocityX > SNAP_VELOCITY && mCurrentScreen > 0) {
                        // Fling hard enough to move left
                        // Don't fling across more than one screen at a time.
                        final int bound = scrolledPos <= whichScreen ?
                                mCurrentScreen - 1 : mCurrentScreen;
                        snapToScreen(Math.min(whichScreen, bound));
                    } else if (isFling && velocityX < -SNAP_VELOCITY &&
                            mCurrentScreen < getChildCount() - 1) {
                        // Fling hard enough to move right
                        // Don't fling across more than one screen at a time.
                        final int bound = scrolledPos >= whichScreen ?
                                mCurrentScreen + 1 : mCurrentScreen;
                        snapToScreen(Math.max(whichScreen, bound));
                    } else {
                        snapToDestination();
                    }
                }
                mTouchState = TOUCH_STATE_REST;
                mActivePointerId = INVALID_POINTER;
                // Intentially fall through to cancel

            case MotionEvent.ACTION_CANCEL:
                mTouchState = TOUCH_STATE_REST;
                mActivePointerId = INVALID_POINTER;
                if (mVelocityTracker != null) {
                    mVelocityTracker.recycle();
                    mVelocityTracker = null;
                }
                break;

            case MotionEventUtils.ACTION_POINTER_UP:
                onSecondaryPointerUp(ev);
                break;
        }

        return true;
    }

    /**
     * Returns the current scroll position as a float value from 0 to the number of screens.
     * Fractional values indicate that the user is mid-scroll or mid-fling, and whole values
     * indicate that the Workspace is currently snapped to a screen.
     */
    float getCurrentScreenFraction() {
        if (!mHasLaidOut) {
            return mCurrentScreen;
        }
        final int scrollX = getScrollX();
        final int screenWidth = getWidth();
        return (float) scrollX / screenWidth;
    }

    void snapToDestination() {
        final int screenWidth = getScrollWidth();
        final int whichScreen = (getScrollX() + (screenWidth / 2)) / screenWidth;

        snapToScreen(whichScreen);
    }

    void snapToScreen(int whichScreen) {
        snapToScreen(whichScreen, false, true);
    }

    void snapToScreen(int whichScreen, boolean fast, boolean notify) {
        if (!mHasLaidOut) { // Can't handle scrolling until we are laid out.
            mDeferredScreenChange = whichScreen;
            mDeferredScreenChangeFast = fast;
            mDeferredNotify = notify;
            return;
        }

        if (mIsVerbose) {
            Log.v(TAG, "Snapping to screen: " + whichScreen);
        }

        whichScreen = Math.max(0, Math.min(whichScreen, getScreenCount() - 1));

        final int screenDelta = Math.abs(whichScreen - mCurrentScreen);

        final boolean screenChanging =
                (mNextScreen != INVALID_SCREEN && mNextScreen != whichScreen) ||
                        (mCurrentScreen != whichScreen);

        mNextScreen = whichScreen;

        View focusedChild = getFocusedChild();
        boolean setTabFocus = false;
        if (focusedChild != null && screenDelta != 0 && focusedChild == getScreenAt(
                mCurrentScreen)) {
            // clearing the focus of the child will cause focus to jump to the tabs,
            // which will in turn cause snapToScreen to be called again with a different
            // value. To prevent this, we temporarily disable the OnTabClickListener
            //if (mTabRow != null) {
            //    mTabRow.setOnTabClickListener(null);
            //}
            //focusedChild.clearFocus();
            //setTabRow(mTabRow); // restore the listener
            //setTabFocus = true;
        }

        final int newX = whichScreen * getScrollWidth();
        final int sX = getScrollX();
        final int delta = newX - sX;
        int duration = screenDelta * 300;
        awakenScrollBars(duration);
        if (duration == 0) {
            duration = Math.abs(delta);
        }
        if (fast) {
            duration = 0;
        }

        if (mNextScreen != mCurrentScreen) {
            // make the current listview hide its filter popup
            final View screenAt = getScreenAt(mCurrentScreen);
            if (screenAt != null) {
                ReflectionUtils.tryInvoke(screenAt, "dispatchDisplayHint",
                        new Class[]{int.class}, View.INVISIBLE);
            } else {
                Log.e(TAG, "Screen at index was null. mCurrentScreen = " + mCurrentScreen);
                return;
            }

            // showing the filter popup for the next listview needs to be delayed
            // until we've fully moved to that listview, since otherwise the
            // popup will appear at the wrong place on the screen
            //removeCallbacks(mFilterWindowEnabler);
            //postDelayed(mFilterWindowEnabler, duration + 10);

            // NOTE: moved to computeScroll and handleScreenChangeCompletion()
        }

        if (!mScroller.isFinished()) {
            mScroller.abortAnimation();
        }
        mScroller.startScroll(sX, 0, delta, 0, duration);
        if (screenChanging && notify) {
            notifyScreenChangeListener(mNextScreen, false);
        }
        invalidate();
    }

    @Override
    protected Parcelable onSaveInstanceState() {
        final SavedState state = new SavedState(super.onSaveInstanceState());
        state.currentScreen = mCurrentScreen;
        return state;
    }

    @Override
    protected void onRestoreInstanceState(Parcelable state) {
        SavedState savedState = (SavedState) state;
        super.onRestoreInstanceState(savedState.getSuperState());
        if (savedState.currentScreen != -1) {
            snapToScreen(savedState.currentScreen, true, true);
        }
    }

    /**
     * @return True is long presses are still allowed for the current touch
     */
    boolean allowLongPress() {
        return mAllowLongPress;
    }

    /**
     * Register a callback to be invoked when the screen is changed, either programmatically or via
     * user interaction.  Will automatically trigger a callback.
     *
     * @param screenChangeListener The callback.
     */
    public void setOnScreenChangeListener(OnScreenChangeListener screenChangeListener) {
        setOnScreenChangeListener(screenChangeListener, true);
    }

    /**
     * Register a callback to be invoked when the screen is changed, either programmatically or via
     * user interaction.
     *
     * @param screenChangeListener The callback.
     * @param notifyImmediately Whether to trigger a notification immediately
     */
    public void setOnScreenChangeListener(OnScreenChangeListener screenChangeListener,
            boolean notifyImmediately) {
        mOnScreenChangeListener = screenChangeListener;
        if (mOnScreenChangeListener != null && notifyImmediately) {
            mOnScreenChangeListener.onScreenChanged(getScreenAt(mCurrentScreen), mCurrentScreen);
        }
    }

    /**
     * Register a callback to be invoked when this Workspace is mid-scroll or mid-fling, either
     * due to user interaction or programmatic changes in the current screen index.
     *
     * @param scrollListener The callback.
     * @param notifyImmediately Whether to trigger a notification immediately
     */
    public void setOnScrollListener(OnScrollListener scrollListener, boolean notifyImmediately) {
        mOnScrollListener = scrollListener;
        if (mOnScrollListener != null && notifyImmediately) {
            mOnScrollListener.onScroll(getCurrentScreenFraction());
        }
    }

    /**
     * Scrolls to the given screen.
     */
    public void setCurrentScreen(int screenIndex) {
        snapToScreen(Math.max(0, Math.min(getScreenCount() - 1, screenIndex)));
    }

    /**
     * Scrolls to the given screen fast (no matter how large the scroll distance is)
     *
     * @param screenIndex
     */
    public void setCurrentScreenNow(int screenIndex) {
        setCurrentScreenNow(screenIndex, true);
    }

    public void setCurrentScreenNow(int screenIndex, boolean notify) {
        snapToScreen(Math.max(0, Math.min(getScreenCount() - 1, screenIndex)), true, notify);
    }

    /**
     * Scrolls to the screen adjacent to the current screen on the left, if it exists. This method
     * is a no-op if the Workspace is currently locked.
     */
    public void scrollLeft() {
        if (mLocked) {
            return;
        }
        if (mScroller.isFinished()) {
            if (mCurrentScreen > 0) {
                snapToScreen(mCurrentScreen - 1);
            }
        } else {
            if (mNextScreen > 0) {
                snapToScreen(mNextScreen - 1);
            }
        }
    }

    /**
     * Scrolls to the screen adjacent to the current screen on the right, if it exists. This method
     * is a no-op if the Workspace is currently locked.
     */
    public void scrollRight() {
        if (mLocked) {
            return;
        }
        if (mScroller.isFinished()) {
            if (mCurrentScreen < getChildCount() - 1) {
                snapToScreen(mCurrentScreen + 1);
            }
        } else {
            if (mNextScreen < getChildCount() - 1) {
                snapToScreen(mNextScreen + 1);
            }
        }
    }

    /**
     * If set, invocations of requestChildRectangleOnScreen() will be ignored.
     */
    public void setIgnoreChildFocusRequests(boolean mIgnoreChildFocusRequests) {
        this.mIgnoreChildFocusRequests = mIgnoreChildFocusRequests;
    }

    public void markViewSelected(View v) {
        mCurrentScreen = indexOfChild(v);
    }

    /**
     * Locks the current screen, preventing users from changing screens by swiping.
     */
    public void lockCurrentScreen() {
        mLocked = true;
    }

    /**
     * Unlocks the current screen, if it was previously locked. See also {@link
     * Workspace#lockCurrentScreen()}.
     */
    public void unlockCurrentScreen() {
        mLocked = false;
    }

    /**
     * Sets the resource ID of the separator drawable to use between adjacent screens.
     */
    public void setSeparator(int resId) {
        if (mSeparatorDrawable != null && resId == 0) {
            // remove existing separators
            mSeparatorDrawable = null;
            int num = getChildCount();
            for (int i = num - 2; i > 0; i -= 2) {
                removeViewAt(i);
            }
            requestLayout();
        } else if (resId != 0) {
            // add or update separators
            if (mSeparatorDrawable == null) {
                // add
                int numsep = getChildCount();
                int insertIndex = 1;
                mSeparatorDrawable = getResources().getDrawable(resId);
                for (int i = 1; i < numsep; i++) {
                    View v = new View(getContext());
                    v.setBackgroundDrawable(mSeparatorDrawable);
                    LayoutParams lp = new LayoutParams(LayoutParams.WRAP_CONTENT,
                            LayoutParams.FILL_PARENT);
                    v.setLayoutParams(lp);
                    addView(v, insertIndex);
                    insertIndex += 2;
                }
                requestLayout();
            } else {
                // update
                mSeparatorDrawable = getResources().getDrawable(resId);
                int num = getChildCount();
                for (int i = num - 2; i > 0; i -= 2) {
                    getChildAt(i).setBackgroundDrawable(mSeparatorDrawable);
                }
                requestLayout();
            }
        }
    }

    private static class SavedState extends BaseSavedState {
        int currentScreen = -1;

        SavedState(Parcelable superState) {
            super(superState);
        }

        private SavedState(Parcel in) {
            super(in);
            currentScreen = in.readInt();
        }

        @Override
        public void writeToParcel(Parcel out, int flags) {
            super.writeToParcel(out, flags);
            out.writeInt(currentScreen);
        }

        public static final Parcelable.Creator<SavedState> CREATOR =
                new Parcelable.Creator<SavedState>() {
                    public SavedState createFromParcel(Parcel in) {
                        return new SavedState(in);
                    }

                    public SavedState[] newArray(int size) {
                        return new SavedState[size];
                    }
                };
    }

    public void addViewToFront(View v) {
        mCurrentScreen++;
        addView(v, 0);
    }

    public void removeViewFromFront() {
        mCurrentScreen--;
        removeViewAt(0);
    }

    public void addViewToBack(View v) {
        addView(v);
    }

    public void removeViewFromBack() {
        removeViewAt(getChildCount() - 1);
    }
}




Java Source Code List

com.google.android.apps.iosched.io.LocalBlocksHandler.java
com.google.android.apps.iosched.io.LocalExecutor.java
com.google.android.apps.iosched.io.LocalRoomsHandler.java
com.google.android.apps.iosched.io.LocalSearchSuggestHandler.java
com.google.android.apps.iosched.io.LocalSessionsHandler.java
com.google.android.apps.iosched.io.LocalTracksHandler.java
com.google.android.apps.iosched.io.RemoteExecutor.java
com.google.android.apps.iosched.io.RemoteSessionsHandler.java
com.google.android.apps.iosched.io.RemoteSpeakersHandler.java
com.google.android.apps.iosched.io.RemoteVendorsHandler.java
com.google.android.apps.iosched.io.RemoteWorksheetsHandler.java
com.google.android.apps.iosched.io.SessionsHandlerTest.java
com.google.android.apps.iosched.io.StubHttpClient.java
com.google.android.apps.iosched.io.XmlHandler.java
com.google.android.apps.iosched.provider.ScheduleContract.java
com.google.android.apps.iosched.provider.ScheduleDatabase.java
com.google.android.apps.iosched.provider.ScheduleProvider.java
com.google.android.apps.iosched.service.SyncService.java
com.google.android.apps.iosched.ui.BaseActivity.java
com.google.android.apps.iosched.ui.BaseMultiPaneActivity.java
com.google.android.apps.iosched.ui.BaseSinglePaneActivity.java
com.google.android.apps.iosched.ui.BulletinActivity.java
com.google.android.apps.iosched.ui.BulletinFragment.java
com.google.android.apps.iosched.ui.DashboardFragment.java
com.google.android.apps.iosched.ui.HomeActivity.java
com.google.android.apps.iosched.ui.MapFragment.java
com.google.android.apps.iosched.ui.ScheduleFragment.java
com.google.android.apps.iosched.ui.SearchActivity.java
com.google.android.apps.iosched.ui.SessionDetailFragment.java
com.google.android.apps.iosched.ui.SessionsFragment.java
com.google.android.apps.iosched.ui.StarredActivity.java
com.google.android.apps.iosched.ui.TagStreamActivity.java
com.google.android.apps.iosched.ui.TagStreamFragment.java
com.google.android.apps.iosched.ui.TracksAdapter.java
com.google.android.apps.iosched.ui.TracksFragment.java
com.google.android.apps.iosched.ui.VendorDetailFragment.java
com.google.android.apps.iosched.ui.VendorsFragment.java
com.google.android.apps.iosched.ui.WhatsOnFragment.java
com.google.android.apps.iosched.ui.phone.MapActivity.java
com.google.android.apps.iosched.ui.phone.ScheduleActivity.java
com.google.android.apps.iosched.ui.phone.SessionDetailActivity.java
com.google.android.apps.iosched.ui.phone.SessionsActivity.java
com.google.android.apps.iosched.ui.phone.TracksActivity.java
com.google.android.apps.iosched.ui.phone.VendorDetailActivity.java
com.google.android.apps.iosched.ui.phone.VendorsActivity.java
com.google.android.apps.iosched.ui.tablet.MapMultiPaneActivity.java
com.google.android.apps.iosched.ui.tablet.NowPlayingMultiPaneActivity.java
com.google.android.apps.iosched.ui.tablet.ScheduleMultiPaneActivity.java
com.google.android.apps.iosched.ui.tablet.SessionsMultiPaneActivity.java
com.google.android.apps.iosched.ui.tablet.TracksDropdownFragment.java
com.google.android.apps.iosched.ui.tablet.VendorsMultiPaneActivity.java
com.google.android.apps.iosched.ui.widget.BezelImageView.java
com.google.android.apps.iosched.ui.widget.BlockView.java
com.google.android.apps.iosched.ui.widget.BlocksLayout.java
com.google.android.apps.iosched.ui.widget.DashboardLayout.java
com.google.android.apps.iosched.ui.widget.ObservableScrollView.java
com.google.android.apps.iosched.ui.widget.TimeRulerView.java
com.google.android.apps.iosched.ui.widget.Workspace.java
com.google.android.apps.iosched.util.ActivityHelperHoneycomb.java
com.google.android.apps.iosched.util.ActivityHelper.java
com.google.android.apps.iosched.util.AnalyticsUtils.java
com.google.android.apps.iosched.util.BitmapUtils.java
com.google.android.apps.iosched.util.CatchNotesHelper.java
com.google.android.apps.iosched.util.DetachableResultReceiver.java
com.google.android.apps.iosched.util.EulaHelper.java
com.google.android.apps.iosched.util.FractionalTouchDelegate.java
com.google.android.apps.iosched.util.Lists.java
com.google.android.apps.iosched.util.Maps.java
com.google.android.apps.iosched.util.MotionEventUtils.java
com.google.android.apps.iosched.util.NotifyingAsyncQueryHandler.java
com.google.android.apps.iosched.util.ParserUtils.java
com.google.android.apps.iosched.util.ReflectionUtils.java
com.google.android.apps.iosched.util.SelectionBuilder.java
com.google.android.apps.iosched.util.Sets.java
com.google.android.apps.iosched.util.SimpleMenuItem.java
com.google.android.apps.iosched.util.SimpleMenu.java
com.google.android.apps.iosched.util.SpreadsheetEntryTest.java
com.google.android.apps.iosched.util.SpreadsheetEntry.java
com.google.android.apps.iosched.util.UIUtils.java
com.google.android.apps.iosched.util.WorksheetEntry.java