วิธีการแปลง Blob เป็น File ใน JavaScript


111

ฉันต้องการอัปโหลดภาพไปยังเซิร์ฟเวอร์ NodeJS ไปยังไดเรกทอรีบางรายการ ฉันใช้connect-busboyโมดูลโหนดสำหรับสิ่งนั้น

ฉันมีdataURLภาพที่ฉันแปลงเป็นหยดโดยใช้รหัสต่อไปนี้:

dataURLToBlob: function(dataURL) {
    var BASE64_MARKER = ';base64,';
    if (dataURL.indexOf(BASE64_MARKER) == -1) {
        var parts = dataURL.split(',');
        var contentType = parts[0].split(':')[1];
        var raw = decodeURIComponent(parts[1]);
        return new Blob([raw], {type: contentType});
    }
    var parts = dataURL.split(BASE64_MARKER);
    var contentType = parts[0].split(':')[1];
    var raw = window.atob(parts[1]);
    var rawLength = raw.length;
    var uInt8Array = new Uint8Array(rawLength);
    for (var i = 0; i < rawLength; ++i) {
        uInt8Array[i] = raw.charCodeAt(i);
    }
    return new Blob([uInt8Array], {type: contentType});
}

ฉันต้องการวิธีแปลงหยดเป็นไฟล์เพื่ออัปโหลดภาพ

ใครช่วยฉันด้วย?


4
ไฟล์เป็น Blobs เพียงแค่ยึดติดกับคุณสมบัติเมตาและคุณก็พร้อมที่จะไป
dandavis

1
blobค่าเริ่มต้นสำหรับหยดเมื่ออัปโหลดมันเป็น ดังนั้นครั้งแรกที่ผมสกัดชื่อของไฟล์ที่ผมได้รับการปลูกพืชและจากนั้นให้เหมือนกันเพื่อให้ไฟล์ที่ถูกตัดขณะที่อัปโหลดไปยังเซิร์ฟเวอร์โดยการทำfilename form.append("blob",blob, filename);
ข้าม

@skip คำตอบของฉันด้านล่างช่วยได้หรือไม่? ดังนั้นโปรดทำเครื่องหมายว่าเป็นคำตอบที่ถูกต้อง
CBarr

คำตอบ:


150

ฟังก์ชั่นนี้แปลงBlobเป็นFileและใช้งานได้ดีสำหรับฉัน

วานิลลา JavaScript

function blobToFile(theBlob, fileName){
    //A Blob() is almost a File() - it's just missing the two properties below which we will add
    theBlob.lastModifiedDate = new Date();
    theBlob.name = fileName;
    return theBlob;
}

TypeScript (ด้วยการพิมพ์ที่เหมาะสม)

public blobToFile = (theBlob: Blob, fileName:string): File => {
    var b: any = theBlob;
    //A Blob() is almost a File() - it's just missing the two properties below which we will add
    b.lastModifiedDate = new Date();
    b.name = fileName;

    //Cast to a File() type
    return <File>theBlob;
}

การใช้งาน

var myBlob = new Blob();

//do stuff here to give the blob some data...

var myFile = blobToFile(myBlob, "my-image.png");

3
ฉันคิดว่าการแคสต์ควรเกิดขึ้นก่อนดังนั้นคุณไม่จำเป็นต้องใช้anyใน TypeScript ดูนี้ตัวอย่างเช่น
styfle

3
สิ่งนี้ใช้ไม่ได้สำหรับวัตถุประสงค์ทั้งหมด การเพิ่ม "ไฟล์" นี้ไปยังการโทรของ ajax จะทำให้ตั้งชื่อไฟล์ไม่ถูกต้อง
Jacob Poul Richardt

13
นี่ไม่ใช่วิธีแก้ปัญหาที่ดีวัตถุที่ผลิตยังคงเป็นหยด
czupe

4
เพียงแค่เพิ่มb.__proto__ = File.prototypeและวิธีการแก้ปัญหาของคุณจะกลายเป็นความฝันที่เทคนิคแม้b instanceOf Fileจะtrue
manuelnaranjo

3
นี่ไม่ควรเป็นคำตอบที่ยอมรับstackoverflow.com/a/53205768/2557517หรือstackoverflow.com/a/31663645/2557517ควรจะเป็น
Kira

204

คุณสามารถใช้ตัวสร้างไฟล์:

var file = new File([myBlob], "name");

ตามข้อกำหนด w3 สิ่งนี้จะต่อท้ายไบต์ที่ blob มีต่อไบต์สำหรับอ็อบเจ็กต์ File ใหม่และสร้างไฟล์ด้วยชื่อที่ระบุ http://www.w3.org/TR/FileAPI/#dfn-file


2
นี่คือสิ่งที่ฉันกำลังมองหาความแตกต่างอย่างหนึ่งคืออาร์กิวเมนต์แรกคืออาร์เรย์จริงๆดังนั้นฉันจึงใช้มันเป็น: var file = new File ([myBlob], "name");
Michaeldcooney

1
ใช่. ฉันกำลังมองหาวิธีแก้ปัญหานี้เหมือนกัน เซิร์ฟเวอร์ Stamplay ไม่สามารถรับชื่อไฟล์จากวัตถุ blob ดังนั้นฉันต้องแปลงกลับเป็นไฟล์ แต่ Safari ไม่รองรับไฟล์ API ... กลับไปที่ 0 ทันที :(
Hugh Hou

4
ตัวสร้างไฟล์ไม่ทำงานเช่นใน PhantomJS:TypeError: FileConstructor is not a constructor (evaluating 'new File([''], 'file.pdf', {'size': 1000, 'type': 'application/pdf'})')
bkimminich

7
ปัจจุบัน Edge มีข้อบกพร่อง (2017-10-04) ที่ขัดขวางการใช้งานตัวสร้างนี้ developer.microsoft.com/en-us/microsoft-edge/platform/issues/…
Rik Martins

8
อย่าลืมเพิ่มประเภทไฟล์มิฉะนั้นจะทำงานไม่ถูกต้อง ไฟล์ใหม่ ([myBlob], "name", {type: "image / jpeg",});
BernardA

34

คำตอบของJoshua P Nixonถูกต้อง แต่ฉันต้องกำหนดวันที่แก้ไขล่าสุดด้วย นี่คือรหัส

var file = new File([blob], "file_name", {lastModified: 1534584790000});

1534584790000คือการประทับเวลา Unix สำหรับ " GMT: วันเสาร์ที่ 18 สิงหาคม 2018 เวลา 09:33:10 น. "


3
คะแนนที่ไม่ทำลายinstanceofเหมือนคำตอบที่ยอมรับ
Matt Jensen

15

สำหรับฉันในการทำงานฉันต้องระบุประเภทอย่างชัดเจนแม้ว่าจะมีอยู่ในหยดโดยทำเช่นนั้น:

const file = new File([blob], 'untitled', { type: blob.type })

14

ตัวแปรที่ทันสมัยของฉัน:

function blob2file(blobData) {
  const fd = new FormData();
  fd.set('a', blobData);
  return fd.get('a');
}

2
ในการเพิ่มชื่อไฟล์ในไฟล์ผลลัพธ์ให้ใช้:fd.set('a', blobData, 'filename')
joaoguerravieira

มันล้มเหลวในอุปกรณ์ IOS บางเครื่องเมื่อfd.set
ร้องเพลง

ขอบคุณ. ใช้งานได้ตอนนี้ฉันไม่ต้องแปลง blob เป็นไฟล์
xreyc_developer22

2

ใช้saveAsกับโปรเจ็กต์ github FileSaver.js

FileSaver.jsใช้saveAs()อินเทอร์เฟซ FileSaver ในเบราว์เซอร์ที่ไม่สนับสนุนโดยกำเนิด


2

ประเภท

public blobToFile = (theBlob: Blob, fileName:string): File => {       
    return new File([theBlob], fileName, { lastModified: new Date().getTime(), type: theBlob.type })
}

Javascript

function blobToFile(theBlob, fileName){       
    return new File([theBlob], fileName, { lastModified: new Date().getTime(), type: theBlob.type })
}

เอาต์พุต

ภาพหน้าจอ

File {name: "fileName", lastModified: 1597081051454, lastModifiedDate: Mon Aug 10 2020 19:37:31 GMT+0200 (Eastern European Standard Time), webkitRelativePath: "", size: 601887, …}
lastModified: 1597081051454
lastModifiedDate: Mon Aug 10 2020 19:37:31 GMT+0200 (Eastern European Standard Time) {}
name: "fileName"
size: 601887
type: "image/png"
webkitRelativePath: ""
__proto__: File
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.