Android手指绘图(一)

package cn.c;
import java.io.OutputStream;
import android.app.Activity;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.Bitmap.CompressFormat;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.net.Uri;
import android.os.Bundle;
import android.view.Display;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.View.OnTouchListener;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.Toast;
/**
 * 注意:
 * 1 event.getX()和event.getRawX()方法的区别
 *   event.getX()是相对于父控件而言
 *   event.getRawX()是相对于屏幕左上角而言
 * 2 在Activity中处理触摸事件采用的是OnTouchEvent()
 *   在View中处理触摸事件是实现OnTouchListener{}接口
 *   覆写其中的onTouch()方法
 * 3 Bug的问题:
 *   计算options.inSampleSize这里有Bug.有的图片过大在
 *   设备图片库中是被旋转处理了的(虽然看上去没有).比如一
 *   张大图片在图片库中看上去是竖屏的,但得到后在ImageView
 *   中显示却是横屏的.
 *   所以这里方便测试直接:
 *   options.inSampleSize=8;
 *   写死了
 * 4 调用bitmap.compress()方法的时候注意若设置格式为PNG,那么则无效.
 *   因为PNG将始终保持所有数据.JPEG是"有损的"编解码器
 */
public class MainActivity extends Activity implements OnTouchListener{
    private ImageView mImageView;
    private Button mButton;
    private Button mSaveButton;
    private Bitmap mRawBitmap;
    private Bitmap mAlteredBitmap;
    private Paint mPaint;
    private Canvas mCanvas;
    private Uri mPhotoFileUri;
    private final int PICK=88;
    float downX=0;
    float downY=0;
    float upX=0;
    float upY=0;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        mImageView=(ImageView) findViewById(R.id.imageView);
        mImageView.setOnTouchListener(this);
        mButton=(Button) findViewById(R.id.button);
        mButton.setOnClickListener(new ButtonOnClickListener());
        mSaveButton=(Button) findViewById(R.id.saveButton);
        mSaveButton.setOnClickListener(new ButtonOnClickListener());
        mPaint=new Paint();
        mPaint.setColor(Color.RED);
        mPaint.setStrokeWidth(5);
    }

	@Override
	protected void onActivityResult(int requestCode, int resultCode, Intent data) {
		super.onActivityResult(requestCode, resultCode, data);
		if (requestCode==PICK) {
			mPhotoFileUri=data.getData();
			Display display=getWindowManager().getDefaultDisplay();
			float w=display.getWidth();
			float h=display.getHeight();
			try {
				BitmapFactory.Options  options=new BitmapFactory.Options();
				options.inJustDecodeBounds=true;
				mRawBitmap=BitmapFactory.decodeStream
			    (getContentResolver().openInputStream(mPhotoFileUri), null, options);
				int widthRation=(int) Math.ceil(options.outHeight/h);
				int heightRation=(int) Math.ceil(options.outWidth/w);
				if (heightRation>1&&widthRation>1) {
					if (heightRation>widthRation) {
						options.inSampleSize=heightRation;
					} else {
						options.inSampleSize=widthRation;
					}
				}
				//Bug所在:
				options.inSampleSize=8;
				options.inJustDecodeBounds=false;
				mRawBitmap=BitmapFactory.decodeStream
			   (getContentResolver().openInputStream(mPhotoFileUri), null, options);
				mAlteredBitmap=Bitmap.createBitmap
				(mRawBitmap.getWidth(), mRawBitmap.getHeight(), mRawBitmap.getConfig());
				mCanvas=new Canvas(mAlteredBitmap);
				Matrix matrix=new Matrix();
				mCanvas.drawBitmap(mRawBitmap, matrix, mPaint);
				mImageView.setImageBitmap(mAlteredBitmap);
			} catch (Exception e) {
			}
		}
	}

	private class ButtonOnClickListener implements OnClickListener {
		public void onClick(View v) {
			switch (v.getId()) {
			case R.id.button:
				Intent intent = new Intent(
				Intent.ACTION_PICK,
				android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
				startActivityForResult(intent, PICK);
				break;
			case R.id.saveButton:
				try {
					if (mPhotoFileUri!=null) {
						OutputStream fos=getContentResolver().openOutputStream(mPhotoFileUri);
						mAlteredBitmap.compress(CompressFormat.JPEG, 90, fos);
						fos.close();
						Toast.makeText(MainActivity.this, "OK", Toast.LENGTH_SHORT).show();
					}

				} catch (Exception e) {
					// TODO: handle exception
				}

				break;
			default:
				break;
			}

		}

	}

	public boolean onTouch(View v, MotionEvent event) {
		int action = event.getAction();
		switch (action) {
		case MotionEvent.ACTION_DOWN:
            downX=event.getX();
            downY=event.getY();
			break;
		case MotionEvent.ACTION_MOVE:
			upX=event.getX();
            upY=event.getY();
            mCanvas.drawLine(downX, downY, upX, upY, mPaint);
            mImageView.invalidate();
            downX=upX;
            downY=upY;
			break;
		case MotionEvent.ACTION_UP:
//            upX=event.getX();
//            upY=event.getY();
//            mCanvas.drawLine(downX, downY, upX, upY, mPaint);
//            mImageView.invalidate();
			break;
		case MotionEvent.ACTION_CANCEL:
			break;

		default:
			break;
		}
		return true;
	}

}
时间: 2024-10-16 16:42:38

Android手指绘图(一)的相关文章

canvas-请问大家Android Canvas绘图填充问题,有专门的填充函数(方法)吗?

问题描述 请问大家Android Canvas绘图填充问题,有专门的填充函数(方法)吗? 已经使用paint画了一个封闭矩形,接下来目的是点击一下该区域后直接填充为所选颜色. 注意:不是直接画一个填充好的. 要用stroke和fill吗? 解决方案 这是没有的,当然你可以用开源别人写好的封装好的代码.你在ontouchevent里面判断一下手指按下的区域是否在该矩形内.如果在,在该区域画一个填充的矩形后更新界面.

Android SurfaceView 绘图覆盖刷新及脏矩形刷新方法

SurfaceView在Android中用作游戏开发是最适宜的,本文就将演示游戏开发中常用的两种绘图刷新策略在SurfaceView中的实现方法. 首先我们来看一下本例需要用到的两个素材图片: bj.jpg就是一个渐变图,用作背景. question.png是一个半透明的图像,我们希望将它放在上面,围绕其圆心不断旋转. 实现代码如下: package SkyD.SurfaceViewTest; import android.app.Activity; import android.content

Android SurfaceView 绘图覆盖刷新及“.NET研究”脏矩形刷新方法

SurfaceView在Android中用作游戏开发是最适宜的,本文就将演示游戏开发中常用的两种绘图刷新策略在SurfaceView中的实现方法. 首先我们来看一下本例需要用到的两个素材图片: bj.jpg就是一个渐变图,用作背景. question.png是一个半透明的图像,我们希望将它放在上面,围绕其圆心不断旋转. 实现代码如下: package SkyD.SurfaceViewTest; import android.app.Activity; import android.content

Evernote 收购手指绘图应用 Penultimate

功能繁多.最为流行的一款跨平台笔记服务 Evernote 近日收购了Penultimate,这是一款漂亮.手感流畅的绘图和手写记事的应用.Penultimate 本身也是排名最佳销售榜第四名的应用.这次收购对于 Penultimate 以及它的用户而言意味着什么呢?首先,当然是好消息.Penultimate 将仍然是一款独立的应用,就像 Evernote 曾经的收购一样(比如 Skitch).其次,与 Evernote 结合肯定会给它带来一些新功能,包括全文搜索和同步.另外一个好消息是,Penu

【android入门】之Android Canvas绘图抗锯齿解决方法

对于Android来说Canvas一般大量用于自定义View和游戏开发中,对于图形的基础绘制类,提供的drawText.drawBitmap以及drawCircle都需要抗锯齿处理才能让人满意,下面Android123将他们分为两大种情况.  一.包含Paint参数情况时,对于drawText.drawBitmap这样的方法,一般最后一个参数为Paint对象,Paint对象一般用于设置笔刷颜色和大小,同时包含了抗锯齿的方法,比如说objPaint.setAntiAlias(true);  这个方

Android群英传笔记——第六章:Android绘图机制与处理技巧

Android群英传笔记--第六章:Android绘图机制与处理技巧 一直在情调,时间都是可以自己调节的,不然世界上哪有这么多牛X的人 今天就开始读第六章了,算日子也刚好一个月了,一个月就读一半,这效率也确实有点低了,自己还要加把劲,争取四月底全部看完,第六章讲的是Android的绘图机制,应该算是比较核心的东西了,不管什么功能,最终都是以图形的方式呈现给用户的,因此,掌握Android的绘图技巧,可以在让你设计应用的时候更加的随心所欲,对Android的理解更高 基本的绘图方法,相信读者都已经

Android开发之实现手指直接拖动图片移动

一.基础知识:  要实现这一效果,需要一个容器来存放Gallrey显示的图片,这里使用一个继承自BaseAdapter类的派生类来装这些图片. 我们需要监听其事件setOnItemClickListener,从而确定用户当前选中的是哪一张图片. 首先,需要将所有要显示的图片的索引存放在一个int型数组中,然后通过setImageResource方法来设置ImageView要显示的图片资源,最后将 每张图片的ImageView显示在屏幕上.   二.代码展示: 1."main.xml"

Android中解决页签手指按下从左到右滑动的bug_Android

有一种方法可以阻止父层的View截获touch事件,就是调用 getParent().requestDisallowInterceptTouchEvent(true);方法. 一旦底层View收到touch的 action后调用这个方法那么父层View就不会再调用onInterceptTouchEvent了,也无法截获以后的action 在ViewPagerIndicator项目中找到TabPageIndicator该类,添加如下代码 @Override public boolean dispa

Android UI效果之绘图篇(一)_Android

最近准备整理一套关于UI效果的文章,算是对这段时间的一个总结,主要讲Android开发中的UI效果设计模块.初步分为一下几个篇幅: Android XML绘图(Shape.Layer.Selector) Android Canvas绘图(canvas.point.porterDuffXfermode.shader) Android 动画详解 Android 自定义控件 今天就当开胃菜,先讲讲最简单的xml绘图,相信这个大家都用的比较熟,这里就当给大家做一个小文档,当那个参数配置忘了,便于查阅 一