跳到主要内容
版本:v7

将照片保存到文件系统

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

文件系统 API

幸运的是,将它们保存到文件系统只需要几个步骤。首先在 usePhotoGallery.tsusePhotoGallery() 方法中创建一个新的类方法 savePicture()

import { ref } from 'vue';
import { Camera, CameraResultType, CameraSource } from '@capacitor/camera';
// 变更:添加导入
import type { Photo } from '@capacitor/camera';

export const usePhotoGallery = () => {
// ...现有代码...

// 变更:添加 `savePicture()` 方法
const savePicture = async (photo: Photo, fileName: string): Promise<UserPhoto> => {
return {
filepath: 'soon...',
webviewPath: 'soon...',
};
};

return {
addNewToGallery,
photos,
};
};

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

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

import { ref } from 'vue';
import { Camera, CameraResultType, CameraSource } from '@capacitor/camera';
import type { Photo } from '@capacitor/camera';

export const usePhotoGallery = () => {
// ...现有代码...

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

const fileName = Date.now() + '.jpeg';
// 变更:添加 `savedImageFile`
// 保存图片并添加到照片集合
const savedImageFile = await savePicture(capturedPhoto, fileName);

photos.value = [savedImageFile, ...photos.value];
};

const savePicture = async (photo: Photo, fileName: string): Promise<UserPhoto> => {
return {
filepath: 'soon...',
webviewPath: 'soon...',
};
};

return {
addNewToGallery,
photos,
};
};

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

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

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

现在,创建一个新的辅助方法 convertBlobToBase64(),用于实现在 Web 上运行所需的逻辑。

import { ref } from 'vue';
import { Camera, CameraResultType, CameraSource } from '@capacitor/camera';
import type { Photo } from '@capacitor/camera';
// 变更:添加导入
import { Filesystem, Directory } from '@capacitor/filesystem';

export const usePhotoGallery = () => {
// ...现有代码...

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

const savedFile = await Filesystem.writeFile({
path: fileName,
data: base64Data,
directory: Directory.Data,
});

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

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

return {
addNewToGallery,
photos,
};
};

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

现在 usePhotoGallery.ts 应该如下所示:

import { ref } from 'vue';
import { Camera, CameraResultType, CameraSource } from '@capacitor/camera';
import type { Photo } from '@capacitor/camera';
import { Filesystem, Directory } from '@capacitor/filesystem';

export const usePhotoGallery = () => {
const photos = ref<UserPhoto[]>([]);

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

const fileName = Date.now() + '.jpeg';
// 保存图片并添加到照片集合
const savedImageFile = await savePicture(capturedPhoto, fileName);

photos.value = [savedImageFile, ...photos.value];
};

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

const savedFile = await Filesystem.writeFile({
path: fileName,
data: base64Data,
directory: Directory.Data,
});

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

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

return {
addNewToGallery,
photos,
};
};

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

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

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