1/* 2 * Copyright (c) 2022-2023 Huawei Device Co., Ltd. 3 * Licensed under the Apache License, Version 2.0 (the "License"); 4 * you may not use this file except in compliance with the License. 5 * You may obtain a copy of the License at 6 * 7 * http://www.apache.org/licenses/LICENSE-2.0 8 * 9 * Unless required by applicable law or agreed to in writing, software 10 * distributed under the License is distributed on an "AS IS" BASIS, 11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 * See the License for the specific language governing permissions and 13 * limitations under the License. 14 */ 15 16import { 17 BigDataConstants, 18 BrowserDataFactory, 19 Log, 20 MediaItem, 21 ReportToBigDataUtil, 22 UserFileManagerAccess 23} from '@ohos/common'; 24import { ImageFilterStack } from './ImageFilterStack'; 25import image from '@ohos.multimedia.image'; 26import fileIO from '@ohos.fileio'; 27import { DateUtil } from './utils/DateUtil'; 28import { Loader } from './Loader'; 29import userFileManager from '@ohos.filemanagement.userFileManager'; 30import type { Album, FileType } from '@ohos/common/src/main/ets/default/access/UserFileManagerAccess'; 31 32const TAG: string = 'editor_Save'; 33 34export class Save { 35 private static readonly QUALITY_95: number = 95; 36 private static readonly PERCENT_100: number = 100; 37 38 constructor() { 39 } 40 41 public static async save(item: MediaItem, albumUri: string, optStack: ImageFilterStack, isReplace: Boolean, 42 callback: Function): Promise<void> { 43 Log.info(TAG, `${JSON.stringify(item)} ${isReplace}`); 44 45 // Get quality of image first, making sure it's succeed even in replace mode 46 let finalQuality: number = await this.getImageQuality(item); 47 Log.debug(TAG, "save: final quality = " + finalQuality); 48 let options = { 49 format: 'image/jpeg', 50 quality: finalQuality 51 }; 52 53 try { 54 let wrapper = await Loader.loadPixelMapWrapper(item); 55 wrapper = optStack.apply(wrapper); 56 Log.debug(TAG, 'Edit and restore operation execution end.'); 57 58 let fileAsset = await this.createFileAsset(item.uri, albumUri, isReplace); 59 Log.info(TAG, `create fileAsset succeed: [${fileAsset}]`); 60 let fd = await UserFileManagerAccess.getInstance().openAsset('RW', fileAsset); 61 if (fd < 0) { 62 Log.error(TAG, 'open asset failed.'); 63 return; 64 } 65 66 let packer = image.createImagePacker(); 67 let buffer = await packer.packing(wrapper.pixelMap, options); 68 Log.info(TAG, 'Format pixelMap data to jpg data end.'); 69 70 await fileIO.write(fd, buffer); 71 Log.info(TAG, 'write jpg file end.'); 72 try { 73 let imageSourceApi: image.ImageSource = image.createImageSource(fd); 74 await new Promise(async (resolve, reject) => { 75 imageSourceApi.modifyImageProperty('CompressedBitsPerPixel', String(finalQuality / Save.PERCENT_100)); 76 resolve(Boolean(true)); 77 }); 78 } catch (e) { 79 Log.error(TAG, 'sth wrong when set CompressedBitsPerPixel value is ' + (finalQuality / Save.PERCENT_100) + 80 ', error is ' + JSON.stringify(e)); 81 } 82 let newUri = fileAsset.uri; 83 Log.info(TAG, `New saved file: ${newUri}`); 84 callback && callback(newUri); 85 await UserFileManagerAccess.getInstance().closeAsset(fd, fileAsset); 86 wrapper && wrapper.release(); 87 await packer.release(); 88 89 } catch (e) { 90 Log.error(TAG, `save catch error: ${JSON.stringify(e)}`); 91 let msg = { 92 'CatchError': JSON.stringify(e) 93 } 94 ReportToBigDataUtil.errEventReport(BigDataConstants.PHOTO_EDIT_SAVE_ERROR_ID, msg); 95 callback && callback(undefined); 96 } 97 } 98 99 private static async createFileAsset(uri: string, albumUri: string, isReplace: Boolean) { 100 let dataImpl = BrowserDataFactory.getFeature(BrowserDataFactory.TYPE_PHOTO); 101 let fileAsset = await dataImpl.getDataByUri(uri); 102 103 if (!fileAsset) { 104 Log.error(TAG, 'get file error'); 105 return null; 106 } 107 let title = DateUtil.nameByDate(isReplace, fileAsset.displayName); 108 if (null == title) { 109 Log.error(TAG, `create picture name failed.`); 110 return null; 111 } 112 let displayName = title + '.jpg'; 113 Log.debug(TAG, `file displayname = ${displayName}`); 114 let favorite: boolean = false; 115 if (isReplace) { 116 favorite = Boolean(await fileAsset.get(userFileManager.ImageVideoKey.FAVORITE.toString())); 117 await UserFileManagerAccess.getInstance().deleteToTrash(uri); 118 Log.debug(TAG, `trash picture file uri ${uri} end.`); 119 } 120 fileAsset = await UserFileManagerAccess.getInstance().createAsset(fileAsset.mediaType, displayName); 121 await fileAsset.favorite(favorite); 122 let album: Album = await UserFileManagerAccess.getInstance().getAlbumByUri(albumUri); 123 let isSystemAlbum = UserFileManagerAccess.getInstance().isSystemAlbum(album); 124 if (!isSystemAlbum) { 125 await UserFileManagerAccess.getInstance().addFileToAlbum(albumUri, fileAsset); 126 } 127 return fileAsset; 128 } 129 130 private static async getImageQuality(item: MediaItem): Promise<number> { 131 let quality = undefined; 132 try { 133 quality = await Loader.getCompressedBitsPerPixel(item); 134 } catch (e) { 135 Log.error(TAG, "sth wrong with getCompressedBitsPerPixel" + e); 136 } 137 let finalQuality: number = quality !== undefined ? quality * Save.PERCENT_100 : Save.QUALITY_95 138 Log.debug(TAG, "save: final quality = " + finalQuality); 139 return finalQuality; 140 } 141}