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;
      }
    });
  }
  /**
   * 计算处于true或是false区域, 并做改变处理
   */
  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);
    }
  }
  /**
   * 改变checked, 根据checked移动游标
   * @param isChecked
   */
  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);
    }
  }
  /**
   * layout游标
   */
  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);
  }
  /**
   * isChecked值改变监听器
   */
  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程序设计有所帮助。

(0)

相关推荐

  • Android实现中国象棋附源码下载

    象棋,很多人多接触过,学者写了一个,大神可以指点一下~直接上代码: 贴出主要代码,想要Demo的点击下载:中国象棋Demo package wyf.ytl; import android.content.Context; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Canvas; import android.graphics.Color; impor

  • Android RecyclerView的Item自定义动画及DefaultItemAnimator源码分析

    这是关于RecyclerView的第二篇,说的是如何自定义Item动画,但是请注意,本文不包含动画的具体实现方法,只是告诉大家如何去自定义动画,如何去参考源代码. 我们知道,RecyclerView默认会使用DefaultItemAnimator,所以如果我们需要自定义动画,那么应该好好的读读这个类的源代码,这样不仅仅是学习怎么自定义,还要学习Android的设计模式. 先弄明白一件事,DefaultItemAnimator继承自SimpleItemAnimator,SimpleItemAnim

  • Android内核源码 在Ubuntu上下载,编译,安装

    从源代码树下载下来的最新Android源代码,是不包括内核代码的,也就是Android源代码工程默认不包含Linux Kernel代码,而是使用预先编译好的内核,也就是prebuilt/android-arm/kernel/kernel-qemu文件.那么,如何才能DIY自己的内核呢?这篇文章一一道来. 一. 首选,参照前一篇在Android源码 在Ubuntu上下载,编译和安装准备好Android源代码目录.       二. 下载Linux Kernel for Android源代码. 1.

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

    本文实例讲述了Android实现软件列表的点击启动另外一个程序功能.分享给大家供大家参考,具体如下: 目前面世的许多软件中有这么一个功能:设备中安装了哪些软件,他们会以一个软件列表清单的形式向用户展示出来. 今天我们就来实现这一功能: 运行环境: motorola defy+ 系统2.3.6 主要 API : PackageInfo,PackageManager,LayoutInflater,ApplicationInfo PackageManger类,它的主要职责是管理应用程序包. 通过它,我

  • 从源码分析Android的Glide库的图片加载流程及特点

    0.基础知识 Glide中有一部分单词,我不知道用什么中文可以确切的表达出含义,用英文单词可能在行文中更加合适,还有一些词在Glide中有特别的含义,我理解的可能也不深入,这里先记录一下. (1)View: 一般情况下,指Android中的View及其子类控件(包括自定义的),尤其指ImageView.这些控件可在上面绘制Drawable (2)Target: Glide中重要的概念,目标.它即可以指封装了一个View的Target(ViewTarget),也可以不包含View(SimpleTa

  • Android Volley框架使用源码分享

    过去在Android上网络通信都是使用的Xutils 因为用它可以顺道处理了图片和网络这两个方面,后来发觉Xutils里面使用的是HttpClient  而Google在6.0的版本上已经把HttpClient废除了,所以开始寻找新的网络框架,okhttp也用过,但是它是在作用在UI线程,使用起来还需要用handler 所以就先用着Volley框架了.  这里我先分析下Volley框架的简单网络请求的源码. 使用Volley请求网络数据的简单过程: RequestQueue queue = Vo

  • Android 日志系统Logger源代码详细介绍

    我们知道,在Android系统中,提供了一个轻量级的日志系统,这个日志系统是以驱动程序的形式实现在内核空间的,而在用户空间分别提供了Java接口和C/C++接口来使用这个日志系统,取决于你编写的是Android应用程序还是系统组件.在前面的文章浅谈Android系统开发中LOG的使用中,已经简要地介绍了在Android应用程序开发中Log的使用方法,在这一篇文章中,我们将更进一步地分析Logger驱动程序的源代码,使得我们对Android日志系统有一个深刻的认识. 既然Android 日志系统是

  • 详解Android中用于线程处理的AsyncTask类的用法及源码

    为什么要用AsyncTask 我们写App都有一个原则,主线程不能够运行需要占用大量CPU时间片的任务,如大量复杂的浮点运算,较大的磁盘IO操作,网络socket等,这些都会导致我们的主线程对用户的响应变得迟钝,甚至ANR,这些会使应用的用户体验变差,但是有时又的确需要执行这些耗时的任务,那么我们通常可以使用AsyncTask或者new Thread 来处理,这样把任务放入工作线程中执行,不会占用主线程的时间片,所以主线程会及时响应用户的操作,如果使用new Thread来执行任务,那么如果需要

  • Android源码 在Ubuntu上下载,编译和安装

    看完了前面说的几本书之后,对Linux Kernel和Android有一定的认识了,是不是心里蠢蠢欲动,想小试牛刀自己编译一把Android源代码了呢?一直习惯使用Windows系统,而Android源代码是不支持在Windows上编译上,于是决定使用虚拟机安装Ubuntu,然后下载.编译和安装Android源代码.      一. 环境准备. 1. 磁盘空间预留20G左右,内存3G,因为一边要跑主机,一边要跑虚拟机,内存要求还是比较高的,这样才会比较流畅. 2. 安装VMWare 7.1.4.

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

    本文实例讲述了Android编程实现可滑动的开关效果.分享给大家供大家参考,具体如下: 闲着没事,把之前写的一个Demo放上来分享下.就是一个开关,实现可滑动和动画效果.不是图片切换. 好了,先上图: 完整实例代码点击此处本站下载. 直接把自定义的这个View代码放上来,有注释应该很好理解: 首先是布局: <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android=&qu

  • Android编程自定义搜索框实现方法【附demo源码下载】

    本文实例讲述了Android编程自定义搜索框实现方法.分享给大家供大家参考,具体如下: 先来看效果图吧~ 分析:这只是模拟了一个静态数据的删除与显示 用EditText+PopupWindow+listView实现的 步骤: 1.先写出搜索框来-activity_mian布局: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://sc

  • Android编程基于Contacts读取联系人的方法(附demo源码)

    本文实例讲述了Android编程基于Contacts读取联系人的方法.分享给大家供大家参考,具体如下: Android Contacts简介: 这里介绍安卓通讯录数据库.包括Android使用Contacts访问SQLite的基本知识,并了解Android SQLite和Contacts的更多信息.谷歌改变了从版本1到版本2的Contacts数据库.下面加以简单介绍. Contacts 读取代码: package com.homer.phone; import java.util.ArrayLi

  • HTML5游戏引擎LTweenLite实现的超帅动画效果(附demo源码下载)

    本文实例讲述了HTML5游戏引擎LTweenLite实现的超帅动画效果.分享给大家供大家参考,具体如下: lufylegend.js是一个开源的HTML5游戏引擎,在游戏中往往会有各种的动画,这些动画有些是flash文件,有些是视频文件,本次就来利用lufylegend制作一个帅气的游戏动画,如下图. 测试连接如下: http://lufylegend.com/demo/effects01/ 一.准备工作 准备工作当然就是引擎的下载了. lufylegend.js引擎官网 http://lufy

  • HTML5+jQuery插件Quicksand实现超酷的星际争霸2兵种分类展示效果(附demo源码下载)

    本文讲述了HTML5+jQuery插件Quicksand实现超酷的星际争霸2兵种分类展示效果.分享给大家供大家参考,具体如下: 因为本人是星际争霸系列游戏的忠实拥簇,所以在今天的jQuery教程中,我们将使用HTML5和jQuery插件Quicksand来创建一个超酷的星际争霸兵种效果图.希望大家喜欢! 先来看看效果图: HTML5代码 首先我们使用HTML5的代码来创建一个html文档,将所需的quicksand类库,及其jquery类库,还有HTML5类库倒入,如下: <!DOCTYPE h

  • Asp.net(C#)读取数据库并生成JS文件制作首页图片切换效果(附demo源码下载)

    本文实例讲述了Asp.net(C#)读取数据库并生成JS文件制作首页图片切换效果的方法.分享给大家供大家参考,具体如下: using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.UI; using System.Web.UI.WebControls; using System.Text; using System.IO; public partial

  • jquery插件jquery.LightBox.js实现点击放大图片并左右点击切换效果(附demo源码下载)

    本文实例讲述了jquery插件jquery.LightBox.js实现点击放大图片并左右点击切换效果.分享给大家供大家参考,具体如下: 该插件乃文章作者所写,目的在于提升作者的js能力,也给一些js菜鸟在使用插件时提供一些便利,老鸟就悠然地飞过吧. 此插件旨在实现目前较为流行的点击放大图片并左右点击切换图片的效果,您可以根据自己的实际需求来设置是否添加左右切换图片的效果.整体代码如下: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transit

  • jQuery实现横向带缓冲的水平运动效果(附demo源码下载)

    本文实例讲述了jQuery实现横向带缓冲的水平运动效果.分享给大家供大家参考,具体如下: 这里使用jQuery生成横向带缓冲的水平运动,用鼠标点一下才能激活,点一下以后可看到Div层在做水平运动,由此可派生出诸多的其它形式的动画效果. 点击此处查看在线演示效果. 具体代码如下: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/x

  • jQuery+css3实现转动的正方形效果(附demo源码下载)

    本文实例讲述了jQuery+css3实现转动的正方形效果.分享给大家供大家参考,具体如下: 主要是应用到了css3中的rotate来控制旋转角度 运行效果截图如下: 点击此处查看在线演示效果. 具体代码如下: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> &l

  • jQuery实现可以控制图片旋转角度效果(附demo源码下载)

    本文实例讲述了jQuery实现可以控制图片旋转角度效果.分享给大家供大家参考,具体如下: 运行效果截图如下: 点击此处查看在线演示效果. 具体代码如下: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://ww

随机推荐