实现一个视图在固定圆圈内移动

2024-02-04 10:18

本文主要是介绍实现一个视图在固定圆圈内移动,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

    我们知道,在android4.0系统原生的滑动锁屏中,用户可以拖动中心的图标在固定的圆圈内移动。本文将以一个小Demo来实现类似这种绕着固定圆圈内拖动图标。

    这个Demo有两个类:

    1. MainActivity:继承Activity,主要用来承载MainView这个自定义ViewGroup类;

    2. MainView:自定义ViewGroup,继承ViewGroup,是本Demo核心类,功能的实现都在这个自定义View类里了;


    MainActivity的代码如下:

package com.example.hu.tvt;import android.app.Activity;
import android.os.Bundle;public class MainActivity extends Activity 
{/** Called when the activity is first created. */@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);//加载MainViewsetContentView(new MainView(this));}
}

  MainView的代码如下:

package com.example.hu.tvt;import android.content.Context;
import android.graphics.Rect;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.view.animation.AlphaAnimation;
import android.view.animation.Animation;
import android.widget.ImageView;
import android.widget.ImageView.ScaleType;public class MainView extends ViewGroup 
{private static final boolean DBG = true;private static final String TAG = "MainView";private int mWidth, mHight;//手机屏幕高度的一半private int mScreenHalfWidth;private int mAlphaViewWidth, mAlphaViewHeight;private int mCenterViewWidth, mCenterViewHeight;private int mCenterViewTop, mCenterViewBottom;private int mAlphaViewTop, mAlphaViewBottom;private boolean mTracking = false;private Rect mCenterViewRect;private ImageView mCenterView, mAlphaView;private AlphaAnimation mAlphaAnimation;public MainView(Context context){super(context);initViews(context);onAnimationStart();}//获取图标,将获取的图标添加入MainView,设置图标的可见性private void initViews(Context context){mAlphaView = new ImageView(context);mAlphaView.setImageResource(R.drawable.alpha);setViewsLayout(mAlphaView);mAlphaView.setVisibility(View.INVISIBLE);mCenterView = new ImageView(context);mCenterView.setImageResource(R.drawable.centure);setViewsLayout(mCenterView);mCenterView.setVisibility(View.VISIBLE);}//设置获取图标的参数,并添加到MainViewprivate void setViewsLayout(ImageView image) {image.setScaleType(ScaleType.CENTER);image.setLayoutParams(new LayoutParams(LayoutParams.WRAP_CONTENT,LayoutParams.WRAP_CONTENT));addView(image);}@Overrideprotected void onLayout(boolean changed, int l, int t, int r, int b) {// TODO Auto-generated method stubif (changed){mWidth = r;mHight = b;//mHalfWidth >> 1为向右位移1位,相当于mHalfWidth / 2。采用位移的原因是计算效率比较高。mScreenHalfWidth = mWidth >> 1;getViewMeasure();//中心图标顶部到屏幕顶部的距离mCenterViewTop = mHight / 2 - (mCenterViewHeight >> 1);//中心图标底部部到屏幕顶部的距离mCenterViewBottom = mHight / 2 + (mCenterViewHeight >> 1);//显示动画的图标顶部到屏幕顶部的距离mAlphaViewTop = mHight / 2 - (mAlphaViewHeight >> 1);//显示动画的图标底部到屏幕顶部的距离mAlphaViewBottom = mHight / 2 + (mAlphaViewHeight >> 1);setChildViewLayout();//创建中心图标所在矩形区域mCenterViewRect = new Rect(mWidth / 2 - mAlphaViewWidth / 2, mAlphaViewTop,mWidth / 2 + mAlphaViewWidth / 2, mAlphaViewBottom);}}//获取中心图片和显示动画图片的宽、高private void getViewMeasure(){mAlphaView.measure(View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED), View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED));//显示动画图标的宽mAlphaViewWidth = mAlphaView.getMeasuredWidth();//显示动画图标的高mAlphaViewHeight = mAlphaView.getMeasuredHeight();mCenterView.measure(View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED), View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED));//中心图标的宽mCenterViewWidth = mCenterView.getMeasuredWidth();//中心图标的高mCenterViewHeight = mCenterView.getMeasuredHeight();}//设置各图标在MainView中的布局private void setChildViewLayout(){mAlphaView.layout(mScreenHalfWidth - mAlphaViewWidth / 2, mAlphaViewTop,mScreenHalfWidth + mAlphaViewWidth / 2, mAlphaViewBottom);mCenterView.layout(mScreenHalfWidth - mCenterViewWidth / 2, mCenterViewTop,mScreenHalfWidth + mCenterViewWidth / 2, mCenterViewBottom);}//停止显示动画@Overrideprotected void onAnimationEnd() {// TODO Auto-generated method stubsuper.onAnimationEnd();if (mAlphaAnimation != null){mAlphaAnimation = null;}mAlphaView.setAnimation(null);}//显示中心图标动画@Overrideprotected void onAnimationStart() {// TODO Auto-generated method stubsuper.onAnimationStart();mAlphaView.setVisibility(View.VISIBLE);if (mAlphaAnimation == null) {mAlphaAnimation = new AlphaAnimation(0.0f, 1.0f);mAlphaAnimation.setDuration(1000);}mAlphaAnimation.setRepeatCount(Animation.INFINITE);mAlphaView.startAnimation(mAlphaAnimation);}//用户手机点下屏幕时首先会先调用执行该函数,然后再执行onTouchEvent@Overridepublic boolean onInterceptTouchEvent(MotionEvent ev) {if(DBG) Log.d(TAG, "onInterceptTouchEvent()");final int action = ev.getAction();final float x = ev.getX();final float y = ev.getY();switch (action) {case MotionEvent.ACTION_DOWN://手指点在中心图标范围区域内if (mCenterViewRect.contains((int) x, (int) y)) {mTracking = true;onAnimationEnd();mAlphaView.setVisibility(View.INVISIBLE);return true;} break;default:break;}return super.onInterceptTouchEvent(ev);}@Overridepublic boolean onTouchEvent(MotionEvent event){/*mTracking为true时,说明中心图标被点击移动* 即只有在中心图标被点击移动的情况下,onTouchEvent* 事件才会触发。*/if (mTracking){final int action = event.getAction();final float nx = event.getX();final float ny = event.getY();switch (action) {case MotionEvent.ACTION_DOWN:break;case MotionEvent.ACTION_MOVE://中心图标移动handleMoveView(nx, ny);break;case MotionEvent.ACTION_UP:mTracking = false;resetMoveView();break;case MotionEvent.ACTION_CANCEL:mTracking = false;resetMoveView();break;}}if(DBG) Log.d(TAG, "onTouchEvent()");return mTracking || super.onTouchEvent(event);}//实现图标在固定圆圈内移动的方法private void handleMoveView(float x, float y){int mHalfCenterViewWidth = mCenterViewWidth >> 1;//Radius为中心图标移动的限定的圆范围区域半径(可根据自己的需要设置大小)int Radius = mCenterViewWidth + mHalfCenterViewWidth;//int Radius = 100;/*若用户手指移动的点与中心点的距离长度大于Radius,则中心图标坐标位置限定在移动区域范围圆弧上。* 一般是用户拖动中心图标,手指移动到限定圆范围区域外。*/if (Math.sqrt(dist2(x - mScreenHalfWidth, y - (mCenterView.getTop() + mCenterViewWidth / 2))) > Radius)		{//原理为x1 / x = r1 / r ,此时的x,y为以Radius为半径的圆圈圆弧上的坐标x = (float) ((Radius / (Math.sqrt(dist2(x - mScreenHalfWidth, y - (mCenterView.getTop() + mHalfCenterViewWidth))))) * (x - mScreenHalfWidth) + mScreenHalfWidth);y = (float) ((Radius / (Math.sqrt(dist2(x - mScreenHalfWidth, y - (mCenterView.getTop() + mHalfCenterViewWidth))))) * (y - (mCenterView.getTop() + mHalfCenterViewWidth)) + mCenterView.getTop() + mHalfCenterViewWidth);}/*图形的坐标是以左上角为基准的,* 所以,为了使手指所在的坐标和图标的中心位置一致,* 中心坐标要减去宽度和高度的一半。*/mCenterView.setX((int)x - mCenterView.getWidth()/2);mCenterView.setY((int)y - mCenterView.getHeight()/2);invalidate();}//平方和计算private float dist2(float dx, float dy){return dx * dx + dy * dy;}//重置中心图标,回到原位置private void resetMoveView(){mCenterView.setX(mWidth / 2 - mCenterViewWidth /2);	mCenterView.setY((mCenterView.getTop() + mCenterViewHeight / 2) - mCenterViewHeight / 2);onAnimationStart();invalidate();}
}

  代码实现总结:

     1. 代码中的handleMoveView函数是计算图标在固定圆圈中移动的核心方法。

     2. 当我们移动中心图标时,会调用onTouchEvent方法,然后会获得用户手指所在屏幕点的移动坐标。

     3. 将获得的移动坐标作为参数传到handleMoveView,在handleMoveView方法里再对坐标的选定判断处理。

     4. 当用户移动图标超出限定的圆圈(移动坐标相对图标原始位置的偏移量形成的直角三角形的斜边大于限定圆圈半径时),那么这时候就将图标移动定格在限定圆圈的圆弧上。

     5. 至于圆弧点的坐标值,可根据获得的用户手指所在屏幕的坐标值(即通过onTouchEvent获得的坐标值)通过比例计算得到相应的圆弧点的坐标值(注:x,y的偏移量和半径连成一个直角三角形,所以可以通过比例等值计算得到圆弧的坐标值)。

     6. 得到圆弧点上的坐标值后,就可以通过设置中心图标的坐标值移动图标到相应的坐标点了。


     ok,关于实现一个视图在固定圆圈内移动就到此结束了,有时间会在此基础上开发一个基于第三方应用的锁屏;

    相关Demo代码下载链接:http://download.csdn.net/detail/stevenhu_223/5591373

这篇关于实现一个视图在固定圆圈内移动的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/677150

相关文章

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

通过SSH隧道实现通过远程服务器上外网

搭建隧道 autossh -M 0 -f -D 1080 -C -N user1@remotehost##验证隧道是否生效,查看1080端口是否启动netstat -tuln | grep 1080## 测试ssh 隧道是否生效curl -x socks5h://127.0.0.1:1080 -I http://www.github.com 将autossh 设置为服务,隧道开机启动

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测 目录 时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测基本介绍程序设计参考资料 基本介绍 MATLAB实现LSTM时间序列未来多步预测-递归预测。LSTM是一种含有LSTM区块(blocks)或其他的一种类神经网络,文献或其他资料中LSTM区块可能被描述成智能网络单元,因为

vue项目集成CanvasEditor实现Word在线编辑器

CanvasEditor实现Word在线编辑器 官网文档:https://hufe.club/canvas-editor-docs/guide/schema.html 源码地址:https://github.com/Hufe921/canvas-editor 前提声明: 由于CanvasEditor目前不支持vue、react 等框架开箱即用版,所以需要我们去Git下载源码,拿到其中两个主

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

基于Springboot + vue 的抗疫物质管理系统的设计与实现

目录 📚 前言 📑摘要 📑系统流程 📚 系统架构设计 📚 数据库设计 📚 系统功能的具体实现    💬 系统登录注册 系统登录 登录界面   用户添加  💬 抗疫列表展示模块     区域信息管理 添加物资详情 抗疫物资列表展示 抗疫物资申请 抗疫物资审核 ✒️ 源码实现 💖 源码获取 😁 联系方式 📚 前言 📑博客主页:

移动硬盘盒:便携与交互的完美结合 PD 充电IC

在数字化时代的浪潮中,数据已成为我们生活中不可或缺的一部分。随着数据的不断增长,人们对于数据存储的需求也在不断增加。传统的存储设备如U盘、光盘等,虽然具有一定的便携性,但在容量和稳定性方面往往难以满足现代人的需求。而移动硬盘,以其大容量、高稳定性和可移动性,成为了数据存储的优选方案。然而,单纯的移动硬盘在携带和使用上仍存在诸多不便,于是,移动硬盘盒应运而生,以其独特的便携性和交互性,成为了数据存储

探索蓝牙协议的奥秘:用ESP32实现高质量蓝牙音频传输

蓝牙(Bluetooth)是一种短距离无线通信技术,广泛应用于各种电子设备之间的数据传输。自1994年由爱立信公司首次提出以来,蓝牙技术已经经历了多个版本的更新和改进。本文将详细介绍蓝牙协议,并通过一个具体的项目——使用ESP32实现蓝牙音频传输,来展示蓝牙协议的实际应用及其优点。 蓝牙协议概述 蓝牙协议栈 蓝牙协议栈是蓝牙技术的核心,定义了蓝牙设备之间如何进行通信。蓝牙协议

VirtualBox中,虚拟系统文件VDI移动或者复制

在安装virtualbox以后有时需要复制,移动虚拟磁盘等操作,这些操作在vmware的虚拟机下面可以直接操作虚拟磁盘即可使用,但是在virtualbox环境 下每个VDI 文件都有一个唯一的uuid,而VirtualBox 不允许注册重复的uuid,所以直接复制的VDI文件是不能拿来使用的,我们就需要使用到virtualbox自带的管理命令来克隆一个VDI,这样通过命令克隆的VDI文件会重

python实现最简单循环神经网络(RNNs)

Recurrent Neural Networks(RNNs) 的模型: 上图中红色部分是输入向量。文本、单词、数据都是输入,在网络里都以向量的形式进行表示。 绿色部分是隐藏向量。是加工处理过程。 蓝色部分是输出向量。 python代码表示如下: rnn = RNN()y = rnn.step(x) # x为输入向量,y为输出向量 RNNs神经网络由神经元组成, python