`

使用Android自带Gallery组件实现CoverFlow

阅读更多

首先声明,源代码转载自国外Neil Davies,使用Apache2.0开源协议,请使用源代码的人自觉遵守协议内容。
本文为Kearnel原创,转载请注明出处。
http://www.eoeandroid.com/thread-39709-1-1.html
以下是正文:

使用过Android自带的gallery组件的人都知道,gallery实现的效果就是拖动浏览一组图片,相比iphone里也是用于拖动浏览图片的coverflow,显然逊色不少。实际上,可以通过扩展gallery,通过伪3D变换可以基本实现coverflow的效果。本文通过源代码解析这一功能的实现。具体代码作用可参照注释。

最终实现效果如下:






要使用gallery,我们必须首先给其指定一个adapter。在这里,我们实现了一个自定义的ImageAdapter,为图片制作倒影效果。
传入参数为context和程序内drawable中的图片ID数组。之后调用其中的createReflectedImages()方法分别创造每一个图像的倒影效果,生成对应的ImageView数组,最后在getView()中返回。

  1. /*
  2. * Copyright (C) 2010 Neil Davies
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. *
  16. * This code is base on the Android Gallery widget and was Created
  17. * by Neil Davies neild001 'at' gmail dot com to be a Coverflow widget
  18. *
  19. * @author Neil Davies
  20. */
  21. public class ImageAdapter extends BaseAdapter {
  22. int mGalleryItemBackground;
  23. private Context mContext;

  24. private Integer[] mImageIds ;

  25. private ImageView[] mImages;

  26. public ImageAdapter(Context c, int[] ImageIds) {
  27. mContext = c;
  28. mImageIds = ImageIds;
  29. mImages = new ImageView[mImageIds.length];
  30. }

  31. public boolean createReflectedImages() {
  32. // The gap we want between the reflection and the original image
  33. final int reflectionGap = 4;

  34. int index = 0;
  35. for (int imageId : mImageIds) {
  36. Bitmap originalImage = BitmapFactory.decodeResource(
  37. mContext.getResources(), imageId);
  38. int width = originalImage.getWidth();
  39. int height = originalImage.getHeight();

  40. // This will not scale but will flip on the Y axis
  41. Matrix matrix = new Matrix();
  42. matrix.preScale(1, -1);

  43. // Create a Bitmap with the flip matrix applied to it.
  44. // We only want the bottom half of the image
  45. Bitmap reflectionImage = Bitmap.createBitmap(originalImage, 0,
  46. height / 2, width, height / 2, matrix, false);

  47. // Create a new bitmap with same width but taller to fit
  48. // reflection
  49. Bitmap bitmapWithReflection = Bitmap.createBitmap(width,
  50. (height + height / 2), Config.ARGB_8888);

  51. // Create a new Canvas with the bitmap that's big enough for
  52. // the image plus gap plus reflection
  53. Canvas canvas = new Canvas(bitmapWithReflection);
  54. // Draw in the original image
  55. canvas.drawBitmap(originalImage, 0, 0, null);
  56. // Draw in the gap
  57. Paint deafaultPaint = new Paint();
  58. canvas.drawRect(0, height, width, height + reflectionGap,
  59. deafaultPaint);
  60. // Draw in the reflection
  61. canvas.drawBitmap(reflectionImage, 0, height + reflectionGap,
  62. null);

  63. // Create a shader that is a linear gradient that covers the
  64. // reflection
  65. Paint paint = new Paint();
  66. LinearGradient shader = new LinearGradient(0,
  67. originalImage.getHeight(), 0,
  68. bitmapWithReflection.getHeight() + reflectionGap,
  69. 0x70ffffff, 0x00ffffff, TileMode.CLAMP);
  70. // Set the paint to use this shader (linear gradient)
  71. paint.setShader(shader);
  72. // Set the Transfer mode to be porter duff and destination in
  73. paint.setXfermode(new PorterDuffXfermode(Mode.DST_IN));
  74. // Draw a rectangle using the paint with our linear gradient
  75. canvas.drawRect(0, height, width,
  76. bitmapWithReflection.getHeight() + reflectionGap, paint);

  77. ImageView imageView = new ImageView(mContext);
  78. imageView.setImageBitmap(bitmapWithReflection);
  79. imageView
  80. .setLayoutParams(new GalleryFlow.LayoutParams(160, 240));
  81. // imageView.setScaleType(ScaleType.MATRIX);
  82. mImages[index++] = imageView;

  83. }
  84. return true;
  85. }

  86. public int getCount() {
  87. return mImageIds.length;
  88. }

  89. public Object getItem(int position) {
  90. return position;
  91. }

  92. public long getItemId(int position) {
  93. return position;
  94. }

  95. public View getView(int position, View convertView, ViewGroup parent) {

  96. // Use this code if you want to load from resources
  97. /*
  98. * ImageView i = new ImageView(mContext);
  99. * i.setImageResource(mImageIds[position]); i.setLayoutParams(new
  100. * CoverFlow.LayoutParams(350,350));
  101. * i.setScaleType(ImageView.ScaleType.CENTER_INSIDE);
  102. *
  103. * //Make sure we set anti-aliasing otherwise we get jaggies
  104. * BitmapDrawable drawable = (BitmapDrawable) i.getDrawable();
  105. * drawable.setAntiAlias(true); return i;
  106. */

  107. return mImages[position];
  108. }

  109. /**
  110. * Returns the size (0.0f to 1.0f) of the views depending on the
  111. * 'offset' to the center.
  112. */
  113. public float getScale(boolean focused, int offset) {
  114. /* Formula: 1 / (2 ^ offset) */
  115. return Math.max(0, 1.0f / (float) Math.pow(2, Math.abs(offset)));
  116. }

  117. }
  118. }
复制代码




仅仅实现了图片的倒影效果还不够,因为在coverflow中图片切换是有旋转和缩放效果的,而自带的gallery中并没有实现。因此,我们扩展自带的gallery,实现自己的galleryflow。在原gallery类中,提供了一个方法getChildStaticTransformation()以实现对图片的变换。我们通过覆写这个方法并在其中调用自定义的transformImageBitmap(“每个图片与gallery中心的距离”)方法,,即可实现每个图片做相应的旋转和缩放。其中使用了camera和matrix用于视图变换。具体可参考代码注释。


  1. public class GalleryFlow extends Gallery {

  2.         /**
  3.          * Graphics Camera used for transforming the matrix of ImageViews
  4.          */
  5.         private Camera mCamera = new Camera();

  6.         /**
  7.          * The maximum angle the Child ImageView will be rotated by
  8.          */
  9.         private int mMaxRotationAngle = 60;

  10.         /**
  11.          * The maximum zoom on the centre Child
  12.          */
  13.         private int mMaxZoom = -120;

  14.         /**
  15.          * The Centre of the Coverflow
  16.          */
  17.         private int mCoveflowCenter;

  18.         public GalleryFlow(Context context) {
  19.                 super(context);
  20.                 this.setStaticTransformationsEnabled(true);
  21.         }

  22.         public GalleryFlow(Context context, AttributeSet attrs) {
  23.                 super(context, attrs);
  24.                 this.setStaticTransformationsEnabled(true);
  25.         }

  26.         public GalleryFlow(Context context, AttributeSet attrs, int defStyle) {
  27.                 super(context, attrs, defStyle);
  28.                 this.setStaticTransformationsEnabled(true);
  29.         }

  30.         /**
  31.          * Get the max rotational angle of the image
  32.          *
  33.          * @return the mMaxRotationAngle
  34.          */
  35.         public int getMaxRotationAngle() {
  36.                 return mMaxRotationAngle;
  37.         }

  38.         /**
  39.          * Set the max rotational angle of each image
  40.          *
  41.          * @param maxRotationAngle
  42.          *            the mMaxRotationAngle to set
  43.          */
  44.         public void setMaxRotationAngle(int maxRotationAngle) {
  45.                 mMaxRotationAngle = maxRotationAngle;
  46.         }

  47.         /**
  48.          * Get the Max zoom of the centre image
  49.          *
  50.          * @return the mMaxZoom
  51.          */
  52.         public int getMaxZoom() {
  53.                 return mMaxZoom;
  54.         }

  55.         /**
  56.          * Set the max zoom of the centre image
  57.          *
  58.          * @param maxZoom
  59.          *            the mMaxZoom to set
  60.          */
  61.         public void setMaxZoom(int maxZoom) {
  62.                 mMaxZoom = maxZoom;
  63.         }

  64.         /**
  65.          * Get the Centre of the Coverflow
  66.          *
  67.          * @return The centre of this Coverflow.
  68.          */
  69.         private int getCenterOfCoverflow() {
  70.                 return (getWidth() - getPaddingLeft() - getPaddingRight()) / 2
  71.                                 + getPaddingLeft();
  72.         }

  73.         /**
  74.          * Get the Centre of the View
  75.          *
  76.          * @return The centre of the given view.
  77.          */
  78.         private static int getCenterOfView(View view) {
  79.                 return view.getLeft() + view.getWidth() / 2;
  80.         }

  81.         /**
  82.          * {@inheritDoc}
  83.          *
  84.          * @see #setStaticTransformationsEnabled(boolean)
  85.          */
  86.         protected boolean getChildStaticTransformation(View child, Transformation t) {

  87.                 final int childCenter = getCenterOfView(child);
  88.                 final int childWidth = child.getWidth();
  89.                 int rotationAngle = 0;

  90.                 t.clear();
  91.                 t.setTransformationType(Transformation.TYPE_MATRIX);

  92.                 if (childCenter == mCoveflowCenter) {
  93.                         transformImageBitmap((ImageView) child, t, 0);
  94.                 } else {
  95.                         rotationAngle = (int) (((float) (mCoveflowCenter - childCenter) / childWidth) * mMaxRotationAngle);
  96.                         if (Math.abs(rotationAngle) > mMaxRotationAngle) {
  97.                                 rotationAngle = (rotationAngle < 0) ? -mMaxRotationAngle
  98.                                                 : mMaxRotationAngle;
  99.                         }
  100.                         transformImageBitmap((ImageView) child, t, rotationAngle);
  101.                 }

  102.                 return true;
  103.         }

  104.         /**
  105.          * This is called during layout when the size of this view has changed. If
  106.          * you were just added to the view hierarchy, you're called with the old
  107.          * values of 0.
  108.          *
  109.          * @param w
  110.          *            Current width of this view.
  111.          * @param h
  112.          *            Current height of this view.
  113.          * @param oldw
  114.          *            Old width of this view.
  115.          * @param oldh
  116.          *            Old height of this view.
  117.          */
  118.         protected void onSizeChanged(int w, int h, int oldw, int oldh) {
  119.                 mCoveflowCenter = getCenterOfCoverflow();
  120.                 super.onSizeChanged(w, h, oldw, oldh);
  121.         }

  122.         /**
  123.          * Transform the Image Bitmap by the Angle passed
  124.          *
  125.          * @param imageView
  126.          *            ImageView the ImageView whose bitmap we want to rotate
  127.          * @param t
  128.          *            transformation
  129.          * @param rotationAngle
  130.          *            the Angle by which to rotate the Bitmap
  131.          */
  132.         private void transformImageBitmap(ImageView child, Transformation t,
  133.                         int rotationAngle) {
  134.                 mCamera.save();
  135.                 final Matrix imageMatrix = t.getMatrix();
  136.                 final int imageHeight = child.getLayoutParams().height;
  137.                 final int imageWidth = child.getLayoutParams().width;
  138.                 final int rotation = Math.abs(rotationAngle);

  139.                 // 在Z轴上正向移动camera的视角,实际效果为放大图片。
  140.                 // 如果在Y轴上移动,则图片上下移动;X轴上对应图片左右移动。
  141.                 mCamera.translate(0.0f, 0.0f, 100.0f);

  142.                 // As the angle of the view gets less, zoom in
  143.                 if (rotation < mMaxRotationAngle) {
  144.                         float zoomAmount = (float) (mMaxZoom + (rotation * 1.5));
  145.                         mCamera.translate(0.0f, 0.0f, zoomAmount);
  146.                 }

  147.                 // 在Y轴上旋转,对应图片竖向向里翻转。
  148.                 // 如果在X轴上旋转,则对应图片横向向里翻转。
  149.                 mCamera.rotateY(rotationAngle);
  150.                 mCamera.getMatrix(imageMatrix);
  151.                 imageMatrix.preTranslate(-(imageWidth / 2), -(imageHeight / 2));
  152.                 imageMatrix.postTranslate((imageWidth / 2), (imageHeight / 2));
  153.                 mCamera.restore();
  154.         }
  155. }
复制代码





代码到这里就结束了。有兴趣的话可以自行调整里面的参数来实现更多更炫的效果。
下面是调用的示例:

  1. public void onCreate(Bundle savedInstanceState) {
  2.                 super.onCreate(savedInstanceState);
  3.                
  4.                
  5.                 setContentView(R.layout.layout_gallery);
  6.                
  7.                 Integer[] images = { R.drawable.img0001, R.drawable.img0030,
  8.                         R.drawable.img0100, R.drawable.img0130, R.drawable.img0200,
  9.                         R.drawable.img0230, R.drawable.img0300, R.drawable.img0330,
  10.                         R.drawable.img0354 };
  11.                
  12.                 ImageAdapter adapter = new ImageAdapter(this, images);
  13.                 adapter.createReflectedImages();

  14.                 GalleryFlow galleryFlow = (GalleryFlow) findViewById(R.id.gallery_flow);
  15.                 galleryFlow.setAdapter(adapter);
  16.                
  17.         }
复制代码






PS1:
可以看出来这样实现的gallery锯齿问题比较严重。可以在createReflectedImages()使用以下代码:

  1. BitmapDrawable bd = new BitmapDrawable(bitmapWithReflection);
  2. bd.setAntiAlias(true);
  3. ...
复制代码

然后用iv.setImageDrawable(bd);
代替iv.setImageBitmap(bitmapWithReflection);
即可基本消除锯齿。

PS2:
ImageAdapter有待确定的MemoryLeak问题,貌似的Bitmap的decode方法会造成ML,使用ImageAdapter时多次旋转屏幕后会出现OOM。目前可以通过将使用完毕的bimap调用recycle()方法和设置null并及时调用system.gc()得到一些改善,但是问题并不明显。

庆祝精华和推荐,增加3个PS~

PS3 ON PS1:
256楼说到,为什么开启抗锯齿后不明显。答案是,锯齿不可能完全消除,但开启抗锯齿后会有很大改善。
另外还说到为什么android不默认开启锯齿,以下是我的一点想法:
插值是我现在所知道的抗锯齿的算法,也就是计算像素间的相关度对其间插入中间像素以达到平滑图像边缘的效果。但这无疑会耗费了大量的运算。
虽然我没有经过测试,但是我猜测,使用antialias后图形性能至少会下降30%。
当然,在这里没有涉及到复杂的图形运算,所以开启抗锯齿不会有很明显的性能影响,但如果你在模拟器或者低端机型上测试就会发现一点问题。


PS4:
有人问到transformImageBitmap()中这俩句话是什么意思:
    imageMatrix.preTranslate(-(imageWidth / 2), -(imageHeight / 2));
    imageMatrix.postTranslate((imageWidth / 2), (imageHeight / 2));
个人的理解如下:
preTranslate相当于在对图像进行任何矩阵变换前先进行preTranslate,postTranslate相反,进行所有变换后再执行postTranlate。
这俩句的意思是:在做任何变换前,先将整个图像从图像的中心点移动到原点((0,0)点),执行变换完毕后再将图像从原点移动到之前的中心点。
如果不加这俩句,任何变换将以图像的原点为变换中心点,加了之后,任何变换都将以图像的中心点为变换中心点。

举个例子,对图像进行旋转,需要俩个参数:一个是旋转的角度,另一个是旋转中心的坐标。旋转中心的坐标影响旋转的效果。这个能明白吗?你拿一根棍子,拿着棍子的一端进行旋转和拿在棍子中间旋转,是不一样的。preTranslate和postTranslate执行后对图像本身不会有影响,影响的是对图像进行变换时的旋转轴。
说了这么多有点绕,其实就是矩阵变换的知识。



PS5 ON PS2:
这个问题在google group下有过很充分的讨论,貌似一般只在debug模式下存在。现在我使用这段代码没有出现OOM问题了。


【示例工程文件见6楼】

欢迎大家转载和评论

 

 

分享到:
评论
1 楼 huxinli 2011-10-10  
效果不错啊,多谢了啊。

相关推荐

    使用Android自带的Gallery控件实现CoverFlow

    在Android开发中,`Gallery`控件是一种非常实用的组件,它可以用来展示一系列横向滑动的项目,类似于图片轮播或者iOS中的CoverFlow效果。在本教程中,我们将深入探讨如何利用Android SDK中的`Gallery`控件来实现类似...

    Android中实现coverflow效果

    本教程将深入探讨如何利用Android的Gallery控件来实现这种酷炫的Coverflow效果。 首先,我们要了解Android的Gallery控件。Gallery是一个水平滚动的视图容器,可以用来展示一系列的项目,如图片或按钮。它的特点是...

    android gallery demo

    从提供的文件名来看,我们可以看到一些关于`Gallery`的源码解析和探究文章,如"使用Android自带Gallery组件实现CoverFlow",这可能涉及到将`Gallery`组件改造为更复杂的CoverFlow效果,类似于iPod的封面翻转。Cover...

    Android实现CoverFlow效果

    下面我们将深入探讨如何在Android中实现CoverFlow效果。 首先,CoverFlow效果的核心在于视图的旋转和平移。在Android中,我们可以使用自定义View或者第三方库来实现这一效果。一种常见的方式是基于`Gallery`组件...

    安卓的伪3D实现,用galler控件实现coverflow,里面都是可以运行的demo

    总的来说,虽然原生的Gallery控件在新版本的Android中已不再推荐使用,但通过自定义和优化,我们依然可以利用它或者其他的视图组件来实现引人注目的Cover Flow效果。这个过程涉及到Android UI编程的多个方面,包括...

    Android 3D CoverFlow 3D Gallery

    "Android 3D CoverFlow 3D Gallery"就是这样一个利用OpenGL ES技术实现的创新性应用组件,它将传统的CoverFlow效果提升到三维空间,为用户呈现更为立体和生动的浏览体验。 CoverFlow最初由Apple公司在iPod和iTunes...

    收集的scorllview和coverflow效果

    在Android开发中,ScrollView和CoverFlow是两种常用的视图组件,它们为用户界面提供了不同的交互体验。ScrollView通常用于实现可滚动的内容区域,而CoverFlow则是一种具有翻页效果的控件,模仿了苹果iOS中的类似功能...

    android实现Iphone的coverFlow图片浏览效果

    可以通过覆写`Gallery`类,然后在其中重写`onTouchEvent()`、`onFling()`等方法,或者使用第三方库如`android-coverflow`来实现更复杂的3D翻转效果。 例如,你可以创建一个新的`Gallery`子类,然后修改其`onDraw()`...

    Android实现CoverFlow效果控件的实例代码

    在Android开发中,实现CoverFlow效果控件是一个常见的需求,它可以为用户界面带来优雅的视觉体验,类似于苹果iOS中的Cover Flow。在这个实例中,我们将会深入理解如何自定义一个这样的控件,包括优化性能和内存管理...

    Android改进版CoverFlow效果控件无限循环

    在原生Android SDK中,并没有内置的CoverFlow控件,因此需要通过自定义ViewGroup,如Gallery或HorizontalScrollView来实现。这个改进版的控件可能是在此基础上进行了优化,解决了原始实现中可能存在的性能问题,并...

    Android coverFlow

    在Android开发中,CoverFlow是一种视觉效果酷炫的界面组件,它模仿了苹果iOS系统中的经典界面元素,用户可以通过滑动来浏览一系列呈现3D旋转效果的图像或卡片。这个组件通常用于展示图片集合,如音乐专辑封面、应用...

    Android 使用Gallery_tabhost实现标签效果图源码.rar

    在这个项目中,`Android 使用Gallery_tabhost实现标签效果图源码.rar`是一个示例,展示了如何结合这两个组件来创建一个动态、交互式的标签切换效果。 `Gallery`是Android提供的一个水平滚动视图,它允许用户通过...

    Android画廊组件实现图片切换

    在Android开发中,画廊组件(Gallery)是一个非常实用的控件,主要用于展示一系列的图片或项目,并允许用户通过左右滑动来切换选择。在这个场景中,我们将探讨如何使用Gallery类以及ImageSwitcher来实现一个典型的...

    Android可循环显示图像的Android Gallery组件用法实例

    需要注意的是,由于`Gallery`在Android API 21及以上版本已被弃用,所以在新版本的Android应用中,开发者通常会使用`RecyclerView`或其他第三方库(如`CarouselView`)来实现类似的功能。但了解`Gallery`组件的使用...

    031_android UI组件之 Gallery画廊控件

    "Gallery画廊控件"是Android SDK中一个独特的视图组件,它允许用户在一个水平滚动的列表中展示项目,通常用于图片或选择项的浏览。在本教程中,我们将深入探讨Gallery控件的用法、属性以及如何自定义它。 首先,...

    android 4.0 Gallery源码

    Gallery使用了ViewPager组件,它是Android 3.0引入的用于实现页面滑动的视图容器。Adapter类则为ViewPager提供数据,负责将图片数据转化为可展示的View。通过重写Adapter的getItem()和getCount()方法,可以实现图片...

    Android学习记录使用Gallery实现炫丽的拖动效果

    在Android开发中,实现炫丽的拖动效果是提升用户体验的重要一环,特别是使用`Gallery`组件可以创建出类似轮播图或者选择器的效果。这篇博客"Android学习记录使用Gallery实现炫丽的拖动效果"深入探讨了如何利用`...

    android--Gallery的实现

    在Android开发中,`Gallery`组件是一个非常有用的控件,它允许用户通过水平滚动来浏览一系列的项目,比如图片、选项卡或者任何其他你想展示的内容。`Gallery`控件是基于`AbsSpinner`类的,提供了类似轮播图的效果,...

Global site tag (gtag.js) - Google Analytics