Как загрузить файл с помощью Angular2
У меня есть приложение WebApi/MVC, для которого я разрабатываю клиент angular2 (вместо MVC). У меня возникли проблемы с пониманием того, как Angular сохраняет файл.
Запрос в порядке (отлично работает с MVC, и мы можем регистрировать полученные данные), но я не могу понять, как сохранить загруженные данные (я в основном следую той же логике, что и в этом посте). Я уверен, что это глупо просто, но пока я просто не понимаю этого.
Код функции компонента приведен ниже. Я пробовал разные альтернативы, путь blob должен быть таким, насколько я понял, но в URL
нет функции createObjectURL
. Я даже не могу найти определение 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
Проблема в том, что observable запускается в другом контексте, поэтому, когда вы пытаетесь создать URL var, у вас есть пустой объект, а не нужный вам BLOB-объект.
Один из многих способов решить эту проблему заключается в следующем:
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' ;
Я надеюсь это тебе поможет.
Ответ 2
Попробуйте это !
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 с помощью "файловой заставки",
import {saveAs as importedSaveAs} from "file-saver";
this.myService.downloadFile(this.id).subscribe(blob => {
importedSaveAs(blob, this.fileName);
}
)
Это работает для меня!
Ответ 3
Если вам не нужно добавлять заголовки в запрос, чтобы загрузить файл в Angular2, вы можете сделать простой:
window.location.href='http://example.com/myuri/report?param=x';
в вашем компоненте.
Ответ 4
Это для людей, которые смотрят, как это сделать, используя HttpClient и файловую заставку:
- Установить файловую заставку
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'));
}
Ответ 5
Как насчет этого?
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();
})
Я мог бы с этим справиться.
нет необходимости в дополнительном пакете.
Ответ 6
Как упоминал Алехандро Корредор, это простая ошибка масштаба. 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 загрузит данные, избегая вложенности и решая проблему async.
//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
Загрузите *.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']);
}
Ответ 8
Для более новых угловых версий:
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');
});
Ответ 9
Загрузка файла через ajax всегда является болезненным процессом, и, на мой взгляд, лучше разрешить сервер и браузер выполнять эту работу по согласованию типов контента.
Я считаю, что лучше всего
<a href="api/sample/download"></a>
сделать это. Это даже не требует открытия новых окон и тому подобного.
Контроллер MVC, как и в вашем примере, может выглядеть следующим образом:
[HttpGet("[action]")]
public async Task<FileContentResult> DownloadFile()
{
// ...
return File(dataStream.ToArray(), "text/plain", "myblob.txt");
}
Ответ 10
Я разделяю решение, которое помогло мне (любое улучшение с благодарностью)
На вашем сервисе "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);
}
На компонентной части вы вызываете услугу, не подписываясь на ответ. Подпишитесь на полный список типов openOffice mime: http://www.openoffice.org/framework/documentation/mimetypes/mimetypes.html
Ответ 11
Для тех, кто использует шаблон Redux
Я добавил в файл-хранитель как @Hector Куэвас, названный в его ответе. Используя 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: [] });
}
Надеюсь, это полезно.
Ответ 12
Будет лучше, если вы попытаетесь вызвать новый метод внутри вашей 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);
}
}
Ответ 13
Чтобы загрузить и показать файлы в формате 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!")
);
}
Ответ 14
Я использую 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.')
});
}
Значение this.downloadUrl было установлено ранее, чтобы указать на api. Я использую это для загрузки вложений, поэтому я знаю id, contentType и filename: я использую 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 возвращает файл из базы данных.
Надеюсь, это поможет кому-то :)
Ответ 15
Обновите ответ Гектора, используя файловую заставку и HttpClient для шага 2:
public downloadFile(file: File): Observable<Blob> {
return this.http.get(file.fullPath, {responseType: 'blob'})
}
Ответ 16
Вот что я сделал в моем случае -
// 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"));
}
На решение ссылаются - здесь
Ответ 17
Я получил решение для загрузки с углового 2 без повреждения, используя весенний mvc и угловой 2
1st- мой тип возврата: - ResponseEntity от java-конца. Здесь я посылаю массив byte [] с возвращаемым типом с контроллера.
2nd- включить файловый менеджер в рабочую область - на индексной странице:
<script src="https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2014-11-29/FileSaver.min.js"></script>
3rd- в компоненте 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. Если вы хотите, чтобы другие форматы изменяли медиатип и имя файла с правильным расширением.
Ответ 18
Сегодня я столкнулся с тем же случаем, мне пришлось загрузить файл 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);
ОБНОВИТЬ:
Удалены ненужные настройки заголовка из сервиса
Ответ 19
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();
});
Ответ 20
Просто введите url
как href
как href
ниже.
<a href="my_url">Download File</a>
Ответ 21
<a href="my_url" download="myfilename">Download file</a>
my_url должен иметь то же происхождение, иначе он будет перенаправлен в это место
Ответ 22
Вы также можете загрузить файл непосредственно из вашего шаблона, где вы используете атрибут загрузки, и в [attr.href]
вы можете указать значение свойства из компонента.
Это простое решение должно работать в большинстве браузеров.
<a download [attr.href]="yourDownloadLink"></a>
Ссылка: https://www.w3schools.com/tags/att_a_download.asp
Ответ 23
Если вы отправляете параметры только на URL, вы можете сделать это следующим образом:
downloadfile(runname: string, type: string): string {
return window.location.href = '${this.files_api + this.title +"/"+ runname + "/?file="+ type}';
}
в сервисе, который получает параметры
Ответ 24
Этот ответ предполагает, что вы не можете загружать файлы напрямую с помощью 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.