ฉันจะดาวน์โหลดไฟล์ด้วย 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-2GB
Matthew B.

@MatthewB หวังว่าคุณจะพูดในสิ่งที่ดีกว่า
สตีฟ

สำหรับการดาวน์โหลดไฟล์ขนาดใหญ่คุณจำเป็นต้องระบุแท็บใหม่เช่นหากจำลองการคลิก <A> เป้าหมายต้องเท่ากับ "_blank" หรือส่งแบบฟอร์ม ฉันไม่คิดว่าจะมีวิธีที่สะอาดในการหลีกเลี่ยงข้อ จำกัด ขนาดไฟล์ขนาดใหญ่ด้วยคำขอสไตล์ Ajax
Matthew B.

คำตอบ:


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()และมันกลับมาอย่างไร?
Burjua

3
@Burjua getReport()ผลตอบแทนthis.http.get(PriceConf.download.url)
ji-ruh

6
ปัญหาที่ฉันมีคือหน้าต่างเปิดและปิดทันทีที่ไม่ได้ดาวน์โหลดไฟล์
Braden Brown

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' ไม่สามารถกำหนดให้กับพารามิเตอร์ประเภท '{ส่วนหัว?: HttpHeaders | {[header: string]: string | สตริง []; };
giveJob

อย่างไรก็ตามไฟล์นี้ไม่เหมาะสำหรับการดาวน์โหลดไฟล์ขนาดใหญ่
Reven

61

หากคุณไม่ต้องการเพิ่มส่วนหัวในคำขอเพื่อดาวน์โหลดไฟล์ใน Angular2 คุณสามารถทำได้ง่าย ๆ :

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

ในองค์ประกอบของคุณ


4
ใครช่วยกรุณาบอกว่าทำไมคำตอบนี้จึงลดลง? หัวข้อคือการดาวน์โหลดไฟล์โดยใช้ angular2 หากวิธีนี้ใช้ในการดาวน์โหลดอย่างง่ายก็ควรทำเครื่องหมายเป็นคำตอบที่ถูกต้อง
Saurabh Shetty

5
@SaurabhShetty สิ่งนี้จะไม่ช่วยในกรณีที่คุณต้องการส่งส่วนหัวที่กำหนดเองจะทำอย่างไรถ้าคุณต้องการส่งโทเค็นการตรวจสอบสิทธิ์ หากคุณดูคำถาม OP คุณจะเห็นว่าเขาใช้authHttp!
A.Akram

6
ฉันเข้าใจ downvotes แต่คำตอบนี้สามารถแก้ไขปัญหาของฉันได้
JoeriShoeby

1
หากคุณให้เซิร์ฟเวอร์ส่งคืน URL ในบางบริบทเซิร์ฟเวอร์อาจเตรียม URL เช่นวัตถุ: MyRecord.Cover หน้าปกอาจเป็น URL ของรูปภาพในเซิร์ฟเวอร์ เมื่อเรียกใช้ get (Myrecord) คุณให้เซิร์ฟเวอร์คืน url ที่เตรียมไว้ (ปก) พร้อมกับโทเค็นความปลอดภัยและชุดส่วนหัวอื่น ๆ
Jens Alenius

2
มันเป็นคำตอบที่ได้ผล เพียงเพราะมันไม่มี <คุณสมบัติที่มีประโยชน์ที่นี่> ที่ไม่ทำให้คำตอบ
gburton

47

นี่สำหรับคนที่กำลังมองหาวิธีการใช้ HttpClient และ file-saver:

  1. ติดตั้งโปรแกรมรักษาไฟล์

npm ติดตั้งโปรแกรมรักษาไฟล์ - บันทึก

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

ดังกล่าวโดยAlejandro Corredorมันเป็นข้อผิดพลาดขอบเขตง่าย ๆ The 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);
}

จากนั้นในส่วนที่เราเพิ่งสมัครและจัดการกับข้อมูลที่แมป มีความเป็นไปได้สองอย่าง ครั้งแรกตามที่แนะนำในโพสต์ต้นฉบับ แต่ต้องการการแก้ไขเล็กน้อยตามที่ระบุไว้โดย Alejandro:

//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
ผมคิดว่าเหตุผลที่ไฟล์ได้รับความเสียหายเป็นเพราะคุณโหลดลงในหยดและคุณต้องการจริงres res._bodyอย่างไรก็ตาม_bodyเป็นตัวแปรส่วนตัวและไม่สามารถเข้าถึงได้ ณ วันนี้.blob()และ.arrayBuffer()บนวัตถุการตอบสนอง http ยังไม่ได้รับการดำเนินการใน Angular 2 text()และjson()เป็นเพียงสองตัวเลือก แต่ทั้งสองจะบิดเบือนร่างกายของคุณ คุณพบทางออกหรือไม่?
sschueller

1
สวัสดี @rll ฉันทำตามขั้นตอนข้างต้นและฉันได้รับการสมัครเมื่อเสร็จสมบูรณ์ ฉันยังคงไม่เห็นไฟล์ที่ดาวน์โหลด ฉันไม่เห็นข้อผิดพลาดเช่นกัน กรุณาช่วย
AishApp

1
ตัวเลือก 2 ตัวช่วยให้ฉันสามารถดาวน์โหลดไฟล์ได้ แต่มันจะโหลดข้อมูลในพื้นหลังก่อน จะทำอย่างไรถ้าฉันมีไฟล์ขนาดใหญ่ที่ต้องดาวน์โหลด
f123

1
โซลูชันของฉันคือใช้<a href=""></a>เพื่อดาวน์โหลดไฟล์
user2061057

1
ฉันรู้ว่านี่เป็นคำตอบเก่า แต่สูงขึ้นในผลการค้นหาและเป็นคำตอบที่ยอมรับได้: บรรทัด `headers.append ('responseType', 'arraybuffer');` ผิด มันเป็นตัวเลือกไม่ใช่ส่วนหัว โปรดแก้ไข Aaaand ... ส่วนหัวถูกสร้างขึ้นและไม่ได้ใช้ ไม่เป็นประโยชน์
Stevo

17

ดาวน์โหลด * .zip solution สำหรับ 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');
  });

ขอบคุณทำงานกับแองกูลาร์ 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
หากคุณมีโทเค็นการเข้าถึงคุณต้องระบุสิ่งนี้ไม่ได้
chris31389

นี่คือธรรมดาและเรียบง่าย แต่ถ้าคุณต้องการพิสูจน์ตัวตนบางอย่างมีความเป็นไปได้ที่จะมีโทเค็นหนึ่งครั้ง ดังนั้นแทนที่จะให้มันเป็นแบบนี้คุณสามารถมี URL เป็น: example.com/myuri/report?tokenid=1234-1233และตรวจสอบโทเค็น ID ในฐานข้อมูล แน่นอนว่ามันไม่ใช่สถานการณ์ง่าย ๆ และทำงานได้ในทุกสถานการณ์ แต่สามารถเป็นวิธีแก้ปัญหาในสถานการณ์ที่คุณสามารถเข้าถึงฐานข้อมูลก่อนที่จะส่งรายงานกลับมาเป็นกระแสข้อมูล ..
GingerBeer

รับ URL ดาวน์โหลดจากเซิร์ฟเวอร์ ดังนั้นเซิร์ฟเวอร์สามารถเตรียม URL ด้วยโทเค็นความปลอดภัยที่เกิดขึ้นครั้งเดียว
Jens Alenius

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 ฉันกำลังใช้สิ่งนี้เพื่อดาวน์โหลดไฟล์แนบดังนั้นฉันจึงรู้ 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 ฉันไม่จำเป็นต้องเพิ่มใน @ 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
    });
};

The journal reducer แม้ว่านี่จะเป็นเพียงการตั้งค่าสถานะที่ถูกต้องที่ใช้ในแอปพลิเคชันของเรา แต่ฉันยังต้องการที่จะเพิ่มลงในเพื่อแสดงรูปแบบที่สมบูรณ์

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

อัปเดตเป็นคำตอบของ Hector โดยใช้ file-saver และ HttpClient สำหรับขั้นตอนที่ 2:

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

3

ฉันได้วิธีแก้ปัญหาสำหรับการดาวน์โหลดจาก angular 2 โดยไม่เสียหายโดยใช้ spring mvc และ angular 2

ประเภทที่หนึ่งของฉันคือ: - ResponseEntityจากปลายจาวา ที่นี่ฉันกำลังส่งไบต์ [] อาร์เรย์มีชนิดส่งคืนจากตัวควบคุม

2nd- เพื่อรวม filesaver ในเวิร์กสเปซของคุณในหน้าดัชนีเป็น:

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

UPDATE:

ลบการตั้งค่าส่วนหัวที่ไม่จำเป็นออกจากบริการ


ถ้าฉันใช้ 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 + (ถ้าคุณสนใจ)

นี่คือตัวอย่างที่สมบูรณ์ที่มีส่วนแอปพลิเคชันและส่วนบริการหลังจาก โปรดทราบว่าเราได้ตั้งค่าการสังเกต: "การตอบสนอง"เพื่อตรวจจับส่วนหัวของชื่อไฟล์ นอกจากนี้โปรดทราบว่าส่วนหัวการจัดการเนื้อหาจะต้องมีการตั้งค่าและเปิดเผยโดยเซิร์ฟเวอร์ไม่เช่นนั้น 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 core พร้อมด้วย 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 หรืออย่างอื่น?
เจย์

มันเป็นโปรโตคอลไฟล์
Harunur Rashid

1

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

my_url ควรมีต้นกำเนิดเดียวกันมิฉะนั้นจะเปลี่ยนเส้นทางไปยังตำแหน่งนั้น


ใช่นั่นจะใช้งานได้กับ html บริสุทธิ์ไม่แน่ใจว่าทำไมต้องค้นหาสิ่งที่ซับซ้อนอื่น ๆ
VIVEK SINGH


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แอ็ตทริบิวต์ในแท็ก anchor ของคุณภายใน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);
  }

หมายเหตุ: คำตอบนี้จะใช้งานได้หากคุณได้รับข้อผิดพลาด "OK" พร้อมรหัสสถานะ 200


0

ฉันเขียนโค้ดบางส่วนที่ได้แรงบันดาลใจจากคำตอบข้างต้นหลายข้อที่ควรทำงานได้ง่ายในสถานการณ์ส่วนใหญ่ที่เซิร์ฟเวอร์ส่งไฟล์ที่มีส่วนหัวการจัดการเนื้อหาโดยไม่มีการติดตั้งของบุคคลที่สามยกเว้น rxjs และเชิงมุม

ก่อนอื่นวิธีเรียกรหัสจากไฟล์คอมโพเนนต์ของคุณ

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 ที่เป็นความลับอีกต่อไป! เราสามารถใช้ชื่อใดก็ตามที่เซิร์ฟเวอร์ให้โดยไม่ต้องระบุอย่างชัดเจนในไคลเอนต์หรือเขียนทับชื่อไฟล์โดยเซิร์ฟเวอร์ (ดังในตัวอย่างนี้) นอกจากนี้เราสามารถเปลี่ยนอัลกอริทึมของการแยกชื่อไฟล์จาก content-disposition เพื่อให้เหมาะกับความต้องการของพวกเขาได้อย่างง่ายดายและทุกอย่างอื่นจะไม่ได้รับผลกระทบ - ในกรณีที่เกิดข้อผิดพลาดระหว่างการแยกมันจะผ่าน 'null' เป็นชื่อไฟล์

ดังที่คำตอบอื่นชี้ให้เห็นแล้ว IE ต้องการการดูแลเป็นพิเศษเช่นเคย แต่ด้วยขอบโครเมียมที่จะมาในอีกไม่กี่เดือนฉันไม่ต้องกังวลเกี่ยวกับเรื่องนี้ในขณะที่สร้างแอพใหม่ (หวังว่า) นอกจากนี้ยังมีเรื่องของการเพิกถอน 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);  }
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.