Andoird实现类似iphone AssistiveTouch的控件的demo
类似Iphone Assistive Touch的控件的实现
网上也有些这方面的控件,不过貌似不怎么好用,或者是论坛需要积分下载,恰好自己在项目中有用到这种控件,就打算自己写一个,也成功实现了这种功能。今天就打算把这个小控件分享到博客上,供大家参考学习。
下面贴出控件的代码:
package com.example.com.sus; import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.Context;
import android.graphics.Rect;
import android.util.AttributeSet;
import android.util.DisplayMetrics;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.Window;
import android.widget.RelativeLayout;
import android.widget.Scroller; /**
* file.test4.WSuspensionButton
*
* <p>as iphone AssistiveTouch</p>
* @author b
* @version 1.0
*
*/
public class WSuspensionButton extends View { private static final int SIZE = 50;
private int size;
private int sizeHalf;
private int stateHeight;
private Scroller mScroller;
private int screenWidth;
private int screenWidthHalf;
private int screenHeight;
private int screenHeightHalf; private int topLine;
private int leftLine;
private int rightLine;
private int bottomLine; public WSuspensionButton(Context context) {
this(context, null);
} public WSuspensionButton(Context context, AttributeSet attrs) {
super(context, attrs);
init();
} private void init() { recordX = 0;
recordY = 0;
stateHeight = 0;
mScroller = new Scroller(getContext());
DisplayMetrics dm = getContext().getResources().getDisplayMetrics();
size = (int)(SIZE * dm.density);
sizeHalf = size / 2;
screenWidth = dm.widthPixels;// 屏幕宽
screenHeight = dm.heightPixels - getStatusBarHeight();
screenWidthHalf = screenWidth / 2;
screenHeightHalf = screenHeight / 2 + stateHeight; // 紧紧只会左右飞
topLine = 0;
bottomLine = screenHeight - topLine;
} /**
* state bar height
* @return
*/
public int getStatusBarHeight() {// 状态栏高度
int result = 0;
int resourceId = getContext().getResources().getIdentifier("status_bar_height",
"dimen", "android");
if (resourceId > 0) {
result = getContext().getResources().getDimensionPixelSize(resourceId);
}
return result;
} public int getSize() {
return size;
} float recordX;
float recordY;
float tempX;
float tempY;
int recordLeft;
int recordTop;
int recordRight;
int recordBottom;
int state;
boolean isMoved;
@SuppressLint("ClickableViewAccessibility")
public boolean onTouchEvent(MotionEvent event) { switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
state = 0;
isMoved = false;
recordX = event.getRawX();
recordY = event.getRawY();
tempX = recordX;
tempY = recordY;
break;
case MotionEvent.ACTION_MOVE:
int intx = (int)(event.getRawX() - tempX);
int inty = (int)(event.getRawY() - tempY);
tempX = event.getRawX();
tempY = event.getRawY();
if(Math.abs(tempX - recordX) < 5 && Math.abs(tempY - recordY) < 5) {
if(state == 0) {
state = 1;
return true;
}
} else {
isMoved = true;
}
if(isMoved && state != 2)
state = 2;
if(state != 2) return true;
int left = getLeft() + intx;
int top = getTop() + inty;
int right = getRight() + intx;
int bottom = getBottom() + inty;
if(left < 0) {
left = 0;
right = getWidth();
}
if(right > screenWidth) {
left = screenWidth - getWidth();
right = screenWidth;
}
if(top < stateHeight) {
top = stateHeight;
bottom = stateHeight + getHeight();
}
if(bottom > screenHeight) {
top = screenHeight - getHeight();
bottom = screenHeight;
}
layout(left, top, right, bottom);
break;
case MotionEvent.ACTION_UP:
if(state != 2) {
if(clickedlistener != null)
clickedlistener.onClick(this);
return true;
}
recordLeft = getLeft();
recordTop = getTop();
recordRight = getRight();
recordBottom = getBottom();
Log.v("AC", "left:"+recordLeft+"\nright:"+recordRight+"\ntop:"+recordTop+"\nbottom:"+recordBottom);
if(getLeft() <= 0 || getTop() <= stateHeight ||
getRight() >= screenWidth || getBottom() >= screenHeight) {
RelativeLayout.LayoutParams params = (RelativeLayout.LayoutParams)getLayoutParams();
int leftX = getLeft();
int topX = getTop();
params.setMargins(getLeft(), getTop(), 0, 0);
setLayoutParams(params);
if(completeMoveListener != null) completeMoveListener.onCompleteMove(this, leftX, topX);
} else {
// 控件中心y,小于topline,向上飞
// 控件中心y,大于bottomLine,向下飞
// 控件中心x,小于屏幕可视中心x,向左飞
// 控件中心x,大于屏幕可视中心x,向右飞
if((getTop() + sizeHalf) <= topLine) {
scrollToTop();
} else if((getBottom() + sizeHalf) >= bottomLine) {
scrollToBottom();
} else if((getLeft() + sizeHalf) < screenWidthHalf) {
scrollToLeft();
} else {
scrollToRight();
}
}
break;
default:
break;
} return true;
} public void computeScroll() {
if(mScroller.computeScrollOffset()) {
layout(recordLeft + mScroller.getCurrX(),
recordTop + mScroller.getCurrY(),
recordRight + mScroller.getCurrX(),
recordBottom + mScroller.getCurrY());
postInvalidate();
if(mScroller.isFinished()) {
RelativeLayout.LayoutParams params = (RelativeLayout.LayoutParams)getLayoutParams();
int left = recordLeft + mScroller.getCurrX();
int top = recordTop + mScroller.getCurrY();
params.setMargins(left, top, 0, 0);
setLayoutParams(params);
if(completeMoveListener != null) completeMoveListener.onCompleteMove(this, left, top);
}
} } private void scrollToLeft() {
mScroller.startScroll(0, 0, -recordLeft, 0);
postInvalidate();
} private void scrollToTop() {
mScroller.startScroll(0, 0, 0, -recordTop + stateHeight);
postInvalidate();
} private void scrollToRight() {
mScroller.startScroll(0, 0, screenWidth - recordRight, 0);
postInvalidate();
} private void scrollToBottom() {
mScroller.startScroll(0, 0, 0, screenHeight - recordBottom);
postInvalidate();
} //-------------------
// 监听
//-------------------
private ClickListener clickedlistener;
public void setClickListener(ClickListener clickedlistener) {
this.clickedlistener = clickedlistener;
}
public interface ClickListener {
public void onClick(View v);
} private CompleteMoveListener completeMoveListener;
public void setCompleteMoveListener(CompleteMoveListener completeMoveListener) {
this.completeMoveListener = completeMoveListener;
}
public interface CompleteMoveListener {
public void onCompleteMove(View v, int left, int top);
}
}
原则上可以向上,向下,向左,向有飞,但实现的时候感觉阉割一下下比较清爽,就只实现了左右飞,判断控件移动方向是根据控件空心位置和限制条件决定的:
private int topLine;
private int bottomLine;
private int leftLine;
private int rightLine
限制条件是1、topLine=0,2、bottomLine=screenHeight-状态栏高度,3、screenHeight=屏幕高-状态栏高,4、屏幕中心x坐标。
下面是main.xml布局文件:
<RelativeLayout 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"
tools:context="com.example.com.sus.MainActivity" > <TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/hello_world" /> <com.example.com.sus.WSuspensionButton
android:id="@+id/btnSus"
android:layout_width="50dp"
android:layout_height="50dp"
/> </RelativeLayout>
下面是在Activity中的初始化设置:
WSuspensionButton suspensionButton = (WSuspensionButton)findViewById(R.id.btnSus);
int size = suspensionButton.getSize();
suspensionButton.setBackground(new ColorDrawable(Color.BLUE));
RelativeLayout.LayoutParams wmParams = new RelativeLayout.LayoutParams(
size, size);
wmParams.leftMargin = preferences.getInt("suspend_btn_x", dm.widthPixels
- size);
wmParams.topMargin = preferences.getInt("suspend_btn_y",
(dm.heightPixels - size) / 2);
suspensionButton.setLayoutParams(wmParams);// 取本地存入的位置信息,并从新设定控件的坐标
suspensionButton
.setClickListener(new WSuspensionButton.ClickListener() { public void onClick(View v) {
showToast();
} private void showToast() {
Toast.makeText(mActivity,
"You clicked speaker button!",
Toast.LENGTH_SHORT).show();
}
});
suspensionButton
.setCompleteMoveListener(new WSuspensionButton.CompleteMoveListener() { public void onCompleteMove(View v, int left, int top) {
saveInLocal(left, top);
} private void saveInLocal(int left, int top) {
SharedPreferences.Editor editor = preferences.edit();
editor.putInt("suspend_btn_x", left);
editor.putInt("suspend_btn_y", top);
editor.commit();
}
});
在初始化该控件的时候,对该控件的大小以及屏幕中显示的位置进行了从新设置,让其显示在靠屏幕右侧的中心位置。
好了,以上就是今天的干货。分享愉快
转载请标明转载地址:http://www.cnblogs.com/swalka/p/5078109.html
Andoird实现类似iphone AssistiveTouch的控件的demo的更多相关文章
- iOS iPhone iPad 各种控件默认高度
iPhone iPad 各种控件默认高度 注意:这些是ios7之前的,ios7之后(包括ios7)有改动,我会在后面标注出来 iPhone和iPad下各种常见控件的宽度和标准是一样的,所以这里就用iP ...
- WPF常用控件应用demo
WPF常用控件应用demo 一.Demo 1.Demo截图如下: 2.demo实现过程 总体布局:因放大缩小窗体,控件很根据空间是否足够改变布局,故用WrapPanel布局. <ScrollVi ...
- 如何在web中实现类似excel的表格控件
Execl功能非常强大,内置的很多函数或公式可以大大提高对数据的加工处理能力.那么在web中有没有类似的控件呢?经过一番搜寻,发现handsontable具备了基本的excel功能支持公式,同时能对数 ...
- .NET在WebForm里实现类似WinForm里面TrackBar控件的效果(AJAX Control Toolkit的使用)
WinForm 里面有一个 TrackBar 控件,表示一个标准的 Windows 跟踪条,是类似于 ScrollBar 控件的可滚动控件.用这个控件可以实现很多可以实时调整的功能,比如最常见的音量调 ...
- iPhone iPad 各种控件默认高度
iPhone和iPad下各种常见控件的宽度和标准是一样的,所以这里就用iPhone说明. 以下是常见的几种控件的高度.Statusbar,Navigationbar和Tabbar的宽度极其图标大小. ...
- Android仿IOS的AssistiveTouch的控件EasyTouch实现
概述: 之前我听到过一则新闻,就是说Ipone中的AssistiveTouch的设计初衷是给残疾人使用的. 而这一功能在亚洲(中国)的使用最为频繁. 虽不知道这新闻的可靠性,但无庸置疑的是它的确给我们 ...
- Android类似日历的翻转控件
最近写了个翻转面板的控件拿出来与大家分享一下,类似日历的那种,写的比较简单有需要的可以直接拿去用.直接上效果图吧,代码我放在百度云了,有问题的话直接回复就好呢,大家一起交流下. http://pan. ...
- Delphi 制作自定义数据感知控件并装入包(dpk文件)中(与DBText类似的数据感知控件)
一.基础知识简介: 1.包的命名介绍: 包的命名没有规则,但是建议遵守包的命名约定:包的命名与包的版本相关,包的名称前面几个字符通常表示作者或公司名,也可以是控件的一个描述词,后面紧跟的Std表示运行 ...
- 【WPF】推荐一款拖拉缩放控件的DEMO
引言 在CodeProject上有个实现了控件拖拉缩放的DEMO,界面很漂亮,里面对Thumb和Adorner运用得很精彩.我觉得,使用WPF的开发者都可以去学习一下.下面放出链接. WPF Diag ...
随机推荐
- Enum的简单了解
Enum可以将一组具名的有限集合创建成一种新的类型,而这些具名的值可以作为常规的程序组件使用. 在创建enum时,编译器会为你生成一个相关的类,这个类继承自java.lang.Enum,所以enum本 ...
- Android----------WindowManager
我们Android平台是一个又一个的Activity组成的,每个Activity有一个或者多个View构成.所以说.当我们想显示一个界面的时候,我们首先想到的是建立一个Activity,然后全部的操作 ...
- 基于Opencv图像处理的时时头像採集试验
2014 4.20 近期想做一个关于图像处理的软件玩玩,可惜也没有什么特别的想法,就当玩玩好了,准备用Opencv开源库实现下简单的功能吧. Opencv是一个专业的图像处理库,里面有非常多基础函数能 ...
- Android 自己定义View (二) 进阶
转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/24300125 继续自己定义View之旅.前面已经介绍过一个自己定义View的基础 ...
- Cocos2d-x v3.0正式版尝鲜体验【3】 Label文本标签
Cocos2d-x在新版本号中增加了新的Label API.和以往不同的是,2.x的版本号是通过三个不同的类来创建不同的文本标签,而如今是模仿着精灵的创建方式.一个类创建不同形式的文本,只是核心内容还 ...
- Ubuntu12.04下使用virtualbox4.3.12 amd64安装XP系统教程
首先第一步打开已安装好的Virtualbox4.3.12,效果图例如以下: 第二步:点击新建进入新建虚拟电脑界面,填写名称,选择类型和版本号(我这里使用的三XP 64bit): 第三步:选择内存大小, ...
- HDU 3639 Hawk-and-Chicken(良好的沟通)
HDU 3639 Hawk-and-Chicken 题目链接 题意:就是在一个有向图上,满足传递关系,比方a->b, b->c,那么c能够得到2的支持,问得到支持最大的是谁,而且输出这些人 ...
- linux+nginx+mysql+php
LNMP(linux+nginx+mysql+php)服务器环境配置 一.简介 Nginx是俄罗斯人编写的十分轻量级的HTTP服务器,Nginx,它的发音为 “engine X”, 是一个高性能的 ...
- C#的百度地图开发(二)转换JSON数据为相应的类
原文:C#的百度地图开发(二)转换JSON数据为相应的类 在<C#的百度地图开发(一)发起HTTP请求>一文中我们向百度提供的API的URL发起请求,并得到了返回的结果,结果是一串JSON ...
- (转)ikvmc的使用
IKVM.NET是一个针对Mono和微软.net框架的java实现,其设计目的是在.NET平台上运行java程序.本文将比较详细的介绍这个工具的原理.使用入门(如何java应用转换为.NET应用.), ...