跳到主要内容
版本:v7

将照片保存到文件系统

现在我们已经能够在应用的第二个标签页上拍摄多张照片并在照片库中显示。然而,这些照片目前并未被永久存储,因此当应用关闭时,它们将被删除。

文件系统 API

幸运的是,将它们保存到文件系统只需要几个步骤。首先在 PhotoService 类中创建一个新的类方法 savePicture()。我们传入 photo 对象,它代表新拍摄的设备照片:

import { Camera, CameraResultType, CameraSource } from '@capacitor/camera';
// 修改:添加导入
import type { Photo } from '@capacitor/camera';

@Injectable({
providedIn: 'root',
})
export class PhotoService {
// ...现有代码...

// 修改:添加 `savePicture()` 方法
private async savePicture(photo: Photo) {
return {
filepath: 'soon...',
webviewPath: 'soon...',
};
}
}

export interface UserPhoto {
filepath: string;
webviewPath?: string;
}

我们可以立即在 addNewToGallery() 中使用这个新方法。

import { Injectable } from '@angular/core';
import { Camera, CameraResultType, CameraSource, Photo } from '@capacitor/camera';

@Injectable({
providedIn: 'root',
})
export class PhotoService {
public photos: UserPhoto[] = [];

// 修改:更新 `addNewToGallery()` 方法
public async addNewToGallery() {
// 拍摄照片
const capturedPhoto = await Camera.getPhoto({
resultType: CameraResultType.Uri,
source: CameraSource.Camera,
quality: 100,
});

// 修改:添加 `savedImageFile`
// 保存图片并添加到照片集合
const savedImageFile = await this.savePicture(capturedPhoto);

// 修改:更新 unshift 数组方法的参数
this.photos.unshift(savedImageFile);
}

private async savePicture(photo: Photo) {
return {
filepath: 'soon...',
webviewPath: 'soon...',
};
}
}

export interface UserPhoto {
filepath: string;
webviewPath?: string;
}

我们将使用 Capacitor 的文件系统 API来保存照片。首先,将照片转换为 base64 格式。

然后,将数据传递给文件系统的 writeFile 方法。回想一下,我们通过将图像的源路径(src)设置为 webviewPath 属性来显示照片。因此,设置 webviewPath 并返回新的 Photo 对象。

现在,创建一个新的辅助方法 convertBlobToBase64(),以实现网页运行所需的逻辑。

import { Injectable } from '@angular/core';
import { Camera, CameraResultType, CameraSource } from '@capacitor/camera';
import type { Photo } from '@capacitor/camera';
// 修改:添加导入
import { Filesystem, Directory } from '@capacitor/filesystem';

@Injectable({
providedIn: 'root',
})
export class PhotoService {
// ...现有代码...

// 修改:更新 `savePicture()` 方法
private async savePicture(photo: Photo) {
// 获取照片,读取为 blob,然后转换为 base64 格式
const response = await fetch(photo.webPath!);
const blob = await response.blob();
const base64Data = (await this.convertBlobToBase64(blob)) as string;

// 将文件写入数据目录
const fileName = Date.now() + '.jpeg';
const savedFile = await Filesystem.writeFile({
path: fileName,
data: base64Data,
directory: Directory.Data,
});

// 使用 webPath 显示新图像而不是 base64,因为它已经加载到内存中
return {
filepath: fileName,
webviewPath: photo.webPath,
};
}

// 修改:添加 `convertBlobToBase64` 方法
private convertBlobToBase64(blob: Blob) {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.onerror = reject;
reader.onload = () => {
resolve(reader.result);
};
reader.readAsDataURL(blob);
});
}
}

export interface UserPhoto {
filepath: string;
webviewPath?: string;
}

现在 photo.service.ts 应该看起来像这样:

import { Injectable } from '@angular/core';
import { Camera, CameraResultType, CameraSource } from '@capacitor/camera';
import type { Photo } from '@capacitor/camera';
import { Filesystem, Directory } from '@capacitor/filesystem';

@Injectable({
providedIn: 'root',
})
export class PhotoService {
public photos: UserPhoto[] = [];

public async addNewToGallery() {
// 拍摄照片
const capturedPhoto = await Camera.getPhoto({
resultType: CameraResultType.Uri,
source: CameraSource.Camera,
quality: 100,
});

// 保存图片并添加到照片集合
const savedImageFile = await this.savePicture(capturedPhoto);

this.photos.unshift(savedImageFile);
}

private async savePicture(photo: Photo) {
// 获取照片,读取为 blob,然后转换为 base64 格式
const response = await fetch(photo.webPath!);
const blob = await response.blob();
const base64Data = (await this.convertBlobToBase64(blob)) as string;

// 将文件写入数据目录
const fileName = Date.now() + '.jpeg';
const savedFile = await Filesystem.writeFile({
path: fileName,
data: base64Data,
directory: Directory.Data,
});

// 使用 webPath 显示新图像而不是 base64,因为它已经加载到内存中
return {
filepath: fileName,
webviewPath: photo.webPath,
};
}

private convertBlobToBase64(blob: Blob) {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.onerror = reject;
reader.onload = () => {
resolve(reader.result);
};
reader.readAsDataURL(blob);
});
}
}

export interface UserPhoto {
filepath: string;
webviewPath?: string;
}

在网页上获取相机照片的 base64 格式似乎比在移动设备上要复杂一些。实际上,我们只是使用内置的网页 API:fetch() 作为一种简洁的方式读取文件为 blob 格式,然后使用 FileReader 的 readAsDataURL() 将照片 blob 转换为 base64。

就是这样!每次拍摄新照片时,它现在都会自动保存到文件系统。接下来,我们将加载并显示已保存的图像。