android仿iosviewpager滑动进度条

author author     2022-12-11     752

关键词:


最近做项目,碰到如下的需求:ViewPager分页,如果是6页(包括6页)就用圆点,如果是6页以上就用进度条来切换。前面一种交互方法最常见,用小圆点来表示当前选中的页面,这些小圆点称为导航点,很多App都是这种实现方式。当用户第一次安装或升级应用时,都会利用导航页面告诉用户当前版本的主要亮点,一般情况下当行页面有三部分组成,背景图片,导航文字和滑动的原点,即下面的效果:

Android仿IOS

这里就不作详细的讲解,大家可以参考我以前写过的博客:

​​ViewPager实现图片轮翻效果​​

今天来实现ViewPager进度条切换,主要逻辑如下:

MainActivity.java

package com.jackie.slidebarviewdemo.activity;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.widget.TextView;

import com.jackie.slidebarviewdemo.R;
import com.jackie.slidebarviewdemo.widget.SlideBarView;

public class MainActivity extends AppCompatActivity
private SlideBarView mSlideBarView;
private TextView mTextView;

@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);

mSlideBarView = (SlideBarView) findViewById(R.id.slide_bar);
mTextView = (TextView) findViewById(R.id.text_view);

mSlideBarView.setTotalPage(80);
mSlideBarView.setOnSlideChangeListener(new SlideBarView.OnSlideChangeListener()
@Override
public void onSlideChange(int page)
mTextView.setText("当前是第" + page + "页");

);

SlideBarView.java

package com.jackie.slidebarviewdemo.widget;

import android.content.Context;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.widget.LinearLayout;
import android.widget.PopupWindow;
import android.widget.RelativeLayout;
import android.widget.TextView;

import com.jackie.slidebarviewdemo.R;
import com.jackie.slidebarviewdemo.utils.ConvertUtils;

/**
* Created by Jackie on 2017/1/17.
*/

public class SlideBarView extends RelativeLayout
private LayoutInflater mInflater;

private RelativeLayout mSlideBarView;
private View mSlideBarBlock;

private PopupWindow mPopupWindow;
private TextView mPopupText;

private int mDp40;

private String mBound = "no"; // no表示没到边界,left为到左边界了,right表示到右边界了

public interface OnSlideChangeListener
void onSlideChange(int page);


private OnSlideChangeListener mOnSlideChangeListener;
public void setOnSlideChangeListener(OnSlideChangeListener onSlideChangeListener)
this.mOnSlideChangeListener = onSlideChangeListener;


public SlideBarView(Context context)
this(context, null);


public SlideBarView(Context context, AttributeSet attrs)
this(context, attrs, 0);


public SlideBarView(Context context, AttributeSet attrs, int defStyleAttr)
super(context, attrs, defStyleAttr);

init(context);
initEvent();


private void init(Context context)
mInflater = LayoutInflater.from(context);
View slideBar = mInflater.inflate(R.layout.slide_bar, null);
LayoutParams params = new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT);
addView(slideBar, params);

mSlideBarView = (RelativeLayout) slideBar.findViewById(R.id.slide_bar_view);
mSlideBarBlock = slideBar.findViewById(R.id.slide_bar_block);

mDp40 = ConvertUtils.dip2px(context, 40);


private void initEvent()
mSlideBarView.setOnTouchListener(new OnTouchListener()
int currentX = 0;
int startX = 0;

@Override
public boolean onTouch(View v, MotionEvent event)
switch (event.getAction())
case MotionEvent.ACTION_DOWN:
currentX = (int) event.getX();
startX = (int) event.getX();

// 设置滑块的滑动, 手指第一次点下去把滑块放到手指上
int downLeft = currentX - mSlideBarBlock.getMeasuredWidth() / 2;
int downTop = mSlideBarBlock.getTop();
int downRight = downLeft + mSlideBarBlock.getWidth();
int downBottom = mSlideBarBlock.getBottom();

//边界检测
if (downLeft < 0)
downLeft = 0;
downRight = mSlideBarBlock.getMeasuredWidth();
else if (downRight > mSlideBarView.getMeasuredWidth())
downLeft = mSlideBarView.getMeasuredWidth() - mSlideBarBlock.getMeasuredWidth();
downRight = mSlideBarView.getMeasuredWidth();


mSlideBarBlock.layout(downLeft, downTop, downRight, downBottom);
break;
case MotionEvent.ACTION_MOVE:
currentX = (int) event.getX();
int currentPage = currentX * mTotalPage / mSlideBarView.getMeasuredWidth();
if (currentPage < 0)
currentPage = 0;
else if (currentPage > mTotalPage)
currentPage = mTotalPage;


// 设置滑块的滑动
int moveLeft = currentX - mSlideBarBlock.getMeasuredWidth() / 2;
int moveTop = mSlideBarBlock.getTop();
int moveRight = moveLeft + mSlideBarBlock.getMeasuredWidth();
int moveBottom = mSlideBarBlock.getBottom();

//边界处理
if (moveLeft < 0)
mBound = "left";

moveLeft = 0;
moveRight = mSlideBarBlock.getMeasuredWidth();
else if (moveRight >= mSlideBarView.getMeasuredWidth())
mBound = "right";

moveLeft = mSlideBarView.getMeasuredWidth() - mSlideBarBlock.getMeasuredWidth();
moveRight = mSlideBarView.getMeasuredWidth();
else
mBound = "no";


mSlideBarBlock.layout(moveLeft, moveTop, moveRight, moveBottom);
startX = currentX;

//设置popupWindow的弹出位置
if (mOnSlideChangeListener != null)
if (currentPage == mTotalPage)
//防止ViewPager越界
currentPage = mTotalPage - 1;


mOnSlideChangeListener.onSlideChange(currentPage);

if (mPopupWindow != null)
mPopupText.setText(currentPage + "");

//设置PopupWindow的滑动
if (!mPopupWindow.isShowing())
int[] location = new int[2];
mSlideBarView.getLocationInWindow(location);
mPopupWindow.showAsDropDown(mSlideBarView, currentX, location[1] - mDp40);
else
if ("no".equals(mBound))
int[] location = new int[2] ;
mSlideBarView.getLocationInWindow(location);
mPopupWindow.update(currentX, location[1] - mDp40, mPopupWindow.getWidth(), mPopupWindow.getHeight(), true);




break;
case MotionEvent.ACTION_UP:
currentX = 0;
startX = 0;
mPopupWindow.dismiss();
break;


return true;

);

// 初始化PopupWindow
View contentView = mInflater.inflate(R.layout.popup_window, null);
mPopupText = (TextView) contentView.findViewById(R.id.popup_text);
mPopupWindow = new PopupWindow(LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT);
mPopupWindow.setContentView(contentView);
mPopupWindow.setOutsideTouchable(true);
mPopupWindow.setBackgroundDrawable(getResources().getDrawable(R.mipmap.popup_window_bg));
mPopupWindow.setAnimationStyle(0);


int mTotalPage = 0;
public void setTotalPage(int totalPage)
this.mTotalPage = totalPage;

相关的单位转化工具,大家可以拷贝到自己的项目中直接使用。

ConvertUtils.java

package com.jackie.slidebarviewdemo.utils;

import android.content.Context;

public class ConvertUtils
public static int dip2px(Context context, float dpValue)
final float scale = context.getResources().getDisplayMetrics().density;
return (int) (dpValue * scale + 0.5f);


public static int px2dip(Context context, float pxValue)
final float scale = context.getResources().getDisplayMetrics().density;
return (int) (pxValue / scale + 0.5f);


public static int px2sp(Context context, float pxValue)
final float fontScale = context.getResources().getDisplayMetrics().scaledDensity;
return (int) (pxValue / fontScale + 0.5f);


public static int sp2px(Context context, float spValue)
final float fontScale = context.getResources().getDisplayMetrics().scaledDensity;
return (int) (spValue * fontScale + 0.5f);

自定义组合控件,然后实现相关的手势,思路很清晰,代码也很详细,这里就直接贴代码了。

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_
android:layout_>

<LinearLayout
android:layout_
android:layout_
android:layout_centerInParent="true"
android:orientation="vertical">

<com.jackie.slidebarviewdemo.widget.SlideBarView
android:id="@+id/slide_bar"
android:layout_
android:layout_
android:layout_marginLeft="20dp"
android:layout_marginRight="20dp"/>

<TextView
android:id="@+id/text_view"
android:layout_
android:layout_
android:layout_gravity="center_horizontal"
android:layout_marginTop="20dp"
android:textColor="#000"
android:textSize="20dp"
android:text="当前是第0页"/>
</LinearLayout>
</RelativeLayout>

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_
android:layout_>

<RelativeLayout
android:id="@+id/slide_bar_view"
android:layout_
android:layout_>

<View
android:layout_
android:layout_
android:layout_centerInParent="true"
android:background="@drawable/shape_slide_bar_bg"/>

<View
android:id="@+id/slide_bar_block"
android:layout_
android:layout_
android:background="#b9b9b9"
android:layout_centerVertical="true" />
</RelativeLayout>
</RelativeLayout>

popup_window.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_
android:layout_>
<RelativeLayout
android:layout_
android:layout_>
<TextView
android:id="@+id/popup_text"
android:layout_
android:layout_
android:textColor="#fff"
android:textSize="16dp"
android:gravity="center"
android:layout_centerInParent="true" />
</RelativeLayout>
</RelativeLayout>

附上相关的资源文件:

shape_slide_bar_bg.xml

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle">
<solid android:color="#dcdcdc" />
<corners android:radius="1dp"/>
</shape>

popup_window_bg.9.png

Android仿IOS



效果如下:

Android仿IOS


androidseekbar使用(代码片段)

本文主要是介绍androidseekbar的使用。一、含义SeekBar是拖动条,是ProgressBar的一个子类,如:音乐播放、音量条、播放进度条等。效果图:二、常用属性和方法属性名含义progress设置该进度条的最大值progressDrawable自... 查看详情

仿ios圆形下载进度条

/***CreatedbyC058on2016/5/25.*/publicclassMyHoriztalProgressBarextendsProgressBar{privatestaticfinalintDEFAULT_REACH_COLOR=0xff24F569;privatestaticfinalintDEFAULT_UNREACH_COLOR=0xffC0C0C0;privatestati 查看详情

如何在 Android 中向 ViewPager 添加进度条

】如何在Android中向ViewPager添加进度条【英文标题】:HowtoAddprogressbartoViewPagerinAndroid【发布时间】:2012-04-1721:50:24【问题描述】:如何在Android中为ViewPager添加进度条?我的情况是我必须在ViewPager中加载imageURL。当用户滑动页面时... 查看详情

进度条和滑动条

1#import"ViewController.h"2#import"ViewController2.h"34@interfaceViewController()56@end78@implementationViewController910@synthesizeslider=_slider;11@synthesizepView=_progressView;1213-(void)viewDidLo 查看详情

libgdx之progressbar(进度条)slider(滑动条)

...或者knob的最大高度决定的,默认的宽度width是140.Slider滑动条,一般在游戏中设置调节音量或者调节游戏难度等等。Slider就是一水平的指示条,允许用户设值。ExtendsProgressBar 查看详情

自定义控件——圆形圆点进度条(仿安全卫士中的一键加速)

一、源代码源代码及demo下载(此进度条开源项目后续会持续添加、更新)二、行动由来在开发交流群中,一童鞋说要实现这个进度条,但在网上没有找到开源项目。看到这个图片,很熟悉吧?有木有想点它的冲动?觉得有点意... 查看详情

如何只实现一个圆形进度条,直到图像被滑动加载

】如何只实现一个圆形进度条,直到图像被滑动加载【英文标题】:Howtoimplementonlyonecircularprogressbaruntilimagesareloadedwithglide【发布时间】:2021-08-2406:49:15【问题描述】:您好,我只想实现一个圆形进度动画,直到从滑行加载图像,... 查看详情

例题进度条效果点击图片滑动效果。(代码片段)

进度条效果<!DOCTYPEhtmlPUBLIC"-//W3C//DTDXHTML1.0Transitional//EN""http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"><htmlxmlns="http://www.w3.org/1999/xhtml"><head><metahttp-eq 查看详情

仿猎豹扫描进度(代码片段)

布局:<RelativeLayoutxmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent&# 查看详情

自定义view-滑动进度条(代码片段)

好久没有写文章啦。记录一下此时的心情,哈哈。先上效果图:布局样式:<declare-styleablename="SlidingScaleBarView"><!--刻度文字的大小--><attrname="scaleTextSize"format="dimensi 查看详情

自定义view-滑动进度条(代码片段)

好久没有写文章啦。记录一下此时的心情,哈哈。先上效果图:布局样式:<declare-styleablename="SlidingScaleBarView"><!--刻度文字的大小--><attrname="scaleTextSize"format="dimensi 查看详情

ue4音乐时长进度条

...建4个变量  Audio(音乐组件)bManualTimeChange(是否滑动) PlayingMovedToPercentage(进度条滑动长度) PlayingSongDuration(音乐总长度)4.创建三个函数 FnSetupTrackDuration(初始化时长) FnUpdateTrackPlayback(设置当前时... 查看详情

ycprogress自定义百分比进度条(代码片段)

目录介绍1.本库优势亮点2.使用介绍2.1圆环百分比进度条2.2直线百分比进度条2.3仿杀毒类型百分比进度条3.注意要点4.效果展示5.其他介绍1.本库优势亮点圆环百分比进度条简便且小巧,支持设置多种属性。可以设置内圆和外圆的颜... 查看详情

androidapp开发音量调节中实现拖动条和滑动条和音频管理器audiomanager讲解及实战(超详细附源码和演示视频)(代码片段)

...动之后用户不能直观的看到当前进度值是多少,为此Android设计了全新的滑动条空间Slider,首先要增加以下配置implementaion'com.google.android.materia:material:1.4.0'运行效果如下可以手动点击控制音量 代码如下Java类packagecom.e... 查看详情

在android进度条中删除进度条背景

】在android进度条中删除进度条背景【英文标题】:Removeprogressbarbackgroundinandroidprogressbar【发布时间】:2014-08-0516:20:51【问题描述】:如何去除灰色背景,只在进度条中显示蓝色进度条。【问题讨论】:见***.com/questions/16893209/…这... 查看详情

进度条与拖动条的使用学习(代码片段)

(一)进度条常用属性详解:android:max:进度条的最大值android:progress:进度条已完成进度值android:progressDrawable:设置轨道对应的Drawable对象android:indeterminate:如果设置成true,则进度条不精确显示进度android:indeterminateDrawable:设置... 查看详情

酷炫进度条自定义seekbar(代码片段)

...c;是这样子的:需求:1、看了ui图可以知道这类似android自带的seekbar控件,2、一个水平进度条和一个圆形进度条;3、圆形进度条显示环形刻度和当前进度值;4、并且圆形进度可滑动操作;最终实现效果ÿ... 查看详情

仿网易新闻鼠标滑动跟随效果

<!DOCTYPEhtml><html><head><metacharset="UTF-8"><title>jQuery仿百度新闻网站导航条特效</title><styletype="text/css">*{ margin:0; padding:0; overflow-x:hidden;overflow-y:aut 查看详情