Як завантажити файл з Angular2 або новішою версією


182

У мене є додаток WebApi / MVC, для якого я розробляю клієнт angular2 (замінити MVC). У мене виникають проблеми з розумінням того, як Angular зберігає файл.

Запит нормальний (добре працює з MVC, і ми можемо реєструвати отримані дані), але я не можу зрозуміти, як зберегти завантажені дані (я в основному дотримуюся тієї ж логіки, що і в цій публікації ). Я впевнений, що це дурно просто, але поки що я просто не розумію цього.

Код функції компонента знаходиться нижче. Я пробував різні варіанти, блоб шлях повинен бути шляхом, наскільки я зрозумів, але немає жодної функції createObjectURLв URL. Я навіть не можу знайти визначення URLвікна, але, мабуть, воно існує. Якщо я використовую FileSaver.jsмодуль, я отримую ту ж помилку. Тож я здогадуюсь, що це щось, що нещодавно змінилося або ще не реалізовано. Як я можу викликати збереження файлу в A2?

downloadfile(type: string){

    let thefile = {};
    this.pservice.downloadfile(this.rundata.name, type)
        .subscribe(data => thefile = new Blob([data], { type: "application/octet-stream" }), //console.log(data),
                    error => console.log("Error downloading the file."),
                    () => console.log('Completed file download.'));

    let url = window.URL.createObjectURL(thefile);
    window.open(url);
}

Для повноти сервіс, який отримує дані, знаходиться нижче, але єдине, що потрібно зробити - це надіслати запит і передати дані без відображення, якщо це вдасться:

downloadfile(runname: string, type: string){
   return this.authHttp.get( this.files_api + this.title +"/"+ runname + "/?file="+ type)
            .catch(this.logAndPassOn);
}

За допомогою цього методу ви не можете завантажувати великі файли. Ви досягнете обмеження пам'яті на вкладку. Це може бути від 1-2 Гб.
Матвій Б.

@MatthewB. хотілося б, щоб ви сказали, що краще
steve

Для великих завантажень файлів потрібно вказати нову вкладку, наприклад, якщо імітувати клавішу <A>, цільова потреба повинна дорівнювати "_blank" або подати форму. Я не думаю, що існує чіткий спосіб обійти велике обмеження розміру файлу за допомогою запитів у стилі Ajax.
Метью Б.

Відповіді:


181

Проблема полягає в тому, що спостережуване працює в іншому контексті, тому при спробі створити var URL-адресу у вас порожній об’єкт, а не крапка, яку ви хочете.

Один із численних способів вирішити це:

this._reportService.getReport().subscribe(data => this.downloadFile(data)),//console.log(data),
                 error => console.log('Error downloading the file.'),
                 () => console.info('OK');

Коли запит буде готовий, він викличе функцію "downloadFile", яка визначається наступним чином:

downloadFile(data: Response) {
  const blob = new Blob([data], { type: 'text/csv' });
  const url= window.URL.createObjectURL(blob);
  window.open(url);
}

blob створений ідеально, і тому URL-адреса var, якщо не відкриється нове вікно, перевірте, що ви вже імпортували 'rxjs / Rx';

  import 'rxjs/Rx' ;

Я сподіваюся, що це може вам допомогти.


9
Що таке this._reportService.getReport()і що повертає?
Буржуа

3
@Burjua getReport()повертається athis.http.get(PriceConf.download.url)
ji-ruh

6
Проблема в мені полягає в тому, що вікно відкривається і закривається негайно, не завантажуючи файл
Бреден Браун

7
Як тут можна встановити ім’я файлу? за замовчуванням воно вибирає числове значення як ім'я
Саурах

8
Я використовував вищевказаний код для завантаження файлу з відповіді API, але я отримую деяку помилку при створенні частини Blob "Тип відповіді не призначається типу Blobpart". Будь ласка, допоможіть, якщо хтось знає це питання
knbibin

92

Спробуйте це !

1 - Встановити залежності для спливаючих вікон збереження / відкриття файлів

npm install file-saver --save
npm install @types/file-saver --save

2- Створіть службу з цією функцією для отримання даних

downloadFile(id): Observable<Blob> {
    let options = new RequestOptions({responseType: ResponseContentType.Blob });
    return this.http.get(this._baseUrl + '/' + id, options)
        .map(res => res.blob())
        .catch(this.handleError)
}

3- У компонентному синтаксичному синтаксичному синтаксичному аналізі виберіть файл

import {saveAs as importedSaveAs} from "file-saver";

  this.myService.downloadFile(this.id).subscribe(blob => {
            importedSaveAs(blob, this.fileName);
        }
    )

Це працює для мене!


1
Я використовував крок 2 у поєднанні з відповіддю від @Alejandro, і він працював без необхідності встановлення файлозбереження ...
Ewert

5
Дякую! Це прекрасно працює! Цікаво, чи зможемо ми отримати ім’я файлу, визначене в заголовку відповіді. Це можливо?
jfajunior

2
помилка Av5 Аргумент типу 'RequestOptions' не призначається параметру типу '{заголовки ?: HttpHeaders | {[заголовок: рядок]: рядок | рядок []; };
giveJob

Однак цей варіант не підходить для завантаження великих файлів.
Ревен

61

Якщо вам не потрібно додавати заголовки в запит, для завантаження файлу в Angular2 ви можете зробити просте :

window.location.href='http://example.com/myuri/report?param=x';

у вашому компоненті.


4
Може хтось скажіть, будь ласка, чому ця відповідь спростована? Тема - завантажити файл за допомогою angular2. Якщо цей метод працює для простого завантаження, його також слід позначити як дійсну відповідь.
Саураб Шетті

5
@SaurabhShetty, це не допоможе у випадку, якщо ви хочете надіслати власні заголовки, що робити, наприклад, якщо ви хочете надіслати маркер автентичності? Якщо ви подивитесь на питання щодо ОП, ви можете бачити, що він використовує authHttp!
А.Акрам

6
Я розумію суті, але ця відповідь вирішила моє питання.
JoeriShoeby

1
Якщо ви дозволите серверу повернути URL-адресу в якомусь контексті, він може підготувати URL-адресу. Наприклад: об'єкт: MyRecord.Cover. Обкладинка може бути URL-адресою для зображення на сервері. Під час виклику get (Myrecord) ви дозволяєте серверу повернути підготовлений URL (Cover) із встановленим маркером безпеки та іншими заголовками.
Єнс Аленіус

2
Це відповідь, яка працює. Просто, адже у нього немає <тут вставити корисну функцію>, що не робить це не відповіддю.
gburton

47

Це для людей, які шукають, як це зробити за допомогою HttpClient та збереження файлів:

  1. Встановити збереження файлів

npm встановити збереження файлів - зберегти

npm встановити @ types / file-saver --save

Клас обслуговування API:

export() {
    return this.http.get(this.download_endpoint, 
        {responseType: 'blob'});
}

Компонент:

import { saveAs } from 'file-saver';
exportPdf() {
    this.api_service.export().subscribe(data => saveAs(data, `pdf report.pdf`));
}

1
Як показати розмір файлів у браузері при запуску завантаження? Я надсилаю розмір файлів як розмір вмісту в заголовку http.
humbleCoder

35

Як щодо цього?

this.http.get(targetUrl,{responseType:ResponseContentType.Blob})
        .catch((err)=>{return [do yourself]})
        .subscribe((res:Response)=>{
          var a = document.createElement("a");
          a.href = URL.createObjectURL(res.blob());
          a.download = fileName;
          // start download
          a.click();
        })

Я міг би це зробити.
не потрібен додатковий пакет.


3
Так просто, але це той, який працює бездоганно. Це не захаращує DOM, не створює жодного елемента. Я поєднав це рішення з деякими з абобів, і це працює як шарм.
Чакс

20

Як зазначав Алехандро Корредор, це проста помилка сфери застосування. subscribeВиконуються асинхронно і openповинні бути розміщені в цьому контексті, так що дані по закінченню завантаження , коли ми викликаємо завантаження.

Однак це є два способи зробити це. Як рекомендують документи, сервіс опікується отриманням та картографуванням даних:

//On the service:
downloadfile(runname: string, type: string){
  var headers = new Headers();
  headers.append('responseType', 'arraybuffer');
  return this.authHttp.get( this.files_api + this.title +"/"+ runname + "/?file="+ type)
            .map(res => new Blob([res],{ type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' }))
            .catch(this.logAndPassOn);
}

Потім на компоненті ми просто підписуємося і обробляємо відображені дані. Є дві можливості. Перший , як було запропоновано в оригіналі публікації, але потребує невеликої корекції, як зазначив Алехандро:

//On the component
downloadfile(type: string){
  this.pservice.downloadfile(this.rundata.name, type)
      .subscribe(data => window.open(window.URL.createObjectURL(data)),
                  error => console.log("Error downloading the file."),
                  () => console.log('Completed file download.'));
  }

Другим способом було б використання FileReader. Логіка однакова, але ми можемо явно чекати, коли FileReader завантажить дані, уникаючи введення та вирішення проблеми асинхронізації.

//On the component using FileReader
downloadfile(type: string){
    var reader = new FileReader();
    this.pservice.downloadfile(this.rundata.name, type)
        .subscribe(res => reader.readAsDataURL(res), 
                    error => console.log("Error downloading the file."),
                    () => console.log('Completed file download.'));

    reader.onloadend = function (e) {
        window.open(reader.result, 'Excel', 'width=20,height=10,toolbar=0,menubar=0,scrollbars=no');
  }
}

Примітка. Я намагаюся завантажити файл Excel, і навіть незважаючи на те, що завантаження запущено (це відповідає на питання), файл пошкоджений. Дивіться відповідь на цю публікацію, щоб уникнути пошкодження файлу.


7
Я думаю, що причина, пошкоджена файлом, полягає в тому, що ви завантажуєте resв блоб, і ви насправді хочете res._body. Однак _bodyце приватна змінна і недоступна. На сьогоднішній день .blob()і .arrayBuffer()на об'єкт відповіді http не було реалізовано в Angular 2., text()і json()є єдиними двома варіантами, але обидва будуть обробляти тіло. Ви знайшли рішення?
sschueller

1
привіт @rll, я дотримувався вищезазначених кроків, і я отримую підписку як завершено. Але я не бачив завантаження файлу. Я також не міг бачити жодної помилки. Будь ласка, допоможіть
AishApp

1
Дві параметри дозволяють мені завантажити файл, але він спочатку завантажує дані у фоновому режимі. Що робити, якщо у мене великий файл, який потрібно завантажити?
f123

1
Моє рішення - просто використовувати <a href=""></a>для завантаження файлу.
користувач2061057

1
Я знаю, що це стара відповідь, але вона високо відповідає результатам пошуку, і це прийнята відповідь: рядок `headers.append ('responseType', 'arraybuffer');` неправильний. Це варіант, а не заголовок. Виправте це. Aaaand ... Заголовки створюються, а не використовуються. Не корисно.
Стево

17

Завантажте * .zip рішення для кутового 2.4.x: ви повинні імпортувати ResponseContentType з '@ angular / http' та змінити responseType на ResponseContentType.ArrayBuffer (за замовчуванням це ResponseContentType.Json)

getZip(path: string, params: URLSearchParams = new URLSearchParams()): Observable<any> {
 let headers = this.setHeaders({
      'Content-Type': 'application/zip',
      'Accept': 'application/zip'
    });

 return this.http.get(`${environment.apiUrl}${path}`, { 
   headers: headers, 
   search: params, 
   responseType: ResponseContentType.ArrayBuffer //magic
 })
          .catch(this.formatErrors)
          .map((res:Response) => res['_body']);
}

16

Для новіших кутових версій:

npm install file-saver --save
npm install @types/file-saver --save


import {saveAs} from 'file-saver/FileSaver';

this.http.get('endpoint/', {responseType: "blob", headers: {'Accept': 'application/pdf'}})
  .subscribe(blob => {
    saveAs(blob, 'download.pdf');
  });

Дякую, працює з Angular 8. Не знаю, чому це було так важко знайти.
MDave

11

Завантаження файлу через ajax - це завжди болючий процес, і, на мій погляд, найкраще дозволити серверу та браузеру виконувати цю роботу щодо перегляду типу вмісту.

Я думаю, що це найкраще

<a href="api/sample/download"></a> 

зробити це. Для цього навіть не потрібно відкривання нових вікон та подібних матеріалів.

Контролер MVC, як у вашому зразку, може бути подібним до наведеного нижче:

[HttpGet("[action]")]
public async Task<FileContentResult> DownloadFile()
{
    // ...
    return File(dataStream.ToArray(), "text/plain", "myblob.txt");
}

1
Ви маєте рацію, але як потім ви можете керувати помилками сервера в одностранному додатку? У випадку помилки, звичайно, служба REST повертає JSON з помилкою, завдяки чому програма відкриває JSON в іншому вікні браузера, що не те, що користувач хоче бачити
Лука

2
Якщо у вас є маркер доступу, вам потрібно вказати, що це не працює
chris31389

Це просто і просто. Але якщо ви хочете зробити деяку автентифікацію, то є можливість мати щось на зразок одноразового маркера. Отже, замість того, щоб мати його так, ви можете мати URL як: example.com/myuri/report?tokenid=1234-1233 І підтвердити ідентифікатор токена в базі даних. Звичайно, це не простий сценарій і працює у всіх ситуаціях, але може бути рішенням у ситуації, коли ви маєте доступ до бази даних, перш ніж повертати звіт як потік ..
GingerBeer

Отримайте URL-адресу завантаження з сервера. Таким чином, сервер може підготувати URL з одноразовим маркером безпеки.
Єнс Аленіус

8

Я використовую Angular 4 з об'єктом 4.3 httpClient. Я змінив відповідь, знайдений у Технічному блозі Js, який створює об'єкт посилання, використовує його для завантаження, а потім знищує.

Клієнт:

doDownload(id: number, contentType: string) {
    return this.http
        .get(this.downloadUrl + id.toString(), { headers: new HttpHeaders().append('Content-Type', contentType), responseType: 'blob', observe: 'body' })
}

downloadFile(id: number, contentType: string, filename:string)  {

    return this.doDownload(id, contentType).subscribe(  
        res => { 
            var url = window.URL.createObjectURL(res);
            var a = document.createElement('a');
            document.body.appendChild(a);
            a.setAttribute('style', 'display: none');
            a.href = url;
            a.download = filename;
            a.click();
            window.URL.revokeObjectURL(url);
            a.remove(); // remove the element
        }, error => {
            console.log('download error:', JSON.stringify(error));
        }, () => {
            console.log('Completed file download.')
        }); 

} 

Значення цього.downloadUrl було встановлено раніше, щоб вказувати на api. Я використовую це для завантаження вкладених файлів, тому я знаю id, contentType та ім'я файлу: я використовую MVC api для повернення файлу:

 [ResponseCache(Location = ResponseCacheLocation.None, NoStore = true)]
    public FileContentResult GetAttachment(Int32 attachmentID)
    { 
        Attachment AT = filerep.GetAttachment(attachmentID);            
        if (AT != null)
        {
            return new FileContentResult(AT.FileBytes, AT.ContentType);  
        }
        else
        { 
            return null;
        } 
    } 

Клас вкладень виглядає приблизно так:

 public class Attachment
{  
    public Int32 AttachmentID { get; set; }
    public string FileName { get; set; }
    public byte[] FileBytes { get; set; }
    public string ContentType { get; set; } 
}

Репозиторій filerep повертає файл із бази даних.

Сподіваюся, це комусь допоможе :)


7

Для тих, хто використовує Redux Pattern

Я додав у збереження файлів як @Hector Cuevas, названий у своїй відповіді. Використовуючи Angular2 v. 2.3.1, мені не потрібно було додавати в @ type / file-saver.

Наступний приклад - завантаження журналу у форматі PDF.

Журнальні дії

public static DOWNLOAD_JOURNALS = '[Journals] Download as PDF';
public downloadJournals(referenceId: string): Action {
 return {
   type: JournalActions.DOWNLOAD_JOURNALS,
   payload: { referenceId: referenceId }
 };
}

public static DOWNLOAD_JOURNALS_SUCCESS = '[Journals] Download as PDF Success';
public downloadJournalsSuccess(blob: Blob): Action {
 return {
   type: JournalActions.DOWNLOAD_JOURNALS_SUCCESS,
   payload: { blob: blob }
 };
}

Ефекти журналу

@Effect() download$ = this.actions$
    .ofType(JournalActions.DOWNLOAD_JOURNALS)
    .switchMap(({payload}) =>
        this._journalApiService.downloadJournal(payload.referenceId)
        .map((blob) => this._actions.downloadJournalsSuccess(blob))
        .catch((err) => handleError(err, this._actions.downloadJournalsFail(err)))
    );

@Effect() downloadJournalSuccess$ = this.actions$
    .ofType(JournalActions.DOWNLOAD_JOURNALS_SUCCESS)
    .map(({payload}) => saveBlobAs(payload.blob, 'journal.pdf'))

Служба журналу

public downloadJournal(referenceId: string): Observable<any> {
    const url = `${this._config.momentumApi}/api/journals/${referenceId}/download`;
    return this._http.getBlob(url);
}

Послуга HTTP

public getBlob = (url: string): Observable<any> => {
    return this.request({
        method: RequestMethod.Get,
        url: url,
        responseType: ResponseContentType.Blob
    });
};

Журнал-редуктор Хоча це встановлює лише правильні стани, які використовуються в нашій програмі, я все-таки хотів додати його, щоб показати повний шаблон.

case JournalActions.DOWNLOAD_JOURNALS: {
  return Object.assign({}, state, <IJournalState>{ downloading: true, hasValidationErrors: false, errors: [] });
}

case JournalActions.DOWNLOAD_JOURNALS_SUCCESS: {
  return Object.assign({}, state, <IJournalState>{ downloading: false, hasValidationErrors: false, errors: [] });
}

Я сподіваюся, що це корисно.


7

Я поділяю рішення, яке мені допомогло (будь-яке поліпшення дуже вдячне)

На вашому сервісі "pservice":

getMyFileFromBackend(typeName: string): Observable<any>{
    let param = new URLSearchParams();
    param.set('type', typeName);
    // setting 'responseType: 2' tells angular that you are loading an arraybuffer
    return this.http.get(http://MYSITE/API/FILEIMPORT, {search: params, responseType: 2})
            .map(res => res.text())
            .catch((error:any) => Observable.throw(error || 'Server error'));
}

Компонентна частина:

downloadfile(type: string){
   this.pservice.getMyFileFromBackend(typename).subscribe(
                    res => this.extractData(res),
                    (error:any) => Observable.throw(error || 'Server error')
                );
}

extractData(res: string){
    // transforme response to blob
    let myBlob: Blob = new Blob([res], {type: 'application/vnd.oasis.opendocument.spreadsheet'}); // replace the type by whatever type is your response

    var fileURL = URL.createObjectURL(myBlob);
    // Cross your fingers at this point and pray whatever you're used to pray
    window.open(fileURL);
}

У складовій частині ви телефонуєте службі, не підписуючись на відповідь. Підписатись на повний перелік типів mime openOffice див .: http://www.openoffice.org/framework/documentation/mimetypes/mimetypes.html


7

Буде краще, якщо ви спробуєте викликати новий метод всередині себе subscribe

this._reportService.getReport()
    .subscribe((data: any) => {
        this.downloadFile(data);
    },
        (error: any) => сonsole.log(error),
        () => console.log('Complete')
    );

Внутрішня downloadFile(data)функція, яку нам потрібно зробитиblock, link, href and file name

downloadFile(data: any, type: number, name: string) {
    const blob = new Blob([data], {type: 'text/csv'});
    const dataURL = window.URL.createObjectURL(blob);

    // IE doesn't allow using a blob object directly as link href
    // instead it is necessary to use msSaveOrOpenBlob
    if (window.navigator && window.navigator.msSaveOrOpenBlob) {
      window.navigator.msSaveOrOpenBlob(blob);
      return;
    }

    const link = document.createElement('a');
    link.href = dataURL;
    link.download = 'export file.csv';
    link.click();

    setTimeout(() => {

      // For Firefox it is necessary to delay revoking the ObjectURL
      window.URL.revokeObjectURL(dataURL);
      }, 100);
    }
}

5

Для завантаження та показу файлів PDF дуже подібний фрагмент коду, як показано нижче:

  private downloadFile(data: Response): void {
    let blob = new Blob([data.blob()], { type: "application/pdf" });
    let url = window.URL.createObjectURL(blob);
    window.open(url);
  }

  public showFile(fileEndpointPath: string): void {
    let reqOpt: RequestOptions = this.getAcmOptions();  //  getAcmOptions is our helper method. Change this line according to request headers you need.
    reqOpt.responseType = ResponseContentType.Blob;
    this.http
      .get(fileEndpointPath, reqOpt)
      .subscribe(
        data => this.downloadFile(data),
        error => alert("Error downloading file!"),
        () => console.log("OK!")
      );
  }

5

Ось що я зробив у своєму випадку -

// service method
downloadFiles(vendorName, fileName) {
    return this.http.get(this.appconstants.filesDownloadUrl, { params: { vendorName: vendorName, fileName: fileName }, responseType: 'arraybuffer' }).map((res: ArrayBuffer) => { return res; })
        .catch((error: any) => _throw('Server error: ' + error));
}

// a controller function which actually downloads the file
saveData(data, fileName) {
    var a = document.createElement("a");
    document.body.appendChild(a);
    a.style = "display: none";
    let blob = new Blob([data], { type: "octet/stream" }),
        url = window.URL.createObjectURL(blob);
    a.href = url;
    a.download = fileName;
    a.click();
    window.URL.revokeObjectURL(url);
}

// a controller function to be called on requesting a download
downloadFiles() {
    this.service.downloadFiles(this.vendorName, this.fileName).subscribe(data => this.saveData(data, this.fileName), error => console.log("Error downloading the file."),
        () => console.info("OK"));
}

На рішення посилається - тут


4

Оновлення відповіді Гектора за допомогою збереження файлів та HttpClient для кроку 2:

public downloadFile(file: File): Observable<Blob> {
    return this.http.get(file.fullPath, {responseType: 'blob'})
}

3

Я отримав рішення для завантаження з кутового 2, не пошкоджуючись, використовуючи пружинні mvc та angular 2

Перший - мій тип повернення: - ResponseEntity від java end. Ось я надсилаю масив byte [] має тип повернення від контролера.

2-е - включити файловий засіб у свою робочу область - на індексну сторінку як:

<script src="https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2014-11-29/FileSaver.min.js"></script>

3-й - на компоненті t написати цей код:

import {ResponseContentType} from '@angular.core';

let headers = new Headers({ 'Content-Type': 'application/json', 'MyApp-Application' : 'AppName', 'Accept': 'application/pdf' });
        let options = new RequestOptions({ headers: headers, responseType: ResponseContentType.Blob });
            this.http
            .post('/project/test/export',
                    somevalue,options)
              .subscribe(data => {

                  var mediaType = 'application/vnd.ms-excel';
                  let blob: Blob = data.blob();
                    window['saveAs'](blob, 'sample.xls');

                });

Це дасть вам формат файлу xls. Якщо ви хочете, щоб інші формати змінили медіатип і ім'я файлу з правильним розширенням.


3

Я зіткнувся з цим самим випадком сьогодні, мені довелося завантажити файл PDF у вигляді вкладеного файлу (файл не повинен відображатися у браузері, а завантажуватися замість нього). Щоб досягти цього, я виявив, що я повинен отримати файл у кутку Blob, і в той же час додати аContent-Disposition заголовок у відповідь.

Це було найпростіше, що я міг отримати (кутовий 7):

Всередині сервісу:

getFile(id: String): Observable<HttpResponse<Blob>> {
  return this.http.get(`./file/${id}`, {responseType: 'blob', observe: 'response'});
}

Потім, коли мені потрібно завантажити файл у складі, я можу просто:

fileService.getFile('123').subscribe((file: HttpResponse<Blob>) => window.location.href = file.url);

ОНОВЛЕННЯ:

Видалено непотрібні параметри заголовка зі служби


Якщо я використовую window.location.href замість window.open Chrome розглядає це як кілька завантажень файлів.
DanO

це не спрацює, якщо у вас є потрібний
авторський

3

Наступний код працював для мене

let link = document.createElement('a');
link.href = data.fileurl; //data is object received as response
link.download = data.fileurl.substr(data.fileurl.lastIndexOf('/') + 1);
link.click();

2

На сьогодні відповіді мені не вистачало розуміння, а також попереджень. Ви можете і слідкувати за несумісністю з IE10 + (якщо вам все одно).

Це повний приклад з додатком та сервісною частиною після. Зауважте, що ми встановлюємо зауваження : "відповідь", щоб зловити заголовок для імені файлу. Також зауважте, що заголовок Content-Disposition повинен бути встановлений та оголений сервером, інакше поточний Angular HttpClient не передасть його. Для цього я додав основний фрагмент коду точки дотнету .

public exportAsExcelFile(dataId: InputData) {
    return this.http.get(this.apiUrl + `event/export/${event.id}`, {
        responseType: "blob",
        observe: "response"
    }).pipe(
        tap(response => {
            this.downloadFile(response.body, this.parseFilename(response.headers.get('Content-Disposition')));
        })
    );
}

private downloadFile(data: Blob, filename: string) {
    const blob = new Blob([data], {type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=utf-8;'});
    if (navigator.msSaveBlob) { // IE 10+
        navigator.msSaveBlob(blob, filename);
    } else {
        const link = document.createElement('a');
        if (link.download !== undefined) {
            // Browsers that support HTML5 download attribute
            const url = URL.createObjectURL(blob);
            link.setAttribute('href', url);
            link.setAttribute('download', filename);
            link.style.visibility = 'hidden';
            document.body.appendChild(link);
            link.click();
            document.body.removeChild(link);
        }
    }
}

private parseFilename(contentDisposition): string {
    if (!contentDisposition) return null;
    let matches = /filename="(.*?)"/g.exec(contentDisposition);

    return matches && matches.length > 1 ? matches[1] : null;
}

Ядро Dotnet із вмістом-диспозицією та MediaType

 private object ConvertFileResponse(ExcelOutputDto excelOutput)
    {
        if (excelOutput != null)
        {
            ContentDisposition contentDisposition = new ContentDisposition
            {
                FileName = excelOutput.FileName.Contains(_excelExportService.XlsxExtension) ? excelOutput.FileName : "TeamsiteExport.xlsx",
                Inline = false
            };
            Response.Headers.Add("Access-Control-Expose-Headers", "Content-Disposition");
            Response.Headers.Add("Content-Disposition", contentDisposition.ToString());
            return File(excelOutput.ExcelSheet, "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
        }
        else
        {
            throw new UserFriendlyException("The excel output was empty due to no events.");
        }
    }

1
 let headers = new Headers({
                'Content-Type': 'application/json',
                'MyApp-Application': 'AppName',
                'Accept': 'application/vnd.ms-excel'
            });
            let options = new RequestOptions({
                headers: headers,
                responseType: ResponseContentType.Blob
            });


this.http.post(this.urlName + '/services/exportNewUpc', localStorageValue, options)
                .subscribe(data => {
                    if (navigator.appVersion.toString().indexOf('.NET') > 0)
                    window.navigator.msSaveBlob(data.blob(), "Export_NewUPC-Items_" + this.selectedcategory + "_" + this.retailname +"_Report_"+this.myDate+".xlsx");

                    else {
                        var a = document.createElement("a");
                        a.href = URL.createObjectURL(data.blob());
                        a.download = "Export_NewUPC-Items_" + this.selectedcategory + "_" + this.retailname +"_Report_"+this.myDate+ ".xlsx";
                        a.click();
                    }
                    this.ui_loader = false;
                    this.selectedexport = 0;
                }, error => {
                    console.log(error.json());
                    this.ui_loader = false;
                    document.getElementById("exceptionerror").click();
                });

1

Простіше кажучи , urlяк , hrefяк показано нижче.

<a href="my_url">Download File</a>

Це працює? Я отримую помилку ... "ERROR TypeError:" Доступ до 'файлу: ///Downloads/test.json' зі сценарію заборонений "".
Jay

Дякую, можете, будь ласка, поділіться, як виглядає URL-адреса? Чи це файл протоколу або http або щось інше?
Джей

Це протокол файлів.
Харунур Рашид

1

<a href="my_url" download="myfilename">Download file</a>

my_url має мати те саме походження, інакше він буде перенаправлений на це місце


так, це буде працювати для чистого html, не знаю, навіщо шукати інші складні
речі


1

Ви також можете завантажити файл безпосередньо з вашого шаблону, де ви використовуєте атрибут завантаження, і [attr.href]ви можете надати значення властивості з компонента. Це просте рішення має працювати в більшості браузерів.

<a download [attr.href]="yourDownloadLink"></a>

Довідка: https://www.w3schools.com/tags/att_a_download.asp


1
Ласкаво просимо до SO! Перевірте, чи корисні мої виправлення (набір тексту та граматики).
B - rian

0

Якщо ви надсилаєте параметри лише до URL-адреси, ви можете це зробити так:

downloadfile(runname: string, type: string): string {
   return window.location.href = `${this.files_api + this.title +"/"+ runname + "/?file="+ type}`;
}

в сервісі, який приймає параметри


0

Ця відповідь говорить про те, що ви не можете завантажувати файли безпосередньо з AJAX, в першу чергу з міркувань безпеки. Тому я опишу, що я роблю в цій ситуації,

01. Додайте hrefатрибут у тег якоря всередині component.htmlфайлу,
наприклад: -

<div>
       <a [href]="fileUrl" mat-raised-button (click)='getGenaratedLetterTemplate(element)'> GENARATE </a>
</div>

02. Виконайте всі наступні кроки, component.tsщоб обійти рівень безпеки, і переведіть діалогове вікно збереження як спливаюче вікно,
наприклад: -

import { environment } from 'environments/environment';
import { DomSanitizer } from '@angular/platform-browser';
export class ViewHrApprovalComponent implements OnInit {
private apiUrl = environment.apiUrl;
  fileUrl
 constructor(
    private sanitizer: DomSanitizer,
    private letterService: LetterService) {}
getGenaratedLetterTemplate(letter) {

    this.data.getGenaratedLetterTemplate(letter.letterId).subscribe(
      // cannot download files directly with AJAX, primarily for security reasons);
    console.log(this.apiUrl + 'getGeneratedLetter/' + letter.letterId);
    this.fileUrl = this.sanitizer.bypassSecurityTrustResourceUrl(this.apiUrl + 'getGeneratedLetter/' + letter.letterId);
  }

Примітка. Ця відповідь спрацює, якщо ви отримуєте помилку "ОК" з кодом статусу 200


0

Ну, я написав фрагмент коду, натхненний багатьма вищезазначеними відповідями, який повинен легко працювати в більшості сценаріїв, коли сервер надсилає файл із заголовком диспозиції вмісту, без сторонніх установок, крім rxjs та angular.

По-перше, як викликати код з файлу компонента

this.httpclient.get(
   `${myBackend}`,
   {
      observe: 'response',
      responseType: 'blob'
   }
).pipe(first())
.subscribe(response => SaveFileResponse(response, 'Custom File Name.extension'));

Як бачите, це в основному середній дзвінок із зворотного зв'язку з кутового, з двома змінами

  1. Я спостерігаю за реакцією замість тіла
  2. Я відверто кажу про те, що відповідь є краплиною

Після отримання файлу з сервера я, в принципі, делегую все завдання збереження файлу до помічникової функції, яку я зберігаю в окремому файлі, та імпорту в той компонент, який мені потрібно

export const SaveFileResponse = 
(response: HttpResponse<Blob>, 
 filename: string = null) => 
{
    //null-checks, just because :P
    if (response == null || response.body == null)
        return;

    let serverProvidesName: boolean = true;
    if (filename != null)
        serverProvidesName = false;

    //assuming the header is something like
    //content-disposition: attachment; filename=TestDownload.xlsx; filename*=UTF-8''TestDownload.xlsx
    if (serverProvidesName)
        try {
            let f: string = response.headers.get('content-disposition').split(';')[1];
            if (f.includes('filename='))
                filename = f.substring(10);
        }
        catch { }
    SaveFile(response.body, filename);
}

//Create an anchor element, attach file to it, and
//programmatically click it. 
export const SaveFile = (blobfile: Blob, filename: string = null) => {
    const a = document.createElement('a');
    a.href = window.URL.createObjectURL(blobfile);
    a.download = filename;
    a.click();
}

Там більше немає нахабних імен GUID! Ми можемо використовувати будь-яке ім’я, яке надає сервер, не вказуючи його явно в клієнті, або перезаписувати ім'я файлу, надане сервером (як у цьому прикладі). Крім того, можна легко змінити алгоритм вилучення імені файлу з вмісту-диспозиції відповідно до їх потреб, а все інше залишиться без змін - у випадку помилки під час такого вилучення воно просто пройде "null" як ім'я файлу.

Як вже було зазначено в іншій відповіді, ІЕ, як завжди, потребує спеціального лікування. Але якщо край із хромом з’явиться через кілька місяців, я б не переживав про це під час створення нових додатків (сподіваюся). Існує також питання про відкликання URL-адреси, але я якось не дуже впевнений у цьому, тож якщо хтось міг би допомогти з цим у коментарях, це було б дивним.


0

Якщо вкладка відкривається і закривається, не завантажуючи нічого, я спробував перейти з макетом якорного посилання, і це працювало.

downloadFile(x: any) {
var newBlob = new Blob([x], { type: "application/octet-stream" });

    // IE doesn't allow using a blob object directly as link href
    // instead it is necessary to use msSaveOrOpenBlob
    if (window.navigator && window.navigator.msSaveOrOpenBlob) {
      window.navigator.msSaveOrOpenBlob(newBlob);
      return;
    }

    // For other browsers: 
    // Create a link pointing to the ObjectURL containing the blob.
    const data = window.URL.createObjectURL(newBlob);

    var link = document.createElement('a');
    link.href = data;
    link.download = "mapped.xlsx";
    // this is necessary as link.click() does not work on the latest firefox
    link.dispatchEvent(new MouseEvent('click', { bubbles: true, cancelable: true, view: window }));

    setTimeout(function () {
      // For Firefox it is necessary to delay revoking the ObjectURL
      window.URL.revokeObjectURL(data);
      link.remove();
    }, 100);  }
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.