仿格瓦拉生活注册界面验证码滑动效果

2024-02-03 18:40

本文主要是介绍仿格瓦拉生活注册界面验证码滑动效果,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近在项目当中有一个滑动验证码动画效果要实现,刚开始看到这个效果感觉还是可以的,但是要实现起来觉得有点麻烦,我想大家一定都和我一样拿到这个需求一定是一顿谷歌,百度啥的,但是搜索了很久都没找到相似的效果,又催的急没办法,卷起袖子干吧,就开始自己造轮子的过程,在做这个需求的过程中,还是很有收获的比如说复习了下高中的函数知识(开玩笑,重要的是复习了下自定义view),我想大家都和一样高中的数学知识应该都忘的差不多了吧,反正我是忘的差不多了,废话不多说了,直接上代码,里面使用到了的图片,大家可以自己搞一点图片进去试试

下图是我截取的一部分,效果就是这样

WX20171218-163655@2x.png

WX20171218-163642@2x.png

import android.content.Context;
import android.support.annotation.AttrRes;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.RelativeLayout;import com.nineoldandroids.view.ViewHelper;public class MyFrameLayout extends FrameLayout {/*** 这里面所说的右边的距离,都是子view 距离父控件的距离,比例右边距离就是说是右边到父控件的距离*/private float down_x;//按下点的距离private ImageView imageView_8;//大框中变化的框private ImageView imageView_7;//第一个框private ImageView imageView_6;//最左边的小框,以下是从左到右private ImageView imageView_5;//第二个小框private ImageView imageView_4;//第三个小框private ImageView imageView_3;//第四个小框private ImageView imageView_2;//第五个小框private ImageView imageView_1;//第六个小框private float frist_left;//大框的左边private float frist_right;//大框的右边private float eight_right;//最右边框的右边距离,也就是第有个小框的右边距离private float frist_center;//最大框的中心点private float position;//表示变化的周期,也是从0-1的变化的距离private float position_key;//表示大框中心点移动到这个点的时候开始变化private float position_7;//大框的中点的位置private int width;//这个view 的宽度private float three_right;//第六个小框的右边距离private float position_center;//两个小框中间距离一半protected boolean isFrist = true;//这里作用就是只能滑动一次private int[] arr = {R.drawable.test_1,R.drawable.test_2,R.drawable.test_3,R.drawable.test_4,R.drawable.test_5,R.drawable.test_6,R.drawable.test_bg};private float five_right;//第5个小框的右边距离private float six_right;//第六个小框的右边距离private float seven_right;//第7个小框的右边距离private float four_right;//第八个小框的右边距离private static int KEY = 0;public MyFrameLayout(@NonNull Context context) {super(context);init(context);}public MyFrameLayout(@NonNull Context context, @Nullable AttributeSet attrs) {super(context, attrs);init(context);}public MyFrameLayout(@NonNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr) {super(context, attrs, defStyleAttr);init(context);}/*** 方便外部重置*/public void reset(){isFrist = true;ViewHelper.setTranslationX(imageView_7, 0);ViewHelper.setTranslationX(imageView_8, 0);}/*** 从外面传进来一个图片数组* 这里里面也可以传递一个数字的数字* @param arr*/public void inview(int[] arr) {this.arr = arr;imageView_1.setImageDrawable(this.getResources().getDrawable(R.drawable.test_1));imageView_2.setImageDrawable(this.getResources().getDrawable(R.drawable.test_2));imageView_3.setImageDrawable(this.getResources().getDrawable(R.drawable.test_3));imageView_4.setImageDrawable(this.getResources().getDrawable(R.drawable.test_4));imageView_5.setImageDrawable(this.getResources().getDrawable(R.drawable.test_5));imageView_6.setImageDrawable(this.getResources().getDrawable(R.drawable.test_6));}private void init(Context context) {View view = View.inflate(context, R.layout.my_framelayout, this);imageView_8 = (ImageView) view.findViewById(R.id.iv_8);imageView_7 = (ImageView) view.findViewById(R.id.iv_7);imageView_6 = (ImageView) view.findViewById(R.id.iv_6);imageView_5 = (ImageView) view.findViewById(R.id.iv_5);imageView_4 = (ImageView) view.findViewById(R.id.iv_4);imageView_3 = (ImageView) view.findViewById(R.id.iv_3);imageView_2 = (ImageView) view.findViewById(R.id.iv_2);imageView_1 = (ImageView) view.findViewById(R.id.iv_1);}@Overridepublic boolean onTouchEvent(MotionEvent event) {switch (event.getAction()) {case MotionEvent.ACTION_DOWN:if (isFrist) {down_x = event.getX();if (down_x - frist_right >= 0 || down_x - frist_left <= 0) {//让其在这个大框的里面才能让其滑动return false;}} else {return false;}break;case MotionEvent.ACTION_MOVE:float currentX = event.getX();if (currentX != down_x) {setTranslation(currentX);}break;case MotionEvent.ACTION_UP:isFrist = false;lister.moveKey(KEY);//回调break;}return true;}/*** 随着手指的移动,一直会调用这个方法* @param transl*/public void setTranslation(float transl) {//实时的获取大框的中心点的位置float move_position = transl - down_x + position_7;//只有大框的中心点位置到达指定的关键点才让其变化,这里就是大框只变化的小框float scX = move_position - position_key;if (scX >= 0) {changeDrawable(move_position,scX);}else {imageView_8.setImageDrawable(null);//其余的地方就用不设置图片}if (transl <= down_x || transl - down_x + position_7 >= width - frist_center) {//这个就是边界点的设置} else {ViewHelper.setTranslationX(imageView_7, transl - down_x);ViewHelper.setTranslationX(imageView_8, transl - down_x);}}/*** 转变image_7的图片背景*/private void changeDrawable(float move_position,float scX){if (move_position <=three_right+position_center  && move_position >= 0) {//第一张图片的设置,以及变化的范围,下面也是相对应imageView_8.setImageDrawable(this.getResources().getDrawable(arr[5]));KEY = 1;} else if (move_position>three_right+position_center && move_position<=four_right+position_center){imageView_8.setImageDrawable(this.getResources().getDrawable(arr[4]));KEY = 2;}else if (move_position>four_right+position_center && move_position<=five_right+position_center){imageView_8.setImageDrawable(this.getResources().getDrawable(arr[3]));KEY = 3;}else if (move_position>five_right+position_center && move_position<=six_right+position_center){imageView_8.setImageDrawable(this.getResources().getDrawable(arr[2]));KEY = 4;}else if (move_position>six_right+position_center && move_position<=seven_right+position_center){imageView_8.setImageDrawable(this.getResources().getDrawable(arr[1]));KEY = 5;}else if (move_position>seven_right+position_center && move_position<=eight_right+position_center){imageView_8.setImageDrawable(this.getResources().getDrawable(arr[0]));KEY = 6;}else {imageView_8.setImageDrawable(null);KEY = 0;}//这就是这个项目的关键点,利用正弦函数,来达到这个效果,至于为啥是正弦我也是恶补了下高中的数学(开玩笑),才有想法//其实这个大家想明白了就很简单ViewHelper.setScaleX(imageView_8, (float) (Math.sin((double) (scX * Math.PI / (position * 2)))));ViewHelper.setScaleY(imageView_8, (float) (Math.sin((double) (scX * Math.PI / (position * 2)))));
//}private TransterListener lister;public interface TransterListener {void moveKey(int key);}public void setLister(TransterListener lister) {this.lister = lister;}@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);width = getWidth();RelativeLayout childAt = (RelativeLayout) MyFrameLayout.this.getChildAt(0);int childCount = childAt.getChildCount();frist_left = childAt.getChildAt(childCount - 2).getLeft();frist_right = childAt.getChildAt(childCount - 2).getRight();//大框的半径frist_center = (frist_right - frist_left) / 2;float three_left = childAt.getChildAt(childCount - 3).getLeft();three_right = childAt.getChildAt(childCount - 3).getRight();four_right = childAt.getChildAt(childCount - 4).getRight();five_right = childAt.getChildAt(childCount - 5).getRight();six_right = childAt.getChildAt(childCount - 6).getRight();seven_right = childAt.getChildAt(childCount - 7).getRight();float center_6 = (three_right - three_left) / 2;//第二框的半径//大框的中心position_7 = frist_right - frist_center;//中间间距的半径position_center = (three_left - frist_right) / 2;position = center_6 + position_center;//表示变化的周期,也是从0-1的变化的距离position_key = three_left - position_center;//节点eight_right = childAt.getChildAt(childCount - 8).getRight();}
}

这个布局也是一个关键的一个地方,图片大家或者背景大家可以自己设置下

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"android:id="@+id/ll_bg"android:layout_width="250dp"android:layout_height="50dp"android:background="@drawable/background_bg"android:gravity="center_vertical"><ImageView
        android:layout_centerVertical="true"android:layout_marginLeft="12dp"android:layout_toRightOf="@+id/iv_2"android:id="@+id/iv_1"android:layout_width="19dp"android:layout_height="19dp"android:src="@drawable/test_1" /><ImageView
        android:layout_centerVertical="true"android:layout_marginLeft="12dp"android:layout_toRightOf="@+id/iv_3"android:id="@+id/iv_2"android:layout_width="19dp"android:layout_height="19dp"android:src="@drawable/test_2" /><ImageView
        android:layout_centerVertical="true"android:layout_marginLeft="12dp"android:layout_toRightOf="@+id/iv_4"android:id="@+id/iv_3"android:layout_width="19dp"android:layout_height="19dp"android:src="@drawable/test_3" /><ImageView
        android:layout_centerVertical="true"android:layout_marginLeft="12dp"android:layout_toRightOf="@+id/iv_5"android:id="@+id/iv_4"android:layout_width="19dp"android:layout_height="19dp"android:src="@drawable/test_4" /><ImageView
        android:layout_centerVertical="true"android:layout_marginLeft="12dp"android:layout_toRightOf="@+id/iv_6"android:id="@+id/iv_5"android:layout_width="19dp"android:layout_height="19dp"android:src="@drawable/test_5" /><ImageView
        android:layout_centerVertical="true"android:layout_marginLeft="12dp"android:layout_toRightOf="@+id/iv_7"android:id="@+id/iv_6"android:layout_width="19dp"android:layout_height="19dp"android:src="@drawable/test_6" /><ImageView
        android:layout_centerVertical="true"android:id="@+id/iv_7"android:layout_marginLeft="5dp"android:layout_width="38dp"android:layout_height="38dp"android:src="@drawable/test_bg" /><ImageView
        android:scaleType="fitCenter"android:layout_marginLeft="10dp"android:layout_centerVertical="true"android:id="@+id/iv_8"android:layout_width="28dp"android:layout_height="28dp"/></RelativeLayout>

这个地方就是显现自定义的地方

import android.app.Activity;
import android.app.Dialog;
import android.content.Context;
import android.graphics.Typeface;
import android.net.Uri;
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.view.Display;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.WindowManager;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.RelativeLayout;
import android.widget.TextView;
import android.widget.Toast;import com.facebook.common.util.UriUtil;
import com.facebook.drawee.backends.pipeline.Fresco;
import com.facebook.drawee.interfaces.DraweeController;
import com.facebook.drawee.view.SimpleDraweeView;import java.util.ArrayList;
import java.util.List;
import java.util.Random;
public class WarmTipsDialog implements MyFrameLayout.TransterListener {private Context mContext;private Dialog dialog;private Display display;private RelativeLayout lLayout_bg;private FrameLayout rl_bg;private View view;private ImageView delete;private TextView mTv;private MyFrameLayout mSBV;public WarmTipsDialog(Context context) {this.mContext = context;WindowManager windowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);display = windowManager.getDefaultDisplay();}public WarmTipsDialog builder() {// 获取Dialog布局view = LayoutInflater.from(mContext).inflate(R.layout.view_warm_tip_layout, null);lLayout_bg = (RelativeLayout) view.findViewById(R.id.ll_bg);delete = (ImageView) view.findViewById(R.id.iv_delete);visibli(view);mSBV = (MyFrameLayout) view.findViewById(R.id.slid_block);mSBV.setLister(this);mTv = (TextView) view.findViewById(R.id.tv_title);
//        // 定义Dialog布局和参数dialog = new Dialog(mContext, R.style.AlertDialogStyle);dialog.setContentView(view);// 调整dialog背景大小lLayout_bg.setLayoutParams(new FrameLayout.LayoutParams((int) (display.getWidth() * 0.9), LinearLayout.LayoutParams.WRAP_CONTENT));delete.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {dismissDialog();}});dialog.setCanceledOnTouchOutside(false);return this;}/*** 显示dialog*/public void showDialog(){if(mContext != null && !((Activity)mContext).isFinishing() && dialog != null){if(!dialog.isShowing()){dialog.show();}}}/*** 取消dialog*/public void dismissDialog(){dialog.dismiss();}@Overridepublic void moveKey(int key) {if (key == 0){mTv.setClickable(true);mTv.setFocusable(true);mTv.setText("验证码错误,点击这里重试");mTv.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {mSBV.reset();}});}else {Toast.makeText(mContext, "我是" + key + "被点击了", Toast.LENGTH_SHORT).show();}}
}

对话框布局

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"android:id="@+id/ll_bg"android:layout_width="match_parent"android:layout_height="wrap_content"android:background="#fff"android:orientation="vertical"><ImageView
            android:id="@+id/iv_delete"android:layout_width="20dp"android:layout_height="20dp"android:layout_alignParentRight="true"android:layout_marginRight="15dp"android:layout_marginTop="15dp"android:src="@drawable/x" /><TextView
            android:id="@+id/tv_title"android:layout_width="match_parent"android:layout_height="wrap_content"android:layout_below="@+id/iv_delete"android:layout_marginTop="50dp"android:gravity="center_horizontal"android:text="@string/tv_title"android:textColor="#333333"android:textSize="20dp" /><MyFrameLayout
            android:id="@+id/slid_block"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_below="@+id/tv_title"android:layout_centerHorizontal="true"android:layout_marginBottom="70dp"android:layout_marginTop="55dp" /></RelativeLayout>

这篇关于仿格瓦拉生活注册界面验证码滑动效果的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

springboot security验证码的登录实例

《springbootsecurity验证码的登录实例》:本文主要介绍springbootsecurity验证码的登录实例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录前言代码示例引入依赖定义验证码生成器定义获取验证码及认证接口测试获取验证码登录总结前言在spring

Python实现自动化接收与处理手机验证码

《Python实现自动化接收与处理手机验证码》在移动互联网时代,短信验证码已成为身份验证、账号注册等环节的重要安全手段,本文将介绍如何利用Python实现验证码的自动接收,识别与转发,需要的可以参考下... 目录引言一、准备工作1.1 硬件与软件需求1.2 环境配置二、核心功能实现2.1 短信监听与获取2.

Flask 验证码自动生成的实现示例

《Flask验证码自动生成的实现示例》本文主要介绍了Flask验证码自动生成的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习... 目录生成图片以及结果处理验证码蓝图html页面展示想必验证码大家都有所了解,但是可以自己定义图片验证码

Spring Cloud之注册中心Nacos的使用详解

《SpringCloud之注册中心Nacos的使用详解》本文介绍SpringCloudAlibaba中的Nacos组件,对比了Nacos与Eureka的区别,展示了如何在项目中引入SpringClo... 目录Naacos服务注册/服务发现引⼊Spring Cloud Alibaba依赖引入Naco编程s依

SpringBoot集成图片验证码框架easy-captcha的详细过程

《SpringBoot集成图片验证码框架easy-captcha的详细过程》本文介绍了如何将Easy-Captcha框架集成到SpringBoot项目中,实现图片验证码功能,Easy-Captcha是... 目录SpringBoot集成图片验证码框架easy-captcha一、引言二、依赖三、代码1. Ea

Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)

《Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)》文章介绍了如何使用dhtmlx-gantt组件来实现公司的甘特图需求,并提供了一个简单的Vue组件示例,文章还分享了一... 目录一、首先 npm 安装插件二、创建一个vue组件三、业务页面内 引用自定义组件:四、dhtmlx

Python自动化处理手机验证码

《Python自动化处理手机验证码》手机验证码是一种常见的身份验证手段,广泛应用于用户注册、登录、交易确认等场景,下面我们来看看如何使用Python自动化处理手机验证码吧... 目录一、获取手机验证码1.1 通过短信接收验证码1.2 使用第三方短信接收服务1.3 使用ADB读取手机短信1.4 通过API获取

Python爬虫selenium验证之中文识别点选+图片验证码案例(最新推荐)

《Python爬虫selenium验证之中文识别点选+图片验证码案例(最新推荐)》本文介绍了如何使用Python和Selenium结合ddddocr库实现图片验证码的识别和点击功能,感兴趣的朋友一起看... 目录1.获取图片2.目标识别3.背景坐标识别3.1 ddddocr3.2 打码平台4.坐标点击5.图

前端原生js实现拖拽排课效果实例

《前端原生js实现拖拽排课效果实例》:本文主要介绍如何实现一个简单的课程表拖拽功能,通过HTML、CSS和JavaScript的配合,我们实现了课程项的拖拽、放置和显示功能,文中通过实例代码介绍的... 目录1. 效果展示2. 效果分析2.1 关键点2.2 实现方法3. 代码实现3.1 html部分3.2

Go路由注册方法详解

《Go路由注册方法详解》Go语言中,http.NewServeMux()和http.HandleFunc()是两种不同的路由注册方式,前者创建独立的ServeMux实例,适合模块化和分层路由,灵活性高... 目录Go路由注册方法1. 路由注册的方式2. 路由器的独立性3. 灵活性4. 启动服务器的方式5.