mirror of
				https://github.com/kevinporetti/UnrealImGui.git
				synced 2025-10-26 19:53:16 +00:00 
			
		
		
		
	Fixed issues with passing input focus between viewport and ImGui Widget.
This commit is contained in:
		
							parent
							
								
									f696393089
								
							
						
					
					
						commit
						c47d911f22
					
				| @ -157,9 +157,8 @@ void FImGuiModuleManager::AddWidgetToViewport(UGameViewportClient* GameViewport) | ||||
| 	// Bind widget's input to context for this world.
 | ||||
| 	Proxy.SetInputState(&ViewportWidget->GetInputState()); | ||||
| 
 | ||||
| 	// High enough z-order guarantees that ImGui output is rendered on top of the game UI.
 | ||||
| 	constexpr int32 IMGUI_WIDGET_Z_ORDER = 10000; | ||||
| 	GameViewport->AddViewportWidgetContent(SNew(SWeakWidget).PossiblyNullContent(ViewportWidget), IMGUI_WIDGET_Z_ORDER); | ||||
| 	// We should always have one viewport per context index at a time (this will be validated by widget).
 | ||||
| 	ViewportWidget->AttachToViewport(GameViewport); | ||||
| } | ||||
| 
 | ||||
| void FImGuiModuleManager::AddWidgetToAllViewports() | ||||
|  | ||||
| @ -12,6 +12,10 @@ | ||||
| #include "Utilities/ScopeGuards.h" | ||||
| 
 | ||||
| 
 | ||||
| // High enough z-order guarantees that ImGui output is rendered on top of the game UI.
 | ||||
| constexpr int32 IMGUI_WIDGET_Z_ORDER = 10000; | ||||
| 
 | ||||
| 
 | ||||
| DEFINE_LOG_CATEGORY_STATIC(LogImGuiWidget, Warning, All); | ||||
| 
 | ||||
| #define TEXT_INPUT_MODE(Val) ((Val) == EInputMode::MouseAndKeyboard ? TEXT("MouseAndKeyboard") : (Val) == EInputMode::MouseOnly ? TEXT("MouseOnly") : TEXT("None")) | ||||
| @ -66,6 +70,23 @@ SImGuiWidget::~SImGuiWidget() | ||||
| 	ModuleManager->OnPostImGuiUpdate().RemoveAll(this); | ||||
| } | ||||
| 
 | ||||
| void SImGuiWidget::AttachToViewport(UGameViewportClient* InGameViewport, bool bResetInput) | ||||
| { | ||||
| 	checkf(InGameViewport, TEXT("Null InGameViewport")); | ||||
| 	checkf(!GameViewport.IsValid() || GameViewport.Get() == InGameViewport, | ||||
| 		TEXT("Widget is attached to another game viewport and will be available for reuse only after this session ") | ||||
| 		TEXT("ends. ContextIndex = %d, CurrentGameViewport = %s, InGameViewport = %s"), | ||||
| 		ContextIndex, *GameViewport->GetName(), InGameViewport->GetName()); | ||||
| 
 | ||||
| 	if (bResetInput) | ||||
| 	{ | ||||
| 		ResetInputState(); | ||||
| 	} | ||||
| 
 | ||||
| 	GameViewport = InGameViewport; | ||||
| 	GameViewport->AddViewportWidgetContent(SNew(SWeakWidget).PossiblyNullContent(SharedThis(this)), IMGUI_WIDGET_Z_ORDER); | ||||
| } | ||||
| 
 | ||||
| void SImGuiWidget::Tick(const FGeometry& AllottedGeometry, const double InCurrentTime, const float InDeltaTime) | ||||
| { | ||||
| 	Super::Tick(AllottedGeometry, InCurrentTime, InDeltaTime); | ||||
| @ -209,6 +230,13 @@ void SImGuiWidget::CopyModifierKeys(const FPointerEvent& MouseEvent) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void SImGuiWidget::ResetInputState() | ||||
| { | ||||
| 	bInputEnabled = false; | ||||
| 	SetVisibilityFromInputEnabled(); | ||||
| 	UpdateInputMode(false, false); | ||||
| } | ||||
| 
 | ||||
| void SImGuiWidget::SetVisibilityFromInputEnabled() | ||||
| { | ||||
| 	// If we don't use input disable hit test to make this widget invisible for cursors hit detection.
 | ||||
| @ -230,21 +258,33 @@ void SImGuiWidget::UpdateInputEnabled() | ||||
| 
 | ||||
| 		SetVisibilityFromInputEnabled(); | ||||
| 
 | ||||
| 		// Setup input to show cursor and take focus when we use input or clear state and pass focus back to viewport
 | ||||
| 		// when we don't.
 | ||||
| 		// Setup input to show cursor and to pass keyboard/user focus between viewport and widget. Note that we should
 | ||||
| 		// only pass focus if it is inside of the parent viewport, otherwise we would be stealing from other viewports
 | ||||
| 		// or windows.
 | ||||
| 		auto& Slate = FSlateApplication::Get(); | ||||
| 		if (bInputEnabled) | ||||
| 		{ | ||||
| 			const auto& ViewportWidget = GameViewport->GetGameViewportWidget(); | ||||
| 			if (ViewportWidget->HasKeyboardFocus() || ViewportWidget->HasFocusedDescendants()) | ||||
| 			{ | ||||
| 				// Remember where is user focus, so we will have an option to restore it.
 | ||||
| 				PreviousUserFocusedWidget = Slate.GetUserFocusedWidget(Slate.GetUserIndexForKeyboard()); | ||||
| 
 | ||||
| 				Slate.ResetToDefaultPointerInputSettings(); | ||||
| 				Slate.SetKeyboardFocus(SharedThis(this)); | ||||
| 			} | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			if (Slate.GetKeyboardFocusedWidget().Get() == this) | ||||
| 			{ | ||||
| 				Slate.SetUserFocusToGameViewport(Slate.GetUserIndexForKeyboard()); | ||||
| 				Slate.ResetToDefaultPointerInputSettings(); | ||||
| 				Slate.SetUserFocus(Slate.GetUserIndexForKeyboard(), | ||||
| 					PreviousUserFocusedWidget.IsValid() ? PreviousUserFocusedWidget.Pin() : GameViewport->GetGameViewportWidget()); | ||||
| 			} | ||||
| 
 | ||||
| 			PreviousUserFocusedWidget.Reset(); | ||||
| 
 | ||||
| 			UpdateInputMode(false, false); | ||||
| 		} | ||||
| 	} | ||||
| @ -338,6 +378,11 @@ FVector2D SImGuiWidget::ComputeDesiredSize(float) const | ||||
| // Controls tweaked for 2-columns layout.
 | ||||
| namespace TwoColumns | ||||
| { | ||||
| 	static void GroupName(const char* Name) | ||||
| 	{ | ||||
| 		ImGui::TextColored({ 0.5f, 0.5f, 0.5f, 1.f }, Name); ImGui::NextColumn(); ImGui::NextColumn(); | ||||
| 	} | ||||
| 
 | ||||
| 	static void Value(const char* Label, int Value) | ||||
| 	{ | ||||
| 		ImGui::Text("%s:", Label); ImGui::NextColumn(); | ||||
| @ -362,12 +407,13 @@ void SImGuiWidget::OnDebugDraw() | ||||
| 	bool bDebug = CVars::DebugWidget.GetValueOnGameThread() > 0; | ||||
| 	if (bDebug) | ||||
| 	{ | ||||
| 		ImGui::SetNextWindowSize(ImVec2(300, 200), ImGuiSetCond_Once); | ||||
| 		ImGui::SetNextWindowSize(ImVec2(380, 320), ImGuiSetCond_Once); | ||||
| 		if (ImGui::Begin("ImGui Widget Debug", &bDebug)) | ||||
| 		{ | ||||
| 			ImGui::Columns(2, nullptr, false); | ||||
| 
 | ||||
| 			TwoColumns::Value("Context Index", ContextIndex); | ||||
| 			TwoColumns::Value("Game Viewport", *GameViewport->GetName()); | ||||
| 
 | ||||
| 			ImGui::Separator(); | ||||
| 
 | ||||
| @ -376,10 +422,33 @@ void SImGuiWidget::OnDebugDraw() | ||||
| 
 | ||||
| 			ImGui::Separator(); | ||||
| 
 | ||||
| 			const float GroupIndent = 5.f; | ||||
| 
 | ||||
| 			TwoColumns::GroupName("Widget"); | ||||
| 			ImGui::Indent(GroupIndent); | ||||
| 			{ | ||||
| 				TwoColumns::Value("Visibility", *GetVisibility().ToString()); | ||||
| 				TwoColumns::Value("Is Hovered", IsHovered()); | ||||
| 				TwoColumns::Value("Is Directly Hovered", IsDirectlyHovered()); | ||||
| 				TwoColumns::Value("Has Keyboard Input", HasKeyboardFocus()); | ||||
| 			} | ||||
| 			ImGui::Unindent(GroupIndent); | ||||
| 
 | ||||
| 			ImGui::Separator(); | ||||
| 
 | ||||
| 			TwoColumns::GroupName("Viewport Widget"); | ||||
| 			ImGui::Indent(GroupIndent); | ||||
| 			{ | ||||
| 				const auto& ViewportWidget = GameViewport->GetGameViewportWidget(); | ||||
| 				TwoColumns::Value("Is Hovered", ViewportWidget->IsHovered()); | ||||
| 				TwoColumns::Value("Is Directly Hovered", ViewportWidget->IsDirectlyHovered()); | ||||
| 				TwoColumns::Value("Has Mouse Capture", ViewportWidget->HasMouseCapture()); | ||||
| 				TwoColumns::Value("Has Keyboard Input", ViewportWidget->HasKeyboardFocus()); | ||||
| 				TwoColumns::Value("Has Focused Descendants", ViewportWidget->HasFocusedDescendants()); | ||||
| 				auto Widget = PreviousUserFocusedWidget.Pin(); | ||||
| 				TwoColumns::Value("Previous User Focused", Widget.IsValid() ? *Widget->GetTypeAsString() : TEXT("None")); | ||||
| 			} | ||||
| 			ImGui::Unindent(GroupIndent); | ||||
| 
 | ||||
| 			ImGui::Columns(1); | ||||
| 		} | ||||
|  | ||||
| @ -32,6 +32,16 @@ public: | ||||
| 	// Get input state associated with this widget.
 | ||||
| 	const FImGuiInputState& GetInputState() const { return InputState; } | ||||
| 
 | ||||
| 	// Get the game viewport to which this widget is attached.
 | ||||
| 	const TWeakObjectPtr<UGameViewportClient>& GetGameViewport() const { return GameViewport; } | ||||
| 
 | ||||
| 	// Attach this widget to a target game viewport.
 | ||||
| 	// Widget can be attached to only one viewport at a time but can be reused after its last viewport becomes invalid
 | ||||
| 	// at the end of a session. Widgets are weakly attached, so once destroyed they are automatically removed.
 | ||||
| 	// @param InGameViewport - Target game viewport
 | ||||
| 	// @param bResetInput - If true (default), input will be reset back to a default state
 | ||||
| 	void AttachToViewport(UGameViewportClient* InGameViewport, bool bResetInput = true); | ||||
| 
 | ||||
| 	//----------------------------------------------------------------------------------------------------
 | ||||
| 	// SWidget overrides
 | ||||
| 	//----------------------------------------------------------------------------------------------------
 | ||||
| @ -76,6 +86,8 @@ private: | ||||
| 	FORCEINLINE void CopyModifierKeys(const FInputEvent& InputEvent); | ||||
| 	FORCEINLINE void CopyModifierKeys(const FPointerEvent& MouseEvent); | ||||
| 
 | ||||
| 	void ResetInputState(); | ||||
| 
 | ||||
| 	// Update visibility based on input enabled state.
 | ||||
| 	void SetVisibilityFromInputEnabled(); | ||||
| 
 | ||||
| @ -94,6 +106,7 @@ private: | ||||
| 	void OnDebugDraw(); | ||||
| 
 | ||||
| 	FImGuiModuleManager* ModuleManager = nullptr; | ||||
| 	TWeakObjectPtr<UGameViewportClient> GameViewport; | ||||
| 
 | ||||
| 	mutable TArray<FSlateVertex> VertexBuffer; | ||||
| 	mutable TArray<SlateIndex> IndexBuffer; | ||||
| @ -104,4 +117,6 @@ private: | ||||
| 	bool bInputEnabled = false; | ||||
| 
 | ||||
| 	FImGuiInputState InputState; | ||||
| 
 | ||||
| 	TWeakPtr<SWidget> PreviousUserFocusedWidget; | ||||
| }; | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Sebastian
						Sebastian