热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

简单实现Android绘图板

这篇文章主要教大家如何简单实现Android绘图板,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文这个实例通过前面学过的Paint、Canvas等2D绘画技术来实现一个简单的Android的绘图板。

具体实现代码:

创建一个名为DrawView的类,该类继承自android.view.View类。在该类中,首先定义程序中所需的属性,然后添加构造方法,并重写onDraw(Canvas canvas)方法:

DrawView.java:

package com.example.test; 
 
import android.content.Context; 
import android.graphics.Bitmap; 
import android.graphics.Canvas; 
import android.graphics.Paint; 
import android.graphics.Path; 
import android.util.AttributeSet; 
import android.view.View; 
 
public class DrawView extends View{ 
 private int view_width=0;//屏幕的宽度 
 private int view_height=0;//屏幕的高度 
 private float preX;//起始点的x坐标 
 private float preY;//起始点的y坐标 
 private Path path;//路径 
 public Paint paint;//画笔 
 Bitmap cacheBitmap=null;//定义一个内存中的图片,该图片将作为缓冲区 
 Canvas cacheCanvas=null;//定义cacheBitmap上的Canvas对象 
 /* 
 * 功能:构造方法 
 * */ 
 public DrawView(Context context, AttributeSet attrs) { 
 super(context, attrs); 
  
 } 
 
 /* 
 * 功能:重写onDraw方法 
 * */ 
 @Override 
 protected void onDraw(Canvas canvas) { 
 super.onDraw(canvas); 
 
 } 
} 

创建布局文件,选择帧布局,并加入上面创建的继承了View的自定义画图控件:
res/layout/main.xml

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 

在DrawView类的构造方法中,首先获取屏幕的高度和宽度,并创建一个与该View相同大小的缓存区,然后创建一个新的画面,并实例化一个路径,再将内存中的位图绘制到cacheCanvas中,最后实例化一个画笔,并设置画笔的相关属性。
关键代码如下:

/* 
 * 功能:构造方法 
 * */ 
 public DrawView(Context context, AttributeSet attrs) { 
 super(context, attrs); 
 view_width=context.getResources().getDisplayMetrics().widthPixels;//获取屏幕宽度 
 view_height=context.getResources().getDisplayMetrics().heightPixels;//获取屏幕高度 
 //创建一个与该View相同大小的缓存区 
 cacheBitmap=Bitmap.createBitmap(view_width,view_height,Config.ARGB_8888); 
 cacheCanvas=new Canvas();//创建一个新的画布 
 path=new Path(); 
 //在cacheCanvas上绘制cacheBitmap 
 cacheCanvas.setBitmap(cacheBitmap); 
 paint=new Paint(Paint.DITHER_FLAG);//Paint.DITHER_FLAG防抖动的 
 paint.setColor(Color.RED); 
 //设置画笔风格 
 paint.setStyle(Paint.Style.STROKE);//设置填充方式为描边 
 paint.setStrokeJoin(Paint.Join.ROUND);//设置笔刷转弯处的连接风格 
 paint.setStrokeCap(Paint.Cap.ROUND);//设置笔刷的图形样式(体现在线的端点上) 
 paint.setStrokeWidth(1);//设置默认笔触的宽度为1像素 
 paint.setAntiAlias(true);//设置抗锯齿效果 
 paint.setDither(true);//使用抖动效果 
 } 

在DrawView类的onDraw()方法中,添加以下代码,用于设置背景颜色、绘制cacheBitmap、绘制路径以及保存当前绘图状态到栈中,并调用restore()方法恢复所保存的状态,关键代码如下:

/* 
 * 功能:重写onDraw方法 
 * */ 
 @Override 
 protected void onDraw(Canvas canvas) { 
 super.onDraw(canvas); 
 canvas.drawColor(0xFFFFFFFF);//设置背景色 
 Paint bmpPaint=new Paint();//采用默认设置创建一个画笔 
 canvas.drawBitmap(cacheBitmap, 0, 0,bmpPaint);//绘制cacheBitmap 
 canvas.drawPath(path, paint);//绘制路径 
 canvas.save(Canvas.ALL_SAVE_FLAG);//保存canvas的状态 
 //恢复canvas之前保存的状态,防止保存后对canvas执行的操作对后续的绘制有影响 
 canvas.restore(); 
 } 

在Draw类中,重写onTouchEvent()方法,为该视图添加触摸事件监听器,在该方法中,首先获取触摸事件发生的位置,然后用switch语句对事件的不同状态添加响应代码,最后调用invalidate()方法更新视图。具体代码如下:

@Override 
 public boolean onTouchEvent(MotionEvent event) { 
 //获取触摸事件发生的位置 
 float x=event.getX(); 
 float y=event.getY(); 
 switch(event.getAction()){ 
  case MotionEvent.ACTION_DOWN: 
  //将绘图的起始点移到(x,y)坐标点的位置 
  path.moveTo(x, y); 
  preX=x; 
  preY=y; 
  break; 
  case MotionEvent.ACTION_MOVE: 
  //保证横竖绘制距离不能超过625 
  float dx=Math.abs(x-preX); 
  float dy=Math.abs(y-preY); 
  if(dx>5||dy>5){ 
   //.quadTo贝塞尔曲线,实现平滑曲线(对比lineTo) 
   //x1,y1为控制点的坐标值,x2,y2为终点的坐标值 
   path.quadTo(preX, preY, (x+preX)/2, (y+preY)/2); 
   preX=x; 
   preY=y; 
  } 
  break; 
  case MotionEvent.ACTION_UP: 
  cacheCanvas.drawPath(path, paint);//绘制路径 
  path.reset(); 
  break; 
 } 
 invalidate(); 
 return true;//返回true,表明处理方法已经处理该事件 
 } 

编写clear()方法,用于实现橡皮擦功能,具体代码如下:

public void clear(){ 
 //设置图形重叠时的处理方式 
 paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); 
 //设置笔触的宽度 
 paint.setStrokeWidth(50); 
 } 

编写保存当前绘图的save方法,在该方法中,调用saveBitmap()方法将当前绘图保存为PNG图片。save()方法的具体代码如下:

public void save(){ 
 try{ 
  saveBitmap("myPitcture"); 
 }catch(IOException e){ 
  e.printStackTrace(); 
 } 
  
 } 

编写保存绘制好的位图的方法saveBitmap(),在该方法中,首先在SD卡上创建一个文件,然后创建一个文件输出流对象,并调用Bitmap类的compress()方法将绘图内容压缩为PNG格式输出到刚刚创建的文件输出流对象中,最后将缓冲区的数据全部写出到输出流中,并关闭文件输出流对象。saveBitmap()方法的具体代码如下:

private void saveBitmap(String fileName) throws IOException { 
 File file=new File(getSDPath()+fileName+".png"); 
 file.createNewFile(); 
 FileOutputStream fileOS=new FileOutputStream(file); 
 //将绘图内容压缩为PNG格式输出到输出流对象中 
 cacheBitmap.compress(Bitmap.CompressFormat.PNG, 100, fileOS); 
 fileOS.flush();//将缓冲区中的数据全部写出到输出流中 
 fileOS.close();//关闭文件输出流对象 
 } 
 
 //获得SD卡的根目录 
 public String getSDPath(){ 
  File sdDir = null; 
  boolean sdCardExist = Environment.getExternalStorageState() 
    .equals(android.os.Environment.MEDIA_MOUNTED); //判断sd卡是否存在 
 
  if (sdCardExist) //如果SD卡存在,则获取跟目录 
  {     
  sdDir = Environment.getExternalStorageDirectory();//获取跟目录 
  } 
  return sdDir.toString(); 
  
 } 

在程序中需要向SD卡上保存文件,那么需要在AndroidManifest.xml文件中赋予相应的权限,具体代码入下:

 
 
 
 

在res目录中,创建一个menu目录,并在该目录中创建一个名称为toolsmenu.xml的菜单资源文件,在该文件中编写实例中所应用的功能菜单,关键代码如下:

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 
  
  
   
   
   
   
   
   
   
  
  
  
  
   
   
   
   
   
   
  
  
  
  
 

其中values/strings.xml中:

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 
 
 test 
 Hello world! 
 Settings 
 画笔颜色 
 红色 
 绿色 
 蓝色 
 画笔宽度 
  
  
  
 擦除绘画 
 保存绘画 
 
 

在默认创建的MainActivity中,为实例添加选项菜单。
首先,重写onCreatOptionsMenu()方法,在该方法中,实例化一个MenuInflater对象,并调用该对象的inflate方法解析toolsmenu.xml的菜单资源文件。具体代码如下:

/* 
 * 创建选项菜单 
 * */ 
 @Override 
 public boolean onCreateOptionsMenu(Menu menu) { 
 MenuInflater inflator=new MenuInflater(this); 
 inflator.inflate(R.menu.toolsmenu, menu); 
 return super.onCreateOptionsMenu(menu); 
 } 

然后,重写onOptionsItemSelected方法,分别对各个菜单项被选择时做出相应的处理,具体代码如下:

/* 
 * 当菜单项被选择时,做出相应的处理 
 * */ 
 @Override 
 public boolean onOptionsItemSelected(MenuItem item) { 
 //获取自定义的绘图视图 
 DrawView dv=(DrawView)findViewById(R.id.drawView1); 
 dv.paint.setXfermode(null);//取消擦除效果 
 dv.paint.setStrokeWidth(1);//初始化画笔的宽度 
 switch(item.getItemId()){ 
  case R.id.red: 
  dv.paint.setColor(Color.RED);//设置笔的颜色为红色 
  item.setChecked(true); 
  break; 
  case R.id.green: 
  dv.paint.setColor(Color.GREEN);//设置笔的颜色为绿色 
  item.setChecked(true); 
  break; 
  case R.id.blue: 
  dv.paint.setColor(Color.BLUE);//设置笔的颜色为蓝色 
  item.setChecked(true); 
  break; 
  case R.id.width_1: 
  dv.paint.setStrokeWidth(1);//设置笔触的宽度为1像素 
  break; 
  case R.id.width_2: 
  dv.paint.setStrokeWidth(5);//设置笔触的宽度为5像素 
  break; 
  case R.id.width_3: 
  dv.paint.setStrokeWidth(10);//设置笔触的宽度为10像素 
  break; 
  case R.id.clear: 
  dv.clear();//擦除绘画 
  break; 
  case R.id.save: 
  dv.save();//保存绘画 
  break; 
 } 
 return true; 
 } 

运行效果如图

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • Android 九宫格布局详解及实现:人人网应用示例
    本文深入探讨了人人网Android应用中独特的九宫格布局设计,解析其背后的GridView实现原理,并提供详细的代码示例。这种布局方式不仅美观大方,而且在现代Android应用中较为少见,值得开发者借鉴。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 深入理解 Oracle 存储函数:计算员工年收入
    本文介绍如何使用 Oracle 存储函数查询特定员工的年收入。我们将详细解释存储函数的创建过程,并提供完整的代码示例。 ... [详细]
  • 本文总结了2018年的关键成就,包括职业变动、购车、考取驾照等重要事件,并分享了读书、工作、家庭和朋友方面的感悟。同时,展望2019年,制定了健康、软实力提升和技术学习的具体目标。 ... [详细]
  • CSS 布局:液态三栏混合宽度布局
    本文介绍了如何使用 CSS 实现液态的三栏布局,其中各栏具有不同的宽度设置。通过调整容器和内容区域的属性,可以实现灵活且响应式的网页设计。 ... [详细]
  • Python自动化处理:从Word文档提取内容并生成带水印的PDF
    本文介绍如何利用Python实现从特定网站下载Word文档,去除水印并添加自定义水印,最终将文档转换为PDF格式。该方法适用于批量处理和自动化需求。 ... [详细]
  • 将Web服务部署到Tomcat
    本文介绍了如何在JDeveloper 12c中创建一个Java项目,并将其打包为Web服务,然后部署到Tomcat服务器。内容涵盖从项目创建、编写Web服务代码、配置相关XML文件到最终的本地部署和验证。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • 本文介绍如何在 Unity 的 XML 配置文件中,将参数传递给自定义生命周期管理器的构造函数。我们将详细探讨 CustomLifetimeManager 类的实现及其配置方法。 ... [详细]
  • 本文详细介绍了 Java 中 org.apache.xmlbeans.SchemaType 类的 getBaseEnumType() 方法,提供了多个代码示例,并解释了其在不同场景下的使用方法。 ... [详细]
  • 本文详细介绍了如何解决MyBatis中常见的BindingException错误,提供了多种排查和修复方法,确保Mapper接口与XML文件的正确配置。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
author-avatar
百万家庭
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有