Android优质索尼滚动相册

虽然索尼手机卖的不怎么样,但是有些东西还是做的挺好的,工业设计就不用说了,索尼的相册的双指任意缩放功能也是尤其炫酷。其桌面小部件滚动相册我觉得也挺好的,比谷歌原生的相册墙功能好多了,网上搜了一下也没发现有人写这个,于是,下面就介绍下我的高A货。

首先是效果图:

主要手势操作有:

1.上/下满速移动,可以上滑/下滑一张图片
 2.上/下快读移动,则根据滑动速度,上滑/下滑多张图片
 3.单击则请求系统图库展示该图片

该小部件的主要优点:在屏幕内的小范围内提供一个很好的图片选择/浏览部件,尤其是切换图片时有很强的靠近/远离动画感,增加好感。

代码分析

刚开始想这个小部件的时候以为是利用多个ImageView叠加实现的效果,例如谷歌原生的该部件就是利用多个ImageView叠加形成的,但是效果远比不上这个。但觉得通过多个ImageView叠加可能会没这么流畅,性能上也不好。该效果本身也比较规律,应该可以通过一个View来实现,达到更好的性能。于是通过View Hierarchy分析,sony这个果然是通过一个View实现的,于是通过如下方式这个小部件。

代码主要由三个部分组成:

•RollImageView:实际的View
 •CellCalculater:用来实时计算每张图片的绘制区域以及透明度,这个是本小部件的核心部件。接口定义如下:  

/**
  * get all rects for drawing image
  * @return
  */
 public Cell[] getCells();

 /**
  *
  * @param distance the motion distance during the period from ACTION_DOWN to this moment
  * @return 0 means no roll, positive number means roll forward and negative means roll backward
  */
 public int setStatus(float distance);

 /**
  * set the dimen of view
  * @param widht
  * @param height
  */
 public void setDimen(int widht, int height);

 /**
  * set to the status for static
  */
 public void setStatic();

•ImageLoader:用来加载图片,提供Bitmap给RollImageView绘制。接口定义如下:  

/**
  * the images shown roll forward
  */
 public void rollForward();

 /**
  * the images shown roll backward
  */
 public void rollBackward();

 /**
  * get bitmaps
  * @return
  */
 public Bitmap[] getBitmap();

 /**
  * use invalidate to invalidate the view
  * @param invalidate
  */
 public void setInvalidate(RollImageView.InvalidateView invalidate);

 /**
  * set the dimen of view
  * @param width
  * @param height
  */
 public void setDimen(int width, int height);

 /**
  * the image path to be show
  * @param paths
  */
 public void setImagePaths(List<String> paths);

 /**
  * get large bitmap while static
  */
 public void loadCurrentLargeBitmap();

下面分析每个部分的核心代码。

RollImageView
View的主要职责是draw各个bitmap以及响应用户的手势操作,相对比较简单。
绘制部分就是把从ImageLoader获得的的各个Bitmap按照从CellCalculater中获得的绘制区域以及透明度绘制到屏幕上,目前本代码实现的比较简单,没有考虑不同尺寸的图片需要进行一些更加协调的显示方式,比如像ImageView.ScaleType中定义的一些显示方式。  

@Override
 public void onDraw(Canvas canvas) {
  super.onDraw(canvas);
  Bitmap[] bitmaps = mImageLoader.getBitmap();
  Cell[] cells = mCellCalculator.getCells(); //得到每张Image的显示区域与透明度
  canvas.translate(getWidth() / 2, 0);
  for (int i = SHOW_CNT - 1; i >= 0; i--) { //从最底层的Image开始绘制
   Bitmap bitmap = bitmaps[i];
   Cell cell = cells[i];
   if (bitmap != null && !bitmap.isRecycled()) {
    mPaint.setAlpha(cell.getAlpha());
    LOG("ondraw " + i + bitmap.getWidth() + " " + cell.getRectF() + " alpha " + cell.getAlpha());
    canvas.drawBitmap(bitmap, null, cell.getRectF(), mPaint);
   }
  }
 }

手势部分采用了GestureListener,主要代码如下:  

@Override
 public boolean onTouchEvent(MotionEvent event) {
  if (event.getPointerCount() > 1) {
   return false;
  }
  mGestureDetector.onTouchEvent(event);
  switch (event.getAction()) {
   case MotionEvent.ACTION_UP: //这里主要用于处理没有触发Fling事件时,使界面保持没有移动的状态
    if(!mIsFling){
     if(mRollResult == CellCalculator.ROLL_FORWARD){
      mImageLoader.rollForward();
     } else if (mRollResult == CellCalculator.ROLL_BACKWARD && !mScrollRollBack){
      mImageLoader.rollBackward();
     }
     LOG("OnGestureListener ACTION_UP setstatic " );
     mCellCalculator.setStatic();
     mImageLoader.loadCurrentLargeBitmap();
    }
    break;
   default:
    break;
  }
  return true;
 }

 //缓慢拖动
 @Override
 public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {
  mScrollDistance += distanceY;
  if(mScrollDistance > 0 && !mScrollRollBack){
   mImageLoader.rollBackward();
   mScrollRollBack = true;
  } else if(mScrollDistance < 0 && mScrollRollBack){
   mImageLoader.rollForward();
   mScrollRollBack = false;
  }
  LOG("OnGestureListener onScroll " + distanceY + " all" + mScrollDistance);
  mRollResult = mCellCalculator.setStatus(-mScrollDistance);
  invalidate();
  return true;
 }

 //快速拖动
 @Override
 public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
  if (Math.abs(velocityY) > MIN_FLING) {
   LOG("OnGestureListener onFling " + velocityY);
   if (mExecutorService == null) {
    mExecutorService = Executors.newSingleThreadExecutor();
   }
   mIsFling = true;
   mExecutorService.submit(new FlingTask(velocityY));
  }
  return true;
 }

 //利用一个异步任务来处理滚动多张Images
 private class FlingTask implements Runnable {

  float mVelocity;
  float mViewHeight;
  int mSleepTime;
  boolean mRollBackward;

  FlingTask(float velocity) {
   mRollBackward = velocity < 0 ? true : false;
   mVelocity = Math.abs(velocity / 4);
   mViewHeight = RollImageView.this.getHeight() / 2;
   mSleepTime = (int)(4000 / Math.abs(velocity) * 100); //the slower velocity of fling, the longer interval for roll
  }

  @Override
  public void run() {
   int i = 0;
   try{
    while (mVelocity > mViewHeight) {
     mCellCalculator.setStatus(mRollBackward ? -mViewHeight : mViewHeight);
     mHandler.sendEmptyMessage(MSG_INVALATE);
     //determines the count of roll. The using of mViewHeight has no strictly logical
     mVelocity -= mViewHeight;
     if (((i++) & 1) == 0) { //roll forward once for every two setStatus
      if(mRollBackward){
       mImageLoader.rollBackward();
      }else {
       mImageLoader.rollForward();
      }
     }
     Thread.sleep(mSleepTime);
    }
    mCellCalculator.setStatic();
    mImageLoader.loadCurrentLargeBitmap();
    mHandler.sendEmptyMessage(MSG_INVALATE);
   } catch(Exception e){

   } finally{

   }
  }
 }

CellCalculater分析

首先阐明下向前移动/向后移动的概念。需要显示的图片路径存储为一个List,假设显示在最前的图片的索引为index,则当前显示的图片为[index,index+3],向前则表示index加1,向后则表示index减1.

CellCalculater的计算情形主要在于用户通过手势操作,表达了需要向前或者向后移动一张图片的意图。在View中能够获取到的只是手势移动的距离,所以在CellCalculater中需要对传进来的移动距离进行处理,输出移动结果。在我的实现中,当移动距离超过图片高度一半的时候,就表示显示的图片需要移动一位,否则当手势操作结束的时候就设置为static状态。主要代码如下:  

public DefaultCellCalculator(int showCnt){
  mCnt = showCnt;
  mCells = new Cell[mCnt];
  mAlphas = new float[mCnt];
  STATIC_ALPHA = new int[mCnt];
  STATIC_ALPHA[mCnt - 1] = 0; //最后一张图的透明度为0
  int alphaUnit = (255 - FIRST_ALPHA) / (mCnt - 2);
  for(int i = mCnt - 2; i >= 0; i--){ //定义静态时每张图的透明度
   STATIC_ALPHA[i] = FIRST_ALPHA + (mCnt - 2 - i) * alphaUnit;
  }
 }

 @Override
 public Cell[] getCells() {
  return mCells;
 }

 //用户手势移动,distance表示移动距离,正负值分别意味着需要向前/向后移动
 @Override
 public int setStatus(float distance) {
  if(distance > 0){
   return calculateForward(distance);
  } else if(distance < 0){
   return calculateBackward(distance);
  } else{
   initCells();
  }
  return 0;
 }

 //设置RollImageView的尺寸,从而计算合适的显示区域
 @Override
 public void setDimen(int widht, int height) {
  mViewWidth = widht;
  mViewHeight = height;
  mWidhtIndent = (int)(WIDHT_INDENT * mViewWidth);
  mWidths = new int[mCnt];
  for(int i = 0; i < mCnt; i++){
   mWidths[i] = mViewWidth - i * mWidhtIndent;
  }
  //每张图片的高度。
  //假如显示四张图,那么在上面会有三个高度落差,然后最底部保留一个高度落差,所以是mcnt-1
  mImageHeight = mViewHeight - (mCnt - 1) * HEIGHT_INDENT;
  LOG("mImageHeight " + mImageHeight);
  initCells();
 }

 //静态时,即用户手势操作结束时
 @Override
 public void setStatic() {
  initCells();
 }

 //用户有需要向前移动一位的趋势
 private int calculateForward(float status){
  float scale = status / mImageHeight;
  LOG("scale " + scale + " mImageHeight " + mImageHeight + " status " + status);
  for(int i = 1; i < mCnt; i++){
   mCells[i].setWidth(interpolate(scale * 3, mWidths[i], mWidths[i - 1])); // *3 使得后面的宽度快速增大,经验值
   mCells[i].moveVertical(interpolate(scale * 10, 0, HEIGHT_INDENT)); //*10使得后面的图片迅速向前,向前的动画感更强
   mCells[i].setAlpha((int)interpolate(scale, STATIC_ALPHA[i], STATIC_ALPHA[i - 1]));
  }
  mCells[0].moveVertical(status);
  mCells[0].setAlpha((int)interpolate(scale, 255, 0));
  if(status >= mImageHeight / 3){
   return ROLL_FORWARD;
  } else {
   return 0;
  }
 }

 //用户有需要向后移动一位的趋势
 private int calculateBackward(float status){
  float scale = Math.abs(status / mImageHeight);
  for(int i = 1; i < mCnt; i++){
   mCells[i].setWidth(interpolate(scale, mWidths[i - 1], mWidths[i]));
   mCells[i].moveVertical(-scale * HEIGHT_INDENT);
   mCells[i].setAlpha((int)interpolate(scale, STATIC_ALPHA[i - 1], STATIC_ALPHA[i]));
  }
  mCells[0].resetRect();
  mCells[0].setWidth(mWidths[0]);
  mCells[0].setHeight(mImageHeight);
  mCells[0].moveVertical(mImageHeight + status);
  mCells[0].setAlpha((int)interpolate(scale, 0, 255));
  if(-status >= mImageHeight / 3){
   return ROLL_BACKWARD;
  } else {
   return 0;
  }
 }

 /**
  * status without move
  */
 private void initCells(){
  int top = -HEIGHT_INDENT;
  for(int i = 0; i < mCnt; i++){
   RectF rectF = new RectF(0,0,0,0);
   rectF.top = top + (mCnt - 1 - i) * HEIGHT_INDENT;
   rectF.bottom = rectF.top + mImageHeight;
   mCells[i] = new Cell(rectF, STATIC_ALPHA[i]);
   mCells[i].setWidth(mWidths[i]);
  }
 }

 //计算差值
 private float interpolate(float scale, float start, float end){
  if(scale > 1){
   scale = 1;
  }
  return start + scale * (end - start);
 }

ImageLoader分析
ImageLoader其实比较简单,主要有如下两点:
 •响应手势操作,处理对应的向前/向后移动时的Bitmap请求
 •当手势还在操作时,应该加载小图,等手势操作结束之后,应该加载大图。因为只有缓慢移动时,需要清晰显示,而快速移动时,显示小图即可,所以需要加载当前index以及向前向后一张图即可。

//加载当前index以及向前向后三张大图
 @Override
 public void loadCurrentLargeBitmap() {
  for(int i = mCurrentIndex - 1; i < mCurrentIndex + 2; i++){
   if(i >= 0 && i < mImagesCnt - 1){
    mBitmapCache.getLargeBitmap(mAllImagePaths[i]);
   }
  }
 }

 //index向前移动一位
 @Override
 public void rollForward() {
  LOG("rollForward");
  mCurrentIndex++;
  if(mCurrentIndex > mImagesCnt - 1){
   mCurrentIndex = mImagesCnt - 1;
  }
  setCurrentPaths();
 }

 //index向后移动一位
 @Override
 public void rollBackward() {
  LOG("rollBackward");
  mCurrentIndex--;
  if(mCurrentIndex < 0){
   mCurrentIndex = 0;
  }
  setCurrentPaths();
 }

 @Override
 public Bitmap[] getBitmap() {
  if(mCurrentPaths != null){
   LOG("getBitmap paths nut null");
   for(int i = mCurrentIndex, j = 0; j < mShowCnt; j++, i++){
    if(i >= 0 && i < mImagesCnt){
     mCurrentBitmaps[j] = mBitmapCache.getBimap(mAllImagePaths[i]);
    } else{
     mCurrentBitmaps[j] = mBitmapCache.getBimap(NO_PATH);
    }
   }
  }
  return mCurrentBitmaps;
 }

最后,所有源代码:https://github.com/willhua/RollImage

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android Zipalign工具优化Android APK应用

    生成的Android应用APK文件最好进行优化,因为APK包的本质是一个zip压缩文档,经过优化能使包内未压缩的数据有序的排列,从而减少应用程序运行时的内存消耗.我们可以使用Zipalign工具进行APK优化. 据Android官方网站的说明,Zipalign是一款重要的优化APK应用程序的工具. 多数软件开发商在正式推出其Android应用程序,都使用Zipalign工具优化APK包.但是,仍然有一些应用程序需要我们自己动手进行Zipalign优化,例如一些个人开发的软件.一些破解版的软件.

  • Android样式的开发:layer-list实例详解

    上图Tab的背景效果,和带阴影的圆角矩形,是怎么实现的呢?大部分的人会让美工切图,用点九图做背景.但是,如果只提供一张图,会怎么样呢?比如,中间的Tab背景红色底线的像素高度为4px,那么,在mdpi设备上显示会符合预期,在hdpi设备上显示时会细了一点点,在xhdpi设备上显示时会再细一点,在xxhdpi上显示时又细了,在xxxhdpi上显示时则更细了.因为在xxxhdpi上,1dp=4px,所以,4px的图,在xxxhdpi设备上显示时,就只剩下1dp了.所以,为了适配好各种分辨率,必须提供

  • Android APK优化工具Zipalign详解

    Android SDK中包含了一个用于优化APK的新工具zipalign.它提高了优化后的Applications与Android系统的交互效率(俗话:"要致富先修路",Android小组重新为Applications与Android系统之间搭建了一条高速公路),从而可以使整个系统的运行速度有了较大的提升.Android小组强烈建议开发者在发布新Apps之前使用zipalign优化工具,而且对于已经发布但不受限于系统版本的Apps,建议用优化后的APK替换现有的版本. 在下面的内容中将

  • 利用Android中BitmapShader制作自带边框的圆形头像

    效果如下: BitmapShader 的简单介绍 关于 Shader是什么,Shader的种类有哪几种以及如何使用不属于本文范畴,对这方面不是很了解的同学,建议先去学习一下 Shader 的基本使用. BitmapShader主要的作用就是 通过Paint对象,对 画布进行指定的Bitmap填充,实现一系列效果,可以有以下三种模式进行选择 1.CLAMP - 拉伸,这里拉伸的是图片的最后一个元素,不断地重复,这个效果,在图片比较小,而所要画的面积比较大的时候会比较明显. 2.REPEAT - 重

  • Android中使用ViewStub实现布局优化

    在Android开发中,View是我们必须要接触的用来展示的技术.通常情况下随着View视图的越来越复杂,整体布局的性能也会随之下降.这里介绍一个在某些场景下提升布局性能的View,它就是ViewStub. ViewStub是什么 ViewStub是View的子类 它不可见,大小为0 用来延迟加载布局资源 注,关于Stub的解释 A stub is a small program routine that substitutes for a longer program, possibly to

  • Android ViewPager实现Banner循环播放

    问题的起源 在项目里,有时候需要实现一个图片轮播的效果,用来展示Banner.同时,图片能循环播放,下面还有一排小圆点来指示当前轮播到哪一页了. 如下图: 分析 · 图片的个数是会变化的,同时小圆点的个数也会跟着图片个数变化 · 每一个page的布局是一样的.变化的就是小圆点的个数,所以需要用代码来动态生成小圆点 编码 布局 首先完成 MainActivity 的布局 activity_main.xml <RelativeLayout xmlns:android="http://schem

  • Android 动态高斯模糊效果教程

    写在前面 最近一直在做毕设项目的准备工作,考虑到可能要用到一个模糊的效果,所以就学习了一些高斯模糊效果的实现.比较有名的就是 FastBlur 以及它衍生的一些优化方案,还有就是今天要说的RenderScript . 因为这东西是现在需要才去学习的,所以关于一些图像处理和渲染问题就不提了.不过在使用的过程中确实能感受到,虽然不同的方案都能实现相同的模糊效果,但是效率差别真的很大. 本篇文章实现的高斯模糊是根据下面这篇文章学习的,先推荐一下.本文内容与其内容差不多,只是稍微讲的详细一点,并修改了代

  • Android自定义圆形倒计时进度条

    效果预览 源代码传送门:https://github.com/yanzhenjie/CircleTextProgressbar 实现与原理 这个文字圆形的进度条我们在很多APP中看到过,比如APP欢迎页倒计时,下载文件倒计时等. 分析下原理,可能有的同学一看到这个自定义View就慌了,这个是不是要继承View啊,是不是要绘制啊之类的,答案是:是的.但是我们也不要担心,实现这个效果实在是so easy.下面就跟我一起来看看核心分析和代码吧. 原理分析 首先我们观察上图,需要几个部分组成: 1. 外

  • AndroidSDK Support自带夜间、日间模式切换详解

    写这篇博客的目的就是教大家利用AndroidSDK自带的support lib来实现APP日间/夜间模式的切换,最近看到好多帖子在做关于这个日夜间模式切换的开源项目,其实AndroidSDK Support中已经有了非常好的支持了. 本文demo下载地址在文章的末尾,看完文档如果还不能实现可以下载玩玩. -------------------------------------------------------------------------------- 效果演示 左是Android 4

  • Android 逆向学习详解及实例

    断断续续的总算的把android开发和逆向的这两本书看完了,虽然没有java,和android开发的基础,但总体感觉起来还是比较能接收的,毕竟都是触类旁通的.当然要深入的话还需要对这门语言的细节特性和奇技淫巧进行挖掘. 这里推荐2本书,个人觉得对android开发入门和android逆向入门比较好的教材: <google android 开发入门与实战> <android 软件安全与逆向分析> 1. 我对android逆向的认识 因为之前有一些windows逆向的基础,在看andr

随机推荐