Как загрузить файл с 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. Если бы ты сказал, что было лучше.
Стив

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

Ответы:


181

Проблема в том, что наблюдаемое работает в другом контексте, поэтому при попытке создать переменную 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
@ Буржуа getReport()возвращаетсяthis.http.get(PriceConf.download.url)
Джи-

6
У меня проблема в том, что окно открывается и закрывается сразу же, не загружая файл
Бреден Браун

7
Как мы можем установить имя файла здесь? по умолчанию он выбирает числовое значение в качестве имени
Saurabh

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- В компоненте проанализируйте BLOB-объект с помощью «File-Saver»

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' не может быть назначен параметру типа '{headers ?: HttpHeaders | {[header: string]: строка | Строка []; };
GiveJob

Этот, однако, не подходит для загрузки больших файлов.
Reven

61

Если вам не нужно добавлять заголовки в запрос, скачать файл в Angular2 вы можете просто :

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

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


4
Может кто-нибудь сказать, пожалуйста, почему этот ответ опущен? Тема состоит в том, чтобы загрузить файл, используя angular2. Если этот метод работает для простой загрузки, он также должен быть помечен как правильный ответ.
Саураб Шетти,

5
@SaurabhShetty, это не поможет в случае, если вы хотите отправить пользовательские заголовки, что, если вы хотите отправить токен аутентификации, например? Если вы посмотрите на вопрос ОП, вы увидите, что он использует authHttp!
А.Акрам

6
Я понимаю отрицательные голоса, но этот ответ решил мою проблему.
JoeriShoeby

1
Если вы позволите серверу вернуть URL в каком-то контексте, сервер может подготовить URL. пример: объект: MyRecord.Cover. Обложка может быть ссылкой на изображение на сервере. При вызове get (Myrecord) вы позволяете серверу вернуть подготовленный url (Cover) с установленным токеном безопасности и другими заголовками.
Йенс Алениус

2
Это ответ, который работает. Просто потому, что у него нет <вставить полезную функцию здесь>, что делает его не ответом.
Гбертон

47

Это для людей, которые ищут, как это сделать, используя HttpClient и File-Saver:

  1. Установить файл-заставку

npm установить файл-заставку --save

npm install @ 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, не создает никаких элементов. Я совместил это решение с некоторыми из вышеперечисленных, и оно работает как шарм.
Chax

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в BLOB-объект и действительно хотите res._body. Однако _bodyэто закрытая переменная и не доступна. На сегодняшний день .blob()и .arrayBuffer()на объекте ответа HTTP не были реализован в угловом 2. text()и json()являются только два варианта , но оба будет искажать ваше тело. Вы нашли решение?
sschueller

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

1
2 варианта позволяют мне загрузить файл, но сначала он загружает данные в фоновом режиме. Что если у меня есть большой файл, который нужно загрузить?
f123

1
Мое решение - просто использовать <a href=""></a>для загрузки файла.
user2061057

1
Я знаю, что это старый ответ, но он находится высоко над результатами поиска и является приемлемым ответом: строка `headers.append ('responseType', 'arraybuffer');` неверна. Это вариант, а не заголовок. Пожалуйста, исправьте это. А-а-а ... Заголовки созданы и не используются. Не полезно.
Стево

17

Загрузите * .zip решение для angular 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 в другом окне браузера, а это не то, что хочет видеть пользователь
Luca

2
Если у вас есть токен доступа, вам нужно
убедиться, что

Это просто и понятно. Но если вы хотите провести некоторую аутентификацию, то есть возможность получить что-то вроде одноразового токена. Таким образом, вместо этого вы можете получить URL-адрес: example.com/myuri/report?tokenid=1234-1233 и проверить идентификатор токена в базе данных. Конечно, это не простой сценарий и работает во всех ситуациях, но может быть решением в ситуации, когда у вас есть доступ к базе данных, прежде чем возвращать отчет в виде потока.
GingerBeer

Получить URL-адрес загрузки с сервера. Таким образом, сервер может подготовить URL с помощью одноразового токена безопасности.
Йенс Алениус

8

Я использую Angular 4 с объектом httpClient 4.3. Я изменил ответ, который нашел в техническом блоге 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.')
        }); 

} 

Значение this.downloadUrl было установлено ранее, чтобы указывать на API. Я использую это для загрузки вложений, поэтому я знаю идентификатор, contentType и имя файла: я использую API MVC, чтобы вернуть файл:

 [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, мне не нужно было добавлять в @ types / 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

Я поделюсь решением, которое помогло мне (любое улучшение очень ценится)

К вашим услугам «Псервис»:

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);
}

В компонентной части вы вызываете сервис, не подписываясь на ответ. Подписка на полный список типов пантомимы 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

Я получил решение для загрузки из Angular 2 без повреждения, используя Spring MVC и Angular 2

1-й - мой тип возврата: - ResponseEntity от конца Java. Здесь я отправляю массив byte [] имеет тип возврата из контроллера.

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

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

3-й компонент ts, напишите этот код:

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 в виде вложения (файл не должен отображаться в браузере, а загружаться вместо него). Чтобы добиться этого, я обнаружил, что мне нужно получить файл в формате Angular Blobи одновременно добавитьContent-Disposition заголовок в ответ.

Это было самое простое, что я мог получить (Angular 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

это не будет работать, если у вас есть токен, необходимый в заголовке
garg10may

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 не будет передавать его. Я добавил фрагмент кода для dotnet ниже.

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, с Content-Disposition & 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>

Это работает? Я получаю ошибку ... "ОШИБКА TypeError:" Доступ к 'file: ///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
Добро пожаловать на ТАК! Пожалуйста, проверьте, полезны ли мои (верстка и грамматика) исправления.
B -

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'));

Как видите, это в основном средний бэкэнд-вызов из angular с двумя изменениями

  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' как имя файла.

Как уже отмечалось в другом ответе, IE, как всегда, нуждается в особом подходе. Но с появлением хрома через несколько месяцев я бы не стал беспокоиться об этом при создании новых приложений (надеюсь). Существует также вопрос об отзыве URL, но я не совсем уверен в этом, так что если кто-то может помочь с этим в комментариях, это было бы здорово.


0

Если вкладка открывается и закрывается без загрузки чего-либо, я попытался перейти с помощью ссылки «mock anchor», и это сработало.

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.