在 Android 上的 Rhodes 中将签名图片保存到文件中

Posted

技术标签:

【中文标题】在 Android 上的 Rhodes 中将签名图片保存到文件中【英文标题】:Save a picture of a signature to a file in Rhodes on Android 【发布时间】:2012-02-19 08:08:08 【问题描述】:

我正在尝试使用针对 android 选项卡的 Rhodes 实现签名捕获。我设法拿到画布并在上面涂鸦。但是我无法将签名保存在所需的位置。 signature_uri 采用默认位置为db/db-files/Image-XXXXXX.png

这是示例中的代码:

def signature_callback
if @params['status'] == 'ok'
  #create signature record in the DB
  signature = SignatureUtil.new('signature_uri'=>@params['signature_uri'])
  signature.save
  puts "new Signature object: " + signature.inspect
end

在我得到的控制台中

APP| RHO serve: /app/Settings/signature_callback
I/APP     (  801): I 01/26/2012 11:36:20:236 0000032e                  APP| Params: "status"=>"ok", "signature_uri"=>"db/db-files/Image_20120126113618375.png", "rho_callback"=>"1"
I/APP     (  801): I 01/26/2012 11:36:20:238 0000032e                  APP| *******************ok****************
I/APP     (  801): I 01/26/2012 11:36:20:238 0000032e                  APP| %%%%%%%%%%%%%%%%%db/db-files/Image_20120126113618375.png%%%%%%%%%%%
I/APP     (  801): I 01/26/2012 11:36:20:239 0000032e                  APP| App error: can't convert Symbol into Integer
I/APP     (  801):  lib/rhom/rhom_object_factory.rb:67:in `[]'
I/APP     (  801): lib/rhom/rhom_object_factory.rb:67:in `initialize'
I/APP     (  801): apps/app/Settings/controller.rb:34:in `new'
I/APP     (  801): apps/app/Settings/controller.rb:34:in `signature_callback'
I/APP     (  801): lib/rho/rhocontroller.rb:91:in `serve'
I/APP     (  801): lib/rho/rhoapplication.rb:209:in `serve'
I/APP     (  801): lib/rho/rho.rb:822:in `serve'
I/APP     (  801): E 01/26/2012 11:36:20:248 00000321                  Net| Request failed. HTTP Code: 500 returned. HTTP Response:         <html>
I/APP     (  801):             <head>
I/APP     (  801):                 <meta name="viewport" content="width=320"/>
I/APP     (  801):             </head>
I/APP     (  801):             <body>
I/APP     (  801):                 <h2>Server Error</h2>
I/APP     (  801):                 <p>
I/APP     (  801): Error: can't convert Symbol into Integer<br/>Trace: <br/

现在我想将它保存在所需的位置,而不是这个数据库存储。请分享您的观点。

【问题讨论】:

你能给我们看看痕迹吗?其他人:请阅读问题。他已经捕获了签名,但他想和罗德一起保存它。这是真正的问题,不是“如何在 Android 上捕获签名”。 【参考方案1】:

只需尝试在签名 uri 中传递 sd 卡位置。 请注意在 manifest 中给予 sd 卡读写权限。

否则

将签名文件从内部包复制到所需位置,但这是重复工作。

【讨论】:

感谢 Manish 的回复,并为迟到的回复道歉。试过了,但没有运气。【参考方案2】:

您是否查看过此页面:http://docs.rhomobile.com/rhodes/device-caps#file-system-access?在我看来,您需要将图像保存到数据库并使用 Rho::RhoApplication::get_blob_path(image.image_uri) 方法打开文件。然后您应该能够使用文件系统 API 创建一个新文件并将图像的内容写入您选择的任何位置。

【讨论】:

这与我们在项目中所做的非常接近。基本上 signature_uri 返回已保存图像的位置(我不知何故无法访问)。我们将此位置作为图像源推送到 Web 服务并在数据库中创建记录。通过正在存储的文件的大小验证正在保存的图像。笔画数与图像大小成正比。【参考方案3】:

下面的代码可能对你有用

import java.io.File;
import java.io.FileOutputStream;
import java.util.Calendar;
import android.app.Activity; 
import android.content.Context;
import android.content.ContextWrapper;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.RectF;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore.Images;
import android.util.AttributeSet;
import android.util.Log;
import android.view.Gravity;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup.LayoutParams;
import android.view.Window;
import android.widget.Button;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.Toast;

 public class CaptureSignature extends Activity 

LinearLayout mContent;
signature mSignature;
Button mClear, mGetSign, mCancel;
public static String tempDir;
public int count = 1;
public String current = null;
private Bitmap mBitmap;
View mView;
File mypath;

private String uniqueId;
private EditText yourName;

@Override
public void onCreate(Bundle savedInstanceState)

    super.onCreate(savedInstanceState);
    this.requestWindowFeature(Window.FEATURE_NO_TITLE);
    setContentView(R.layout.signature);

    tempDir = Environment.getExternalStorageDirectory() + "/" + getResources().getString(R.string.external_dir) + "/";
    ContextWrapper cw = new ContextWrapper(getApplicationContext());
    File directory = cw.getDir(getResources().getString(R.string.external_dir), Context.MODE_PRIVATE);

    prepareDirectory();
    uniqueId = getTodaysDate() + "_" + getCurrentTime() + "_" + Math.random();
    current = uniqueId + ".png";
    mypath= new File(directory,current);


    mContent = (LinearLayout) findViewById(R.id.linearLayout);
    mSignature = new signature(this, null);
    mSignature.setBackgroundColor(Color.WHITE);
    mContent.addView(mSignature, LayoutParams.FILL_PARENT, LayoutParams.FILL_PARENT);
    mClear = (Button)findViewById(R.id.clear);
    mGetSign = (Button)findViewById(R.id.getsign);
    mGetSign.setEnabled(false);
    mCancel = (Button)findViewById(R.id.cancel);
    mView = mContent;

    yourName = (EditText) findViewById(R.id.yourName);

    mClear.setOnClickListener(new OnClickListener()
           
        public void onClick(View v)
        
            Log.v("log_tag", "Panel Cleared");
            mSignature.clear();
            mGetSign.setEnabled(false);
        
    );

    mGetSign.setOnClickListener(new OnClickListener()
           
        public void onClick(View v)
        
            Log.v("log_tag", "Panel Saved");
            boolean error = captureSignature();
            if(!error)
                mView.setDrawingCacheEnabled(true);
                mSignature.save(mView);
                Bundle b = new Bundle();
                b.putString("status", "done");
                Intent intent = new Intent();
                intent.putExtras(b);
                setResult(RESULT_OK,intent);  
                finish();
            
        
    );

    mCancel.setOnClickListener(new OnClickListener()
           
        public void onClick(View v)
        
            Log.v("log_tag", "Panel Canceled");
            Bundle b = new Bundle();
            b.putString("status", "cancel");
            Intent intent = new Intent();
            intent.putExtras(b);
            setResult(RESULT_OK,intent); 
            finish();
        
    );



@Override
protected void onDestroy() 
    Log.w("GetSignature", "onDestory");
    super.onDestroy();


private boolean captureSignature() 

    boolean error = false;
    String errorMessage = "";


    if(yourName.getText().toString().equalsIgnoreCase(""))
        errorMessage = errorMessage + "Please enter your Name\n";
        error = true;
      

    if(error)
        Toast toast = Toast.makeText(this, errorMessage, Toast.LENGTH_SHORT);
        toast.setGravity(Gravity.TOP, 105, 50);
        toast.show();
    

    return error;


private String getTodaysDate() 

    final Calendar c = Calendar.getInstance();
    int todaysDate =     (c.get(Calendar.YEAR) * 10000) +
    ((c.get(Calendar.MONTH) + 1) * 100) +
    (c.get(Calendar.DAY_OF_MONTH));
    Log.w("DATE:",String.valueOf(todaysDate));
    return(String.valueOf(todaysDate));



private String getCurrentTime() 

    final Calendar c = Calendar.getInstance();
    int currentTime =     (c.get(Calendar.HOUR_OF_DAY) * 10000) +
    (c.get(Calendar.MINUTE) * 100) +
    (c.get(Calendar.SECOND));
    Log.w("TIME:",String.valueOf(currentTime));
    return(String.valueOf(currentTime));




private boolean prepareDirectory()

    try
    
        if (makedirs())
        
            return true;
         else 
            return false;
        
     catch (Exception e)
    
        e.printStackTrace();
        Toast.makeText(this, "Could not initiate File System.. Is Sdcard mounted properly?", 1000).show();
        return false;
    


private boolean makedirs()

    File tempdir = new File(tempDir);
    if (!tempdir.exists())
        tempdir.mkdirs();

    if (tempdir.isDirectory())
    
        File[] files = tempdir.listFiles();
        for (File file : files)
        
            if (!file.delete())
            
                System.out.println("Failed to delete " + file);
            
        
    
    return (tempdir.isDirectory());


public class signature extends View

    private static final float STROKE_WIDTH = 5f;
    private static final float HALF_STROKE_WIDTH = STROKE_WIDTH / 2;
    private Paint paint = new Paint();
    private Path path = new Path();

    private float lastTouchX;
    private float lastTouchY;
    private final RectF dirtyRect = new RectF();

    public signature(Context context, AttributeSet attrs)
    
        super(context, attrs);
        paint.setAntiAlias(true);
        paint.setColor(Color.BLACK);
        paint.setStyle(Paint.Style.STROKE);
        paint.setStrokeJoin(Paint.Join.ROUND);
        paint.setStrokeWidth(STROKE_WIDTH);
    

    public void save(View v)
    
        Log.v("log_tag", "Width: " + v.getWidth());
        Log.v("log_tag", "Height: " + v.getHeight());
        if(mBitmap == null)
        
            mBitmap =  Bitmap.createBitmap (mContent.getWidth(), mContent.getHeight(), Bitmap.Config.RGB_565);;
        
        Canvas canvas = new Canvas(mBitmap);
        try
        
            FileOutputStream mFileOutStream = new FileOutputStream(mypath);

            v.draw(canvas);
            mBitmap.compress(Bitmap.CompressFormat.PNG, 90, mFileOutStream);
            mFileOutStream.flush();
            mFileOutStream.close();
            String url = Images.Media.insertImage(getContentResolver(), mBitmap, "title", null);
            Log.v("log_tag","url: " + url);
            //In case you want to delete the file
            //boolean deleted = mypath.delete();
            //Log.v("log_tag","deleted: " + mypath.toString() + deleted);
            //If you want to convert the image to string use base64 converter

        
        catch(Exception e)
        
            Log.v("log_tag", e.toString());
        
    

    public void clear()
    
        path.reset();
        invalidate();
    

    @Override
    protected void onDraw(Canvas canvas)
    
        canvas.drawPath(path, paint);
    

    @Override
    public boolean onTouchEvent(MotionEvent event)
    
        float eventX = event.getX();
        float eventY = event.getY();
        mGetSign.setEnabled(true);

        switch (event.getAction())
        
        case MotionEvent.ACTION_DOWN:
            path.moveTo(eventX, eventY);
            lastTouchX = eventX;
            lastTouchY = eventY;
            return true;

        case MotionEvent.ACTION_MOVE:

        case MotionEvent.ACTION_UP:

            resetDirtyRect(eventX, eventY);
            int historySize = event.getHistorySize();
            for (int i = 0; i < historySize; i++)
            
                float historicalX = event.getHistoricalX(i);
                float historicalY = event.getHistoricalY(i);
                expandDirtyRect(historicalX, historicalY);
                path.lineTo(historicalX, historicalY);
            
            path.lineTo(eventX, eventY);
            break;

        default:
            debug("Ignored touch event: " + event.toString());
            return false;
        

        invalidate((int) (dirtyRect.left - HALF_STROKE_WIDTH),
                (int) (dirtyRect.top - HALF_STROKE_WIDTH),
                (int) (dirtyRect.right + HALF_STROKE_WIDTH),
                (int) (dirtyRect.bottom + HALF_STROKE_WIDTH));

        lastTouchX = eventX;
        lastTouchY = eventY;

        return true;
    

    private void debug(String string)
    

    private void expandDirtyRect(float historicalX, float historicalY)
    
        if (historicalX < dirtyRect.left)
        
            dirtyRect.left = historicalX;
        
        else if (historicalX > dirtyRect.right)
        
            dirtyRect.right = historicalX;
        

        if (historicalY < dirtyRect.top)
        
            dirtyRect.top = historicalY;
        
        else if (historicalY > dirtyRect.bottom)
        
            dirtyRect.bottom = historicalY;
        
    

    private void resetDirtyRect(float eventX, float eventY)
    
        dirtyRect.left = Math.min(lastTouchX, eventX);
        dirtyRect.right = Math.max(lastTouchX, eventX);
        dirtyRect.top = Math.min(lastTouchY, eventY);
        dirtyRect.bottom = Math.max(lastTouchY, eventY);
    

【讨论】:

不是在寻找活动类/android 代码片段好友。还是谢谢

以上是关于在 Android 上的 Rhodes 中将签名图片保存到文件中的主要内容,如果未能解决你的问题,请参考以下文章

AR入门系列-03-在unity中将调试好的Vuforia项目导出为APK

Android 上的自签名 SSL 接受

如何信任 Android 上的自签名证书?

如何在 Android 上的 TextView 中将两行文本居中?

18.5 设置Android程序的图标和签名文件

android 怎么查看签名文件