4

如标题中所述,我正在尝试使用 rails 的 Active Storage 从嵌套在 rails 表单中的元素将图像上传到我的 S3 存储桶。到目前为止,我已经能够使用 <%= f.input :signature, type: file_field(:user, :signature), %> Active Storage 上传图像。Userhas_one_attached :signature。_ 当我使用 file_field 时图像正确上传,所以这不是问题的一部分。

到目前为止,我simple_form有:

  <div class="signature_pad text-center form-group">
    <div class="signature_pad_heading">
      Enter your Signature:
    </div>
    <div class="signature_pad_body">
      <canvas id="signature_pad_input" height="145px" width="370px" style="height: 145px; width: 370px;" class="border" />
    </div>
    <div class="signature_pad_footer">
      <button type="button" class="btn btn-default" onclick="signaturePad.clear()">Clear</button>
    </div>
  </div>

  <%= f.input :signature, type: file_field(:user, :signature), value: "", as: :hidden %>

  <%= f.submit "Save", class:'btn-primary btn-lg btn-md-wide',  id: "signature_pad_save"  %>

我的javascript是:

<script src="https://cdn.jsdelivr.net/npm/signature_pad@2.3.2/dist/signature_pad.min.js"></script>
<script>
const canvas = document.querySelector("canvas");
const signaturePad = new SignaturePad(canvas);

$('#signature_pad_save').click(function(event) {
    if (signaturePad.isEmpty()){
        alert('Please enter your signature.');
        event.preventDefault();
    } else {
        $('#user_signature').val(
            JSON.parse(
                signaturePad.toDataURL()  
        );
    }});
</script>

使用 .toDataURL 我可以获得图像的 base64,我读过的所有内容似乎都指出这就是我需要通过 Active-Storage 发送到 S3 的全部内容。

最后:

我使用 .file_field 时发送的内容

"signature"=>"<ActionDispatch::Http::UploadedFile:0x007f7a02ad4ef8 @tempfile=#<Tempfile:/tmp/RackMultipart20180903-3527-kked3g.png>, @original_filename="signature1.png", @content_type="image/png", @headers="Content-Disposition: form-data; name=\"user[signature]\"; filename=\"signature1.png\"\r\nContent-Type: image/png\r\n">},"

当我尝试在表单提交之前插入值时发送的内容

"signature"=>"#<ActiveStorage::Attached::One:0x007f7a01c8b4f0>"}
4

2 回答 2

2

是的,这是可能的。

直接上传由DirectUpload班级处理(或ActiveStorage.DirectUpload取决于您访问事物的方式)。通常DirectUpload想要 aFile从中读取图像但足够方便, aFile是 aBlob所以你几乎可以使用 aBlob来代替。Active Storage 只需要它的“文件”中的一些东西,并且几乎所有这些东西都存在于 中Blob,唯一缺少的是一个name属性,您可以自己添加它。

所以现在我们需要Blob从画布中取出一个。这实际上很容易,因为画布有一个toBlob方法

HTMLCanvasElement.toBlob()方法创建一个Blob代表画布中包含的图像的对象;这个文件可以缓存在磁盘上,也可以由用户代理决定存储在内存中。

首先照常设置直接上传。然后为大致如下所示的整体表单添加您自己的提交处理程序:

const form   = your_form_element;
const canvas = your_canvas_element;
const input  = your_file_input;

canvas.toBlob(blob => {
  // Fake out DirectUpload by manually adding a name.
  blob.name = input.files[0].name;

  const uploader = new DirectUpload(blob, input.dataset.directUploadUrl);
  uploader.create((error, blob) => {
    if(error) {
      // Handle the error.
    }
    else {
      // Add the <input type="hidden"> with the signature.
      const hiddenField = document.createElement('input')
      hiddenField.setAttribute('type', 'hidden');
      hiddenField.setAttribute('value', blob.signed_id);
      form.appendChild(hiddenField);

      // And submit away...
      form.submit();
    }
  });
}, 'image/jpeg', 0.95);

假设您需要 JPEG,您可以根据需要使用其他内容类型,也可以将其与原始内容相匹配。我还遗漏了一些常见的提交处理程序样板。如果您设置了通常的全局直接上传处理程序,那么它们将在此处用于进度条等;如果你没有这些,那么你必须自己处理,上面链接的指南对所有这些都有指示。

于 2019-02-15T02:01:23.120 回答
0

我以前不得不做类似的事情。imagemagick我通过rmagickgem将 base64 字符串转换回图像文件。

有关代码,请参阅此问题的第三个答案:如何使用 ruby​​ 将 base64 字符串保存为图像

然后,您必须将其添加到控制器的create和方法中,并使用此处描述的方法。update@user.signature.attach

于 2018-09-05T05:13:02.367 回答