`
ranlic
  • 浏览: 46797 次
社区版块
存档分类
最新评论

Android Demos 示例BitmapMesh中部分方法的理解

 
阅读更多

 

package practice.com;

import android.os.Bundle;
import android.view.*;
import android.content.Context;
import android.util.FloatMath;
import android.graphics.*;

public class BitmapMesh extends GraphicsActivity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(new SampleView(this));
    }
    
    private static class SampleView extends View {
    	private static final int WIDTH = 20;
    	private static final int HEIGHT = 20;
    	private static final int COUNT = (WIDTH + 1)*(HEIGHT +1);
    	
    	private final Bitmap mBitmap;
    	private final Bitmap mBitmap1;
    	private final float[] mVerts = new float[COUNT*2];
    	private final float[] mOrig = new float[COUNT*2];
    	
    	private final Matrix mMatrix = new Matrix();
    	private final Matrix mInverse = new Matrix();
    	
    	private static void setXY(float[] array , int index , float x, float y){
    		array[index*2 + 0] = x;
    		array[index*2 + 1] = y;
    	}
    	
    	public SampleView(Context context){
    		
    		super(context);
    		setFocusable(true);
    	
    		mBitmap = BitmapFactory.decodeResource(getResources(), R.drawable.beach);
    		mBitmap1 = Bitmap.createBitmap(mBitmap.getWidth(), 
    				mBitmap.getHeight(), Bitmap.Config.ARGB_8888);
    		drawImageWithGrid(mBitmap);
    		float w = mBitmap.getWidth();
    		float h = mBitmap.getHeight();
    		
    		int index = 0;
    		for(int y = 0; y <= HEIGHT; y++) {
    			float fy = h * y / HEIGHT; 
    			for(int x = 0; x <= WIDTH; x++){
    				float fx = w * x / WIDTH;
    				setXY(mVerts, index, fx, fy);
    				setXY(mOrig, index, fx, fy);
    				index += 1;
    			}
    		}
    		
    		mMatrix.setTranslate(10, 10);
    		mMatrix.invert(mInverse);
    		
    	}
    	// 此处借鉴自引路蜂
    	private void drawImageWithGrid(Bitmap image){
    		
    		Canvas canvas = new Canvas(mBitmap1);
    		float w = mBitmap1.getWidth();
    		float h = mBitmap1.getHeight();
    		int xCount = (int)w/WIDTH;
    		int yCount = (int)h/HEIGHT;
    		Paint paint = new Paint();
    		canvas.drawBitmap(image, 0, 0, paint);
    		paint.setStyle(Paint.Style.STROKE);
    		paint.setStrokeWidth(1);
    		paint.setColor(0x8000FF00);
    		for(int i = 0; i < xCount;i++){
    			for(int j = 0; j < yCount; j++){
    				canvas.drawRect(i*WIDTH, j*HEIGHT,i*WIDTH+WIDTH,j*HEIGHT+HEIGHT, paint);
    				
    			}
    		}
    	}

		@Override
		protected void onDraw(Canvas canvas) {
			// TODO Auto-generated method stub
			canvas.drawColor(0xFFCCCCCC);
			
			canvas.concat(mMatrix);
		    canvas.drawBitmapMesh(mBitmap1, WIDTH, HEIGHT, mVerts, 0, null, 0, null);
		}
		
		private void warp(float cx, float cy){
			final float K = 100000;
			float[] src = mOrig;
			float[] dst = mVerts;
			
			for(int i = 0; i < COUNT*2; i += 2){
				float x = src[i+0];
				float y = src[i+1];
				float dx = cx - x;
				float dy = cy - y;
				float dd = dx*dx + dy*dy;
				float d = FloatMath.sqrt(dd);
				float pull = K / (dd + 0.000001f);
				
				pull /= (d + 0.000001f);
				
				if(pull >= 1){
					dst[i+0] = cx;
					dst[i+1] = cy;
				}else{
					dst[i+0] = x + dx * pull;
					dst[i+1] = y + dy * pull;
				}
			}
		}
    	
    	private int mLastWarpX = -9999;
    	private int mLastWarpY;
    	
    	public boolean onTouchEvent (MotionEvent event){
    		float[] pt = {event.getX(), event.getY()};
    		mInverse.mapPoints(pt);
    		
    		int x = (int)pt[0];
    		int y = (int)pt[1];
    		
    		if(mLastWarpX != x || mLastWarpY != y){
    			mLastWarpX = x;
    			mLastWarpY = y;
    			warp(pt[0], pt[1]);
    			invalidate();
    		}
    		return true;
    	}
    }
}

 

上面为Android Demos 中BitmapMesh示例的代码(本人借鉴引路蜂Android ApiDemos示例解析(55)示例解析,为图像添加了绿色格线,方便清楚观察桌布扭曲情况),其中部分代码理解如下:

 

1 本例涉及到的矩阵算法

  mMatrix.invert(mInverse) 

  这个是把给定矩阵(本代码中指mMatrix)取逆矩阵,并将结果写入参数矩阵中(本代码指mInverse).

     mInverse.mapPoints(pt)

   这个方法是将指定的点集(pt)按照矩阵参数(mInverse)进行图形平移,旋转,放大,缩小等操作后再写入点集(pt)中去.

 在本例中,因为图片相对于屏幕已经按照矩阵mMatrix进行了平移(mMatrix.setTranslate(10, 10)),因此触摸屏幕所取得的点坐标(pt)实际以屏幕左上角为原点的坐标值,需要把此点坐标变换为以图片左上角为原点的坐标值.第一个方法是取得mMatrix矩阵的逆操作(矩阵mInverse),第二个方法是将触摸屏幕所取得的屏幕点坐标按照矩阵mInverse变换为图片(本例图片beach)中的点坐标,然后方能按照算法进行图片扭曲处理.

 

 

2 图片扭曲算法

 

  本例中的扭曲算法是以触摸点(float[] pt = {event.getX(), event.getY()})为目标点,其它网格点

 

float[] src = mOrig;
float x = src[i+0];
float y = src[i+1];

  距目标点(pt)的直线距离的远近为考量

 

float dx = cx - x;
float dy = cy - y;
float dd = dx*dx + dy*dy;
float d = FloatMath.sqrt(dd);
float pull = K / (dd + 0.000001f);
				
pull /= (d + 0.000001f);
 

 

 较近处,向目标点塌缩

if(pull >= 1){
dst[i+0] = cx;
dst[i+1] = cy;

 

 

 较远处,修正数值,向目标点靠近,与目标点越远,修改值越小

else{
  dst[i+0] = x + dx * pull;
  dst[i+1] = y + dy * pull;

 

 

本例运行结果如下:

 

 

 

  • 大小: 143.4 KB
分享到:
评论

相关推荐

    android apidemos示例解析

    Android ApiDemos是一个官方提供的示例应用程序集合,旨在帮助开发者理解并掌握Android SDK的各种API用法。通过这些示例,开发者可以学习到如何在实际项目中应用不同的功能和技术。本文将对部分示例进行详细解析,...

    ApiDemos示例源码

    ApiDemos是Android平台提供的一款示例应用,它包含了Android SDK中的各种API功能展示,对于开发者来说,这是一个非常宝贵的资源库,可以帮助我们深入理解和学习Android系统的API用法。本文将详细介绍ApiDemos中的...

    Android ApiDemos4.4 示例解析

    Android SDK 中带有很多例子,其中ApiDemo 详细介绍了Android 平台主要API,分成了 · App · Content · Graphics · Media · OS · Text · Views 几个大类,每个大类又分为几个小类,一共大约有200 多个...

    android官方示例Support-V7 Demos整合

    【Android官方示例Support-V7 Demos整合】是Android开发者们的重要参考资料,它涵盖了多个Android Support Library v7组件的示例代码,旨在帮助开发者更好地理解和使用这些库。在这个压缩包中,包含了多个子项目的...

    Android6.0 Api Demos

    这些示例旨在帮助开发者更好地理解和学习如何在实际应用中使用Android的新特性和API。下面将详细探讨其中的一些关键知识点。 1. **权限管理(Runtime Permissions)** 在Android 6.0中,权限管理有了重大变化,...

    最新Android apidemos

    Android API Demos是Google官方提供的一款用于展示Android SDK中各种API功能和用法的应用程序,它涵盖了从基础控件到高级特性的全方位示例,是开发者学习Android开发的重要资源。在最新的版本中,API Demos不仅包含...

    android demos

    "Android Demos"是一个集合了各种Android开发示例的应用程序,它包含了丰富的代码实例,旨在帮助开发者更好地理解和学习Android平台的各种功能和API的使用。这个压缩包中的"ApiDemos"很可能是一个官方或非官方的实现...

    Android ApiDemos示例解析(26):App->Notification->IncomingMessage

    本文将深入解析`ApiDemos`中的一个特定示例——`App-&gt;Notification-&gt;IncomingMessage`,帮助开发者更好地理解和应用Android的通知功能。 通知(Notification)是Android系统中一种关键的用户界面元素,它在状态栏中...

    android ApiDemos

    Android API Demos是一款由谷歌官方提供的开源项目,它包含了大量Android SDK中的API示例代码,旨在帮助开发者更好地理解和学习如何在实际应用中使用Android的各种功能和API。该项目覆盖了从基础组件到高级特性的全...

    Android ApiDemos

    `Android ApiDemos` 是Android系统提供的一款官方示例程序,它集合了Android SDK中的各种API用法,是开发者学习和理解Android开发的关键资源。这个项目旨在通过实例代码来演示Android API的各种功能和组件,帮助...

    Android开发入门与实例 Demos

    进入实例部分,"demos"目录下的文件很可能是各种示例代码,涵盖了Android开发中的常见功能和组件: - **活动(Activity)**:展示用户界面,例如MainActivity.java,展示了如何创建一个基本的活动。 - **布局(Layout)...

    Android ApiDemos apk

    Android ApiDemos apk是Android开发者们熟悉的一个示例程序,它包含了Android SDK中的各种API功能演示,为开发者提供了丰富的学习资源。这个应用程序旨在帮助开发者更好地理解和掌握Android平台的各种功能和特性,...

    android Support4Demos

    这些示例可以帮助开发者深入理解如何在实际项目中使用`Fragment`和`ViewPager`,以实现更丰富的用户体验。 总之,`android-support-v4`库的`Fragment`和`ViewPager`是Android开发中的重要工具,它们帮助开发者创建...

    support4Demos_AndroidStudio版

    在Android开发中,`support4Demos`是一个重要的学习资源,它包含了Google官方支持库v4版本的各种组件和功能的示例代码。这个压缩包文件“support4Demos”是针对Android Studio优化过的版本,解决了原版在导入时可能...

    android1.6 apiDemos

    2. 阅读源代码,理解示例背后的实现逻辑。 3. 结合官方文档,进一步学习API的详细用法。 4. 尝试修改示例,创造自己的应用场景。 总之,"android1.6 apiDemos"是开发者学习Android 1.6及早期版本API的重要参考资料...

    android_ApiDemos

    Android ApiDemos是Android开发者学习和理解Android API的重要资源,它包含了丰富的示例代码,涵盖了Android SDK中的各种API功能。这个程序是专为Android 7(API级别24)设计的,提供了详尽的实践教程,帮助开发者...

    android ApiDemos不报错版本

    Android ApiDemos是Android平台上的一个官方示例项目,它为开发者提供了丰富的API演示,涵盖了Android系统中的各种控件和功能,是学习和理解Android开发的宝贵资源。这个不报错版本确保了无论是通过虚拟机还是真机...

    Android ApiDemos2.1

    通过深入研究ApiDemos 2.1,开发者可以了解到早期Android开发的基本方法和最佳实践,这对于理解后来的Android版本和API更新也是有帮助的,因为许多核心概念和原理是延续的。即使在当前版本的Android中,很多基础概念...

    android的ApiDemos

    API Demos 是 Google 为了 Android 开发者所提供的一个 Android API 合集,其中包含了很多的 API 范例,同时遵循了良好的代码规范,是一个值得开发者研究和学习的典型。android的ApiDemos,需要解压缩后使用。

Global site tag (gtag.js) - Google Analytics