/* * Copyright (c) 2022 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "js_keyboard_delegate_setting.h" #include "event_checker.h" #include "input_method_ability.h" #include "inputmethod_trace.h" #include "js_callback_handler.h" #include "js_keyboard_controller_engine.h" #include "js_text_input_client_engine.h" #include "js_util.h" #include "js_utils.h" #include "key_event_napi.h" #include "napi/native_api.h" #include "napi/native_node_api.h" namespace OHOS { namespace MiscServices { constexpr size_t ARGC_ONE = 1; constexpr size_t ARGC_TWO = 2; const std::string JsKeyboardDelegateSetting::KDS_CLASS_NAME = "KeyboardDelegate"; thread_local napi_ref JsKeyboardDelegateSetting::KDSRef_ = nullptr; std::mutex JsKeyboardDelegateSetting::keyboardMutex_; std::shared_ptr JsKeyboardDelegateSetting::keyboardDelegate_{ nullptr }; std::mutex JsKeyboardDelegateSetting::eventHandlerMutex_; std::shared_ptr JsKeyboardDelegateSetting::handler_{ nullptr }; napi_value JsKeyboardDelegateSetting::Init(napi_env env, napi_value exports) { napi_property_descriptor descriptor[] = { DECLARE_NAPI_PROPERTY("OPTION_ASCII", GetJsConstProperty(env, static_cast(20))), DECLARE_NAPI_PROPERTY("OPTION_NONE", GetJsConstProperty(env, static_cast(0))), DECLARE_NAPI_PROPERTY("OPTION_AUTO_CAP_CHARACTERS", GetJsConstProperty(env, static_cast(2))), DECLARE_NAPI_PROPERTY("OPTION_AUTO_CAP_SENTENCES", GetJsConstProperty(env, static_cast(8))), DECLARE_NAPI_PROPERTY("OPTION_AUTO_WORDS", GetJsConstProperty(env, static_cast(4))), DECLARE_NAPI_PROPERTY("OPTION_MULTI_LINE", GetJsConstProperty(env, static_cast(1))), DECLARE_NAPI_PROPERTY("OPTION_NO_FULLSCREEN", GetJsConstProperty(env, static_cast(10))), DECLARE_NAPI_PROPERTY("CURSOR_UP", GetJsConstProperty(env, static_cast(1))), DECLARE_NAPI_PROPERTY("CURSOR_DOWN", GetJsConstProperty(env, static_cast(2))), DECLARE_NAPI_PROPERTY("CURSOR_LEFT", GetJsConstProperty(env, static_cast(3))), DECLARE_NAPI_PROPERTY("CURSOR_RIGHT", GetJsConstProperty(env, static_cast(4))), DECLARE_NAPI_PROPERTY("FLAG_SELECTING", GetJsConstProperty(env, static_cast(2))), DECLARE_NAPI_PROPERTY("FLAG_SINGLE_LINE", GetJsConstProperty(env, static_cast(1))), DECLARE_NAPI_PROPERTY("DISPLAY_MODE_PART", GetJsConstProperty(env, static_cast(0))), DECLARE_NAPI_PROPERTY("DISPLAY_MODE_FULL", GetJsConstProperty(env, static_cast(1))), DECLARE_NAPI_PROPERTY("WINDOW_TYPE_INPUT_METHOD_FLOAT", GetJsConstProperty(env, static_cast(2105))), DECLARE_NAPI_FUNCTION("createKeyboardDelegate", CreateKeyboardDelegate), DECLARE_NAPI_FUNCTION("getKeyboardDelegate", GetKeyboardDelegate), }; NAPI_CALL(env, napi_define_properties(env, exports, sizeof(descriptor) / sizeof(napi_property_descriptor), descriptor)); napi_property_descriptor properties[] = { DECLARE_NAPI_FUNCTION("on", Subscribe), DECLARE_NAPI_FUNCTION("off", UnSubscribe), }; napi_value cons = nullptr; NAPI_CALL(env, napi_define_class(env, KDS_CLASS_NAME.c_str(), KDS_CLASS_NAME.size(), JsConstructor, nullptr, sizeof(properties) / sizeof(napi_property_descriptor), properties, &cons)); NAPI_CALL(env, napi_create_reference(env, cons, 1, &KDSRef_)); NAPI_CALL(env, napi_set_named_property(env, exports, KDS_CLASS_NAME.c_str(), cons)); return exports; }; napi_value JsKeyboardDelegateSetting::GetJsConstProperty(napi_env env, uint32_t num) { napi_value jsNumber = nullptr; napi_create_int32(env, num, &jsNumber); return jsNumber; }; std::shared_ptr JsKeyboardDelegateSetting::GetKeyboardDelegateSetting() { if (keyboardDelegate_ == nullptr) { std::lock_guard lock(keyboardMutex_); if (keyboardDelegate_ == nullptr) { auto delegate = std::make_shared(); if (delegate == nullptr) { IMSA_HILOGE("keyboard delegate is nullptr!"); return nullptr; } keyboardDelegate_ = delegate; } } return keyboardDelegate_; } bool JsKeyboardDelegateSetting::InitKeyboardDelegate() { if (!InputMethodAbility::GetInstance().IsCurrentIme()) { return false; } auto delegate = GetKeyboardDelegateSetting(); if (delegate == nullptr) { return false; } InputMethodAbility::GetInstance().SetKdListener(delegate); { std::lock_guard lock(eventHandlerMutex_); handler_ = AppExecFwk::EventHandler::Current(); } return true; } napi_value JsKeyboardDelegateSetting::JsConstructor(napi_env env, napi_callback_info cbinfo) { napi_value thisVar = nullptr; NAPI_CALL(env, napi_get_cb_info(env, cbinfo, nullptr, nullptr, &thisVar, nullptr)); auto delegate = GetKeyboardDelegateSetting(); if (delegate == nullptr || !InitKeyboardDelegate()) { IMSA_HILOGE("failed to get delegate!"); napi_value result = nullptr; napi_get_null(env, &result); return result; } napi_status status = napi_wrap( env, thisVar, delegate.get(), [](napi_env env, void *nativeObject, void *hint) {}, nullptr, nullptr); if (status != napi_ok) { IMSA_HILOGE("failed to wrap: %{public}d!", status); return nullptr; } return thisVar; }; napi_value JsKeyboardDelegateSetting::CreateKeyboardDelegate(napi_env env, napi_callback_info info) { return GetKDInstance(env, info); } napi_value JsKeyboardDelegateSetting::GetKeyboardDelegate(napi_env env, napi_callback_info info) { return GetKDInstance(env, info); } napi_value JsKeyboardDelegateSetting::GetKDInstance(napi_env env, napi_callback_info info) { napi_value instance = nullptr; napi_value cons = nullptr; if (napi_get_reference_value(env, KDSRef_, &cons) != napi_ok) { IMSA_HILOGE("failed to get reference value!"); return nullptr; } if (napi_new_instance(env, cons, 0, nullptr, &instance) != napi_ok) { IMSA_HILOGE("failed to new instance!"); return nullptr; } return instance; } void JsKeyboardDelegateSetting::RegisterListener(napi_value callback, std::string type, std::shared_ptr callbackObj) { IMSA_HILOGD("RegisterListener %{public}s", type.c_str()); std::lock_guard lock(mutex_); if (jsCbMap_.empty() || jsCbMap_.find(type) == jsCbMap_.end()) { IMSA_HILOGD("methodName %{public}s is not registered!", type.c_str()); } auto callbacks = jsCbMap_[type]; bool ret = std::any_of(callbacks.begin(), callbacks.end(), [&callback](std::shared_ptr cb) { if (cb == nullptr) { return false; } return JsUtils::Equals(cb->env_, callback, cb->callback_, cb->threadId_); }); if (ret) { IMSA_HILOGD("JsKeyboardDelegateSetting callback already registered!"); return; } IMSA_HILOGI("add %{public}s callbackObj into jsCbMap_.", type.c_str()); jsCbMap_[type].push_back(std::move(callbackObj)); } void JsKeyboardDelegateSetting::UnRegisterListener(napi_value callback, std::string type) { IMSA_HILOGI("unregister listener: %{public}s.", type.c_str()); std::lock_guard lock(mutex_); if (jsCbMap_.empty() || jsCbMap_.find(type) == jsCbMap_.end()) { IMSA_HILOGE("methodName %{public}s is not unregistered!", type.c_str()); return; } if (callback == nullptr) { jsCbMap_.erase(type); IMSA_HILOGE("callback is nullptr!"); return; } for (auto item = jsCbMap_[type].begin(); item != jsCbMap_[type].end(); item++) { if ((callback != nullptr) && (JsUtils::Equals((*item)->env_, callback, (*item)->callback_, (*item)->threadId_))) { jsCbMap_[type].erase(item); break; } } if (jsCbMap_[type].empty()) { jsCbMap_.erase(type); } } napi_value JsKeyboardDelegateSetting::Subscribe(napi_env env, napi_callback_info info) { size_t argc = ARGC_TWO; napi_value argv[ARGC_TWO] = { nullptr }; napi_value thisVar = nullptr; void *data = nullptr; NAPI_CALL(env, napi_get_cb_info(env, info, &argc, argv, &thisVar, &data)); std::string type; // 2 means least param num. if (argc < 2 || !JsUtil::GetValue(env, argv[0], type) || !EventChecker::IsValidEventType(EventSubscribeModule::KEYBOARD_DELEGATE, type) || JsUtil::GetType(env, argv[1]) != napi_function) { IMSA_HILOGE("subscribe failed, type: %{public}s!", type.c_str()); return nullptr; } IMSA_HILOGD("subscribe type: %{public}s.", type.c_str()); auto engine = reinterpret_cast(JsUtils::GetNativeSelf(env, info)); if (engine == nullptr) { return nullptr; } std::shared_ptr callback = std::make_shared(env, argv[1], std::this_thread::get_id(), AppExecFwk::EventHandler::Current()); engine->RegisterListener(argv[ARGC_ONE], type, callback); napi_value result = nullptr; napi_get_null(env, &result); return result; } napi_value JsKeyboardDelegateSetting::UnSubscribe(napi_env env, napi_callback_info info) { size_t argc = ARGC_TWO; napi_value argv[ARGC_TWO] = { nullptr }; napi_value thisVar = nullptr; void *data = nullptr; NAPI_CALL(env, napi_get_cb_info(env, info, &argc, argv, &thisVar, &data)); std::string type; // 1 means least param num. if (argc < 1 || !JsUtil::GetValue(env, argv[0], type) || !EventChecker::IsValidEventType(EventSubscribeModule::KEYBOARD_DELEGATE, type)) { IMSA_HILOGE("unsubscribe failed, type: %{public}s!", type.c_str()); return nullptr; } // if the second param is not napi_function/napi_null/napi_undefined, return auto paramType = JsUtil::GetType(env, argv[1]); if (paramType != napi_function && paramType != napi_null && paramType != napi_undefined) { return nullptr; } // if the second param is napi_function, delete it, else delete all argv[1] = paramType == napi_function ? argv[1] : nullptr; IMSA_HILOGD("unsubscribe type: %{public}s.", type.c_str()); auto delegate = reinterpret_cast(JsUtils::GetNativeSelf(env, info)); if (delegate == nullptr) { return nullptr; } delegate->UnRegisterListener(argv[ARGC_ONE], type); napi_value result = nullptr; napi_get_null(env, &result); return result; } napi_value JsKeyboardDelegateSetting::GetResultOnKeyEvent(napi_env env, int32_t keyCode, int32_t keyStatus) { napi_value KeyboardDelegate = nullptr; NAPI_CALL(env, napi_create_object(env, &KeyboardDelegate)); napi_value jsKeyCode = nullptr; NAPI_CALL(env, napi_create_int32(env, keyCode, &jsKeyCode)); NAPI_CALL(env, napi_set_named_property(env, KeyboardDelegate, "keyCode", jsKeyCode)); napi_value jsKeyAction = nullptr; NAPI_CALL(env, napi_create_int32(env, keyStatus, &jsKeyAction)); NAPI_CALL(env, napi_set_named_property(env, KeyboardDelegate, "keyAction", jsKeyAction)); return KeyboardDelegate; } bool JsKeyboardDelegateSetting::OnDealKeyEvent( const std::shared_ptr &keyEvent, uint64_t cbId, const sptr &channelObject) { if (keyEvent == nullptr) { IMSA_HILOGE("keyEvent is nullptr"); return false; } auto eventHandler = GetEventHandler(); if (eventHandler == nullptr) { IMSA_HILOGE("eventHandler is nullptr!"); return false; } auto keyEventEntry = GetEntry("keyEvent", [keyEvent](UvEntry &entry) { entry.fullKeyEventPara = keyEvent; }); KeyEventPara para{ keyEvent->GetKeyCode(), keyEvent->GetKeyAction(), false }; std::string type = (keyEvent->GetKeyAction() == ARGC_TWO ? "keyDown" : "keyUp"); auto keyCodeEntry = GetEntry(type, [¶](UvEntry &entry) { entry.keyEventPara = { para.keyCode, para.keyStatus, para.isOnKeyEvent }; }); if (keyEventEntry == nullptr && keyCodeEntry == nullptr) { IMSA_HILOGW("key event callback is not registered."); return false; } IMSA_HILOGD("run in."); auto task = [keyEvent, keyEventEntry, keyCodeEntry, cbId, channelObject]() { DealKeyEvent(keyEvent, keyEventEntry, keyCodeEntry, cbId, channelObject); }; eventHandler->PostTask(task, "OnDealKeyEvent", 0, AppExecFwk::EventQueue::Priority::VIP); return true; } void JsKeyboardDelegateSetting::DealKeyEvent(const std::shared_ptr &keyEvent, const std::shared_ptr &keyEventEntry, const std::shared_ptr &keyCodeEntry, uint64_t cbId, const sptr &channelObject) { bool isKeyEventConsumed = false; bool isKeyCodeConsumed = false; if (keyEventEntry != nullptr) { auto getKeyEventProperty = [keyEventEntry](napi_env env, napi_value *args, uint8_t argc) -> bool { if (argc == 0) { return false; } napi_value keyEventObject{}; auto result = napi_create_object(env, &keyEventObject); CHECK_RETURN((result == napi_ok) && (keyEventObject != nullptr), "create object", false); result = MMI::KeyEventNapi::CreateKeyEvent(env, keyEventEntry->fullKeyEventPara, keyEventObject); CHECK_RETURN((result == napi_ok) && (keyEventObject != nullptr), "create key event object", false); // 0 means the first param of callback. args[0] = keyEventObject; return true; }; // 1 means callback has one param. JsCallbackHandler::Traverse(keyEventEntry->vecCopy, { 1, getKeyEventProperty }, isKeyEventConsumed); } if (keyCodeEntry != nullptr) { auto getKeyEventProperty = [keyCodeEntry](napi_env env, napi_value *args, uint8_t argc) -> bool { InputMethodSyncTrace tracer("Create parameter"); if (argc == 0) { return false; } napi_value jsObject = GetResultOnKeyEvent(env, keyCodeEntry->keyEventPara.keyCode, keyCodeEntry->keyEventPara.keyStatus); if (jsObject == nullptr) { IMSA_HILOGE("jsObject is nullptr!"); return false; } // 0 means the first param of callback. args[0] = jsObject; return true; }; // 1 means callback has one param. JsCallbackHandler::Traverse(keyCodeEntry->vecCopy, { 1, getKeyEventProperty }, isKeyCodeConsumed); } bool consumeResult = isKeyEventConsumed || isKeyCodeConsumed; if (!consumeResult) { if (keyEvent != nullptr && keyEvent->GetKeyAction() == MMI::KeyEvent::KEY_ACTION_DOWN) { IMSA_HILOGW("keyEvent is not consumed by ime"); } consumeResult = InputMethodAbility::GetInstance().HandleUnconsumedKey(keyEvent); } auto ret = InputMethodAbility::GetInstance().HandleKeyEventResult(cbId, consumeResult, channelObject); if (ret != ErrorCode::NO_ERROR) { IMSA_HILOGE("handle keyEvent failed:%{public}d", ret); } } bool JsKeyboardDelegateSetting::OnKeyEvent(const std::shared_ptr &keyEvent, sptr &consumer) { std::string type = "keyEvent"; auto entry = GetEntry(type, [keyEvent, &consumer](UvEntry &entry) { entry.fullKeyEventPara = keyEvent; entry.keyEvenetConsumer = consumer; }); if (entry == nullptr) { return false; } auto eventHandler = GetEventHandler(); if (eventHandler == nullptr) { IMSA_HILOGE("eventHandler is nullptr!"); return false; } IMSA_HILOGI("run in."); StartAsync("OnFullKeyEvent", static_cast(TraceTaskId::ON_FULL_KEY_EVENT)); auto task = [entry, this]() { auto getKeyEventProperty = [entry](napi_env env, napi_value *args, uint8_t argc) -> bool { InputMethodSyncTrace tracer("Create parameter"); if (argc == 0) { return false; } napi_value keyEventObject{}; auto result = napi_create_object(env, &keyEventObject); CHECK_RETURN((result == napi_ok) && (keyEventObject != nullptr), "create object", false); result = MMI::KeyEventNapi::CreateKeyEvent(env, entry->fullKeyEventPara, keyEventObject); CHECK_RETURN((result == napi_ok) && (keyEventObject != nullptr), "create key event object", false); // 0 means the first param of callback. args[0] = keyEventObject; return true; }; bool isConsumed = false; // 1 means callback has one param. JsCallbackHandler::Traverse(entry->vecCopy, { 1, getKeyEventProperty }, isConsumed); auto consumer = entry->keyEvenetConsumer; if (consumer != nullptr) { IMSA_HILOGE("consumer result: %{public}d!", isConsumed); OnKeyEventConsumeResult(isConsumed, consumer); } FinishAsync("OnFullKeyEvent", static_cast(TraceTaskId::ON_FULL_KEY_EVENT)); }; eventHandler->PostTask(task, type, 0, AppExecFwk::EventQueue::Priority::VIP); return true; } bool JsKeyboardDelegateSetting::OnKeyEvent(int32_t keyCode, int32_t keyStatus, sptr &consumer) { KeyEventPara para{ keyCode, keyStatus, false }; std::string type = (keyStatus == ARGC_TWO ? "keyDown" : "keyUp"); auto entry = GetEntry(type, [¶, &consumer](UvEntry &entry) { entry.keyEventPara = { para.keyCode, para.keyStatus, para.isOnKeyEvent }; entry.keyEvenetConsumer = consumer; }); if (entry == nullptr) { return false; } auto eventHandler = GetEventHandler(); if (eventHandler == nullptr) { IMSA_HILOGE("eventHandler is nullptr!"); return false; } IMSA_HILOGI("run in."); StartAsync("OnKeyEvent", static_cast(TraceTaskId::ON_KEY_EVENT)); auto task = [entry, this]() { InputMethodSyncTrace tracer("OnkeyEvent UV_QUEUE_WORK"); auto getKeyEventProperty = [entry](napi_env env, napi_value *args, uint8_t argc) -> bool { InputMethodSyncTrace tracer("Create parameter"); if (argc == 0) { return false; } napi_value jsObject = GetResultOnKeyEvent(env, entry->keyEventPara.keyCode, entry->keyEventPara.keyStatus); if (jsObject == nullptr) { IMSA_HILOGE("jsObject is nullptr!"); return false; } // 0 means the first param of callback. args[0] = jsObject; return true; }; bool isConsumed = false; // 1 means callback has one param. JsCallbackHandler::Traverse(entry->vecCopy, { 1, getKeyEventProperty }, isConsumed); auto consumer = entry->keyEvenetConsumer; if (consumer != nullptr) { IMSA_HILOGE("consumer result: %{public}d!", isConsumed); OnKeyCodeConsumeResult(isConsumed, consumer); } FinishAsync("OnKeyEvent", static_cast(TraceTaskId::ON_KEY_EVENT)); }; eventHandler->PostTask(task, type, 0, AppExecFwk::EventQueue::Priority::VIP); return true; } void JsKeyboardDelegateSetting::OnKeyEventConsumeResult(bool isConsumed, sptr consumer) { IMSA_HILOGI("result: %{public}d.", isConsumed); keyEventConsume_ = true; keyEventResult_ = isConsumed; if (keyCodeConsume_) { consumer->OnKeyEventResult(keyCodeResult_ || keyEventResult_); keyEventConsume_ = false; keyEventResult_ = false; } } void JsKeyboardDelegateSetting::OnKeyCodeConsumeResult(bool isConsumed, sptr consumer) { IMSA_HILOGI("result: %{public}d.", isConsumed); keyCodeConsume_ = true; keyCodeResult_ = isConsumed; if (keyEventConsume_) { consumer->OnKeyEventResult(keyCodeResult_ || keyEventResult_); keyCodeConsume_ = false; keyCodeResult_ = false; } } void JsKeyboardDelegateSetting::OnCursorUpdate(int32_t positionX, int32_t positionY, int32_t height) { CursorPara para{ positionX, positionY, height }; std::string type = "cursorContextChange"; auto entry = GetEntry(type, [¶](UvEntry &entry) { entry.curPara.positionX = para.positionX; entry.curPara.positionY = para.positionY; entry.curPara.height = para.height; }); if (entry == nullptr) { return; } auto eventHandler = GetEventHandler(); if (eventHandler == nullptr) { IMSA_HILOGE("eventHandler is nullptr!"); return; } IMSA_HILOGD("the cursor for x: %{public}d, y: %{public}d, height: %{public}d.", positionX, positionY, height); auto task = [entry]() { auto paramGetter = [&entry](napi_env env, napi_value *args, uint8_t argc) -> bool { if (argc < 3) { return false; } // 0 means the first param of callback. napi_create_int32(env, entry->curPara.positionX, &args[0]); // 1 means the second param of callback. napi_create_int32(env, entry->curPara.positionY, &args[1]); // 2 means the third param of callback. napi_create_int32(env, entry->curPara.height, &args[2]); return true; }; // 3 means callback has three params. JsCallbackHandler::Traverse(entry->vecCopy, { 3, paramGetter }); }; eventHandler->PostTask(task, type, 0, AppExecFwk::EventQueue::Priority::VIP); } void JsKeyboardDelegateSetting::OnSelectionChange(int32_t oldBegin, int32_t oldEnd, int32_t newBegin, int32_t newEnd) { SelectionPara para{ oldBegin, oldEnd, newBegin, newEnd }; std::string type = "selectionChange"; auto entry = GetEntry(type, [¶](UvEntry &entry) { entry.selPara.oldBegin = para.oldBegin; entry.selPara.oldEnd = para.oldEnd; entry.selPara.newBegin = para.newBegin; entry.selPara.newEnd = para.newEnd; }); if (entry == nullptr) { return; } auto eventHandler = GetEventHandler(); if (eventHandler == nullptr) { IMSA_HILOGE("eventHandler is nullptr!"); return; } IMSA_HILOGD("the selection for oldBegin: %{public}d, oldEnd: %{public}d, newBegin: %{public}d, newEnd: " "%{public}d.", oldBegin, oldEnd, newBegin, newEnd); auto task = [entry]() { auto paramGetter = [entry](napi_env env, napi_value *args, uint8_t argc) -> bool { if (argc < 4) { return false; } // 0 means the first param of callback. napi_create_int32(env, entry->selPara.oldBegin, &args[0]); // 1 means the second param of callback. napi_create_int32(env, entry->selPara.oldEnd, &args[1]); // 2 means the third param of callback. napi_create_int32(env, entry->selPara.newBegin, &args[2]); // 3 means the fourth param of callback. napi_create_int32(env, entry->selPara.newEnd, &args[3]); return true; }; // 4 means callback has four params. JsCallbackHandler::Traverse(entry->vecCopy, { 4, paramGetter }); }; eventHandler->PostTask(task, type, 0, AppExecFwk::EventQueue::Priority::VIP); } void JsKeyboardDelegateSetting::OnTextChange(const std::string &text) { std::string type = "textChange"; auto entry = GetEntry(type, [&text](UvEntry &entry) { entry.text = text; }); if (entry == nullptr) { IMSA_HILOGE("failed to get uv entry!"); return; } auto eventHandler = GetEventHandler(); if (eventHandler == nullptr) { IMSA_HILOGE("eventHandler is nullptr!"); return; } IMSA_HILOGD("run in."); auto task = [entry]() { auto getTextChangeProperty = [entry](napi_env env, napi_value *args, uint8_t argc) -> bool { if (argc == 0) { return false; } // 0 means the first param of callback. napi_create_string_utf8(env, entry->text.c_str(), NAPI_AUTO_LENGTH, &args[0]); return true; }; // 1 means callback has one param. JsCallbackHandler::Traverse(entry->vecCopy, { 1, getTextChangeProperty }); }; eventHandler->PostTask(task, type, 0, AppExecFwk::EventQueue::Priority::VIP); } void JsKeyboardDelegateSetting::OnEditorAttributeChange(const InputAttribute &inputAttribute) { std::string type = "editorAttributeChanged"; auto entry = GetEntry(type, [&inputAttribute](UvEntry &entry) { entry.inputAttribute = inputAttribute; }); if (entry == nullptr) { return; } auto eventHandler = GetEventHandler(); if (eventHandler == nullptr) { IMSA_HILOGE("eventHandler is nullptr!"); return; } IMSA_HILOGD("enterKeyType: %{public}d, inputPattern: %{public}d, previewSupport: %{public}d", inputAttribute.enterKeyType, inputAttribute.inputPattern, inputAttribute.isTextPreviewSupported); auto task = [entry]() { auto paramGetter = [entry](napi_env env, napi_value *args, uint8_t argc) -> bool { if (argc == 0) { return false; } napi_value jsObject = JsInputAttribute::Write(env, entry->inputAttribute); if (jsObject == JsUtil::Const::Null(env)) { IMSA_HILOGE("jsObject is nullptr!"); return false; } // 0 means the first param of callback. args[0] = jsObject; return true; }; // 1 means callback has one param. JsCallbackHandler::Traverse(entry->vecCopy, { 1, paramGetter }); }; eventHandler->PostTask(task, type, 0, AppExecFwk::EventQueue::Priority::VIP); } std::shared_ptr JsKeyboardDelegateSetting::GetEventHandler() { std::lock_guard lock(eventHandlerMutex_); return handler_; } std::shared_ptr JsKeyboardDelegateSetting::GetEntry(const std::string &type, EntrySetter entrySetter) { IMSA_HILOGD("start, type: %{public}s", type.c_str()); std::shared_ptr entry = nullptr; { std::lock_guard lock(mutex_); if (jsCbMap_[type].empty()) { IMSA_HILOGD("%{public}s cb-vector is empty.", type.c_str()); return nullptr; } entry = std::make_shared(jsCbMap_[type], type); } if (entrySetter != nullptr) { entrySetter(*entry); } return entry; } } // namespace MiscServices } // namespace OHOS