// Copyright 1998-2019 Epic Games, Inc. All Rights Reserved. #include "InputRouter.h" UInputRouter::UInputRouter() { ActiveLeftCapture = nullptr; ActiveLeftCaptureOwner = nullptr; ActiveRightCapture = nullptr; ActiveRightCaptureOwner = nullptr; ActiveKeyboardCapture = nullptr; ActiveKeyboardCaptureOwner = nullptr; bAutoInvalidateOnHover = false; bAutoInvalidateOnCapture = false; ActiveInputBehaviors = NewObject(this, "InputBehaviors"); } void UInputRouter::Initialize(IToolsContextTransactionsAPI* TransactionsAPIIn) { this->TransactionsAPI = TransactionsAPIIn; } void UInputRouter::Shutdown() { this->TransactionsAPI = nullptr; } void UInputRouter::RegisterSource(IInputBehaviorSource* Source) { ActiveInputBehaviors->Add(Source->GetInputBehaviors(), Source); } void UInputRouter::DeregisterSource(IInputBehaviorSource* Source) { ActiveInputBehaviors->RemoveBySource(Source); } void UInputRouter::PostInputEvent(const FInputDeviceState& Input) { if (ActiveInputBehaviors->IsEmpty()) { return; } if (Input.IsFromDevice(EInputDevices::Mouse)) { PostInputEvent_Mouse(Input); } else if (Input.IsFromDevice(EInputDevices::Keyboard)) { PostInputEvent_Keyboard(Input); } else { unimplemented(); TransactionsAPI->PostMessage(TEXT("UInteractiveToolManager::PostInputEvent - input device is not currently supported."), EToolMessageLevel::Internal); return; } } // // Keyboard event handling // void UInputRouter::PostInputEvent_Keyboard(const FInputDeviceState& Input) { if (ActiveKeyboardCapture != nullptr) { HandleCapturedKeyboardInput(Input); } else { ActiveKeyboardCaptureData = FInputCaptureData(); CheckForKeyboardCaptures(Input); } } void UInputRouter::CheckForKeyboardCaptures(const FInputDeviceState& Input) { TArray CaptureRequests; ActiveInputBehaviors->CollectWantsCapture(Input, CaptureRequests); if (CaptureRequests.Num() == 0) { return; } CaptureRequests.StableSort(); bool bAccepted = false; for (int i = 0; i < CaptureRequests.Num() && bAccepted == false; ++i) { FInputCaptureUpdate Result = CaptureRequests[i].Source->BeginCapture(Input, EInputCaptureSide::Left); if (Result.State == EInputCaptureState::Begin) { ActiveKeyboardCapture = Result.Source; ActiveKeyboardCaptureOwner = CaptureRequests[i].Owner; ActiveKeyboardCaptureData = Result.Data; bAccepted = true; } } } void UInputRouter::HandleCapturedKeyboardInput(const FInputDeviceState& Input) { if (ActiveKeyboardCapture == nullptr) { return; } FInputCaptureUpdate Result = ActiveKeyboardCapture->UpdateCapture(Input, ActiveKeyboardCaptureData); if (Result.State == EInputCaptureState::End) { ActiveKeyboardCapture = nullptr; ActiveKeyboardCaptureOwner = nullptr; ActiveKeyboardCaptureData = FInputCaptureData(); } else if (Result.State != EInputCaptureState::Continue) { TransactionsAPI->PostMessage(TEXT("UInteractiveToolManager::HandleCapturedKeyboardInput - unexpected capture state!"), EToolMessageLevel::Internal); } if (bAutoInvalidateOnCapture) { TransactionsAPI->PostInvalidation(); } } // // Mouse event handling // void UInputRouter::PostInputEvent_Mouse(const FInputDeviceState& Input) { if (ActiveLeftCapture != nullptr) { HandleCapturedMouseInput(Input); } else { ActiveLeftCaptureData = FInputCaptureData(); CheckForMouseCaptures(Input); } // update hover if nobody is capturing if (ActiveLeftCapture == nullptr && ActiveRightCapture == nullptr) { bool bProcessed = ActiveInputBehaviors->UpdateHover(Input); if (bProcessed && bAutoInvalidateOnHover) { TransactionsAPI->PostInvalidation(); } } } void UInputRouter::PostHoverInputEvent(const FInputDeviceState& Input) { LastHoverInput = Input; bool bProcessed = ActiveInputBehaviors->UpdateHover(Input); if (bProcessed && bAutoInvalidateOnHover) { TransactionsAPI->PostInvalidation(); } } bool UInputRouter::HasActiveMouseCapture() const { return (ActiveLeftCapture != nullptr); } void UInputRouter::CheckForMouseCaptures(const FInputDeviceState& Input) { TArray CaptureRequests; ActiveInputBehaviors->CollectWantsCapture(Input, CaptureRequests); if (CaptureRequests.Num() == 0) { return; } CaptureRequests.StableSort(); bool bAccepted = false; for (int i = 0; i < CaptureRequests.Num() && bAccepted == false; ++i) { FInputCaptureUpdate Result = CaptureRequests[i].Source->BeginCapture(Input, EInputCaptureSide::Left); if (Result.State == EInputCaptureState::Begin) { // end outstanding hovers ActiveInputBehaviors->EndHover(Input); ActiveLeftCapture = Result.Source; ActiveLeftCaptureOwner = CaptureRequests[i].Owner; ActiveLeftCaptureData = Result.Data; bAccepted = true; } } } void UInputRouter::HandleCapturedMouseInput(const FInputDeviceState& Input) { if (ActiveLeftCapture == nullptr) { return; } // have active capture - give it this event FInputCaptureUpdate Result = ActiveLeftCapture->UpdateCapture(Input, ActiveLeftCaptureData); if (Result.State == EInputCaptureState::End) { ActiveLeftCapture = nullptr; ActiveLeftCaptureOwner = nullptr; ActiveLeftCaptureData = FInputCaptureData(); } else if (Result.State != EInputCaptureState::Continue) { TransactionsAPI->PostMessage(TEXT("UInteractiveToolManager::HandleCapturedMouseInput - unexpected capture state!"), EToolMessageLevel::Internal); } if (bAutoInvalidateOnCapture) { TransactionsAPI->PostInvalidation(); } } void UInputRouter::ForceTerminateAll() { if (ActiveKeyboardCapture != nullptr) { ActiveKeyboardCapture->ForceEndCapture(ActiveKeyboardCaptureData); ActiveKeyboardCapture = nullptr; ActiveKeyboardCaptureOwner = nullptr; ActiveKeyboardCaptureData = FInputCaptureData(); } if (ActiveLeftCapture != nullptr) { ActiveLeftCapture->ForceEndCapture(ActiveLeftCaptureData); ActiveLeftCapture = nullptr; ActiveLeftCaptureOwner = nullptr; ActiveLeftCaptureData = FInputCaptureData(); } if (ActiveRightCapture != nullptr) { ActiveRightCapture->ForceEndCapture(ActiveRightCaptureData); ActiveRightCapture = nullptr; ActiveRightCaptureOwner = nullptr; ActiveRightCaptureData = FInputCaptureData(); } ActiveInputBehaviors->EndHover(LastHoverInput); } void UInputRouter::ForceTerminateSource(IInputBehaviorSource* Source) { if (ActiveKeyboardCapture != nullptr && ActiveKeyboardCaptureOwner == Source) { ActiveKeyboardCapture->ForceEndCapture(ActiveKeyboardCaptureData); ActiveKeyboardCapture = nullptr; ActiveKeyboardCaptureOwner = nullptr; ActiveKeyboardCaptureData = FInputCaptureData(); } if (ActiveLeftCapture != nullptr && ActiveLeftCaptureOwner == Source) { ActiveLeftCapture->ForceEndCapture(ActiveLeftCaptureData); ActiveLeftCapture = nullptr; ActiveLeftCaptureOwner = nullptr; ActiveLeftCaptureData = FInputCaptureData(); } if (ActiveRightCapture != nullptr && ActiveRightCaptureOwner == Source) { ActiveRightCapture->ForceEndCapture(ActiveRightCaptureData); ActiveRightCapture = nullptr; ActiveRightCaptureOwner = nullptr; ActiveRightCaptureData = FInputCaptureData(); } }