HarmonyOS 鸿蒙Next:利用SaveButton组件显示图片到相册的方法demo,支持组件截图、url网络图片、base64格式图片。注意事项:1、不支持自定义SaveButton样式。2、带Scroll时下载按钮遮挡部分无法保存。
HarmonyOS 鸿蒙Next:利用SaveButton组件显示图片到相册的方法demo,支持组件截图、url网络图片、base64格式图片。注意事项:1、不支持自定义SaveButton样式。2、带Scroll时下载按钮遮挡部分无法保存。
import photoAccessHelper from ‘@ohos.file.photoAccessHelper’;
import fs from ‘@ohos.file.fs’;
import { common } from ‘@kit.AbilityKit’;
import { componentSnapshot, promptAction } from ‘@kit.ArkUI’;
import { image } from ‘@kit.ImageKit’;
import { BusinessError } from ‘@kit.BasicServicesKit’;
import { http } from ‘@kit.NetworkKit’;
import { util } from ‘@kit.ArkTS’;
@Entry
@Component
struct Page09 {
inviteQrCodeID: string = “inviteQrCodeID”
@State imageUrl: string =
“https://img1.baidu.com/it/u=1268271089,1175168242&fm=253&fmt=auto&app=120&f=JPEG?w=506&h=500”
@State base64Str: string =
‘’
build() {
Scroll() {
Column({ space: 10 }) {
Column({ space: 10 }) {
Text(‘下载组件截图图片到相册’)
Column() {
Column() {
Text(<span class="hljs-string"><span class="hljs-string">'标题测试'</span></span>).fontSize(<span class="hljs-string"><span class="hljs-string">'36lpx'</span></span>).height(<span class="hljs-string"><span class="hljs-string">'120lpx'</span></span>).fontColor(<span class="hljs-string"><span class="hljs-string">"#2E2E2E"</span></span>)
QRCode(<span class="hljs-string"><span class="hljs-string">'https://www.huawei.com/'</span></span>)
.width(<span class="hljs-string"><span class="hljs-string">'300lpx'</span></span>)
.height(<span class="hljs-string"><span class="hljs-string">'300lpx'</span></span>)
.margin({ top: <span class="hljs-string"><span class="hljs-string">'25lpx'</span></span> })
.draggable(<span class="hljs-literal"><span class="hljs-literal">false</span></span>)
.width(<span class="hljs-number"><span class="hljs-number">140</span></span>)
.height(<span class="hljs-number"><span class="hljs-number">140</span></span>)
}.padding({ left: <span class="hljs-string"><span class="hljs-string">'42lpx'</span></span>, right: <span class="hljs-string"><span class="hljs-string">'42lpx'</span></span>, bottom: <span class="hljs-string"><span class="hljs-string">'20lpx'</span></span> })
Text(<span class="hljs-string"><span class="hljs-string">'点按下载保存至相册'</span></span>)
.textAlign(TextAlign.Center)
.padding({ left: <span class="hljs-string"><span class="hljs-string">'125lpx'</span></span>, right: <span class="hljs-string"><span class="hljs-string">'125lpx'</span></span> })
.fontColor(<span class="hljs-string"><span class="hljs-string">"#4B4B4B"</span></span>)
.fontSize(<span class="hljs-string"><span class="hljs-string">'32lpx'</span></span>)
.margin({ bottom: <span class="hljs-string"><span class="hljs-string">'70lpx'</span></span> })
}
.id(<span class="hljs-keyword"><span class="hljs-keyword">this</span></span>.inviteQrCodeID)
.padding(<span class="hljs-string"><span class="hljs-string">'20lpx'</span></span>)
.margin({ top: <span class="hljs-string"><span class="hljs-string">'30lpx'</span></span>, bottom: <span class="hljs-string"><span class="hljs-string">'30lpx'</span></span> })
.backgroundColor(Color.White)
.borderRadius(<span class="hljs-string"><span class="hljs-string">'30lpx'</span></span>)
.alignItems(HorizontalAlign.Center)
.justifyContent(FlexAlign.Center)
SaveButton().onClick(async (event: ClickEvent, result: SaveButtonOnClickResult) => {
<span class="hljs-keyword"><span class="hljs-keyword">if</span></span> (result === SaveButtonOnClickResult.SUCCESS) {
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> context: common.UIAbilityContext = getContext(<span class="hljs-keyword"><span class="hljs-keyword">this</span></span>) as common.UIAbilityContext;
<span class="hljs-comment"><span class="hljs-comment">// 免去权限申请和权限请求等环节,获得临时授权,保存对应图片</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> helper = photoAccessHelper.getPhotoAccessHelper(context);
<span class="hljs-keyword"><span class="hljs-keyword">try</span></span> {
<span class="hljs-comment"><span class="hljs-comment">// onClick触发后5秒内通过createAsset接口创建图片文件,5秒后createAsset权限收回。</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> uri = await helper.createAsset(photoAccessHelper.PhotoType.IMAGE, <span class="hljs-string"><span class="hljs-string">'jpg'</span></span>);
<span class="hljs-comment"><span class="hljs-comment">// 使用uri打开文件,可以持续写入内容,写入过程不受时间限制</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> file = await fs.open(uri, fs.OpenMode.READ_WRITE | fs.OpenMode.CREATE);
componentSnapshot.get(<span class="hljs-keyword"><span class="hljs-keyword">this</span></span>.inviteQrCodeID).then((pixelMap) => {
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> packOpts: image.PackingOption = { format: <span class="hljs-string"><span class="hljs-string">'image/png'</span></span>, quality: <span class="hljs-number"><span class="hljs-number">100</span></span> }
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> imagePacker: image.ImagePacker = image.createImagePacker();
<span class="hljs-keyword"><span class="hljs-keyword">return</span></span> imagePacker.packToFile(pixelMap, file.fd, packOpts).finally(() => {
imagePacker.release(); <span class="hljs-comment"><span class="hljs-comment">//释放</span></span>
fs.close(file.fd);
promptAction.showToast({
message: <span class="hljs-string"><span class="hljs-string">'图片已保存至相册'</span></span>,
duration: <span class="hljs-number"><span class="hljs-number">2000</span></span>
});
});
})
} <span class="hljs-keyword"><span class="hljs-keyword">catch</span></span> (error) {
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> err: BusinessError = error as BusinessError;
console.error(`Failed to save photo. Code is ${err.code}, message is ${err.message}`);
}
} <span class="hljs-keyword"><span class="hljs-keyword">else</span></span> {
promptAction.showToast({
message: <span class="hljs-string"><span class="hljs-string">'设置权限失败!'</span></span>,
duration: <span class="hljs-number"><span class="hljs-number">2000</span></span>
});
}
})
}.borderWidth(<span class="hljs-number"><span class="hljs-number">1</span></span>).borderStyle(BorderStyle.Dotted).backgroundColor(Color.Pink).padding(<span class="hljs-string"><span class="hljs-string">'50lpx'</span></span>)
Column({ space: <span class="hljs-number"><span class="hljs-number">10</span></span> }) {
Text(<span class="hljs-string"><span class="hljs-string">'下载网络图片到相册'</span></span>)
<span class="hljs-comment"><span class="hljs-comment">/**
* 需要在 src/main/module.json5
* 添加网络权限
* {
"module": {
"requestPermissions": [
{
"name": "ohos.permission.INTERNET"
},
],
*/</span></span>
Image(<span class="hljs-keyword"><span class="hljs-keyword">this</span></span>.imageUrl).width(<span class="hljs-string"><span class="hljs-string">'100lpx'</span></span>)
SaveButton().onClick(async (event: ClickEvent, result: SaveButtonOnClickResult) => {
<span class="hljs-keyword"><span class="hljs-keyword">if</span></span> (result === SaveButtonOnClickResult.SUCCESS) {
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> context: common.UIAbilityContext = getContext(<span class="hljs-keyword"><span class="hljs-keyword">this</span></span>) as common.UIAbilityContext;
<span class="hljs-comment"><span class="hljs-comment">// 免去权限申请和权限请求等环节,获得临时授权,保存对应图片</span></span>
<span class="hljs-comment"><span class="hljs-comment">// savePhotoToGallery(context);</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> helper = photoAccessHelper.getPhotoAccessHelper(context);
<span class="hljs-keyword"><span class="hljs-keyword">try</span></span> {
http.createHttp().request(
<span class="hljs-keyword"><span class="hljs-keyword">this</span></span>.imageUrl,
{ expectDataType: http.HttpDataType.ARRAY_BUFFER }
).then(async (res) => {
console.info(<span class="hljs-string"><span class="hljs-string">'res'</span></span>, <span class="hljs-built_in"><span class="hljs-built_in">JSON</span></span>.stringify(res))
<span class="hljs-comment"><span class="hljs-comment">// 将图片资源转为像素图(PixelMap)</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> pixelMap = await image.createImageSource(res.result as <span class="hljs-built_in"><span class="hljs-built_in">ArrayBuffer</span></span>).createPixelMap()
<span class="hljs-comment"><span class="hljs-comment">// onClick触发后5秒内通过createAsset接口创建图片文件,5秒后createAsset权限收回。</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> uri = await helper.createAsset(photoAccessHelper.PhotoType.IMAGE, <span class="hljs-string"><span class="hljs-string">'jpg'</span></span>);
<span class="hljs-comment"><span class="hljs-comment">// 使用uri打开文件,可以持续写入内容,写入过程不受时间限制</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> file = await fs.open(uri, fs.OpenMode.READ_WRITE | fs.OpenMode.CREATE);
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> packOpts: image.PackingOption = { format: <span class="hljs-string"><span class="hljs-string">'image/png'</span></span>, quality: <span class="hljs-number"><span class="hljs-number">100</span></span> }
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> imagePacker: image.ImagePacker = image.createImagePacker();
<span class="hljs-keyword"><span class="hljs-keyword">return</span></span> imagePacker.packToFile(pixelMap, file.fd, packOpts).finally(() => {
imagePacker.release(); <span class="hljs-comment"><span class="hljs-comment">//释放</span></span>
fs.close(file.fd);
promptAction.showToast({
message: <span class="hljs-string"><span class="hljs-string">'图片已保存至相册'</span></span>,
duration: <span class="hljs-number"><span class="hljs-number">2000</span></span>
});
});
}).catch(() => {
console.info(<span class="hljs-string"><span class="hljs-string">'catch'</span></span>)
})
} <span class="hljs-keyword"><span class="hljs-keyword">catch</span></span> (error) {
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> err: BusinessError = error as BusinessError;
console.error(`Failed to save photo. Code is ${err.code}, message is ${err.message}`);
}
} <span class="hljs-keyword"><span class="hljs-keyword">else</span></span> {
promptAction.showToast({
message: <span class="hljs-string"><span class="hljs-string">'设置权限失败!'</span></span>,
duration: <span class="hljs-number"><span class="hljs-number">2000</span></span>
});
}
})
}.borderWidth(<span class="hljs-number"><span class="hljs-number">1</span></span>).borderStyle(BorderStyle.Dotted).backgroundColor(Color.Pink).padding(<span class="hljs-string"><span class="hljs-string">'50lpx'</span></span>)
Column({ space: <span class="hljs-number"><span class="hljs-number">10</span></span> }) {
Text(<span class="hljs-string"><span class="hljs-string">'下载base64图片到相册'</span></span>)
Text(<span class="hljs-string"><span class="hljs-string">'注意1:有些base64的格式图片显示不出来,\n是因为前缀没加data:image/png;base64,'</span></span>).textAlign(TextAlign.Center)
Text(<span class="hljs-string"><span class="hljs-string">"注意2:下载到相册的base64字符串不能有'data:image/jpeg;base64,'这样的前缀。所以我这里用正则去掉了前缀"</span></span>).textAlign(TextAlign.Center)
SaveButton().onClick(async (event: ClickEvent, result: SaveButtonOnClickResult) => {
<span class="hljs-keyword"><span class="hljs-keyword">if</span></span> (result === SaveButtonOnClickResult.SUCCESS) {
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> context: common.UIAbilityContext = getContext(<span class="hljs-keyword"><span class="hljs-keyword">this</span></span>) as common.UIAbilityContext;
<span class="hljs-comment"><span class="hljs-comment">// 免去权限申请和权限请求等环节,获得临时授权,保存对应图片</span></span>
<span class="hljs-comment"><span class="hljs-comment">// savePhotoToGallery(context);</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> helper = photoAccessHelper.getPhotoAccessHelper(context);
<span class="hljs-keyword"><span class="hljs-keyword">try</span></span> {
<span class="hljs-comment"><span class="hljs-comment">// 正则表达式用于匹配 "data:image/*;base64," 这样的前缀</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> prefixRegex = <span class="hljs-regexp"><span class="hljs-regexp">/^data:image\/[a-zA-Z]+;base64,/</span></span>;
<span class="hljs-comment"><span class="hljs-comment">// 使用 replace 方法去除匹配到的前缀</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> base64String = <span class="hljs-keyword"><span class="hljs-keyword">this</span></span>.base64Str.replace(prefixRegex, <span class="hljs-string"><span class="hljs-string">''</span></span>)
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> buffer: <span class="hljs-built_in"><span class="hljs-built_in">ArrayBuffer</span></span> =
<span class="hljs-keyword"><span class="hljs-keyword">new</span></span> util.Base64Helper().decodeSync(base64String, util.Type.MIME).buffer as <span class="hljs-built_in"><span class="hljs-built_in">ArrayBuffer</span></span>;
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> imageSource = image.createImageSource(buffer);
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> pixelMap = await imageSource.createPixelMap({ editable: <span class="hljs-literal"><span class="hljs-literal">true</span></span> });
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> opts: image.PackingOption = { format: <span class="hljs-string"><span class="hljs-string">"image/jpeg"</span></span>, quality: <span class="hljs-number"><span class="hljs-number">100</span></span> };
<span class="hljs-comment"><span class="hljs-comment">// onClick触发后5秒内通过createAsset接口创建图片文件,5秒后createAsset权限收回。</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> uri = await helper.createAsset(photoAccessHelper.PhotoType.IMAGE, <span class="hljs-string"><span class="hljs-string">'jpg'</span></span>);
<span class="hljs-comment"><span class="hljs-comment">// 使用uri打开文件,可以持续写入内容,写入过程不受时间限制</span></span>
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> file = await fs.open(uri, fs.OpenMode.READ_WRITE | fs.OpenMode.CREATE);
<span class="hljs-keyword"><span class="hljs-keyword">let</span></span> packOpts: image.PackingOption = { format: <span class="hljs-string"><span class="hljs-string">'image/png'</span></span>, quality: <span class="hljs-number"><span class="hljs-number">100</span></span> }
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> imagePacker: image.ImagePacker = image.createImagePacker();
<span class="hljs-keyword"><span class="hljs-keyword">return</span></span> imagePacker.packToFile(pixelMap, file.fd, packOpts).finally(() => {
imagePacker.release(); <span class="hljs-comment"><span class="hljs-comment">//释放</span></span>
fs.close(file.fd);
promptAction.showToast({
message: <span class="hljs-string"><span class="hljs-string">'图片已保存至相册'</span></span>,
duration: <span class="hljs-number"><span class="hljs-number">2000</span></span>
});
});
} <span class="hljs-keyword"><span class="hljs-keyword">catch</span></span> (error) {
<span class="hljs-keyword"><span class="hljs-keyword">const</span></span> err: BusinessError = error as BusinessError;
console.error(`Failed to save photo. Code is ${err.code}, message is ${err.message}`);
}
} <span class="hljs-keyword"><span class="hljs-keyword">else</span></span> {
console.info(`result:${<span class="hljs-built_in"><span class="hljs-built_in">JSON</span></span>.stringify(result)}`)
promptAction.showToast({
message: <span class="hljs-string"><span class="hljs-string">'设置权限失败!'</span></span>,
duration: <span class="hljs-number"><span class="hljs-number">2000</span></span>
});
}
})
}
.borderWidth(<span class="hljs-number"><span class="hljs-number">1</span></span>).borderStyle(BorderStyle.Dotted).backgroundColor(Color.Pink).padding(<span class="hljs-string"><span class="hljs-string">'50lpx'</span></span>)
}.width(<span class="hljs-string"><span class="hljs-string">'100%'</span></span>)
}.width(<span class="hljs-string"><span class="hljs-string">'100%'</span></span>)
}
}
<button style="position: absolute; padding: 4px 8px 0px; cursor: pointer; top: 8px; right: 8px; font-size: 14px;">复制</button>
注意事项:
1、样式不支持自定义图标,试过用opacity修改透明度,然后添加背景来实现自定义样式,结果也失败了,opacity(1)点击生效,opacity(0.9)后点击按钮就不生效了。
参考:https://developer.huawei.com/consumer/cn/doc/harmonyos-faqs-V5/faqs-arkui-301-V5
2、如果带Scroll时,下载按钮被遮挡一部分,也无法保存到相册。
比如下面这样遮挡一点点下载按钮后,点击就没办法保存到相册了。
【参考方案】
1、官方文档:
2、PixelMap和base64的相互转换
参考文档:https://developer.huawei.com/consumer/cn/doc/harmonyos-faqs-V5/faqs-image-15-V5
关于HarmonyOS 鸿蒙Next:利用SaveButton组件显示图片到相册的方法demo,支持组件截图、url网络图片、base64格式图片。注意事项:1、不支持自定义SaveButton样式。2、带Scroll时下载按钮遮挡部分无法保存。的问题,您也可以访问:https://www.itying.com/category-93-b0.html 联系官网客服。
更多关于HarmonyOS 鸿蒙Next:利用SaveButton组件显示图片到相册的方法demo,支持组件截图、url网络图片、base64格式图片。注意事项:1、不支持自定义SaveButton样式。2、带Scroll时下载按钮遮挡部分无法保存。的实战系列教程也可以访问 https://www.itying.com/category-93-b0.html
有要学HarmonyOS AI的同学吗,联系我:https://www.itying.com/goods-1206.html
更多关于HarmonyOS 鸿蒙Next:利用SaveButton组件显示图片到相册的方法demo,支持组件截图、url网络图片、base64格式图片。注意事项:1、不支持自定义SaveButton样式。2、带Scroll时下载按钮遮挡部分无法保存。的实战系列教程也可以访问 https://www.itying.com/category-93-b0.html
用我的例子有问题吗?如果没问题, 那可能的原因有, 你修改了SaveButton的透明度比如SaveButton.opacity(0.9), 你在SaveButton上面加了遮罩,按钮被阻挡了一部分。