成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

OpenHarmony源碼解析之多模輸入子系統(一)

系統 OpenHarmony
本篇文章基于社區weekly_20230207的代碼,對多模輸入客戶端注冊監聽流程和多模服務端事件派發流程作了簡單介紹。相信大家通過本文,對多模輸入子系統能有一個大致了解。

??想了解更多關于開源的內容,請訪問:??

??51CTO 開源基礎軟件社區??

??https://ost.51cto.com??

1、簡介

多模輸入子系統是 OpenHarmony 輸入事件管理框架。多模輸入服務接收多種類型輸入設備(觸摸屏、鼠標、鍵盤、觸摸板等)的輸入事件,通過歸一/標準化處理后,分發給多模客戶端(應用,系統服務)。多模輸入還提供事件注入接口,該接口目前僅對系統應用開放。

多模輸入子系統分為框架部分和服務部分:框架部分封裝了各種接口給其他子系統和應用來調用;服務部分實現了這些接口,并且實現了事件派發處理的核心邏輯。這兩個部分運行在不同進程中,根據具體接口,通過socket或者binder ipc機制進行通信。

(1)主要模塊交互圖

OpenHarmony源碼解析之多模輸入子系統(一)-開源基礎軟件社區

(2)代碼目錄

/foundation/multimodalinput/input
├── frameworks # napi接口代碼,客戶端實現代碼
├── interfaces # 對外接口存放目錄
└── native # 對外native層接口存放目錄
└── innerkits # 對系統內部子系統提供native層接口存放目錄
├── service # 服務端代碼
├── sa_profile # 服務啟動配置文件
├── tools # 輸入事件注入工具
├── uinput # 輸入事件注入模塊
├── util # socket相關工具類

2、多模客戶端啟動流程

(1)時序圖

OpenHarmony源碼解析之多模輸入子系統(一)-開源基礎軟件社區

說明:

  • Ability生命周期函數OnStart()中會去創建WindowImpl實例,WindowImpl::Create()中調用InputTransferStation::AddInputWindow()創建InputEventListener并注冊到InputManagerImpl中。后續收到多模服務端發送來的輸入事件之后會通過回調InputEventListener的接口函數,把事件上報到窗口管理,窗口管理再把事件進一步上報給ArkUI。
  • InputManagerImpl::SetWindowInputEventConsumer()方法中會去初始化多模Socket客戶端,用于接收多模服務端發來的輸入事件。

(2)ArkUI何時注冊的窗口管理輸入事件回調?

AceAbility::OnStart()方法中先調用基類Ability::OnStart()方法走完上述時序圖的流程,然后調用如下代碼段,創建AceWindowListener,并調用WindowImpl::SetInputEventConsumer()注冊輸入事件回調。

OHOS::sptr<OHOS::Rosen::Window> window = Ability::GetWindow();
std::shared_ptr<AceAbility> self = std::static_pointer_cast<AceAbility>(shared_from_this());
OHOS::sptr<AceWindowListener> aceWindowListener = new AceWindowListener(self);
// register surface change callback and window mode change callback
window->RegisterWindowChangeListener(aceWindowListener);
// register drag event callback
window->RegisterDragListener(aceWindowListener);
// register Occupied Area callback
window->RegisterOccupiedAreaChangeListener(aceWindowListener);
// register ace ability handler callback
window->SetAceAbilityHandler(aceWindowListener);
// register input consumer callback
std::shared_ptr<AceWindowListener> aceInputConsumer = std::make_shared<AceWindowListener>(self);
window->SetInputEventConsumer(aceInputConsumer);

3、多模輸入服務

(1)多模服務初始化流程

OpenHarmony源碼解析之多模輸入子系統(一)-開源基礎軟件社區

說明:

  • MMIService::OnThread()中會起循環,等待并處理epoll事件。接收到libinput相關的epoll事件后,調用LibinputAdapter::EventDispatch()處理input事件。
void MMIService::OnThread()
{
SetThreadName(std::string("mmi_service"));
uint64_t tid = GetThisThreadId();
delegateTasks_.SetWorkerThreadId(tid);
MMI_HILOGI("Main worker thread start. tid:%{public}" PRId64 "", tid);
#ifdef OHOS_RSS_CLIENT
tid_.store(tid);
#endif
libinputAdapter_.RetriggerHotplugEvents();
libinputAdapter_.ProcessPendingEvents();
while (state_ == ServiceRunningState::STATE_RUNNING) {
epoll_event ev[MAX_EVENT_SIZE] = {};
int32_t timeout = TimerMgr->CalcNextDelay();
MMI_HILOGD("timeout:%{public}d", timeout);
int32_t count = EpollWait(ev[0], MAX_EVENT_SIZE, timeout, mmiFd_);
for (int32_t i = 0; i < count && state_ == ServiceRunningState::STATE_RUNNING; i++) {
auto mmiEd = reinterpret_cast<mmi_epoll_event*>(ev[i].data.ptr);
CHKPC(mmiEd);
if (mmiEd->event_type == EPOLL_EVENT_INPUT) {
libinputAdapter_.EventDispatch(ev[i]);//處理input事件
} else if (mmiEd->event_type == EPOLL_EVENT_SOCKET) {
OnEpollEvent(ev[i]);
} else if (mmiEd->event_type == EPOLL_EVENT_SIGNAL) {
OnSignalEvent(mmiEd->fd);
} else if (mmiEd->event_type == EPOLL_EVENT_ETASK) {
OnDelegateTask(ev[i]);
} else {
MMI_HILOGW("Unknown epoll event type:%{public}d", mmiEd->event_type);
}
}
TimerMgr->ProcessTimers();
if (state_ != ServiceRunningState::STATE_RUNNING) {
break;
}
}
MMI_HILOGI("Main worker thread stop. tid:%{public}" PRId64 "", tid);
}
  • InputEventHandler::BuildInputHandlerChain()會創建IInputEventHandler對象鏈,用于處理libinput上報的input事件。類圖如下:
  • InputEventHandler::OnEvent(void event)調用。
    EventNormalizeHandler::HandleEvent(libinput_event
     event)開始按順序處理輸入事件。
  • EventNormalizeHandler把libinput_event標準化成各種InputEvent(KeyEvent,PointerEvent,AxisEvent),并傳遞給下一級 EventFilterHandler處理。
  • EventFilterHandler會過濾一些事件,否則繼續往下傳遞。
  • EventInterceptorHandler事件攔截器,攔截成功不會繼續往下傳。
  • KeyCommandHandler根據配置文件,對一些特殊按鍵,拉起特定應用界面,或者對電源鍵,音量鍵做特殊處理,否則繼續往下傳遞。
  • KeySubscriberHandler應用訂閱的組合按鍵(應用通過inputConsumer.on接口訂閱)處理,否則繼續往下傳遞。
  • EventMonitorHandler事件跟蹤器,把事件分發給跟蹤者并繼續往下傳。
  • EventDispatchHandler通過socket把事件派發給應用。

4、多模輸入touch事件派發流程

OpenHarmony源碼解析之多模輸入子系統(一)-開源基礎軟件社區

說明:
MMIService收到libinput上報的input事件后,會調用InputEventHandler::OnEvent來處理輸入事件。最終EventDispatchHandler通過socket把事件派發給目標應用進程。

5、如何確定輸入事件派發的目標進程?

多模服務端InputWindowsManager類中有如下成員變量。

DisplayGroupInfo displayGroupInfo_;
std::map<int32_t, WindowInfo> touchItemDownInfos_;

DisplayGroupInfo中包含了當前獲焦的窗口id,以z軸排序的窗口信息列表,物理屏幕信息列表等。displayGroupInfo_信息由窗口管理服務調用。
MMI::InputManager::GetInstance()->UpdateDisplayInfo(displayGroupInfo_)接口設置。

struct DisplayGroupInfo {
int32_t width; //Width of the logical display
int32_t height; //Height of the logical display
int32_t focusWindowId; //ID of the focus window
//List of window information of the logical display arranged in Z order, with the top window at the top
std::vector<WindowInfo> windowsInfo;
std::vector<DisplayInfo> displaysInfo; //Physical screen information list
};

以鍵盤按鍵事件為例。

收到libinput上報的輸入事件之后,最終走到EventDispatchHandler::DispatchKeyEventPid(UDSServer& udsServer, std::shared_ptr<KeyEvent> key)函數。

簡化的調用流程如下:

EventDispatchHandler::DispatchKeyEventPid() =>
InputWindowsManager::UpdateTarget() =>
InputWindowsManager::GetPidAndUpdateTarget()
int32_t InputWindowsManager::GetPidAndUpdateTarget(std::shared_ptr<InputEvent> inputEvent)
{
CALL_DEBUG_ENTER;
CHKPR(inputEvent, INVALID_PID);
const int32_t focusWindowId = displayGroupInfo_.focusWindowId;
WindowInfo* windowInfo = nullptr;
for (auto &item : displayGroupInfo_.windowsInfo) {
if (item.id == focusWindowId) {
windowInfo = &item;
break;
}
}
CHKPR(windowInfo, INVALID_PID);
inputEvent->SetTargetWindowId(windowInfo->id);
inputEvent->SetAgentWindowId(windowInfo->agentWindowId);
MMI_HILOGD("focusWindowId:%{public}d, pid:%{public}d", focusWindowId, windowInfo->pid);
return windowInfo->pid;
}

InputWindowsManager::GetPidAndUpdateTarget()函數中把當前獲焦windowId信息設置到InputEvent中,并且返回目標窗口所在進程pid,有了目標進程pid,就可以獲取到目標進程對應的socket會話的服務端fd,把事件派發給目標進程。

touch事件目標窗口信息的獲取和按鍵事件不同,感興趣的可以自己查看代碼。

6、總結

本篇文章基于社區weekly_20230207的代碼,對多模輸入客戶端注冊監聽流程和多模服務端事件派發流程作了簡單介紹。相信大家通過本文,對多模輸入子系統能有一個大致了解。

??想了解更多關于開源的內容,請訪問:??

??51CTO 開源基礎軟件社區??

??https://ost.51cto.com??

責任編輯:jianghua 來源: 51CTO 開源基礎軟件社區
相關推薦

2021-09-17 14:38:58

鴻蒙HarmonyOS應用

2021-09-13 15:15:18

鴻蒙HarmonyOS應用

2023-06-28 15:00:02

開源鴻蒙輸入系統架構

2022-02-17 20:57:07

OpenHarmon操作系統鴻蒙

2022-01-06 16:17:58

鴻蒙HarmonyOS應用

2021-11-08 15:04:47

鴻蒙HarmonyOS應用

2021-12-17 16:42:09

鴻蒙HarmonyOS應用

2021-09-18 14:40:37

鴻蒙HarmonyOS應用

2023-04-12 15:31:11

系統服務管理鴻蒙

2022-01-10 15:30:11

鴻蒙HarmonyOS應用

2022-05-10 11:17:27

電話子系統數據服務模塊

2021-11-18 10:28:03

鴻蒙HarmonyOS應用

2022-05-24 15:46:51

Wi-FiSTA模式

2022-01-20 14:33:29

openharmonwayland協議鴻蒙

2022-01-13 10:11:59

鴻蒙HarmonyOS應用

2022-05-20 10:32:49

事件循環器事件隊列鴻蒙

2022-03-18 16:07:04

Graphic子系統鴻蒙

2022-01-20 11:04:31

Linux DRMOpenHarmon鴻蒙

2022-05-30 15:08:33

包管理子系統包安裝模塊

2022-02-14 14:47:11

SystemUIOpenHarmon鴻蒙
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 自拍偷拍第一页 | 亚洲精品一二三区 | 午夜黄色影院 | 青青草在线视频免费观看 | 91综合网 | 国产成人一区二区三区 | 久久久久久久久久久久久久国产 | 免费三级网站 | 综合国产 | 欧美一级久久久猛烈a大片 日韩av免费在线观看 | 久久精品毛片 | 成年人视频在线免费观看 | 国产一区二区三区精品久久久 | 亚洲国产成人精品久久久国产成人一区 | 精品久久久久久 | 日韩三级电影一区二区 | 国产视频一视频二 | 国产精品久久久久久 | 亚洲一区二区三区四区五区午夜 | 国产综合第一页 | 91精品一区二区三区久久久久 | 黄色三级毛片 | japan21xxxxhd美女 日本欧美国产在线 | 久久久久久久国产 | 美国黄色毛片 | a在线视频 | 日韩欧美不卡 | 午夜精品一区二区三区在线视频 | 久久久久久网站 | 国产九九精品 | 人人爽日日躁夜夜躁尤物 | 国产成人叼嘿视频在线观看 | www.婷婷 | 国产精品自在线 | 国产成人久久久 | 欧美精品中文 | 国产日产精品一区二区三区四区 | 免费不卡一区 | 中文字幕在线播放不卡 | 天天综合网天天综合 | 91精品久久久久久久久久小网站 |