• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 }