mirror of
https://github.com/Drezil/imgui.git
synced 2024-11-24 20:57:00 +00:00
parent
b8e56dce83
commit
7374b96f5c
@ -41,6 +41,7 @@ Breaking Changes:
|
|||||||
- Reworked IO mouse input API: (#4858) [@thedmd, @ocornut]
|
- Reworked IO mouse input API: (#4858) [@thedmd, @ocornut]
|
||||||
- Added io.AddMousePosEvent(), AddMouseButtonEvent(), AddMouseWheelEvent() functions,
|
- Added io.AddMousePosEvent(), AddMouseButtonEvent(), AddMouseWheelEvent() functions,
|
||||||
obsoleting writing directly to io.MousePos, io.MouseDown[], io.MouseWheel, etc.
|
obsoleting writing directly to io.MousePos, io.MouseDown[], io.MouseWheel, etc.
|
||||||
|
- This enable input queue trickling to support low framerates. (#2787, #1992, #3383, #2525, #1320)
|
||||||
- Reworked IO keyboard input API: (#2625, #3724) [@thedmd, @ocornut]
|
- Reworked IO keyboard input API: (#2625, #3724) [@thedmd, @ocornut]
|
||||||
- Added io.AddKeyEvent() function, obsoleting writing directly to io.KeyMap[], io.KeysDown[] arrays.
|
- Added io.AddKeyEvent() function, obsoleting writing directly to io.KeyMap[], io.KeysDown[] arrays.
|
||||||
- Added io.AddKeyModsEvent() function, obsoleting writing directly to io.KeyCtrl, io.KeyShift etc.
|
- Added io.AddKeyModsEvent() function, obsoleting writing directly to io.KeyCtrl, io.KeyShift etc.
|
||||||
@ -94,6 +95,9 @@ Breaking Changes:
|
|||||||
|
|
||||||
Other Changes:
|
Other Changes:
|
||||||
|
|
||||||
|
- IO: Added event based input queue API, which now trickles events to support low framerates. [@thedmd, @ocornut]
|
||||||
|
Previously the most common issue case (button presses in low framerates) was handled by backend. This is now
|
||||||
|
handled by core automatically for all kind of inputs. (#4858, #2787, #1992, #3383, #2525, #1320)
|
||||||
- Fixed a situation where CTRL+Tab or Modal can occasionally lead to the creation of ImDrawCmd with zero triangles,
|
- Fixed a situation where CTRL+Tab or Modal can occasionally lead to the creation of ImDrawCmd with zero triangles,
|
||||||
which would makes the draw operation of some backends assert (e.g. Metal with debugging). (#4857)
|
which would makes the draw operation of some backends assert (e.g. Metal with debugging). (#4857)
|
||||||
- Tables, ImDrawListSplitter: Fixed erroneously stripping trailing ImDrawList::AddCallback() when submitted in
|
- Tables, ImDrawListSplitter: Fixed erroneously stripping trailing ImDrawList::AddCallback() when submitted in
|
||||||
|
219
imgui.cpp
219
imgui.cpp
@ -1136,6 +1136,7 @@ ImGuiIO::ImGuiIO()
|
|||||||
#else
|
#else
|
||||||
ConfigMacOSXBehaviors = false;
|
ConfigMacOSXBehaviors = false;
|
||||||
#endif
|
#endif
|
||||||
|
ConfigInputEventQueue = true;
|
||||||
ConfigInputTextCursorBlink = true;
|
ConfigInputTextCursorBlink = true;
|
||||||
ConfigWindowsResizeFromEdges = true;
|
ConfigWindowsResizeFromEdges = true;
|
||||||
ConfigWindowsMoveFromTitleBarOnly = false;
|
ConfigWindowsMoveFromTitleBarOnly = false;
|
||||||
@ -1162,10 +1163,19 @@ ImGuiIO::ImGuiIO()
|
|||||||
// Pass in translated ASCII characters for text input.
|
// Pass in translated ASCII characters for text input.
|
||||||
// - with glfw you can get those from the callback set in glfwSetCharCallback()
|
// - with glfw you can get those from the callback set in glfwSetCharCallback()
|
||||||
// - on Windows you can get those using ToAscii+keyboard state, or via the WM_CHAR message
|
// - on Windows you can get those using ToAscii+keyboard state, or via the WM_CHAR message
|
||||||
|
// FIXME: Should in theory be called "AddCharacterEvent()" to be consistent with new API
|
||||||
void ImGuiIO::AddInputCharacter(unsigned int c)
|
void ImGuiIO::AddInputCharacter(unsigned int c)
|
||||||
{
|
{
|
||||||
if (c != 0)
|
ImGuiContext& g = *GImGui;
|
||||||
InputQueueCharacters.push_back(c <= IM_UNICODE_CODEPOINT_MAX ? (ImWchar)c : IM_UNICODE_CODEPOINT_INVALID);
|
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
||||||
|
if (c == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
ImGuiInputEvent e;
|
||||||
|
e.Type = ImGuiInputEventType_Char;
|
||||||
|
e.Source = ImGuiInputSource_Keyboard;
|
||||||
|
e.Text.Char = c;
|
||||||
|
g.InputEventsQueue.push_back(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
// UTF16 strings use surrogate pairs to encode codepoints >= 0x10000, so
|
// UTF16 strings use surrogate pairs to encode codepoints >= 0x10000, so
|
||||||
@ -1178,7 +1188,7 @@ void ImGuiIO::AddInputCharacterUTF16(ImWchar16 c)
|
|||||||
if ((c & 0xFC00) == 0xD800) // High surrogate, must save
|
if ((c & 0xFC00) == 0xD800) // High surrogate, must save
|
||||||
{
|
{
|
||||||
if (InputQueueSurrogate != 0)
|
if (InputQueueSurrogate != 0)
|
||||||
InputQueueCharacters.push_back(IM_UNICODE_CODEPOINT_INVALID);
|
AddInputCharacter(IM_UNICODE_CODEPOINT_INVALID);
|
||||||
InputQueueSurrogate = c;
|
InputQueueSurrogate = c;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -1188,7 +1198,7 @@ void ImGuiIO::AddInputCharacterUTF16(ImWchar16 c)
|
|||||||
{
|
{
|
||||||
if ((c & 0xFC00) != 0xDC00) // Invalid low surrogate
|
if ((c & 0xFC00) != 0xDC00) // Invalid low surrogate
|
||||||
{
|
{
|
||||||
InputQueueCharacters.push_back(IM_UNICODE_CODEPOINT_INVALID);
|
AddInputCharacter(IM_UNICODE_CODEPOINT_INVALID);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -1201,7 +1211,7 @@ void ImGuiIO::AddInputCharacterUTF16(ImWchar16 c)
|
|||||||
|
|
||||||
InputQueueSurrogate = 0;
|
InputQueueSurrogate = 0;
|
||||||
}
|
}
|
||||||
InputQueueCharacters.push_back(cp);
|
AddInputCharacter((unsigned)cp);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ImGuiIO::AddInputCharactersUTF8(const char* utf8_chars)
|
void ImGuiIO::AddInputCharactersUTF8(const char* utf8_chars)
|
||||||
@ -1211,7 +1221,7 @@ void ImGuiIO::AddInputCharactersUTF8(const char* utf8_chars)
|
|||||||
unsigned int c = 0;
|
unsigned int c = 0;
|
||||||
utf8_chars += ImTextCharFromUtf8(&c, utf8_chars, NULL);
|
utf8_chars += ImTextCharFromUtf8(&c, utf8_chars, NULL);
|
||||||
if (c != 0)
|
if (c != 0)
|
||||||
InputQueueCharacters.push_back((ImWchar)c);
|
AddInputCharacter(c);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1240,12 +1250,13 @@ void ImGuiIO::ClearInputKeys()
|
|||||||
// Queue a new key down/up event.
|
// Queue a new key down/up event.
|
||||||
// - ImGuiKey key: Translated key (as in, generally ImGuiKey_A matches the key end-user would use to emit an 'A' character)
|
// - ImGuiKey key: Translated key (as in, generally ImGuiKey_A matches the key end-user would use to emit an 'A' character)
|
||||||
// - bool down: Is the key down? use false to signify a key release.
|
// - bool down: Is the key down? use false to signify a key release.
|
||||||
// FIXME: In the current version this is setting key data immediately. This will evolve into a trickling queue.
|
|
||||||
void ImGuiIO::AddKeyEvent(ImGuiKey key, bool down)
|
void ImGuiIO::AddKeyEvent(ImGuiKey key, bool down)
|
||||||
{
|
{
|
||||||
//if (e->Down) { IMGUI_DEBUG_LOG("AddKeyEvent() Key='%s' %d, NativeKeycode = %d, NativeScancode = %d\n", ImGui::GetKeyName(e->Key), e->Down, e->NativeKeycode, e->NativeScancode); }
|
//if (e->Down) { IMGUI_DEBUG_LOG("AddKeyEvent() Key='%s' %d, NativeKeycode = %d, NativeScancode = %d\n", ImGui::GetKeyName(e->Key), e->Down, e->NativeKeycode, e->NativeScancode); }
|
||||||
if (key == ImGuiKey_None)
|
if (key == ImGuiKey_None)
|
||||||
return;
|
return;
|
||||||
|
ImGuiContext& g = *GImGui;
|
||||||
|
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
||||||
IM_ASSERT(ImGui::IsNamedKey(key)); // Backend needs to pass a valid ImGuiKey_ constant. 0..511 values are legacy native key codes which are not accepted by this API.
|
IM_ASSERT(ImGui::IsNamedKey(key)); // Backend needs to pass a valid ImGuiKey_ constant. 0..511 values are legacy native key codes which are not accepted by this API.
|
||||||
|
|
||||||
// Verify that backend isn't mixing up using new io.AddKeyEvent() api and old io.KeysDown[] + io.KeyMap[] data.
|
// Verify that backend isn't mixing up using new io.AddKeyEvent() api and old io.KeysDown[] + io.KeyMap[] data.
|
||||||
@ -1254,12 +1265,15 @@ void ImGuiIO::AddKeyEvent(ImGuiKey key, bool down)
|
|||||||
if (BackendUsingLegacyKeyArrays == -1)
|
if (BackendUsingLegacyKeyArrays == -1)
|
||||||
for (int n = ImGuiKey_NamedKey_BEGIN; n < ImGuiKey_NamedKey_END; n++)
|
for (int n = ImGuiKey_NamedKey_BEGIN; n < ImGuiKey_NamedKey_END; n++)
|
||||||
IM_ASSERT(KeyMap[n] == -1 && "Backend needs to either only use io.AddKeyEvent(), either only fill legacy io.KeysDown[] + io.KeyMap[]. Not both!");
|
IM_ASSERT(KeyMap[n] == -1 && "Backend needs to either only use io.AddKeyEvent(), either only fill legacy io.KeysDown[] + io.KeyMap[]. Not both!");
|
||||||
#endif
|
|
||||||
BackendUsingLegacyKeyArrays = 0;
|
BackendUsingLegacyKeyArrays = 0;
|
||||||
|
#endif
|
||||||
|
|
||||||
// Write key
|
ImGuiInputEvent e;
|
||||||
const int keydata_index = (key - ImGuiKey_KeysData_OFFSET);
|
e.Type = ImGuiInputEventType_Key;
|
||||||
KeysData[keydata_index].Down = down;
|
e.Source = ImGuiInputSource_Keyboard;
|
||||||
|
e.Key.Key = key;
|
||||||
|
e.Key.Down = down;
|
||||||
|
g.InputEventsQueue.push_back(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
// [Optional] Call after AddKeyEvent().
|
// [Optional] Call after AddKeyEvent().
|
||||||
@ -1287,11 +1301,14 @@ void ImGuiIO::SetKeyEventNativeData(ImGuiKey key, int native_keycode, int native
|
|||||||
|
|
||||||
void ImGuiIO::AddKeyModsEvent(ImGuiKeyModFlags modifiers)
|
void ImGuiIO::AddKeyModsEvent(ImGuiKeyModFlags modifiers)
|
||||||
{
|
{
|
||||||
KeyMods = modifiers;
|
ImGuiContext& g = *GImGui;
|
||||||
KeyCtrl = (modifiers & ImGuiKeyModFlags_Ctrl) != 0;
|
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
||||||
KeyShift = (modifiers & ImGuiKeyModFlags_Shift) != 0;
|
|
||||||
KeyAlt = (modifiers & ImGuiKeyModFlags_Alt) != 0;
|
ImGuiInputEvent e;
|
||||||
KeySuper = (modifiers & ImGuiKeyModFlags_Super) != 0;
|
e.Type = ImGuiInputEventType_KeyMods;
|
||||||
|
e.Source = ImGuiInputSource_Keyboard;
|
||||||
|
e.KeyMods.Mods = modifiers;
|
||||||
|
g.InputEventsQueue.push_back(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Queue a mouse move event
|
// Queue a mouse move event
|
||||||
@ -1299,7 +1316,13 @@ void ImGuiIO::AddMousePosEvent(float x, float y)
|
|||||||
{
|
{
|
||||||
ImGuiContext& g = *GImGui;
|
ImGuiContext& g = *GImGui;
|
||||||
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
||||||
g.IO.MousePos = ImVec2(x, y);
|
|
||||||
|
ImGuiInputEvent e;
|
||||||
|
e.Type = ImGuiInputEventType_MousePos;
|
||||||
|
e.Source = ImGuiInputSource_Mouse;
|
||||||
|
e.MousePos.PosX = x;
|
||||||
|
e.MousePos.PosY = y;
|
||||||
|
g.InputEventsQueue.push_back(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ImGuiIO::AddMouseWheelEvent(float wheel_x, float wheel_y)
|
void ImGuiIO::AddMouseWheelEvent(float wheel_x, float wheel_y)
|
||||||
@ -1308,8 +1331,13 @@ void ImGuiIO::AddMouseWheelEvent(float wheel_x, float wheel_y)
|
|||||||
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
||||||
if (wheel_x == 0.0f && wheel_y == 0.0f)
|
if (wheel_x == 0.0f && wheel_y == 0.0f)
|
||||||
return;
|
return;
|
||||||
g.IO.MouseWheelH += wheel_x;
|
|
||||||
g.IO.MouseWheel += wheel_y;
|
ImGuiInputEvent e;
|
||||||
|
e.Type = ImGuiInputEventType_MouseWheel;
|
||||||
|
e.Source = ImGuiInputSource_Mouse;
|
||||||
|
e.MouseWheel.WheelX = wheel_x;
|
||||||
|
e.MouseWheel.WheelY = wheel_y;
|
||||||
|
g.InputEventsQueue.push_back(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ImGuiIO::AddMouseButtonEvent(int mouse_button, bool down)
|
void ImGuiIO::AddMouseButtonEvent(int mouse_button, bool down)
|
||||||
@ -1317,14 +1345,24 @@ void ImGuiIO::AddMouseButtonEvent(int mouse_button, bool down)
|
|||||||
ImGuiContext& g = *GImGui;
|
ImGuiContext& g = *GImGui;
|
||||||
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
||||||
IM_ASSERT(mouse_button >= 0 && mouse_button < ImGuiMouseButton_COUNT);
|
IM_ASSERT(mouse_button >= 0 && mouse_button < ImGuiMouseButton_COUNT);
|
||||||
g.IO.MouseDown[mouse_button] = down;
|
|
||||||
|
ImGuiInputEvent e;
|
||||||
|
e.Type = ImGuiInputEventType_MouseButton;
|
||||||
|
e.Source = ImGuiInputSource_Mouse;
|
||||||
|
e.MouseButton.Button = mouse_button;
|
||||||
|
e.MouseButton.Down = down;
|
||||||
|
g.InputEventsQueue.push_back(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ImGuiIO::AddFocusEvent(bool focused)
|
void ImGuiIO::AddFocusEvent(bool focused)
|
||||||
{
|
{
|
||||||
// We intentionally overwrite this and process in NewFrame(), in order to give a chance
|
ImGuiContext& g = *GImGui;
|
||||||
// to multi-viewports backends to queue AddFocusEvent(false),AddFocusEvent(true) in same frame.
|
IM_ASSERT(&g.IO == this && "Can only add events to current context.");
|
||||||
AppFocusLost = !focused;
|
|
||||||
|
ImGuiInputEvent e;
|
||||||
|
e.Type = ImGuiInputEventType_Focus;
|
||||||
|
e.AppFocused.Focused = focused;
|
||||||
|
g.InputEventsQueue.push_back(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
@ -4261,13 +4299,9 @@ void ImGui::NewFrame()
|
|||||||
//if (g.IO.AppFocusLost)
|
//if (g.IO.AppFocusLost)
|
||||||
// ClosePopupsExceptModals();
|
// ClosePopupsExceptModals();
|
||||||
|
|
||||||
// Clear buttons state when focus is lost
|
// Process input queue (trickle as many events as possible)
|
||||||
// (this is useful so e.g. releasing Alt after focus loss on Alt-Tab doesn't trigger the Alt menu toggle)
|
g.InputEventsTrail.resize(0);
|
||||||
if (g.IO.AppFocusLost)
|
UpdateInputEvents(g.IO.ConfigInputEventQueue);
|
||||||
{
|
|
||||||
g.IO.ClearInputKeys();
|
|
||||||
g.IO.AppFocusLost = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update keyboard input state
|
// Update keyboard input state
|
||||||
UpdateKeyboardInputs();
|
UpdateKeyboardInputs();
|
||||||
@ -7680,6 +7714,131 @@ static const char* GetInputSourceName(ImGuiInputSource source)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Process input queue
|
||||||
|
// - trickle_fast_inputs = false : process all events, turn into flattened input state (e.g. successive down/up/down/up will be lost)
|
||||||
|
// - trickle_fast_inputs = true : process as many events as possible (successive down/up/down/up will be trickled over several frames so nothing is lost) (new feature in 1.87)
|
||||||
|
void ImGui::UpdateInputEvents(bool trickle_fast_inputs)
|
||||||
|
{
|
||||||
|
ImGuiContext& g = *GImGui;
|
||||||
|
ImGuiIO& io = g.IO;
|
||||||
|
|
||||||
|
bool mouse_moved = false, mouse_wheeled = false, key_changed = false, text_inputed = false;
|
||||||
|
int mouse_button_changed = 0x00, key_mods_changed = 0x00;
|
||||||
|
ImBitArray<ImGuiKey_KeysData_SIZE> key_changed_mask;
|
||||||
|
|
||||||
|
int event_n = 0;
|
||||||
|
for (; event_n < g.InputEventsQueue.Size; event_n++)
|
||||||
|
{
|
||||||
|
const ImGuiInputEvent* e = &g.InputEventsQueue[event_n];
|
||||||
|
if (e->Type == ImGuiInputEventType_MousePos)
|
||||||
|
{
|
||||||
|
if (io.MousePos.x != e->MousePos.PosX || io.MousePos.y != e->MousePos.PosY)
|
||||||
|
{
|
||||||
|
// Trickling Rule: Stop processing queued events if we already handled a mouse button change
|
||||||
|
if (trickle_fast_inputs && (mouse_button_changed != 0 || mouse_wheeled || key_changed || key_mods_changed || text_inputed))
|
||||||
|
break;
|
||||||
|
io.MousePos = ImVec2(e->MousePos.PosX, e->MousePos.PosY);
|
||||||
|
mouse_moved = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (e->Type == ImGuiInputEventType_MouseButton)
|
||||||
|
{
|
||||||
|
const ImGuiMouseButton button = e->MouseButton.Button;
|
||||||
|
IM_ASSERT(button >= 0 && button < ImGuiMouseButton_COUNT);
|
||||||
|
if (io.MouseDown[button] != e->MouseButton.Down)
|
||||||
|
{
|
||||||
|
// Trickling Rule: Stop processing queued events if we got multiple action on the same button
|
||||||
|
if (trickle_fast_inputs && ((mouse_button_changed & (1 << button)) || mouse_wheeled))
|
||||||
|
break;
|
||||||
|
io.MouseDown[button] = e->MouseButton.Down;
|
||||||
|
mouse_button_changed |= (1 << button);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (e->Type == ImGuiInputEventType_MouseWheel)
|
||||||
|
{
|
||||||
|
if (e->MouseWheel.WheelX != 0.0f || e->MouseWheel.WheelY != 0.0f)
|
||||||
|
{
|
||||||
|
// Trickling Rule: Stop processing queued events if we got multiple action on the event
|
||||||
|
if (trickle_fast_inputs && (mouse_wheeled || mouse_button_changed != 0))
|
||||||
|
break;
|
||||||
|
io.MouseWheelH += e->MouseWheel.WheelX;
|
||||||
|
io.MouseWheel += e->MouseWheel.WheelY;
|
||||||
|
mouse_wheeled = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (e->Type == ImGuiInputEventType_Key)
|
||||||
|
{
|
||||||
|
IM_ASSERT(e->Key.Key != ImGuiKey_None);
|
||||||
|
const int keydata_index = (e->Key.Key - ImGuiKey_KeysData_OFFSET);
|
||||||
|
ImGuiKeyData* keydata = &io.KeysData[keydata_index];
|
||||||
|
if (keydata->Down != e->Key.Down)
|
||||||
|
{
|
||||||
|
// Trickling Rule: Stop processing queued events if we got multiple action on the same button
|
||||||
|
if (trickle_fast_inputs && (key_changed_mask.TestBit(keydata_index) || text_inputed || mouse_button_changed != 0))
|
||||||
|
break;
|
||||||
|
keydata->Down = e->Key.Down;
|
||||||
|
key_changed = true;
|
||||||
|
key_changed_mask.SetBit(keydata_index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (e->Type == ImGuiInputEventType_KeyMods)
|
||||||
|
{
|
||||||
|
const ImGuiKeyModFlags modifiers = e->KeyMods.Mods;
|
||||||
|
if (io.KeyMods != modifiers)
|
||||||
|
{
|
||||||
|
// Trickling Rule: Stop processing queued events if we got multiple action on the same button
|
||||||
|
ImGuiKeyModFlags modifiers_that_are_changing = (io.KeyMods ^ modifiers);
|
||||||
|
if (trickle_fast_inputs && (key_mods_changed & modifiers_that_are_changing) != 0)
|
||||||
|
break;
|
||||||
|
io.KeyMods = modifiers;
|
||||||
|
io.KeyCtrl = (modifiers & ImGuiKeyModFlags_Ctrl) != 0;
|
||||||
|
io.KeyShift = (modifiers & ImGuiKeyModFlags_Shift) != 0;
|
||||||
|
io.KeyAlt = (modifiers & ImGuiKeyModFlags_Alt) != 0;
|
||||||
|
io.KeySuper = (modifiers & ImGuiKeyModFlags_Super) != 0;
|
||||||
|
key_mods_changed |= modifiers_that_are_changing;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (e->Type == ImGuiInputEventType_Char)
|
||||||
|
{
|
||||||
|
// Trickling Rule: Stop processing queued events if keys/mouse have been interacted with
|
||||||
|
if (trickle_fast_inputs && (key_changed || mouse_button_changed != 0 || mouse_moved || mouse_wheeled))
|
||||||
|
break;
|
||||||
|
unsigned int c = e->Text.Char;
|
||||||
|
io.InputQueueCharacters.push_back(c <= IM_UNICODE_CODEPOINT_MAX ? (ImWchar)c : IM_UNICODE_CODEPOINT_INVALID);
|
||||||
|
text_inputed = true;
|
||||||
|
}
|
||||||
|
else if (e->Type == ImGuiInputEventType_Focus)
|
||||||
|
{
|
||||||
|
// We intentionally overwrite this and process lower, in order to give a chance
|
||||||
|
// to multi-viewports backends to queue AddFocusEvent(false) + AddFocusEvent(true) in same frame.
|
||||||
|
io.AppFocusLost = !e->AppFocused.Focused;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
IM_ASSERT(0 && "Unknown event!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Record trail (for domain-specific applications wanting to access a precise trail)
|
||||||
|
for (int n = 0; n < event_n; n++)
|
||||||
|
g.InputEventsTrail.push_back(g.InputEventsQueue[n]);
|
||||||
|
|
||||||
|
// Remaining events will be processed on the next frame
|
||||||
|
if (event_n == g.InputEventsQueue.Size)
|
||||||
|
g.InputEventsQueue.resize(0);
|
||||||
|
else
|
||||||
|
g.InputEventsQueue.erase(g.InputEventsQueue.Data, g.InputEventsQueue.Data + event_n);
|
||||||
|
|
||||||
|
// Clear buttons state when focus is lost
|
||||||
|
// (this is useful so e.g. releasing Alt after focus loss on Alt-Tab doesn't trigger the Alt menu toggle)
|
||||||
|
if (g.IO.AppFocusLost)
|
||||||
|
{
|
||||||
|
g.IO.ClearInputKeys();
|
||||||
|
g.IO.AppFocusLost = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// [SECTION] ERROR CHECKING
|
// [SECTION] ERROR CHECKING
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
|
3
imgui.h
3
imgui.h
@ -65,7 +65,7 @@ Index of this file:
|
|||||||
// Version
|
// Version
|
||||||
// (Integer encoded as XYYZZ for use in #if preprocessor conditionals. Work in progress versions typically starts at XYY99 then bounce up to XYY00, XYY01 etc. when release tagging happens)
|
// (Integer encoded as XYYZZ for use in #if preprocessor conditionals. Work in progress versions typically starts at XYY99 then bounce up to XYY00, XYY01 etc. when release tagging happens)
|
||||||
#define IMGUI_VERSION "1.87 WIP"
|
#define IMGUI_VERSION "1.87 WIP"
|
||||||
#define IMGUI_VERSION_NUM 18607
|
#define IMGUI_VERSION_NUM 18608
|
||||||
#define IMGUI_CHECKVERSION() ImGui::DebugCheckVersionAndDataLayout(IMGUI_VERSION, sizeof(ImGuiIO), sizeof(ImGuiStyle), sizeof(ImVec2), sizeof(ImVec4), sizeof(ImDrawVert), sizeof(ImDrawIdx))
|
#define IMGUI_CHECKVERSION() ImGui::DebugCheckVersionAndDataLayout(IMGUI_VERSION, sizeof(ImGuiIO), sizeof(ImGuiStyle), sizeof(ImVec2), sizeof(ImVec4), sizeof(ImDrawVert), sizeof(ImDrawIdx))
|
||||||
#define IMGUI_HAS_TABLE
|
#define IMGUI_HAS_TABLE
|
||||||
|
|
||||||
@ -1944,6 +1944,7 @@ struct ImGuiIO
|
|||||||
// Miscellaneous options
|
// Miscellaneous options
|
||||||
bool MouseDrawCursor; // = false // Request ImGui to draw a mouse cursor for you (if you are on a platform without a mouse cursor). Cannot be easily renamed to 'io.ConfigXXX' because this is frequently used by backend implementations.
|
bool MouseDrawCursor; // = false // Request ImGui to draw a mouse cursor for you (if you are on a platform without a mouse cursor). Cannot be easily renamed to 'io.ConfigXXX' because this is frequently used by backend implementations.
|
||||||
bool ConfigMacOSXBehaviors; // = defined(__APPLE__) // OS X style: Text editing cursor movement using Alt instead of Ctrl, Shortcuts using Cmd/Super instead of Ctrl, Line/Text Start and End using Cmd+Arrows instead of Home/End, Double click selects by word instead of selecting whole text, Multi-selection in lists uses Cmd/Super instead of Ctrl.
|
bool ConfigMacOSXBehaviors; // = defined(__APPLE__) // OS X style: Text editing cursor movement using Alt instead of Ctrl, Shortcuts using Cmd/Super instead of Ctrl, Line/Text Start and End using Cmd+Arrows instead of Home/End, Double click selects by word instead of selecting whole text, Multi-selection in lists uses Cmd/Super instead of Ctrl.
|
||||||
|
bool ConfigInputEventQueue; // = true // Enable input queue trickling: some types of events submitted during the same frame (e.g. button down + up) will be spread over multiple frames, improving interactions with low framerates.
|
||||||
bool ConfigInputTextCursorBlink; // = true // Enable blinking cursor (optional as some users consider it to be distracting).
|
bool ConfigInputTextCursorBlink; // = true // Enable blinking cursor (optional as some users consider it to be distracting).
|
||||||
bool ConfigDragClickToInputText; // = false // [BETA] Enable turning DragXXX widgets into text input with a simple mouse click-release (without moving). Not desirable on devices without a keyboard.
|
bool ConfigDragClickToInputText; // = false // [BETA] Enable turning DragXXX widgets into text input with a simple mouse click-release (without moving). Not desirable on devices without a keyboard.
|
||||||
bool ConfigWindowsResizeFromEdges; // = true // Enable resizing of windows from their edges and from the lower-left corner. This requires (io.BackendFlags & ImGuiBackendFlags_HasMouseCursors) because it needs mouse cursor feedback. (This used to be a per-window ImGuiWindowFlags_ResizeFromAnySide flag)
|
bool ConfigWindowsResizeFromEdges; // = true // Enable resizing of windows from their edges and from the lower-left corner. This requires (io.BackendFlags & ImGuiBackendFlags_HasMouseCursors) because it needs mouse cursor feedback. (This used to be a per-window ImGuiWindowFlags_ResizeFromAnySide flag)
|
||||||
|
@ -460,8 +460,10 @@ void ImGui::ShowDemoWindow(bool* p_open)
|
|||||||
}
|
}
|
||||||
ImGui::CheckboxFlags("io.ConfigFlags: NoMouseCursorChange", &io.ConfigFlags, ImGuiConfigFlags_NoMouseCursorChange);
|
ImGui::CheckboxFlags("io.ConfigFlags: NoMouseCursorChange", &io.ConfigFlags, ImGuiConfigFlags_NoMouseCursorChange);
|
||||||
ImGui::SameLine(); HelpMarker("Instruct backend to not alter mouse cursor shape and visibility.");
|
ImGui::SameLine(); HelpMarker("Instruct backend to not alter mouse cursor shape and visibility.");
|
||||||
|
ImGui::Checkbox("io.ConfigInputEventQueue", &io.ConfigInputEventQueue);
|
||||||
|
ImGui::SameLine(); HelpMarker("Enable input queue trickling: some types of events submitted during the same frame (e.g. button down + up) will be spread over multiple frames, improving interactions with low framerates.");
|
||||||
ImGui::Checkbox("io.ConfigInputTextCursorBlink", &io.ConfigInputTextCursorBlink);
|
ImGui::Checkbox("io.ConfigInputTextCursorBlink", &io.ConfigInputTextCursorBlink);
|
||||||
ImGui::SameLine(); HelpMarker("Enable blinking cursor (optional as some users consider it to be distracting)");
|
ImGui::SameLine(); HelpMarker("Enable blinking cursor (optional as some users consider it to be distracting).");
|
||||||
ImGui::Checkbox("io.ConfigDragClickToInputText", &io.ConfigDragClickToInputText);
|
ImGui::Checkbox("io.ConfigDragClickToInputText", &io.ConfigDragClickToInputText);
|
||||||
ImGui::SameLine(); HelpMarker("Enable turning DragXXX widgets into text input with a simple mouse click-release (without moving).");
|
ImGui::SameLine(); HelpMarker("Enable turning DragXXX widgets into text input with a simple mouse click-release (without moving).");
|
||||||
ImGui::Checkbox("io.ConfigWindowsResizeFromEdges", &io.ConfigWindowsResizeFromEdges);
|
ImGui::Checkbox("io.ConfigWindowsResizeFromEdges", &io.ConfigWindowsResizeFromEdges);
|
||||||
|
@ -199,8 +199,10 @@ namespace ImStb
|
|||||||
// Debug Logging for selected systems. Remove the '((void)0) //' to enable.
|
// Debug Logging for selected systems. Remove the '((void)0) //' to enable.
|
||||||
//#define IMGUI_DEBUG_LOG_POPUP IMGUI_DEBUG_LOG // Enable log
|
//#define IMGUI_DEBUG_LOG_POPUP IMGUI_DEBUG_LOG // Enable log
|
||||||
//#define IMGUI_DEBUG_LOG_NAV IMGUI_DEBUG_LOG // Enable log
|
//#define IMGUI_DEBUG_LOG_NAV IMGUI_DEBUG_LOG // Enable log
|
||||||
|
//#define IMGUI_DEBUG_LOG_IO IMGUI_DEBUG_LOG // Enable log
|
||||||
#define IMGUI_DEBUG_LOG_POPUP(...) ((void)0) // Disable log
|
#define IMGUI_DEBUG_LOG_POPUP(...) ((void)0) // Disable log
|
||||||
#define IMGUI_DEBUG_LOG_NAV(...) ((void)0) // Disable log
|
#define IMGUI_DEBUG_LOG_NAV(...) ((void)0) // Disable log
|
||||||
|
#define IMGUI_DEBUG_LOG_IO(...) ((void)0) // Disable log
|
||||||
|
|
||||||
// Static Asserts
|
// Static Asserts
|
||||||
#define IM_STATIC_ASSERT(_COND) static_assert(_COND, "")
|
#define IM_STATIC_ASSERT(_COND) static_assert(_COND, "")
|
||||||
@ -896,6 +898,19 @@ enum ImGuiPlotType
|
|||||||
ImGuiPlotType_Histogram
|
ImGuiPlotType_Histogram
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum ImGuiInputEventType
|
||||||
|
{
|
||||||
|
ImGuiInputEventType_None = 0,
|
||||||
|
ImGuiInputEventType_MousePos,
|
||||||
|
ImGuiInputEventType_MouseWheel,
|
||||||
|
ImGuiInputEventType_MouseButton,
|
||||||
|
ImGuiInputEventType_Key,
|
||||||
|
ImGuiInputEventType_KeyMods,
|
||||||
|
ImGuiInputEventType_Char,
|
||||||
|
ImGuiInputEventType_Focus,
|
||||||
|
ImGuiInputEventType_COUNT
|
||||||
|
};
|
||||||
|
|
||||||
enum ImGuiInputSource
|
enum ImGuiInputSource
|
||||||
{
|
{
|
||||||
ImGuiInputSource_None = 0,
|
ImGuiInputSource_None = 0,
|
||||||
@ -907,6 +922,34 @@ enum ImGuiInputSource
|
|||||||
ImGuiInputSource_COUNT
|
ImGuiInputSource_COUNT
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// FIXME: Structures in the union below need to be declared as anonymous unions appears to be an extension?
|
||||||
|
// Using ImVec2() would fail on Clang 'union member 'MousePos' has a non-trivial default constructor'
|
||||||
|
struct ImGuiInputEventMousePos { float PosX, PosY; };
|
||||||
|
struct ImGuiInputEventMouseWheel { float WheelX, WheelY; };
|
||||||
|
struct ImGuiInputEventMouseButton { int Button; bool Down; };
|
||||||
|
struct ImGuiInputEventKey { ImGuiKey Key; bool Down; };
|
||||||
|
struct ImGuiInputEventKeyMods { ImGuiKeyModFlags Mods; };
|
||||||
|
struct ImGuiInputEventText { unsigned int Char; };
|
||||||
|
struct ImGuiInputEventAppFocused { bool Focused; };
|
||||||
|
|
||||||
|
struct ImGuiInputEvent
|
||||||
|
{
|
||||||
|
ImGuiInputEventType Type;
|
||||||
|
ImGuiInputSource Source;
|
||||||
|
union
|
||||||
|
{
|
||||||
|
ImGuiInputEventMousePos MousePos; // if Type == ImGuiInputEventType_MousePos
|
||||||
|
ImGuiInputEventMouseWheel MouseWheel; // if Type == ImGuiInputEventType_MouseWheel
|
||||||
|
ImGuiInputEventMouseButton MouseButton; // if Type == ImGuiInputEventType_MouseButton
|
||||||
|
ImGuiInputEventKey Key; // if Type == ImGuiInputEventType_Key
|
||||||
|
ImGuiInputEventKeyMods KeyMods; // if Type == ImGuiInputEventType_Modifiers
|
||||||
|
ImGuiInputEventText Text; // if Type == ImGuiInputEventType_Text
|
||||||
|
ImGuiInputEventAppFocused AppFocused; // if Type == ImGuiInputEventType_Focus
|
||||||
|
};
|
||||||
|
|
||||||
|
ImGuiInputEvent() { memset(this, 0, sizeof(*this)); }
|
||||||
|
};
|
||||||
|
|
||||||
// FIXME-NAV: Clarify/expose various repeat delay/rate
|
// FIXME-NAV: Clarify/expose various repeat delay/rate
|
||||||
enum ImGuiInputReadMode
|
enum ImGuiInputReadMode
|
||||||
{
|
{
|
||||||
@ -1498,6 +1541,8 @@ struct ImGuiContext
|
|||||||
bool Initialized;
|
bool Initialized;
|
||||||
bool FontAtlasOwnedByContext; // IO.Fonts-> is owned by the ImGuiContext and will be destructed along with it.
|
bool FontAtlasOwnedByContext; // IO.Fonts-> is owned by the ImGuiContext and will be destructed along with it.
|
||||||
ImGuiIO IO;
|
ImGuiIO IO;
|
||||||
|
ImVector<ImGuiInputEvent> InputEventsQueue; // Input events which will be tricked/written into IO structure.
|
||||||
|
ImVector<ImGuiInputEvent> InputEventsTrail; // Past input events processed in NewFrame(). This is to allow domain-specific application to access e.g mouse/pen trail.
|
||||||
ImGuiStyle Style;
|
ImGuiStyle Style;
|
||||||
ImFont* Font; // (Shortcut) == FontStack.empty() ? IO.Font : FontStack.back()
|
ImFont* Font; // (Shortcut) == FontStack.empty() ? IO.Font : FontStack.back()
|
||||||
float FontSize; // (Shortcut) == FontBaseSize * g.CurrentWindow->FontWindowScale == window->FontSize(). Text height for current window.
|
float FontSize; // (Shortcut) == FontBaseSize * g.CurrentWindow->FontWindowScale == window->FontSize(). Text height for current window.
|
||||||
@ -2453,6 +2498,7 @@ namespace ImGui
|
|||||||
IMGUI_API void Shutdown(ImGuiContext* context); // Since 1.60 this is a _private_ function. You can call DestroyContext() to destroy the context created by CreateContext().
|
IMGUI_API void Shutdown(ImGuiContext* context); // Since 1.60 this is a _private_ function. You can call DestroyContext() to destroy the context created by CreateContext().
|
||||||
|
|
||||||
// NewFrame
|
// NewFrame
|
||||||
|
IMGUI_API void UpdateInputEvents(bool trickle_fast_inputs);
|
||||||
IMGUI_API void UpdateHoveredWindowAndCaptureFlags();
|
IMGUI_API void UpdateHoveredWindowAndCaptureFlags();
|
||||||
IMGUI_API void StartMouseMovingWindow(ImGuiWindow* window);
|
IMGUI_API void StartMouseMovingWindow(ImGuiWindow* window);
|
||||||
IMGUI_API void UpdateMouseMovingWindowNewFrame();
|
IMGUI_API void UpdateMouseMovingWindowNewFrame();
|
||||||
|
Loading…
Reference in New Issue
Block a user