Android编程实现的手写板和涂鸦功能

本文实例讲述了Android编程实现的手写板和涂鸦功能。分享给大家供大家参考,具体如下:

下面仿一个Android手写板和涂鸦的功能,直接上代码:

write_pad.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:greendroid="http://schemas.android.com/apk/res/com.cyrilmottier.android.gdcatalog"
  android:layout_width="fill_parent"
  android:layout_height="fill_parent"
  android:orientation="vertical" >
  <FrameLayout
    android:id="@+id/tablet_view"
    android:layout_width="fill_parent"
    android:layout_height="300dp" >
  </FrameLayout>
  <LinearLayout
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:background="@android:drawable/bottom_bar"
    android:paddingTop="4dp" >
    <Button
      android:id="@+id/write_pad_ok"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:layout_weight="1"
      android:text="确定" />
    <Button
      android:id="@+id/write_pad_clear"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:layout_weight="1"
      android:text="清除" />
    <Button
      android:id="@+id/write_pad_cancel"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:layout_weight="1"
      android:text="取消" />
  </LinearLayout>
</LinearLayout>

这个是手写板的主要布局文件,能够手写的部分是一个FrameLayout。下面有确定、清除和取消按钮,用来保存和擦除签名。

主要代码逻辑如下:

MainActivity.java

package com.jackie.handwriting;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import android.app.Activity;
import android.graphics.Bitmap;
import android.os.Bundle;
import android.os.Environment;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.ImageView;
import android.widget.TextView;
public class MainActivity extends Activity {
  private ImageView mIVSign;
  private TextView mTVSign;
  private Bitmap mSignBitmap;
  /** Called when the activity is first created. */
  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    mIVSign = (ImageView) findViewById(R.id.iv_sign);
    mTVSign = (TextView) findViewById(R.id.tv_sign);
    mTVSign.setOnClickListener(new OnClickListener() {
      @Override
      public void onClick(View view) {
        WritePadDialog mWritePadDialog = new WritePadDialog(
            MainActivity.this, new WriteDialogListener() {
              @Override
              public void onPaintDone(Object object) {
                mSignBitmap = (Bitmap) object;
                createSignFile();
                mIVSign.setImageBitmap(mSignBitmap);
                mTVSign.setVisibility(View.GONE);
              }
            });
        mWritePadDialog.show();
      }
    });
  }
  //创建签名文件
  private void createSignFile() {
    ByteArrayOutputStream baos = null;
    FileOutputStream fos = null;
    String path = null;
    File file = null;
    try {
      path = Environment.getExternalStorageDirectory() + File.separator + System.currentTimeMillis() + ".jpg";
      file = new File(path);
      fos = new FileOutputStream(file);
      baos = new ByteArrayOutputStream();
      //如果设置成Bitmap.compress(CompressFormat.JPEG, 100, fos) 图片的背景都是黑色的
      mSignBitmap.compress(Bitmap.CompressFormat.PNG, 100, baos);
      byte[] b = baos.toByteArray();
      if (b != null) {
        fos.write(b);
      }
    } catch (IOException e) {
      e.printStackTrace();
    } finally {
      try {
        if (fos != null) {
          fos.close();
        }
        if (baos != null) {
          baos.close();
        }
      } catch (IOException e) {
        e.printStackTrace();
      }
    }
  }
}

PaintView.java

package com.jackie.handwriting;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Bitmap.Config;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.PorterDuff;
import android.view.MotionEvent;
import android.view.View;
public class PaintView extends View {
  private Paint mPaint;
  private Path mPath;
  private Bitmap mBitmap;
  private Canvas mCanvas;
  private int screenWidth, screenHeight;
  private float currentX, currentY;
  public PaintView(Context context, int screenWidth, int screenHeight) {
    super(context);
    this.screenWidth = screenWidth;
    this.screenHeight = screenHeight;
    init();
  }
  private void init() {
    mPaint = new Paint();
    mPaint.setAntiAlias(true); // 去除锯齿
    mPaint.setStrokeWidth(5);
    mPaint.setStyle(Paint.Style.STROKE);
    mPaint.setColor(Color.BLACK);
    mPath = new Path();
    mBitmap = Bitmap.createBitmap(screenWidth, screenHeight, Config.ARGB_8888);
    mCanvas = new Canvas(mBitmap);
//   mCanvas.drawColor(Color.WHITE);
  }
  @Override
  protected void onDraw(Canvas canvas) {
    canvas.drawBitmap(mBitmap, 0, 0, null);
    canvas.drawPath(mPath, mPaint);
  }
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    float x = event.getX();
    float y = event.getY();
    switch (event.getAction()) {
    case MotionEvent.ACTION_DOWN:
      currentX = x;
      currentY = y;
      mPath.moveTo(currentX, currentY);
      break;
    case MotionEvent.ACTION_MOVE:
      currentX = x;
      currentY = y;
      mPath.quadTo(currentX, currentY, x, y); // 画线
      break;
    case MotionEvent.ACTION_UP:
      mCanvas.drawPath(mPath, mPaint);
      break;
    }
    invalidate();
    return true;
  }
  public Bitmap getPaintBitmap() {
    return resizeImage(mBitmap, 320, 480);
  }
  public Path getPath() {
    return mPath;
  }
  // 缩放
  public static Bitmap resizeImage(Bitmap bitmap, int width, int height) {
    int originWidth = bitmap.getWidth();
    int originHeight = bitmap.getHeight();
    float scaleWidth = ((float) width) / originWidth;
    float scaleHeight = ((float) height) / originHeight;
    Matrix matrix = new Matrix();
    matrix.postScale(scaleWidth, scaleHeight);
    Bitmap resizedBitmap = Bitmap.createBitmap(bitmap, 0, 0, originWidth,
        originHeight, matrix, true);
    return resizedBitmap;
  }
  //清除画板
  public void clear() {
    if (mCanvas != null) {
      mPath.reset();
      mCanvas.drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR);
      invalidate();
    }
  }
}

WritePadDialog.java

package com.jackie.handwriting;
import android.app.Dialog;
import android.content.Context;
import android.os.Bundle;
import android.util.DisplayMetrics;
import android.view.View;
import android.view.Window;
import android.widget.Button;
import android.widget.FrameLayout;
import android.widget.Toast;
public class WritePadDialog extends Dialog {
  private Context mContext;
  private WriteDialogListener mWriteDialogListener;
  private PaintView mPaintView;
  private FrameLayout mFrameLayout;
  private Button mBtnOK, mBtnClear, mBtnCancel;
  public WritePadDialog(Context context,
      WriteDialogListener writeDialogListener) {
    super(context);
    this.mContext = context;
    this.mWriteDialogListener = writeDialogListener;
  }
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    requestWindowFeature(Window.FEATURE_NO_TITLE); //无标题
    setContentView(R.layout.write_pad);
    mFrameLayout = (FrameLayout) findViewById(R.id.tablet_view);
    // 获取屏幕尺寸
    DisplayMetrics mDisplayMetrics = new DisplayMetrics();
    getWindow().getWindowManager().getDefaultDisplay().getMetrics(mDisplayMetrics);
    int screenWidth = mDisplayMetrics.widthPixels;
    int screenHeight = mDisplayMetrics.heightPixels;
    mPaintView = new PaintView(mContext, screenWidth, screenHeight);
    mFrameLayout.addView(mPaintView);
    mPaintView.requestFocus();
    mBtnOK = (Button) findViewById(R.id.write_pad_ok);
    mBtnOK.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        if (mPaintView.getPath().isEmpty()) {
          Toast.makeText(mContext, "请写下你的大名", Toast.LENGTH_SHORT).show();
          return;
        }
        mWriteDialogListener.onPaintDone(mPaintView.getPaintBitmap());
        dismiss();
      }
    });
    mBtnClear = (Button) findViewById(R.id.write_pad_clear);
    mBtnClear.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        mPaintView.clear();
      }
    });
    mBtnCancel = (Button) findViewById(R.id.write_pad_cancel);
    mBtnCancel.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        cancel();
      }
    });
  }
}

WriteDilogListener.java

package com.jackie.handwriting;
/**
 * 监听手写板对话框
 * @author chengcj1
 *
 */
public interface WriteDialogListener {
  public void onPaintDone(Object object);
}

效果如下:

更多关于Android相关内容感兴趣的读者可查看本站专题:《Android图形与图像处理技巧总结》、《Android开发入门与进阶教程》、《Android调试技巧与常见问题解决方法汇总》、《Android基本组件用法总结》、《Android视图View技巧总结》、《Android布局layout技巧总结》及《Android控件用法总结》

希望本文所述对大家Android程序设计有所帮助。

您可能感兴趣的文章:

  • Android编程实现在Bitmap上涂鸦效果
  • Android 开发实例简单涂鸦板
  • Android 使用Path实现涂鸦功能
  • Android实现Path平滑的涂鸦效果实例
  • android实现简单的画画板实例代码
  • Android采用双缓冲技术实现画板
  • Android自定义SurfaceView实现画板功能
  • Android实现画板、写字板功能(附源码下载)
  • Android多媒体之画画板开发案例分享
  • 双缓冲技术实现Android 画板应用
  • Android画画板的制作方法
(0)

相关推荐

  • Android 开发实例简单涂鸦板

    在Android上开发一些小应用既可以积累知识又可以增加乐趣,与任务式开发不同,所以想到在Android系统上实现一个简单的涂鸦板,这是我们练手的一种好的方法.   涂鸦板应用的代码实现 新建工程MyWall,修改/res/layout/main.xml文件,在里面添加一个SurfaceView和两个Button,用到了RelativeLayout布局,完整的main.xml文件如下: XML/HTML代码 <?xml version="1.0" encoding="u

  • Android编程实现在Bitmap上涂鸦效果

    本文实例讲述了Android编程实现在Bitmap上涂鸦效果.分享给大家供大家参考,具体如下: 布局文件: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" andro

  • Android画画板的制作方法

    本文实例为大家分享了Android画画板展示的具体代码,供大家参考,具体内容如下 main.xml布局 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:layout_w

  • 双缓冲技术实现Android 画板应用

    什么是双缓冲技术?双缓冲技术就是当用户操作界面完成后,会有一个缓冲区保存用户操作的结果. 为什么要使用双缓冲技术?拿Android 游戏开发来说,界面贞每次都是全部重画的,也就说画了新的,旧的就没了,所以需要使用双缓冲技术保存之前的内容. 如何实现双缓冲?使用一个Bitmap对象保留之前的画布即可. package com.example.phonegaptest; import android.content.Context; import android.graphics.Bitmap; i

  • Android多媒体之画画板开发案例分享

    先看看效果: 其实画画板的原理很简单,就是首先记录下按下屏幕的点,然后每移动一下就让这两次移动的点连线,周而复始,图像就由很多条直线构成了. 核心代码 : public class MainActivity extends Activity implements OnClickListener,OnSeekBarChangeListener { private View red_view,green_view,blue_view; //控制画笔颜色的三块区域 private SeekBar se

  • Android 使用Path实现涂鸦功能

    今天实现一个涂鸦效果,会分几步实现,这里有一个重要的知识点就是图层,要理解这个,不然你看这篇博客,很迷茫,迷茫的苍茫的天涯是我的爱,先从简单的需求做起,绘制一条线,代码如下: package com.tuya; import android.content.Context; import android.graphics.Canvas; import android.graphics.Paint; import android.graphics.Path; import android.util

  • Android采用双缓冲技术实现画板

    本文实例为大家分享了Android实现画板的具体代码,采用的技术是双缓冲技术,供大家参考,具体内容如下 1.双缓冲技术的概念 所谓的双缓冲技术其实很简单,当程序需要在指定的View上进行绘制时,程序并不需要直接绘制到该View组件,而是先绘制到一个内存中的Bitmap图片上(就是缓冲),等内存中的Bitmap绘制好之后,再一次性将Bitmap绘制到View组件上. 2.Android采用双缓冲实现画板  实现的思路: 1).定义一个内存中图片,将他作为缓冲区Bitmap cacheBitmap

  • Android实现画板、写字板功能(附源码下载)

    前言 本文给大家分享一个使用Android开发写字板功能Dem.简单操作内存中的图像.对图像进行简单的处理.绘制直线.以达到写字板的效果 效果图如下 XML布局代码 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="

  • Android自定义SurfaceView实现画板功能

    接触了这么久的View,总不能一直停留在View里,现在开始呢,就要学习一个新的知识点:SurfaceView,实际上SurfaceView与View的原理都差不多,只是效率和渲染方式上,SurfaceView要优于View,这也是我们写这个的原因.今天就看看这个SurfaceView,好了,下面就是今天要说的效果. 界面很简单,就是一个按钮以及一个画板,先看看界面的代码吧 <LinearLayout xmlns:android="http://schemas.android.com/ap

  • android实现简单的画画板实例代码

    直接看代码,注释都写清楚了 复制代码 代码如下: public class MainActivity extends Activity { private ImageView iv; private Bitmap baseBitmap; private Canvas canvas; private Paint paint; @Override protected void onCreate(Bundle savedInstanceState) {  super.onCreate(savedIns

  • Android实现Path平滑的涂鸦效果实例

    前言 在最近的一个项目中做了一个涂鸦的效果,手指快速移动,会出现折线,这篇文章记录笔触优化.下面话不多说了,来一起看看详细的介绍吧. 优化前 优化 设计到的类:Paint,Path Path类记录了坐标点集合决定线条轨迹,Paint决定怎么画 Paint处理 //连接的外边缘以圆弧的方式相交 paint.setStrokeJoin(Paint.Join.ROUND); //线条结束处绘制一个半圆 paint.setStrokeCap(Paint.Cap.ROUND); Path处理 这里用的到有

随机推荐