自定义ios风格开
来源:互联网 发布:开票系统数据更新失败 编辑:程序博客网 时间:2024/05/29 14:11
-
-自定义iOS风格开关,开发中经常使用,废话不多说,看效果。
效果展示
-
-
-
-
view
-
SlideSwitchView.java
-
package com.example.administrator.myapplication.slideswitch;import android.animation.Animator;import android.animation.AnimatorListenerAdapter;import android.animation.ValueAnimator;import android.animation.ValueAnimator.AnimatorUpdateListener;import android.content.Context;import android.content.res.TypedArray;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;import com.example.administrator.myapplication.R;/** * Description: 自定义ios风格开关 * Copyright : Copyright (c) 2016 * Email : jusenr@163.com * Company : * Author : Jusenr * Date : 2016/7/23 0023 下午 9:00 */public class SlideSwitchView extends View { public static final int SHAPE_RECT = 1; public static final int SHAPE_CIRCLE = 2; private static final int RIM_SIZE = 6; private static final int DEFAULT_COLOR_THEME = Color.parseColor("#ff00ee00"); // 3 attributes private int color_theme; 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 max_left; private int min_left; private int frontRect_left; private int frontRect_left_begin = RIM_SIZE; private int eventStartX; private int eventLastX; private int diffX = 0; private boolean slideable = true; private SlideListener listener; public SlideSwitchView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); listener = null; paint = new Paint(); paint.setAntiAlias(true); TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.slideswitch); color_theme = a.getColor(R.styleable.slideswitch_themeColor, DEFAULT_COLOR_THEME); isOpen = a.getBoolean(R.styleable.slideswitch_isOpen, false); shape = a.getInt(R.styleable.slideswitch_shape, SHAPE_RECT); a.recycle(); } public SlideSwitchView(Context context, AttributeSet attrs) { this(context, attrs, 0); } public SlideSwitchView(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(); } public void initDrawingVal() { int width = getMeasuredWidth(); int height = getMeasuredHeight(); backCircleRect = new RectF(); frontCircleRect = new RectF(); frontRect = new Rect(); backRect = new Rect(0, 0, width, height); min_left = RIM_SIZE; if (shape == SHAPE_RECT) max_left = width / 2; else max_left = width - (height - 2 * RIM_SIZE) - RIM_SIZE; if (isOpen) { frontRect_left = max_left; alpha = 255; } else { frontRect_left = RIM_SIZE; alpha = 0; } frontRect_left_begin = frontRect_left; } public 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(Color.GRAY); canvas.drawRect(backRect, paint); paint.setColor(color_theme); paint.setAlpha(alpha); canvas.drawRect(backRect, paint); frontRect.set(frontRect_left, RIM_SIZE, frontRect_left + getMeasuredWidth() / 2 - RIM_SIZE, getMeasuredHeight() - RIM_SIZE); paint.setColor(Color.WHITE); canvas.drawRect(frontRect, paint); } else { // draw circle int radius; radius = backRect.height() / 2 - RIM_SIZE; paint.setColor(Color.GRAY); backCircleRect.set(backRect); canvas.drawRoundRect(backCircleRect, radius, radius, paint); paint.setColor(color_theme); paint.setAlpha(alpha); canvas.drawRoundRect(backCircleRect, radius, radius, paint); frontRect.set(frontRect_left, RIM_SIZE, frontRect_left + backRect.height() - 2 * RIM_SIZE, backRect.height() - RIM_SIZE); frontCircleRect.set(frontRect); paint.setColor(Color.WHITE); canvas.drawRoundRect(frontCircleRect, radius, radius, paint); } } @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 + frontRect_left_begin; tempX = (tempX > max_left ? max_left : tempX); tempX = (tempX < min_left ? min_left : tempX); if (tempX >= min_left && tempX <= max_left) { frontRect_left = tempX; alpha = (int) (255 * (float) tempX / (float) max_left); invalidateView(); } break; case MotionEvent.ACTION_UP: int wholeX = (int) (event.getRawX() - eventStartX); frontRect_left_begin = frontRect_left; boolean toRight; toRight = (frontRect_left_begin > max_left / 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.listener = listener; } public void moveToDest(final boolean toRight) { ValueAnimator toDestAnim = ValueAnimator.ofInt(frontRect_left, toRight ? max_left : min_left); toDestAnim.setDuration(500); toDestAnim.setInterpolator(new AccelerateDecelerateInterpolator()); toDestAnim.start(); toDestAnim.addUpdateListener(new AnimatorUpdateListener() { @Override public void onAnimationUpdate(ValueAnimator animation) { frontRect_left = (Integer) animation.getAnimatedValue(); alpha = (int) (255 * (float) frontRect_left / (float) max_left); invalidateView(); } }); toDestAnim.addListener(new AnimatorListenerAdapter() { @Override public void onAnimationEnd(Animator animation) { if (toRight) { isOpen = true; if (listener != null) listener.open(); frontRect_left_begin = max_left; } else { isOpen = false; if (listener != null) listener.close(); frontRect_left_begin = min_left; } } }); } public void setState(boolean isOpen) { this.isOpen = isOpen; initDrawingVal(); invalidateView(); if (listener != null) if (isOpen == true) { listener.open(); } else { listener.close(); } } 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; } public interface SlideListener { public void open(); public void close(); }}
activity
-
SlideSwitchActivity.java
-测试使用的activity
-通过实现SlideListener接口来监听开关状态,从而作相应的操作。
package com.example.administrator.myapplication.slideswitch;import android.os.Bundle;import android.support.v7.app.AppCompatActivity;import android.widget.Toast;import com.example.administrator.myapplication.R;/** * Description: 自定义ios风格开关Test * Copyright : Copyright (c) 2016 * Email : jusenr@163.com * Company : * Author : Jusenr * Date : 2016/7/23 0023 下午 9:00 */public class SlideSwitchActivity extends AppCompatActivity implements SlideSwitchView.SlideListener { public static boolean isOpen; private SlideSwitchView ss_switch; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_slide_switch); ss_switch = (SlideSwitchView) findViewById(R.id.ss_switch); //设置开关初始状态 ss_switch.setState(isOpen); } @Override public void open() { isOpen = true; ss_switch.setState(isOpen); Toast.makeText(this, "open", Toast.LENGTH_SHORT).show(); } @Override public void close() { isOpen = false; ss_switch.setState(isOpen); Toast.makeText(this, "close", Toast.LENGTH_SHORT).show(); }}
XML
-
activity_slide_switch.xml
-activity布局文件
-开关属性
slideswitch:shape=”circle” –设置开关的样式在这里有两种circle和rect
slideswitch:isOpen=”false” –设置开关状态,即OFF/ON
<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:slideswitch="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent" android:gravity="center_horizontal"> <LinearLayout android:layout_width="wrap_content" android:layout_marginTop="20dp" android:layout_height="wrap_content"> <!-- 开关 --> <com.example.administrator.myapplication.slideswitch.SlideSwitchView android:id="@+id/ss_switch" android:layout_width="70dip" android:layout_height="35dip" android:layout_alignParentRight="true" android:layout_centerVertical="true" android:layout_marginRight="20dp" slideswitch:isOpen="false" slideswitch:shape="circle" slideswitch:themeColor="#4CD964" /> </LinearLayout></LinearLayout>
-
arrays.xml
-自定义开关的相关属性设置
<?xml version="1.0" encoding="utf-8"?><resources> <!--自定义IOS开关属性--> <declare-styleable name="slideswitch"> <attr name="themeColor" format="color" /> <attr name="isOpen" format="boolean" /> <attr name="shape"> <enum name="rect" value="1" /> <enum name="circle" value="2" /> </attr> </declare-styleable></resources>
0 0
- 自定义ios风格开
- Android自定义iOS风格Dialog
- 在Android中自定义IOS风格的按钮
- Android 自定义View学习(3)--仿IOS风格滑动按钮
- Android自定义View-------IOS风格的滑动开关
- 自定义EditText实现类iOS风格搜索框
- 【Android自定义控件】仿IOS风格的搜索框
- 自定义EditText实现类iOS风格搜索框
- 【Android自定义控件】仿IOS风格的搜索框
- Android开篇之高仿ios风格的自定义Dialog
- ios之定制iOS 7中的导航栏和状态栏/颜色/风格/字体/个性自定义
- 自定义MyEclipse代码风格
- 自定义Android Toasts风格
- 自定义Android Toasts风格
- 自定义JTabbedPane 标签风格
- Android Gallery自定义风格
- DISCUZ风格自定义变量
- 自定义风格的SeekBar
- 二进制文件与文本文件的区别
- ngx_pool_t
- iOS 分享成功后不走回调方法
- Java毕业设计——基于Java web的停车场管理系统
- DLNA第一步UPnP协议栈
- 自定义ios风格开
- MDK5使用技巧
- 将一个long型的数字转化为字节数组(利用IO流)
- 为什么TCP连接需要三次握手?
- android studio 使用记录
- C#控制台 Trim 删除字符串首字母前面的空格与尾字母后面的空格
- HDU 5088 Revenge of Nim II(尼姆博弈,高斯消元)
- 求多边形的核(模板)
- UIfont字号与像素对应表