Android自定义控件(一)——开关控件
Google 在 API 14 开始才新增了Switch 控件。
因此,我们可以选择自己封装一个Switch 。
效果如图:
View主要代码:
- public class SwitchView extends LinearLayout {
- private ImageView maskImage; // 开关遮盖图片
- private boolean open; // 开关当前状态
- private boolean isAninFinish = true; // 动画是否结束
- private float x; // 记录ACTION_DOWN时候的横坐标
- private boolean isChangedByTouch = false; // 是否在一次事件中已经切换过状态
- private OnSwitchChangeListener switchChangeListener; // 监控开关状态
- public interface OnSwitchChangeListener {
- void onSwitchChanged(boolean open);
- }
- public SwitchView(Context context, AttributeSet attrs) {
- super(context, attrs);
- init();
- }
- public SwitchView(Context context) {
- super(context);
- init();
- }
- private void init() {
- setBackgroundResource(R.drawable.switch_bg);
- maskImage = new ImageView(getContext());
- maskImage.setLayoutParams(new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));
- maskImage.setImageResource(R.drawable.switch_mask);
- addView(maskImage);
- }
- public boolean getSwitchStatus() {
- return open;
- }
- public void setSwitchStatus(boolean isOpen) {
- this.open = isOpen;
- if (isOpen) {
- setGravity(Gravity.RIGHT);
- } else {
- setGravity(Gravity.LEFT);
- }
- }
- @Override
- public boolean onTouchEvent(MotionEvent event) {
- switch (event.getAction()) {
- case MotionEvent.ACTION_DOWN: {
- x = event.getX();
- break;
- }
- case MotionEvent.ACTION_MOVE: {
- if (event.getX() - x > 5 && !open) { // 向右
- changeStatus();
- } else if (event.getX() - x < -5 && open) { // 向左
- changeStatus();
- }
- break;
- }
- case MotionEvent.ACTION_UP: {
- if (Math.abs(event.getX() - x) <= 5) {
- changeStatus();
- }
- isChangedByTouch = false;
- break;
- }
- case MotionEvent.ACTION_CANCEL: {
- isChangedByTouch = false;
- break;
- }
- }
- return true;
- }
- private void changeStatus() {
- if (isAninFinish && !isChangedByTouch) {
- isChangedByTouch = true;
- open = !open;
- isAninFinish = false;
- if (switchChangeListener != null) {
- switchChangeListener.onSwitchChanged(open);
- }
- changeOpenStatusWithAnim(open);
- }
- }
- private void changeOpenStatusWithAnim(boolean open) {
- if (open) {
- // 左到右
- Animation leftToRight = new TranslateAnimation(Animation.RELATIVE_TO_SELF, 0,
- Animation.ABSOLUTE, getWidth() - maskImage.getWidth(),
- Animation.RELATIVE_TO_SELF, 0,
- Animation.RELATIVE_TO_SELF, 0);
- leftToRight.setDuration(300);
- leftToRight.setAnimationListener(new AnimationListener() {
- @Override
- public void onAnimationStart(Animation animation) {
- }
- @Override
- public void onAnimationRepeat(Animation animation) {
- }
- @Override
- public void onAnimationEnd(Animation animation) {
- maskImage.clearAnimation();
- setGravity(Gravity.RIGHT);
- isAninFinish = true;
- }
- });
- maskImage.startAnimation(leftToRight);
- } else {
- // 右到左
- Animation rightToLeft = new TranslateAnimation(Animation.RELATIVE_TO_SELF, 0,
- Animation.ABSOLUTE, maskImage.getWidth() - getWidth(),
- Animation.RELATIVE_TO_SELF, 0,
- Animation.RELATIVE_TO_SELF, 0);
- rightToLeft.setDuration(300);
- rightToLeft.setAnimationListener(new AnimationListener() {
- @Override
- public void onAnimationStart(Animation animation) {
- }
- @Override
- public void onAnimationRepeat(Animation animation) {
- }
- @Override
- public void onAnimationEnd(Animation animation) {
- maskImage.clearAnimation();
- setGravity(Gravity.LEFT);
- isAninFinish = true;
- }
- });
- maskImage.startAnimation(rightToLeft);
- }
- }
- public OnSwitchChangeListener getSwitchChangeListener() {
- return switchChangeListener;
- }
- public void setOnSwitchChangeListener(OnSwitchChangeListener switchChangeListener) {
- this.switchChangeListener = switchChangeListener;
- }
- }
测试用代码:
- public class MainActivity extends Activity {
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- SwitchView switchView = (SwitchView) findViewById(R.id.switchview);
- switchView.setSwitchStatus(true);
- switchView.setOnSwitchChangeListener(new OnSwitchChangeListener() {
- @Override
- public void onSwitchChanged(boolean open) {
- Toast.makeText(MainActivity.this, "开关状态:" + open, Toast.LENGTH_SHORT).show();
- }
- });
- }
- }
测试用布局:
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:padding="20dp"
- >
- <TextView
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="自定义开关状态:"
- />
- <com.fancyy.switchview.SwitchView
- android:layout_marginLeft="20dp"
- android:id="@+id/switchview"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content" />
- </LinearLayout>
测试代码下载:http://download.csdn.net/detail/a105865708/6800519
Android自定义控件(一)——开关控件的更多相关文章
- Android自定义控件之日历控件
标签: android 控件 日历 应用 需求 2015年09月26日 22:21:54 25062人阅读 评论(109) 收藏 举报 分类: Android自定义控件系列(7) 版权声明:转载注 ...
- android自己定义开关控件
近日在android项目要使用开关控件.可是android中自带的开关控件不太惬意,所以就打算通过自己定义View写一个开关控件 ios的开关控件当然就是我要仿照的目标. 先上图: waterma ...
- Android自定义控件1--自定义控件介绍
Android控件基本介绍 Android本身提供了很多控件比如我们常用的有文本控件TextView和EditText:按钮控件Button和ImageButton状态开关按钮ToggleButton ...
- Android - 自定义控件之圆形控件
自定义控件 - 圈圈 Android L: Android Studio 效果:能够自定义圆圈半径和位置:设定点击效果:改变背景颜色 下面是demo图 点击前: 点击后: 自定义控件一般要继承View ...
- android 仿ios开关控件
ios一些控件还是挺美丽的,可是对android程序猿来说可能比較苦逼,由于ios一些看起来简单的效果对android来说可能就没那么简单了,可是没办法非常多产品都是拿ios的一些控件叫android ...
- Android开发技巧——自定义控件之组合控件
Android开发技巧--自定义控件之组合控件 我准备在接下来一段时间,写一系列有关Android自定义控件的博客,包括如何进行各种自定义,并分享一下我所知道的其中的技巧,注意点等. 还是那句老话,尽 ...
- Android常用酷炫控件(开源项目)github地址汇总
转载一个很牛逼的控件收集帖... 第一部分 个性化控件(View) 主要介绍那些不错个性化的 View,包括 ListView.ActionBar.Menu.ViewPager.Gallery.Gri ...
- Android 常用炫酷控件(开源项目)git地址汇总
第一部分 个性化控件(View) 主要介绍那些不错个性化的 View,包括 ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView.P ...
- MaterialEditText——Android Material Design EditText控件
MaterialEditText是Android Material Design EditText控件.可以定制浮动标签.主要颜色.默认的错误颜色等. 随着 Material Design 的到来, ...
- (转载)Android自定义标签列表控件LabelsView解析
Android自定义标签列表控件LabelsView解析 作者 donkingliang 关注 2017.03.15 20:59* 字数 759 阅读 406评论 0喜欢 3 无论是在移动端的App, ...
随机推荐
- CentOS NFS的安装配置、启动及mount挂载方法
一.环境介绍: 服务器:centos 192.168.1.225 客户端:centos 192.168.1.226 二.安装: NFS的安装配置:centos 5 : yum -y install n ...
- ubuntu安装python3.5
ubuntu14.04系统会自带python2.7,请不要卸载它.不同版本的Python可以共存在一个系统上. 卸载之后,桌面系统会被影响. (1)sudo add-apt-repository pp ...
- 广播接收者 BroadcastReceiver 示例-1
广播机制概述 Android广播分为两个方面:广播发送者和广播接收者,通常情况下,BroadcastReceiver指的就是广播接收者.广播作为Android组件间的通信方式,可以使用的场景如下: 1 ...
- sql练习总结(一)
最近在学sql,遇到了这么一道题: 写出一条Sql语句:取出表A中第31到第40记录(SQLServer,以自动增长的ID作为主键,注意:ID可能不是连续的. 把所能想到的实现方法都做了一遍: 1.用 ...
- Linux_x64安装Oracle11g(完整版)
一.修改操作系统核心参数 在Root用户下执行以下步骤: 1)修改用户的SHELL的限制,修改/etc/security/limits.conf文件 输入命令:vi /etc/security/lim ...
- C/C++中new关键字是否加括号的区别
代码: #include <iostream> using namespace std; class A{ public: int a; }; int main(){ A *a1 = ne ...
- javascript中的一些基本方法收藏
W3C DOM 什么是DOM,DOM其实就是把一个HTML或者XML等符合W3C标准的文档内容模拟成一个JAVA对象,这样才能给JAVA或者JS来操作.下面是JS中模拟出的内置DOM对象documen ...
- append与remove的简单使用
点击Add More按钮页面会自动添加一个输入框和Remove按钮,点击Remove按钮则此行元素将被移除. <!DOCTYPE html> <html lang="en& ...
- 浅谈PCB敷铜的“弊与利”
敷铜作为PCB设计的一个重要环节,不管是国产的青越锋PCB设计软件,还国外的一些Protel,PowerPCB都提供了智能敷铜功能,那么怎样才能敷好铜,我将自己一些想法与大家一起分享,希望能给同行带来 ...
- Powershell使用管道
管道并不是什么新事物,以前的Cmd控制台也有重定向的命令,例如Dir | More可以将结果分屏显示.传统的Cmd管道是基于文本的,但是Powershell是基于对象. PS> ls | Sor ...