Android编程实现手绘及保存为图片的方法(附demo源码下载)

本文实例讲述了Android编程实现手绘及保存为图片的方法。分享给大家供大家参考,具体如下:

运行效果图预览:

应 yzuo_08 要求做了此Demo,跟以前那个手写板Demo不同的是可以将画布的内容保存为图片。

附上关键代码:

MainView.java

package com.tszy.views;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Bitmap.CompressFormat;
import android.graphics.Bitmap.Config;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
public class MainView extends View {
 private Paint paint;
 private Canvas cacheCanvas;
 private Bitmap cachebBitmap;
 private Path path;
 private int clr_bg, clr_fg;
 public MainView(Context context, AttributeSet attrs) {
  super(context, attrs);
  clr_bg = Color.WHITE;
  clr_fg = Color.CYAN;
  paint = new Paint();
  paint.setAntiAlias(true); // 抗锯齿
  paint.setStrokeWidth(3); // 线条宽度
  paint.setStyle(Paint.Style.STROKE); // 画轮廓
  paint.setColor(clr_fg); // 颜色
  path = new Path();
  // 创建一张屏幕大小的位图,作为缓冲
  cachebBitmap = Bitmap.createBitmap(480, 800, Config.ARGB_8888);
  cacheCanvas = new Canvas(cachebBitmap);
  cacheCanvas.drawColor(clr_bg);
 }
 public MainView(Context context) {
  super(context);
 }
 @Override
 protected void onDraw(Canvas canvas) {
  canvas.drawColor(clr_bg);
  // 绘制上一次的,否则不连贯
  canvas.drawBitmap(cachebBitmap, 0, 0, null);
  canvas.drawPath(path, paint);
 }
 /**
  * 清空画布
  */
 public void clear() {
  path.reset();
  cacheCanvas.drawColor(clr_bg);
  invalidate();
 }
 /**
  * 将画布的内容保存到文件
  * @param filename
  * @throws FileNotFoundException
  */
 public void saveToFile(String filename) throws FileNotFoundException {
  File f = new File(filename);
  if(f.exists())
   throw new RuntimeException("文件:" + filename + " 已存在!");
  FileOutputStream fos = new FileOutputStream(new File(filename));
  //将 bitmap 压缩成其他格式的图片数据
  cachebBitmap.compress(CompressFormat.PNG, 50, fos);
  try {
   fos.close();
  } catch (IOException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  }
 }
 private float cur_x, cur_y;
 private boolean isMoving;
 @Override
 public boolean onTouchEvent(MotionEvent event) {
  // TODO Auto-generated method stub
  float x = event.getX();
  float y = event.getY();
  switch (event.getAction()) {
   case MotionEvent.ACTION_DOWN : {
    cur_x = x;
    cur_y = y;
    path.moveTo(cur_x, cur_y);
    isMoving = true;
    break;
   }
   case MotionEvent.ACTION_MOVE : {
    if (!isMoving)
     break;
    // 二次曲线方式绘制
    path.quadTo(cur_x, cur_y, x, y);
    // 下面这个方法貌似跟上面一样
    // path.lineTo(x, y);
    cur_x = x;
    cur_y = y;
    break;
   }
   case MotionEvent.ACTION_UP : {
    // 鼠标弹起保存最后状态
    cacheCanvas.drawPath(path, paint);
    path.reset();
    isMoving = false;
    break;
   }
  }
  // 通知刷新界面
  invalidate();
  return true;
 }
}

Activity 代码:

@Override
public void onClick(View v) {
  // TODO Auto-generated method stub
  switch (v.getId()) {
   case R.id.iv_btn_clear :
    view.clear();
    break;
   case R.id.iv_btn_save : {
    try {
     String sdState = Environment.getExternalStorageState(); // 判断sd卡是否存在
     // 检查SD卡是否可用
     if (!sdState.equals(android.os.Environment.MEDIA_MOUNTED)) {
      Toast.makeText(this, "SD卡未准备好!", Toast.LENGTH_SHORT).show();
      break;
     }
     //获取系统图片存储路径
     File path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);
     // Make sure the Pictures directory exists.
     path.mkdirs();
     //根据当前时间生成图片名称
     Calendar c = Calendar.getInstance();
     String name = ""
       + c.get(Calendar.YEAR) + c.get(Calendar.MONTH) + c.get(Calendar.DAY_OF_MONTH)
       + c.get(Calendar.HOUR_OF_DAY) + c.get(Calendar.MINUTE) + c.get(Calendar.SECOND)
        + ".png";
     //合成完整路径,注意 / 分隔符
     String string = path.getPath() + "/" + name;
     view.saveToFile(string);
     Toast.makeText(this, "保存成功!\n文件保存在:" + string, Toast.LENGTH_LONG).show();
    } catch (FileNotFoundException e) {
     Toast.makeText(this, "保存失败!\n" + e, Toast.LENGTH_LONG).show();
    }
    break;
   }
  }
}

没什么难度,主要是将Bitmap转PNG图片那里,找了一会发现 Canvas 没有直接或间接保存的方法,刚好这里我使用了双缓冲,另一块画布的内容位图自己创建的,很自然想到将这个画布的位图保存为文件即可。

再查看 Bitmap 有个 compress(CompressFormat format, int quality,OutputStream stream) 方法,很明显将文件输出流传给这个方法就OK

@Override
public void onClick(View v) {
  // TODO Auto-generated method stub
  switch (v.getId()) {
   case R.id.iv_btn_clear :
    view.clear();
    break;
   case R.id.iv_btn_save : {
    try {
     String sdState = Environment.getExternalStorageState(); // 判断sd卡是否存在
     // 检查SD卡是否可用
     if (!sdState.equals(android.os.Environment.MEDIA_MOUNTED)) {
      Toast.makeText(this, "SD卡未准备好!", Toast.LENGTH_SHORT).show();
      break;
     }
     //获取系统图片存储路径
     File path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);
     // Make sure the Pictures directory exists.
     path.mkdirs();
     //根据当前时间生成图片名称
     Calendar c = Calendar.getInstance();
     String name = ""
       + c.get(Calendar.YEAR) + c.get(Calendar.MONTH) + c.get(Calendar.DAY_OF_MONTH)
       + c.get(Calendar.HOUR_OF_DAY) + c.get(Calendar.MINUTE) + c.get(Calendar.SECOND)
        + ".png";
     //合成完整路径,注意 / 分隔符
     String string = path.getPath() + "/" + name;
     view.saveToFile(string);
     Toast.makeText(this, "保存成功!\n文件保存在:" + string, Toast.LENGTH_LONG).show();
    } catch (FileNotFoundException e) {
     Toast.makeText(this, "保存失败!\n" + e, Toast.LENGTH_LONG).show();
    }
    break;
   }
  }
}

完整实例代码点击此处本站下载。

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

(0)

相关推荐

  • Android中使用HttpURLConnection实现GET POST JSON数据与下载图片

    Android6.0中把Apache HTTP Client所有的包与类都标记为deprecated不再建议使用所有跟HTTP相关的数据请求与提交操作都通过HttpURLConnection类实现,现实是很多Android开发者一直都Apache HTTP Client来做andoird客户端与后台HTTP接口数据交互,小编刚刚用HttpURLConnection做了一个android的APP,不小心踩到了几个坑,总结下最常用的就通过HttpURLConnection来POST提交JSON数据与

  • Android编程滑动效果之Gallery+GridView实现图片预览功能(附demo源码下载)

    本文实例讲述了Android编程滑动效果之Gallery+GridView实现图片预览功能.分享给大家供大家参考,具体如下: Android系统自带一个GridView和Gallery两个控件,GridView网格显示,Gallery单个浏览,两者结合起来可以真正实现Gallery浏览图片效果. 本示例通过GridView和Gallery两个控件,模仿实现一个完整的仿Gallery图像集的图片浏览效果.效果图如下: 1.GridView 首先,自定义一个GridImageAdapter图片适配器

  • Android 下载网络图片并显示到本地

    Android下载网络图片的流程是: 发送网络请求->将图片以流的形式下载下来->将流转换为Bitmap并赋给ImageView控件. 注意点 最新的Android系统不可以在主线程上请求网络,需要使用线程来请求 下载图片属于耗时任务,最优做法是放在一个AsyncTask中操作 设计思路 1.网络请求:该例中需要下载的文件类型是图片类型,可以将网络请求获取的数据类型转换为Bitmap已供ImageView直接使用,但是一个合理的网络请求类的设计是将下载的数据类型转换为最基本的InputStre

  • Android使用okHttp(get方式)下载图片

    一.首先下载Jar包 https://github.com/square/okhttp 如果使用android studio只需要加入依赖compile 'com.squareup.okhttp3:okhttp:3.2.0' 二.下载一张图片并显示 使用的是hanlder的方式 package com.liunan.okhttpdemo2; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import

  • Android编程实现图片的上传和下载功能示例

    本文实例讲述了Android编程实现图片的上传和下载功能.分享给大家供大家参考,具体如下: 在实现一个Android的WEB服务客户端,比如微博,论坛客户端时,经常会使用到图片的上传和下载.在这里介绍如何利用HttpClient实现图片的上传和下载功能. 1 图片上传:上传图片时,首先获得图片的路径,创建文件,并将图片转化为字节流写入到request,并发送该请求. 客户端代码: File file = new File(imageUrl); String httpUrl = httpDomai

  • SimpleCommand实现图片下载(二)

    使用simplecommand下载网络图片,并显示到ImageView控件上. 1 在app module的build.gradle将simplecommand框架进行导入,具体请看第1节的步骤1和2 2 设置网络图片的Url地址:这里以请求百度logo图片为例 String imageUrl = "https://www.baidu.com/img/bd_logo1.png"; 3 构建图片下载的进度监听器 ImageLoader.ProgressListener listener

  • Android中使用七牛云存储进行图片上传下载的实例代码

    Android开发中的图片存储本来就是比较耗时耗地的事情,而使用第三方的七牛云,便可以很好的解决这些后顾之忧,最近我也是在学习七牛的SDK,将使用过程在这记录下来,方便以后使用. 先说一下七牛云的存储原理,上面这幅图片是官方给出的原理图,表述当然比较清晰了. 可以看出,要进行图片上传的话可以分为五大步: 1. 客户端用户登录到APP的账号系统里面: 2. 客户端上传文件之前,需要向业务服务器申请七牛的上传凭证,这个凭证由业务服务器使用七牛提供的服务端SDK生成: 3. 客户端使用七牛提供的客户端

  • Android使用缓存机制实现文件下载及异步请求图片加三级缓存

    首先给大家介绍Android使用缓存机制实现文件下载 在下载文件或者在线浏览文件时,或者为了保证文件下载的正确性,需要使用缓存机制,常使用SoftReference来实现. SoftReference的特点是它的一个实例保存对一个Java对象的软引用,该软引用的存在不妨碍垃圾收集线程对该Java对象的回收.也就是说,一旦SoftReference保存了对一个Java对象的软引用后,在垃圾线程对这个Java对象回收前,SoftReference类所提供的get()方法返回Java对象的强引用.另外

  • Android 利用ViewPager实现图片可以左右循环滑动效果附代码下载

    首先给大家展示靓照,对效果图感兴趣的朋友可以继续往下阅读哦. ViewPager这个小demo实现的是可以左右循环滑动图片,下面带索引,滑到最后一页在往右滑动就要第一页,第一页往左滑动就到最后一页,上面是效果图,用美女图片是我一贯的作风,呵呵  1.    首先看一些layout下的xml <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width=&qu

  • Android中Glide实现超简单的图片下载功能

    本文介绍了Glide实现超简单的图片下载功能,具体步骤如下: 添加依赖 compile 'com.github.bumptech.glide:glide:3.7.0' 添加权限 <uses-permission android:name="android.permission.INTERNET"/> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/&

随机推荐