在其它手机端,若想保存图片到相册,需要申请对应的权限,而鸿蒙中对应的权限为受限开放权限,普通应用一般不让申请,这个时候我们可以使用安全保存控件来临时申请权限,用于保存图片到相册。
受限开放权限
应用权限分为三类,一类是对所有应用开放,所有应用均可申请使用;一类是受限开放权限,仅少量符合特殊场景的应用可在通过审批后,使用受限权限;最后一类是仅对 MDM(Mobile Device Management)设备管理应用开放。
保存图片到相机涉及到的权限是 ohos.permission.WRITE_IMAGEVIDEO,仅特殊场景与功能才可申请此权限,例如应用需要克隆、备份或同步图片/视频类文件,其它场景下使用安全控件来临时申请权限。
使用安全控件保存本地图片到相机
我们先使用安全控件让用户点击临时获取权限,获取到权限后,再使用 photoAccessHelper 来将我们本地的图片保存在相册,示例如下
import { photoAccessHelper } from '@kit.MediaLibraryKit'
@Entry
@ComponentV2
struct Index {
build() {
Column() {
SaveButton({ icon: SaveIconStyle.LINES, text: SaveDescription.SAVE_TO_GALLERY, buttonType: ButtonType.Capsule })
.onClick(() => {
this.savePhotoToGallery().then(() => {
this.getUIContext().getPromptAction().showToast({ message: '保存成功' })
}).catch((err: Error) => {
this.getUIContext().getPromptAction().showToast({ message: err.message })
})
})
}
}
public async savePhotoToGallery(): Promise<void> {
const ctx = getContext()
const helper = photoAccessHelper.getPhotoAccessHelper(ctx)
const src = ctx.resourceDir + '/icon.png'
const request = photoAccessHelper.MediaAssetChangeRequest.createImageAssetRequest(ctx, src)
return helper.applyChanges(request)
}
}
复制代码
以上示例请保证 icon.png 在本地真实的存在。当我们使用 helper 调用 applyChanges 时,因为是在安全控件点击后调用的,临时获取权限,可以正常执行,若不在安全控件内,则需保证已获取对应的权限。
## 使用安全控件保存服务端图片到相机
服务端图片我们一般使用下载服务将图片下载到本地,若本地不需要备份,则直接将下载好的图片 buffer 保存到相册即可。我们将本地图片转成 buffer 来模拟服务端下载后的图片,再使用 photoAccessHelper 创建一个相册图片资源,并将我们的图片 buffer 写入到这个图片资源中,就可以将图片保存到相册了,示例如下
import { photoAccessHelper } from '@kit.MediaLibraryKit'
import fs from '@ohos.file.fs'
@Entry
@ComponentV2
struct Index {
build() {
Column() {
SaveButton({ icon: SaveIconStyle.LINES, text: SaveDescription.SAVE_TO_GALLERY, buttonType: ButtonType.Capsule })
.onClick(() => {
this.savePhotoToGallery().then(() => {
this.getUIContext().getPromptAction().showToast({ message: '保存成功' })
}).catch((err: Error) => {
this.getUIContext().getPromptAction().showToast({ message: err.message })
})
})
}
}
public async savePhotoToGallery(): Promise<void> {
const ctx = getContext()
const helper = photoAccessHelper.getPhotoAccessHelper(ctx)
return Promise.all([helper.createAsset(photoAccessHelper.PhotoType.IMAGE, 'png').then((uri) => {
return fs.open(uri, fs.OpenMode.READ_WRITE | fs.OpenMode.CREATE)
}), ctx.resourceManager.getMediaContent($r('app.media.app_icon').id, 0)]).then((array) => {
fs.writeSync(array[0].fd, array[1].buffer)
fs.closeSync(array[0].fd)
})
}
}
复制代码
使用保存确认弹窗保存图片
前面的方式都是强依赖于安全控件,但在有些场景下,我们没办法使用安全控件,比如在 H5 页面中,再比如在 Flutter 页面中等等,这个时候我们可以借助保存确认弹窗来保存图片。当我们通过 photoAccessHelper 调用 showAssetsCreationDialog 时,系统会弹出一个确认弹窗,用户点击允许,则我们可以将图片保存到相册,若用户点击禁止,则不能保存图片到相册。使用这种方式,我们就不用强依赖于安全控件了,示例如下
import { photoAccessHelper } from '@kit.MediaLibraryKit'
import fs from '@ohos.file.fs'
@Entry
@ComponentV2
struct Index {
@Local isShowHome: boolean = false
build() {
Column() {
Button('保存图片到相册').onClick(()=>{
this.savePhotoToGallery().then(() => {
this.getUIContext().getPromptAction().showToast({ message: '保存成功' })
}).catch((err: Error) => {
this.getUIContext().getPromptAction().showToast({ message: err.message })
})
})
}
}
public async savePhotoToGallery(): Promise<void> {
const ctx = getContext()
const helper = photoAccessHelper.getPhotoAccessHelper(ctx)
const src = ctx.resourceDir + '/icon.png'
const desFileUris = await helper.showAssetsCreationDialog([src], [{
title: 'test',
fileNameExtension: 'png',
photoType: photoAccessHelper.PhotoType.IMAGE
}])
const desFile = fs.openSync(desFileUris[0], fs.OpenMode.WRITE_ONLY)
const srcFile = fs.openSync(src, fs.OpenMode.READ_ONLY)
fs.copyFileSync(srcFile.fd, desFile.fd)
fs.closeSync(srcFile)
fs.closeSync(desFile)
}
}
复制代码
以上示例请保存 icon.png 在本地真实存在。
评论