• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2007 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.widget;
18 
19 import com.android.internal.R;
20 
21 import android.annotation.Widget;
22 import android.content.Context;
23 import android.content.res.TypedArray;
24 import android.graphics.Rect;
25 import android.util.AttributeSet;
26 import android.util.Log;
27 import android.view.GestureDetector;
28 import android.view.Gravity;
29 import android.view.HapticFeedbackConstants;
30 import android.view.KeyEvent;
31 import android.view.MotionEvent;
32 import android.view.View;
33 import android.view.ViewConfiguration;
34 import android.view.ViewGroup;
35 import android.view.SoundEffectConstants;
36 import android.view.ContextMenu.ContextMenuInfo;
37 import android.view.animation.Transformation;
38 
39 /**
40  * A view that shows items in a center-locked, horizontally scrolling list.
41  * <p>
42  * The default values for the Gallery assume you will be using
43  * {@link android.R.styleable#Theme_galleryItemBackground} as the background for
44  * each View given to the Gallery from the Adapter. If you are not doing this,
45  * you may need to adjust some Gallery properties, such as the spacing.
46  * <p>
47  * Views given to the Gallery should use {@link Gallery.LayoutParams} as their
48  * layout parameters type.
49  *
50  * @attr ref android.R.styleable#Gallery_animationDuration
51  * @attr ref android.R.styleable#Gallery_spacing
52  * @attr ref android.R.styleable#Gallery_gravity
53  */
54 @Widget
55 public class Gallery extends AbsSpinner implements GestureDetector.OnGestureListener {
56 
57     private static final String TAG = "Gallery";
58 
59     private static final boolean localLOGV = false;
60 
61     /**
62      * Duration in milliseconds from the start of a scroll during which we're
63      * unsure whether the user is scrolling or flinging.
64      */
65     private static final int SCROLL_TO_FLING_UNCERTAINTY_TIMEOUT = 250;
66 
67     /**
68      * Horizontal spacing between items.
69      */
70     private int mSpacing = 0;
71 
72     /**
73      * How long the transition animation should run when a child view changes
74      * position, measured in milliseconds.
75      */
76     private int mAnimationDuration = 400;
77 
78     /**
79      * The alpha of items that are not selected.
80      */
81     private float mUnselectedAlpha;
82 
83     /**
84      * Left most edge of a child seen so far during layout.
85      */
86     private int mLeftMost;
87 
88     /**
89      * Right most edge of a child seen so far during layout.
90      */
91     private int mRightMost;
92 
93     private int mGravity;
94 
95     /**
96      * Helper for detecting touch gestures.
97      */
98     private GestureDetector mGestureDetector;
99 
100     /**
101      * The position of the item that received the user's down touch.
102      */
103     private int mDownTouchPosition;
104 
105     /**
106      * The view of the item that received the user's down touch.
107      */
108     private View mDownTouchView;
109 
110     /**
111      * Executes the delta scrolls from a fling or scroll movement.
112      */
113     private FlingRunnable mFlingRunnable = new FlingRunnable();
114 
115     /**
116      * Sets mSuppressSelectionChanged = false. This is used to set it to false
117      * in the future. It will also trigger a selection changed.
118      */
119     private Runnable mDisableSuppressSelectionChangedRunnable = new Runnable() {
120         public void run() {
121             mSuppressSelectionChanged = false;
122             selectionChanged();
123         }
124     };
125 
126     /**
127      * When fling runnable runs, it resets this to false. Any method along the
128      * path until the end of its run() can set this to true to abort any
129      * remaining fling. For example, if we've reached either the leftmost or
130      * rightmost item, we will set this to true.
131      */
132     private boolean mShouldStopFling;
133 
134     /**
135      * The currently selected item's child.
136      */
137     private View mSelectedChild;
138 
139     /**
140      * Whether to continuously callback on the item selected listener during a
141      * fling.
142      */
143     private boolean mShouldCallbackDuringFling = true;
144 
145     /**
146      * Whether to callback when an item that is not selected is clicked.
147      */
148     private boolean mShouldCallbackOnUnselectedItemClick = true;
149 
150     /**
151      * If true, do not callback to item selected listener.
152      */
153     private boolean mSuppressSelectionChanged;
154 
155     /**
156      * If true, we have received the "invoke" (center or enter buttons) key
157      * down. This is checked before we action on the "invoke" key up, and is
158      * subsequently cleared.
159      */
160     private boolean mReceivedInvokeKeyDown;
161 
162     private AdapterContextMenuInfo mContextMenuInfo;
163 
164     /**
165      * If true, this onScroll is the first for this user's drag (remember, a
166      * drag sends many onScrolls).
167      */
168     private boolean mIsFirstScroll;
169 
Gallery(Context context)170     public Gallery(Context context) {
171         this(context, null);
172     }
173 
Gallery(Context context, AttributeSet attrs)174     public Gallery(Context context, AttributeSet attrs) {
175         this(context, attrs, R.attr.galleryStyle);
176     }
177 
Gallery(Context context, AttributeSet attrs, int defStyle)178     public Gallery(Context context, AttributeSet attrs, int defStyle) {
179         super(context, attrs, defStyle);
180 
181         mGestureDetector = new GestureDetector(context, this);
182         mGestureDetector.setIsLongpressEnabled(true);
183 
184         TypedArray a = context.obtainStyledAttributes(
185                 attrs, com.android.internal.R.styleable.Gallery, defStyle, 0);
186 
187         int index = a.getInt(com.android.internal.R.styleable.Gallery_gravity, -1);
188         if (index >= 0) {
189             setGravity(index);
190         }
191 
192         int animationDuration =
193                 a.getInt(com.android.internal.R.styleable.Gallery_animationDuration, -1);
194         if (animationDuration > 0) {
195             setAnimationDuration(animationDuration);
196         }
197 
198         int spacing =
199                 a.getDimensionPixelOffset(com.android.internal.R.styleable.Gallery_spacing, 0);
200         setSpacing(spacing);
201 
202         float unselectedAlpha = a.getFloat(
203                 com.android.internal.R.styleable.Gallery_unselectedAlpha, 0.5f);
204         setUnselectedAlpha(unselectedAlpha);
205 
206         a.recycle();
207 
208         // We draw the selected item last (because otherwise the item to the
209         // right overlaps it)
210         mGroupFlags |= FLAG_USE_CHILD_DRAWING_ORDER;
211 
212         mGroupFlags |= FLAG_SUPPORT_STATIC_TRANSFORMATIONS;
213     }
214 
215     /**
216      * Whether or not to callback on any {@link #getOnItemSelectedListener()}
217      * while the items are being flinged. If false, only the final selected item
218      * will cause the callback. If true, all items between the first and the
219      * final will cause callbacks.
220      *
221      * @param shouldCallback Whether or not to callback on the listener while
222      *            the items are being flinged.
223      */
setCallbackDuringFling(boolean shouldCallback)224     public void setCallbackDuringFling(boolean shouldCallback) {
225         mShouldCallbackDuringFling = shouldCallback;
226     }
227 
228     /**
229      * Whether or not to callback when an item that is not selected is clicked.
230      * If false, the item will become selected (and re-centered). If true, the
231      * {@link #getOnItemClickListener()} will get the callback.
232      *
233      * @param shouldCallback Whether or not to callback on the listener when a
234      *            item that is not selected is clicked.
235      * @hide
236      */
setCallbackOnUnselectedItemClick(boolean shouldCallback)237     public void setCallbackOnUnselectedItemClick(boolean shouldCallback) {
238         mShouldCallbackOnUnselectedItemClick = shouldCallback;
239     }
240 
241     /**
242      * Sets how long the transition animation should run when a child view
243      * changes position. Only relevant if animation is turned on.
244      *
245      * @param animationDurationMillis The duration of the transition, in
246      *        milliseconds.
247      *
248      * @attr ref android.R.styleable#Gallery_animationDuration
249      */
setAnimationDuration(int animationDurationMillis)250     public void setAnimationDuration(int animationDurationMillis) {
251         mAnimationDuration = animationDurationMillis;
252     }
253 
254     /**
255      * Sets the spacing between items in a Gallery
256      *
257      * @param spacing The spacing in pixels between items in the Gallery
258      *
259      * @attr ref android.R.styleable#Gallery_spacing
260      */
setSpacing(int spacing)261     public void setSpacing(int spacing) {
262         mSpacing = spacing;
263     }
264 
265     /**
266      * Sets the alpha of items that are not selected in the Gallery.
267      *
268      * @param unselectedAlpha the alpha for the items that are not selected.
269      *
270      * @attr ref android.R.styleable#Gallery_unselectedAlpha
271      */
setUnselectedAlpha(float unselectedAlpha)272     public void setUnselectedAlpha(float unselectedAlpha) {
273         mUnselectedAlpha = unselectedAlpha;
274     }
275 
276     @Override
getChildStaticTransformation(View child, Transformation t)277     protected boolean getChildStaticTransformation(View child, Transformation t) {
278 
279         t.clear();
280         t.setAlpha(child == mSelectedChild ? 1.0f : mUnselectedAlpha);
281 
282         return true;
283     }
284 
285     @Override
computeHorizontalScrollExtent()286     protected int computeHorizontalScrollExtent() {
287         // Only 1 item is considered to be selected
288         return 1;
289     }
290 
291     @Override
computeHorizontalScrollOffset()292     protected int computeHorizontalScrollOffset() {
293         // Current scroll position is the same as the selected position
294         return mSelectedPosition;
295     }
296 
297     @Override
computeHorizontalScrollRange()298     protected int computeHorizontalScrollRange() {
299         // Scroll range is the same as the item count
300         return mItemCount;
301     }
302 
303     @Override
checkLayoutParams(ViewGroup.LayoutParams p)304     protected boolean checkLayoutParams(ViewGroup.LayoutParams p) {
305         return p instanceof LayoutParams;
306     }
307 
308     @Override
generateLayoutParams(ViewGroup.LayoutParams p)309     protected ViewGroup.LayoutParams generateLayoutParams(ViewGroup.LayoutParams p) {
310         return new LayoutParams(p);
311     }
312 
313     @Override
generateLayoutParams(AttributeSet attrs)314     public ViewGroup.LayoutParams generateLayoutParams(AttributeSet attrs) {
315         return new LayoutParams(getContext(), attrs);
316     }
317 
318     @Override
generateDefaultLayoutParams()319     protected ViewGroup.LayoutParams generateDefaultLayoutParams() {
320         /*
321          * Gallery expects Gallery.LayoutParams.
322          */
323         return new Gallery.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT,
324                 ViewGroup.LayoutParams.WRAP_CONTENT);
325     }
326 
327     @Override
onLayout(boolean changed, int l, int t, int r, int b)328     protected void onLayout(boolean changed, int l, int t, int r, int b) {
329         super.onLayout(changed, l, t, r, b);
330 
331         /*
332          * Remember that we are in layout to prevent more layout request from
333          * being generated.
334          */
335         mInLayout = true;
336         layout(0, false);
337         mInLayout = false;
338     }
339 
340     @Override
getChildHeight(View child)341     int getChildHeight(View child) {
342         return child.getMeasuredHeight();
343     }
344 
345     /**
346      * Tracks a motion scroll. In reality, this is used to do just about any
347      * movement to items (touch scroll, arrow-key scroll, set an item as selected).
348      *
349      * @param deltaX Change in X from the previous event.
350      */
trackMotionScroll(int deltaX)351     void trackMotionScroll(int deltaX) {
352 
353         if (getChildCount() == 0) {
354             return;
355         }
356 
357         boolean toLeft = deltaX < 0;
358 
359         int limitedDeltaX = getLimitedMotionScrollAmount(toLeft, deltaX);
360         if (limitedDeltaX != deltaX) {
361             // The above call returned a limited amount, so stop any scrolls/flings
362             mFlingRunnable.endFling(false);
363             onFinishedMovement();
364         }
365 
366         offsetChildrenLeftAndRight(limitedDeltaX);
367 
368         detachOffScreenChildren(toLeft);
369 
370         if (toLeft) {
371             // If moved left, there will be empty space on the right
372             fillToGalleryRight();
373         } else {
374             // Similarly, empty space on the left
375             fillToGalleryLeft();
376         }
377 
378         // Clear unused views
379         mRecycler.clear();
380 
381         setSelectionToCenterChild();
382 
383         invalidate();
384     }
385 
386     int getLimitedMotionScrollAmount(boolean motionToLeft, int deltaX) {
387         int extremeItemPosition = motionToLeft ? mItemCount - 1 : 0;
388         View extremeChild = getChildAt(extremeItemPosition - mFirstPosition);
389 
390         if (extremeChild == null) {
391             return deltaX;
392         }
393 
394         int extremeChildCenter = getCenterOfView(extremeChild);
395         int galleryCenter = getCenterOfGallery();
396 
397         if (motionToLeft) {
398             if (extremeChildCenter <= galleryCenter) {
399 
400                 // The extreme child is past his boundary point!
401                 return 0;
402             }
403         } else {
404             if (extremeChildCenter >= galleryCenter) {
405 
406                 // The extreme child is past his boundary point!
407                 return 0;
408             }
409         }
410 
411         int centerDifference = galleryCenter - extremeChildCenter;
412 
413         return motionToLeft
414                 ? Math.max(centerDifference, deltaX)
415                 : Math.min(centerDifference, deltaX);
416     }
417 
418     /**
419      * Offset the horizontal location of all children of this view by the
420      * specified number of pixels.
421      *
422      * @param offset the number of pixels to offset
423      */
424     private void offsetChildrenLeftAndRight(int offset) {
425         for (int i = getChildCount() - 1; i >= 0; i--) {
426             getChildAt(i).offsetLeftAndRight(offset);
427         }
428     }
429 
430     /**
431      * @return The center of this Gallery.
432      */
getCenterOfGallery()433     private int getCenterOfGallery() {
434         return (getWidth() - mPaddingLeft - mPaddingRight) / 2 + mPaddingLeft;
435     }
436 
437     /**
438      * @return The center of the given view.
439      */
getCenterOfView(View view)440     private static int getCenterOfView(View view) {
441         return view.getLeft() + view.getWidth() / 2;
442     }
443 
444     /**
445      * Detaches children that are off the screen (i.e.: Gallery bounds).
446      *
447      * @param toLeft Whether to detach children to the left of the Gallery, or
448      *            to the right.
449      */
detachOffScreenChildren(boolean toLeft)450     private void detachOffScreenChildren(boolean toLeft) {
451         int numChildren = getChildCount();
452         int firstPosition = mFirstPosition;
453         int start = 0;
454         int count = 0;
455 
456         if (toLeft) {
457             final int galleryLeft = mPaddingLeft;
458             for (int i = 0; i < numChildren; i++) {
459                 final View child = getChildAt(i);
460                 if (child.getRight() >= galleryLeft) {
461                     break;
462                 } else {
463                     count++;
464                     mRecycler.put(firstPosition + i, child);
465                 }
466             }
467         } else {
468             final int galleryRight = getWidth() - mPaddingRight;
469             for (int i = numChildren - 1; i >= 0; i--) {
470                 final View child = getChildAt(i);
471                 if (child.getLeft() <= galleryRight) {
472                     break;
473                 } else {
474                     start = i;
475                     count++;
476                     mRecycler.put(firstPosition + i, child);
477                 }
478             }
479         }
480 
481         detachViewsFromParent(start, count);
482 
483         if (toLeft) {
484             mFirstPosition += count;
485         }
486     }
487 
488     /**
489      * Scrolls the items so that the selected item is in its 'slot' (its center
490      * is the gallery's center).
491      */
scrollIntoSlots()492     private void scrollIntoSlots() {
493 
494         if (getChildCount() == 0 || mSelectedChild == null) return;
495 
496         int selectedCenter = getCenterOfView(mSelectedChild);
497         int targetCenter = getCenterOfGallery();
498 
499         int scrollAmount = targetCenter - selectedCenter;
500         if (scrollAmount != 0) {
501             mFlingRunnable.startUsingDistance(scrollAmount);
502         } else {
503             onFinishedMovement();
504         }
505     }
506 
onFinishedMovement()507     private void onFinishedMovement() {
508         if (mSuppressSelectionChanged) {
509             mSuppressSelectionChanged = false;
510 
511             // We haven't been callbacking during the fling, so do it now
512             super.selectionChanged();
513         }
514         invalidate();
515     }
516 
517     @Override
selectionChanged()518     void selectionChanged() {
519         if (!mSuppressSelectionChanged) {
520             super.selectionChanged();
521         }
522     }
523 
524     /**
525      * Looks for the child that is closest to the center and sets it as the
526      * selected child.
527      */
setSelectionToCenterChild()528     private void setSelectionToCenterChild() {
529 
530         View selView = mSelectedChild;
531         if (mSelectedChild == null) return;
532 
533         int galleryCenter = getCenterOfGallery();
534 
535         // Common case where the current selected position is correct
536         if (selView.getLeft() <= galleryCenter && selView.getRight() >= galleryCenter) {
537             return;
538         }
539 
540         // TODO better search
541         int closestEdgeDistance = Integer.MAX_VALUE;
542         int newSelectedChildIndex = 0;
543         for (int i = getChildCount() - 1; i >= 0; i--) {
544 
545             View child = getChildAt(i);
546 
547             if (child.getLeft() <= galleryCenter && child.getRight() >=  galleryCenter) {
548                 // This child is in the center
549                 newSelectedChildIndex = i;
550                 break;
551             }
552 
553             int childClosestEdgeDistance = Math.min(Math.abs(child.getLeft() - galleryCenter),
554                     Math.abs(child.getRight() - galleryCenter));
555             if (childClosestEdgeDistance < closestEdgeDistance) {
556                 closestEdgeDistance = childClosestEdgeDistance;
557                 newSelectedChildIndex = i;
558             }
559         }
560 
561         int newPos = mFirstPosition + newSelectedChildIndex;
562 
563         if (newPos != mSelectedPosition) {
564             setSelectedPositionInt(newPos);
565             setNextSelectedPositionInt(newPos);
566             checkSelectionChanged();
567         }
568     }
569 
570     /**
571      * Creates and positions all views for this Gallery.
572      * <p>
573      * We layout rarely, most of the time {@link #trackMotionScroll(int)} takes
574      * care of repositioning, adding, and removing children.
575      *
576      * @param delta Change in the selected position. +1 means the selection is
577      *            moving to the right, so views are scrolling to the left. -1
578      *            means the selection is moving to the left.
579      */
580     @Override
layout(int delta, boolean animate)581     void layout(int delta, boolean animate) {
582 
583         int childrenLeft = mSpinnerPadding.left;
584         int childrenWidth = mRight - mLeft - mSpinnerPadding.left - mSpinnerPadding.right;
585 
586         if (mDataChanged) {
587             handleDataChanged();
588         }
589 
590         // Handle an empty gallery by removing all views.
591         if (mItemCount == 0) {
592             resetList();
593             return;
594         }
595 
596         // Update to the new selected position.
597         if (mNextSelectedPosition >= 0) {
598             setSelectedPositionInt(mNextSelectedPosition);
599         }
600 
601         // All views go in recycler while we are in layout
602         recycleAllViews();
603 
604         // Clear out old views
605         //removeAllViewsInLayout();
606         detachAllViewsFromParent();
607 
608         /*
609          * These will be used to give initial positions to views entering the
610          * gallery as we scroll
611          */
612         mRightMost = 0;
613         mLeftMost = 0;
614 
615         // Make selected view and center it
616 
617         /*
618          * mFirstPosition will be decreased as we add views to the left later
619          * on. The 0 for x will be offset in a couple lines down.
620          */
621         mFirstPosition = mSelectedPosition;
622         View sel = makeAndAddView(mSelectedPosition, 0, 0, true);
623 
624         // Put the selected child in the center
625         int selectedOffset = childrenLeft + (childrenWidth / 2) - (sel.getWidth() / 2);
626         sel.offsetLeftAndRight(selectedOffset);
627 
628         fillToGalleryRight();
629         fillToGalleryLeft();
630 
631         // Flush any cached views that did not get reused above
632         mRecycler.clear();
633 
634         invalidate();
635         checkSelectionChanged();
636 
637         mDataChanged = false;
638         mNeedSync = false;
639         setNextSelectedPositionInt(mSelectedPosition);
640 
641         updateSelectedItemMetadata();
642     }
643 
fillToGalleryLeft()644     private void fillToGalleryLeft() {
645         int itemSpacing = mSpacing;
646         int galleryLeft = mPaddingLeft;
647 
648         // Set state for initial iteration
649         View prevIterationView = getChildAt(0);
650         int curPosition;
651         int curRightEdge;
652 
653         if (prevIterationView != null) {
654             curPosition = mFirstPosition - 1;
655             curRightEdge = prevIterationView.getLeft() - itemSpacing;
656         } else {
657             // No children available!
658             curPosition = 0;
659             curRightEdge = mRight - mLeft - mPaddingRight;
660             mShouldStopFling = true;
661         }
662 
663         while (curRightEdge > galleryLeft && curPosition >= 0) {
664             prevIterationView = makeAndAddView(curPosition, curPosition - mSelectedPosition,
665                     curRightEdge, false);
666 
667             // Remember some state
668             mFirstPosition = curPosition;
669 
670             // Set state for next iteration
671             curRightEdge = prevIterationView.getLeft() - itemSpacing;
672             curPosition--;
673         }
674     }
675 
fillToGalleryRight()676     private void fillToGalleryRight() {
677         int itemSpacing = mSpacing;
678         int galleryRight = mRight - mLeft - mPaddingRight;
679         int numChildren = getChildCount();
680         int numItems = mItemCount;
681 
682         // Set state for initial iteration
683         View prevIterationView = getChildAt(numChildren - 1);
684         int curPosition;
685         int curLeftEdge;
686 
687         if (prevIterationView != null) {
688             curPosition = mFirstPosition + numChildren;
689             curLeftEdge = prevIterationView.getRight() + itemSpacing;
690         } else {
691             mFirstPosition = curPosition = mItemCount - 1;
692             curLeftEdge = mPaddingLeft;
693             mShouldStopFling = true;
694         }
695 
696         while (curLeftEdge < galleryRight && curPosition < numItems) {
697             prevIterationView = makeAndAddView(curPosition, curPosition - mSelectedPosition,
698                     curLeftEdge, true);
699 
700             // Set state for next iteration
701             curLeftEdge = prevIterationView.getRight() + itemSpacing;
702             curPosition++;
703         }
704     }
705 
706     /**
707      * Obtain a view, either by pulling an existing view from the recycler or by
708      * getting a new one from the adapter. If we are animating, make sure there
709      * is enough information in the view's layout parameters to animate from the
710      * old to new positions.
711      *
712      * @param position Position in the gallery for the view to obtain
713      * @param offset Offset from the selected position
714      * @param x X-coordintate indicating where this view should be placed. This
715      *        will either be the left or right edge of the view, depending on
716      *        the fromLeft paramter
717      * @param fromLeft Are we posiitoning views based on the left edge? (i.e.,
718      *        building from left to right)?
719      * @return A view that has been added to the gallery
720      */
makeAndAddView(int position, int offset, int x, boolean fromLeft)721     private View makeAndAddView(int position, int offset, int x,
722             boolean fromLeft) {
723 
724         View child;
725 
726         if (!mDataChanged) {
727             child = mRecycler.get(position);
728             if (child != null) {
729                 // Can reuse an existing view
730                 int childLeft = child.getLeft();
731 
732                 // Remember left and right edges of where views have been placed
733                 mRightMost = Math.max(mRightMost, childLeft
734                         + child.getMeasuredWidth());
735                 mLeftMost = Math.min(mLeftMost, childLeft);
736 
737                 // Position the view
738                 setUpChild(child, offset, x, fromLeft);
739 
740                 return child;
741             }
742         }
743 
744         // Nothing found in the recycler -- ask the adapter for a view
745         child = mAdapter.getView(position, null, this);
746 
747         // Position the view
748         setUpChild(child, offset, x, fromLeft);
749 
750         return child;
751     }
752 
753     /**
754      * Helper for makeAndAddView to set the position of a view and fill out its
755      * layout paramters.
756      *
757      * @param child The view to position
758      * @param offset Offset from the selected position
759      * @param x X-coordintate indicating where this view should be placed. This
760      *        will either be the left or right edge of the view, depending on
761      *        the fromLeft paramter
762      * @param fromLeft Are we posiitoning views based on the left edge? (i.e.,
763      *        building from left to right)?
764      */
setUpChild(View child, int offset, int x, boolean fromLeft)765     private void setUpChild(View child, int offset, int x, boolean fromLeft) {
766 
767         // Respect layout params that are already in the view. Otherwise
768         // make some up...
769         Gallery.LayoutParams lp = (Gallery.LayoutParams)
770             child.getLayoutParams();
771         if (lp == null) {
772             lp = (Gallery.LayoutParams) generateDefaultLayoutParams();
773         }
774 
775         addViewInLayout(child, fromLeft ? -1 : 0, lp);
776 
777         child.setSelected(offset == 0);
778 
779         // Get measure specs
780         int childHeightSpec = ViewGroup.getChildMeasureSpec(mHeightMeasureSpec,
781                 mSpinnerPadding.top + mSpinnerPadding.bottom, lp.height);
782         int childWidthSpec = ViewGroup.getChildMeasureSpec(mWidthMeasureSpec,
783                 mSpinnerPadding.left + mSpinnerPadding.right, lp.width);
784 
785         // Measure child
786         child.measure(childWidthSpec, childHeightSpec);
787 
788         int childLeft;
789         int childRight;
790 
791         // Position vertically based on gravity setting
792         int childTop = calculateTop(child, true);
793         int childBottom = childTop + child.getMeasuredHeight();
794 
795         int width = child.getMeasuredWidth();
796         if (fromLeft) {
797             childLeft = x;
798             childRight = childLeft + width;
799         } else {
800             childLeft = x - width;
801             childRight = x;
802         }
803 
804         child.layout(childLeft, childTop, childRight, childBottom);
805     }
806 
807     /**
808      * Figure out vertical placement based on mGravity
809      *
810      * @param child Child to place
811      * @return Where the top of the child should be
812      */
calculateTop(View child, boolean duringLayout)813     private int calculateTop(View child, boolean duringLayout) {
814         int myHeight = duringLayout ? mMeasuredHeight : getHeight();
815         int childHeight = duringLayout ? child.getMeasuredHeight() : child.getHeight();
816 
817         int childTop = 0;
818 
819         switch (mGravity) {
820         case Gravity.TOP:
821             childTop = mSpinnerPadding.top;
822             break;
823         case Gravity.CENTER_VERTICAL:
824             int availableSpace = myHeight - mSpinnerPadding.bottom
825                     - mSpinnerPadding.top - childHeight;
826             childTop = mSpinnerPadding.top + (availableSpace / 2);
827             break;
828         case Gravity.BOTTOM:
829             childTop = myHeight - mSpinnerPadding.bottom - childHeight;
830             break;
831         }
832         return childTop;
833     }
834 
835     @Override
onTouchEvent(MotionEvent event)836     public boolean onTouchEvent(MotionEvent event) {
837 
838         // Give everything to the gesture detector
839         boolean retValue = mGestureDetector.onTouchEvent(event);
840 
841         int action = event.getAction();
842         if (action == MotionEvent.ACTION_UP) {
843             // Helper method for lifted finger
844             onUp();
845         } else if (action == MotionEvent.ACTION_CANCEL) {
846             onCancel();
847         }
848 
849         return retValue;
850     }
851 
852     /**
853      * {@inheritDoc}
854      */
onSingleTapUp(MotionEvent e)855     public boolean onSingleTapUp(MotionEvent e) {
856 
857         if (mDownTouchPosition >= 0) {
858 
859             // An item tap should make it selected, so scroll to this child.
860             scrollToChild(mDownTouchPosition - mFirstPosition);
861 
862             // Also pass the click so the client knows, if it wants to.
863             if (mShouldCallbackOnUnselectedItemClick || mDownTouchPosition == mSelectedPosition) {
864                 performItemClick(mDownTouchView, mDownTouchPosition, mAdapter
865                         .getItemId(mDownTouchPosition));
866             }
867 
868             return true;
869         }
870 
871         return false;
872     }
873 
874     /**
875      * {@inheritDoc}
876      */
onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY)877     public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
878 
879         if (!mShouldCallbackDuringFling) {
880             // We want to suppress selection changes
881 
882             // Remove any future code to set mSuppressSelectionChanged = false
883             removeCallbacks(mDisableSuppressSelectionChangedRunnable);
884 
885             // This will get reset once we scroll into slots
886             if (!mSuppressSelectionChanged) mSuppressSelectionChanged = true;
887         }
888 
889         // Fling the gallery!
890         mFlingRunnable.startUsingVelocity((int) -velocityX);
891 
892         return true;
893     }
894 
895     /**
896      * {@inheritDoc}
897      */
onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY)898     public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {
899 
900         if (localLOGV) Log.v(TAG, String.valueOf(e2.getX() - e1.getX()));
901 
902         /*
903          * Now's a good time to tell our parent to stop intercepting our events!
904          * The user has moved more than the slop amount, since GestureDetector
905          * ensures this before calling this method. Also, if a parent is more
906          * interested in this touch's events than we are, it would have
907          * intercepted them by now (for example, we can assume when a Gallery is
908          * in the ListView, a vertical scroll would not end up in this method
909          * since a ListView would have intercepted it by now).
910          */
911         mParent.requestDisallowInterceptTouchEvent(true);
912 
913         // As the user scrolls, we want to callback selection changes so related-
914         // info on the screen is up-to-date with the gallery's selection
915         if (!mShouldCallbackDuringFling) {
916             if (mIsFirstScroll) {
917                 /*
918                  * We're not notifying the client of selection changes during
919                  * the fling, and this scroll could possibly be a fling. Don't
920                  * do selection changes until we're sure it is not a fling.
921                  */
922                 if (!mSuppressSelectionChanged) mSuppressSelectionChanged = true;
923                 postDelayed(mDisableSuppressSelectionChangedRunnable, SCROLL_TO_FLING_UNCERTAINTY_TIMEOUT);
924             }
925         } else {
926             if (mSuppressSelectionChanged) mSuppressSelectionChanged = false;
927         }
928 
929         // Track the motion
930         trackMotionScroll(-1 * (int) distanceX);
931 
932         mIsFirstScroll = false;
933         return true;
934     }
935 
936     /**
937      * {@inheritDoc}
938      */
onDown(MotionEvent e)939     public boolean onDown(MotionEvent e) {
940 
941         // Kill any existing fling/scroll
942         mFlingRunnable.stop(false);
943 
944         // Get the item's view that was touched
945         mDownTouchPosition = pointToPosition((int) e.getX(), (int) e.getY());
946 
947         if (mDownTouchPosition >= 0) {
948             mDownTouchView = getChildAt(mDownTouchPosition - mFirstPosition);
949             mDownTouchView.setPressed(true);
950         }
951 
952         // Reset the multiple-scroll tracking state
953         mIsFirstScroll = true;
954 
955         // Must return true to get matching events for this down event.
956         return true;
957     }
958 
959     /**
960      * Called when a touch event's action is MotionEvent.ACTION_UP.
961      */
onUp()962     void onUp() {
963 
964         if (mFlingRunnable.mScroller.isFinished()) {
965             scrollIntoSlots();
966         }
967 
968         dispatchUnpress();
969     }
970 
971     /**
972      * Called when a touch event's action is MotionEvent.ACTION_CANCEL.
973      */
onCancel()974     void onCancel() {
975         onUp();
976     }
977 
978     /**
979      * {@inheritDoc}
980      */
onLongPress(MotionEvent e)981     public void onLongPress(MotionEvent e) {
982 
983         if (mDownTouchPosition < 0) {
984             return;
985         }
986 
987         performHapticFeedback(HapticFeedbackConstants.LONG_PRESS);
988         long id = getItemIdAtPosition(mDownTouchPosition);
989         dispatchLongPress(mDownTouchView, mDownTouchPosition, id);
990     }
991 
992     // Unused methods from GestureDetector.OnGestureListener below
993 
994     /**
995      * {@inheritDoc}
996      */
onShowPress(MotionEvent e)997     public void onShowPress(MotionEvent e) {
998     }
999 
1000     // Unused methods from GestureDetector.OnGestureListener above
1001 
dispatchPress(View child)1002     private void dispatchPress(View child) {
1003 
1004         if (child != null) {
1005             child.setPressed(true);
1006         }
1007 
1008         setPressed(true);
1009     }
1010 
dispatchUnpress()1011     private void dispatchUnpress() {
1012 
1013         for (int i = getChildCount() - 1; i >= 0; i--) {
1014             getChildAt(i).setPressed(false);
1015         }
1016 
1017         setPressed(false);
1018     }
1019 
1020     @Override
dispatchSetSelected(boolean selected)1021     public void dispatchSetSelected(boolean selected) {
1022         /*
1023          * We don't want to pass the selected state given from its parent to its
1024          * children since this widget itself has a selected state to give to its
1025          * children.
1026          */
1027     }
1028 
1029     @Override
dispatchSetPressed(boolean pressed)1030     protected void dispatchSetPressed(boolean pressed) {
1031 
1032         // Show the pressed state on the selected child
1033         if (mSelectedChild != null) {
1034             mSelectedChild.setPressed(pressed);
1035         }
1036     }
1037 
1038     @Override
getContextMenuInfo()1039     protected ContextMenuInfo getContextMenuInfo() {
1040         return mContextMenuInfo;
1041     }
1042 
1043     @Override
showContextMenuForChild(View originalView)1044     public boolean showContextMenuForChild(View originalView) {
1045 
1046         final int longPressPosition = getPositionForView(originalView);
1047         if (longPressPosition < 0) {
1048             return false;
1049         }
1050 
1051         final long longPressId = mAdapter.getItemId(longPressPosition);
1052         return dispatchLongPress(originalView, longPressPosition, longPressId);
1053     }
1054 
1055     @Override
showContextMenu()1056     public boolean showContextMenu() {
1057 
1058         if (isPressed() && mSelectedPosition >= 0) {
1059             int index = mSelectedPosition - mFirstPosition;
1060             View v = getChildAt(index);
1061             return dispatchLongPress(v, mSelectedPosition, mSelectedRowId);
1062         }
1063 
1064         return false;
1065     }
1066 
dispatchLongPress(View view, int position, long id)1067     private boolean dispatchLongPress(View view, int position, long id) {
1068         boolean handled = false;
1069 
1070         if (mOnItemLongClickListener != null) {
1071             handled = mOnItemLongClickListener.onItemLongClick(this, mDownTouchView,
1072                     mDownTouchPosition, id);
1073         }
1074 
1075         if (!handled) {
1076             mContextMenuInfo = new AdapterContextMenuInfo(view, position, id);
1077             handled = super.showContextMenuForChild(this);
1078         }
1079 
1080         if (handled) {
1081             performHapticFeedback(HapticFeedbackConstants.LONG_PRESS);
1082         }
1083 
1084         return handled;
1085     }
1086 
1087     @Override
dispatchKeyEvent(KeyEvent event)1088     public boolean dispatchKeyEvent(KeyEvent event) {
1089         // Gallery steals all key events
1090         return event.dispatch(this);
1091     }
1092 
1093     /**
1094      * Handles left, right, and clicking
1095      * @see android.view.View#onKeyDown
1096      */
1097     @Override
onKeyDown(int keyCode, KeyEvent event)1098     public boolean onKeyDown(int keyCode, KeyEvent event) {
1099         switch (keyCode) {
1100 
1101         case KeyEvent.KEYCODE_DPAD_LEFT:
1102             if (movePrevious()) {
1103                 playSoundEffect(SoundEffectConstants.NAVIGATION_LEFT);
1104             }
1105             return true;
1106 
1107         case KeyEvent.KEYCODE_DPAD_RIGHT:
1108             if (moveNext()) {
1109                 playSoundEffect(SoundEffectConstants.NAVIGATION_RIGHT);
1110             }
1111             return true;
1112 
1113         case KeyEvent.KEYCODE_DPAD_CENTER:
1114         case KeyEvent.KEYCODE_ENTER:
1115             mReceivedInvokeKeyDown = true;
1116             // fallthrough to default handling
1117         }
1118 
1119         return super.onKeyDown(keyCode, event);
1120     }
1121 
1122     @Override
onKeyUp(int keyCode, KeyEvent event)1123     public boolean onKeyUp(int keyCode, KeyEvent event) {
1124         switch (keyCode) {
1125         case KeyEvent.KEYCODE_DPAD_CENTER:
1126         case KeyEvent.KEYCODE_ENTER: {
1127 
1128             if (mReceivedInvokeKeyDown) {
1129                 if (mItemCount > 0) {
1130 
1131                     dispatchPress(mSelectedChild);
1132                     postDelayed(new Runnable() {
1133                         public void run() {
1134                             dispatchUnpress();
1135                         }
1136                     }, ViewConfiguration.getPressedStateDuration());
1137 
1138                     int selectedIndex = mSelectedPosition - mFirstPosition;
1139                     performItemClick(getChildAt(selectedIndex), mSelectedPosition, mAdapter
1140                             .getItemId(mSelectedPosition));
1141                 }
1142             }
1143 
1144             // Clear the flag
1145             mReceivedInvokeKeyDown = false;
1146 
1147             return true;
1148         }
1149         }
1150 
1151         return super.onKeyUp(keyCode, event);
1152     }
1153 
movePrevious()1154     boolean movePrevious() {
1155         if (mItemCount > 0 && mSelectedPosition > 0) {
1156             scrollToChild(mSelectedPosition - mFirstPosition - 1);
1157             return true;
1158         } else {
1159             return false;
1160         }
1161     }
1162 
moveNext()1163     boolean moveNext() {
1164         if (mItemCount > 0 && mSelectedPosition < mItemCount - 1) {
1165             scrollToChild(mSelectedPosition - mFirstPosition + 1);
1166             return true;
1167         } else {
1168             return false;
1169         }
1170     }
1171 
scrollToChild(int childPosition)1172     private boolean scrollToChild(int childPosition) {
1173         View child = getChildAt(childPosition);
1174 
1175         if (child != null) {
1176             int distance = getCenterOfGallery() - getCenterOfView(child);
1177             mFlingRunnable.startUsingDistance(distance);
1178             return true;
1179         }
1180 
1181         return false;
1182     }
1183 
1184     @Override
setSelectedPositionInt(int position)1185     void setSelectedPositionInt(int position) {
1186         super.setSelectedPositionInt(position);
1187 
1188         // Updates any metadata we keep about the selected item.
1189         updateSelectedItemMetadata();
1190     }
1191 
updateSelectedItemMetadata()1192     private void updateSelectedItemMetadata() {
1193 
1194         View oldSelectedChild = mSelectedChild;
1195 
1196         View child = mSelectedChild = getChildAt(mSelectedPosition - mFirstPosition);
1197         if (child == null) {
1198             return;
1199         }
1200 
1201         child.setSelected(true);
1202         child.setFocusable(true);
1203 
1204         if (hasFocus()) {
1205             child.requestFocus();
1206         }
1207 
1208         // We unfocus the old child down here so the above hasFocus check
1209         // returns true
1210         if (oldSelectedChild != null) {
1211 
1212             // Make sure its drawable state doesn't contain 'selected'
1213             oldSelectedChild.setSelected(false);
1214 
1215             // Make sure it is not focusable anymore, since otherwise arrow keys
1216             // can make this one be focused
1217             oldSelectedChild.setFocusable(false);
1218         }
1219 
1220     }
1221 
1222     /**
1223      * Describes how the child views are aligned.
1224      * @param gravity
1225      *
1226      * @attr ref android.R.styleable#Gallery_gravity
1227      */
setGravity(int gravity)1228     public void setGravity(int gravity)
1229     {
1230         if (mGravity != gravity) {
1231             mGravity = gravity;
1232             requestLayout();
1233         }
1234     }
1235 
1236     @Override
getChildDrawingOrder(int childCount, int i)1237     protected int getChildDrawingOrder(int childCount, int i) {
1238         int selectedIndex = mSelectedPosition - mFirstPosition;
1239 
1240         // Just to be safe
1241         if (selectedIndex < 0) return i;
1242 
1243         if (i == childCount - 1) {
1244             // Draw the selected child last
1245             return selectedIndex;
1246         } else if (i >= selectedIndex) {
1247             // Move the children to the right of the selected child earlier one
1248             return i + 1;
1249         } else {
1250             // Keep the children to the left of the selected child the same
1251             return i;
1252         }
1253     }
1254 
1255     @Override
onFocusChanged(boolean gainFocus, int direction, Rect previouslyFocusedRect)1256     protected void onFocusChanged(boolean gainFocus, int direction, Rect previouslyFocusedRect) {
1257         super.onFocusChanged(gainFocus, direction, previouslyFocusedRect);
1258 
1259         /*
1260          * The gallery shows focus by focusing the selected item. So, give
1261          * focus to our selected item instead. We steal keys from our
1262          * selected item elsewhere.
1263          */
1264         if (gainFocus && mSelectedChild != null) {
1265             mSelectedChild.requestFocus(direction);
1266         }
1267 
1268     }
1269 
1270     /**
1271      * Responsible for fling behavior. Use {@link #startUsingVelocity(int)} to
1272      * initiate a fling. Each frame of the fling is handled in {@link #run()}.
1273      * A FlingRunnable will keep re-posting itself until the fling is done.
1274      *
1275      */
1276     private class FlingRunnable implements Runnable {
1277         /**
1278          * Tracks the decay of a fling scroll
1279          */
1280         private Scroller mScroller;
1281 
1282         /**
1283          * X value reported by mScroller on the previous fling
1284          */
1285         private int mLastFlingX;
1286 
FlingRunnable()1287         public FlingRunnable() {
1288             mScroller = new Scroller(getContext());
1289         }
1290 
startCommon()1291         private void startCommon() {
1292             // Remove any pending flings
1293             removeCallbacks(this);
1294         }
1295 
startUsingVelocity(int initialVelocity)1296         public void startUsingVelocity(int initialVelocity) {
1297             if (initialVelocity == 0) return;
1298 
1299             startCommon();
1300 
1301             int initialX = initialVelocity < 0 ? Integer.MAX_VALUE : 0;
1302             mLastFlingX = initialX;
1303             mScroller.fling(initialX, 0, initialVelocity, 0,
1304                     0, Integer.MAX_VALUE, 0, Integer.MAX_VALUE);
1305             post(this);
1306         }
1307 
1308         public void startUsingDistance(int distance) {
1309             if (distance == 0) return;
1310 
1311             startCommon();
1312 
1313             mLastFlingX = 0;
1314             mScroller.startScroll(0, 0, -distance, 0, mAnimationDuration);
1315             post(this);
1316         }
1317 
1318         public void stop(boolean scrollIntoSlots) {
1319             removeCallbacks(this);
1320             endFling(scrollIntoSlots);
1321         }
1322 
1323         private void endFling(boolean scrollIntoSlots) {
1324             /*
1325              * Force the scroller's status to finished (without setting its
1326              * position to the end)
1327              */
1328             mScroller.forceFinished(true);
1329 
1330             if (scrollIntoSlots) scrollIntoSlots();
1331         }
1332 
1333         public void run() {
1334 
1335             if (mItemCount == 0) {
1336                 endFling(true);
1337                 return;
1338             }
1339 
1340             mShouldStopFling = false;
1341 
1342             final Scroller scroller = mScroller;
1343             boolean more = scroller.computeScrollOffset();
1344             final int x = scroller.getCurrX();
1345 
1346             // Flip sign to convert finger direction to list items direction
1347             // (e.g. finger moving down means list is moving towards the top)
1348             int delta = mLastFlingX - x;
1349 
1350             // Pretend that each frame of a fling scroll is a touch scroll
1351             if (delta > 0) {
1352                 // Moving towards the left. Use first view as mDownTouchPosition
1353                 mDownTouchPosition = mFirstPosition;
1354 
1355                 // Don't fling more than 1 screen
1356                 delta = Math.min(getWidth() - mPaddingLeft - mPaddingRight - 1, delta);
1357             } else {
1358                 // Moving towards the right. Use last view as mDownTouchPosition
1359                 int offsetToLast = getChildCount() - 1;
1360                 mDownTouchPosition = mFirstPosition + offsetToLast;
1361 
1362                 // Don't fling more than 1 screen
1363                 delta = Math.max(-(getWidth() - mPaddingRight - mPaddingLeft - 1), delta);
1364             }
1365 
1366             trackMotionScroll(delta);
1367 
1368             if (more && !mShouldStopFling) {
1369                 mLastFlingX = x;
1370                 post(this);
1371             } else {
1372                endFling(true);
1373             }
1374         }
1375 
1376     }
1377 
1378     /**
1379      * Gallery extends LayoutParams to provide a place to hold current
1380      * Transformation information along with previous position/transformation
1381      * info.
1382      *
1383      */
1384     public static class LayoutParams extends ViewGroup.LayoutParams {
LayoutParams(Context c, AttributeSet attrs)1385         public LayoutParams(Context c, AttributeSet attrs) {
1386             super(c, attrs);
1387         }
1388 
LayoutParams(int w, int h)1389         public LayoutParams(int w, int h) {
1390             super(w, h);
1391         }
1392 
LayoutParams(ViewGroup.LayoutParams source)1393         public LayoutParams(ViewGroup.LayoutParams source) {
1394             super(source);
1395         }
1396     }
1397 }
1398