CogEngine: console window improvements

This commit is contained in:
Arnaud Jamin
2025-01-31 23:33:22 -05:00
parent 2ede6b4cf9
commit 0429a7b313
3 changed files with 179 additions and 130 deletions
@@ -25,6 +25,7 @@ void FCogEngineWindow_Console::Initialize()
bNoPadding = true;
bHasMenu = true;
bHasWidget = 1;
SelectedCommandIndex = -1;
RefreshCommandList();
}
@@ -95,28 +96,14 @@ void FCogEngineWindow_Console::RenderContent()
RenderConsoleTextInput();
}
if (ImGui::BeginChild("Commands", ImVec2(0.0f, -1.0f), ImGuiChildFlags_NavFlattened))
const ImVec2 Size = IsWindowRenderedInMainMenu() ? ImVec2(0, ImGui::GetFontSize() * 20) : ImVec2(0.0f, -1.0f);
if (ImGui::BeginChild("Commands", Size, ImGuiChildFlags_NavFlattened))
{
const float Indent = ImGui::GetFontSize() * 0.5f;
ImGui::Indent(Indent);
RenderCommandList();
if (ImGui::IsWindowFocused(ImGuiFocusedFlags_RootAndChildWindows))
{
if (bIsWindowFocused == false)
{
bRequestTextInputFocus = true;
}
bIsWindowFocused = true;
HandleInputs();
}
else
{
bIsWindowFocused = false;
}
ImGui::Unindent(Indent);
}
@@ -146,7 +133,9 @@ void FCogEngineWindow_Console::RenderConsoleTextInput()
ImGuiInputTextFlags_EnterReturnsTrue
| ImGuiInputTextFlags_EscapeClearsAll
| ImGuiInputTextFlags_CallbackCompletion
| ImGuiInputTextFlags_CallbackEdit;
| ImGuiInputTextFlags_CallbackHistory
| ImGuiInputTextFlags_CallbackEdit
| ImGuiInputTextFlags_CallbackAlways;
if (FCogWindowWidgets::InputTextWithHint("##Command", "Command", CurrentUserInput, InputFlags, &OnTextInputCallbackStub, this))
{
@@ -155,45 +144,108 @@ void FCogEngineWindow_Console::RenderConsoleTextInput()
}
ImGui::SetItemDefaultFocus();
//-------------------------------------------------------------------------------------------------
// In Widget mode, do not want to show the command list as soon as the input text has focus,
// but wait for the user to click on the input text (or interact with it). This is because
// we want the text input to always have focus so the user can directly type text if he wants to,
// but if he doesn't the command list should not clutter the screen.
//-------------------------------------------------------------------------------------------------
if (bIsRenderingWidget && ImGui::IsItemActive() && ImGui::IsMouseClicked(ImGuiMouseButton_Left))
{
bPopupCommandListOnWidgetMode = true;
}
if (ImGui::IsItemActive() == false
&& bRequestTextInputFocus
&& ImGui::IsWindowFocused(ImGuiFocusedFlags_RootAndChildWindows)
&& IsWindowRenderedInMainMenu() == false)
/*&& ImGui::IsWindowFocused(ImGuiFocusedFlags_RootAndChildWindows)*/
/*&& IsWindowRenderedInMainMenu() == false*/)
{
ImGui::SetKeyboardFocusHere(-1);
//bRequestTextInputFocus = false;
}
if (ImGui::IsItemActive())
{
bRequestTextInputFocus = false;
bRequestTextInputFocus = false;
}
}
//--------------------------------------------------------------------------------------------------------------------------
void FCogEngineWindow_Console::HandleInputs()
int FCogEngineWindow_Console::OnTextInputCallback(ImGuiInputTextCallbackData* InData)
{
if (ImGui::IsKeyPressed(ImGuiKey_DownArrow))
bool DoCompletion = false;
if (InData->EventFlag == ImGuiInputTextFlags_CallbackHistory)
{
SelectedCommandIndex += 1;
bScrollDirection = -1;
DoCompletion = true;
if (SelectedCommandIndex >= CommandList.Num())
if (InData->EventKey == ImGuiKey_UpArrow)
{
SelectedCommandIndex = 0;
SelectedCommandIndex -= 1;
bScroll = true;
if (SelectedCommandIndex < 0)
{
SelectedCommandIndex = CommandList.Num() - 1;
}
}
else if (InData->EventKey == ImGuiKey_DownArrow)
{
SelectedCommandIndex += 1;
bScroll = true;
if (SelectedCommandIndex >= CommandList.Num())
{
SelectedCommandIndex = 0;
}
}
}
else if (ImGui::IsKeyPressed(ImGuiKey_UpArrow))
else if (InData->EventFlag == ImGuiInputTextFlags_CallbackCompletion)
{
SelectedCommandIndex -= 1;
bScrollDirection = 1;
DoCompletion = true;
}
else if (InData->EventFlag == ImGuiInputTextFlags_CallbackEdit)
{
CurrentUserInput = FString(InData->Buf);
RefreshCommandList();
if (SelectedCommandIndex < 0)
if (bIsRenderingWidget)
{
SelectedCommandIndex = CommandList.Num() - 1;
//ImGui::SetScrollHereY(1.0f);
bPopupCommandListOnWidgetMode = true;
}
}
else if (InData->EventFlag == ImGuiInputTextFlags_CallbackAlways)
{
if (bSetBufferToSelectedCommand)
{
DoCompletion = true;
bSetBufferToSelectedCommand = false;
}
}
if (DoCompletion && CommandList.IsValidIndex(SelectedCommandIndex))
{
const FString SelectedCommand = CommandList[SelectedCommandIndex];
const FString CleanupSelectedCommand = SelectedCommand.TrimEnd();
const auto& CommandStr = StringCast<ANSICHAR>(*CleanupSelectedCommand);
InData->DeleteChars(0, InData->BufTextLen);
InData->InsertChars(0, CommandStr.Get());
InData->InsertChars(InData->CursorPos, " ");
if (bIsRenderingWidget)
{
bPopupCommandListOnWidgetMode = true;
}
}
return 0;
}
//--------------------------------------------------------------------------------------------------------------------------
int FCogEngineWindow_Console::OnTextInputCallbackStub(ImGuiInputTextCallbackData* InData)
{
FCogEngineWindow_Console& ConsoleWindow = *static_cast<FCogEngineWindow_Console*>(InData->UserData);
return ConsoleWindow.OnTextInputCallback(InData);
}
//--------------------------------------------------------------------------------------------------------------------------
@@ -214,13 +266,7 @@ void FCogEngineWindow_Console::RenderCommandList()
{
ImGui::PushID(Index);
const FString& CommandName = CommandList[Index];
RenderCommand(CommandName, Index);
if (SelectedCommandIndex == Index)
{
SelectedCommand = CommandName;
}
ImGui::PopID();
}
}
@@ -272,29 +318,39 @@ void FCogEngineWindow_Console::RenderCommand(const FString& CommandName, const i
{
ImGui::PushStyleColor(ImGuiCol_Text, FCogImguiHelper::ToImVec4(Config->HistoryColor));
}
ImGuiSelectableFlags Flags =
ImGuiSelectableFlags_AllowDoubleClick // Double click executes the selected command
| ImGuiSelectableFlags_SelectOnClick; // Need to focus the console text input right away, otherwise the Selectable take back the focus on mouse release
if (ImGui::Selectable(CommandNameStr.Get(), &IsSelected, ImGuiSelectableFlags_AllowDoubleClick))
if (ImGui::Selectable(CommandNameStr.Get(), &IsSelected, Flags))
{
SelectedCommandIndex = Index;
CurrentUserInput = CommandName;
bRequestTextInputFocus = true;
bSetBufferToSelectedCommand = true;
if (ImGui::IsMouseDoubleClicked(ImGuiMouseButton_Left))
{
ExecuteCommand(CommandName);
}
}
if (Index < NumHistoryCommands)
{
ImGui::PopStyleColor();
}
}
if (NumHistoryCommands > 0 && Index == NumHistoryCommands - 1)
{
ImGui::Separator();
}
if (IsSelected)
{
SelectedCommand = CommandName;
if (bScrollDirection != 0)
if (bScroll)
{
ImGui::SetScrollHereY(1.0f);
bScrollDirection = 0;
bScroll = false;
}
}
@@ -369,36 +425,10 @@ void FCogEngineWindow_Console::RefreshCommandList()
CommandList.Append(Commands);
SelectedCommandIndex = 0;
}
//--------------------------------------------------------------------------------------------------------------------------
int FCogEngineWindow_Console::OnTextInputCallback(ImGuiInputTextCallbackData* InData)
{
if (InData->EventFlag == ImGuiInputTextFlags_CallbackCompletion)
{
FString CleanupCommand = SelectedCommand.TrimEnd();
const auto& CommandStr = StringCast<ANSICHAR>(*CleanupCommand);
InData->DeleteChars(0, InData->BufTextLen);
InData->InsertChars(0, CommandStr.Get());
InData->InsertChars(InData->CursorPos, " ");
TextInputState = ImGuiInputTextFlags_CallbackCompletion;
}
else if (InData->EventFlag == ImGuiInputTextFlags_CallbackEdit)
{
CurrentUserInput = FString(InData->Buf);
RefreshCommandList();
TextInputState = ImGuiInputTextFlags_CallbackCompletion;
}
return 0;
}
//--------------------------------------------------------------------------------------------------------------------------
int FCogEngineWindow_Console::OnTextInputCallbackStub(ImGuiInputTextCallbackData* InData)
{
FCogEngineWindow_Console& ConsoleWindow = *static_cast<FCogEngineWindow_Console*>(InData->UserData);
return ConsoleWindow.OnTextInputCallback(InData);
//-------------------------------------------------------------------------------------
// Reset to -1 so the next down arrow will select the first entry in history/command
//-------------------------------------------------------------------------------------
SelectedCommandIndex = -1;
}
//--------------------------------------------------------------------------------------------------------------------------
@@ -415,6 +445,11 @@ void FCogEngineWindow_Console::ExecuteCommand(const FString& InCommand)
GEngine->DeferredCommands.Add(CleanupCommand);
}
if (bIsRenderingWidget)
{
bPopupCommandListOnWidgetMode = false;
}
CurrentUserInput = FString();
RefreshCommandList();
}
@@ -422,52 +457,50 @@ void FCogEngineWindow_Console::ExecuteCommand(const FString& InCommand)
//--------------------------------------------------------------------------------------------------------------------------
void FCogEngineWindow_Console::RenderMainMenuWidget()
{
bIsRenderingWidget = true;
const ImGuiContext& g = *GImGui;
const ImGuiStyle& Style = g.Style;
const ImGuiWindow* Window = ImGui::GetCurrentWindow();
ImVec2 TooltipPos = Window->DC.CursorPos;
TooltipPos.y += Window->MenuBarHeight;
const float Width = ImGui::GetFontSize() * 20;
ImGui::SetNextItemWidth(Width);
ImGui::SetNextItemWidth(-1);
RenderConsoleTextInput();
const float Width = ImGui::GetItemRectSize().x;
const bool IsTextInputActive = ImGui::IsItemActive();
if (ImGui::IsItemActivated())
if (ImGui::IsWindowAppearing())
{
bShowCommandsInWidget = true;
bRequestTextInputFocus = true;
}
if (bShowCommandsInWidget)
const bool IsTextInputActive = ImGui::IsItemActive();
if (bPopupCommandListOnWidgetMode)
{
ImGui::SetNextWindowBgAlpha(g.Style.Colors[ImGuiCol_PopupBg].w * 0.60f);
ImGui::SetNextWindowSize(ImVec2(Width, ImGui::GetFontSize() * 30), ImGuiCond_Always);
ImGui::SetNextWindowBgAlpha(g.Style.Colors[ImGuiCol_PopupBg].w * 0.90f);
ImGui::SetNextWindowSize(ImVec2(Width, ImGui::GetFontSize() * 30), ImGuiCond_FirstUseEver);
ImGui::SetNextWindowPos(TooltipPos, ImGuiCond_Always);
ImGuiWindowFlags flags =
ImGuiWindowFlags_NoTitleBar
//| ImGuiWindowFlags_NoNav
| ImGuiWindowFlags_NoMove;
//| ImGuiWindowFlags_NoBringToFrontOnFocus
//| ImGuiWindowFlags_NoFocusOnAppearing;
| ImGuiWindowFlags_NoMove
| ImGuiWindowFlags_NoFocusOnAppearing; // We want the console input text to keep the focus.
if (ImGui::Begin("ConsoleTooltip", 0, flags))
if (ImGui::Begin("ConsoleTooltip", nullptr, flags))
{
RenderCommandList();
HandleInputs();
RenderCommandList();
const bool IsWindowFocused = ImGui::IsWindowFocused(ImGuiFocusedFlags_RootAndChildWindows);
if (IsWindowFocused)
const bool IsWindowFocused = ImGui::IsWindowFocused(ImGuiFocusedFlags_RootWindow | ImGuiFocusedFlags_NoPopupHierarchy);
if (IsTextInputActive == false && IsWindowFocused == false && bRequestTextInputFocus == false)
{
bRequestTextInputFocus = true;
}
if (IsTextInputActive == false && IsWindowFocused == false)
{
bShowCommandsInWidget = false;
bPopupCommandListOnWidgetMode = false;
}
}
ImGui::End();
}
bIsRenderingWidget = false;
}
@@ -31,7 +31,6 @@ private:
static FString GetConsoleCommandHelp(const FString& InCommandName);
void RenderConsoleTextInput();
void HandleInputs();
void RenderCommand(const FString& CommandName, int32 Index);
@@ -41,27 +40,27 @@ private:
void ExecuteCommand(const FString& InCommand);
int32 SelectedCommandIndex = 0;
int32 SelectedCommandIndex = -1;
FString SelectedCommand;
TArray<FString> CommandList;
int32 NumHistoryCommands = 0;
FString CurrentUserInput;
int32 bScrollDirection = 0;
bool bScroll = false;
bool bRequestTextInputFocus = false;
bool bIsWindowFocused = false;
bool bShowCommandsInWidget = false;
bool bPopupCommandListOnWidgetMode = false;
bool bIsRenderingWidget = false;
bool bSetBufferToSelectedCommand = false;
TWeakObjectPtr<UCogEngineConfig_Console> Config;
ImGuiInputTextFlags TextInputState;
};
//--------------------------------------------------------------------------------------------------------------------------
@@ -372,7 +372,16 @@ void UCogWindowManager::RenderMainMenu()
IsRenderingInMainMenu = true;
if (ImGui::BeginMainMenuBar())
//-----------------------------------------------------------------------------------------------
// Prevent having a small gap on the right of the main menu, where some widgets are displayed
//-----------------------------------------------------------------------------------------------
ImGui::PushStyleVarX(ImGuiStyleVar_WindowPadding, 0.0f);
const bool ShowMainMenu = ImGui::BeginMainMenuBar();
ImGui::PopStyleVar();
if (ShowMainMenu)
{
for (FMenu& Menu : MainMenu.SubMenus)
{
@@ -406,42 +415,46 @@ void UCogWindowManager::RenderMainMenu()
ImGui::EndMenu();
}
RenderWidgets();
ImGui::EndMainMenuBar();
}
}
IsRenderingInMainMenu = false;
}
//--------------------------------------------------------------------------------------------------------------------------
void UCogWindowManager::RenderWidgets()
{
int32 numVisibleWidgets = 0;
int32 NumVisibleWidgets = 0;
for (int i = 0; i < Widgets.Num(); ++i)
{
FCogWindow* Window = Widgets[i];
if (Window->GetIsWidgetVisible())
{
numVisibleWidgets++;
NumVisibleWidgets++;
}
}
if (numVisibleWidgets == 0)
if (NumVisibleWidgets == 0)
{ return; }
int32 numColumns = numVisibleWidgets;
const bool AddLeftColumn = Settings->WidgetAlignment == ECogWidgetAlignment::Right
|| Settings->WidgetAlignment == ECogWidgetAlignment::Manual
|| Settings->WidgetAlignment == ECogWidgetAlignment::Center;
if (Settings->WidgetAlignment == ECogWidgetAlignment::Right)
{
numColumns++;
}
else if (Settings->WidgetAlignment == ECogWidgetAlignment::Center)
{
numColumns += 2;
}
const bool AddRightColumn = Settings->WidgetAlignment == ECogWidgetAlignment::Center;
int32 NumColumns = NumVisibleWidgets;
if (AddLeftColumn)
{
NumColumns++;
}
if (AddRightColumn)
{
NumColumns ++;
}
ImGuiTableFlags Flags = ImGuiTableFlags_None;
if (Settings->ShowWidgetBorders)
@@ -461,27 +474,29 @@ void UCogWindowManager::RenderWidgets()
Flags |= ImGuiTableFlags_NoBordersInBodyUntilResize;
}
}
ImGui::PushStyleVarX(ImGuiStyleVar_CellPadding, 0.0f);
if (ImGui::BeginTable("Widgets", numColumns, Flags))
if (ImGui::BeginTable("Widgets", NumColumns, Flags))
{
if (Settings->WidgetAlignment == ECogWidgetAlignment::Right || Settings->WidgetAlignment == ECogWidgetAlignment::Center)
if (AddLeftColumn)
{
ImGui::TableSetupColumn("Stretch", ImGuiTableColumnFlags_WidthStretch);
}
for (int i = 0; i < numVisibleWidgets; ++i)
for (int i = 0; i < NumVisibleWidgets; ++i)
{
ImGui::TableSetupColumn("Fixed", ImGuiTableColumnFlags_WidthFixed);
}
if (Settings->WidgetAlignment == ECogWidgetAlignment::Center)
if (AddRightColumn)
{
ImGui::TableSetupColumn("Stretch", ImGuiTableColumnFlags_WidthStretch);
}
ImGui::TableNextRow();
if (Settings->WidgetAlignment == ECogWidgetAlignment::Right || Settings->WidgetAlignment == ECogWidgetAlignment::Center)
if (AddLeftColumn)
{
ImGui::TableNextColumn();
}
@@ -498,13 +513,15 @@ void UCogWindowManager::RenderWidgets()
Window->RenderMainMenuWidget();
}
if (Settings->WidgetAlignment == ECogWidgetAlignment::Center)
if (AddRightColumn)
{
ImGui::TableNextColumn();
}
ImGui::EndTable();
}
ImGui::PopStyleVar();
}
//--------------------------------------------------------------------------------------------------------------------------