1 /*
2 * Copyright (C) 2024 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
16 #include "core/common/ai/ai_write_adapter.h"
17
18 #include "array_wrapper.h"
19 #include "bool_wrapper.h"
20 #include "int_wrapper.h"
21 #include "want.h"
22 #include "core/components_ng/pattern/text/span/span_string.h"
23
24 namespace OHOS::Ace {
25 const std::pair<std::string, std::string> UI_ENTENSION_TYPE = {"ability.want.params.uiExtensionType", "sys/commonUI"};
26 const std::u16string BOUNDARY_SYMBOLS = u",.?,。?!";
27 const std::string API_VERSION = "apiVersion";
28 const std::string RESULT_BUFFER = "resultBuffer";
29 const std::string SHEET_DISMISS = "sheetDismiss";
30 const std::string PROCESS_ID = "processId";
31 const std::string MAX_CONTENT_LENGTH = "maxContentLength";
32 const std::string FIRST_HANDLE_RECT = "firstHandleRect";
33 const std::string SECOND_HANDLE_RECT = "secondHandleRect";
34 const std::string IS_AI_SUPPORT_METADATA = "isAiSupport";
35 const std::string SELECT_CONTENT_LENGTH = "selectContentLength";
36 const std::string REQUEST_LONG_CONTENT = "requestLongContent";
37 const std::string LONG_SENTENCE_BUFFER = "longSentenceBuffer";
38 const std::string LONG_SELECT_START = "longSelectStart";
39 const std::string LONG_SELECT_END = "longSelectEnd";
40 const std::string KEY_PACKAGE_NAME = "keyPackageApp";
41 const std::string START_COMPONONT_TYPE = "startComponentType";
42
IsSentenceBoundary(const char16_t value)43 bool AIWriteAdapter::IsSentenceBoundary(const char16_t value)
44 {
45 for (char16_t item: BOUNDARY_SYMBOLS) {
46 if (value == item) {
47 return true;
48 }
49 }
50 return false;
51 }
52
GetSelectLengthOnlyText(const std::u16string & content)53 uint32_t AIWriteAdapter::GetSelectLengthOnlyText(const std::u16string& content)
54 {
55 uint32_t length = 0;
56 for (uint32_t i = 0; i < content.length(); i++) {
57 if (content[i] != u' ' && content[i] != u'\n') {
58 length++;
59 }
60 }
61 return length;
62 }
63
CloseModalUIExtension()64 void AIWriteAdapter::CloseModalUIExtension()
65 {
66 TAG_LOGI(AceLogTag::ACE_RICH_TEXT, "CloseModalUIExtension.");
67 auto context = pipelineContext_.Upgrade();
68 CHECK_NULL_VOID(context);
69 auto overlayManager = context->GetOverlayManager();
70 CHECK_NULL_VOID(overlayManager);
71 overlayManager->CloseModalUIExtension(sessionId_);
72 SetAIWrite(false);
73 }
74
ShowModalUIExtension(const AIWriteInfo & info,std::function<void (std::vector<uint8_t> &)> resultCallback)75 void AIWriteAdapter::ShowModalUIExtension(const AIWriteInfo& info,
76 std::function<void(std::vector<uint8_t>&)> resultCallback)
77 {
78 TAG_LOGI(AceLogTag::ACE_RICH_TEXT, "ShowModalUIExtension.");
79 AAFwk::Want want;
80 SetWantParams(info, want);
81 Ace::ModalUIExtensionCallbacks callbacks;
82 BindModalUIExtensionCallback(callbacks, resultCallback);
83
84 auto context = pipelineContext_.Upgrade();
85 CHECK_NULL_VOID(context);
86 auto overlayManager = context->GetOverlayManager();
87 CHECK_NULL_VOID(overlayManager);
88 Ace::ModalUIExtensionConfig config;
89 sessionId_ = overlayManager->CreateModalUIExtension(want, callbacks, config);
90 aiWriteInfo_ = info;
91 }
92
SetWantParams(const AIWriteInfo & info,AAFwk::Want & want)93 void AIWriteAdapter::SetWantParams(const AIWriteInfo& info, AAFwk::Want& want)
94 {
95 auto apiVersion = AceApplicationInfo::GetInstance().GetApiTargetVersion();
96 want.SetElementName(bundleName_, abilityName_);
97 want.SetParam(UI_ENTENSION_TYPE.first, UI_ENTENSION_TYPE.second);
98 want.SetParam(API_VERSION, static_cast<int>(apiVersion));
99 want.SetParam(PROCESS_ID, getpid());
100 want.SetParam(MAX_CONTENT_LENGTH, info.maxLength);
101 want.SetParam(SELECT_CONTENT_LENGTH, info.selectLength);
102 want.SetParam(FIRST_HANDLE_RECT, info.firstHandle);
103 want.SetParam(SECOND_HANDLE_RECT, info.secondHandle);
104 want.SetParam(KEY_PACKAGE_NAME, AceApplicationInfo::GetInstance().GetPackageName());
105 want.SetParam(START_COMPONONT_TYPE, info.componentType);
106 }
107
BindModalUIExtensionCallback(Ace::ModalUIExtensionCallbacks & callbacks,std::function<void (std::vector<uint8_t> &)> resultCallback)108 void AIWriteAdapter::BindModalUIExtensionCallback(
109 Ace::ModalUIExtensionCallbacks& callbacks, std::function<void(std::vector<uint8_t>&)> resultCallback)
110 {
111 callbacks.onResult = [](int32_t code, const AAFwk::Want& want) {
112 TAG_LOGD(AceLogTag::ACE_UIEXTENSIONCOMPONENT, "UIExtension onResult, code: %{public}d", code);
113 };
114 callbacks.onDestroy = [weak = WeakClaim(this)]() {
115 TAG_LOGD(AceLogTag::ACE_UIEXTENSIONCOMPONENT, "UIExtension onDestroy.");
116 auto aiWriteAdapter = weak.Upgrade();
117 CHECK_NULL_VOID(aiWriteAdapter);
118 aiWriteAdapter->CloseModalUIExtension();
119 };
120 callbacks.onError = [weak = WeakClaim(this)](int32_t code, const std::string& name, const std::string& message) {
121 TAG_LOGE(AceLogTag::ACE_UIEXTENSIONCOMPONENT,
122 "UIExtension onError, code: %{public}d, name: %{public}s, message: %{public}s",
123 code, name.c_str(), message.c_str());
124 auto aiWriteAdapter = weak.Upgrade();
125 CHECK_NULL_VOID(aiWriteAdapter);
126 aiWriteAdapter->CloseModalUIExtension();
127 };
128 callbacks.onRelease = [](int32_t code) {
129 TAG_LOGD(AceLogTag::ACE_UIEXTENSIONCOMPONENT, "UIExtension onRelease, code: %{public}d", code);
130 };
131 callbacks.onRemoteReady = [weak = WeakClaim(this)](const std::shared_ptr<Ace::ModalUIExtensionProxy>& proxy) {
132 TAG_LOGD(AceLogTag::ACE_UIEXTENSIONCOMPONENT, "UIExtension onRemoteReady.");
133 auto aiWriteAdapter = weak.Upgrade();
134 CHECK_NULL_VOID(aiWriteAdapter);
135 aiWriteAdapter->SetModalUIExtensionProxy(proxy);
136 };
137 callbacks.onReceive = [weak = WeakClaim(this), cb = std::move(resultCallback)]
138 (const AAFwk::WantParams& wantParams) {
139 auto aiWriteAdapter = weak.Upgrade();
140 CHECK_NULL_VOID(aiWriteAdapter);
141 auto isSheetClose = false;
142 auto isRequest = false;
143 auto result = aiWriteAdapter->GetBufferParam(RESULT_BUFFER, wantParams);
144 isSheetClose = aiWriteAdapter->GetBoolParam(SHEET_DISMISS, wantParams);
145 isRequest = aiWriteAdapter->GetBoolParam(REQUEST_LONG_CONTENT, wantParams);
146 if (isRequest) {
147 aiWriteAdapter->SendData();
148 return;
149 }
150 if (cb && !result.empty()) {
151 cb(result);
152 return;
153 }
154 if (isSheetClose) {
155 aiWriteAdapter->CloseModalUIExtension();
156 }
157 };
158 }
159
SendData()160 void AIWriteAdapter::SendData()
161 {
162 auto proxy = GetModalUIExtensionProxy();
163 AAFwk::WantParams wantParams;
164 SetArrayParam(wantParams, LONG_SENTENCE_BUFFER, aiWriteInfo_.sentenceBuffer);
165 wantParams.SetParam(LONG_SELECT_START, AAFwk::Integer::Box(aiWriteInfo_.start));
166 wantParams.SetParam(LONG_SELECT_END, AAFwk::Integer::Box(aiWriteInfo_.end));
167 proxy->SendData(wantParams);
168 }
169
SetArrayParam(AAFwk::WantParams & wantParams,const std::string & key,const std::vector<uint8_t> & value)170 void AIWriteAdapter::SetArrayParam(
171 AAFwk::WantParams& wantParams, const std::string& key, const std::vector<uint8_t>& value)
172 {
173 std::vector<int> valueVec(value.size());
174 std::transform(value.begin(), value.end(), valueVec.begin(),
175 [](uint8_t x) { return static_cast<int>(x); });
176 size_t size = valueVec.size();
177 sptr<AAFwk::IArray> ao = new (std::nothrow) AAFwk::Array(size, AAFwk::g_IID_IInteger);
178 if (ao == nullptr) {
179 return;
180 }
181 for (size_t i = 0; i < size; i++) {
182 ao->Set(i, AAFwk::Integer::Box(value[i]));
183 }
184 wantParams.SetParam(key, ao);
185 }
186
GetBufferParam(const std::string & key,const AAFwk::WantParams & wantParams)187 std::vector<uint8_t> AIWriteAdapter::GetBufferParam(const std::string& key, const AAFwk::WantParams& wantParams)
188 {
189 std::vector<uint8_t> array;
190 auto value = wantParams.GetParam(key);
191 AAFwk::IArray *ao = AAFwk::IArray::Query(value);
192 if (ao != nullptr && AAFwk::Array::IsIntegerArray(ao)) {
193 auto func = [&](AAFwk::IInterface *object) {
194 CHECK_NULL_VOID(object);
195 AAFwk::IInteger *value = AAFwk::IInteger::Query(object);
196 CHECK_NULL_VOID(value);
197 array.emplace_back(AAFwk::Integer::Unbox(value));
198 };
199 AAFwk::Array::ForEach(ao, func);
200 }
201 return array;
202 }
203
GetBoolParam(const std::string & key,const AAFwk::WantParams & wantParams)204 bool AIWriteAdapter::GetBoolParam(const std::string& key, const AAFwk::WantParams& wantParams)
205 {
206 auto value = wantParams.GetParam(key);
207 AAFwk::IBoolean *bo = AAFwk::IBoolean::Query(value);
208 if (bo != nullptr) {
209 return AAFwk::Boolean::Unbox(bo);
210 }
211 return false;
212 }
213 }