自定義Switch

不多說剖毯,直接上代碼吧!

package com.smartisan.muse.ui.widget;

/**
 * Created by Administrator on 2018/3/28.
 */

import android.animation.ValueAnimator;
import android.content.Context;
import android.graphics.drawable.Drawable;
import android.graphics.drawable.TransitionDrawable;
import android.util.AttributeSet;
import android.util.Log;
import android.view.Gravity;
import android.view.View;
import android.view.ViewGroup;
import android.view.ViewTreeObserver;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.LinearLayout;

import com.smartisan.muse.R;

import static android.content.ContentValues.TAG;


/**
 * 自定義switch開關(guān)
 */

public class MySwitch extends FrameLayout {
    private Context context;
    private final static int MP = ViewGroup.LayoutParams.MATCH_PARENT;
    private final static int WC = ViewGroup.LayoutParams.WRAP_CONTENT;
    private Drawable bg_on;
    private Drawable bg_off;
    private ImageView ball;
    private ImageView switchBg;
    private int local_state = 0;//本地狀態(tài) 0:關(guān)  1:開
    private int mWidth;
    private int mHeight;
    private int mBallWidth;


    public MySwitch(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        init(context);
        this.context = context;
    }

    public MySwitch(Context context, AttributeSet attrs) {
        super(context, attrs);
        init(context);
        this.context = context;
    }

    public MySwitch(Context context) {
        super(context);
        init(context);
        this.context = context;
    }

    private void init(Context context) {

        /**整體布局預(yù)設(shè)**/
        bg_on = getResources().getDrawable(R.drawable.switch_on_bg);
        bg_off = getResources().getDrawable(R.drawable.switch_off_bg);
        FrameLayout.LayoutParams FParams;
//                =new FrameLayout.LayoutParams(WC,getRealPixel(59));
//        setLayoutParams(FParams);

        ImageView switch_default_Bg = new ImageView(context);
        FParams = new FrameLayout.LayoutParams(WC, WC);
        switch_default_Bg.setImageResource(R.drawable.switch_off_bg);
        addView(switch_default_Bg, FParams);


        switchBg = new ImageView(context);
        FParams = new FrameLayout.LayoutParams(WC, WC);
        addView(switchBg, FParams);


        //開關(guān)圓球
        LinearLayout ballLayout = new LinearLayout(context);
        ballLayout.setOrientation(LinearLayout.HORIZONTAL);
        FParams = new FrameLayout.LayoutParams(WC, WC);
        FParams.gravity = Gravity.BOTTOM;
        addView(ballLayout, FParams);

        ball = new ImageView(context);
        LinearLayout.LayoutParams lParams = new LinearLayout.LayoutParams(WC, WC);
        ball.setImageResource(R.drawable.switch_thumb);
        ballLayout.addView(ball, lParams);

        turnOff(0);

      //獲取控件寬高
        post(new Runnable() {
            @Override
            public void run() {
                mWidth = getWidth();
                mHeight = getHeight();
                mBallWidth =ball.getWidth();
                Log.d(TAG, "onGlobalLayout: width = " + mWidth + "   height = " + mHeight+" mBallWidth:"+mBallWidth);
            }
        });

        setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                if (local_state == 0) {
                    turnOn(200);
                } else {
                    turnOff(200);
                }
                if (onSwitchClickListener != null) {
                    onSwitchClickListener.onClickSwitch(local_state);
                }

            }
        });

    }



    /**
     * 需要詳細配置請看 {@link #turnOn(int time)}
     * 打開按鈕佑吝,帶漸變動畫
     *
     * @param time 漸變時間
     *             0 馬上設(shè)置成打開狀態(tài)
     */
    private void turnOn(int time) {
//        ballLayout.setGravity(Gravity.RIGHT);
        switchBg.setImageDrawable(getResources().getDrawable(android.R.color.transparent));
//        ball.setPadding(getRealPixel(80), 0, 0, 0);
//        switchBg.setBackgroundResource(R.drawable.switch_on_bg);
        startAnimator(time,0,mWidth-mBallWidth,ball);
        local_state = 1;

        TransitionDrawable td;

        if (time == 0) {
            td = new TransitionDrawable(new Drawable[]{bg_on, bg_on});
        } else {
            td = new TransitionDrawable(new Drawable[]{getResources().getDrawable(android.R.color.transparent), bg_on});
        }

        td.setCrossFadeEnabled(true);
        td.startTransition(time);
        switchBg.setImageDrawable(td);

    }


    /**
     * 需要詳細配置請看 {@link #turnOff(int time)}
     * 打開按鈕唱歧,帶漸變動畫
     *
     * @param time 漸變時間
     *             0 馬上設(shè)置成關(guān)閉狀態(tài)
     */
    private void turnOff(int time) {
//        ballLayout.setGravity(Gravity.RIGHT);
        switchBg.setImageDrawable(getResources().getDrawable(android.R.color.transparent));
//        ball.setPadding(0, 0, getRealPixel(80), 0);
//        switchBg.setBackgroundResource(R.drawable.switch_off_bg);
        startAnimator(time,mWidth-mBallWidth,0,ball);
        local_state = 0;
        TransitionDrawable td;

        if (time == 0) {
            td = new TransitionDrawable(new Drawable[]{getResources().getDrawable(android.R.color.transparent), getResources().getDrawable(android.R.color.transparent)});
        } else {
            td = new TransitionDrawable(new Drawable[]{bg_on, getResources().getDrawable(android.R.color.transparent)});
        }


        td.setCrossFadeEnabled(true);
        td.startTransition(time);
        switchBg.setImageDrawable(td);

    }

    //以漸變效果岖免,在200ms內(nèi)打開開關(guān)
    public void transitionOn() {
        turnOn(200);
    }

    //以漸變效果,在200ms內(nèi)關(guān)閉開關(guān)
    public void transitionOff() {
        turnOff(200);
    }

    //無漸變效果合愈,強制打開開關(guān)
    public void switchOn() {
        turnOn(0);
    }

    //無漸變效果叮贩,強制關(guān)閉開關(guān)
    public void switchOff() {
        turnOff(0);
    }

    //獲取開關(guān)本地狀態(tài)
    public int getLocalState() {
        return this.local_state;
    }

    //點擊時回傳switch 狀態(tài),0為關(guān)閉,1為打開
    public interface OnSwitchClickListener {
        void onClickSwitch(int local_state);
    }

    private OnSwitchClickListener onSwitchClickListener;

    public void setOnSwitchClickListener(OnSwitchClickListener l) {
        this.onSwitchClickListener = l;
    }



    private void startAnimator(int time, final int from, final int to, final View v) {
        final ValueAnimator animator = ValueAnimator.ofInt(from, to);
        animator.setDuration(time);
        animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                /**
                 * 通過這樣一個監(jiān)聽事件想暗,我們就可以獲取
                 * 到ValueAnimator每一步所產(chǎn)生的值妇汗。
                 *
                 * 通過調(diào)用getAnimatedValue()獲取到每個時間因子所產(chǎn)生的Value。
                 * */
                Integer value = (Integer) animation.getAnimatedValue();
                v.setPadding(value, 0, 0, 0);


            }
        });
        animator.start();
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末说莫,一起剝皮案震驚了整個濱河市杨箭,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌储狭,老刑警劉巖互婿,帶你破解...
    沈念sama閱讀 212,599評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件捣郊,死亡現(xiàn)場離奇詭異,居然都是意外死亡慈参,警方通過查閱死者的電腦和手機呛牲,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,629評論 3 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來驮配,“玉大人娘扩,你說我怎么就攤上這事∽扯停” “怎么了琐旁?”我有些...
    開封第一講書人閱讀 158,084評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長猜绣。 經(jīng)常有香客問我灰殴,道長,這世上最難降的妖魔是什么掰邢? 我笑而不...
    開封第一講書人閱讀 56,708評論 1 284
  • 正文 為了忘掉前任牺陶,我火速辦了婚禮,結(jié)果婚禮上辣之,老公的妹妹穿的比我還像新娘掰伸。我一直安慰自己,他們只是感情好怀估,可當(dāng)我...
    茶點故事閱讀 65,813評論 6 386
  • 文/花漫 我一把揭開白布碱工。 她就那樣靜靜地躺著,像睡著了一般奏夫。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上历筝,一...
    開封第一講書人閱讀 50,021評論 1 291
  • 那天酗昼,我揣著相機與錄音,去河邊找鬼梳猪。 笑死麻削,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的春弥。 我是一名探鬼主播呛哟,決...
    沈念sama閱讀 39,120評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼匿沛!你這毒婦竟也來了扫责?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,866評論 0 268
  • 序言:老撾萬榮一對情侶失蹤逃呼,失蹤者是張志新(化名)和其女友劉穎鳖孤,沒想到半個月后者娱,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,308評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡苏揣,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,633評論 2 327
  • 正文 我和宋清朗相戀三年黄鳍,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片平匈。...
    茶點故事閱讀 38,768評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡框沟,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出增炭,到底是詐尸還是另有隱情忍燥,我是刑警寧澤,帶...
    沈念sama閱讀 34,461評論 4 333
  • 正文 年R本政府宣布弟跑,位于F島的核電站灾前,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏孟辑。R本人自食惡果不足惜哎甲,卻給世界環(huán)境...
    茶點故事閱讀 40,094評論 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望饲嗽。 院中可真熱鬧炭玫,春花似錦、人聲如沸貌虾。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,850評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽尽狠。三九已至衔憨,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間袄膏,已是汗流浹背践图。 一陣腳步聲響...
    開封第一講書人閱讀 32,082評論 1 267
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留沉馆,地道東北人码党。 一個月前我還...
    沈念sama閱讀 46,571評論 2 362
  • 正文 我出身青樓,卻偏偏與公主長得像斥黑,于是被迫代替她去往敵國和親揖盘。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,666評論 2 350

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