Android 实现手写板技术
Android手写板和涂鸦的功能,代码如下:
<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>
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();
}
}
}
}
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();
}
}
}
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();
}
});
}
}
/**
* 监听手写板对话框
* @author chengcj1
*
*/
public interface WriteDialogListener {
public void onPaintDone(Object object);
}
Android 实现手写板技术的更多相关文章
- Android的NDK技术
Android的NDK技术
- 本人讲课时录制的Android应用开发技术教学视频
网盘地址:http://yun.baidu.com/pcloud/album/info?query_uk=1963923831&album_id=3523786484935252365 本人讲 ...
- 【转】Android 防破解技术简介
http://www.cnblogs.com/likeandroid/p/4888808.html Android 防破解技术简介 这几年随着互联网的不断发展,Android App 也越来越多!但是 ...
- Android推送技术研究
前言 最近研究Android推送的实现, 研究了两天一夜, 有了一点收获, 写下来既为了分享, 也为了吐槽. 需要说明的是有些东西偏底层硬件和通信行业, 我对这些一窍不通, 只能说说自己的理解. 为什 ...
- 全面了解Android热修复技术
WeTest 导读 本文探讨了Android热修复技术的发展脉络,现状及其未来. 热修复技术概述 热修复技术在近年来飞速发展,尤其是在InstantRun方案推出之后,各种热修复技术竞相涌现.国内大部 ...
- Android热修复技术原理详解(最新最全版本)
本文框架 什么是热修复? 热修复框架分类 技术原理及特点 Tinker框架解析 各框架对比图 总结 通过阅读本文,你会对热修复技术有更深的认知,本文会列出各类框架的优缺点以及技术原理,文章末尾简单 ...
- 聊聊真实的 Android TV 开发技术栈
智能电视越来越普及了,华为说四月发布智能电视跳票了,一加也说今后要布局智能电视,在智能电视方向,小米已经算是先驱了.但是还有不少开发把智能电视简单的理解成手机屏幕的放大,其实这两者并不一样. 一.序 ...
- Android 插件化技术窥探
在Android 插件化技术中(宿主app和插件app设置相同的sharedUserId),动态加载apk有两种方式: 一种是将资源主题包的apk安装到手机上再读取apk内的资源,这种方式的原理是将宿 ...
- Android 防破解技术简介
Android 防破解技术简介 这几年随着互联网的不断发展,Android App 也越来越多!但是随之而来的问题也越来越多,这其中比较令人头疼的问题就是:有些不法分子利用反编译技术破解 App,修改 ...
随机推荐
- Sybase采用load table加载文本数据中的部分字段数据
LOAD TABLE语句实现从数据文件中装载部分列数据的方法.转载自:https://www.cnblogs.com/lizm166/p/8116475.html(有修改) LOAD TABLE:从数 ...
- mvc:view-controller标签使用
mvc:view-controller可以在不需要Controller处理request的情况,转向到设置的View,完成无Controller的path和view的直接映射. 1.重定向 <m ...
- 自动滚动标签marquee
<marquee>标签,它是成对出现的标签,首标签<marquee>和尾标签</marquee>之间的内容就是滚动内容.<marquee>标签的属性主要 ...
- js方法用来获取路径传参上所带的参数
//js方法用来获取路径传参上所带的参数 function GetQueryString(param) { var reg = new RegExp("(^|&)" + p ...
- springmvc 怎么响应json数据
springmvc 怎么响应json数据@Controller@RequestMapping("/items") class ItemsController{ @RequestM ...
- 20175213 2018-2019-2 《Java程序设计》第3周学习总结
## 教材学习内容总结 在第三周的学习过程中,我学习了第四章的内容. 第四章内容总结: 1.类是组成Java源文件的基本元素,一个源文件是由若干个类组成的. 2.成员变量分为实例变量和类变量.类变量被 ...
- 深度学习中,使用regularization正则化(weight_decay)的好处,loss=nan
刚开始训练一个模型,自己就直接用了,而且感觉训练的数据量也挺大的,因此就没有使用正则化, 可能用的少的原因,我也就不用了,后面,训练到一定程度,accuracy不上升,loss不下降,老是出现loss ...
- aio,nio ,io 心得
1.nio 流的过程有几个,连接,可读,读 ,返回 :连接了不一定可读,等待浪费时间,这些时间可以去读其他的连接,selector是管理,管理全部测一下可不可读,只对可读的连接进行读取.同时,nio有 ...
- cdn是否缓存了网站内容,如何查看
查看网站是否缓存,本例以cdnbest的节点程序,浏览器是firefox为例 打开浏览器后按 F12键,再打开网站,如下图显示 HIT就表示有缓存,没有缓存就会显示MISS
- 大数据入门到精通12--spark dataframe 注册成hive 的临时表
一.获得最初的数据并形成dataframe val ny= sc.textFile("data/new_york/")val header=ny.firstval filterNY ...