FileDocCategorySizeDatePackage
WindowAlignment.javaAPI DocAndroid 5.1 API10495Thu Mar 12 22:22:56 GMT 2015android.support.v17.leanback.widget

WindowAlignment.java

/*
 * Copyright (C) 2014 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 android.support.v17.leanback.widget;

import static android.support.v17.leanback.widget.BaseGridView.WINDOW_ALIGN_LOW_EDGE;
import static android.support.v17.leanback.widget.BaseGridView.WINDOW_ALIGN_HIGH_EDGE;
import static android.support.v17.leanback.widget.BaseGridView.WINDOW_ALIGN_BOTH_EDGE;
import static android.support.v17.leanback.widget.BaseGridView.WINDOW_ALIGN_OFFSET_PERCENT_DISABLED;

import static android.support.v7.widget.RecyclerView.HORIZONTAL;

import android.view.View;

/**
 * Maintains Window Alignment information of two axis.
 */
class WindowAlignment {

    /**
     * Maintains alignment information in one direction.
     */
    public static class Axis {
        /**
         * mScrollCenter is used to calculate dynamic transformation based on how far a view
         * is from the mScrollCenter. For example, the views with center close to mScrollCenter
         * will be scaled up.
         */
        private float mScrollCenter;
        /** 
         * Right or bottom edge of last child. 
         */
        private int mMaxEdge;
        /** 
         * Left or top edge of first child, typically should be zero.
         */
        private int mMinEdge;
        /**
         * Max Scroll value
         */
        private int mMaxScroll;
        /**
         * Min Scroll value
         */
        private int mMinScroll;

        private int mWindowAlignment = WINDOW_ALIGN_BOTH_EDGE;

        private int mWindowAlignmentOffset = 0;

        private float mWindowAlignmentOffsetPercent = 50f;

        private int mSize;

        private int mPaddingLow;

        private int mPaddingHigh;

        private boolean mReversedFlow;

        private String mName; // for debugging

        public Axis(String name) {
            reset();
            mName = name;
        }

        final public int getWindowAlignment() {
            return mWindowAlignment;
        }

        final public void setWindowAlignment(int windowAlignment) {
            mWindowAlignment = windowAlignment;
        }

        final public int getWindowAlignmentOffset() {
            return mWindowAlignmentOffset;
        }

        final public void setWindowAlignmentOffset(int offset) {
            mWindowAlignmentOffset = offset;
        }

        final public void setWindowAlignmentOffsetPercent(float percent) {
            if ((percent < 0 || percent > 100)
                    && percent != WINDOW_ALIGN_OFFSET_PERCENT_DISABLED) {
                throw new IllegalArgumentException();
            }
            mWindowAlignmentOffsetPercent = percent;
        }

        final public float getWindowAlignmentOffsetPercent() {
            return mWindowAlignmentOffsetPercent;
        }

        final public int getScrollCenter() {
            return (int) mScrollCenter;
        }

        /** set minEdge,  Integer.MIN_VALUE means unknown*/
        final public void setMinEdge(int minEdge) {
            mMinEdge = minEdge;
        }

        final public int getMinEdge() {
            return mMinEdge;
        }

        /** set minScroll,  Integer.MIN_VALUE means unknown*/
        final public void setMinScroll(int minScroll) {
            mMinScroll = minScroll;
        }

        final public int getMinScroll() {
            return mMinScroll;
        }

        final public void invalidateScrollMin() {
            mMinEdge = Integer.MIN_VALUE;
            mMinScroll = Integer.MIN_VALUE;
        }

        /** update max edge,  Integer.MAX_VALUE means unknown*/
        final public void setMaxEdge(int maxEdge) {
            mMaxEdge = maxEdge;
        }

        final public int getMaxEdge() {
            return mMaxEdge;
        }

        /** update max scroll,  Integer.MAX_VALUE means unknown*/
        final public void setMaxScroll(int maxScroll) {
            mMaxScroll = maxScroll;
        }

        final public int getMaxScroll() {
            return mMaxScroll;
        }

        final public void invalidateScrollMax() {
            mMaxEdge = Integer.MAX_VALUE;
            mMaxScroll = Integer.MAX_VALUE;
        }

        final public float updateScrollCenter(float scrollTarget) {
            mScrollCenter = scrollTarget;
            return scrollTarget;
        }

        private void reset() {
            mScrollCenter = Integer.MIN_VALUE;
            mMinEdge = Integer.MIN_VALUE;
            mMaxEdge = Integer.MAX_VALUE;
        }

        final public boolean isMinUnknown() {
            return mMinEdge == Integer.MIN_VALUE;
        }

        final public boolean isMaxUnknown() {
            return mMaxEdge == Integer.MAX_VALUE;
        }

        final public void setSize(int size) {
            mSize = size;
        }

        final public int getSize() {
            return mSize;
        }

        final public void setPadding(int paddingLow, int paddingHigh) {
            mPaddingLow = paddingLow;
            mPaddingHigh = paddingHigh;
        }

        final public int getPaddingLow() {
            return mPaddingLow;
        }

        final public int getPaddingHigh() {
            return mPaddingHigh;
        }

        final public int getClientSize() {
            return mSize - mPaddingLow - mPaddingHigh;
        }

        final public int getSystemScrollPos(boolean isAtMin, boolean isAtMax) {
            return getSystemScrollPos((int) mScrollCenter, isAtMin, isAtMax);
        }

        final public int getSystemScrollPos(int scrollCenter, boolean isAtMin, boolean isAtMax) {
            int middlePosition;
            if (!mReversedFlow) {
                if (mWindowAlignmentOffset >= 0) {
                    middlePosition = mWindowAlignmentOffset - mPaddingLow;
                } else {
                    middlePosition = mSize + mWindowAlignmentOffset - mPaddingLow;
                }
                if (mWindowAlignmentOffsetPercent != WINDOW_ALIGN_OFFSET_PERCENT_DISABLED) {
                    middlePosition += (int) (mSize * mWindowAlignmentOffsetPercent / 100);
                }
            } else {
                if (mWindowAlignmentOffset >= 0) {
                    middlePosition = mSize - mWindowAlignmentOffset - mPaddingLow;
                } else {
                    middlePosition = - mWindowAlignmentOffset - mPaddingLow;
                }
                if (mWindowAlignmentOffsetPercent != WINDOW_ALIGN_OFFSET_PERCENT_DISABLED) {
                    middlePosition -= (int) (mSize * mWindowAlignmentOffsetPercent / 100);
                }
            }
            int clientSize = getClientSize();
            int afterMiddlePosition = clientSize - middlePosition;
            boolean isMinUnknown = isMinUnknown();
            boolean isMaxUnknown = isMaxUnknown();
            if (!isMinUnknown && !isMaxUnknown &&
                    (mWindowAlignment & WINDOW_ALIGN_BOTH_EDGE) == WINDOW_ALIGN_BOTH_EDGE) {
                if (mMaxEdge - mMinEdge <= clientSize) {
                    // total children size is less than view port and we want to align
                    // both edge:  align first child to start edge of view port
                    return mReversedFlow ? mMaxEdge - mPaddingLow - clientSize
                            : mMinEdge - mPaddingLow;
                }
            }
            if (!isMinUnknown) {
                if ((!mReversedFlow ? (mWindowAlignment & WINDOW_ALIGN_LOW_EDGE) != 0
                     : (mWindowAlignment & WINDOW_ALIGN_HIGH_EDGE) != 0)
                        && (isAtMin || scrollCenter - mMinEdge <= middlePosition)) {
                    // scroll center is within half of view port size: align the start edge
                    // of first child to the start edge of view port
                    return mMinEdge - mPaddingLow;
                }
            }
            if (!isMaxUnknown) {
                if ((!mReversedFlow ? (mWindowAlignment & WINDOW_ALIGN_HIGH_EDGE) != 0
                        : (mWindowAlignment & WINDOW_ALIGN_LOW_EDGE) != 0)
                        && (isAtMax || mMaxEdge - scrollCenter <= afterMiddlePosition)) {
                    // scroll center is very close to the end edge of view port : align the
                    // end edge of last children (plus expanded size) to view port's end
                    return mMaxEdge - mPaddingLow - clientSize;
                }
            }
            // else put scroll center in middle of view port
            return scrollCenter - middlePosition - mPaddingLow;
        }

        final public void setReversedFlow(boolean reversedFlow) {
            mReversedFlow = reversedFlow;
        }

        @Override
        public String toString() {
            return "center: " + mScrollCenter + " min:" + mMinEdge +
                    " max:" + mMaxEdge;
        }

    }

    private int mOrientation = HORIZONTAL;

    final public Axis vertical = new Axis("vertical");

    final public Axis horizontal = new Axis("horizontal");

    private Axis mMainAxis = horizontal;

    private Axis mSecondAxis = vertical;

    final public Axis mainAxis() {
        return mMainAxis;
    }

    final public Axis secondAxis() {
        return mSecondAxis;
    }

    final public void setOrientation(int orientation) {
        mOrientation = orientation;
        if (mOrientation == HORIZONTAL) {
            mMainAxis = horizontal;
            mSecondAxis = vertical;
        } else {
            mMainAxis = vertical;
            mSecondAxis = horizontal;
        }
    }

    final public int getOrientation() {
        return mOrientation;
    }

    final public void reset() {
        mainAxis().reset();
    }

    @Override
    public String toString() {
        return new StringBuffer().append("horizontal=")
                .append(horizontal.toString())
                .append("; vertical=")
                .append(vertical.toString())
                .toString();
    }

}