Home | History | Annotate | Download | only in widget
      1 /*
      2  * Copyright (C) 2006 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.widget.ScrollBarUtils;
     20 
     21 import android.annotation.NonNull;
     22 import android.graphics.Canvas;
     23 import android.graphics.ColorFilter;
     24 import android.graphics.PixelFormat;
     25 import android.graphics.Rect;
     26 import android.graphics.drawable.Drawable;
     27 
     28 /**
     29  * This is only used by View for displaying its scroll bars. It should probably
     30  * be moved in to the view package since it is used in that lower-level layer.
     31  * For now, we'll hide it so it can be cleaned up later.
     32  *
     33  * {@hide}
     34  */
     35 public class ScrollBarDrawable extends Drawable implements Drawable.Callback {
     36     private Drawable mVerticalTrack;
     37     private Drawable mHorizontalTrack;
     38     private Drawable mVerticalThumb;
     39     private Drawable mHorizontalThumb;
     40 
     41     private int mRange;
     42     private int mOffset;
     43     private int mExtent;
     44 
     45     private boolean mVertical;
     46     private boolean mBoundsChanged;
     47     private boolean mRangeChanged;
     48     private boolean mAlwaysDrawHorizontalTrack;
     49     private boolean mAlwaysDrawVerticalTrack;
     50     private boolean mMutated;
     51 
     52     private int mAlpha = 255;
     53     private boolean mHasSetAlpha;
     54 
     55     private ColorFilter mColorFilter;
     56     private boolean mHasSetColorFilter;
     57 
     58     /**
     59      * Indicate whether the horizontal scrollbar track should always be drawn
     60      * regardless of the extent. Defaults to false.
     61      *
     62      * @param alwaysDrawTrack Whether the track should always be drawn
     63      *
     64      * @see #getAlwaysDrawHorizontalTrack()
     65      */
     66     public void setAlwaysDrawHorizontalTrack(boolean alwaysDrawTrack) {
     67         mAlwaysDrawHorizontalTrack = alwaysDrawTrack;
     68     }
     69 
     70     /**
     71      * Indicate whether the vertical scrollbar track should always be drawn
     72      * regardless of the extent. Defaults to false.
     73      *
     74      * @param alwaysDrawTrack Whether the track should always be drawn
     75      *
     76      * @see #getAlwaysDrawVerticalTrack()
     77      */
     78     public void setAlwaysDrawVerticalTrack(boolean alwaysDrawTrack) {
     79         mAlwaysDrawVerticalTrack = alwaysDrawTrack;
     80     }
     81 
     82     /**
     83      * @return whether the vertical scrollbar track should always be drawn
     84      *         regardless of the extent.
     85      *
     86      * @see #setAlwaysDrawVerticalTrack(boolean)
     87      */
     88     public boolean getAlwaysDrawVerticalTrack() {
     89         return mAlwaysDrawVerticalTrack;
     90     }
     91 
     92     /**
     93      * @return whether the horizontal scrollbar track should always be drawn
     94      *         regardless of the extent.
     95      *
     96      * @see #setAlwaysDrawHorizontalTrack(boolean)
     97      */
     98     public boolean getAlwaysDrawHorizontalTrack() {
     99         return mAlwaysDrawHorizontalTrack;
    100     }
    101 
    102     public void setParameters(int range, int offset, int extent, boolean vertical) {
    103         if (mVertical != vertical) {
    104             mVertical = vertical;
    105 
    106             mBoundsChanged = true;
    107         }
    108 
    109         if (mRange != range || mOffset != offset || mExtent != extent) {
    110             mRange = range;
    111             mOffset = offset;
    112             mExtent = extent;
    113 
    114             mRangeChanged = true;
    115         }
    116     }
    117 
    118     @Override
    119     public void draw(Canvas canvas) {
    120         final boolean vertical = mVertical;
    121         final int extent = mExtent;
    122         final int range = mRange;
    123 
    124         boolean drawTrack = true;
    125         boolean drawThumb = true;
    126         if (extent <= 0 || range <= extent) {
    127             drawTrack = vertical ? mAlwaysDrawVerticalTrack : mAlwaysDrawHorizontalTrack;
    128             drawThumb = false;
    129         }
    130 
    131         final Rect r = getBounds();
    132         if (canvas.quickReject(r.left, r.top, r.right, r.bottom, Canvas.EdgeType.AA)) {
    133             return;
    134         }
    135 
    136         if (drawTrack) {
    137             drawTrack(canvas, r, vertical);
    138         }
    139 
    140         if (drawThumb) {
    141             final int scrollBarLength = vertical ? r.height() : r.width();
    142             final int thickness = vertical ? r.width() : r.height();
    143             final int thumbLength =
    144                     ScrollBarUtils.getThumbLength(scrollBarLength, thickness, extent, range);
    145             final int thumbOffset =
    146                     ScrollBarUtils.getThumbOffset(scrollBarLength, thumbLength, extent, range,
    147                             mOffset);
    148 
    149             drawThumb(canvas, r, thumbOffset, thumbLength, vertical);
    150         }
    151     }
    152 
    153     @Override
    154     protected void onBoundsChange(Rect bounds) {
    155         super.onBoundsChange(bounds);
    156         mBoundsChanged = true;
    157     }
    158 
    159     @Override
    160     public boolean isStateful() {
    161         return (mVerticalTrack != null && mVerticalTrack.isStateful())
    162                 || (mVerticalThumb != null && mVerticalThumb.isStateful())
    163                 || (mHorizontalTrack != null && mHorizontalTrack.isStateful())
    164                 || (mHorizontalThumb != null && mHorizontalThumb.isStateful())
    165                 || super.isStateful();
    166     }
    167 
    168     @Override
    169     protected boolean onStateChange(int[] state) {
    170         boolean changed = super.onStateChange(state);
    171         if (mVerticalTrack != null) {
    172             changed |= mVerticalTrack.setState(state);
    173         }
    174         if (mVerticalThumb != null) {
    175             changed |= mVerticalThumb.setState(state);
    176         }
    177         if (mHorizontalTrack != null) {
    178             changed |= mHorizontalTrack.setState(state);
    179         }
    180         if (mHorizontalThumb != null) {
    181             changed |= mHorizontalThumb.setState(state);
    182         }
    183         return changed;
    184     }
    185 
    186     private void drawTrack(Canvas canvas, Rect bounds, boolean vertical) {
    187         final Drawable track;
    188         if (vertical) {
    189             track = mVerticalTrack;
    190         } else {
    191             track = mHorizontalTrack;
    192         }
    193 
    194         if (track != null) {
    195             if (mBoundsChanged) {
    196                 track.setBounds(bounds);
    197             }
    198             track.draw(canvas);
    199         }
    200     }
    201 
    202     private void drawThumb(Canvas canvas, Rect bounds, int offset, int length, boolean vertical) {
    203         final boolean changed = mRangeChanged || mBoundsChanged;
    204         if (vertical) {
    205             if (mVerticalThumb != null) {
    206                 final Drawable thumb = mVerticalThumb;
    207                 if (changed) {
    208                     thumb.setBounds(bounds.left, bounds.top + offset,
    209                             bounds.right, bounds.top + offset + length);
    210                 }
    211 
    212                 thumb.draw(canvas);
    213             }
    214         } else {
    215             if (mHorizontalThumb != null) {
    216                 final Drawable thumb = mHorizontalThumb;
    217                 if (changed) {
    218                     thumb.setBounds(bounds.left + offset, bounds.top,
    219                             bounds.left + offset + length, bounds.bottom);
    220                 }
    221 
    222                 thumb.draw(canvas);
    223             }
    224         }
    225     }
    226 
    227     public void setVerticalThumbDrawable(Drawable thumb) {
    228         if (mVerticalThumb != null) {
    229             mVerticalThumb.setCallback(null);
    230         }
    231 
    232         propagateCurrentState(thumb);
    233         mVerticalThumb = thumb;
    234     }
    235 
    236     public void setVerticalTrackDrawable(Drawable track) {
    237         if (mVerticalTrack != null) {
    238             mVerticalTrack.setCallback(null);
    239         }
    240 
    241         propagateCurrentState(track);
    242         mVerticalTrack = track;
    243     }
    244 
    245     public void setHorizontalThumbDrawable(Drawable thumb) {
    246         if (mHorizontalThumb != null) {
    247             mHorizontalThumb.setCallback(null);
    248         }
    249 
    250         propagateCurrentState(thumb);
    251         mHorizontalThumb = thumb;
    252     }
    253 
    254     public void setHorizontalTrackDrawable(Drawable track) {
    255         if (mHorizontalTrack != null) {
    256             mHorizontalTrack.setCallback(null);
    257         }
    258 
    259         propagateCurrentState(track);
    260         mHorizontalTrack = track;
    261     }
    262 
    263     private void propagateCurrentState(Drawable d) {
    264         if (d != null) {
    265             if (mMutated) {
    266                 d.mutate();
    267             }
    268 
    269             d.setState(getState());
    270             d.setCallback(this);
    271 
    272             if (mHasSetAlpha) {
    273                 d.setAlpha(mAlpha);
    274             }
    275 
    276             if (mHasSetColorFilter) {
    277                 d.setColorFilter(mColorFilter);
    278             }
    279         }
    280     }
    281 
    282     public int getSize(boolean vertical) {
    283         if (vertical) {
    284             return mVerticalTrack != null ? mVerticalTrack.getIntrinsicWidth() :
    285                     mVerticalThumb != null ? mVerticalThumb.getIntrinsicWidth() : 0;
    286         } else {
    287             return mHorizontalTrack != null ? mHorizontalTrack.getIntrinsicHeight() :
    288                     mHorizontalThumb != null ? mHorizontalThumb.getIntrinsicHeight() : 0;
    289         }
    290     }
    291 
    292     @Override
    293     public ScrollBarDrawable mutate() {
    294         if (!mMutated && super.mutate() == this) {
    295             if (mVerticalTrack != null) {
    296                 mVerticalTrack.mutate();
    297             }
    298             if (mVerticalThumb != null) {
    299                 mVerticalThumb.mutate();
    300             }
    301             if (mHorizontalTrack != null) {
    302                 mHorizontalTrack.mutate();
    303             }
    304             if (mHorizontalThumb != null) {
    305                 mHorizontalThumb.mutate();
    306             }
    307             mMutated = true;
    308         }
    309         return this;
    310     }
    311 
    312     @Override
    313     public void setAlpha(int alpha) {
    314         mAlpha = alpha;
    315         mHasSetAlpha = true;
    316 
    317         if (mVerticalTrack != null) {
    318             mVerticalTrack.setAlpha(alpha);
    319         }
    320         if (mVerticalThumb != null) {
    321             mVerticalThumb.setAlpha(alpha);
    322         }
    323         if (mHorizontalTrack != null) {
    324             mHorizontalTrack.setAlpha(alpha);
    325         }
    326         if (mHorizontalThumb != null) {
    327             mHorizontalThumb.setAlpha(alpha);
    328         }
    329     }
    330 
    331     @Override
    332     public int getAlpha() {
    333         return mAlpha;
    334     }
    335 
    336     @Override
    337     public void setColorFilter(ColorFilter colorFilter) {
    338         mColorFilter = colorFilter;
    339         mHasSetColorFilter = true;
    340 
    341         if (mVerticalTrack != null) {
    342             mVerticalTrack.setColorFilter(colorFilter);
    343         }
    344         if (mVerticalThumb != null) {
    345             mVerticalThumb.setColorFilter(colorFilter);
    346         }
    347         if (mHorizontalTrack != null) {
    348             mHorizontalTrack.setColorFilter(colorFilter);
    349         }
    350         if (mHorizontalThumb != null) {
    351             mHorizontalThumb.setColorFilter(colorFilter);
    352         }
    353     }
    354 
    355     @Override
    356     public ColorFilter getColorFilter() {
    357         return mColorFilter;
    358     }
    359 
    360     @Override
    361     public int getOpacity() {
    362         return PixelFormat.TRANSLUCENT;
    363     }
    364 
    365     @Override
    366     public void invalidateDrawable(@NonNull Drawable who) {
    367         invalidateSelf();
    368     }
    369 
    370     @Override
    371     public void scheduleDrawable(@NonNull Drawable who, @NonNull Runnable what, long when) {
    372         scheduleSelf(what, when);
    373     }
    374 
    375     @Override
    376     public void unscheduleDrawable(@NonNull Drawable who, @NonNull Runnable what) {
    377         unscheduleSelf(what);
    378     }
    379 
    380     @Override
    381     public String toString() {
    382         return "ScrollBarDrawable: range=" + mRange + " offset=" + mOffset +
    383                " extent=" + mExtent + (mVertical ? " V" : " H");
    384     }
    385 }
    386 
    387 
    388