• Android 一个3D相册源码


    我们专业课有Android的学习,最后老师让做一个简单的Android应用程序.我在网上找些资料,加上自己改造一下做了一个3D相册.

    程序仿照Android的相册功能,调用Gallery类对相片进行浏览.实现3D效果.其中不足的地方就是里面的图片都是写死的.这一点可以改进.程序中使用Png格式图片,注意图片不要太大,500*500一下最好.

    image

    首先:

    GalleryFlow.java 类 用来实现图片放映查看效果.

       1: package com.android.CustomGallery;
       2:  
       3: import android.content.Context;
       4: import android.graphics.Camera;
       5: import android.graphics.Matrix;
       6: import android.util.AttributeSet;
       7: import android.util.Log;
       8: import android.view.View;
       9: import android.view.animation.Transformation;
      10: import android.widget.Gallery;
      11: import android.widget.ImageView;
      12:  
      13: public class GalleryFlow extends Gallery {
      14:  
      15:         /**
      16:          * Graphics Camera used for transforming the matrix of ImageViews
      17:          */
      18:         private Camera mCamera = new Camera();
      19:  
      20:         /**
      21:          * The maximum angle the Child ImageView will be rotated by
      22:          */
      23:         private int mMaxRotationAngle = 60;
      24:  
      25:         /**
      26:          * The maximum zoom on the centre Child
      27:          */
      28:         private int mMaxZoom = -120;
      29:  
      30:         /**
      31:          * The Centre of the Coverflow
      32:          */
      33:         private int mCoveflowCenter;
      34:  
      35:         public GalleryFlow(Context context) {
      36:                 super(context);
      37:                 this.setStaticTransformationsEnabled(true);
      38:         }
      39:  
      40:         public GalleryFlow(Context context, AttributeSet attrs) {
      41:                 super(context, attrs);
      42:                 this.setStaticTransformationsEnabled(true);
      43:         }
      44:  
      45:         public GalleryFlow(Context context, AttributeSet attrs, int defStyle) {
      46:                 super(context, attrs, defStyle);
      47:                 this.setStaticTransformationsEnabled(true);
      48:         }
      49:  
      50:         /**
      51:          * Get the max rotational angle of the image
      52:          * 
      53:          * @return the mMaxRotationAngle
      54:          */
      55:         public int getMaxRotationAngle() {
      56:                 return mMaxRotationAngle;
      57:         }
      58:  
      59:         /**
      60:          * Set the max rotational angle of each image
      61:          * 
      62:          * @param maxRotationAngle
      63:          *            the mMaxRotationAngle to set
      64:          */
      65:         public void setMaxRotationAngle(int maxRotationAngle) {
      66:                 mMaxRotationAngle = maxRotationAngle;
      67:         }
      68:  
      69:         /**
      70:          * Get the Max zoom of the centre image
      71:          * 
      72:          * @return the mMaxZoom
      73:          */
      74:         public int getMaxZoom() {
      75:                 return mMaxZoom;
      76:         }
      77:  
      78:         /**
      79:          * Set the max zoom of the centre image
      80:          * 
      81:          * @param maxZoom
      82:          *            the mMaxZoom to set
      83:          */
      84:         public void setMaxZoom(int maxZoom) {
      85:                 mMaxZoom = maxZoom;
      86:         }
      87:  
      88:         /**
      89:          * Get the Centre of the Coverflow
      90:          * 
      91:          * @return The centre of this Coverflow.
      92:          */
      93:         private int getCenterOfCoverflow() {
      94:                 //Log.e("CoverFlow Width+Height", getWidth() + "*" + getHeight());
      95:                 return (getWidth() - getPaddingLeft() - getPaddingRight()) / 2
      96:                                 + getPaddingLeft();
      97:         }
      98:  
      99:         /**
     100:          * Get the Centre of the View
     101:          * 
     102:          * @return The centre of the given view.
     103:          */
     104:         private static int getCenterOfView(View view) {
     105:                 /*Log.e("ChildView Width+Height", view.getWidth() + "*"
     106:                                 + view.getHeight());*/
     107:                 return view.getLeft() + view.getWidth() / 2;
     108:         }
     109:  
     110:         /**
     111:          * {@inheritDoc}
     112:          * 
     113:          * @see #setStaticTransformationsEnabled(boolean)
     114:          */
     115:         protected boolean getChildStaticTransformation(View child, Transformation t) {
     116:  
     117:                 final int childCenter = getCenterOfView(child);
     118:                 final int childWidth = child.getWidth();
     119:                 int rotationAngle = 0;
     120:  
     121:                 t.clear();
     122:                 t.setTransformationType(Transformation.TYPE_MATRIX);
     123:  
     124:                 if (childCenter == mCoveflowCenter) {
     125:                         transformImageBitmap((ImageView) child, t, 0);
     126:                 } else {
     127:                         rotationAngle = (int) (((float) (mCoveflowCenter - childCenter) / childWidth) * mMaxRotationAngle);
     128:                         if (Math.abs(rotationAngle) > mMaxRotationAngle) {
     129:                                 rotationAngle = (rotationAngle < 0) ? -mMaxRotationAngle
     130:                                                 : mMaxRotationAngle;
     131:                         }
     132:                         transformImageBitmap((ImageView) child, t, rotationAngle);
     133:                 }
     134:  
     135:                 return true;
     136:         }
     137:  
     138:         /**
     139:          * This is called during layout when the size of this view has changed. If
     140:          * you were just added to the view hierarchy, you're called with the old
     141:          * values of 0.
     142:          * 
     143:          * @param w
     144:          *            Current width of this view.
     145:          * @param h
     146:          *            Current height of this view.
     147:          * @param oldw
     148:          *            Old width of this view.
     149:          * @param oldh
     150:          *            Old height of this view.
     151:          */
     152:         protected void onSizeChanged(int w, int h, int oldw, int oldh) {
     153:                 mCoveflowCenter = getCenterOfCoverflow();
     154:                 super.onSizeChanged(w, h, oldw, oldh);
     155:         }
     156:  
     157:         /**
     158:          * Transform the Image Bitmap by the Angle passed
     159:          * 
     160:          * @param imageView
     161:          *            ImageView the ImageView whose bitmap we want to rotate
     162:          * @param t
     163:          *            transformation
     164:          * @param rotationAngle
     165:          *            the Angle by which to rotate the Bitmap
     166:          */
     167:         private void transformImageBitmap(ImageView child, Transformation t,
     168:                         int rotationAngle) {
     169:                 mCamera.save();
     170:                 final Matrix imageMatrix = t.getMatrix();
     171:                 final int imageHeight = child.getLayoutParams().height;
     172:                 final int imageWidth = child.getLayoutParams().width;
     173:                 final int rotation = Math.abs(rotationAngle);
     174:  
     175:                 // 在Z轴上正向移动camera的视角,实际效果为放大图片。
     176:                 // 如果在Y轴上移动,则图片上下移动;X轴上对应图片左右移动。
     177:                 mCamera.translate(0.0f, 0.0f, 100.0f);
     178:  
     179:                 // As the angle of the view gets less, zoom in
     180:                 if (rotation < mMaxRotationAngle) {
     181:                         float zoomAmount = (float) (mMaxZoom + (rotation * 1.5));
     182:                         mCamera.translate(0.0f, 0.0f, zoomAmount);
     183:                 }
     184:  
     185:                 // 在Y轴上旋转,对应图片竖向向里翻转。
     186:                 // 如果在X轴上旋转,则对应图片横向向里翻转。
     187:                 mCamera.rotateY(rotationAngle);
     188:                 mCamera.getMatrix(imageMatrix);
     189:                 // Preconcats matrix相当于右乘矩阵,Postconcats matrix相当于左乘矩阵。
     190:                 imageMatrix.preTranslate(-(imageWidth / 2), -(imageHeight / 2));
     191:                 imageMatrix.postTranslate((imageWidth / 2), (imageHeight / 2));
     192:                 mCamera.restore();
     193:         }
     194: }

    ageAdapter.java 图片适配器类,用来将文件中的图片资源加载到应用程序

       1: package com.android.CustomGallery;
       2:  
       3: import android.content.Context;
       4: import android.graphics.Bitmap;
       5: import android.graphics.BitmapFactory;
       6: import android.graphics.Canvas;
       7: import android.graphics.LinearGradient;
       8: import android.graphics.Matrix;
       9: import android.graphics.Paint;
      10: import android.graphics.PorterDuffXfermode;
      11: import android.graphics.Bitmap.Config;
      12: import android.graphics.PorterDuff.Mode;
      13: import android.graphics.Shader.TileMode;
      14: import android.graphics.drawable.BitmapDrawable;
      15: import android.view.View;
      16: import android.view.ViewGroup;
      17: import android.widget.BaseAdapter;
      18: import android.widget.ImageView;
      19:  
      20: /*
      21: * Copyright (C) 2010 Neil Davies
      22: *
      23: * Licensed under the Apache License, Version 2.0 (the "License");
      24: * you may not use this file except in compliance with the License.
      25: * You may obtain a copy of the License at
      26: *
      27: * http://www.apache.org/licenses/LICENSE-2.0
      28: *
      29: * Unless required by applicable law or agreed to in writing, software
      30: * distributed under the License is distributed on an "AS IS" BASIS,
      31: * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
      32: * See the License for the specific language governing permissions and
      33: * limitations under the License.
      34: * 
      35: * This code is base on the Android Gallery widget and was Created 
      36: * by Neil Davies neild001 'at' gmail dot com to be a Coverflow widget
      37: * 
      38: * @author Neil Davies
      39: */
      40: public class ImageAdapter extends BaseAdapter {
      41:         int mGalleryItemBackground;
      42:         private Context mContext;
      43:  
      44:         private int[] mImageIds;
      45:  
      46:         private ImageView[] mImages;
      47:  
      48:         public ImageAdapter(Context c, int[] ImageIds) {
      49:                 mContext = c;
      50:                 mImageIds = ImageIds;
      51:                 mImages = new ImageView[mImageIds.length];
      52:         }
      53:  
      54:         public boolean createReflectedImages() {
      55:                 // The gap we want between the reflection and the original image
      56:                 final int reflectionGap = 4;
      57:                 
      58:  
      59:  
      60:                 int index = 0;
      61:                 for (int imageId : mImageIds) {
      62:                         Bitmap originalImage = BitmapFactory.decodeResource(mContext
      63:                                         .getResources(), imageId);
      64:                         int width = originalImage.getWidth();
      65:                         int height = originalImage.getHeight();
      66:  
      67:                         // This will not scale but will flip on the Y axis
      68:                         Matrix matrix = new Matrix();
      69:                         matrix.preScale(1, -1);
      70:  
      71:                         // Create a Bitmap with the flip matrix applied to it.
      72:                         // We only want the bottom half of the image
      73:                         Bitmap reflectionImage = Bitmap.createBitmap(originalImage, 0,
      74:                                         height / 2, width, height / 2, matrix, false);
      75:  
      76:                         // Create a new bitmap with same width but taller to fit
      77:                         // reflection
      78:                         Bitmap bitmapWithReflection = Bitmap.createBitmap(width,
      79:                                         (height + height / 2), Config.ARGB_8888);
      80:  
      81:                         // Create a new Canvas with the bitmap that's big enough for
      82:                         // the image plus gap plus reflection
      83:                         Canvas canvas = new Canvas(bitmapWithReflection);
      84:                         // Draw in the original image
      85:                         canvas.drawBitmap(originalImage, 0, 0, null);
      86:                         // Draw in the gap
      87:                         Paint deafaultPaint = new Paint();
      88:                         canvas.drawRect(0, height, width, height + reflectionGap,
      89:                                         deafaultPaint);
      90:                         // Draw in the reflection
      91:                         canvas.drawBitmap(reflectionImage, 0, height + reflectionGap, null);
      92:  
      93:                         // Create a shader that is a linear gradient that covers the
      94:                         // reflection
      95:                         Paint paint = new Paint();
      96:                         LinearGradient shader = new LinearGradient(0, originalImage
      97:                                         .getHeight(), 0, bitmapWithReflection.getHeight()
      98:                                         + reflectionGap, 0x70ffffff, 0x00ffffff, TileMode.CLAMP);
      99:                         // Set the paint to use this shader (linear gradient)
     100:                         paint.setShader(shader);
     101:                         // Set the Transfer mode to be porter duff and destination in
     102:                         paint.setXfermode(new PorterDuffXfermode(Mode.DST_IN));
     103:                         // Draw a rectangle using the paint with our linear gradient
     104:                         canvas.drawRect(0, height, width, bitmapWithReflection.getHeight()
     105:                                         + reflectionGap, paint);
     106:                         //解决图片的锯齿现象
     107:                         BitmapDrawable bd = new BitmapDrawable(bitmapWithReflection);
     108:                         bd.setAntiAlias(true);
     109:  
     110:                         ImageView imageView = new ImageView(mContext);
     111:                         //imageView.setImageBitmap(bitmapWithReflection);
     112:                         imageView.setImageDrawable(bd);
     113:                         imageView.setLayoutParams(new GalleryFlow.LayoutParams(160, 240));
     114:                         // imageView.setScaleType(ScaleType.MATRIX);
     115:                         mImages[index++] = imageView;
     116:  
     117:                 }
     118:                 return true;
     119:         }
     120:  
     121:         public int getCount() {
     122:                 return mImageIds.length;
     123:         }
     124:  
     125:         public Object getItem(int position) {
     126:                 return position;
     127:         }
     128:  
     129:         public long getItemId(int position) {
     130:                 return position;
     131:         }
     132:  
     133:         public View getView(int position, View convertView, ViewGroup parent) {
     134:  
     135:                 // Use this code if you want to load from resources
     136:                 /*
     137:                  * ImageView i = new ImageView(mContext);
     138:                  * i.setImageResource(mImageIds[position]); i.setLayoutParams(new
     139:                  * CoverFlow.LayoutParams(350,350));
     140:                  * i.setScaleType(ImageView.ScaleType.CENTER_INSIDE);
     141:                  * 
     142:                  * //Make sure we set anti-aliasing otherwise we get jaggies
     143:                  * BitmapDrawable drawable = (BitmapDrawable) i.getDrawable();
     144:                  * drawable.setAntiAlias(true); return i;
     145:                  */
     146:  
     147:                 return mImages[position];
     148:         }
     149:  
     150:         /**
     151:          * Returns the size (0.0f to 1.0f) of the views depending on the 'offset' to
     152:          * the center.
     153:          */
     154:         public float getScale(boolean focused, int offset) {
     155:                 /* Formula: 1 / (2 ^ offset) */
     156:                 return Math.max(0, 1.0f / (float) Math.pow(2, Math.abs(offset)));
     157:         }
     158:  
     159: }
     

    MainActivity.java 用来显示界面

       1: package com.android.CustomGallery;
       2:  
       3: /**
       4: * 一个实现了3D效果的Gallery,就像iPhone中的相册浏览一样炫……
       5: */
       6: import android.app.Activity;
       7: import android.os.Bundle;
       8:  
       9: public class MainActivity extends Activity {
      10:         /** Called when the activity is first created. */
      11:         @Override
      12:         public void onCreate(Bundle savedInstanceState) {
      13:                 super.onCreate(savedInstanceState);
      14:  
      15:                 setContentView(R.layout.main);
      16:  
      17:                 int[] images = { R.drawable.photo1, R.drawable.photo2,
      18:                                 R.drawable.photo3, R.drawable.photo4, R.drawable.photo5,
      19:                                 R.drawable.photo6, R.drawable.photo7, R.drawable.photo8, };
      20:  
      21:                 ImageAdapter adapter = new ImageAdapter(this, images);
      22:                 adapter.createReflectedImages();
      23:  
      24:                 GalleryFlow galleryFlow = (GalleryFlow) findViewById(R.id.gallery_flow);
      25:                 galleryFlow.setAdapter(adapter);
      26:  
      27:         }
      28: }

    main.xml 程序设置文件

       1: "1.0" encoding="utf-8"?>
       2: "http://schemas.android.com/apk/res/android"
       3:     android:orientation="vertical"
       4:     android:layout_width="fill_parent"
       5:     android:layout_height="fill_parent"
       6:     >
       7:   
       8:         android:id="@+id/gallery_flow"
       9:         android:layout_width="fill_parent"
      10:         android:layout_height="fill_parent"
      11:         />
  • 相关阅读:
    【iOS开发每日小笔记(十二)】仿Facebook登录界面 错误提示抖动 利用CAAnimation设置动画效果
    《跑酷好基友》(英文名:BothLive)隐私政策声明
    【iOS开发每日小笔记(十一)】iOS8更新留下的“坑” NSAttributedString设置下划线 NSUnderlineStyleAttributeName 属性必须为NSNumber
    【iOS开发每日小笔记(十)】自制带圆框的头像 利用在CALayer设置“寄宿图”
    【iOS开发每日小笔记(九)】在子线程中使用runloop,正确操作NSTimer计时的注意点 三种可选方法
    2018/3/29
    2018/3/26
    2018/3/25
    BZOJ[3091] 城市旅行
    2018/3/23
  • 原文地址:https://www.cnblogs.com/yangkai-cn/p/4016822.html
Copyright © 2020-2023  润新知