我正在尝试使用 Bezier 绘制两个数字签名,如上图所示。当我触摸并尝试绘制线时,结果是点线但没有得到连续线。
使用 simple signature 完成的简单签名
但我想使用带有触摸压力的贝塞尔曲线创建更平滑的曲线。
tried with this link
SignatureViewDemo.java
public class SignatureViewDemo extends View {
private int color = Color.BLACK;
private Bitmap m_Bitmap;
private final Paint m_BorderPaint;
private Canvas m_Canvas;
private Point m_CropBotRight;
private Point m_CropTopLeft;
private float m_CurrentX;
private float m_CurrentY;
private final float m_DesiredDash;
private float m_LastWidth = 6.5F;
private Paint m_PenPaint;
private int m_PointIndex = 0;
private ArrayList<Point> m_Points = new ArrayList<Point>();
private final float m_StrokeWidth;
boolean m_Empty;
public SignatureViewDemo(Context paramContext) {
this(paramContext, null);
}
public SignatureViewDemo(Context paramContext,
AttributeSet paramAttributeSet) {
this(paramContext, paramAttributeSet, 0);
}
public SignatureViewDemo(Context paramContext,
AttributeSet paramAttributeSet, int paramInt) {
super(paramContext, paramAttributeSet, paramInt);
setFocusable(true);
this.m_PenPaint = new Paint();
this.m_PenPaint.setAntiAlias(true);
this.m_PenPaint.setColor(Color.BLACK);
this.m_PenPaint.setStrokeWidth(5.0F);
this.m_PenPaint.setStrokeJoin(Paint.Join.ROUND);
this.m_PenPaint.setStrokeCap(Paint.Cap.ROUND);
this.m_CurrentY = (0.0F / 0.0F);
this.m_CurrentX = (0.0F / 0.0F);
this.m_StrokeWidth = 5.0F;
this.m_DesiredDash = 10.0F;
this.m_BorderPaint = new Paint();
this.m_BorderPaint.setColor(Color.BLACK);
this.m_BorderPaint.setStyle(Paint.Style.STROKE);
this.m_BorderPaint.setStrokeWidth(this.m_StrokeWidth);
}
public void addBezier(Bezier paramBezier, float paramFloat1,
float paramFloat2) {
if (this.m_Bitmap == null) {
this.m_Bitmap = Bitmap.createBitmap(getWidth(), getHeight(),
Bitmap.Config.ARGB_8888);
this.m_Canvas = new Canvas(this.m_Bitmap);
}
paramBezier.draw(this.m_Canvas, this.m_PenPaint, paramFloat1,
paramFloat2);
}
public void addPoint(Point paramPoint) {
if ((paramPoint.getX() < this.m_CropTopLeft.getX())
&& (paramPoint.getX() >= 0.0F))
this.m_CropTopLeft.setX(paramPoint.getX());
if ((paramPoint.getY() < this.m_CropTopLeft.getY())
&& (paramPoint.getY() >= 0.0F))
this.m_CropTopLeft.setY(paramPoint.getY());
if ((paramPoint.getX() > this.m_CropBotRight.getX())
&& (paramPoint.getX() <= this.m_Canvas.getWidth()))
this.m_CropBotRight.setX(paramPoint.getX());
if ((paramPoint.getY() > this.m_CropBotRight.getY())
&& (paramPoint.getY() <= this.m_Canvas.getHeight()))
this.m_CropBotRight.setY(paramPoint.getY());
this.m_Points.add(paramPoint);
drawPoints();
}
// public void clear() {
// if (this.m_Canvas == null)
// this.m_Canvas.drawColor(0, PorterDuff.Mode.CLEAR);
// invalidate();
// // return;
// }
public void clear() {
if (this.m_Canvas == null)
return;
while (m_Empty) {
m_Canvas.drawColor(0, PorterDuff.Mode.CLEAR);
m_Empty = true;
invalidate();
return;
}
}
public void drawBitmap(Bitmap paramBitmap) {
clear();
if ((paramBitmap != null) && (this.m_Canvas != null)
&& (this.m_Canvas.getWidth() != 0)
&& (this.m_Canvas.getHeight() != 0)) {
Matrix localMatrix = new Matrix();
localMatrix.setRectToRect(
new RectF(0.0F, 0.0F, paramBitmap.getWidth(), paramBitmap
.getHeight()),
new RectF(0.0F, 0.0F, this.m_Canvas.getWidth(),
this.m_Canvas.getHeight()),
Matrix.ScaleToFit.CENTER);
this.m_Canvas.drawBitmap(paramBitmap, localMatrix, null);
m_Empty = false;
}
invalidate();
}
public void drawPoints() {
if ((m_Points.size() >= 4)
&& (4 + this.m_PointIndex <= this.m_Points.size())) {
Point localPoint1 = (Point) this.m_Points.get(this.m_PointIndex);
Point localPoint2 = (Point) this.m_Points
.get(1 + this.m_PointIndex);
Point localPoint3 = (Point) this.m_Points
.get(2 + this.m_PointIndex);
Point localPoint4 = (Point) this.m_Points
.get(3 + this.m_PointIndex);
Bezier localBezier = new Bezier(localPoint1, localPoint2,
localPoint3, localPoint4);
localBezier.setColor(Color.GREEN);
float f = strokeWidth(8.0F / localPoint4.velocityFrom(localPoint1));
addBezier(localBezier, this.m_LastWidth, f);
invalidate();
this.m_LastWidth = f;
this.m_PointIndex = (3 + this.m_PointIndex);
m_Empty = false;
}
}
public boolean isEmpty() {
return m_Empty;
}
public Bitmap getBitmap() {
return this.m_Bitmap;
}
public int getColor() {
return this.color;
}
protected void onDraw(Canvas paramCanvas) {
if (this.m_Bitmap != null)
paramCanvas.drawBitmap(this.m_Bitmap, 0.0F, 0.0F, null);
}
protected void onMeasure(int paramInt1, int paramInt2) {
int i = View.MeasureSpec.getSize(paramInt1);
int j = View.MeasureSpec.getSize(paramInt2);
this.m_CropTopLeft = new Point(i, j);
this.m_CropBotRight = new Point(0.0F, 0.0F);
setMeasuredDimension(i, j);
}
protected void onSizeChanged(int paramInt1, int paramInt2, int paramInt3,
int paramInt4) {
Bitmap localBitmap = Bitmap.createBitmap(paramInt1, paramInt2,
Bitmap.Config.ARGB_8888);
this.m_Canvas = new Canvas(localBitmap);
float f1 = 2.0F * (this.m_Canvas.getWidth() + this.m_Canvas.getHeight() - 2.0F * this.m_StrokeWidth);
float f2 = f1
* this.m_DesiredDash
/ (Math.round(f1 / (4.0F * this.m_DesiredDash)) * (4.0F * this.m_DesiredDash));
Paint localPaint = this.m_BorderPaint;
float[] arrayOfFloat = new float[2];
arrayOfFloat[0] = f2;
arrayOfFloat[1] = f2;
localPaint.setPathEffect(new DashPathEffect(arrayOfFloat, f2 / 2.0F));
clear();
if (this.m_Bitmap != null) {
Rect localRect = new Rect(0, 0, this.m_Canvas.getWidth(),
this.m_Canvas.getHeight());
this.m_Canvas.drawBitmap(this.m_Bitmap, null, localRect, null);
m_Empty = false;
}
this.m_Bitmap = localBitmap;
}
public boolean onTouchEvent(MotionEvent paramMotionEvent) {
int i = 0xFF & paramMotionEvent.getAction();
if (i == 0) {
this.m_CurrentX = paramMotionEvent.getX();
this.m_CurrentY = paramMotionEvent.getY();
addPoint(new Point(this.m_CurrentX, this.m_CurrentY,
paramMotionEvent.getEventTime()));
getParent().requestDisallowInterceptTouchEvent(true);
}
// while (m_Empty) {
if ((i == 1) || (i == 3)) {
this.m_CurrentY = (0.0F / 0.0F);
this.m_CurrentX = (0.0F / 0.0F);
this.m_Points.clear();
this.m_PointIndex = 0;
getParent().requestDisallowInterceptTouchEvent(false);
}
// if ((this.m_Points.size() < 4) || (4 + this.m_PointIndex >
// this.m_Points.size()))
// while (1 + this.m_PointIndex <= this.m_Points.size())
drawPoints();
if ((i == 2) || (i == 1)) {
for (int j = 0; j < paramMotionEvent.getHistorySize(); j++)
addPoint(new Point(paramMotionEvent.getHistoricalX(j),
paramMotionEvent.getHistoricalY(j),
paramMotionEvent.getHistoricalEventTime(j)));
addPoint(new Point(paramMotionEvent.getX(),
paramMotionEvent.getY(), paramMotionEvent.getEventTime()));
}
// }
return true;
}
public void setColor(int paramInt) {
this.color = Color.BLACK;
}
public Point getCropBotRight() {
return this.m_CropBotRight;
}
public Point getCropTopLeft() {
return this.m_CropTopLeft;
}
// public float strokeWidth(float paramFloat) {
// if (paramFloat > 11.0F)
// paramFloat = 10.0F;
// if (paramFloat < 5.0F)
// paramFloat = 6.0F;
// return paramFloat;
// }
public float strokeWidth(float paramFloat) {
if (paramFloat > 11.0F)
paramFloat = 10.0F;
while (m_Empty) {
if (paramFloat < 5.0F)
paramFloat = 6.0F;
return paramFloat;
}
return paramFloat;
}
}
Point.java
private long time;
private float x;
private float y;
public Point(float paramFloat1, float paramFloat2) {
this.x = paramFloat1;
this.y = paramFloat2;
}
public Point(float paramFloat1, float paramFloat2, long paramLong) {
this.x = paramFloat1;
this.y = paramFloat2;
this.time = paramLong;
}
protected float distanceTo(Point paramPoint) {
float f1 = this.x - paramPoint.getX();
float f2 = this.y - paramPoint.getY();
return FloatMath.sqrt(f1 * f1 + f2 * f2);
}
public long getTime() {
return this.time;
}
public float getX() {
return this.x;
}
public float getY() {
return this.y;
}
public void setX(float paramFloat) {
this.x = paramFloat;
}
public void setY(float paramFloat) {
this.y = paramFloat;
}
public float velocityFrom(Point paramPoint) {
return distanceTo(paramPoint) / (this.time - paramPoint.time);
}
}
Bezier.java
public class Bezier {
private Point controlPointOne;
private Point controlPointTwo;
private int drawSteps;
private Point endPoint;
private int mColor;
private Point startPoint;
public Bezier() {
}
public Bezier(Point paramPoint1, Point paramPoint2, Point paramPoint3,
Point paramPoint4) {
this.startPoint = paramPoint1;
this.controlPointOne = paramPoint2;
this.controlPointTwo = paramPoint3;
this.endPoint = paramPoint4;
this.drawSteps = ((int) (paramPoint1.distanceTo(paramPoint2)
+ paramPoint2.distanceTo(paramPoint3) + paramPoint3
.distanceTo(paramPoint4)));
}
public void draw(Canvas paramCanvas, Paint paramPaint, float paramFloat1,
float paramFloat2) {
float f1 = paramFloat2 - paramFloat1;
for (int i = 0; i < this.drawSteps; i++) {
float f2 = i / this.drawSteps;
float f3 = f2 * f2;
float f4 = f3 * f2;
float f5 = 1.0F - f2;
float f6 = f5 * f5;
float f7 = f6 * f5;
float f8 = f7 * this.startPoint.getX() + f2 * (3.0F * f6)
* this.controlPointOne.getX() + f3 * (3.0F * f5)
* this.controlPointTwo.getX() + f4 * this.endPoint.getX();
float f9 = f7 * this.startPoint.getY() + f2 * (3.0F * f6)
* this.controlPointOne.getY() + f3 * (3.0F * f5)
* this.controlPointTwo.getY() + f4 * this.endPoint.getY();
paramPaint.setColor(getColor());
paramPaint.setStrokeWidth(paramFloat1 + f4 * f1);
paramCanvas.drawPoint(f8, f9, paramPaint);
}
}
public int getColor() {
return this.mColor;
}
public Point getControlPointOne() {
return this.controlPointOne;
}
public Point getControlPointTwo() {
return this.controlPointTwo;
}
public int getDrawSteps() {
return this.drawSteps;
}
public Point getEndPoint() {
return this.endPoint;
}
public Point getStartPoint() {
return this.startPoint;
}
public void setColor(int paramInt) {
this.mColor = Color.BLACK;
}
public void setControlPointOne(Point paramPoint) {
this.controlPointOne = paramPoint;
}
public void setControlPointTwo(Point paramPoint) {
this.controlPointTwo = paramPoint;
}
public void setDrawSteps(int paramInt) {
this.drawSteps = paramInt;
}
public void setEndPoint(Point paramPoint) {
this.endPoint = paramPoint;
}
public void setStartPoint(Point paramPoint) {
this.startPoint = paramPoint;
}
}
最佳答案
将Bezier类中的draw方法改为:
public void draw(Canvas canvas, Paint paint, float startWidth, float endWidth) {
float originalWidth = paint.getStrokeWidth();
float widthDelta = endWidth - startWidth;
for (int i = 0; i < drawSteps; i++) {
float t = ((float) i) / drawSteps;
float tt = t * t;
float ttt = tt * t;
float u = 1 - t;
float uu = u * u;
float uuu = uu * u;
float x = uuu * startPoint.x;
x += 3 * uu * t * getControlPointOne().x;
x += 3 * u * tt * getControlPointTwo().x;
x += ttt * endPoint.x;
float y = uuu * startPoint.y;
y += 3 * uu * t * getControlPointOne().y;
y += 3 * u * tt * getControlPointTwo().y;
y += ttt * endPoint.y;
paint.setStrokeWidth(startWidth + ttt * widthDelta);
canvas.drawPoint(x, y, paint);
}
paint.setStrokeWidth(originalWidth);
}
关于java - Android:使用贝塞尔的数字签名,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16455896/
我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div
我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看rubyzip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于
我正在尝试使用ruby和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t
我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h
我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po