【问题标题】:How to get the download Url of firebase file storage如何获取firebase文件存储的下载Url
【发布时间】:2018-08-29 06:42:04
【问题描述】:

我无法理解获取下载 URl 的过程,有人可以很好地解释一下吗? 所以我在这里有这个上传组件:

import { Component, OnInit } from '@angular/core';
import { AngularFireStorage, AngularFireUploadTask } from 
'angularfire2/storage';
import { AngularFirestore } from 'angularfire2/firestore'; 
import { Observable } from 'rxjs/Observable';
import { tap, filter, switchMap } from 'rxjs/operators';
import { storage } from 'firebase/storage';


@Component({
  selector: 'file-upload',
  templateUrl: './file-upload.component.html',
  styleUrls: ['./file-upload.component.scss']
})
export class FileUploadComponent {

  // Main task   
  task: AngularFireUploadTask;

  // Progress monitoring
  percentage: Observable<number>;

  snapshot: Observable<any>;

  // Download URL
  downloadURL: Observable<string>;

  // State for dropzone CSS toggling 
  isHovering: boolean;

  constructor(private storage: AngularFireStorage, private db: AngularFirestore) { }

  toggleHover(event: boolean) {
    this.isHovering = event;
  }

  startUpload(event: FileList) {
    // The File object
    const file = event.item(0)

    // Client-side validation example
    if (file.type.split('/')[0] !== 'image') { 
      console.error('unsupported file type :( ')
      return;
    }

    // The storage path
    const path = `test/${new Date().getTime()}_${file.name}`;

    // Totally optional metadata
    const customMetadata = { app: 'My AngularFire-powered PWA!' };

    // The main task
    this.task = this.storage.upload(path, file, { customMetadata })

    // Progress monitoring
    this.percentage = this.task.percentageChanges();
    this.snapshot   = this.task.snapshotChanges().pipe(
      tap(snap => {
        console.log(snap)
        if (snap.bytesTransferred === snap.totalBytes) {
          // Update firestore on completion
          this.db.collection('photos').add( { path, size: snap.totalBytes })
        }
      })
    )


    // The file's download URL
    this.downloadURL = this.task.downloadURL(); 
    console.log(this.downloadURL)

  const ref = this.storage.ref(path);
  this.task = ref.put(file, {customMetadata});

  this.downloadURL = this.task.snapshotChanges().pipe(
    filter(snap => snap.state === storage.TaskState.SUCCESS),
    switchMap(() => ref.getDownloadURL())
  )
  console.log(this.downloadURL);
}

  // Determines if the upload task is active
  isActive(snapshot) {
    return snapshot.state === 'running' && snapshot.bytesTransferred < snapshot.totalBytes
  }}

我尝试控制获取下载 URL 的假定方式,但它是空的,我已经看到了一些其他方法来完成它,但似乎无法正确完成。即使使用 snapshot.downloadURL,下载 URL 也始终为空。 以下是 包.json:

{
  "name": "storage-app",
  "version": "0.0.0",
  "license": "MIT",
  "scripts": {
    "ng": "ng",
    "start": "ng serve",
    "build": "ng build",
    "test": "ng test",
    "lint": "ng lint",
    "e2e": "ng e2e"
  },
  "private": true,
  "dependencies": {
    "@angular/animations": "^6.0.3",
    "@angular/common": "^6.0.3",
    "@angular/compiler": "^6.0.3",
    "@angular/core": "^6.0.3",
    "@angular/forms": "^6.0.3",
    "@angular/http": "^6.0.3",
    "@angular/platform-browser": "^6.0.3",
    "@angular/platform-browser-dynamic": "^6.0.3",
    "@angular/platform-server": "^6.0.3",
    "@angular/router": "^6.0.3",
    "angularfire2": "5.0.0-rc.6",
    "core-js": "^2.5.4",
    "firebase": "4.12.1",
    "rxjs": "^6.0.0",
    "rxjs-compat": "^6.2.2",
    "zone.js": "^0.8.26"
  },
  "devDependencies": {
    "@angular/cli": "^6.0.8",
    "@angular/compiler-cli": "^6.0.3",
    "@angular/language-service": "^6.0.3",
    "@types/jasmine": "~2.5.53",
    "@types/jasminewd2": "~2.0.2",
    "@types/node": "~6.0.60",
    "codelyzer": "^4.0.1",
    "jasmine-core": "~2.6.2",
    "jasmine-spec-reporter": "~4.1.0",
    "karma": "~1.7.0",
    "karma-chrome-launcher": "~2.1.1",
    "karma-cli": "~1.0.1",
    "karma-coverage-istanbul-reporter": "^1.2.1",
    "karma-jasmine": "~1.1.0",
    "karma-jasmine-html-reporter": "^0.2.2",
    "protractor": "~5.1.2",
    "ts-node": "~3.2.0",
    "tslint": "~5.7.0",
    "typescript": "~2.7.2",
    "@angular-devkit/build-angular": "~0.6.8"
  }
}

提前致谢

【问题讨论】:

标签: javascript angular firebase


【解决方案1】:

您可以从存储参考中检索下载网址:

loading = false;

uploadFile(event) {

    this.loading = true;

    const file = event.target.files[0];
    // give it a random file name
    const path = Math.random().toString(36).substring(7); 
    const storageRef = this.storage.ref(path);
    const task = this.storage.upload(path, file);

    return from(task).pipe(
      switchMap(() => storageRef.getDownloadURL()),
      tap(url => {
          // use url here, e.g. assign it to a model
      }),
      mergeMap(() => {
          // e.g. make api call, e.g. save the model 
      }),
      finalize(() => this.loading = false)
    ).subscribe(() => {
      // success
    }, error => {
      // failure
    });
  }

我正在使用 angularfire 5.0.0-rc.11

【讨论】:

  • mergeMap 中的箭头函数抛出此错误:'() => void' 类型的参数不可分配给 '(value: any, index: number) => ObservableInput 类型的参数'。类型 'void' 不可分配给类型 'ObservableInput' 谢谢
  • 你能解释一下tap和mergemap中需要的工作吗?
  • @JawadFarooqi 正如错误所说,您从 mergeMap 返回 void 但它期待一个可观察的。如果不需要,可以移除 tap 和 mergeMap,并在成功回调中检索 url:subscribe(url => {})。如果您想制作例如,您可以使用 mergeMap将 url 存储为副作用的 api 调用。
【解决方案2】:

这里有一个简单的例子来帮助你理解如何(取自 AngularFire2 GitHub):

uploadPercent: Observable < number > ;
downloadURL: Observable < string > ;

constructor(
  private storage: AngularFireStorage
) {
}

uploadFile(event) {
  const file = event.target.files[0];
  const filePath = 'files';
  const fileRef = this.storage.ref(filePath);
  const task = this.storage.upload(filePath, file);

  // observe percentage changes
  this.uploadPercent = task.percentageChanges();
  // get notified when the download URL is available
  task.snapshotChanges().pipe(
      finalize(() => this.downloadURL = fileRef.getDownloadURL())
    )
    .subscribe()
}

下面是这个模板:

<input type="file" (change)="uploadFile($event)" />
<div>{{ uploadPercent | async }}</div>
<a [href]="downloadURL | async">{{ downloadURL | async }}</a>

发生了什么:

我们正在处理文件输入的change 事件。完成后,我们将获取要上传的文件。然后,我们通过调用ref 并将其传递给我们的文件的路径,在我们的 Firebase 存储上创建一个文件引用。这将有助于稍后检索文件下载 URL。

之后,我们通过在 storage 上调用 upload 并将文件路径和要上传的文件传递给它来创建一个 AngularFireUplaodTask。

在这个任务中,我们可以通过在上传任务上调用percentageChanges来检查文件上传百分比。这又是一个 Observable,因此我们使用 async 管道监听并打印 DOM 上的变化。

finalize 将在上传任务完成时触发。那时我们就可以通过在我们之前创建的fileRef 上调用getDownloadURL 来获取下载网址。

您可以查看此StackBlitz 了解更多信息。

【讨论】:

  • 如果用户中断上传或出现错误,是否会调用 finalize?
【解决方案3】:

现在无法在上传的即时结果中访问下载 URL。此更改是在几个月前对 Firebase 客户端 SDK 进行的。

相反,您必须在上传完成后调用 getDownloadURL(或任何用于该 JavaScript 函数的 Angular 绑定)来获取 URL 作为第二个请求。

【讨论】:

    【解决方案4】:

    Angularfire 提供了这个超级方便的管道,getDownloadURL:

    <img [src]="'users/davideast.jpg' | getDownloadURL" />
    

    官方docs.

    结束。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2018-12-28
      • 1970-01-01
      • 2021-08-22
      • 2018-03-23
      • 2019-08-19
      相关资源
      最近更新 更多