如何将图像发布为xhr / ajax请求? (在iOS 10.0.2+中失败)

时间:2016-11-26 03:51:29

标签: javascript ios xmlhttprequest

使用XHR请求上传图像(来自dataUrl格式)似乎在iOS(10.0.2+)上失败,其他浏览器正在运行:最新的Chrome(54.0.2840.71),Firefox(50.0)。

请求似乎在JS中正确构造,但请求中没有附加实际数据。

var dataUrl = "";

// create base64 encoded version of image
let parts = dataUrl.split(',');
let mime = parts[0].match(/:(.*?);/)[1];
let bstr = atob(parts[1]);
let n = bstr.length;
let u8arr = new Uint8Array(n);
  while (n--) {
  u8arr[n] = bstr.charCodeAt(n);
}
let blob = new window.Blob([u8arr], {type: mime});
file = new window.File([blob], file.name);

// create formdata for uploading
let data = new FormData();
data.append('file', file, file.name);


// send the post request
var xhr = new XMLHttpRequest;
xhr.onload = function (oEvent) {
  // Uploaded.
  if ( oEvent.target.status == '200') {
    console.log('upload with XHR success');
  } else {
    console.log('upload with XHR error');
  }
};
xhr.open("POST", '/upload-endpoint', false);
xhr.send(data);

2 个答案:

答案 0 :(得分:0)

如何将图像上传到RESTful api的主要思路是使用multipart/form-data

如果您使用Swift 3.0,可以使用URLSession

尝试以下代码
import Cocoa

fileprivate extension ImageUploader {
    static let urlString = "/upload-endpoint"
    static let boundary = "---------------------------14737809831460499985746342449"
    static let contentType = "multipart/form-data; boundary=\(boundary)"
}

typealias CompletionBlock = (_ data: Data?, _ response: URLResponse?, _ error: Error?) -> Swift.Void

class ImageUploader {

    func recognize(with paths: [String], completionBlock: @escaping CompletionBlock) {

        guard paths.count > 0 else { return }

        let path = paths.first!

        guard let image = NSImage(contentsOfFile: paths.first!) else { return }

        let imgRep: NSBitmapImageRep = image.representations.first as! NSBitmapImageRep
        let data = imgRep.representation(using: .JPEG, properties: [:])

        var request = URLRequest(url: URL(string: ImageUploader.urlString)!)
        request.httpMethod = "POST"
        request.timeoutInterval = 10

        request.addValue(ImageUploader.contentType, forHTTPHeaderField: "Content-Type")
        var postbody = Data()
        postbody.append("\r\n--\(ImageUploader.boundary)\r\n".data(using: .utf8)!)
        postbody.append("Content-Disposition: form-data; name=\"file\"; filename=\"file\"\r\n".data(using: .utf8)!)
        postbody.append("Content-Type: application/octet-stream\r\n\r\n".data(using: .utf8)!)
        postbody.append(data!)
        postbody.append("\r\n--\(ImageUploader.boundary)\r\n".data(using: .utf8)!)

        request.httpBody = postbody

        let task = URLSession.shared.dataTask(with: request) { data, response, error in
            completionBlock(data, response, error)
        }
        task.resume()

    }

}

答案 1 :(得分:0)

也许尝试使用fetch而不是xhr?

let dataUrl = "";
let {name} = file

// First convert base64 to blob
fetch(dataUrl)
.then(res => res.blob())
.then(blob => {

    // Now upload the blob
    let fd = new FormData()
    fd.append('file', blob, name)

    fetch('/upload-endpoint', {method: 'POST', body: fd})
    .then(res => {
        console.log('upload with fetch success:', res.ok)
    })

})