Angular - POST загруженный файл

Я использую Angular, TypeScript, чтобы отправить файл вместе с данными JSON на сервер.

Ниже мой код:

import {Component, View, NgFor, FORM_DIRECTIVES, FormBuilder, ControlGroup} from 'angular2/angular2';
import {Http, Response, Headers} from 'http/http';


@Component({ selector: 'file-upload' })
@View({
directives: [FORM_DIRECTIVES],
template: `
<h3>File Upload</h3>

<div>
     Select file:
    <input type="file" (change)="changeListener($event)">
</div>

`
})
export class FileUploadCmp {

public file: File;
public url: string;
headers: Headers;


constructor(public http: Http) {
    console.log('file upload Initialized');
    //set the header as multipart        
    this.headers = new Headers();
    this.headers.set('Content-Type', 'multipart/form-data');
    this.url = 'http://localhost:8080/test';
}

//onChange file listener
changeListener($event): void {
    this.postFile($event.target);
}

//send post file to server 
postFile(inputValue: any): void {

    var formData = new FormData();
    formData.append("name", "Name");
    formData.append("file",  inputValue.files[0]);

    this.http.post(this.url +,
      formData ,
        {
            headers: this.headers

        });
}

}

Как я могу преобразовать formData в String и отправить его на сервер? Я помню, что в AngularJS (v1) вы использовали бы transformRequest.


person Kamal    schedule 06.09.2015    source источник
comment
нельзя преобразовать данные перед строкой this.http.post? http является наблюдаемым и принимает параметры, попробуйте выполнить поиск в источнике, который вы можете найти, например, github.com/angular/angular/blob/   -  person Angular University    schedule 07.09.2015
comment
Были ли у вас какие-то успехи в этом вопросе?   -  person Canastro    schedule 16.09.2015
comment
На GitHub есть ожидающий запрос функции github.com/angular/angular/issues/2803   -  person martin    schedule 25.09.2015
comment
Смотрите здесь, я решил это после долгой борьбы с 2017.8.25, stackoverflow.com/a/45879409/2803344   -  person Belter    schedule 25.08.2017


Ответы (5)


Посмотрите на мой код, но знайте. Я использую async / await, потому что последняя бета-версия Chrome может читать любой код es6, который получает TypeScript с компиляцией. Итак, вы должны заменить asyns / await на .then().

Обработчик изменения ввода:

/**
 * @param fileInput
 */
public psdTemplateSelectionHandler (fileInput: any){
    let FileList: FileList = fileInput.target.files;

    for (let i = 0, length = FileList.length; i < length; i++) {
        this.psdTemplates.push(FileList.item(i));
    }

    this.progressBarVisibility = true;
}

Отправить обработчик:

public async psdTemplateUploadHandler (): Promise<any> {
    let result: any;

    if (!this.psdTemplates.length) {
        return;
    }

    this.isSubmitted = true;

    this.fileUploadService.getObserver()
        .subscribe(progress => {
            this.uploadProgress = progress;
        });

    try {
        result = await this.fileUploadService.upload(this.uploadRoute, this.psdTemplates);
    } catch (error) {
        document.write(error)
    }

    if (!result['images']) {
        return;
    }

    this.saveUploadedTemplatesData(result['images']);
    this.redirectService.redirect(this.redirectRoute);
}

FileUploadService. Эта служба также хранит свойство $ прогресс в процессе загрузки, а в других местах вы можете подписаться на него и получать новое значение каждые 500 мс.

import { Component } from 'angular2/core';
import { Injectable } from 'angular2/core';
import { Observable } from 'rxjs/Observable';
import 'rxjs/add/operator/share';

@Injectable()
export class FileUploadService {
/**
 * @param Observable<number>
 */
private progress$: Observable<number>;

/**
 * @type {number}
 */
private progress: number = 0;

private progressObserver: any;

constructor () {
    this.progress$ = new Observable(observer => {
        this.progressObserver = observer
    });
}

/**
 * @returns {Observable<number>}
 */
public getObserver (): Observable<number> {
    return this.progress$;
}

/**
 * Upload files through XMLHttpRequest
 *
 * @param url
 * @param files
 * @returns {Promise<T>}
 */
public upload (url: string, files: File[]): Promise<any> {
    return new Promise((resolve, reject) => {
        let formData: FormData = new FormData(),
            xhr: XMLHttpRequest = new XMLHttpRequest();

        for (let i = 0; i < files.length; i++) {
            formData.append("uploads[]", files[i], files[i].name);
        }

        xhr.onreadystatechange = () => {
            if (xhr.readyState === 4) {
                if (xhr.status === 200) {
                    resolve(JSON.parse(xhr.response));
                } else {
                    reject(xhr.response);
                }
            }
        };

        FileUploadService.setUploadUpdateInterval(500);

        xhr.upload.onprogress = (event) => {
            this.progress = Math.round(event.loaded / event.total * 100);

            this.progressObserver.next(this.progress);
        };

        xhr.open('POST', url, true);
        xhr.send(formData);
    });
}

/**
 * Set interval for frequency with which Observable inside Promise will share data with subscribers.
 *
 * @param interval
 */
private static setUploadUpdateInterval (interval: number): void {
    setInterval(() => {}, interval);
}
}
person Качалов Тимофей    schedule 20.02.2016
comment
Мне также нужно было добавить: xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest');, чтобы он работал - person patad; 12.04.2016
comment
@patad: у тебя с этим проблемы? Когда я загружаю XML-файл, он автоматически добавляет '------ WebKitFormBoundaryklb8qUzyCQJSI440 Content-Disposition: form-data; name = file 'внутри моего файла. Я не хочу добавлять в файл дополнительные данные во время передачи, вы знаете, как с этим поступить? А для загруженного изображения, такого как jpg, после загрузки это уже не был файл jpg. Итак, мой вопрос: как сохранить исходный файл во время передачи? - person Ng2-Fun; 29.04.2016
comment
Зачем нам эта вещь setInterval()? - person jrub; 22.05.2016

Рассмотрение этой проблемы https://github.com/angular/angular/issues/10424, angular2 http пока не поддерживает загрузку файлов.

Для очень простой загрузки файлов я создал такую ​​служебную функцию в качестве обходного пути (используя ответ Тимофей):

  uploadFile(file:File):Promise<MyEntity> {
    return new Promise((resolve, reject) => {

        let xhr:XMLHttpRequest = new XMLHttpRequest();
        xhr.onreadystatechange = () => {
            if (xhr.readyState === 4) {
                if (xhr.status === 200) {
                    resolve(<MyEntity>JSON.parse(xhr.response));
                } else {
                    reject(xhr.response);
                }
            }
        };

        xhr.open('POST', this.getServiceUrl(), true);

        let formData = new FormData();
        formData.append("file", file, file.name);
        xhr.send(formData);
    });
}
person Andrii Karaivanskyi    schedule 27.02.2016
comment
Большой! Однако, чтобы он работал, мне нужно было добавить xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest');. - person patad; 12.04.2016
comment
Спасибо! Если требуется авторизация, добавьте это после того, как у меня сработала строка xhr.open: xhr.setRequestHeader('Authorization', 'Bearer ' + __your_auth_key__); - person Steve; 04.05.2016
comment
Поддерживается ли это сейчас? - person Manish Jain; 29.09.2016
comment
@ManishJain проверьте статус проблемы https://github.com/angular/http/issues/75 Пока он открыт. - person Andrii Karaivanskyi; 03.10.2016
comment
@AndreyKarayvansky, есть ли обновленная ссылка? Я получаю страницу 404 и пытаюсь найти соответствующую проблему. - person cbros2008; 25.10.2016
comment
Ссылка обновлена ​​- но по-прежнему нет прямой поддержки прогресса, потому что fetch не хватает этой функции, а Angular не хочет привязывать http API к XHR ... - person Jan Nielsen; 17.03.2017
comment
Я использую новый Angular 4, и, похоже, они этого еще не исправили. Либо это, либо я упустил суть. Дело в том, что файл так и не прошел. Благодаря вашей функции у меня все заработало. Спасибо, мужик. - person Paulo Pedroso; 28.07.2017

ваш служебный файл http:

import { Injectable } from "@angular/core";
import { ActivatedRoute, Router } from '@angular/router';
import { Http, Headers, Response, Request, RequestMethod, URLSearchParams, RequestOptions } from "@angular/http";
import {Observable} from 'rxjs/Rx';
import { Constants } from './constants';
declare var $: any;

@Injectable()
export class HttpClient {
  requestUrl: string;
  responseData: any;
  handleError: any;

  constructor(private router: Router, 
  private http: Http, 
  private constants: Constants, 
  ) {
    this.http = http;
  }

  postWithFile (url: string, postData: any, files: File[]) {

    let headers = new Headers();
    let formData:FormData = new FormData();
    formData.append('files', files[0], files[0].name);
    // For multiple files
    // for (let i = 0; i < files.length; i++) {
    //     formData.append(`files[]`, files[i], files[i].name);
    // }

    if(postData !=="" && postData !== undefined && postData !==null){
      for (var property in postData) {
          if (postData.hasOwnProperty(property)) {
              formData.append(property, postData[property]);
          }
      }
    }
    var returnReponse = new Promise((resolve, reject) => {
      this.http.post(this.constants.root_dir + url, formData, {
        headers: headers
      }).subscribe(
          res => {
            this.responseData = res.json();
            resolve(this.responseData);
          },
          error => {
            this.router.navigate(['/login']);
            reject(error);
          }
      );
    });
    return returnReponse;
  }
}

вызовите свою функцию (файл компонента):

onChange(event) {
    let file = event.srcElement.files;
    let postData = {field1:"field1", field2:"field2"}; // Put your form data variable. This is only example.
    this._service.postWithFile(this.baseUrl + "add-update",postData,file).then(result => {
        console.log(result);
    });
}

ваш html-код:

<input type="file" class="form-control" name="documents" (change)="onChange($event)" [(ngModel)]="stock.documents" #documents="ngModel">
person Bharat Chauhan    schedule 14.11.2016
comment
Один из единственных ответов, который на самом деле использует angular - person csga5000; 05.05.2017
comment
Как я могу это исправить error TS2339: Property 'files' does not exist on type 'Element'. спасибо - person Muhammad Shahzad; 13.05.2017
comment
onChange() {...} должно быть onChange(event) {...} исправит ошибку: файлы свойств не существуют;) - person Muhammad Shahzad; 14.05.2017
comment
у меня не работает. он не может найти файл на внутреннем узле, который использует multer. - person srv_sud; 16.01.2019

В моем проекте я использую XMLHttpRequest для отправки multipart / form-data. Думаю, вам подойдет.

и загрузчик код

let xhr = new XMLHttpRequest();
xhr.open('POST', 'http://www.example.com/rest/api', true);
xhr.withCredentials = true;
xhr.send(formData);

Вот пример: https://github.com/wangzilong/angular2-multipartForm

person Zilong Wang    schedule 04.02.2016
comment
Просто ссылка на вашу собственную библиотеку (или утилиту) - не лучший ответ. Ссылки на него, объяснение, почему он решает проблему, предоставление кода, использующего его для этого, и отказ от ответственности - лучший ответ. См.: Как сделать ссылку на внешний ресурс удобной для сообщества способом? - person Tunaki; 04.02.2016

Во-первых, вам нужно создать свой собственный встроенный TS-класс, поскольку класс FormData в настоящее время не поддерживается:

var data : {
  name: string;
  file: File;
} = {
  name: "Name",
  file: inputValue.files[0]
  };

Затем вы отправляете его на сервер с помощью JSON.stringify (data)

let opts: RequestOptions = new RequestOptions();
opts.method = RequestMethods.Post;
opts.headers = headers;
this.http.post(url,JSON.stringify(data),opts);
person Philip    schedule 11.12.2015
comment
developer.mozilla.org/en-US/docs/Web/ API / Поддержка на самом деле не так уж и плоха. Я предполагаю, что JSON.stringify будет иметь некоторые проблемы, если загруженные файлы окажутся двоичными или чем-то в этом роде ... - person csga5000; 05.05.2017