github:https://github.com/nickeyCode/RoundImageViewPager

说实话不知道怎么描述这个效果,在网页上见得跟多,公司要求做这个效果得时候不知道怎么用文字描述找不到对应的dome只好自己写。

先上图

大概效果就是这个。主要用的的知识点就是viewpager的自定义动画。

项目目录:

roundimg是圆形图片,继承ImageView的,上网好多可以搜索得到

viewpager主要分成三部分

一是viewpager本身,设置adapter,绑定监听器等。

二是adapter,继承PagerAdapter,用法跟listview的差不多。

三是动画类,继承PageTransformer。

首先看看最核心的动画类(能做到这个效果就是根据对应的动画变动)

HeadViewPagerTransformer.Java

public class HeadViewPagerTransformer implements PageTransformer{
private static final float MIN_SCALE = 0.75f;
//主要是设置在不同位置上的VIEW的活动动画
@Override
public void transformPage(View view, float position) {
// TODO Auto-generated method stub
int pageWidth = view.getWidth(); if (position < -) { // [-Infinity,-1)
view.setAlpha();
}
else if (position <= ) { // [-1,0]
view.setAlpha();
view.setTranslationX();
float x = -1.0f * (2f / 3f) * pageWidth * position;
view.setTranslationX(x);
float scaleFactor = MIN_SCALE + ( - MIN_SCALE) * ( - Math.abs(position));
view.setScaleX(scaleFactor);
view.setScaleY(scaleFactor);
} else if (position <= ) { // (0,1]
view.setAlpha();
float x = -1.0f * (2f / 3f) * pageWidth * position;
view.setTranslationX(x);
float scaleFactor = MIN_SCALE + ( - MIN_SCALE) * ( - Math.abs(position));
view.setScaleX(scaleFactor);
view.setScaleY(scaleFactor); }
} }

因为在这个类中,viewpager中view都有对应的位置编号,在正中间显示的view位置是0

在左边的view位置是-1,在右边的view位置是1.(相当于一个坐标轴)

只要viewpager发生滑动,就会调用tansFromPager();position之说以是float类型,是因为如果发生滑动,位置就会有对应的变化,而变化精确到0.0001.

在函数中使用if-else来设定在不同位置区间中的view的动画变化;

if (position < -) { // [-Infinity,-1)
view.setAlpha();
}

这是负无穷到-1的区间,当然,如果你的viewpager缓存的view只有三个的话,这个就没有作用了,因为最多只有三个view,多出来就销毁了。

else if (position <= ) { // [-1,0]
view.setAlpha();
view.setTranslationX();
float x = -1.0f * (2f / 3f) * pageWidth * position;
view.setTranslationX(x);
float scaleFactor = MIN_SCALE + ( - MIN_SCALE) * ( - Math.abs(position));
view.setScaleX(scaleFactor);
view.setScaleY(scaleFactor); }

这是-1到0的区间,就是左边的view到中间或中间的view到左边的动画效果,这里主要是做了两个动画变化,一是大小,二是位置。

这两个变化公式是根据位置的变化与动画数值的关系,解二元一次方程求出来的(初中数学知识。。。。)

具体方式就是balbalblabalbalb。。。。。(不多说)

同理

else if (position <= ) { // (0,1]
view.setAlpha();
float x = -1.0f * (2f / 3f) * pageWidth * position;
view.setTranslationX(x);
float scaleFactor = MIN_SCALE + ( - MIN_SCALE) * ( - Math.abs(position)); }

0到1的区间一样。在实际动画设计的过程中,公式是需要变动的。

剩下的都是普通的viewpager使用,设置adapter  (HeadViewPagerAdapter.java)

public class HeadViewPagerAdapter extends PagerAdapter {

    private Context mContext;
private List<MyImageView> mList; public HeadViewPagerAdapter(Context context,List<MyImageView> list){
this.mContext = context;
this.mList = list;
} @Override
public int getCount() {
// TODO Auto-generated method stub
return mList.size();
} @Override
public boolean isViewFromObject(View arg0, Object arg1) {
// TODO Auto-generated method stub
return arg0 == arg1;
}
//当缓存view的数量超过上限时,会销毁最先的一个
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
// TODO Auto-generated method stub
//Log.d("remove", mImageViews[position].hashCode() + "");
container.removeView(mList.get(position));
}
//添加View
@Override
public Object instantiateItem(ViewGroup container, int position) {
// TODO Auto-generated method stub
container.addView(mList.get(position),);
return mList.get(position);
} }

还有就是对应的绑定:HeadViewPager.java

public class HeadViewPager extends FrameLayout {

    private Context mContext;
private ViewPager mViewPager;
private List<Integer> mImageIds;
private List<MyImageView> mImageViews;
private ViewGroup mViewGroup;
private List<ImageView> tips;
private int tipsChoseImgId;
private int tipsUnchoseImgId; public HeadViewPager(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
// TODO Auto-generated constructor stub
creatView(context);
} public HeadViewPager(Context context, AttributeSet attrs) {
super(context, attrs);
// TODO Auto-generated constructor stub
creatView(context);
} public HeadViewPager(Context context) {
super(context);
// TODO Auto-generated constructor stub
creatView(context);
} public HeadViewPager(Context context,List<MyImageView> imgageList) {
super(context);
// TODO Auto-generated constructor stub
creatView(context,imgageList);
} public void creatView(Context context){
this.mContext = context;
LayoutInflater.from(context).inflate(R.layout.head_view_pager, this);
mViewPager = (ViewPager)findViewById(R.id.viewpager);
mViewGroup = (ViewGroup)findViewById(R.id.viewgroup);
mImageViews = new ArrayList<MyImageView>();
mImageIds = new ArrayList<Integer>();
tips = new ArrayList<ImageView>();
tipsChoseImgId = R.drawable.img_bg_chose;
tipsUnchoseImgId = R.drawable.img_bg_unchose;
build();
} public void creatView(Context context,List<MyImageView> imgageList){
this.mContext = context;
LayoutInflater.from(context).inflate(R.layout.head_view_pager, this);
mViewPager = (ViewPager)findViewById(R.id.viewpager);
mViewGroup = (ViewGroup)findViewById(R.id.viewgroup);
mImageViews = imgageList;
mImageIds = new ArrayList<Integer>();
tips = new ArrayList<ImageView>();
tipsChoseImgId = R.drawable.img_bg_chose;
tipsUnchoseImgId = R.drawable.img_bg_unchose;
build();
} public void build(){
buildTips();
mViewPager.setAdapter(new HeadViewPagerAdapter(mContext,mImageViews));
//设置默认显示页面为第0页
mViewPager.setCurrentItem();
//设置选择页面时的动画
mViewPager.setPageTransformer(true, new HeadViewPagerTransformer());
//设置缓存View的个数,默认是3个,这表示缓存了5个
mViewPager.setOffscreenPageLimit();
//页面发生改变的监听器
mViewPager.setOnPageChangeListener(new OnPageChangeListener() {
//选择发生改变
@Override
public void onPageSelected(int arg0) {
// TODO Auto-generated method stub
changeTips(arg0);
}
//有滑动操作
@Override
public void onPageScrolled(int arg0, float arg1, int arg2) {
// TODO Auto-generated method stub }
//滑动操作或选择改变
@Override
public void onPageScrollStateChanged(int arg0) {
// TODO Auto-generated method stub }
}); }
//初始化底部导航圆点条
public void buildTips(){
for (int i = ; i < mImageViews.size() ; i ++){
ImageView imageView = new ImageView(mContext);
imageView.setLayoutParams(new LayoutParams(,));
if(i == ){
imageView.setBackgroundResource(tipsChoseImgId);
}else{
imageView.setBackgroundResource(tipsUnchoseImgId);
}
LinearLayout.LayoutParams layoutParams = new LinearLayout.LayoutParams(new ViewGroup.LayoutParams(,));
layoutParams.leftMargin = ;
layoutParams.rightMargin = ;
tips.add(imageView);
mViewGroup.addView(imageView, layoutParams);
}
}
//当选定的页面发生改变时,导航条也对应改变
public void changeTips(int index){
for (int i = ; i < tips.size() ; i ++){
if(i == index){
tips.get(i).setBackgroundResource(tipsChoseImgId);
}else{
tips.get(i).setBackgroundResource(tipsUnchoseImgId);
}
}
} public Context getmContext() {
return mContext;
} public void setmContext(Context mContext) {
this.mContext = mContext;
} public ViewPager getmViewPager() {
return mViewPager;
} public void setmViewPager(ViewPager mViewPager) {
this.mViewPager = mViewPager;
} public List<MyImageView> getmImageViews() {
return mImageViews;
}
//改变图片队列时,要更新整个viewPager
public void setmImageViews(List<MyImageView> mImageViews) {
this.mImageViews = mImageViews;
this.mViewPager.notify();
this.mViewPager.setCurrentItem(); } public ViewGroup getmViewGroup() {
return mViewGroup;
} public void setmViewGroup(ViewGroup mViewGroup) {
this.mViewGroup = mViewGroup;
} public int getTipsChoseImgId() {
return tipsChoseImgId;
} public void setTipsChoseImgId(int tipsChoseImgId) {
this.tipsChoseImgId = tipsChoseImgId;
} public int getTipsUnchoseImgId() {
return tipsUnchoseImgId;
} public void setTipsUnchoseImgId(int tipsUnchoseImgId) {
this.tipsUnchoseImgId = tipsUnchoseImgId;
}
}

样例项目资源:

http://download.csdn.NET/detail/nickey_1314/8932807

点击打开链接

Android 自定义viewpager 三张图片在同一屏幕轮播的效果的更多相关文章

  1. Android 使用ViewPager 做的半吊子的图片轮播

    Android 使用ViewPager 做的半吊子的图片轮播 效果图 虽然不咋样,但是最起码的功能是实现了,下面我们来一步步的实现它. 界面 下面我们来分析一下界面的构成 整体的布局: 因为我们要做出 ...

  2. Android项目实战(四十七):轮播图效果Viewpager

    简易.常用的轮播图效果ViewPager ,老技术了,记一笔留着以后ctrl C + ctrl V    需求如下: 不定张个数的ImagView轮播,右下角显示轮播点图标,每隔固定时间切换下一张,最 ...

  3. Android使用ViewPager实现左右循环滑动及轮播效果

    边界的时候会看到一个不能翻页的动画,可能影响用户体验.此外,某些区域性的ViewPager(例如展示广告或者公告之类的ViewPager),可能需要自动轮播的效果,即用户在不用滑动的情况下就能够看到其 ...

  4. 巧用ViewPager 打造不一样的广告轮播切换效果

    一.概述 如果大家关注了我的微信公众号的话,一定知道我在5月6号的时候推送了一篇文章,文章名为Android超高仿QQ附近的人搜索展示(一),通过该文可以利用ViewPager实现单页显示多个Item ...

  5. Android使用ViewPager实现无限循环滑动及轮播(附源代码)

    MainActivity例如以下: package cc.ww; import java.util.ArrayList; import android.app.Activity; import and ...

  6. Android自定义视图三:给自定义视图添加“流畅”的动画

    这个系列是老外写的,干货!翻译出来一起学习.如有不妥,不吝赐教! Android自定义视图一:扩展现有的视图,添加新的XML属性 Android自定义视图二:如何绘制内容 Android自定义视图三: ...

  7. Android中使用开源框架android-image-indicator实现图片轮播部署

    之前的博文中有介绍关于图片轮播的实现方式,分别为(含超链接): 1.<Android中使用ViewFlipper实现屏幕切换> 2.<Android中使用ViewPager实现屏幕页 ...

  8. Android开发工程师文集-Fragment,适配器,轮播图,ScrollView,Gallery 图片浏览器,Android常用布局样式

    Android开发工程师文集-Fragment,适配器,轮播图,ScrollView,Gallery 图片浏览器,Android常用布局样式 Fragment FragmentManager frag ...

  9. android 自定义Viewpager实现无限循环

    ; i < imageUrls.length; i ++){ ADInfo info = new ADInfo(); info.setUrl(imageUrls[i]); info.setCon ...

随机推荐

  1. WinForm容器内控件批量效验是否同意为空?设置是否仅仅读?设置是否可用等方法分享

    WinForm容器内控件批量效验是否同意为空?设置是否仅仅读?设置是否可用等方法分享 在WinForm程序中,我们有时须要对某容器内的全部控件做批量操作.如批量推断是否同意为空?批量设置为仅仅读.批量 ...

  2. 研读:AirBag Boosting Smartphone Resistance to Malware Infection

  3. 云server之间实时文件同步和文件备份的最简单高效的免费方案

     分布于不同云计算中心的多台云server,通常须要进行文件同步.以满足业务的须要. 传统的文件同步方案,部署繁琐.同步实时性差.无法令人惬意. 端端Clouduolc,一款纯p2p方式的文件实时 ...

  4. 132.try throw catch介绍

    #include <iostream> using namespace std; //try尝试执行,抛出throw,throw之后语句不再执行 //catch处理throw的异常 voi ...

  5. Java多线程编程模式实战指南(二):Immutable Object模式--转载

    本文由本人首次发布在infoq中文站上:http://www.infoq.com/cn/articles/java-multithreaded-programming-mode-immutable-o ...

  6. 理解class.forName()(good--字节码层面)

    使用jdbc方式连接数据库时会使用一句代码Class.forName(String className).这句话是什么意思呢?首先说一点Class.forName(String className)这 ...

  7. Uva 1605 Building for UN【构造法】

    题意:给出n个国家,给它们分配办公室,使得任意两个国家都有一对相邻的格子 看的紫书,最开始看的时候不理解 后来还是搜了题解--- 发现是这样的 比如说5个国家 应该输出 AAAA BBBB CCCC ...

  8. SpringBoot学习笔记(5)----SpringBoot中异常处理的三种方法

    对于异常的处理,Spring Boot中提供默认的一个异常处理界面,如下图: 但是在实际的运用开发中,这样的页面显然是不友好的,Spring Boot也提供了自定义异常处理的方式,如下总结三种一场处理 ...

  9. Dropout 上

    From <白话深度学习与TensorFlow> Dropout 顾名思义是“丢弃”,在一轮训练阶段丢弃一部分网络节点,比如可以在其中的某些层上临时关闭一些节点,让他们既不输入也不输出,这 ...

  10. Unknown column 't_user.id' in 'where clause'(通过字段名删除不了数据)

    创建员工信息表t_user CREATE TABLE t_user( id INT PRIMARY KEY AUTO_INCREMENT, username VARCHAR(20) , passwor ...