我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Android编程实现可滑动的开关效果(附demo源码下载)

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Android编程实现可滑动的开关效果(附demo源码下载)

本文实例讲述了Android编程实现可滑动的开关效果。分享给大家供大家参考,具体如下:

闲着没事,把之前写的一个Demo放上来分享下。就是一个开关,实现可滑动和动画效果。不是图片切换。

好了,先上图:

完整实例代码点击此处本站下载。

直接把自定义的这个View代码放上来,有注释应该很好理解:
首先是布局:


<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:id="@+id/sv_container"
  android:layout_width="230dip"
  android:layout_height="38dip"
  android:background="@drawable/usage_list_dark" >
  <ImageView
    android:id="@+id/iv_switch_cursor"
    android:layout_width="120dip"
    android:layout_height="36dip"
    android:layout_centerVertical="true"
    android:layout_marginLeft="0.5dip"
    android:layout_marginRight="0.5dip"
    android:background="@drawable/usage_list_green" />
  <LinearLayout
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:gravity="center" >
    <TextView
      android:id="@+id/switch_text_true"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:layout_weight="1"
      android:gravity="center"
      android:text="开" />
    <TextView
      android:id="@+id/switch_text_false"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:layout_weight="1"
      android:gravity="center"
      android:text="关" />
  </LinearLayout>
</RelativeLayout>

接着是这个View的代码,继承自LinearLayout :


package com.lxb.switchdemo;
import android.content.Context;
import android.graphics.Color;
import android.os.Handler;
import android.os.Message;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.animation.Animation;
import android.view.animation.Animation.AnimationListener;
import android.view.animation.LinearInterpolator;
import android.view.animation.TranslateAnimation;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.RelativeLayout;
import android.widget.TextView;
public class SwitchView extends LinearLayout implements OnClickListener {
  private static final int FLAG_MOVE_TRUE = 1; // 向左滑动标识
  private static final int FLAG_MOVE_FALSE = 2; // 向右滑动标识
  private static final int HANDLE_LAYOUT_CURSOR = 100; // 处理调用开关的layout方法
  private Context context; // 上下文对象
  private RelativeLayout sv_container; // SwitchView的外层Layout
  private ImageView iv_switch_cursor; // 开关邮标的ImageView
  private TextView switch_text_true; // true的文字信息控件
  private TextView switch_text_false; // false的文字信息控件
  private boolean isChecked = true; // 是否已开
  private boolean checkedChange = false; // isChecked是否有改变
  private OnCheckedChangeListener onCheckedChangeListener; // 用于监听isChecked是否有改变
  private int margin = 1; // 游标离边缘位置(这个值视图片而定, 主要是为了图片能显示正确)
  private int bg_left; // 背景左
  private int bg_right; // 背景右
  private int cursor_left; // 游标左部
  private int cursor_top; // 游标顶部
  private int cursor_right; // 游标右部
  private int cursor_bottom; // 游标底部
  private Animation animation; // 移动动画
  private int currentFlag = FLAG_MOVE_TRUE; // 当前移动方向flag
  public SwitchView(Context context) {
    super(context);
    this.context = context;
    initView();
  }
  @Override
  protected void onLayout(boolean changed, int l, int t, int r, int b) {
    super.onLayout(changed, l, t, r, b);
    // 获取所需要的值
    bg_left = sv_container.getLeft();
    bg_right = sv_container.getRight();
    cursor_left = iv_switch_cursor.getLeft();
    cursor_top = iv_switch_cursor.getTop();
    cursor_right = iv_switch_cursor.getRight();
    cursor_bottom = iv_switch_cursor.getBottom();
  }
  private Handler mHandler = new Handler() {
    @Override
    public void handleMessage(Message msg) {
      switch(msg.what) {
      case HANDLE_LAYOUT_CURSOR:
        iv_switch_cursor.layout(cursor_left, cursor_top, cursor_right, cursor_bottom);
        break;
      }
    }
  };
  public void onClick(View v) {
    // 控件点击时触发改变checked值
    if(v == this) {
      changeChecked(!isChecked);
    }
  }
  
  private void initView() {
    LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    View view = inflater.inflate(R.layout.switch_view, this);
    view.setOnClickListener(this);
    sv_container = (RelativeLayout) view.findViewById(R.id.sv_container);
    switch_text_true = (TextView) view.findViewById(R.id.switch_text_true);
    switch_text_false = (TextView) view.findViewById(R.id.switch_text_false);
    changeTextColor();
    iv_switch_cursor = (ImageView) view.findViewById(R.id.iv_switch_cursor);
    iv_switch_cursor.setClickable(false);
    iv_switch_cursor.setOnTouchListener(new OnTouchListener() {
      int lastX; // 最后的X坐标
      public boolean onTouch(View v, MotionEvent event) {
        switch(event.getAction()) {
        case MotionEvent.ACTION_DOWN:
          lastX = (int) event.getRawX();
          cursor_left = v.getLeft();
          cursor_top = v.getTop();
          cursor_right = v.getRight();
          cursor_bottom = v.getBottom();
          break;
        case MotionEvent.ACTION_MOVE:
          int dx = (int) event.getRawX() - lastX;
          cursor_left = v.getLeft() + dx;
          cursor_right = v.getRight() + dx;
          // 超出边界处理
          if(cursor_left <= bg_left + margin) {
            cursor_left = bg_left + margin;
            cursor_right = cursor_left + v.getWidth();
          }
          if(cursor_right >= bg_right - margin) {
            cursor_right = bg_right - margin;
            cursor_left = cursor_right - v.getWidth();
          }
          v.layout(cursor_left, cursor_top, cursor_right, cursor_bottom);
          lastX = (int) event.getRawX();
          break;
        case MotionEvent.ACTION_UP:
          calculateIscheck();
          break;
        }
        return true;
      }
    });
  }
  
  private void calculateIscheck() {
    float center = (float) ((bg_right - bg_left) / 2.0);
    float cursor_center = (float) ((cursor_right - cursor_left) / 2.0);
    if(cursor_left + cursor_center <= center) {
      changeChecked(true);
    } else {
      changeChecked(false);
    }
  }
  
  private void changeChecked(boolean isChecked) {
    if(this.isChecked != isChecked) {
      checkedChange = true;
    } else {
      checkedChange = false;
    }
    if(isChecked) {
      currentFlag = FLAG_MOVE_TRUE;
    } else {
      currentFlag = FLAG_MOVE_FALSE;
    }
    cursorMove();
  }
  
  private void cursorMove() {
    // 这里说明一点, 动画本可设置animation.setFillAfter(true)
    // 令动画进行完后停在最后位置. 但这里使用这样方式的话.
    // 再次拖动图片会出现异常(具体原因我没找到)
    // 所以最后只能使用onAnimationEnd回调方式再layout游标
    animation = null;
    final int toX;
    if(currentFlag == FLAG_MOVE_TRUE) {
      toX = cursor_left - bg_left - margin;
      animation = new TranslateAnimation(0, -toX, 0, 0);
    } else {
      toX = bg_right - margin - cursor_right;
      animation = new TranslateAnimation(0, toX, 0, 0);
    }
    animation.setDuration(100);
    animation.setInterpolator(new LinearInterpolator());
    animation.setAnimationListener(new AnimationListener() {
      public void onAnimationStart(Animation animation) {
      }
      public void onAnimationRepeat(Animation animation) {
      }
      public void onAnimationEnd(Animation animation) {
        // 计算动画完成后游标应在的位置
        if(currentFlag == FLAG_MOVE_TRUE) {
          cursor_left -= toX;
          cursor_right = cursor_left + iv_switch_cursor.getWidth();
        } else {
          cursor_right = bg_right - margin;
          cursor_left = cursor_right - iv_switch_cursor.getWidth();
        }
        // 这里不能马上layout游标正确位置, 否则会有一点点闪屏
        // 为了美观, 这里迟了一点点调用layout方法, 便不会闪屏
        mHandler.sendEmptyMessageDelayed(HANDLE_LAYOUT_CURSOR, 5);
        // 这里是根据是不是改变了isChecked值进行一些操作
        if(checkedChange) {
          isChecked = !isChecked;
          if(onCheckedChangeListener != null) {
            onCheckedChangeListener.onCheckedChanged(isChecked);
          }
          changeTextColor();
        }
      }
    });
    iv_switch_cursor.startAnimation(animation);
  }
  
  private void changeTextColor() {
    if(isChecked) {
      switch_text_true.setTextColor(Color.WHITE);
      switch_text_false.setTextColor(Color.GRAY);
    } else {
      switch_text_true.setTextColor(Color.GRAY);
      switch_text_false.setTextColor(Color.WHITE);
    }
  }
  
  private void layoutCursor() {
    if(isChecked) {
      cursor_left = bg_left + margin;
      cursor_right = bg_left + margin + iv_switch_cursor.getWidth();
    } else {
      cursor_left = bg_right - margin - iv_switch_cursor.getWidth();
      cursor_right = bg_right - margin;
    }
    iv_switch_cursor.layout(cursor_left, cursor_top, cursor_right, cursor_bottom);
  }
  
  public interface OnCheckedChangeListener {
    void onCheckedChanged(boolean isChecked);
  }
  public boolean isChecked() {
    return isChecked;
  }
  public void setChecked(boolean isChecked) {
    if(this.isChecked != isChecked) {
      this.isChecked = isChecked;
      if(onCheckedChangeListener != null) {
        onCheckedChangeListener.onCheckedChanged(isChecked);
      }
      layoutCursor();
    }
  }
  public void setOnCheckedChangeListener(
      OnCheckedChangeListener onCheckedChangeListener) {
    this.onCheckedChangeListener = onCheckedChangeListener;
  }
}

最后是Activity使用这个View:


package com.lxb.switchdemo;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.LinearLayout;
import android.widget.TextView;
import android.widget.Toast;
import com.lxb.switchdemo.SwitchView.OnCheckedChangeListener;
public class Switch_demoActivity extends Activity implements OnClickListener {
  private LinearLayout layout;
  private TextView tv_showcheck;
  private SwitchView sv;
  private Button btn_set_true;
  private Button btn_set_false;
  private Button btn_getstate;
  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    layout = (LinearLayout) findViewById(R.id.layout);
    tv_showcheck = (TextView) findViewById(R.id.tv_showcheck);
    sv = new SwitchView(this);
    tv_showcheck.setText("当前状态: " + getState(sv.isChecked()));
    sv.setOnCheckedChangeListener(new OnCheckedChangeListener() {
      public void onCheckedChanged(boolean isChecked) {
        tv_showcheck.setText("当前状态: " + getState(isChecked));
      }
    });
    layout.addView(sv);
    btn_set_true = (Button) findViewById(R.id.btn_set_true);
    btn_set_false = (Button) findViewById(R.id.btn_set_false);
    btn_getstate = (Button) findViewById(R.id.btn_getstate);
    btn_set_true.setOnClickListener(this);
    btn_set_false.setOnClickListener(this);
    btn_getstate.setOnClickListener(this);
  }
  public void onClick(View v) {
    switch(v.getId()) {
    case R.id.btn_set_true:
      sv.setChecked(true);
      break;
    case R.id.btn_set_false:
      sv.setChecked(false);
      break;
    case R.id.btn_getstate:
      Toast.makeText(Switch_demoActivity.this,
          sv.isChecked() + "", Toast.LENGTH_SHORT).show();
      break;
    }
  }
  private String getState(boolean state) {
    if(state) {
      return "开";
    }
    return "关";
  }
}

实现起来还是很简单的,主要还是坐标什么的需要计算与调整。

当然可能还会有一些BUG存在,有需要的可以下下来自行修改,也可以和我讨论。

更多关于Android相关内容感兴趣的读者可查看本站专题:《Android通信方式总结》、《Android调试技巧与常见问题解决方法汇总》、《Android开发入门与进阶教程》、《Android多媒体操作技巧汇总(音频,视频,录音等)》、《Android基本组件用法总结》、《Android视图View技巧总结》、《Android布局layout技巧总结》及《Android控件用法总结》

希望本文所述对大家Android程序设计有所帮助。

您可能感兴趣的文章:Android内核源码 在Ubuntu上下载,编译,安装Android源码 在Ubuntu上下载,编译和安装Android RecyclerView的Item自定义动画及DefaultItemAnimator源码分析Android实现软件列表的点击启动另外一个程序功能【附demo源码下载】Android实现中国象棋附源码下载Android Volley框架使用源码分享详解Android中用于线程处理的AsyncTask类的用法及源码从源码分析Android的Glide库的图片加载流程及特点Android 日志系统Logger源代码详细介绍


免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Android编程实现可滑动的开关效果(附demo源码下载)

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android编程实现可滑动的开关效果(附demo源码下载)

本文实例讲述了Android编程实现可滑动的开关效果。分享给大家供大家参考,具体如下: 闲着没事,把之前写的一个Demo放上来分享下。就是一个开关,实现可滑动和动画效果。不是图片切换。 好了,先上图:完整实例代码点击此处本站下载。 直接把自
2022-06-06

Android编程滑动效果之倒影效果实现方法(附demo源码下载)

本文实例讲述了Android编程滑动效果之倒影效果实现方法。分享给大家供大家参考,具体如下: 前面介绍了使用《Android编程实现3D滑动旋转效果的方法》,现在介绍图片倒影实现,先看效果图这里主要通过自定义Gallery和ImageAda
2022-06-06

Android编程滑动效果之Gallery+GridView实现图片预览功能(附demo源码下载)

本文实例讲述了Android编程滑动效果之Gallery+GridView实现图片预览功能。分享给大家供大家参考,具体如下: Android系统自带一个GridView和Gallery两个控件,GridView网格显示,Gallery单个浏
2022-06-06

Android TreeView效果实现方法(附demo源码下载)

本文实例讲述了Android TreeView效果实现方法。分享给大家供大家参考,具体如下: 应该说很多的操作系统上面都提供了TreeView空间,实现树形结构,这个树形结构的应用时很广泛的,而Google开发Android的时候出于And
2022-06-06

Android编程实现简易弹幕效果示例【附demo源码下载】

本文实例讲述了Android编程实现简易弹幕效果。分享给大家供大家参考,具体如下: 首先上效果图,类似于360检测到骚扰电话页面:布局很简单,上面是一个RelativeLayout,下面一个Button. 功能: (1)弹幕生成后自动从右侧
2022-06-06

Android编程实现仿易信精美弹出框效果【附demo源码下载】

本文实例讲述了Android编程实现仿易信精美弹出框效果。分享给大家供大家参考,具体如下: 截图:动画效果介绍: 1.点击ActionBar上“+”按钮,菜单从上方弹出(带反弹效果); 2.再次点击“+”、点击空白区域或者点击返回键,菜单向
2022-06-06

Android开发进阶自定义控件之滑动开关实现方法【附demo源码下载】

本文实例讲述了Android开发进阶自定义控件之滑动开关实现方法。分享给大家供大家参考,具体如下: 自定义开关控件Android自定义控件一般有三种方式 1、继承Android固有的控件,在Android原生控件的基础上,进行添加功能和逻辑
2022-06-06

Android编程实现TextView垂直自动滚动功能【附demo源码下载】

本文实例讲述了Android编程实现TextView垂直自动滚动功能。分享给大家供大家参考,具体如下: 在做android 应用的开发的时候,横向滚动或者要做出跑马灯的效果很简单,textview本身的属性就支持,只要设置准确就会滚动,开发
2022-06-06

Android实现基于滑动的SQLite数据分页加载技术(附demo源码下载)

本文实例讲述了Android实现基于滑动的SQLite数据分页加载技术。分享给大家供大家参考,具体如下: main.xml如下:2022-06-06

Android编程实现仿优酷旋转菜单效果(附demo源码)

本文实例讲述了Android编程实现仿优酷旋转菜单效果。分享给大家供大家参考,具体如下: 首先,看下效果:不好意思,不会制作动态图片,只好上传静态的了,如果谁会,请教教我吧。 首先,看下xml文件:
2022-06-06

Android仿zaker用手向上推动的特效开发【推动门效果】(附demo源码下载)

本文实例讲述了Android仿zaker用手向上推动的特效开发。分享给大家供大家参考,具体如下: 最近在商店下载了zaker ,闲暇时拿来看看新闻!发现每次打开软件进入主界面时有个界面,需要你把它往上滑到一定距离才能进入到主界面。每次进入软
2022-06-06

Android编程实现仿美团或淘宝的多级分类菜单效果示例【附demo源码下载】

本文实例讲述了Android编程实现仿美团或淘宝的多级分类菜单效果。分享给大家供大家参考,具体如下: 这里要实现的是诸如美团/淘宝/百度糯米 多级分类菜单效果。当分类数量非常多时可以考虑采用两级分类,而诸如美团这种表现方式是一个不错的选择。
2022-06-06

Android编程实现仿iphone抖动效果的方法(附源码)

本文实例讲述了Android编程实现仿iphone抖动效果的方法。分享给大家供大家参考,具体如下: 布局文件:
2022-06-06

Android编程实现仿QQ发表说说,上传照片及弹出框效果【附demo源码下载】

本文实例讲述了Android编程实现仿QQ发表说说,上传照片及弹出框效果。分享给大家供大家参考,具体如下: 代码很简单,主要就是几个动画而已,图标什么的就随便找了几个,效果图: 动画说明: 1.点击右上角按钮,菜单从顶部下拉弹出,同时背
2022-06-06

Android 利用ViewPager实现图片可以左右循环滑动效果附代码下载

首先给大家展示靓照,对效果图感兴趣的朋友可以继续往下阅读哦。ViewPager这个小demo实现的是可以左右循环滑动图片,下面带索引,滑到最后一页在往右滑动就要第一页,第一页往左滑动就到最后一页,上面是效果图,用美女图片是我一贯的作风,呵呵
2022-06-06

Android编程实现手绘及保存为图片的方法(附demo源码下载)

本文实例讲述了Android编程实现手绘及保存为图片的方法。分享给大家供大家参考,具体如下: 运行效果图预览:应 yzuo_08 要求做了此Demo,跟以前那个手写板Demo不同的是可以将画布的内容保存为图片。 附上关键代码: MainVi
2022-06-06

Android程序自动更新功能模块的实现方法【附完整demo源码下载】

本文实例讲述了Android程序自动更新功能模块的实现方法。分享给大家供大家参考,具体如下: 在程序启动的时候检测服务器上有没有对应版本更新,如果有更新,提示用户是否更新。 在程序启动的时候首先调用更新模块检测服务器上存放的版本号跟当前程序
2022-06-06

Android开发之自定义view实现通讯录列表A~Z字母提示效果【附demo源码下载】

本文实例讲述了Android开发之自定义view实现通讯录列表A~Z字母提示效果。分享给大家供大家参考,具体如下:开发工具:eclipse运行环境:htc G9 android2.3.3话不多说,先看效果图其实左右边的A~Z是一个自定义的V
2023-05-30

Android实现软件列表的点击启动另外一个程序功能【附demo源码下载】

本文实例讲述了Android实现软件列表的点击启动另外一个程序功能。分享给大家供大家参考,具体如下: 目前面世的许多软件中有这么一个功能:设备中安装了哪些软件,他们会以一个软件列表清单的形式向用户展示出来。 今天我们就来实现这一功能: 运行
2022-06-06

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录