Как в bootstrap - vue преобразовать FileReader в blob и загрузить на сервер? - PullRequest
0 голосов
/ 17 февраля 2020

В приложении @ vue / cli 4.1.1 я использую bootstrap - vue и компонент b-form-file для загрузки изображений

https://bootstrap-vue.js.org/docs/components/form-file/#multiple -файлы

с определением:

<b-form-file
    id="upload_ad_image"
    v-model="new_upload_ad_image"
    :state="Boolean(new_upload_ad_image)"
    placeholder="Choose a file or drop it here..."
    drop-placeholder="Drop file here..."
    accept="image/jpeg, image/png, image/gif"
></b-form-file>

<div ref="uploaded_img_preview" id="uploaded_img_preview" class="m-2" >Uploaded image preview :</div>

Я нашел фрагмент https://codepen.io/Tenderfeel/pen/rgqWXR и, используя его, показываю выбранный файл в своей форме для предварительного просмотра. Далее мне нужно загрузить его на сервер. У меня есть опыт загрузки изображений в виде блога с использованием кода вроде:

fetch(this.taskRow.imageFile.blob).then(function (response) {
    if (response.ok) {
        return response.blob().then(function (imageBlob) {
            let imageUploadData = new FormData()
            imageUploadData.append('id', self.taskRow.id)
            imageUploadData.append('image', imageBlob)
            imageUploadData.append('image_filename', self.taskRow.imageFile.name)

Но мне нужно преобразовать загружаемое изображение в блоб. Я использую метод, когда изображение выбрано: Но получена ошибка: Ошибка в обратном вызове для наблюдателя "new_upload_ad_image": "InvalidStateError: Не удалось выполнить readAsDataURL" для FileReader ": Объект уже занят чтением BLOB-объектов

watch: {
    new_upload_ad_image(val) {
        if (!val) return;
        if (this.previewImg) {
            this.previewImg.remove();
        }
        const img = document.createElement("img");
        img.classList.add("obj");
        img.file = this.new_upload_ad_image;
        console.log('img.file::')
        console.log(img.file)

        this.previewImg = img;
        console.log('this.$refs.uploaded_img_preview::')
        console.log(this.$refs.uploaded_img_preview)

        console.log('img::')
        console.log(img)

        this.$refs.uploaded_img_preview.appendChild(img);

        const fileReader = new FileReader();
        fileReader.onload = (e) => {
            this.previewImg.src = e.target.result;
        };
        fileReader.readAsDataURL(this.new_upload_ad_image);
        console.log('fileReader::')
        console.log(fileReader)

        let blobObj= fileReader.readAsDataURL(img.file) // RAISE ERROR :

        console.log('blobObj::')
        console.log(blobObj)

    }
},

Что я вижу в консоли: https://imgur.com/a/2EZxq9C

Как получить блоб и загрузить его на сервер?

МОДИФИЦИРОВАННЫЙ БЛОК:

определив входной файл с идентификатором id = "upload_ad_image":

<b-form-file
    id="upload_ad_image"
    v-model="new_upload_ad_image"
    :state="Boolean(new_upload_ad_image)"
    placeholder="Choose a file or drop it here..."
    drop-placeholder="Drop file here..."
    accept="image/jpeg, image/png, image/gif"
></b-form-file>

Я запускаю fetch и вижу, что blob-изображение недействительно и файл создан, но он недействителен. У меня есть:

var self = this
const upload_ad_image = document.getElementById('upload_ad_image')
console.log('upload_ad_image::')
console.log(upload_ad_image)
console.log('upload_ad_image.files::')
console.log(upload_ad_image.files[0])

if (typeof upload_ad_image.files[0] == 'undefined') {
    self.showPopupMessage('Ad image upload', 'Invalid image !', 'warn')
    return
}
fetch(upload_ad_image.files[0].blob).then(function (response) {
    if (response.ok) {
        return response.blob().then(function (imageBlob) {
            console.log('imageBlob::')
            console.log(imageBlob) // Looks like this var has invalid content(printscreen below)!

            let imageUploadData = new FormData()
            imageUploadData.append('ad_id', self.editableAd.id)
            imageUploadData.append('main', self.new_upload_ad_image_main)
            imageUploadData.append('info', self.new_upload_ad_image_info)
            imageUploadData.append('image', imageBlob)
            imageUploadData.append('image_filename', upload_ad_image.files[0].name)

Я вижу в консоли: https://imgur.com/a/4ees55C

Что не так и как это можно исправить?

"bootstrap-vue": "^2.3.0",
"vue": "^2.6.11",

Спасибо!

1 Ответ

1 голос
/ 29 февраля 2020

Файл объекты наследуются от интерфейса Blob .

Все, что вы можете сделать с Blob, вы также можете сделать с File, поэтому в своем коде вы можете напрямую добавить this.new_upload_ad_image к своим FormData.

const inp = document.getElementById('inp');
inp.onchange = (evt) => {
  const formData = new FormData();
  formData.append( 'my-file', inp.files[0], 'file.ext' );
  //  check it's correctly in the FormData
  console.log( [...formData.entries()] );
};
<input type="file" id="inp">
...