1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
5  * in compliance with the License. You may obtain a copy of the License at
6  *
7  * http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software distributed under the License
10  * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
11  * or implied. See the License for the specific language governing permissions and limitations under
12  * the License.
13  */
14 
15 package androidx.leanback.widget;
16 
17 import static androidx.leanback.widget.BaseGridView.WINDOW_ALIGN_BOTH_EDGE;
18 import static androidx.leanback.widget.BaseGridView.WINDOW_ALIGN_HIGH_EDGE;
19 import static androidx.leanback.widget.BaseGridView.WINDOW_ALIGN_LOW_EDGE;
20 import static androidx.leanback.widget.BaseGridView.WINDOW_ALIGN_OFFSET_PERCENT_DISABLED;
21 import static androidx.recyclerview.widget.RecyclerView.HORIZONTAL;
22 
23 /**
24  * Maintains Window Alignment information of two axis.
25  */
26 class WindowAlignment {
27 
28     /**
29      * Maintains alignment information in one direction.
30      */
31     public static class Axis {
32         /**
33          * Right or bottom edge of last child.
34          */
35         private int mMaxEdge;
36         /**
37          * Left or top edge of first child
38          */
39         private int mMinEdge;
40         /**
41          * Scroll distance to align last child, it defines limit of scroll.
42          */
43         private int mMaxScroll;
44         /**
45          * Scroll distance to align first child, it defines limit of scroll.
46          */
47         private int mMinScroll;
48 
49         static final int PF_KEYLINE_OVER_LOW_EDGE = 1;
50         static final int PF_KEYLINE_OVER_HIGH_EDGE = 1 << 1;
51 
52         /**
53          * By default we prefer low edge over keyline, prefer keyline over high edge.
54          */
55         private int mPreferredKeyLine = PF_KEYLINE_OVER_HIGH_EDGE;
56 
57         private int mWindowAlignment = WINDOW_ALIGN_BOTH_EDGE;
58 
59         private int mWindowAlignmentOffset = 0;
60 
61         private float mWindowAlignmentOffsetPercent = 50f;
62 
63         private int mSize;
64 
65         /**
66          * Padding at the min edge, it is the left or top padding.
67          */
68         private int mPaddingMin;
69 
70         /**
71          * Padding at the max edge, it is the right or bottom padding.
72          */
73         private int mPaddingMax;
74 
75         private boolean mReversedFlow;
76 
77         private String mName; // for debugging
78 
Axis(String name)79         public Axis(String name) {
80             reset();
81             mName = name;
82         }
83 
getWindowAlignment()84         public final int getWindowAlignment() {
85             return mWindowAlignment;
86         }
87 
setWindowAlignment(int windowAlignment)88         public final void setWindowAlignment(int windowAlignment) {
89             mWindowAlignment = windowAlignment;
90         }
91 
setPreferKeylineOverLowEdge(boolean keylineOverLowEdge)92         final void setPreferKeylineOverLowEdge(boolean keylineOverLowEdge) {
93             mPreferredKeyLine = keylineOverLowEdge
94                     ? mPreferredKeyLine | PF_KEYLINE_OVER_LOW_EDGE
95                     : mPreferredKeyLine & ~PF_KEYLINE_OVER_LOW_EDGE;
96         }
97 
setPreferKeylineOverHighEdge(boolean keylineOverHighEdge)98         final void setPreferKeylineOverHighEdge(boolean keylineOverHighEdge) {
99             mPreferredKeyLine = keylineOverHighEdge
100                     ? mPreferredKeyLine | PF_KEYLINE_OVER_HIGH_EDGE
101                     : mPreferredKeyLine & ~PF_KEYLINE_OVER_HIGH_EDGE;
102         }
103 
isPreferKeylineOverHighEdge()104         final boolean isPreferKeylineOverHighEdge() {
105             return (mPreferredKeyLine & PF_KEYLINE_OVER_HIGH_EDGE) != 0;
106         }
107 
isPreferKeylineOverLowEdge()108         final boolean isPreferKeylineOverLowEdge() {
109             return (mPreferredKeyLine & PF_KEYLINE_OVER_LOW_EDGE) != 0;
110         }
111 
getWindowAlignmentOffset()112         public final int getWindowAlignmentOffset() {
113             return mWindowAlignmentOffset;
114         }
115 
setWindowAlignmentOffset(int offset)116         public final void setWindowAlignmentOffset(int offset) {
117             mWindowAlignmentOffset = offset;
118         }
119 
setWindowAlignmentOffsetPercent(float percent)120         public final void setWindowAlignmentOffsetPercent(float percent) {
121             if ((percent < 0 || percent > 100)
122                     && percent != WINDOW_ALIGN_OFFSET_PERCENT_DISABLED) {
123                 throw new IllegalArgumentException();
124             }
125             mWindowAlignmentOffsetPercent = percent;
126         }
127 
getWindowAlignmentOffsetPercent()128         public final float getWindowAlignmentOffsetPercent() {
129             return mWindowAlignmentOffsetPercent;
130         }
131 
132         /**
133          * Returns scroll distance to align min child.
134          */
getMinScroll()135         public final int getMinScroll() {
136             return mMinScroll;
137         }
138 
invalidateScrollMin()139         public final void invalidateScrollMin() {
140             mMinEdge = Integer.MIN_VALUE;
141             mMinScroll = Integer.MIN_VALUE;
142         }
143 
144         /**
145          * Returns scroll distance to align max child.
146          */
getMaxScroll()147         public final int getMaxScroll() {
148             return mMaxScroll;
149         }
150 
invalidateScrollMax()151         public final void invalidateScrollMax() {
152             mMaxEdge = Integer.MAX_VALUE;
153             mMaxScroll = Integer.MAX_VALUE;
154         }
155 
reset()156         void reset() {
157             mMinEdge = Integer.MIN_VALUE;
158             mMaxEdge = Integer.MAX_VALUE;
159         }
160 
isMinUnknown()161         public final boolean isMinUnknown() {
162             return mMinEdge == Integer.MIN_VALUE;
163         }
164 
isMaxUnknown()165         public final boolean isMaxUnknown() {
166             return mMaxEdge == Integer.MAX_VALUE;
167         }
168 
setSize(int size)169         public final void setSize(int size) {
170             mSize = size;
171         }
172 
getSize()173         public final int getSize() {
174             return mSize;
175         }
176 
setPadding(int paddingMin, int paddingMax)177         public final void setPadding(int paddingMin, int paddingMax) {
178             mPaddingMin = paddingMin;
179             mPaddingMax = paddingMax;
180         }
181 
getPaddingMin()182         public final int getPaddingMin() {
183             return mPaddingMin;
184         }
185 
getPaddingMax()186         public final int getPaddingMax() {
187             return mPaddingMax;
188         }
189 
getClientSize()190         public final int getClientSize() {
191             return mSize - mPaddingMin - mPaddingMax;
192         }
193 
calculateKeyline()194         final int calculateKeyline() {
195             int keyLine;
196             if (!mReversedFlow) {
197                 if (mWindowAlignmentOffset >= 0) {
198                     keyLine = mWindowAlignmentOffset;
199                 } else {
200                     keyLine = mSize + mWindowAlignmentOffset;
201                 }
202                 if (mWindowAlignmentOffsetPercent != WINDOW_ALIGN_OFFSET_PERCENT_DISABLED) {
203                     keyLine += (int) (mSize * mWindowAlignmentOffsetPercent / 100);
204                 }
205             } else {
206                 if (mWindowAlignmentOffset >= 0) {
207                     keyLine = mSize - mWindowAlignmentOffset;
208                 } else {
209                     keyLine = -mWindowAlignmentOffset;
210                 }
211                 if (mWindowAlignmentOffsetPercent != WINDOW_ALIGN_OFFSET_PERCENT_DISABLED) {
212                     keyLine -= (int) (mSize * mWindowAlignmentOffsetPercent / 100);
213                 }
214             }
215             return keyLine;
216         }
217 
218         /**
219          * Returns scroll distance to move viewCenterPosition to keyLine.
220          */
calculateScrollToKeyLine(int viewCenterPosition, int keyLine)221         final int calculateScrollToKeyLine(int viewCenterPosition, int keyLine) {
222             return viewCenterPosition - keyLine;
223         }
224 
225         /**
226          * Update {@link #getMinScroll()} and {@link #getMaxScroll()}
227          */
updateMinMax(int minEdge, int maxEdge, int minChildViewCenter, int maxChildViewCenter)228         public final void updateMinMax(int minEdge, int maxEdge,
229                 int minChildViewCenter, int maxChildViewCenter) {
230             mMinEdge = minEdge;
231             mMaxEdge = maxEdge;
232             final int clientSize = getClientSize();
233             final int keyLine = calculateKeyline();
234             final boolean isMinUnknown = isMinUnknown();
235             final boolean isMaxUnknown = isMaxUnknown();
236             if (!isMinUnknown) {
237                 if (!mReversedFlow ? (mWindowAlignment & WINDOW_ALIGN_LOW_EDGE) != 0
238                         : (mWindowAlignment & WINDOW_ALIGN_HIGH_EDGE) != 0) {
239                     // calculate scroll distance to move current mMinEdge to padding at min edge
240                     mMinScroll = mMinEdge - mPaddingMin;
241                 } else  {
242                     // calculate scroll distance to move min child center to key line
243                     mMinScroll = calculateScrollToKeyLine(minChildViewCenter, keyLine);
244                 }
245             }
246             if (!isMaxUnknown) {
247                 if (!mReversedFlow ? (mWindowAlignment & WINDOW_ALIGN_HIGH_EDGE) != 0
248                         : (mWindowAlignment & WINDOW_ALIGN_LOW_EDGE) != 0) {
249                     // calculate scroll distance to move current mMaxEdge to padding at max edge
250                     mMaxScroll = mMaxEdge - mPaddingMin - clientSize;
251                 } else  {
252                     // calculate scroll distance to move max child center to key line
253                     mMaxScroll = calculateScrollToKeyLine(maxChildViewCenter, keyLine);
254                 }
255             }
256             if (!isMaxUnknown && !isMinUnknown) {
257                 if (!mReversedFlow) {
258                     if ((mWindowAlignment & WINDOW_ALIGN_LOW_EDGE) != 0) {
259                         if (isPreferKeylineOverLowEdge()) {
260                             // if we prefer key line, might align max child to key line for
261                             // minScroll
262                             mMinScroll = Math.min(mMinScroll,
263                                     calculateScrollToKeyLine(maxChildViewCenter, keyLine));
264                         }
265                         // don't over scroll max
266                         mMaxScroll = Math.max(mMinScroll, mMaxScroll);
267                     } else if ((mWindowAlignment & WINDOW_ALIGN_HIGH_EDGE) != 0) {
268                         if (isPreferKeylineOverHighEdge()) {
269                             // if we prefer key line, might align min child to key line for
270                             // maxScroll
271                             mMaxScroll = Math.max(mMaxScroll,
272                                     calculateScrollToKeyLine(minChildViewCenter, keyLine));
273                         }
274                         // don't over scroll min
275                         mMinScroll = Math.min(mMinScroll, mMaxScroll);
276                     }
277                 } else {
278                     if ((mWindowAlignment & WINDOW_ALIGN_LOW_EDGE) != 0) {
279                         if (isPreferKeylineOverLowEdge()) {
280                             // if we prefer key line, might align min child to key line for
281                             // maxScroll
282                             mMaxScroll = Math.max(mMaxScroll,
283                                     calculateScrollToKeyLine(minChildViewCenter, keyLine));
284                         }
285                         // don't over scroll min
286                         mMinScroll = Math.min(mMinScroll, mMaxScroll);
287                     } else if ((mWindowAlignment & WINDOW_ALIGN_HIGH_EDGE) != 0) {
288                         if (isPreferKeylineOverHighEdge()) {
289                             // if we prefer key line, might align max child to key line for
290                             // minScroll
291                             mMinScroll = Math.min(mMinScroll,
292                                     calculateScrollToKeyLine(maxChildViewCenter, keyLine));
293                         }
294                         // don't over scroll max
295                         mMaxScroll = Math.max(mMinScroll, mMaxScroll);
296                     }
297                 }
298             }
299         }
300 
301         /**
302          * Get scroll distance of align an item (depends on ALIGN_LOW_EDGE, ALIGN_HIGH_EDGE or the
303          * item should be aligned to key line). The scroll distance will be capped by
304          * {@link #getMinScroll()} and {@link #getMaxScroll()}.
305          */
getScroll(int viewCenter)306         public final int getScroll(int viewCenter) {
307             final int size = getSize();
308             final int keyLine = calculateKeyline();
309             final boolean isMinUnknown = isMinUnknown();
310             final boolean isMaxUnknown = isMaxUnknown();
311             if (!isMinUnknown) {
312                 final int keyLineToMinEdge = keyLine - mPaddingMin;
313                 if ((!mReversedFlow ? (mWindowAlignment & WINDOW_ALIGN_LOW_EDGE) != 0
314                      : (mWindowAlignment & WINDOW_ALIGN_HIGH_EDGE) != 0)
315                         && (viewCenter - mMinEdge <= keyLineToMinEdge)) {
316                     // view center is before key line: align the min edge (first child) to padding.
317                     int alignToMin = mMinEdge - mPaddingMin;
318                     // Also we need make sure don't over scroll
319                     if (!isMaxUnknown && alignToMin > mMaxScroll) {
320                         alignToMin = mMaxScroll;
321                     }
322                     return alignToMin;
323                 }
324             }
325             if (!isMaxUnknown) {
326                 final int keyLineToMaxEdge = size - keyLine - mPaddingMax;
327                 if ((!mReversedFlow ? (mWindowAlignment & WINDOW_ALIGN_HIGH_EDGE) != 0
328                         : (mWindowAlignment & WINDOW_ALIGN_LOW_EDGE) != 0)
329                         && (mMaxEdge - viewCenter <= keyLineToMaxEdge)) {
330                     // view center is after key line: align the max edge (last child) to padding.
331                     int alignToMax = mMaxEdge - (size - mPaddingMax);
332                     // Also we need make sure don't over scroll
333                     if (!isMinUnknown && alignToMax < mMinScroll) {
334                         alignToMax = mMinScroll;
335                     }
336                     return alignToMax;
337                 }
338             }
339             // else put view center at key line.
340             return calculateScrollToKeyLine(viewCenter, keyLine);
341         }
342 
setReversedFlow(boolean reversedFlow)343         public final void setReversedFlow(boolean reversedFlow) {
344             mReversedFlow = reversedFlow;
345         }
346 
347         @Override
toString()348         public String toString() {
349             return " min:" + mMinEdge + " " + mMinScroll + " max:" + mMaxEdge + " " + mMaxScroll;
350         }
351 
352     }
353 
354     private int mOrientation = HORIZONTAL;
355 
356     public final Axis vertical = new Axis("vertical");
357 
358     public final Axis horizontal = new Axis("horizontal");
359 
360     private Axis mMainAxis = horizontal;
361 
362     private Axis mSecondAxis = vertical;
363 
mainAxis()364     public final Axis mainAxis() {
365         return mMainAxis;
366     }
367 
secondAxis()368     public final Axis secondAxis() {
369         return mSecondAxis;
370     }
371 
setOrientation(int orientation)372     public final void setOrientation(int orientation) {
373         mOrientation = orientation;
374         if (mOrientation == HORIZONTAL) {
375             mMainAxis = horizontal;
376             mSecondAxis = vertical;
377         } else {
378             mMainAxis = vertical;
379             mSecondAxis = horizontal;
380         }
381     }
382 
getOrientation()383     public final int getOrientation() {
384         return mOrientation;
385     }
386 
reset()387     public final void reset() {
388         mainAxis().reset();
389     }
390 
391     @Override
toString()392     public String toString() {
393         return "horizontal=" + horizontal + "; vertical=" + vertical;
394     }
395 
396 }
397