Android 自定義實(shí)現(xiàn)switch開(kāi)關(guān)按鈕

在開(kāi)發(fā)我們經(jīng)常會(huì)使用到自定義實(shí)現(xiàn)控件, 今天呢就跟大家說(shuō)一說(shuō)switch開(kāi)關(guān)按鈕的自定義實(shí)現(xiàn)悼瘾。

為了大家的方便赴邻,直接把代碼貼出來(lái),不想那么麻煩自己去寫(xiě)的朋友可以直接copy到你們的項(xiàng)目中使用哦紊选,顏色轿腺、大小都可以根據(jù)自己的需求改一改就行啦!

下面貼出整個(gè)自定義類(lèi)的代碼:

import android.animation.Animator;
import android.animation.AnimatorListenerAdapter;
import android.animation.ValueAnimator;
import android.animation.ValueAnimator.AnimatorUpdateListener;
import android.annotation.SuppressLint;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.graphics.RectF;
import android.os.Bundle;
import android.os.Looper;
import android.os.Parcelable;
import android.support.v4.view.MotionEventCompat;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.animation.AccelerateDecelerateInterpolator;

public class SlideSwitcher extends View {

public static final int SHAPE_RECT = 1;
public static final int SHAPE_CIRCLE = 2;
private static final int RIM_SIZE = 4;
private static final int DEFAULT_OPEN_COLOR = 0xFFC80000;
private static final int DEFAULT_CLOSE_COLOR = 0xFFD0D0D0;

// 4 attributes
private int openColor;
private int closeColor;
private boolean isOpen;
private int shape;

// varials of drawing
private Paint paint;
private Rect backRect;
private Rect frontRect;
private RectF frontCircleRect;
private RectF backCircleRect;
private int alpha;
private int maxLeft;
private int minLeft;
private int frontRectLeft;
private int frontRectLeftBegin = RIM_SIZE;
private int eventStartX;
private int eventLastX;
private int diffX = 0;
private boolean slideable = true;
private SlideListener mSlideListener;

public interface SlideListener {
    void onStatusChanged(SlideSwitcher view, boolean isOpen);
}

public SlideSwitcher(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);

    paint = new Paint();
    paint.setAntiAlias(true);

    openColor = DEFAULT_OPEN_COLOR;
    closeColor = DEFAULT_CLOSE_COLOR;
    isOpen = false;
    shape = SHAPE_CIRCLE;
}

public SlideSwitcher(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
}

public SlideSwitcher(Context context) {
    this(context, null);
}

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

    int width = measureDimension(280, widthMeasureSpec);
    int height = measureDimension(140, heightMeasureSpec);
    if (shape == SHAPE_CIRCLE) {
        if (width < height) width = height * 2;
    }
    setMeasuredDimension(width, height);
    initDrawingVal();
}

private void initDrawingVal() {
    int width = getMeasuredWidth();
    int height = getMeasuredHeight();

    backCircleRect = new RectF();
    frontCircleRect = new RectF();
    frontRect = new Rect();
    backRect = new Rect(0, 0, width, height);
    minLeft = RIM_SIZE;
    if (shape == SHAPE_RECT)
        maxLeft = width / 2;
    else
        maxLeft = width - (height - 2 * RIM_SIZE) - RIM_SIZE;
    if (isOpen) {
        frontRectLeft = maxLeft;
        alpha = 255;
    } else {
        frontRectLeft = RIM_SIZE;
        alpha = 0;
    }
    frontRectLeftBegin = frontRectLeft;
}

private int measureDimension(int defaultSize, int measureSpec) {
    int result;
    int specMode = MeasureSpec.getMode(measureSpec);
    int specSize = MeasureSpec.getSize(measureSpec);
    if (specMode == MeasureSpec.EXACTLY) {
        result = specSize;
    } else {
        result = defaultSize; // UNSPECIFIED
        if (specMode == MeasureSpec.AT_MOST) {
            result = Math.min(result, specSize);
        }
    }
    return result;
}

@Override
protected void onDraw(Canvas canvas) {
    if (shape == SHAPE_RECT) {
        paint.setColor(closeColor);
        canvas.drawRect(backRect, paint);
        paint.setColor(openColor);
        paint.setAlpha(alpha);
        canvas.drawRect(backRect, paint);
        frontRect
                .set(frontRectLeft, RIM_SIZE, frontRectLeft + getMeasuredWidth() / 2 - RIM_SIZE, getMeasuredHeight() - RIM_SIZE);
        paint.setColor(Color.WHITE);
        canvas.drawRect(frontRect, paint);
    } else {
        // draw circle
        int radius;
        radius = backRect.height() / 2;
        paint.setColor(closeColor);
        backCircleRect.set(backRect);
        canvas.drawRoundRect(backCircleRect, radius, radius, paint);
        paint.setColor(openColor);
        paint.setAlpha(alpha);
        canvas.drawRoundRect(backCircleRect, radius, radius, paint);
        frontRect.set(frontRectLeft, RIM_SIZE, frontRectLeft + backRect.height() - 2 * RIM_SIZE, backRect.height() - RIM_SIZE);
        frontCircleRect.set(frontRect);
        paint.setColor(Color.WHITE);
        canvas.drawRoundRect(frontCircleRect, radius, radius, paint);
    }
}

@SuppressLint("ClickableViewAccessibility")
@Override
public boolean onTouchEvent(MotionEvent event) {
    if (slideable == false)
        return super.onTouchEvent(event);
    int action = MotionEventCompat.getActionMasked(event);
    switch (action) {
        case MotionEvent.ACTION_DOWN:
            eventStartX = (int) event.getRawX();
            break;
        case MotionEvent.ACTION_MOVE:
            eventLastX = (int) event.getRawX();
            diffX = eventLastX - eventStartX;
            int tempX = diffX + frontRectLeftBegin;
            tempX = (tempX > maxLeft ? maxLeft : tempX);
            tempX = (tempX < minLeft ? minLeft : tempX);
            if (tempX >= minLeft && tempX <= maxLeft) {
                frontRectLeft = tempX;
                alpha = (int) (255 * (float) tempX / (float) maxLeft);
                invalidateView();
            }
            break;
        case MotionEvent.ACTION_UP:
            int wholeX = (int) (event.getRawX() - eventStartX);
            frontRectLeftBegin = frontRectLeft;
            boolean toRight;
            toRight = (frontRectLeftBegin > maxLeft / 2 ? true : false);
            if (Math.abs(wholeX) < 3) {
                toRight = !toRight;
            }
            moveToDest(toRight);
            break;
        default:
            break;
    }
    return true;
}

/**
 * draw again
 */
private void invalidateView() {
    if (Looper.getMainLooper() == Looper.myLooper()) {
        invalidate();
    } else {
        postInvalidate();
    }
}

public void setSlideListener(SlideListener listener) {
    this.mSlideListener = listener;
}

public void moveToDest(final boolean toRight) {
    ValueAnimator toDestAnim = ValueAnimator.ofInt(frontRectLeft, toRight ? maxLeft : minLeft);
    toDestAnim.setDuration(300);
    toDestAnim.setInterpolator(new AccelerateDecelerateInterpolator());
    toDestAnim.start();
    toDestAnim.addUpdateListener(new AnimatorUpdateListener() {

        @Override
        public void onAnimationUpdate(ValueAnimator animation) {
            frontRectLeft = (Integer) animation.getAnimatedValue();
            alpha = (int) (255 * (float) frontRectLeft / (float) maxLeft);
            invalidateView();
        }
    });
    toDestAnim.addListener(new AnimatorListenerAdapter() {
        @Override
        public void onAnimationEnd(Animator animation) {
            if (toRight) {
                isOpen = true;
                frontRectLeftBegin = maxLeft;
            } else {
                isOpen = false;
                frontRectLeftBegin = minLeft;
            }

            if (mSlideListener != null)
                mSlideListener.onStatusChanged(SlideSwitcher.this, isOpen);
        }
    });
}

public void setOpen(boolean isOpen) {
    setOpen(isOpen, false);
}

public void setOpen(boolean isOpen, boolean invokListener) {
    this.isOpen = isOpen;
    initDrawingVal();
    invalidateView();
    if (invokListener && mSlideListener != null) mSlideListener.onStatusChanged(this, isOpen);
}

public void setShapeType(int shapeType) {
    this.shape = shapeType;
}

public void setSlideable(boolean slideable) {
    this.slideable = slideable;
}

@Override
protected void onRestoreInstanceState(Parcelable state) {
    if (state instanceof Bundle) {
        Bundle bundle = (Bundle) state;
        this.isOpen = bundle.getBoolean("isOpen");
        state = bundle.getParcelable("instanceState");
    }
    super.onRestoreInstanceState(state);
}

@Override
protected Parcelable onSaveInstanceState() {
    Bundle bundle = new Bundle();
    bundle.putParcelable("instanceState", super.onSaveInstanceState());
    bundle.putBoolean("isOpen", this.isOpen);
    return bundle;
  }
}

下面是布局文件:

//在布局中使用
 <com.***.SlideSwitcher
            android:id="@+id/switch_btn"
            android:layout_width="48dp"
            android:layout_height="28dp"
            android:layout_alignParentRight="true"
            android:layout_centerVertical="true" />
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市丛楚,隨后出現(xiàn)的幾起案子族壳,更是在濱河造成了極大的恐慌,老刑警劉巖趣些,帶你破解...
    沈念sama閱讀 221,430評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件仿荆,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡坏平,警方通過(guò)查閱死者的電腦和手機(jī)拢操,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,406評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)舶替,“玉大人令境,你說(shuō)我怎么就攤上這事」说桑” “怎么了舔庶?”我有些...
    開(kāi)封第一講書(shū)人閱讀 167,834評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)陈醒。 經(jīng)常有香客問(wèn)我惕橙,道長(zhǎng),這世上最難降的妖魔是什么钉跷? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,543評(píng)論 1 296
  • 正文 為了忘掉前任弥鹦,我火速辦了婚禮,結(jié)果婚禮上爷辙,老公的妹妹穿的比我還像新娘彬坏。我一直安慰自己,他們只是感情好膝晾,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,547評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布栓始。 她就那樣靜靜地躺著,像睡著了一般玷犹。 火紅的嫁衣襯著肌膚如雪混滔。 梳的紋絲不亂的頭發(fā)上洒疚,一...
    開(kāi)封第一講書(shū)人閱讀 52,196評(píng)論 1 308
  • 那天,我揣著相機(jī)與錄音坯屿,去河邊找鬼油湖。 笑死,一個(gè)胖子當(dāng)著我的面吹牛领跛,可吹牛的內(nèi)容都是我干的乏德。 我是一名探鬼主播,決...
    沈念sama閱讀 40,776評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼吠昭,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼喊括!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起矢棚,我...
    開(kāi)封第一講書(shū)人閱讀 39,671評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤郑什,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后蒲肋,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體蘑拯,經(jīng)...
    沈念sama閱讀 46,221評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,303評(píng)論 3 340
  • 正文 我和宋清朗相戀三年兜粘,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了申窘。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,444評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡孔轴,死狀恐怖剃法,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情路鹰,我是刑警寧澤贷洲,帶...
    沈念sama閱讀 36,134評(píng)論 5 350
  • 正文 年R本政府宣布郊霎,位于F島的核電站绰姻,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜趣斤,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,810評(píng)論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望黎休。 院中可真熱鬧浓领,春花似錦、人聲如沸势腮。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,285評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)捎拯。三九已至泪幌,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背祸泪。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,399評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工吗浩, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人没隘。 一個(gè)月前我還...
    沈念sama閱讀 48,837評(píng)論 3 376
  • 正文 我出身青樓懂扼,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親右蒲。 傳聞我的和親對(duì)象是個(gè)殘疾皇子阀湿,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,455評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,277評(píng)論 25 707
  • 用兩張圖告訴你,為什么你的 App 會(huì)卡頓? - Android - 掘金 Cover 有什么料瑰妄? 從這篇文章中你...
    hw1212閱讀 12,744評(píng)論 2 59
  • 1陷嘴、通過(guò)CocoaPods安裝項(xiàng)目名稱(chēng)項(xiàng)目信息 AFNetworking網(wǎng)絡(luò)請(qǐng)求組件 FMDB本地?cái)?shù)據(jù)庫(kù)組件 SD...
    陽(yáng)明先生_X自主閱讀 15,988評(píng)論 3 119
  • 一、Windows API 1间坐、進(jìn)程 創(chuàng)建進(jìn)程參考文檔:CreateProcess 注意:本章內(nèi)容所有使用的函數(shù)都...
    卍卍_卐卐閱讀 2,162評(píng)論 0 0
  • 混職場(chǎng)的都知道罩旋,我們基本上都是會(huì)議動(dòng)物,特別是我們這些個(gè)做項(xiàng)目的眶诈,不是在開(kāi)會(huì)涨醋,就是在去開(kāi)會(huì)的路上。我想大家最討厭的...
    哪兒黑閱讀 855評(píng)論 0 12