From dac9266f6ff7b212ad2654ac91668ed6a35b1096 Mon Sep 17 00:00:00 2001 From: ocornut Date: Mon, 9 Mar 2015 18:26:48 +0000 Subject: [PATCH 01/44] Fixed warnings --- imgui.cpp | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 6cb7be3e..76047136 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -5548,7 +5548,7 @@ static bool InputTextFilterCharacter(unsigned int* p_char, ImGuiInputTextFlags f if (flags & ImGuiInputTextFlags_CharsUppercase) if (c >= 'a' && c <= 'z') - *p_char = (c += 'A'-'a'); + *p_char = (c += (unsigned int)('A'-'a')); if (flags & ImGuiInputTextFlags_CharsNoBlank) if (ImCharIsSpace(c)) @@ -5560,7 +5560,7 @@ static bool InputTextFilterCharacter(unsigned int* p_char, ImGuiInputTextFlags f ImGuiTextEditCallbackData callback_data; memset(&callback_data, 0, sizeof(ImGuiTextEditCallbackData)); callback_data.EventFlag = ImGuiInputTextFlags_CallbackCharFilter; - callback_data.EventChar = c; + callback_data.EventChar = (ImWchar)c; callback_data.Flags = flags; callback_data.UserData = user_data; if (callback(&callback_data) != 0) @@ -9300,6 +9300,8 @@ static void ShowExampleAppFixedOverlay(bool* opened) static void ShowExampleAppManipulatingWindowTitle(bool* opened) { + (void)opened; + // By default, Windows are uniquely identified by their title. // You can use the "##" and "###" markers to manipulate the display/ID. Read FAQ at the top of this file! From 4b1a18c2c9370a7093ca9752c38cf75ede5a85b4 Mon Sep 17 00:00:00 2001 From: ocornut Date: Mon, 9 Mar 2015 18:36:03 +0000 Subject: [PATCH 02/44] Fixed more warnings from @bkaradzic --- imgui.cpp | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 76047136..4368cc7b 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -1835,7 +1835,7 @@ void ImGui::NewFrame() // Are we using inputs? Tell user so they can capture/discard the inputs away from the rest of their application. // When clicking outside of a window we assume the click is owned by the application and won't request capture. int mouse_earliest_button_down = -1; - for (size_t i = 0; i < IM_ARRAYSIZE(g.IO.MouseDown); i++) + for (int i = 0; i < IM_ARRAYSIZE(g.IO.MouseDown); i++) { if (g.IO.MouseClicked[i]) g.IO.MouseDownOwned[i] = (g.HoveredWindow != NULL); @@ -5636,9 +5636,9 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT { // Recycle existing cursor/selection/undo stack but clamp position // Note a single mouse click will override the cursor/position immediately by calling stb_textedit_click handler. - edit_state.StbState.cursor = ImMin(edit_state.StbState.cursor, edit_state.CurLenW); - edit_state.StbState.select_start = ImMin(edit_state.StbState.select_start, edit_state.CurLenW); - edit_state.StbState.select_end = ImMin(edit_state.StbState.select_end, edit_state.CurLenW); + edit_state.StbState.cursor = ImMin(edit_state.StbState.cursor, (int)edit_state.CurLenW); + edit_state.StbState.select_start = ImMin(edit_state.StbState.select_start, (int)edit_state.CurLenW); + edit_state.StbState.select_end = ImMin(edit_state.StbState.select_end, (int)edit_state.CurLenW); } if (focus_requested_by_tab || (user_clicked && is_ctrl_down)) select_all = true; @@ -5735,7 +5735,7 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT if (g.IO.SetClipboardTextFn) { const int ib = edit_state.HasSelection() ? ImMin(edit_state.StbState.select_start, edit_state.StbState.select_end) : 0; - const int ie = edit_state.HasSelection() ? ImMax(edit_state.StbState.select_start, edit_state.StbState.select_end) : edit_state.CurLenW; + const int ie = edit_state.HasSelection() ? ImMax(edit_state.StbState.select_start, edit_state.StbState.select_end) : (int)edit_state.CurLenW; ImTextStrToUtf8(text_tmp_utf8, IM_ARRAYSIZE(text_tmp_utf8), edit_state.Text+ib, edit_state.Text+ie); g.IO.SetClipboardTextFn(text_tmp_utf8); } From 5d1c38150931b4f191d4b23dd908c80873e9860e Mon Sep 17 00:00:00 2001 From: ocornut Date: Wed, 11 Mar 2015 16:44:54 +0000 Subject: [PATCH 03/44] Update version number, added IMGUI_VERSION, ImGui::GetVersion() (#127) --- imgui.cpp | 7 ++++++- imgui.h | 5 ++++- 2 files changed, 10 insertions(+), 2 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 4368cc7b..266e898c 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -1,4 +1,4 @@ -// ImGui library v1.35 +// ImGui library v1.36 wip // See ImGui::ShowTestWindow() for sample code. // Read 'Programmer guide' below for notes on how to setup ImGui in your codebase. // Get latest version at https://github.com/ocornut/imgui @@ -1713,6 +1713,11 @@ static void MarkSettingsDirty() g.SettingsDirtyTimer = g.IO.IniSavingRate; } +const char* ImGui::GetVersion() +{ + return IMGUI_VERSION; +} + // Internal state access - if you want to share ImGui state between modules (e.g. DLL) or allocate it yourself // Note that we still point to some static data and members (such as GFontAtlas), so the state instance you end up using will point to the static data within its module void* ImGui::GetInternalState() diff --git a/imgui.h b/imgui.h index f4f36f18..6efe3673 100644 --- a/imgui.h +++ b/imgui.h @@ -1,4 +1,4 @@ -// ImGui library v1.35 +// ImGui library v1.36 wip // See .cpp file for documentation. // See ImGui::ShowTestWindow() for sample code. // Read 'Programmer guide' in .cpp for notes on how to setup ImGui in your codebase. @@ -13,6 +13,8 @@ #include // NULL, malloc #include // memset, memmove +#define IMGUI_VERSION "1.36 wip" + // Define assertion handler. #ifndef IM_ASSERT #include @@ -369,6 +371,7 @@ namespace ImGui IMGUI_API void ColorConvertHSVtoRGB(float h, float s, float v, float& out_r, float& out_g, float& out_b); // Internal state access - if you want to share ImGui state between modules (e.g. DLL) or allocate it yourself + IMGUI_API const char* GetVersion(); IMGUI_API void* GetInternalState(); IMGUI_API size_t GetInternalStateSize(); IMGUI_API void SetInternalState(void* state, bool construct = false); From 58dc3f97c54a104d52cd11bfceb69fcb52ce448e Mon Sep 17 00:00:00 2001 From: ocornut Date: Wed, 11 Mar 2015 16:45:58 +0000 Subject: [PATCH 04/44] WIP marker in uppercase.. (#127) --- imgui.cpp | 2 +- imgui.h | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 266e898c..3f509448 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -1,4 +1,4 @@ -// ImGui library v1.36 wip +// ImGui library v1.36 WIP // See ImGui::ShowTestWindow() for sample code. // Read 'Programmer guide' below for notes on how to setup ImGui in your codebase. // Get latest version at https://github.com/ocornut/imgui diff --git a/imgui.h b/imgui.h index 6efe3673..8b9b7de0 100644 --- a/imgui.h +++ b/imgui.h @@ -1,4 +1,4 @@ -// ImGui library v1.36 wip +// ImGui library v1.36 WIP // See .cpp file for documentation. // See ImGui::ShowTestWindow() for sample code. // Read 'Programmer guide' in .cpp for notes on how to setup ImGui in your codebase. @@ -13,7 +13,7 @@ #include // NULL, malloc #include // memset, memmove -#define IMGUI_VERSION "1.36 wip" +#define IMGUI_VERSION "1.36 WIP" // Define assertion handler. #ifndef IM_ASSERT From 09115264249ba04baa539e2b0823c4ffdd997e90 Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 09:49:38 +0000 Subject: [PATCH 05/44] Added io.KeyAlt + support in examples apps Currently unused but supported by sample so that more people will have it set up when menus needs them --- examples/directx11_example/imgui_impl_dx11.cpp | 1 + examples/directx11_example/main.cpp | 4 ++++ examples/directx9_example/imgui_impl_dx9.cpp | 1 + examples/directx9_example/main.cpp | 4 ++++ examples/opengl3_example/imgui_impl_glfw_gl3.cpp | 1 + examples/opengl_example/imgui_impl_glfw.cpp | 1 + imgui.cpp | 4 +++- imgui.h | 1 + 8 files changed, 16 insertions(+), 1 deletion(-) diff --git a/examples/directx11_example/imgui_impl_dx11.cpp b/examples/directx11_example/imgui_impl_dx11.cpp index 727b5eeb..f6ee6b5d 100644 --- a/examples/directx11_example/imgui_impl_dx11.cpp +++ b/examples/directx11_example/imgui_impl_dx11.cpp @@ -455,6 +455,7 @@ void ImGui_ImplDX11_NewFrame() // Read keyboard modifiers inputs io.KeyCtrl = (GetKeyState(VK_CONTROL) & 0x8000) != 0; io.KeyShift = (GetKeyState(VK_SHIFT) & 0x8000) != 0; + io.KeyAlt = (GetKeyState(VK_MENU) & 0x8000) != 0; // io.KeysDown : filled by WM_KEYDOWN/WM_KEYUP events // io.MousePos : filled by WM_MOUSEMOVE events // io.MouseDown : filled by WM_*BUTTON* events diff --git a/examples/directx11_example/main.cpp b/examples/directx11_example/main.cpp index 0620245b..571fe1da 100644 --- a/examples/directx11_example/main.cpp +++ b/examples/directx11_example/main.cpp @@ -117,6 +117,10 @@ LRESULT WINAPI WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) ImGui_ImplDX11_CreateDeviceObjects(); } return 0; + case WM_SYSCOMMAND: + if ((wParam & 0xfff0) == SC_KEYMENU) // Disable ALT application menu + return 0; + break; case WM_DESTROY: PostQuitMessage(0); return 0; diff --git a/examples/directx9_example/imgui_impl_dx9.cpp b/examples/directx9_example/imgui_impl_dx9.cpp index e47ddea9..b64b70eb 100644 --- a/examples/directx9_example/imgui_impl_dx9.cpp +++ b/examples/directx9_example/imgui_impl_dx9.cpp @@ -276,6 +276,7 @@ void ImGui_ImplDX9_NewFrame() // Read keyboard modifiers inputs io.KeyCtrl = (GetKeyState(VK_CONTROL) & 0x8000) != 0; io.KeyShift = (GetKeyState(VK_SHIFT) & 0x8000) != 0; + io.KeyAlt = (GetKeyState(VK_MENU) & 0x8000) != 0; // io.KeysDown : filled by WM_KEYDOWN/WM_KEYUP events // io.MousePos : filled by WM_MOUSEMOVE events // io.MouseDown : filled by WM_*BUTTON* events diff --git a/examples/directx9_example/main.cpp b/examples/directx9_example/main.cpp index 75892ffb..5e0e58d6 100644 --- a/examples/directx9_example/main.cpp +++ b/examples/directx9_example/main.cpp @@ -30,6 +30,10 @@ LRESULT WINAPI WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) ImGui_ImplDX9_CreateDeviceObjects(); } return 0; + case WM_SYSCOMMAND: + if ((wParam & 0xfff0) == SC_KEYMENU) // Disable ALT application menu + return 0; + break; case WM_DESTROY: PostQuitMessage(0); return 0; diff --git a/examples/opengl3_example/imgui_impl_glfw_gl3.cpp b/examples/opengl3_example/imgui_impl_glfw_gl3.cpp index adc2d673..fa669d9c 100644 --- a/examples/opengl3_example/imgui_impl_glfw_gl3.cpp +++ b/examples/opengl3_example/imgui_impl_glfw_gl3.cpp @@ -143,6 +143,7 @@ void ImGui_ImplGlfwGL3_KeyCallback(GLFWwindow* window, int key, int scancode, in io.KeysDown[key] = false; io.KeyCtrl = (mods & GLFW_MOD_CONTROL) != 0; io.KeyShift = (mods & GLFW_MOD_SHIFT) != 0; + io.KeyAlt = (mods & GLFW_MOD_ALT) != 0; } void ImGui_ImplGlfwGL3_CharCallback(GLFWwindow* window, unsigned int c) diff --git a/examples/opengl_example/imgui_impl_glfw.cpp b/examples/opengl_example/imgui_impl_glfw.cpp index 2be01de0..a6b9cb40 100644 --- a/examples/opengl_example/imgui_impl_glfw.cpp +++ b/examples/opengl_example/imgui_impl_glfw.cpp @@ -123,6 +123,7 @@ void ImGui_ImplGlFw_KeyCallback(GLFWwindow* window, int key, int scancode, int a io.KeysDown[key] = false; io.KeyCtrl = (mods & GLFW_MOD_CONTROL) != 0; io.KeyShift = (mods & GLFW_MOD_SHIFT) != 0; + io.KeyAlt = (mods & GLFW_MOD_ALT) != 0; } void ImGui_ImplGlfw_CharCallback(GLFWwindow* window, unsigned int c) diff --git a/imgui.cpp b/imgui.cpp index 3f509448..2b45781c 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -314,6 +314,7 @@ - input: support track pad style scrolling & slider edit. - tooltip: move to fit within screen (e.g. when mouse cursor is right of the screen). - portability: big-endian test/support (github issue #81) + - misc: let user copy any window content to clipboard easily (CTRL+C on windows? while moving it? context menu?) - misc: mark printf compiler attributes on relevant functions - misc: provide a way to compile out the entire implementation while providing a dummy API (e.g. #define IMGUI_DUMMY_IMPL) - misc: double-clicking on title bar to minimize isn't consistent, perhaps move to single-click on left-most collapse icon? @@ -4014,7 +4015,7 @@ static bool ButtonBehaviour(const ImGuiAabb& bb, const ImGuiID& id, bool* out_ho if (hovered) { g.HoveredId = id; - if (allow_key_modifiers || (!g.IO.KeyCtrl && !g.IO.KeyShift)) + if (allow_key_modifiers || (!g.IO.KeyCtrl && !g.IO.KeyShift && !g.IO.KeyAlt)) { if (g.IO.MouseClicked[0]) { @@ -8582,6 +8583,7 @@ void ImGui::ShowTestWindow(bool* opened) ImGui::Text("ImGui says hello."); //ImGui::Text("MousePos (%g, %g)", ImGui::GetIO().MousePos.x, ImGui::GetIO().MousePos.y); //ImGui::Text("MouseWheel %d", ImGui::GetIO().MouseWheel); + ImGui::Text("KeyMods %s%s%s", ImGui::GetIO().KeyCtrl ? "CTRL" : "", ImGui::GetIO().KeyShift ? "SHIFT" : "", ImGui::GetIO().KeyAlt? "ALT" : ""); //ImGui::Text("WantCaptureMouse: %d", ImGui::GetIO().WantCaptureMouse); //ImGui::Text("WantCaptureKeyboard: %d", ImGui::GetIO().WantCaptureKeyboard); diff --git a/imgui.h b/imgui.h index 8b9b7de0..fb24d89f 100644 --- a/imgui.h +++ b/imgui.h @@ -601,6 +601,7 @@ struct ImGuiIO bool MouseDrawCursor; // Request ImGui to draw a mouse cursor for you (if you are on a platform without a mouse cursor). bool KeyCtrl; // Keyboard modifier pressed: Control bool KeyShift; // Keyboard modifier pressed: Shift + bool KeyAlt; // Keyboard modifier pressed: Alt bool KeysDown[512]; // Keyboard keys that are pressed (in whatever storage order you naturally have access to keyboard data) ImWchar InputCharacters[16+1]; // List of characters input (translated by user from keypress+keyboard state). Fill using AddInputCharacter() helper. From e2734d23ef9c02c0bdd7783955c00785e64344ad Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 11:16:11 +0000 Subject: [PATCH 06/44] Child window don't take focus when they first appear (focus left to parent) --- imgui.cpp | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 2b45781c..53d2d9af 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -2807,8 +2807,9 @@ bool ImGui::Begin(const char* name, bool* p_opened, const ImVec2& size, float bg window->Visible = true; // New windows appears in front - if (window->LastFrameDrawn < current_frame - 1) - FocusWindow(window); + if (!(flags & ImGuiWindowFlags_ChildWindow)) + if (window->LastFrameDrawn < current_frame - 1) + FocusWindow(window); window->LastFrameDrawn = current_frame; window->ClipRectStack.resize(0); @@ -8583,7 +8584,7 @@ void ImGui::ShowTestWindow(bool* opened) ImGui::Text("ImGui says hello."); //ImGui::Text("MousePos (%g, %g)", ImGui::GetIO().MousePos.x, ImGui::GetIO().MousePos.y); //ImGui::Text("MouseWheel %d", ImGui::GetIO().MouseWheel); - ImGui::Text("KeyMods %s%s%s", ImGui::GetIO().KeyCtrl ? "CTRL" : "", ImGui::GetIO().KeyShift ? "SHIFT" : "", ImGui::GetIO().KeyAlt? "ALT" : ""); + //ImGui::Text("KeyMods %s%s%s", ImGui::GetIO().KeyCtrl ? "CTRL" : "", ImGui::GetIO().KeyShift ? "SHIFT" : "", ImGui::GetIO().KeyAlt? "ALT" : ""); //ImGui::Text("WantCaptureMouse: %d", ImGui::GetIO().WantCaptureMouse); //ImGui::Text("WantCaptureKeyboard: %d", ImGui::GetIO().WantCaptureKeyboard); From 1c1a55f6b1f8228b5ae2ea4af2d66eb6e629e931 Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 12:25:43 +0000 Subject: [PATCH 07/44] Added IsRootWindowFocused(), IsRootWindowOrAnyChildFocused(). Renamed GetWindowIsFocused() to IsWindowFocused() --- imgui.cpp | 29 ++++++++++++++++++++++------- imgui.h | 5 ++++- 2 files changed, 26 insertions(+), 8 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 53d2d9af..9e102876 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -129,8 +129,9 @@ Occasionally introducing changes that are breaking the API. The breakage are generally minor and easy to fix. Here is a change-log of API breaking changes, if you are using one of the functions listed, expect to have to fix some code. + - 2015/03/13 (1.36) - renamed GetWindowIsFocused() to IsWindowFocused(). Kept inline redirection function. - 2015/03/08 (1.35) - renamed style.ScrollBarWidth to style.ScrollbarWidth - - 2015/02/27 (1.34) - renamed OpenNextNode(bool) to SetNextTreeNodeOpened(bool, ImGuiSetCond), kept inline redirection function. + - 2015/02/27 (1.34) - renamed OpenNextNode(bool) to SetNextTreeNodeOpened(bool, ImGuiSetCond). Kept inline redirection function. - 2015/02/27 (1.34) - renamed ImGuiSetCondition_*** to ImGuiSetCond_***, and _FirstUseThisSession becomes _Once. - 2015/02/11 (1.32) - changed text input callback ImGuiTextEditCallback return type from void-->int. reserved for future use, return 0 for now. - 2015/02/10 (1.32) - renamed GetItemWidth() to CalcItemWidth() to clarify its evolving behavior @@ -1138,12 +1139,12 @@ struct ImGuiWindow bool SkipItems; // == Visible && !Collapsed int AutoFitFrames; bool AutoFitOnlyGrows; - int SetWindowPosAllowFlags; // bit ImGuiSetCond_*** specify if SetWindowPos() call is allowed with this particular flag. - int SetWindowSizeAllowFlags; // bit ImGuiSetCond_*** specify if SetWindowSize() call is allowed with this particular flag. - int SetWindowCollapsedAllowFlags; // bit ImGuiSetCond_*** specify if SetWindowCollapsed() call is allowed with this particular flag. + int SetWindowPosAllowFlags; // bit ImGuiSetCond_*** specify if SetWindowPos() call will succeed with this particular flag. + int SetWindowSizeAllowFlags; // bit ImGuiSetCond_*** specify if SetWindowSize() call will succeed with this particular flag. + int SetWindowCollapsedAllowFlags; // bit ImGuiSetCond_*** specify if SetWindowCollapsed() call will succeed with this particular flag. - ImGuiDrawContext DC; - ImVector IDStack; + ImGuiDrawContext DC; // Temporary per-window data, reset at the beginning of the frame + ImVector IDStack; // ID stack. ID are hashes seeded with the value at the top of the stack ImVector ClipRectStack; // Scissoring / clipping rectangle. x1, y1, x2, y2. ImGuiAabb ClippedAabb; // = ClipRectStack.front() after setup in Begin() int LastFrameDrawn; @@ -3481,13 +3482,27 @@ const char* ImGui::GetStyleColName(ImGuiCol idx) return "Unknown"; } -bool ImGui::GetWindowIsFocused() +bool ImGui::IsWindowFocused() { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); return g.FocusedWindow == window; } +bool ImGui::IsRootWindowFocused() +{ + ImGuiState& g = *GImGui; + ImGuiWindow* root_window = GetCurrentWindow()->RootWindow; + return g.FocusedWindow == root_window; +} + +bool ImGui::IsRootWindowOrAnyChildFocused() +{ + ImGuiState& g = *GImGui; + ImGuiWindow* root_window = GetCurrentWindow()->RootWindow; + return g.FocusedWindow->RootWindow == root_window; +} + float ImGui::GetWindowWidth() { ImGuiWindow* window = GetCurrentWindow(); diff --git a/imgui.h b/imgui.h index fb24d89f..869fde9b 100644 --- a/imgui.h +++ b/imgui.h @@ -166,7 +166,6 @@ namespace ImGui IMGUI_API bool BeginChild(const char* str_id, const ImVec2& size = ImVec2(0,0), bool border = false, ImGuiWindowFlags extra_flags = 0); // size==0.0f: use remaining window size, size<0.0f: use remaining window size minus abs(size). on each axis. IMGUI_API bool BeginChild(ImGuiID id, const ImVec2& size = ImVec2(0,0), bool border = false, ImGuiWindowFlags extra_flags = 0); // " IMGUI_API void EndChild(); - IMGUI_API bool GetWindowIsFocused(); IMGUI_API ImVec2 GetContentRegionMax(); // window or current column boundaries, in windows coordinates IMGUI_API ImVec2 GetWindowContentRegionMin(); // window boundaries, in windows coordinates IMGUI_API ImVec2 GetWindowContentRegionMax(); @@ -346,6 +345,9 @@ namespace ImGui IMGUI_API bool IsItemHovered(); // was the last item hovered by mouse? IMGUI_API bool IsItemActive(); // was the last item active? (e.g. button being held, text field being edited- items that don't interact will always return false) IMGUI_API bool IsAnyItemActive(); // + IMGUI_API bool IsWindowFocused(); // is current window focused (differentiate child windows from each others) + IMGUI_API bool IsRootWindowFocused(); // is current root window focused + IMGUI_API bool IsRootWindowOrAnyChildFocused(); // is current root window or any of its child (including current window) focused IMGUI_API ImVec2 GetItemBoxMin(); // get bounding box of last item IMGUI_API ImVec2 GetItemBoxMax(); // get bounding box of last item IMGUI_API bool IsClipped(const ImVec2& item_size); // to perform coarse clipping on user's side (as an optimization) @@ -379,6 +381,7 @@ namespace ImGui // Obsolete (will be removed) IMGUI_API void GetDefaultFontData(const void** fnt_data, unsigned int* fnt_size, const void** png_data, unsigned int* png_size); static inline void OpenNextNode(bool open) { ImGui::SetNextTreeNodeOpened(open, 0); } + static inline bool GetWindowIsFocused() { return ImGui::IsWindowFocused(); } } // namespace ImGui From 153454a0fb656a74f5790912986de71bcfe01b11 Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 12:28:23 +0000 Subject: [PATCH 08/44] Example console keeps focus on input box at all times --- imgui.cpp | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 9e102876..35b50820 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -9478,7 +9478,7 @@ struct ExampleAppConsole ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(0,0)); static ImGuiTextFilter filter; filter.Draw("Filter (\"incl,-excl\") (\"error\")", 180); - if (ImGui::IsItemHovered()) ImGui::SetKeyboardFocusHere(-1); // Auto focus on hover + //if (ImGui::IsItemHovered()) ImGui::SetKeyboardFocusHere(-1); // Auto focus on hover ImGui::PopStyleVar(); ImGui::Separator(); @@ -9516,7 +9516,9 @@ struct ExampleAppConsole strcpy(InputBuf, ""); } - if (ImGui::IsItemHovered()) ImGui::SetKeyboardFocusHere(-1); // Auto focus on hover + // Demonstrate keeping auto focus on the input box + if (ImGui::IsItemHovered() || (ImGui::IsRootWindowOrAnyChildFocused() && !ImGui::IsAnyItemActive() && !ImGui::IsMouseClicked(0))) + ImGui::SetKeyboardFocusHere(-1); // Auto focus ImGui::End(); } From 55093ea8fe0143b65ed1e1c22337c5c11016f58a Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 12:44:04 +0000 Subject: [PATCH 09/44] Separator() logs itself as text --- imgui.cpp | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/imgui.cpp b/imgui.cpp index 35b50820..3b0a6f86 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -6487,6 +6487,10 @@ void ImGui::Separator() window->DrawList->AddLine(bb.Min, bb.Max, window->Color(ImGuiCol_Border)); + ImGuiState& g = *GImGui; + if (g.LogEnabled) + ImGui::LogText(STR_NEWLINE "--------------------------------"); + if (window->DC.ColumnsCount > 1) { PushColumnClipRect(); From ceb4da20388254124394fd76e3e6987628fae76e Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 13:04:10 +0000 Subject: [PATCH 10/44] Comments --- imgui.cpp | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/imgui.cpp b/imgui.cpp index 3b0a6f86..c97b499d 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -3105,6 +3105,15 @@ bool ImGui::Begin(const char* name, bool* p_opened, const ImVec2& size, float bg // Save clipped aabb so we can access it in constant-time in FindHoveredWindow() window->ClippedAabb = window->Aabb(); window->ClippedAabb.Clip(window->ClipRectStack.front()); + + // Pressing CTRL+C while holding on a window copy its content to the clipboard + // This works but 1. doesn't handle multiple Begin/End pairs, 2. recursing into another Begin/End pair - so we need to work that out and add better logging scope. + // Maybe we can support CTRL+C on every element? + /* + if (g.ActiveId == move_id) + if (g.IO.KeyCtrl && IsKeyPressedMap(ImGuiKey_C)) + ImGui::LogToClipboard(); + */ } // Inner clipping rectangle @@ -6530,7 +6539,7 @@ static void ItemSize(ImVec2 size, ImVec2* adjust_vertical_offset) window->DC.CurrentLineHeight = 0.0f; } -static void ItemSize(const ImGuiAabb& bb, ImVec2* adjust_start_offset) +static inline void ItemSize(const ImGuiAabb& bb, ImVec2* adjust_start_offset) { ItemSize(bb.GetSize(), adjust_start_offset); } From 4d78c5c9aba58447eec046a77dd0551b4d1922a7 Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 13:15:01 +0000 Subject: [PATCH 11/44] Comments and reduced ImGuiTextEditState.InitialText[] to 3 KB (assuming average use case of UTF-8 uses 3 bytes charracters) --- imgui.cpp | 2 +- imgui.h | 3 +-- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index c97b499d..5f7bd5fa 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -958,7 +958,7 @@ struct ImGuiTextEditState { ImGuiID Id; // widget id owning the text state ImWchar Text[1024]; // edit buffer, we need to persist but can't guarantee the persistence of the user-provided buffer. so we copy into own buffer. - char InitialText[1024*4+1]; // backup of end-user buffer at the time of focus (in UTF-8, unaltered) + char InitialText[1024*3+1]; // backup of end-user buffer at the time of focus (in UTF-8, unaltered) size_t CurLenA, CurLenW; // we need to maintain our buffer length in both UTF-8 and wchar format. size_t BufSizeA; // end-user buffer size, <= 1024 (or increase above) float Width; // widget width diff --git a/imgui.h b/imgui.h index 869fde9b..0ad58bf1 100644 --- a/imgui.h +++ b/imgui.h @@ -693,7 +693,6 @@ struct ImGuiTextBuffer ImVector Buf; ImGuiTextBuffer() { Buf.push_back(0); } - ~ImGuiTextBuffer() { } const char* begin() const { return &Buf.front(); } const char* end() const { return &Buf.back(); } // Buf is zero-terminated, so end() will point on the zero-terminator size_t size() const { return Buf.size()-1; } @@ -902,7 +901,7 @@ struct ImFontAtlas IMGUI_API void SetTexID(void* id) { TexID = id; } // Helpers to retrieve list of common Unicode ranges (2 value per range, values are inclusive, zero-terminated list) - // (Those functions could be static, aren't so simple use case doesn't have to refer to the ImFontAtlas:: type ever if in their code) + // (Those functions could be static but aren't so most users don't have to refer to the ImFontAtlas:: name ever if in their code; just using io.Fonts->) IMGUI_API const ImWchar* GetGlyphRangesDefault(); // Basic Latin, Extended Latin IMGUI_API const ImWchar* GetGlyphRangesJapanese(); // Default + Hiragana, Katakana, Half-Width, Selection of 1946 Ideographs IMGUI_API const ImWchar* GetGlyphRangesChinese(); // Japanese + full set of about 21000 CJK Unified Ideographs From aaca73de156a92ad28620c9aea96eff23844cbc0 Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 13:23:03 +0000 Subject: [PATCH 12/44] Refactor SliderFloat() internals into a helper SliderFloatAsInputText() + some comments --- imgui.cpp | 61 ++++++++++++++++++++++++++++++++++--------------------- 1 file changed, 38 insertions(+), 23 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 5f7bd5fa..4c18cefb 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4680,8 +4680,45 @@ static void ApplyNumericalTextInput(const char* buf, float *v) *v = op_v; } +static bool SliderFloatAsInputText(const char* label, float* v, const ImGuiID& id, int decimal_precision) +{ + ImGuiState& g = *GImGui; + ImGuiWindow* window = GetCurrentWindow(); + + char text_buf[64]; + ImFormatString(text_buf, IM_ARRAYSIZE(text_buf), "%.*f", decimal_precision, *v); + + SetActiveId(g.SliderAsInputTextId); + g.HoveredId = 0; + + // Our replacement widget will override the focus ID (registered previously to allow for a TAB focus to happen) + window->FocusItemUnregister(); + + bool value_changed = ImGui::InputText(label, text_buf, IM_ARRAYSIZE(text_buf), ImGuiInputTextFlags_CharsDecimal | ImGuiInputTextFlags_AutoSelectAll); + if (g.SliderAsInputTextId == 0) + { + // First frame + IM_ASSERT(g.ActiveId == id); // InputText ID expected to match the Slider ID (else we'd need to store them both, which is also possible) + g.SliderAsInputTextId = g.ActiveId; + g.HoveredId = id; + } + else if (g.ActiveId != g.SliderAsInputTextId) + { + // Release + g.SliderAsInputTextId = 0; + } + if (value_changed) + { + ApplyNumericalTextInput(text_buf, v); + } + return value_changed; +} + // Use power!=1.0 for logarithmic sliders. // Adjust display_format to decorate the value with a prefix or a suffix. +// "%.3f" 1.234 +// "%5.2f secs" 01.23 secs +// "Gold: %.0f" Gold: 1 bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, const char* display_format, float power) { ImGuiState& g = *GImGui; @@ -4777,29 +4814,7 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c bool value_changed = false; if (start_text_input || (g.ActiveId == id && id == g.SliderAsInputTextId)) { - char text_buf[64]; - ImFormatString(text_buf, IM_ARRAYSIZE(text_buf), "%.*f", decimal_precision, *v); - - SetActiveId(g.SliderAsInputTextId); - g.HoveredId = 0; - window->FocusItemUnregister(); // Our replacement slider will override the focus ID (registered previously to allow for a TAB focus to happen) - value_changed = ImGui::InputText(label, text_buf, IM_ARRAYSIZE(text_buf), ImGuiInputTextFlags_CharsDecimal | ImGuiInputTextFlags_AutoSelectAll); - if (g.SliderAsInputTextId == 0) - { - // First frame - IM_ASSERT(g.ActiveId == id); // InputText ID expected to match the Slider ID (else we'd need to store them both, which is also possible) - g.SliderAsInputTextId = g.ActiveId; - g.HoveredId = id; - } - else if (g.ActiveId != g.SliderAsInputTextId) - { - // Release - g.SliderAsInputTextId = 0; - } - if (value_changed) - { - ApplyNumericalTextInput(text_buf, v); - } + value_changed = SliderFloatAsInputText(label, v, id, decimal_precision); return value_changed; } From 110d96034b5d04b28c37faa498c43db5f7014085 Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 13:28:30 +0000 Subject: [PATCH 13/44] SliderFloat() fast-path when power=1.0f (no powf() calls) also makes code easier to read --- imgui.cpp | 63 ++++++++++++++++++++++++++++++++----------------------- 1 file changed, 37 insertions(+), 26 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 4c18cefb..e29e3185 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4763,6 +4763,7 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c const bool tab_focus_requested = window->FocusItemRegister(g.ActiveId == id); const bool is_unbound = v_min == -FLT_MAX || v_min == FLT_MAX || v_max == -FLT_MAX || v_max == FLT_MAX; + const bool is_logarithmic = abs(power - 1.0f) > 0.0001f; const float grab_size_in_units = 1.0f; // In 'v' units. Probably needs to be parametrized, based on a 'v_step' value? decimal precision? float grab_size_in_pixels; @@ -4830,28 +4831,33 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c { const float normalized_pos = ImClamp((g.IO.MousePos.x - slider_effective_x1) / slider_effective_w, 0.0f, 1.0f); - // Linear slider - //float new_value = ImLerp(v_min, v_max, normalized_pos); - - // Account for logarithmic scale on both sides of the zero float new_value; - if (normalized_pos < linear_zero_pos) + if (is_logarithmic) { - // Negative: rescale to the negative range before powering - float a = 1.0f - (normalized_pos / linear_zero_pos); - a = powf(a, power); - new_value = ImLerp(ImMin(v_max,0.f), v_min, a); + // Account for logarithmic scale on both sides of the zero + if (normalized_pos < linear_zero_pos) + { + // Negative: rescale to the negative range before powering + float a = 1.0f - (normalized_pos / linear_zero_pos); + a = powf(a, power); + new_value = ImLerp(ImMin(v_max,0.f), v_min, a); + } + else + { + // Positive: rescale to the positive range before powering + float a; + if (fabsf(linear_zero_pos - 1.0f) > 1.e-6) + a = (normalized_pos - linear_zero_pos) / (1.0f - linear_zero_pos); + else + a = normalized_pos; + a = powf(a, power); + new_value = ImLerp(ImMax(v_min,0.0f), v_max, a); + } } else { - // Positive: rescale to the positive range before powering - float a; - if (fabsf(linear_zero_pos - 1.0f) > 1.e-6) - a = (normalized_pos - linear_zero_pos) / (1.0f - linear_zero_pos); - else - a = normalized_pos; - a = powf(a, power); - new_value = ImLerp(ImMax(v_min,0.0f), v_max, a); + // Linear slider + new_value = ImLerp(v_min, v_max, normalized_pos); } // Round past decimal precision @@ -4879,21 +4885,26 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c if (!is_unbound) { - // Linear slider - // const float grab_t = (ImClamp(*v, v_min, v_max) - v_min) / (v_max - v_min); - // Calculate slider grab positioning float grab_t; - float v_clamped = ImClamp(*v, v_min, v_max); - if (v_clamped < 0.0f) + if (is_logarithmic) { - const float f = 1.0f - (v_clamped - v_min) / (ImMin(0.0f,v_max) - v_min); - grab_t = (1.0f - powf(f, 1.0f/power)) * linear_zero_pos; + float v_clamped = ImClamp(*v, v_min, v_max); + if (v_clamped < 0.0f) + { + const float f = 1.0f - (v_clamped - v_min) / (ImMin(0.0f,v_max) - v_min); + grab_t = (1.0f - powf(f, 1.0f/power)) * linear_zero_pos; + } + else + { + const float f = (v_clamped - ImMax(0.0f,v_min)) / (v_max - ImMax(0.0f,v_min)); + grab_t = linear_zero_pos + powf(f, 1.0f/power) * (1.0f - linear_zero_pos); + } } else { - const float f = (v_clamped - ImMax(0.0f,v_min)) / (v_max - ImMax(0.0f,v_min)); - grab_t = linear_zero_pos + powf(f, 1.0f/power) * (1.0f - linear_zero_pos); + // Linear slider + grab_t = (ImClamp(*v, v_min, v_max) - v_min) / (v_max - v_min); } // Draw From eff466d77d6318c99f51a2e25ed2f34bf5bf44f8 Mon Sep 17 00:00:00 2001 From: ocornut Date: Fri, 13 Mar 2015 13:38:58 +0000 Subject: [PATCH 14/44] Fixed parsing of decimal precision back from format string when using %% --- imgui.cpp | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/imgui.cpp b/imgui.cpp index e29e3185..f2f3b89d 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4735,9 +4735,10 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c // Parse display precision back from the display format string int decimal_precision = 3; - if (const char* p = strchr(display_format, '%')) + for (const char* p = display_format; p = strchr(p, '%'); ) { p++; + if (p[0] == '%') { p ++; continue; } // Ignore "%%" while (*p >= '0' && *p <= '9') p++; if (*p == '.') @@ -4746,6 +4747,7 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c if (decimal_precision < 0 || decimal_precision > 10) decimal_precision = 3; } + break; } const ImVec2 label_size = CalcTextSize(label, NULL, true); From 72855becabb989146b319fab2e385f8972bdcad6 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 09:36:15 +0000 Subject: [PATCH 15/44] Slider code syntax tweaks --- imgui.cpp | 85 +++++++++++++++++++++++++++---------------------------- 1 file changed, 41 insertions(+), 44 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index f2f3b89d..36c3f0f5 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4827,56 +4827,53 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c // Process clicking on the slider if (g.ActiveId == id) { - if (g.IO.MouseDown[0]) + if (!is_unbound && g.IO.MouseDown[0]) { - if (!is_unbound) - { - const float normalized_pos = ImClamp((g.IO.MousePos.x - slider_effective_x1) / slider_effective_w, 0.0f, 1.0f); + const float normalized_pos = ImClamp((g.IO.MousePos.x - slider_effective_x1) / slider_effective_w, 0.0f, 1.0f); - float new_value; - if (is_logarithmic) + float new_value; + if (is_logarithmic) + { + // Account for logarithmic scale on both sides of the zero + if (normalized_pos < linear_zero_pos) { - // Account for logarithmic scale on both sides of the zero - if (normalized_pos < linear_zero_pos) - { - // Negative: rescale to the negative range before powering - float a = 1.0f - (normalized_pos / linear_zero_pos); - a = powf(a, power); - new_value = ImLerp(ImMin(v_max,0.f), v_min, a); - } + // Negative: rescale to the negative range before powering + float a = 1.0f - (normalized_pos / linear_zero_pos); + a = powf(a, power); + new_value = ImLerp(ImMin(v_max,0.f), v_min, a); + } + else + { + // Positive: rescale to the positive range before powering + float a; + if (fabsf(linear_zero_pos - 1.0f) > 1.e-6) + a = (normalized_pos - linear_zero_pos) / (1.0f - linear_zero_pos); else - { - // Positive: rescale to the positive range before powering - float a; - if (fabsf(linear_zero_pos - 1.0f) > 1.e-6) - a = (normalized_pos - linear_zero_pos) / (1.0f - linear_zero_pos); - else - a = normalized_pos; - a = powf(a, power); - new_value = ImLerp(ImMax(v_min,0.0f), v_max, a); - } - } - else - { - // Linear slider - new_value = ImLerp(v_min, v_max, normalized_pos); + a = normalized_pos; + a = powf(a, power); + new_value = ImLerp(ImMax(v_min,0.0f), v_max, a); } + } + else + { + // Linear slider + new_value = ImLerp(v_min, v_max, normalized_pos); + } - // Round past decimal precision - // 0->1, 1->0.1, 2->0.01, etc. - // So when our value is 1.99999 with a precision of 0.001 we'll end up rounding to 2.0 - const float min_step = 1.0f / powf(10.0f, (float)decimal_precision); - const float remainder = fmodf(new_value, min_step); - if (remainder <= min_step*0.5f) - new_value -= remainder; - else - new_value += (min_step - remainder); + // Round past decimal precision + // 0->1, 1->0.1, 2->0.01, etc. + // So when our value is 1.99999 with a precision of 0.001 we'll end up rounding to 2.0 + const float min_step = 1.0f / powf(10.0f, (float)decimal_precision); + const float remainder = fmodf(new_value, min_step); + if (remainder <= min_step*0.5f) + new_value -= remainder; + else + new_value += (min_step - remainder); - if (*v != new_value) - { - *v = new_value; - value_changed = true; - } + if (*v != new_value) + { + *v = new_value; + value_changed = true; } } else @@ -4915,7 +4912,7 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c window->DrawList->AddRectFilled(grab_bb.Min, grab_bb.Max, window->Color(g.ActiveId == id ? ImGuiCol_SliderGrabActive : ImGuiCol_SliderGrab)); } - // Draw value using user-provided display format so user can add prefix/suffix/decorations to the value. + // Display value using user-provided display format so user can add prefix/suffix/decorations to the value. char value_buf[64]; char* value_buf_end = value_buf + ImFormatString(value_buf, IM_ARRAYSIZE(value_buf), display_format, *v); const ImVec2 value_text_size = CalcTextSize(value_buf, value_buf_end, true); From 834085c90c46ea4663233f667aadae25599b8135 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 10:00:11 +0000 Subject: [PATCH 16/44] Examples: Updated to GLFW 3.1 --- .../opengl3_example/opengl3_example.vcxproj | 4 +- .../opengl_example/glfw/include/GLFW/glfw3.h | 1932 +++++++++++++---- .../glfw/include/GLFW/glfw3native.h | 212 +- .../opengl_example/glfw/lib-msvc100/glfw3.lib | Bin 121560 -> 0 bytes .../opengl_example/glfw/lib-msvc110/glfw3.lib | Bin 121720 -> 0 bytes .../glfw/lib-vc2010-32/glfw3.lib | Bin 0 -> 129546 bytes .../opengl_example/opengl_example.vcxproj | 4 +- 7 files changed, 1701 insertions(+), 451 deletions(-) delete mode 100644 examples/opengl_example/glfw/lib-msvc100/glfw3.lib delete mode 100644 examples/opengl_example/glfw/lib-msvc110/glfw3.lib create mode 100644 examples/opengl_example/glfw/lib-vc2010-32/glfw3.lib diff --git a/examples/opengl3_example/opengl3_example.vcxproj b/examples/opengl3_example/opengl3_example.vcxproj index cda336d2..3dd9afdf 100644 --- a/examples/opengl3_example/opengl3_example.vcxproj +++ b/examples/opengl3_example/opengl3_example.vcxproj @@ -54,7 +54,7 @@ true - $(ProjectDir)..\opengl_example\glfw\lib-msvc100;$(ProjectDir)..\opengl_example\glew\lib\Release\Win32;%(AdditionalLibraryDirectories) + $(ProjectDir)..\opengl_example\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) opengl32.lib;glfw3.lib;%(AdditionalDependencies) Console msvcrt.lib @@ -72,7 +72,7 @@ true true true - $(ProjectDir)..\opengl_example\glfw\lib-msvc100;$(ProjectDir)..\opengl_example\glew\lib\Release\Win32;%(AdditionalLibraryDirectories) + $(ProjectDir)..\opengl_example\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) opengl32.lib;glfw3.lib;%(AdditionalDependencies) Console diff --git a/examples/opengl_example/glfw/include/GLFW/glfw3.h b/examples/opengl_example/glfw/include/GLFW/glfw3.h index 8b11c2fc..89414491 100644 --- a/examples/opengl_example/glfw/include/GLFW/glfw3.h +++ b/examples/opengl_example/glfw/include/GLFW/glfw3.h @@ -1,5 +1,5 @@ /************************************************************************* - * GLFW 3.0 - www.glfw.org + * GLFW 3.1 - www.glfw.org * A library for OpenGL, window and input *------------------------------------------------------------------------ * Copyright (c) 2002-2006 Marcus Geelnard @@ -38,23 +38,27 @@ extern "C" { * Doxygen documentation *************************************************************************/ -/*! @defgroup clipboard Clipboard support - */ /*! @defgroup context Context handling + * + * This is the reference documentation for context related functions. For more + * information, see the @ref context. */ -/*! @defgroup error Error handling - */ -/*! @defgroup init Initialization and version information +/*! @defgroup init Initialization, version and errors + * + * This is the reference documentation for initialization and termination of + * the library, version management and error handling. For more information, + * see the @ref intro. */ /*! @defgroup input Input handling + * + * This is the reference documentation for input related functions and types. + * For more information, see the @ref input. */ /*! @defgroup monitor Monitor handling * * This is the reference documentation for monitor related functions and types. * For more information, see the @ref monitor. */ -/*! @defgroup time Time input - */ /*! @defgroup window Window handling * * This is the reference documentation for window related functions and types, @@ -140,63 +144,81 @@ extern "C" { /* Include the chosen client API headers. */ #if defined(__APPLE_CC__) - #if defined(GLFW_INCLUDE_GLCOREARB) - #include - #elif !defined(GLFW_INCLUDE_NONE) - #define GL_GLEXT_LEGACY - #include + #if defined(GLFW_INCLUDE_GLCOREARB) + #include + #if defined(GLFW_INCLUDE_GLEXT) + #include #endif - #if defined(GLFW_INCLUDE_GLU) - #include + #elif !defined(GLFW_INCLUDE_NONE) + #if !defined(GLFW_INCLUDE_GLEXT) + #define GL_GLEXT_LEGACY #endif + #include + #endif + #if defined(GLFW_INCLUDE_GLU) + #include + #endif #else - #if defined(GLFW_INCLUDE_GLCOREARB) - #include - #elif defined(GLFW_INCLUDE_ES1) - #include - #elif defined(GLFW_INCLUDE_ES2) - #include - #elif defined(GLFW_INCLUDE_ES3) - #include - #elif !defined(GLFW_INCLUDE_NONE) - #include + #if defined(GLFW_INCLUDE_GLCOREARB) + #include + #elif defined(GLFW_INCLUDE_ES1) + #include + #if defined(GLFW_INCLUDE_GLEXT) + #include #endif - #if defined(GLFW_INCLUDE_GLU) - #include + #elif defined(GLFW_INCLUDE_ES2) + #include + #if defined(GLFW_INCLUDE_GLEXT) + #include #endif + #elif defined(GLFW_INCLUDE_ES3) + #include + #if defined(GLFW_INCLUDE_GLEXT) + #include + #endif + #elif defined(GLFW_INCLUDE_ES31) + #include + #if defined(GLFW_INCLUDE_GLEXT) + #include + #endif + #elif !defined(GLFW_INCLUDE_NONE) + #include + #if defined(GLFW_INCLUDE_GLEXT) + #include + #endif + #endif + #if defined(GLFW_INCLUDE_GLU) + #include + #endif #endif #if defined(GLFW_DLL) && defined(_GLFW_BUILD_DLL) - /* GLFW_DLL is defined by users of GLFW when compiling programs that will link - * to the DLL version of the GLFW library. _GLFW_BUILD_DLL is defined by the - * GLFW configuration header when compiling the DLL version of the library. + /* GLFW_DLL must be defined by applications that are linking against the DLL + * version of the GLFW library. _GLFW_BUILD_DLL is defined by the GLFW + * configuration header when compiling the DLL version of the library. */ #error "You must not have both GLFW_DLL and _GLFW_BUILD_DLL defined" #endif +/* GLFWAPI is used to declare public API functions for export + * from the DLL / shared library / dynamic library. + */ #if defined(_WIN32) && defined(_GLFW_BUILD_DLL) - - /* We are building a Win32 DLL */ + /* We are building GLFW as a Win32 DLL */ #define GLFWAPI __declspec(dllexport) - #elif defined(_WIN32) && defined(GLFW_DLL) - - /* We are calling a Win32 DLL */ + /* We are calling GLFW as a Win32 DLL */ #if defined(__LCC__) #define GLFWAPI extern #else #define GLFWAPI __declspec(dllimport) #endif - #elif defined(__GNUC__) && defined(_GLFW_BUILD_DLL) - + /* We are building GLFW as a shared / dynamic library */ #define GLFWAPI __attribute__((visibility("default"))) - #else - - /* We are either building/calling a static lib or we are non-win32 */ + /* We are building or calling GLFW as a static library */ #define GLFWAPI - #endif /* -------------------- END SYSTEM/COMPILER SPECIFIC --------------------- */ @@ -220,27 +242,36 @@ extern "C" { * backward-compatible. * @ingroup init */ -#define GLFW_VERSION_MINOR 0 +#define GLFW_VERSION_MINOR 1 /*! @brief The revision number of the GLFW library. * * This is incremented when a bug fix release is made that does not contain any * API changes. * @ingroup init */ -#define GLFW_VERSION_REVISION 3 +#define GLFW_VERSION_REVISION 0 /*! @} */ /*! @name Key and button actions * @{ */ -/*! @brief The key or button was released. +/*! @brief The key or mouse button was released. + * + * The key or mouse button was released. + * * @ingroup input */ #define GLFW_RELEASE 0 -/*! @brief The key or button was pressed. +/*! @brief The key or mouse button was pressed. + * + * The key or mouse button was pressed. + * * @ingroup input */ #define GLFW_PRESS 1 /*! @brief The key was held down until it repeated. + * + * The key was held down until it repeated. + * * @ingroup input */ #define GLFW_REPEAT 2 @@ -248,20 +279,22 @@ extern "C" { /*! @defgroup keys Keyboard keys * - * These key codes are inspired by the *USB HID Usage Tables v1.12* (p. 53-60), - * but re-arranged to map to 7-bit ASCII for printable keys (function keys are - * put in the 256+ range). + * See [key input](@ref input_key) for how these are used. * - * The naming of the key codes follow these rules: - * - The US keyboard layout is used - * - Names of printable alpha-numeric characters are used (e.g. "A", "R", - * "3", etc.) - * - For non-alphanumeric characters, Unicode:ish names are used (e.g. - * "COMMA", "LEFT_SQUARE_BRACKET", etc.). Note that some names do not - * correspond to the Unicode standard (usually for brevity) - * - Keys that lack a clear US mapping are named "WORLD_x" - * - For non-printable keys, custom names are used (e.g. "F4", - * "BACKSPACE", etc.) + * These key codes are inspired by the _USB HID Usage Tables v1.12_ (p. 53-60), + * but re-arranged to map to 7-bit ASCII for printable keys (function keys are + * put in the 256+ range). + * + * The naming of the key codes follow these rules: + * - The US keyboard layout is used + * - Names of printable alpha-numeric characters are used (e.g. "A", "R", + * "3", etc.) + * - For non-alphanumeric characters, Unicode:ish names are used (e.g. + * "COMMA", "LEFT_SQUARE_BRACKET", etc.). Note that some names do not + * correspond to the Unicode standard (usually for brevity) + * - Keys that lack a clear US mapping are named "WORLD_x" + * - For non-printable keys, custom names are used (e.g. "F4", + * "BACKSPACE", etc.) * * @ingroup input * @{ @@ -398,6 +431,9 @@ extern "C" { /*! @} */ /*! @defgroup mods Modifier key flags + * + * See [key input](@ref input_key) for how these are used. + * * @ingroup input * @{ */ @@ -417,6 +453,9 @@ extern "C" { /*! @} */ /*! @defgroup buttons Mouse buttons + * + * See [mouse button input](@ref input_mouse_button) for how these are used. + * * @ingroup input * @{ */ #define GLFW_MOUSE_BUTTON_1 0 @@ -434,6 +473,9 @@ extern "C" { /*! @} */ /*! @defgroup joysticks Joysticks + * + * See [joystick input](@ref joystick) for how these are used. + * * @ingroup input * @{ */ #define GLFW_JOYSTICK_1 0 @@ -456,36 +498,128 @@ extern "C" { /*! @} */ /*! @defgroup errors Error codes - * @ingroup error + * + * See [error handling](@ref error_handling) for how these are used. + * + * @ingroup init * @{ */ /*! @brief GLFW has not been initialized. + * + * This occurs if a GLFW function was called that may not be called unless the + * library is [initialized](@ref intro_init). + * + * @par Analysis + * Application programmer error. Initialize GLFW before calling any function + * that requires initialization. */ #define GLFW_NOT_INITIALIZED 0x00010001 /*! @brief No context is current for this thread. + * + * This occurs if a GLFW function was called that needs and operates on the + * current OpenGL or OpenGL ES context but no context is current on the calling + * thread. One such function is @ref glfwSwapInterval. + * + * @par Analysis + * Application programmer error. Ensure a context is current before calling + * functions that require a current context. */ #define GLFW_NO_CURRENT_CONTEXT 0x00010002 -/*! @brief One of the enum parameters for the function was given an invalid - * enum. +/*! @brief One of the arguments to the function was an invalid enum value. + * + * One of the arguments to the function was an invalid enum value, for example + * requesting [GLFW_RED_BITS](@ref window_hints_fb) with @ref + * glfwGetWindowAttrib. + * + * @par Analysis + * Application programmer error. Fix the offending call. */ #define GLFW_INVALID_ENUM 0x00010003 -/*! @brief One of the parameters for the function was given an invalid value. +/*! @brief One of the arguments to the function was an invalid value. + * + * One of the arguments to the function was an invalid value, for example + * requesting a non-existent OpenGL or OpenGL ES version like 2.7. + * + * Requesting a valid but unavailable OpenGL or OpenGL ES version will instead + * result in a @ref GLFW_VERSION_UNAVAILABLE error. + * + * @par Analysis + * Application programmer error. Fix the offending call. */ #define GLFW_INVALID_VALUE 0x00010004 /*! @brief A memory allocation failed. + * + * A memory allocation failed. + * + * @par Analysis + * A bug in GLFW or the underlying operating system. Report the bug to our + * [issue tracker](https://github.com/glfw/glfw/issues). */ #define GLFW_OUT_OF_MEMORY 0x00010005 /*! @brief GLFW could not find support for the requested client API on the * system. + * + * GLFW could not find support for the requested client API on the system. + * + * @par Analysis + * The installed graphics driver does not support the requested client API, or + * does not support it via the chosen context creation backend. Below are + * a few examples. + * + * @par + * Some pre-installed Windows graphics drivers do not support OpenGL. AMD only + * supports OpenGL ES via EGL, while Nvidia and Intel only supports it via + * a WGL or GLX extension. OS X does not provide OpenGL ES at all. The Mesa + * EGL, OpenGL and OpenGL ES libraries do not interface with the Nvidia binary + * driver. */ #define GLFW_API_UNAVAILABLE 0x00010006 -/*! @brief The requested client API version is not available. +/*! @brief The requested OpenGL or OpenGL ES version is not available. + * + * The requested OpenGL or OpenGL ES version (including any requested profile + * or context option) is not available on this machine. + * + * @par Analysis + * The machine does not support your requirements. If your application is + * sufficiently flexible, downgrade your requirements and try again. + * Otherwise, inform the user that their machine does not match your + * requirements. + * + * @par + * Future invalid OpenGL and OpenGL ES versions, for example OpenGL 4.8 if 5.0 + * comes out before the 4.x series gets that far, also fail with this error and + * not @ref GLFW_INVALID_VALUE, because GLFW cannot know what future versions + * will exist. */ #define GLFW_VERSION_UNAVAILABLE 0x00010007 /*! @brief A platform-specific error occurred that does not match any of the * more specific categories. + * + * A platform-specific error occurred that does not match any of the more + * specific categories. + * + * @par Analysis + * A bug in GLFW or the underlying operating system. Report the bug to our + * [issue tracker](https://github.com/glfw/glfw/issues). */ #define GLFW_PLATFORM_ERROR 0x00010008 -/*! @brief The clipboard did not contain data in the requested format. +/*! @brief The requested format is not supported or available. + * + * If emitted during window creation, the requested pixel format is not + * supported. + * + * If emitted when querying the clipboard, the contents of the clipboard could + * not be converted to the requested format. + * + * @par Analysis + * If emitted during window creation, one or more + * [hard constraints](@ref window_hints_hard) did not match any of the + * available pixel formats. If your application is sufficiently flexible, + * downgrade your requirements and try again. Otherwise, inform the user that + * their machine does not match your requirements. + * + * @par + * If emitted when querying the clipboard, ignore the error or report it to + * the user, as appropriate. */ #define GLFW_FORMAT_UNAVAILABLE 0x00010009 /*! @} */ @@ -495,6 +629,8 @@ extern "C" { #define GLFW_RESIZABLE 0x00020003 #define GLFW_VISIBLE 0x00020004 #define GLFW_DECORATED 0x00020005 +#define GLFW_AUTO_ICONIFY 0x00020006 +#define GLFW_FLOATING 0x00020007 #define GLFW_RED_BITS 0x00021001 #define GLFW_GREEN_BITS 0x00021002 @@ -511,6 +647,7 @@ extern "C" { #define GLFW_SAMPLES 0x0002100D #define GLFW_SRGB_CAPABLE 0x0002100E #define GLFW_REFRESH_RATE 0x0002100F +#define GLFW_DOUBLEBUFFER 0x00021010 #define GLFW_CLIENT_API 0x00022001 #define GLFW_CONTEXT_VERSION_MAJOR 0x00022002 @@ -520,6 +657,7 @@ extern "C" { #define GLFW_OPENGL_FORWARD_COMPAT 0x00022006 #define GLFW_OPENGL_DEBUG_CONTEXT 0x00022007 #define GLFW_OPENGL_PROFILE 0x00022008 +#define GLFW_CONTEXT_RELEASE_BEHAVIOR 0x00022009 #define GLFW_OPENGL_API 0x00030001 #define GLFW_OPENGL_ES_API 0x00030002 @@ -540,9 +678,54 @@ extern "C" { #define GLFW_CURSOR_HIDDEN 0x00034002 #define GLFW_CURSOR_DISABLED 0x00034003 +#define GLFW_ANY_RELEASE_BEHAVIOR 0 +#define GLFW_RELEASE_BEHAVIOR_FLUSH 0x00035001 +#define GLFW_RELEASE_BEHAVIOR_NONE 0x00035002 + +/*! @defgroup shapes Standard cursor shapes + * + * See [standard cursor creation](@ref cursor_standard) for how these are used. + * + * @ingroup input + * @{ */ + +/*! @brief The regular arrow cursor shape. + * + * The regular arrow cursor. + */ +#define GLFW_ARROW_CURSOR 0x00036001 +/*! @brief The text input I-beam cursor shape. + * + * The text input I-beam cursor shape. + */ +#define GLFW_IBEAM_CURSOR 0x00036002 +/*! @brief The crosshair shape. + * + * The crosshair shape. + */ +#define GLFW_CROSSHAIR_CURSOR 0x00036003 +/*! @brief The hand shape. + * + * The hand shape. + */ +#define GLFW_HAND_CURSOR 0x00036004 +/*! @brief The horizontal resize arrow shape. + * + * The horizontal resize arrow shape. + */ +#define GLFW_HRESIZE_CURSOR 0x00036005 +/*! @brief The vertical resize arrow shape. + * + * The vertical resize arrow shape. + */ +#define GLFW_VRESIZE_CURSOR 0x00036006 +/*! @} */ + #define GLFW_CONNECTED 0x00040001 #define GLFW_DISCONNECTED 0x00040002 +#define GLFW_DONT_CARE -1 + /************************************************************************* * GLFW API types @@ -573,6 +756,14 @@ typedef struct GLFWmonitor GLFWmonitor; */ typedef struct GLFWwindow GLFWwindow; +/*! @brief Opaque cursor object. + * + * Opaque cursor object. + * + * @ingroup cursor + */ +typedef struct GLFWcursor GLFWcursor; + /*! @brief The function signature for error callbacks. * * This is the function signature for error callback functions. @@ -582,7 +773,7 @@ typedef struct GLFWwindow GLFWwindow; * * @sa glfwSetErrorCallback * - * @ingroup error + * @ingroup init */ typedef void (* GLFWerrorfun)(int,const char*); @@ -590,7 +781,7 @@ typedef void (* GLFWerrorfun)(int,const char*); * * This is the function signature for window position callback functions. * - * @param[in] window The window that the user moved. + * @param[in] window The window that was moved. * @param[in] xpos The new x-coordinate, in screen coordinates, of the * upper-left corner of the client area of the window. * @param[in] ypos The new y-coordinate, in screen coordinates, of the @@ -606,7 +797,7 @@ typedef void (* GLFWwindowposfun)(GLFWwindow*,int,int); * * This is the function signature for window size callback functions. * - * @param[in] window The window that the user resized. + * @param[in] window The window that was resized. * @param[in] width The new width, in screen coordinates, of the window. * @param[in] height The new height, in screen coordinates, of the window. * @@ -644,9 +835,9 @@ typedef void (* GLFWwindowrefreshfun)(GLFWwindow*); * * This is the function signature for window focus callback functions. * - * @param[in] window The window that was focused or defocused. - * @param[in] focused `GL_TRUE` if the window was focused, or `GL_FALSE` if - * it was defocused. + * @param[in] window The window that gained or lost input focus. + * @param[in] focused `GL_TRUE` if the window was given input focus, or + * `GL_FALSE` if it lost it. * * @sa glfwSetWindowFocusCallback * @@ -706,8 +897,8 @@ typedef void (* GLFWmousebuttonfun)(GLFWwindow*,int,int,int); * This is the function signature for cursor position callback functions. * * @param[in] window The window that received the event. - * @param[in] xpos The new x-coordinate of the cursor. - * @param[in] ypos The new y-coordinate of the cursor. + * @param[in] xpos The new x-coordinate, in screen coordinates, of the cursor. + * @param[in] ypos The new y-coordinate, in screen coordinates, of the cursor. * * @sa glfwSetCursorPosCallback * @@ -750,7 +941,7 @@ typedef void (* GLFWscrollfun)(GLFWwindow*,double,double); * @param[in] window The window that received the event. * @param[in] key The [keyboard key](@ref keys) that was pressed or released. * @param[in] scancode The system-specific scancode of the key. - * @param[in] action @ref GLFW_PRESS, @ref GLFW_RELEASE or @ref GLFW_REPEAT. + * @param[in] action `GLFW_PRESS`, `GLFW_RELEASE` or `GLFW_REPEAT`. * @param[in] mods Bit field describing which [modifier keys](@ref mods) were * held down. * @@ -765,7 +956,7 @@ typedef void (* GLFWkeyfun)(GLFWwindow*,int,int,int,int); * This is the function signature for Unicode character callback functions. * * @param[in] window The window that received the event. - * @param[in] character The Unicode code point of the character. + * @param[in] codepoint The Unicode code point of the character. * * @sa glfwSetCharCallback * @@ -773,6 +964,38 @@ typedef void (* GLFWkeyfun)(GLFWwindow*,int,int,int,int); */ typedef void (* GLFWcharfun)(GLFWwindow*,unsigned int); +/*! @brief The function signature for Unicode character with modifiers + * callbacks. + * + * This is the function signature for Unicode character with modifiers callback + * functions. It is called for each input character, regardless of what + * modifier keys are held down. + * + * @param[in] window The window that received the event. + * @param[in] codepoint The Unicode code point of the character. + * @param[in] mods Bit field describing which [modifier keys](@ref mods) were + * held down. + * + * @sa glfwSetCharModsCallback + * + * @ingroup input + */ +typedef void (* GLFWcharmodsfun)(GLFWwindow*,unsigned int,int); + +/*! @brief The function signature for file drop callbacks. + * + * This is the function signature for file drop callbacks. + * + * @param[in] window The window that received the event. + * @param[in] count The number of dropped files. + * @param[in] names The UTF-8 encoded path names of the dropped files. + * + * @sa glfwSetDropCallback + * + * @ingroup input + */ +typedef void (* GLFWdropfun)(GLFWwindow*,int,const char**); + /*! @brief The function signature for monitor configuration callbacks. * * This is the function signature for monitor configuration callback functions. @@ -838,6 +1061,21 @@ typedef struct GLFWgammaramp unsigned int size; } GLFWgammaramp; +/*! @brief Image data. + */ +typedef struct GLFWimage +{ + /*! The width, in pixels, of this image. + */ + int width; + /*! The height, in pixels, of this image. + */ + int height; + /*! The pixel data of this image, arranged left-to-right, top-to-bottom. + */ + unsigned char* pixels; +} GLFWimage; + /************************************************************************* * GLFW API functions @@ -846,56 +1084,70 @@ typedef struct GLFWgammaramp /*! @brief Initializes the GLFW library. * * This function initializes the GLFW library. Before most GLFW functions can - * be used, GLFW must be initialized, and before a program terminates GLFW + * be used, GLFW must be initialized, and before an application terminates GLFW * should be terminated in order to free any resources allocated during or * after initialization. * * If this function fails, it calls @ref glfwTerminate before returning. If it - * succeeds, you should call @ref glfwTerminate before the program exits. + * succeeds, you should call @ref glfwTerminate before the application exits. * * Additional calls to this function after successful initialization but before - * termination will succeed but will do nothing. + * termination will return `GL_TRUE` immediately. * - * @return `GL_TRUE` if successful, or `GL_FALSE` if an error occurred. + * @return `GL_TRUE` if successful, or `GL_FALSE` if an + * [error](@ref error_handling) occurred. * - * @par New in GLFW 3 - * This function no longer registers @ref glfwTerminate with `atexit`. - * - * @note This function may only be called from the main thread. - * - * @note This function may take several seconds to complete on some systems, - * while on other systems it may take only a fraction of a second to complete. - * - * @note **Mac OS X:** This function will change the current directory of the + * @remarks __OS X:__ This function will change the current directory of the * application to the `Contents/Resources` subdirectory of the application's - * bundle, if present. + * bundle, if present. This can be disabled with a + * [compile-time option](@ref compile_options_osx). * + * @remarks __X11:__ If the `LC_CTYPE` category of the current locale is set to + * `"C"` then the environment's locale will be applied to that category. This + * is done because character input will not function when `LC_CTYPE` is set to + * `"C"`. If another locale was set before this function was called, it will + * be left untouched. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref intro_init * @sa glfwTerminate * + * @since Added in GLFW 1.0. + * * @ingroup init */ GLFWAPI int glfwInit(void); /*! @brief Terminates the GLFW library. * - * This function destroys all remaining windows, frees any allocated resources - * and sets the library to an uninitialized state. Once this is called, you - * must again call @ref glfwInit successfully before you will be able to use - * most GLFW functions. + * This function destroys all remaining windows and cursors, restores any + * modified gamma ramps and frees any other allocated resources. Once this + * function is called, you must again call @ref glfwInit successfully before + * you will be able to use most GLFW functions. * * If GLFW has been successfully initialized, this function should be called - * before the program exits. If initialization fails, there is no need to call - * this function, as it is called by @ref glfwInit before it returns failure. + * before the application exits. If initialization fails, there is no need to + * call this function, as it is called by @ref glfwInit before it returns + * failure. * * @remarks This function may be called before @ref glfwInit. * - * @note This function may only be called from the main thread. - * * @warning No window's context may be current on another thread when this * function is called. * + * @par Reentrancy + * This function may not be called from a callback. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref intro_init * @sa glfwInit * + * @since Added in GLFW 1.0. + * * @ingroup init */ GLFWAPI void glfwTerminate(void); @@ -906,46 +1158,55 @@ GLFWAPI void glfwTerminate(void); * library. It is intended for when you are using GLFW as a shared library and * want to ensure that you are using the minimum required version. * + * Any or all of the version arguments may be `NULL`. This function always + * succeeds. + * * @param[out] major Where to store the major version number, or `NULL`. * @param[out] minor Where to store the minor version number, or `NULL`. * @param[out] rev Where to store the revision number, or `NULL`. * * @remarks This function may be called before @ref glfwInit. * - * @remarks This function may be called from any thread. + * @par Thread Safety + * This function may be called from any thread. * + * @sa @ref intro_version * @sa glfwGetVersionString * + * @since Added in GLFW 1.0. + * * @ingroup init */ GLFWAPI void glfwGetVersion(int* major, int* minor, int* rev); /*! @brief Returns a string describing the compile-time configuration. * - * This function returns a static string generated at compile-time according to - * which configuration macros were defined. This is intended for use when - * submitting bug reports, to allow developers to see which code paths are - * enabled in a binary. + * This function returns the compile-time generated + * [version string](@ref intro_version_string) of the GLFW library binary. It + * describes the version, platform, compiler and any platform-specific + * compile-time options. * - * The format of the string is as follows: - * - The version of GLFW - * - The name of the window system API - * - The name of the context creation API - * - Any additional options or APIs + * __Do not use the version string__ to parse the GLFW library version. The + * @ref glfwGetVersion function already provides the version of the running + * library binary. * - * For example, when compiling GLFW 3.0 with MinGW using the Win32 and WGL - * back ends, the version string may look something like this: - * - * 3.0.0 Win32 WGL MinGW + * This function always succeeds. * * @return The GLFW version string. * * @remarks This function may be called before @ref glfwInit. * - * @remarks This function may be called from any thread. + * @par Pointer Lifetime + * The returned string is static and compile-time generated. * + * @par Thread Safety + * This function may be called from any thread. + * + * @sa @ref intro_version * @sa glfwGetVersion * + * @since Added in GLFW 3.0. + * * @ingroup init */ GLFWAPI const char* glfwGetVersionString(void); @@ -955,23 +1216,31 @@ GLFWAPI const char* glfwGetVersionString(void); * This function sets the error callback, which is called with an error code * and a human-readable description each time a GLFW error occurs. * + * The error callback is called on the thread where the error occurred. If you + * are using GLFW from multiple threads, your error callback needs to be + * written accordingly. + * + * Because the description string may have been generated specifically for that + * error, it is not guaranteed to be valid after the callback has returned. If + * you wish to use it after the callback returns, you need to make a copy. + * + * Once set, the error callback remains set even after the library has been + * terminated. + * * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set. * * @remarks This function may be called before @ref glfwInit. * - * @note The error callback is called by the thread where the error was - * generated. If you are using GLFW from multiple threads, your error callback - * needs to be written accordingly. + * @par Thread Safety + * This function may only be called from the main thread. * - * @note Because the description string provided to the callback may have been - * generated specifically for that error, it is not guaranteed to be valid - * after the callback has returned. If you wish to use it after that, you need - * to make your own copy of it before returning. + * @sa @ref error_handling * - * @ingroup error + * @since Added in GLFW 3.0. + * + * @ingroup init */ GLFWAPI GLFWerrorfun glfwSetErrorCallback(GLFWerrorfun cbfun); @@ -980,19 +1249,25 @@ GLFWAPI GLFWerrorfun glfwSetErrorCallback(GLFWerrorfun cbfun); * This function returns an array of handles for all currently connected * monitors. * - * @param[out] count Where to store the size of the returned array. This is - * set to zero if an error occurred. - * @return An array of monitor handles, or `NULL` if an error occurred. + * @param[out] count Where to store the number of monitors in the returned + * array. This is set to zero if an error occurred. + * @return An array of monitor handles, or `NULL` if an + * [error](@ref error_handling) occurred. * - * @note The returned array is allocated and freed by GLFW. You should not - * free it yourself. + * @par Pointer Lifetime + * The returned array is allocated and freed by GLFW. You should not free it + * yourself. It is guaranteed to be valid only until the monitor configuration + * changes or the library is terminated. * - * @note The returned array is valid only until the monitor configuration - * changes. See @ref glfwSetMonitorCallback to receive notifications of - * configuration changes. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref monitor_monitors + * @sa @ref monitor_event * @sa glfwGetPrimaryMonitor * + * @since Added in GLFW 3.0. + * * @ingroup monitor */ GLFWAPI GLFWmonitor** glfwGetMonitors(int* count); @@ -1002,10 +1277,17 @@ GLFWAPI GLFWmonitor** glfwGetMonitors(int* count); * This function returns the primary monitor. This is usually the monitor * where elements like the Windows task bar or the OS X menu bar is located. * - * @return The primary monitor, or `NULL` if an error occurred. + * @return The primary monitor, or `NULL` if an [error](@ref error_handling) + * occurred. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref monitor_monitors * @sa glfwGetMonitors * + * @since Added in GLFW 3.0. + * * @ingroup monitor */ GLFWAPI GLFWmonitor* glfwGetPrimaryMonitor(void); @@ -1015,10 +1297,20 @@ GLFWAPI GLFWmonitor* glfwGetPrimaryMonitor(void); * This function returns the position, in screen coordinates, of the upper-left * corner of the specified monitor. * + * Any or all of the position arguments may be `NULL`. If an error occurs, all + * non-`NULL` position arguments will be set to zero. + * * @param[in] monitor The monitor to query. * @param[out] xpos Where to store the monitor x-coordinate, or `NULL`. * @param[out] ypos Where to store the monitor y-coordinate, or `NULL`. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref monitor_properties + * + * @since Added in GLFW 3.0. + * * @ingroup monitor */ GLFWAPI void glfwGetMonitorPos(GLFWmonitor* monitor, int* xpos, int* ypos); @@ -1028,31 +1320,55 @@ GLFWAPI void glfwGetMonitorPos(GLFWmonitor* monitor, int* xpos, int* ypos); * This function returns the size, in millimetres, of the display area of the * specified monitor. * - * @param[in] monitor The monitor to query. - * @param[out] width Where to store the width, in mm, of the monitor's display - * area, or `NULL`. - * @param[out] height Where to store the height, in mm, of the monitor's - * display area, or `NULL`. + * Some systems do not provide accurate monitor size information, either + * because the monitor + * [EDID](https://en.wikipedia.org/wiki/Extended_display_identification_data) + * data is incorrect or because the driver does not report it accurately. * - * @note Some operating systems do not provide accurate information, either - * because the monitor's EDID data is incorrect, or because the driver does not - * report it accurately. + * Any or all of the size arguments may be `NULL`. If an error occurs, all + * non-`NULL` size arguments will be set to zero. + * + * @param[in] monitor The monitor to query. + * @param[out] widthMM Where to store the width, in millimetres, of the + * monitor's display area, or `NULL`. + * @param[out] heightMM Where to store the height, in millimetres, of the + * monitor's display area, or `NULL`. + * + * @remarks __Windows:__ The OS calculates the returned physical size from the + * current resolution and system DPI instead of querying the monitor EDID data. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref monitor_properties + * + * @since Added in GLFW 3.0. * * @ingroup monitor */ -GLFWAPI void glfwGetMonitorPhysicalSize(GLFWmonitor* monitor, int* width, int* height); +GLFWAPI void glfwGetMonitorPhysicalSize(GLFWmonitor* monitor, int* widthMM, int* heightMM); /*! @brief Returns the name of the specified monitor. * * This function returns a human-readable name, encoded as UTF-8, of the - * specified monitor. + * specified monitor. The name typically reflects the make and model of the + * monitor and is not guaranteed to be unique among the connected monitors. * * @param[in] monitor The monitor to query. - * @return The UTF-8 encoded name of the monitor, or `NULL` if an error - * occurred. + * @return The UTF-8 encoded name of the monitor, or `NULL` if an + * [error](@ref error_handling) occurred. * - * @note The returned string is allocated and freed by GLFW. You should not - * free it yourself. + * @par Pointer Lifetime + * The returned string is allocated and freed by GLFW. You should not free it + * yourself. It is valid until the specified monitor is disconnected or the + * library is terminated. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref monitor_properties + * + * @since Added in GLFW 3.0. * * @ingroup monitor */ @@ -1066,12 +1382,19 @@ GLFWAPI const char* glfwGetMonitorName(GLFWmonitor* monitor); * * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). * - * @bug **X11:** This callback is not yet called on monitor configuration + * @bug __X11:__ This callback is not yet called on monitor configuration * changes. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref monitor_event + * + * @since Added in GLFW 3.0. + * * @ingroup monitor */ GLFWAPI GLFWmonitorfun glfwSetMonitorCallback(GLFWmonitorfun cbfun); @@ -1086,16 +1409,25 @@ GLFWAPI GLFWmonitorfun glfwSetMonitorCallback(GLFWmonitorfun cbfun); * @param[in] monitor The monitor to query. * @param[out] count Where to store the number of video modes in the returned * array. This is set to zero if an error occurred. - * @return An array of video modes, or `NULL` if an error occurred. + * @return An array of video modes, or `NULL` if an + * [error](@ref error_handling) occurred. * - * @note The returned array is allocated and freed by GLFW. You should not - * free it yourself. + * @par Pointer Lifetime + * The returned array is allocated and freed by GLFW. You should not free it + * yourself. It is valid until the specified monitor is disconnected, this + * function is called again for that monitor or the library is terminated. * - * @note The returned array is valid only until this function is called again - * for the specified monitor. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref monitor_modes * @sa glfwGetVideoMode * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Changed to return an array of modes for a specific monitor. + * * @ingroup monitor */ GLFWAPI const GLFWvidmode* glfwGetVideoModes(GLFWmonitor* monitor, int* count); @@ -1103,17 +1435,26 @@ GLFWAPI const GLFWvidmode* glfwGetVideoModes(GLFWmonitor* monitor, int* count); /*! @brief Returns the current mode of the specified monitor. * * This function returns the current video mode of the specified monitor. If - * you are using a full screen window, the return value will therefore depend - * on whether it is focused. + * you have created a full screen window for that monitor, the return value + * will depend on whether that window is iconified. * * @param[in] monitor The monitor to query. - * @return The current mode of the monitor, or `NULL` if an error occurred. + * @return The current mode of the monitor, or `NULL` if an + * [error](@ref error_handling) occurred. * - * @note The returned struct is allocated and freed by GLFW. You should not - * free it yourself. + * @par Pointer Lifetime + * The returned array is allocated and freed by GLFW. You should not free it + * yourself. It is valid until the specified monitor is disconnected or the + * library is terminated. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref monitor_modes * @sa glfwGetVideoModes * + * @since Added in GLFW 3.0. Replaces `glfwGetDesktopMode`. + * * @ingroup monitor */ GLFWAPI const GLFWvidmode* glfwGetVideoMode(GLFWmonitor* monitor); @@ -1126,19 +1467,37 @@ GLFWAPI const GLFWvidmode* glfwGetVideoMode(GLFWmonitor* monitor); * @param[in] monitor The monitor whose gamma ramp to set. * @param[in] gamma The desired exponent. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref monitor_gamma + * + * @since Added in GLFW 3.0. + * * @ingroup monitor */ GLFWAPI void glfwSetGamma(GLFWmonitor* monitor, float gamma); -/*! @brief Retrieves the current gamma ramp for the specified monitor. +/*! @brief Returns the current gamma ramp for the specified monitor. * - * This function retrieves the current gamma ramp of the specified monitor. + * This function returns the current gamma ramp of the specified monitor. * * @param[in] monitor The monitor to query. - * @return The current gamma ramp, or `NULL` if an error occurred. + * @return The current gamma ramp, or `NULL` if an + * [error](@ref error_handling) occurred. * - * @note The value arrays of the returned ramp are allocated and freed by GLFW. - * You should not free them yourself. + * @par Pointer Lifetime + * The returned structure and its arrays are allocated and freed by GLFW. You + * should not free them yourself. They are valid until the specified monitor + * is disconnected, this function is called again for that monitor or the + * library is terminated. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref monitor_gamma + * + * @since Added in GLFW 3.0. * * @ingroup monitor */ @@ -1146,13 +1505,25 @@ GLFWAPI const GLFWgammaramp* glfwGetGammaRamp(GLFWmonitor* monitor); /*! @brief Sets the current gamma ramp for the specified monitor. * - * This function sets the current gamma ramp for the specified monitor. + * This function sets the current gamma ramp for the specified monitor. The + * original gamma ramp for that monitor is saved by GLFW the first time this + * function is called and is restored by @ref glfwTerminate. * * @param[in] monitor The monitor whose gamma ramp to set. * @param[in] ramp The gamma ramp to use. * * @note Gamma ramp sizes other than 256 are not supported by all hardware. * + * @par Pointer Lifetime + * The specified gamma ramp is copied before this function returns. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref monitor_gamma + * + * @since Added in GLFW 3.0. + * * @ingroup monitor */ GLFWAPI void glfwSetGammaRamp(GLFWmonitor* monitor, const GLFWgammaramp* ramp); @@ -1162,10 +1533,14 @@ GLFWAPI void glfwSetGammaRamp(GLFWmonitor* monitor, const GLFWgammaramp* ramp); * This function resets all window hints to their * [default values](@ref window_hints_values). * - * @note This function may only be called from the main thread. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref window_hints * @sa glfwWindowHint * + * @since Added in GLFW 3.0. + * * @ingroup window */ GLFWAPI void glfwDefaultWindowHints(void); @@ -1175,52 +1550,70 @@ GLFWAPI void glfwDefaultWindowHints(void); * This function sets hints for the next call to @ref glfwCreateWindow. The * hints, once set, retain their values until changed by a call to @ref * glfwWindowHint or @ref glfwDefaultWindowHints, or until the library is - * terminated with @ref glfwTerminate. + * terminated. * * @param[in] target The [window hint](@ref window_hints) to set. * @param[in] hint The new value of the window hint. * - * @par New in GLFW 3 - * Hints are no longer reset to their default values on window creation. To - * set default hint values, use @ref glfwDefaultWindowHints. - * - * @note This function may only be called from the main thread. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref window_hints * @sa glfwDefaultWindowHints * + * @since Added in GLFW 3.0. Replaces `glfwOpenWindowHint`. + * * @ingroup window */ GLFWAPI void glfwWindowHint(int target, int hint); /*! @brief Creates a window and its associated context. * - * This function creates a window and its associated context. Most of the - * options controlling how the window and its context should be created are - * specified through @ref glfwWindowHint. + * This function creates a window and its associated OpenGL or OpenGL ES + * context. Most of the options controlling how the window and its context + * should be created are specified with [window hints](@ref window_hints). * * Successful creation does not change which context is current. Before you - * can use the newly created context, you need to make it current using @ref - * glfwMakeContextCurrent. + * can use the newly created context, you need to + * [make it current](@ref context_current). For information about the `share` + * parameter, see @ref context_sharing. * - * Note that the created window and context may differ from what you requested, - * as not all parameters and hints are + * The created window, framebuffer and context may differ from what you + * requested, as not all parameters and hints are * [hard constraints](@ref window_hints_hard). This includes the size of the - * window, especially for full screen windows. To retrieve the actual - * attributes of the created window and context, use queries like @ref + * window, especially for full screen windows. To query the actual attributes + * of the created window, framebuffer and context, use queries like @ref * glfwGetWindowAttrib and @ref glfwGetWindowSize. * - * To create a full screen window, you need to specify the monitor to use. If - * no monitor is specified, windowed mode will be used. Unless you have a way - * for the user to choose a specific monitor, it is recommended that you pick - * the primary monitor. For more information on how to retrieve monitors, see - * @ref monitor_monitors. + * To create a full screen window, you need to specify the monitor the window + * will cover. If no monitor is specified, windowed mode will be used. Unless + * you have a way for the user to choose a specific monitor, it is recommended + * that you pick the primary monitor. For more information on how to query + * connected monitors, see @ref monitor_monitors. * - * To create the window at a specific position, make it initially invisible - * using the `GLFW_VISIBLE` window hint, set its position and then show it. + * For full screen windows, the specified size becomes the resolution of the + * window's _desired video mode_. As long as a full screen window has input + * focus, the supported video mode most closely matching the desired video mode + * is set for the specified monitor. For more information about full screen + * windows, including the creation of so called _windowed full screen_ or + * _borderless full screen_ windows, see @ref window_windowed_full_screen. * - * If a full screen window is active, the screensaver is prohibited from + * By default, newly created windows use the placement recommended by the + * window system. To create the window at a specific position, make it + * initially invisible using the [GLFW_VISIBLE](@ref window_hints_wnd) window + * hint, set its [position](@ref window_pos) and then [show](@ref window_hide) + * it. + * + * If a full screen window has input focus, the screensaver is prohibited from * starting. * + * Window systems put limits on window sizes. Very large or very small window + * dimensions may be overridden by the window system on creation. Check the + * actual [size](@ref window_size) after creation. + * + * The [swap interval](@ref buffer_swap) is not set during window creation and + * the initial value may vary depending on driver settings and defaults. + * * @param[in] width The desired width, in screen coordinates, of the window. * This must be greater than zero. * @param[in] height The desired height, in screen coordinates, of the window. @@ -1230,23 +1623,47 @@ GLFWAPI void glfwWindowHint(int target, int hint); * windowed mode. * @param[in] share The window whose context to share resources with, or `NULL` * to not share resources. - * @return The handle of the created window, or `NULL` if an error occurred. + * @return The handle of the created window, or `NULL` if an + * [error](@ref error_handling) occurred. * - * @remarks **Windows:** If the executable has an icon resource named + * @remarks __Windows:__ Window creation will fail if the Microsoft GDI + * software OpenGL implementation is the only one available. + * + * @remarks __Windows:__ If the executable has an icon resource named * `GLFW_ICON,` it will be set as the icon for the window. If no such icon is * present, the `IDI_WINLOGO` icon will be used instead. * - * @remarks **Mac OS X:** The GLFW window has no icon, as it is not a document + * @remarks __Windows:__ The context to share resources with may not be current + * on any other thread. + * + * @remarks __OS X:__ The GLFW window has no icon, as it is not a document * window, but the dock icon will be the same as the application bundle's icon. - * Also, the first time a window is opened the menu bar is populated with - * common commands like Hide, Quit and About. The (minimal) about dialog uses - * information from the application's bundle. For more information on bundles, - * see the Bundle Programming Guide provided by Apple. + * For more information on bundles, see the + * [Bundle Programming Guide](https://developer.apple.com/library/mac/documentation/CoreFoundation/Conceptual/CFBundles/) + * in the Mac Developer Library. * - * @note This function may only be called from the main thread. + * @remarks __OS X:__ The first time a window is created the menu bar is + * populated with common commands like Hide, Quit and About. The About entry + * opens a minimal about dialog with information from the application's bundle. + * The menu bar can be disabled with a + * [compile-time option](@ref compile_options_osx). * + * @remarks __X11:__ There is no mechanism for setting the window icon yet. + * + * @remarks __X11:__ Some window managers will not respect the placement of + * initially hidden windows. + * + * @par Reentrancy + * This function may not be called from a callback. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_creation * @sa glfwDestroyWindow * + * @since Added in GLFW 3.0. Replaces `glfwOpenWindow`. + * * @ingroup window */ GLFWAPI GLFWwindow* glfwCreateWindow(int width, int height, const char* title, GLFWmonitor* monitor, GLFWwindow* share); @@ -1256,19 +1673,25 @@ GLFWAPI GLFWwindow* glfwCreateWindow(int width, int height, const char* title, G * This function destroys the specified window and its context. On calling * this function, no further callbacks will be called for that window. * - * @param[in] window The window to destroy. - * - * @note This function may only be called from the main thread. - * - * @note This function may not be called from a callback. - * - * @note If the window's context is current on the main thread, it is + * If the context of the specified window is current on the main thread, it is * detached before being destroyed. * - * @warning The window's context must not be current on any other thread. + * @param[in] window The window to destroy. * + * @note The context of the specified window must not be current on any other + * thread when this function is called. + * + * @par Reentrancy + * This function may not be called from a callback. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_creation * @sa glfwCreateWindow * + * @since Added in GLFW 3.0. Replaces `glfwCloseWindow`. + * * @ingroup window */ GLFWAPI void glfwDestroyWindow(GLFWwindow* window); @@ -1280,7 +1703,12 @@ GLFWAPI void glfwDestroyWindow(GLFWwindow* window); * @param[in] window The window to query. * @return The value of the close flag. * - * @remarks This function may be called from secondary threads. + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @sa @ref window_close + * + * @since Added in GLFW 3.0. * * @ingroup window */ @@ -1295,7 +1723,12 @@ GLFWAPI int glfwWindowShouldClose(GLFWwindow* window); * @param[in] window The window whose flag to change. * @param[in] value The new value. * - * @remarks This function may be called from secondary threads. + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @sa @ref window_close + * + * @since Added in GLFW 3.0. * * @ingroup window */ @@ -1309,7 +1742,15 @@ GLFWAPI void glfwSetWindowShouldClose(GLFWwindow* window, int value); * @param[in] window The window whose title to change. * @param[in] title The UTF-8 encoded window title. * - * @note This function may only be called from the main thread. + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_title + * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. * * @ingroup window */ @@ -1320,14 +1761,23 @@ GLFWAPI void glfwSetWindowTitle(GLFWwindow* window, const char* title); * This function retrieves the position, in screen coordinates, of the * upper-left corner of the client area of the specified window. * + * Any or all of the position arguments may be `NULL`. If an error occurs, all + * non-`NULL` position arguments will be set to zero. + * * @param[in] window The window to query. * @param[out] xpos Where to store the x-coordinate of the upper-left corner of * the client area, or `NULL`. * @param[out] ypos Where to store the y-coordinate of the upper-left corner of * the client area, or `NULL`. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_pos * @sa glfwSetWindowPos * + * @since Added in GLFW 3.0. + * * @ingroup window */ GLFWAPI void glfwGetWindowPos(GLFWwindow* window, int* xpos, int* ypos); @@ -1335,31 +1785,30 @@ GLFWAPI void glfwGetWindowPos(GLFWwindow* window, int* xpos, int* ypos); /*! @brief Sets the position of the client area of the specified window. * * This function sets the position, in screen coordinates, of the upper-left - * corner of the client area of the window. + * corner of the client area of the specified windowed mode window. If the + * window is a full screen window, this function does nothing. * - * If the specified window is a full screen window, this function does nothing. + * __Do not use this function__ to move an already visible window unless you + * have very good reasons for doing so, as it will confuse and annoy the user. * - * If you wish to set an initial window position you should create a hidden - * window (using @ref glfwWindowHint and `GLFW_VISIBLE`), set its position and - * then show it. + * The window manager may put limits on what positions are allowed. GLFW + * cannot and should not override these limits. * * @param[in] window The window to query. * @param[in] xpos The x-coordinate of the upper-left corner of the client area. * @param[in] ypos The y-coordinate of the upper-left corner of the client area. * - * @note It is very rarely a good idea to move an already visible window, as it - * will confuse and annoy the user. - * - * @note This function may only be called from the main thread. - * - * @note The window manager may put limits on what positions are allowed. - * - * @bug **X11:** Some window managers ignore the set position of hidden (i.e. - * unmapped) windows, instead placing them where it thinks is appropriate once - * they are shown. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref window_pos * @sa glfwGetWindowPos * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. + * * @ingroup window */ GLFWAPI void glfwSetWindowPos(GLFWwindow* window, int xpos, int ypos); @@ -1367,7 +1816,11 @@ GLFWAPI void glfwSetWindowPos(GLFWwindow* window, int xpos, int ypos); /*! @brief Retrieves the size of the client area of the specified window. * * This function retrieves the size, in screen coordinates, of the client area - * of the specified window. + * of the specified window. If you wish to retrieve the size of the + * framebuffer of the window in pixels, see @ref glfwGetFramebufferSize. + * + * Any or all of the size arguments may be `NULL`. If an error occurs, all + * non-`NULL` size arguments will be set to zero. * * @param[in] window The window whose size to retrieve. * @param[out] width Where to store the width, in screen coordinates, of the @@ -1375,8 +1828,17 @@ GLFWAPI void glfwSetWindowPos(GLFWwindow* window, int xpos, int ypos); * @param[out] height Where to store the height, in screen coordinates, of the * client area, or `NULL`. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_size * @sa glfwSetWindowSize * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. + * * @ingroup window */ GLFWAPI void glfwGetWindowSize(GLFWwindow* window, int* width, int* height); @@ -1391,16 +1853,24 @@ GLFWAPI void glfwGetWindowSize(GLFWwindow* window, int* width, int* height); * the context is unaffected, the bit depths of the framebuffer remain * unchanged. * + * The window manager may put limits on what sizes are allowed. GLFW cannot + * and should not override these limits. + * * @param[in] window The window to resize. * @param[in] width The desired width of the specified window. * @param[in] height The desired height of the specified window. * - * @note This function may only be called from the main thread. - * - * @note The window manager may put limits on what window sizes are allowed. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref window_size * @sa glfwGetWindowSize * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. + * * @ingroup window */ GLFWAPI void glfwSetWindowSize(GLFWwindow* window, int width, int height); @@ -1408,7 +1878,11 @@ GLFWAPI void glfwSetWindowSize(GLFWwindow* window, int width, int height); /*! @brief Retrieves the size of the framebuffer of the specified window. * * This function retrieves the size, in pixels, of the framebuffer of the - * specified window. + * specified window. If you wish to retrieve the size of the window in screen + * coordinates, see @ref glfwGetWindowSize. + * + * Any or all of the size arguments may be `NULL`. If an error occurs, all + * non-`NULL` size arguments will be set to zero. * * @param[in] window The window whose framebuffer to query. * @param[out] width Where to store the width, in pixels, of the framebuffer, @@ -1416,74 +1890,140 @@ GLFWAPI void glfwSetWindowSize(GLFWwindow* window, int width, int height); * @param[out] height Where to store the height, in pixels, of the framebuffer, * or `NULL`. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_fbsize * @sa glfwSetFramebufferSizeCallback * + * @since Added in GLFW 3.0. + * * @ingroup window */ GLFWAPI void glfwGetFramebufferSize(GLFWwindow* window, int* width, int* height); +/*! @brief Retrieves the size of the frame of the window. + * + * This function retrieves the size, in screen coordinates, of each edge of the + * frame of the specified window. This size includes the title bar, if the + * window has one. The size of the frame may vary depending on the + * [window-related hints](@ref window_hints_wnd) used to create it. + * + * Because this function retrieves the size of each window frame edge and not + * the offset along a particular coordinate axis, the retrieved values will + * always be zero or positive. + * + * Any or all of the size arguments may be `NULL`. If an error occurs, all + * non-`NULL` size arguments will be set to zero. + * + * @param[in] window The window whose frame size to query. + * @param[out] left Where to store the size, in screen coordinates, of the left + * edge of the window frame, or `NULL`. + * @param[out] top Where to store the size, in screen coordinates, of the top + * edge of the window frame, or `NULL`. + * @param[out] right Where to store the size, in screen coordinates, of the + * right edge of the window frame, or `NULL`. + * @param[out] bottom Where to store the size, in screen coordinates, of the + * bottom edge of the window frame, or `NULL`. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_size + * + * @since Added in GLFW 3.1. + * + * @ingroup window + */ +GLFWAPI void glfwGetWindowFrameSize(GLFWwindow* window, int* left, int* top, int* right, int* bottom); + /*! @brief Iconifies the specified window. * - * This function iconifies/minimizes the specified window, if it was previously - * restored. If it is a full screen window, the original monitor resolution is - * restored until the window is restored. If the window is already iconified, - * this function does nothing. + * This function iconifies (minimizes) the specified window if it was + * previously restored. If the window is already iconified, this function does + * nothing. + * + * If the specified window is a full screen window, the original monitor + * resolution is restored until the window is restored. * * @param[in] window The window to iconify. * - * @note This function may only be called from the main thread. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref window_iconify * @sa glfwRestoreWindow * + * @since Added in GLFW 2.1. + * + * @par + * __GLFW 3:__ Added window handle parameter. + * * @ingroup window */ GLFWAPI void glfwIconifyWindow(GLFWwindow* window); /*! @brief Restores the specified window. * - * This function restores the specified window, if it was previously - * iconified/minimized. If it is a full screen window, the resolution chosen - * for the window is restored on the selected monitor. If the window is - * already restored, this function does nothing. + * This function restores the specified window if it was previously iconified + * (minimized). If the window is already restored, this function does nothing. + * + * If the specified window is a full screen window, the resolution chosen for + * the window is restored on the selected monitor. * * @param[in] window The window to restore. * - * @note This function may only be called from the main thread. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref window_iconify * @sa glfwIconifyWindow * + * @since Added in GLFW 2.1. + * + * @par + * __GLFW 3:__ Added window handle parameter. + * * @ingroup window */ GLFWAPI void glfwRestoreWindow(GLFWwindow* window); /*! @brief Makes the specified window visible. * - * This function makes the specified window visible, if it was previously + * This function makes the specified window visible if it was previously * hidden. If the window is already visible or is in full screen mode, this * function does nothing. * * @param[in] window The window to make visible. * - * @note This function may only be called from the main thread. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref window_hide * @sa glfwHideWindow * + * @since Added in GLFW 3.0. + * * @ingroup window */ GLFWAPI void glfwShowWindow(GLFWwindow* window); /*! @brief Hides the specified window. * - * This function hides the specified window, if it was previously visible. If + * This function hides the specified window if it was previously visible. If * the window is already hidden or is in full screen mode, this function does * nothing. * * @param[in] window The window to hide. * - * @note This function may only be called from the main thread. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref window_hide * @sa glfwShowWindow * + * @since Added in GLFW 3.0. + * * @ingroup window */ GLFWAPI void glfwHideWindow(GLFWwindow* window); @@ -1494,7 +2034,15 @@ GLFWAPI void glfwHideWindow(GLFWwindow* window); * in full screen on. * * @param[in] window The window to query. - * @return The monitor, or `NULL` if the window is in windowed mode. + * @return The monitor, or `NULL` if the window is in windowed mode or an error + * occurred. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_monitor + * + * @since Added in GLFW 3.0. * * @ingroup window */ @@ -1502,13 +2050,22 @@ GLFWAPI GLFWmonitor* glfwGetWindowMonitor(GLFWwindow* window); /*! @brief Returns an attribute of the specified window. * - * This function returns an attribute of the specified window. There are many - * attributes, some related to the window and others to its context. + * This function returns the value of an attribute of the specified window or + * its OpenGL or OpenGL ES context. * * @param[in] window The window to query. * @param[in] attrib The [window attribute](@ref window_attribs) whose value to * return. - * @return The value of the attribute, or zero if an error occurred. + * @return The value of the attribute, or zero if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_attribs + * + * @since Added in GLFW 3.0. Replaces `glfwGetWindowParam` and + * `glfwGetGLVersion`. * * @ingroup window */ @@ -1523,8 +2080,14 @@ GLFWAPI int glfwGetWindowAttrib(GLFWwindow* window, int attrib); * @param[in] window The window whose pointer to set. * @param[in] pointer The new value. * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @sa @ref window_userptr * @sa glfwGetWindowUserPointer * + * @since Added in GLFW 3.0. + * * @ingroup window */ GLFWAPI void glfwSetWindowUserPointer(GLFWwindow* window, void* pointer); @@ -1536,8 +2099,14 @@ GLFWAPI void glfwSetWindowUserPointer(GLFWwindow* window, void* pointer); * * @param[in] window The window whose pointer to return. * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @sa @ref window_userptr * @sa glfwSetWindowUserPointer * + * @since Added in GLFW 3.0. + * * @ingroup window */ GLFWAPI void* glfwGetWindowUserPointer(GLFWwindow* window); @@ -1551,8 +2120,15 @@ GLFWAPI void* glfwGetWindowUserPointer(GLFWwindow* window); * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_pos + * + * @since Added in GLFW 3.0. * * @ingroup window */ @@ -1567,8 +2143,18 @@ GLFWAPI GLFWwindowposfun glfwSetWindowPosCallback(GLFWwindow* window, GLFWwindow * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_size + * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. Updated callback signature. * * @ingroup window */ @@ -1588,12 +2174,22 @@ GLFWAPI GLFWwindowsizefun glfwSetWindowSizeCallback(GLFWwindow* window, GLFWwind * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). * - * @remarks **Mac OS X:** Selecting Quit from the application menu will + * @remarks __OS X:__ Selecting Quit from the application menu will * trigger the close callback for all windows. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_close + * + * @since Added in GLFW 2.5. + * + * @par + * __GLFW 3:__ Added window handle parameter. Updated callback signature. + * * @ingroup window */ GLFWAPI GLFWwindowclosefun glfwSetWindowCloseCallback(GLFWwindow* window, GLFWwindowclosefun cbfun); @@ -1611,12 +2207,18 @@ GLFWAPI GLFWwindowclosefun glfwSetWindowCloseCallback(GLFWwindow* window, GLFWwi * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). * - * @note On compositing window systems such as Aero, Compiz or Aqua, where the - * window contents are saved off-screen, this callback may be called only very - * infrequently or never at all. + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_refresh + * + * @since Added in GLFW 2.5. + * + * @par + * __GLFW 3:__ Added window handle parameter. Updated callback signature. * * @ingroup window */ @@ -1625,18 +2227,25 @@ GLFWAPI GLFWwindowrefreshfun glfwSetWindowRefreshCallback(GLFWwindow* window, GL /*! @brief Sets the focus callback for the specified window. * * This function sets the focus callback of the specified window, which is - * called when the window gains or loses focus. + * called when the window gains or loses input focus. * - * After the focus callback is called for a window that lost focus, synthetic - * key and mouse button release events will be generated for all such that had - * been pressed. For more information, see @ref glfwSetKeyCallback and @ref - * glfwSetMouseButtonCallback. + * After the focus callback is called for a window that lost input focus, + * synthetic key and mouse button release events will be generated for all such + * that had been pressed. For more information, see @ref glfwSetKeyCallback + * and @ref glfwSetMouseButtonCallback. * * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_focus + * + * @since Added in GLFW 3.0. * * @ingroup window */ @@ -1650,8 +2259,15 @@ GLFWAPI GLFWwindowfocusfun glfwSetWindowFocusCallback(GLFWwindow* window, GLFWwi * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_iconify + * + * @since Added in GLFW 3.0. * * @ingroup window */ @@ -1665,8 +2281,15 @@ GLFWAPI GLFWwindowiconifyfun glfwSetWindowIconifyCallback(GLFWwindow* window, GL * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref window_fbsize + * + * @since Added in GLFW 3.0. * * @ingroup window */ @@ -1674,99 +2297,164 @@ GLFWAPI GLFWframebuffersizefun glfwSetFramebufferSizeCallback(GLFWwindow* window /*! @brief Processes all pending events. * - * This function processes only those events that have already been received - * and then returns immediately. Processing events will cause the window and - * input callbacks associated with those events to be called. + * This function processes only those events that are already in the event + * queue and then returns immediately. Processing events will cause the window + * and input callbacks associated with those events to be called. * - * This function is not required for joystick input to work. + * On some platforms, a window move, resize or menu operation will cause event + * processing to block. This is due to how event processing is designed on + * those platforms. You can use the + * [window refresh callback](@ref window_refresh) to redraw the contents of + * your window when necessary during such operations. * - * @par New in GLFW 3 - * This function is no longer called by @ref glfwSwapBuffers. You need to call - * it or @ref glfwWaitEvents yourself. + * On some platforms, certain events are sent directly to the application + * without going through the event queue, causing callbacks to be called + * outside of a call to one of the event processing functions. * - * @note This function may only be called from the main thread. + * Event processing is not required for joystick input to work. * - * @note This function may not be called from a callback. + * @par Reentrancy + * This function may not be called from a callback. * - * @note On some platforms, certain callbacks may be called outside of a call - * to one of the event processing functions. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref events * @sa glfwWaitEvents * + * @since Added in GLFW 1.0. + * * @ingroup window */ GLFWAPI void glfwPollEvents(void); -/*! @brief Waits until events are pending and processes them. +/*! @brief Waits until events are queued and processes them. * - * This function puts the calling thread to sleep until at least one event has - * been received. Once one or more events have been received, it behaves as if - * @ref glfwPollEvents was called, i.e. the events are processed and the - * function then returns immediately. Processing events will cause the window - * and input callbacks associated with those events to be called. + * This function puts the calling thread to sleep until at least one event is + * available in the event queue. Once one or more events are available, + * it behaves exactly like @ref glfwPollEvents, i.e. the events in the queue + * are processed and the function then returns immediately. Processing events + * will cause the window and input callbacks associated with those events to be + * called. * * Since not all events are associated with callbacks, this function may return * without a callback having been called even if you are monitoring all * callbacks. * - * This function is not required for joystick input to work. + * On some platforms, a window move, resize or menu operation will cause event + * processing to block. This is due to how event processing is designed on + * those platforms. You can use the + * [window refresh callback](@ref window_refresh) to redraw the contents of + * your window when necessary during such operations. * - * @note This function may only be called from the main thread. + * On some platforms, certain callbacks may be called outside of a call to one + * of the event processing functions. * - * @note This function may not be called from a callback. + * If no windows exist, this function returns immediately. For synchronization + * of threads in applications that do not create windows, use your threading + * library of choice. * - * @note On some platforms, certain callbacks may be called outside of a call - * to one of the event processing functions. + * Event processing is not required for joystick input to work. * + * @par Reentrancy + * This function may not be called from a callback. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref events * @sa glfwPollEvents * + * @since Added in GLFW 2.5. + * * @ingroup window */ GLFWAPI void glfwWaitEvents(void); +/*! @brief Posts an empty event to the event queue. + * + * This function posts an empty event from the current thread to the event + * queue, causing @ref glfwWaitEvents to return. + * + * If no windows exist, this function returns immediately. For synchronization + * of threads in applications that do not create windows, use your threading + * library of choice. + * + * @par Thread Safety + * This function may be called from any thread. + * + * @sa @ref events + * @sa glfwWaitEvents + * + * @since Added in GLFW 3.1. + * + * @ingroup window + */ +GLFWAPI void glfwPostEmptyEvent(void); + /*! @brief Returns the value of an input option for the specified window. + * + * This function returns the value of an input option for the specified window. + * The mode must be one of `GLFW_CURSOR`, `GLFW_STICKY_KEYS` or + * `GLFW_STICKY_MOUSE_BUTTONS`. * * @param[in] window The window to query. * @param[in] mode One of `GLFW_CURSOR`, `GLFW_STICKY_KEYS` or * `GLFW_STICKY_MOUSE_BUTTONS`. * + * @par Thread Safety + * This function may only be called from the main thread. + * * @sa glfwSetInputMode * + * @since Added in GLFW 3.0. + * * @ingroup input */ GLFWAPI int glfwGetInputMode(GLFWwindow* window, int mode); /*! @brief Sets an input option for the specified window. + * + * This function sets an input mode option for the specified window. The mode + * must be one of `GLFW_CURSOR`, `GLFW_STICKY_KEYS` or + * `GLFW_STICKY_MOUSE_BUTTONS`. + * + * If the mode is `GLFW_CURSOR`, the value must be one of the following cursor + * modes: + * - `GLFW_CURSOR_NORMAL` makes the cursor visible and behaving normally. + * - `GLFW_CURSOR_HIDDEN` makes the cursor invisible when it is over the client + * area of the window but does not restrict the cursor from leaving. + * - `GLFW_CURSOR_DISABLED` hides and grabs the cursor, providing virtual + * and unlimited cursor movement. This is useful for implementing for + * example 3D camera controls. + * + * If the mode is `GLFW_STICKY_KEYS`, the value must be either `GL_TRUE` to + * enable sticky keys, or `GL_FALSE` to disable it. If sticky keys are + * enabled, a key press will ensure that @ref glfwGetKey returns `GLFW_PRESS` + * the next time it is called even if the key had been released before the + * call. This is useful when you are only interested in whether keys have been + * pressed but not when or in which order. + * + * If the mode is `GLFW_STICKY_MOUSE_BUTTONS`, the value must be either + * `GL_TRUE` to enable sticky mouse buttons, or `GL_FALSE` to disable it. If + * sticky mouse buttons are enabled, a mouse button press will ensure that @ref + * glfwGetMouseButton returns `GLFW_PRESS` the next time it is called even if + * the mouse button had been released before the call. This is useful when you + * are only interested in whether mouse buttons have been pressed but not when + * or in which order. + * * @param[in] window The window whose input mode to set. * @param[in] mode One of `GLFW_CURSOR`, `GLFW_STICKY_KEYS` or * `GLFW_STICKY_MOUSE_BUTTONS`. * @param[in] value The new value of the specified input mode. * - * If `mode` is `GLFW_CURSOR`, the value must be one of the supported input - * modes: - * - `GLFW_CURSOR_NORMAL` makes the cursor visible and behaving normally. - * - `GLFW_CURSOR_HIDDEN` makes the cursor invisible when it is over the client - * area of the window. - * - `GLFW_CURSOR_DISABLED` disables the cursor and removes any limitations on - * cursor movement. - * - * If `mode` is `GLFW_STICKY_KEYS`, the value must be either `GL_TRUE` to - * enable sticky keys, or `GL_FALSE` to disable it. If sticky keys are - * enabled, a key press will ensure that @ref glfwGetKey returns @ref - * GLFW_PRESS the next time it is called even if the key had been released - * before the call. This is useful when you are only interested in whether - * keys have been pressed but not when or in which order. - * - * If `mode` is `GLFW_STICKY_MOUSE_BUTTONS`, the value must be either `GL_TRUE` - * to enable sticky mouse buttons, or `GL_FALSE` to disable it. If sticky - * mouse buttons are enabled, a mouse button press will ensure that @ref - * glfwGetMouseButton returns @ref GLFW_PRESS the next time it is called even - * if the mouse button had been released before the call. This is useful when - * you are only interested in whether mouse buttons have been pressed but not - * when or in which order. + * @par Thread Safety + * This function may only be called from the main thread. * * @sa glfwGetInputMode * + * @since Added in GLFW 3.0. Replaces `glfwEnable` and `glfwDisable`. + * * @ingroup input */ GLFWAPI void glfwSetInputMode(GLFWwindow* window, int mode, int value); @@ -1776,22 +2464,34 @@ GLFWAPI void glfwSetInputMode(GLFWwindow* window, int mode, int value); * * This function returns the last state reported for the specified key to the * specified window. The returned state is one of `GLFW_PRESS` or - * `GLFW_RELEASE`. The higher-level state `GLFW_REPEAT` is only reported to + * `GLFW_RELEASE`. The higher-level action `GLFW_REPEAT` is only reported to * the key callback. * * If the `GLFW_STICKY_KEYS` input mode is enabled, this function returns - * `GLFW_PRESS` the first time you call this function after a key has been - * pressed, even if the key has already been released. + * `GLFW_PRESS` the first time you call it for a key that was pressed, even if + * that key has already been released. * * The key functions deal with physical keys, with [key tokens](@ref keys) * named after their use on the standard US keyboard layout. If you want to * input text, use the Unicode character callback instead. * + * The [modifier key bit masks](@ref mods) are not key tokens and cannot be + * used with this function. + * * @param[in] window The desired window. - * @param[in] key The desired [keyboard key](@ref keys). + * @param[in] key The desired [keyboard key](@ref keys). `GLFW_KEY_UNKNOWN` is + * not a valid key for this function. * @return One of `GLFW_PRESS` or `GLFW_RELEASE`. * - * @note `GLFW_KEY_UNKNOWN` is not a valid key for this function. + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref input_key + * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. * * @ingroup input */ @@ -1801,25 +2501,37 @@ GLFWAPI int glfwGetKey(GLFWwindow* window, int key); * window. * * This function returns the last state reported for the specified mouse button - * to the specified window. + * to the specified window. The returned state is one of `GLFW_PRESS` or + * `GLFW_RELEASE`. * * If the `GLFW_STICKY_MOUSE_BUTTONS` input mode is enabled, this function - * returns `GLFW_PRESS` the first time you call this function after a mouse - * button has been pressed, even if the mouse button has already been released. + * `GLFW_PRESS` the first time you call it for a mouse button that was pressed, + * even if that mouse button has already been released. * * @param[in] window The desired window. * @param[in] button The desired [mouse button](@ref buttons). * @return One of `GLFW_PRESS` or `GLFW_RELEASE`. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref input_mouse_button + * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. + * * @ingroup input */ GLFWAPI int glfwGetMouseButton(GLFWwindow* window, int button); -/*! @brief Retrieves the last reported cursor position, relative to the client - * area of the window. +/*! @brief Retrieves the position of the cursor relative to the client area of + * the window. * - * This function returns the last reported position of the cursor to the - * specified window. + * This function returns the position of the cursor, in screen coordinates, + * relative to the upper-left corner of the client area of the specified + * window. * * If the cursor is disabled (with `GLFW_CURSOR_DISABLED`) then the cursor * position is unbounded and limited only by the minimum and maximum values of @@ -1829,26 +2541,42 @@ GLFWAPI int glfwGetMouseButton(GLFWwindow* window, int button); * `floor` function. Casting directly to an integer type works for positive * coordinates, but fails for negative ones. * + * Any or all of the position arguments may be `NULL`. If an error occurs, all + * non-`NULL` position arguments will be set to zero. + * * @param[in] window The desired window. * @param[out] xpos Where to store the cursor x-coordinate, relative to the * left edge of the client area, or `NULL`. * @param[out] ypos Where to store the cursor y-coordinate, relative to the to * top edge of the client area, or `NULL`. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref cursor_pos * @sa glfwSetCursorPos * + * @since Added in GLFW 3.0. Replaces `glfwGetMousePos`. + * * @ingroup input */ GLFWAPI void glfwGetCursorPos(GLFWwindow* window, double* xpos, double* ypos); -/*! @brief Sets the position of the cursor, relative to the client area of the window. +/*! @brief Sets the position of the cursor, relative to the client area of the + * window. * - * This function sets the position of the cursor. The specified window must be - * focused. If the window does not have focus when this function is called, it - * fails silently. + * This function sets the position, in screen coordinates, of the cursor + * relative to the upper-left corner of the client area of the specified + * window. The window must have input focus. If the window does not have + * input focus when this function is called, it fails silently. * - * If the cursor is disabled (with `GLFW_CURSOR_DISABLED`) then the cursor - * position is unbounded and limited only by the minimum and maximum values of + * __Do not use this function__ to implement things like camera controls. GLFW + * already provides the `GLFW_CURSOR_DISABLED` cursor mode that hides the + * cursor, transparently re-centers it and provides unconstrained cursor + * motion. See @ref glfwSetInputMode for more information. + * + * If the cursor mode is `GLFW_CURSOR_DISABLED` then the cursor position is + * unconstrained and limited only by the minimum and maximum values of * a `double`. * * @param[in] window The desired window. @@ -1857,15 +2585,138 @@ GLFWAPI void glfwGetCursorPos(GLFWwindow* window, double* xpos, double* ypos); * @param[in] ypos The desired y-coordinate, relative to the top edge of the * client area. * + * @remarks __X11:__ Due to the asynchronous nature of a modern X desktop, it + * may take a moment for the window focus event to arrive. This means you will + * not be able to set the cursor position directly after window creation. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref cursor_pos * @sa glfwGetCursorPos * + * @since Added in GLFW 3.0. Replaces `glfwSetMousePos`. + * * @ingroup input */ GLFWAPI void glfwSetCursorPos(GLFWwindow* window, double xpos, double ypos); +/*! @brief Creates a custom cursor. + * + * Creates a new custom cursor image that can be set for a window with @ref + * glfwSetCursor. The cursor can be destroyed with @ref glfwDestroyCursor. + * Any remaining cursors are destroyed by @ref glfwTerminate. + * + * The pixels are 32-bit little-endian RGBA, i.e. eight bits per channel. They + * are arranged canonically as packed sequential rows, starting from the + * top-left corner. + * + * The cursor hotspot is specified in pixels, relative to the upper-left corner + * of the cursor image. Like all other coordinate systems in GLFW, the X-axis + * points to the right and the Y-axis points down. + * + * @param[in] image The desired cursor image. + * @param[in] xhot The desired x-coordinate, in pixels, of the cursor hotspot. + * @param[in] yhot The desired y-coordinate, in pixels, of the cursor hotspot. + * + * @return The handle of the created cursor, or `NULL` if an + * [error](@ref error_handling) occurred. + * + * @par Pointer Lifetime + * The specified image data is copied before this function returns. + * + * @par Reentrancy + * This function may not be called from a callback. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref cursor_object + * @sa glfwDestroyCursor + * @sa glfwCreateStandardCursor + * + * @since Added in GLFW 3.1. + * + * @ingroup input + */ +GLFWAPI GLFWcursor* glfwCreateCursor(const GLFWimage* image, int xhot, int yhot); + +/*! @brief Creates a cursor with a standard shape. + * + * Returns a cursor with a [standard shape](@ref shapes), that can be set for + * a window with @ref glfwSetCursor. + * + * @param[in] shape One of the [standard shapes](@ref shapes). + * + * @return A new cursor ready to use or `NULL` if an + * [error](@ref error_handling) occurred. + * + * @par Reentrancy + * This function may not be called from a callback. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref cursor_object + * @sa glfwCreateCursor + * + * @since Added in GLFW 3.1. + * + * @ingroup input + */ +GLFWAPI GLFWcursor* glfwCreateStandardCursor(int shape); + +/*! @brief Destroys a cursor. + * + * This function destroys a cursor previously created with @ref + * glfwCreateCursor. Any remaining cursors will be destroyed by @ref + * glfwTerminate. + * + * @param[in] cursor The cursor object to destroy. + * + * @par Reentrancy + * This function may not be called from a callback. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref cursor_object + * @sa glfwCreateCursor + * + * @since Added in GLFW 3.1. + * + * @ingroup input + */ +GLFWAPI void glfwDestroyCursor(GLFWcursor* cursor); + +/*! @brief Sets the cursor for the window. + * + * This function sets the cursor image to be used when the cursor is over the + * client area of the specified window. The set cursor will only be visible + * when the [cursor mode](@ref cursor_mode) of the window is + * `GLFW_CURSOR_NORMAL`. + * + * On some platforms, the set cursor may not be visible unless the window also + * has input focus. + * + * @param[in] window The window to set the cursor for. + * @param[in] cursor The cursor to set, or `NULL` to switch back to the default + * arrow cursor. + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref cursor_object + * + * @since Added in GLFW 3.1. + * + * @ingroup input + */ +GLFWAPI void glfwSetCursor(GLFWwindow* window, GLFWcursor* cursor); + /*! @brief Sets the key callback. * - * This function sets the key callback of the specific window, which is called + * This function sets the key callback of the specified window, which is called * when a key is pressed, repeated or released. * * The key functions deal with physical keys, with layout independent @@ -1873,16 +2724,16 @@ GLFWAPI void glfwSetCursorPos(GLFWwindow* window, double xpos, double ypos); * layout. If you want to input text, use the * [character callback](@ref glfwSetCharCallback) instead. * - * When a window loses focus, it will generate synthetic key release events - * for all pressed keys. You can tell these events from user-generated events - * by the fact that the synthetic ones are generated after the window has lost - * focus, i.e. `GLFW_FOCUSED` will be false and the focus callback will have - * already been called. + * When a window loses input focus, it will generate synthetic key release + * events for all pressed keys. You can tell these events from user-generated + * events by the fact that the synthetic ones are generated after the focus + * loss event has been processed, i.e. after the + * [window focus callback](@ref glfwSetWindowFocusCallback) has been called. * * The scancode of a key is specific to that platform or sometimes even to that * machine. Scancodes are intended to allow users to bind keys that don't have * a GLFW key token. Such keys have `key` set to `GLFW_KEY_UNKNOWN`, their - * state is not saved and so it cannot be retrieved with @ref glfwGetKey. + * state is not saved and so it cannot be queried with @ref glfwGetKey. * * Sometimes GLFW needs to generate synthetic key events, in which case the * scancode may be zero. @@ -1890,8 +2741,18 @@ GLFWAPI void glfwSetCursorPos(GLFWwindow* window, double xpos, double ypos); * @param[in] window The window whose callback to set. * @param[in] cbfun The new key callback, or `NULL` to remove the currently * set callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref input_key + * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. Updated callback signature. * * @ingroup input */ @@ -1899,11 +2760,56 @@ GLFWAPI GLFWkeyfun glfwSetKeyCallback(GLFWwindow* window, GLFWkeyfun cbfun); /*! @brief Sets the Unicode character callback. * - * This function sets the character callback of the specific window, which is + * This function sets the character callback of the specified window, which is * called when a Unicode character is input. * - * The character callback is intended for text input. If you want to know - * whether a specific key was pressed or released, use the + * The character callback is intended for Unicode text input. As it deals with + * characters, it is keyboard layout dependent, whereas the + * [key callback](@ref glfwSetKeyCallback) is not. Characters do not map 1:1 + * to physical keys, as a key may produce zero, one or more characters. If you + * want to know whether a specific physical key was pressed or released, see + * the key callback instead. + * + * The character callback behaves as system text input normally does and will + * not be called if modifier keys are held down that would prevent normal text + * input on that platform, for example a Super (Command) key on OS X or Alt key + * on Windows. There is a + * [character with modifiers callback](@ref glfwSetCharModsCallback) that + * receives these events. + * + * @param[in] window The window whose callback to set. + * @param[in] cbfun The new callback, or `NULL` to remove the currently set + * callback. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref input_char + * + * @since Added in GLFW 2.4. + * + * @par + * __GLFW 3:__ Added window handle parameter. Updated callback signature. + * + * @ingroup input + */ +GLFWAPI GLFWcharfun glfwSetCharCallback(GLFWwindow* window, GLFWcharfun cbfun); + +/*! @brief Sets the Unicode character with modifiers callback. + * + * This function sets the character with modifiers callback of the specified + * window, which is called when a Unicode character is input regardless of what + * modifier keys are used. + * + * The character with modifiers callback is intended for implementing custom + * Unicode character input. For regular Unicode text input, see the + * [character callback](@ref glfwSetCharCallback). Like the character + * callback, the character with modifiers callback deals with characters and is + * keyboard layout dependent. Characters do not map 1:1 to physical keys, as + * a key may produce zero, one or more characters. If you want to know whether + * a specific physical key was pressed or released, see the * [key callback](@ref glfwSetKeyCallback) instead. * * @param[in] window The window whose callback to set. @@ -1912,26 +2818,43 @@ GLFWAPI GLFWkeyfun glfwSetKeyCallback(GLFWwindow* window, GLFWkeyfun cbfun); * @return The previously set callback, or `NULL` if no callback was set or an * error occurred. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref input_char + * + * @since Added in GLFW 3.1. + * * @ingroup input */ -GLFWAPI GLFWcharfun glfwSetCharCallback(GLFWwindow* window, GLFWcharfun cbfun); +GLFWAPI GLFWcharmodsfun glfwSetCharModsCallback(GLFWwindow* window, GLFWcharmodsfun cbfun); /*! @brief Sets the mouse button callback. * * This function sets the mouse button callback of the specified window, which * is called when a mouse button is pressed or released. * - * When a window loses focus, it will generate synthetic mouse button release - * events for all pressed mouse buttons. You can tell these events from - * user-generated events by the fact that the synthetic ones are generated - * after the window has lost focus, i.e. `GLFW_FOCUSED` will be false and the - * focus callback will have already been called. + * When a window loses input focus, it will generate synthetic mouse button + * release events for all pressed mouse buttons. You can tell these events + * from user-generated events by the fact that the synthetic ones are generated + * after the focus loss event has been processed, i.e. after the + * [window focus callback](@ref glfwSetWindowFocusCallback) has been called. * * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref input_mouse_button + * + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. Updated callback signature. * * @ingroup input */ @@ -1941,13 +2864,21 @@ GLFWAPI GLFWmousebuttonfun glfwSetMouseButtonCallback(GLFWwindow* window, GLFWmo * * This function sets the cursor position callback of the specified window, * which is called when the cursor is moved. The callback is provided with the - * position relative to the upper-left corner of the client area of the window. + * position, in screen coordinates, relative to the upper-left corner of the + * client area of the window. * * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref cursor_pos + * + * @since Added in GLFW 3.0. Replaces `glfwSetMousePosCallback`. * * @ingroup input */ @@ -1962,8 +2893,15 @@ GLFWAPI GLFWcursorposfun glfwSetCursorPosCallback(GLFWwindow* window, GLFWcursor * @param[in] window The window whose callback to set. * @param[in] cbfun The new callback, or `NULL` to remove the currently set * callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref cursor_enter + * + * @since Added in GLFW 3.0. * * @ingroup input */ @@ -1981,20 +2919,61 @@ GLFWAPI GLFWcursorenterfun glfwSetCursorEnterCallback(GLFWwindow* window, GLFWcu * @param[in] window The window whose callback to set. * @param[in] cbfun The new scroll callback, or `NULL` to remove the currently * set callback. - * @return The previously set callback, or `NULL` if no callback was set or an - * error occurred. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref scrolling + * + * @since Added in GLFW 3.0. Replaces `glfwSetMouseWheelCallback`. * * @ingroup input */ GLFWAPI GLFWscrollfun glfwSetScrollCallback(GLFWwindow* window, GLFWscrollfun cbfun); +/*! @brief Sets the file drop callback. + * + * This function sets the file drop callback of the specified window, which is + * called when one or more dragged files are dropped on the window. + * + * Because the path array and its strings may have been generated specifically + * for that event, they are not guaranteed to be valid after the callback has + * returned. If you wish to use them after the callback returns, you need to + * make a deep copy. + * + * @param[in] window The window whose callback to set. + * @param[in] cbfun The new file drop callback, or `NULL` to remove the + * currently set callback. + * @return The previously set callback, or `NULL` if no callback was set or the + * library had not been [initialized](@ref intro_init). + * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref path_drop + * + * @since Added in GLFW 3.1. + * + * @ingroup input + */ +GLFWAPI GLFWdropfun glfwSetDropCallback(GLFWwindow* window, GLFWdropfun cbfun); + /*! @brief Returns whether the specified joystick is present. * * This function returns whether the specified joystick is present. * - * @param[in] joy The joystick to query. + * @param[in] joy The [joystick](@ref joysticks) to query. * @return `GL_TRUE` if the joystick is present, or `GL_FALSE` otherwise. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref joystick + * + * @since Added in GLFW 3.0. Replaces `glfwGetJoystickParam`. + * * @ingroup input */ GLFWAPI int glfwJoystickPresent(int joy); @@ -2002,17 +2981,24 @@ GLFWAPI int glfwJoystickPresent(int joy); /*! @brief Returns the values of all axes of the specified joystick. * * This function returns the values of all axes of the specified joystick. + * Each element in the array is a value between -1.0 and 1.0. * - * @param[in] joy The joystick to query. - * @param[out] count Where to store the size of the returned array. This is - * set to zero if an error occurred. + * @param[in] joy The [joystick](@ref joysticks) to query. + * @param[out] count Where to store the number of axis values in the returned + * array. This is set to zero if an error occurred. * @return An array of axis values, or `NULL` if the joystick is not present. * - * @note The returned array is allocated and freed by GLFW. You should not - * free it yourself. + * @par Pointer Lifetime + * The returned array is allocated and freed by GLFW. You should not free it + * yourself. It is valid until the specified joystick is disconnected, this + * function is called again for that joystick or the library is terminated. * - * @note The returned array is valid only until the next call to @ref - * glfwGetJoystickAxes for that joystick. + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref joystick_axis + * + * @since Added in GLFW 3.0. Replaces `glfwGetJoystickPos`. * * @ingroup input */ @@ -2021,17 +3007,27 @@ GLFWAPI const float* glfwGetJoystickAxes(int joy, int* count); /*! @brief Returns the state of all buttons of the specified joystick. * * This function returns the state of all buttons of the specified joystick. + * Each element in the array is either `GLFW_PRESS` or `GLFW_RELEASE`. * - * @param[in] joy The joystick to query. - * @param[out] count Where to store the size of the returned array. This is - * set to zero if an error occurred. + * @param[in] joy The [joystick](@ref joysticks) to query. + * @param[out] count Where to store the number of button states in the returned + * array. This is set to zero if an error occurred. * @return An array of button states, or `NULL` if the joystick is not present. * - * @note The returned array is allocated and freed by GLFW. You should not - * free it yourself. + * @par Pointer Lifetime + * The returned array is allocated and freed by GLFW. You should not free it + * yourself. It is valid until the specified joystick is disconnected, this + * function is called again for that joystick or the library is terminated. * - * @note The returned array is valid only until the next call to @ref - * glfwGetJoystickButtons for that joystick. + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref joystick_button + * + * @since Added in GLFW 2.2. + * + * @par + * __GLFW 3:__ Changed to return a dynamic array. * * @ingroup input */ @@ -2040,16 +3036,24 @@ GLFWAPI const unsigned char* glfwGetJoystickButtons(int joy, int* count); /*! @brief Returns the name of the specified joystick. * * This function returns the name, encoded as UTF-8, of the specified joystick. + * The returned string is allocated and freed by GLFW. You should not free it + * yourself. * - * @param[in] joy The joystick to query. + * @param[in] joy The [joystick](@ref joysticks) to query. * @return The UTF-8 encoded name of the joystick, or `NULL` if the joystick * is not present. * - * @note The returned string is allocated and freed by GLFW. You should not - * free it yourself. + * @par Pointer Lifetime + * The returned string is allocated and freed by GLFW. You should not free it + * yourself. It is valid until the specified joystick is disconnected, this + * function is called again for that joystick or the library is terminated. * - * @note The returned string is valid only until the next call to @ref - * glfwGetJoystickName for that joystick. + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref joystick_name + * + * @since Added in GLFW 3.0. * * @ingroup input */ @@ -2058,40 +3062,50 @@ GLFWAPI const char* glfwGetJoystickName(int joy); /*! @brief Sets the clipboard to the specified string. * * This function sets the system clipboard to the specified, UTF-8 encoded - * string. The string is copied before returning, so you don't have to retain - * it afterwards. + * string. * * @param[in] window The window that will own the clipboard contents. * @param[in] string A UTF-8 encoded string. * - * @note This function may only be called from the main thread. + * @par Pointer Lifetime + * The specified string is copied before this function returns. * + * @par Thread Safety + * This function may only be called from the main thread. + * + * @sa @ref clipboard * @sa glfwGetClipboardString * - * @ingroup clipboard + * @since Added in GLFW 3.0. + * + * @ingroup input */ GLFWAPI void glfwSetClipboardString(GLFWwindow* window, const char* string); -/*! @brief Retrieves the contents of the clipboard as a string. +/*! @brief Returns the contents of the clipboard as a string. * * This function returns the contents of the system clipboard, if it contains * or is convertible to a UTF-8 encoded string. * * @param[in] window The window that will request the clipboard contents. * @return The contents of the clipboard as a UTF-8 encoded string, or `NULL` - * if an error occurred. + * if an [error](@ref error_handling) occurred. * - * @note This function may only be called from the main thread. + * @par Pointer Lifetime + * The returned string is allocated and freed by GLFW. You should not free it + * yourself. It is valid until the next call to @ref + * glfwGetClipboardString or @ref glfwSetClipboardString, or until the library + * is terminated. * - * @note The returned string is allocated and freed by GLFW. You should not - * free it yourself. - * - * @note The returned string is valid only until the next call to @ref - * glfwGetClipboardString or @ref glfwSetClipboardString. + * @par Thread Safety + * This function may only be called from the main thread. * + * @sa @ref clipboard * @sa glfwSetClipboardString * - * @ingroup clipboard + * @since Added in GLFW 3.0. + * + * @ingroup input */ GLFWAPI const char* glfwGetClipboardString(GLFWwindow* window); @@ -2101,15 +3115,21 @@ GLFWAPI const char* glfwGetClipboardString(GLFWwindow* window); * been set using @ref glfwSetTime, the timer measures time elapsed since GLFW * was initialized. * - * @return The current value, in seconds, or zero if an error occurred. + * The resolution of the timer is system dependent, but is usually on the order + * of a few micro- or nanoseconds. It uses the highest-resolution monotonic + * time source on each supported platform. * - * @remarks This function may be called from secondary threads. + * @return The current value, in seconds, or zero if an + * [error](@ref error_handling) occurred. * - * @note The resolution of the timer is system dependent, but is usually on the - * order of a few micro- or nanoseconds. It uses the highest-resolution - * monotonic time source on each supported platform. + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. * - * @ingroup time + * @sa @ref time + * + * @since Added in GLFW 1.0. + * + * @ingroup input */ GLFWAPI double glfwGetTime(void); @@ -2120,44 +3140,61 @@ GLFWAPI double glfwGetTime(void); * * @param[in] time The new value, in seconds. * - * @note The resolution of the timer is system dependent, but is usually on the - * order of a few micro- or nanoseconds. It uses the highest-resolution - * monotonic time source on each supported platform. + * @par Thread Safety + * This function may only be called from the main thread. * - * @ingroup time + * @sa @ref time + * + * @since Added in GLFW 2.2. + * + * @ingroup input */ GLFWAPI void glfwSetTime(double time); /*! @brief Makes the context of the specified window current for the calling * thread. * - * This function makes the context of the specified window current on the - * calling thread. A context can only be made current on a single thread at - * a time and each thread can have only a single current context at a time. + * This function makes the OpenGL or OpenGL ES context of the specified window + * current on the calling thread. A context can only be made current on + * a single thread at a time and each thread can have only a single current + * context at a time. + * + * By default, making a context non-current implicitly forces a pipeline flush. + * On machines that support `GL_KHR_context_flush_control`, you can control + * whether a context performs this flush by setting the + * [GLFW_CONTEXT_RELEASE_BEHAVIOR](@ref window_hints_ctx) window hint. * * @param[in] window The window whose context to make current, or `NULL` to * detach the current context. * - * @remarks This function may be called from secondary threads. + * @par Thread Safety + * This function may be called from any thread. * + * @sa @ref context_current * @sa glfwGetCurrentContext * + * @since Added in GLFW 3.0. + * * @ingroup context */ GLFWAPI void glfwMakeContextCurrent(GLFWwindow* window); /*! @brief Returns the window whose context is current on the calling thread. * - * This function returns the window whose context is current on the calling - * thread. + * This function returns the window whose OpenGL or OpenGL ES context is + * current on the calling thread. * * @return The window whose context is current, or `NULL` if no window's * context is current. * - * @remarks This function may be called from secondary threads. + * @par Thread Safety + * This function may be called from any thread. * + * @sa @ref context_current * @sa glfwMakeContextCurrent * + * @since Added in GLFW 3.0. + * * @ingroup context */ GLFWAPI GLFWwindow* glfwGetCurrentContext(void); @@ -2170,24 +3207,28 @@ GLFWAPI GLFWwindow* glfwGetCurrentContext(void); * * @param[in] window The window whose buffers to swap. * - * @remarks This function may be called from secondary threads. - * - * @par New in GLFW 3 - * This function no longer calls @ref glfwPollEvents. You need to call it or - * @ref glfwWaitEvents yourself. + * @par Thread Safety + * This function may be called from any thread. * + * @sa @ref buffer_swap * @sa glfwSwapInterval * - * @ingroup context + * @since Added in GLFW 1.0. + * + * @par + * __GLFW 3:__ Added window handle parameter. + * + * @ingroup window */ GLFWAPI void glfwSwapBuffers(GLFWwindow* window); /*! @brief Sets the swap interval for the current context. * * This function sets the swap interval for the current context, i.e. the - * number of screen updates to wait before swapping the buffers of a window and - * returning from @ref glfwSwapBuffers. This is sometimes called 'vertical - * synchronization', 'vertical retrace synchronization' or 'vsync'. + * number of screen updates to wait from the time @ref glfwSwapBuffers was + * called before swapping the buffers and returning. This is sometimes called + * _vertical synchronization_, _vertical retrace synchronization_ or just + * _vsync_. * * Contexts that support either of the `WGL_EXT_swap_control_tear` and * `GLX_EXT_swap_control_tear` extensions also accept negative swap intervals, @@ -2196,17 +3237,29 @@ GLFWAPI void glfwSwapBuffers(GLFWwindow* window); * glfwExtensionSupported. For more information about swap tearing, see the * extension specifications. * + * A context must be current on the calling thread. Calling this function + * without a current context will cause a @ref GLFW_NO_CURRENT_CONTEXT error. + * * @param[in] interval The minimum number of screen updates to wait for * until the buffers are swapped by @ref glfwSwapBuffers. * - * @remarks This function may be called from secondary threads. + * @note This function is not called during window creation, leaving the swap + * interval set to whatever is the default on that platform. This is done + * because some swap interval extensions used by GLFW do not allow the swap + * interval to be reset to zero once it has been set to a non-zero value. * * @note Some GPU drivers do not honor the requested swap interval, either * because of user settings that override the request or due to bugs in the * driver. * + * @par Thread Safety + * This function may be called from any thread. + * + * @sa @ref buffer_swap * @sa glfwSwapBuffers * + * @since Added in GLFW 1.0. + * * @ingroup context */ GLFWAPI void glfwSwapInterval(int interval); @@ -2214,19 +3267,28 @@ GLFWAPI void glfwSwapInterval(int interval); /*! @brief Returns whether the specified extension is available. * * This function returns whether the specified - * [OpenGL or context creation API extension](@ref context_glext) is supported - * by the current context. For example, on Windows both the OpenGL and WGL - * extension strings are checked. + * [API extension](@ref context_glext) is supported by the current OpenGL or + * OpenGL ES context. It searches both for OpenGL and OpenGL ES extension and + * platform-specific context creation API extensions. + * + * A context must be current on the calling thread. Calling this function + * without a current context will cause a @ref GLFW_NO_CURRENT_CONTEXT error. + * + * As this functions retrieves and searches one or more extension strings each + * call, it is recommended that you cache its results if it is going to be used + * frequently. The extension strings will not change during the lifetime of + * a context, so there is no danger in doing this. * * @param[in] extension The ASCII encoded name of the extension. * @return `GL_TRUE` if the extension is available, or `GL_FALSE` otherwise. * - * @remarks This function may be called from secondary threads. + * @par Thread Safety + * This function may be called from any thread. * - * @note As this functions searches one or more extension strings on each call, - * it is recommended that you cache its results if it's going to be used - * frequently. The extension strings will not change during the lifetime of - * a context, so there is no danger in doing this. + * @sa @ref context_glext + * @sa glfwGetProcAddress + * + * @since Added in GLFW 1.0. * * @ingroup context */ @@ -2239,15 +3301,27 @@ GLFWAPI int glfwExtensionSupported(const char* extension); * [client API or extension function](@ref context_glext), if it is supported * by the current context. * + * A context must be current on the calling thread. Calling this function + * without a current context will cause a @ref GLFW_NO_CURRENT_CONTEXT error. + * * @param[in] procname The ASCII encoded name of the function. * @return The address of the function, or `NULL` if the function is - * unavailable. + * unavailable or an [error](@ref error_handling) occurred. * - * @remarks This function may be called from secondary threads. + * @note The addresses of a given function is not guaranteed to be the same + * between contexts. * - * @note The addresses of these functions are not guaranteed to be the same for - * all contexts, especially if they use different client APIs or even different - * context creation hints. + * @par Pointer Lifetime + * The returned function pointer is valid until the context is destroyed or the + * library is terminated. + * + * @par Thread Safety + * This function may be called from any thread. + * + * @sa @ref context_glext + * @sa glfwExtensionSupported + * + * @since Added in GLFW 1.0. * * @ingroup context */ diff --git a/examples/opengl_example/glfw/include/GLFW/glfw3native.h b/examples/opengl_example/glfw/include/GLFW/glfw3native.h index d570f587..b3ce7482 100644 --- a/examples/opengl_example/glfw/include/GLFW/glfw3native.h +++ b/examples/opengl_example/glfw/include/GLFW/glfw3native.h @@ -1,5 +1,5 @@ /************************************************************************* - * GLFW 3.0 - www.glfw.org + * GLFW 3.1 - www.glfw.org * A library for OpenGL, window and input *------------------------------------------------------------------------ * Copyright (c) 2002-2006 Marcus Geelnard @@ -40,13 +40,13 @@ extern "C" { /*! @defgroup native Native access * - * **By using the native API, you assert that you know what you're doing and - * how to fix problems caused by using it. If you don't, you shouldn't be - * using it.** + * **By using the native access functions you assert that you know what you're + * doing and how to fix problems caused by using them. If you don't, you + * shouldn't be using them.** * * Before the inclusion of @ref glfw3native.h, you must define exactly one - * window API macro and exactly one context API macro. Failure to do this - * will cause a compile-time error. + * window system API macro and exactly one context creation API macro. Failure + * to do this will cause a compile-time error. * * The available window API macros are: * * `GLFW_EXPOSE_NATIVE_WIN32` @@ -71,8 +71,13 @@ extern "C" { *************************************************************************/ #if defined(GLFW_EXPOSE_NATIVE_WIN32) + // This is a workaround for the fact that glfw3.h needs to export APIENTRY (for + // example to allow applications to correctly declare a GL_ARB_debug_output + // callback) but windows.h assumes no one will define APIENTRY before it does + #undef APIENTRY #include #elif defined(GLFW_EXPOSE_NATIVE_COCOA) + #include #if defined(__OBJC__) #import #else @@ -80,8 +85,9 @@ extern "C" { #endif #elif defined(GLFW_EXPOSE_NATIVE_X11) #include + #include #else - #error "No window API specified" + #error "No window API selected" #endif #if defined(GLFW_EXPOSE_NATIVE_WGL) @@ -93,7 +99,7 @@ extern "C" { #elif defined(GLFW_EXPOSE_NATIVE_EGL) #include #else - #error "No context API specified" + #error "No context API selected" #endif @@ -102,8 +108,49 @@ extern "C" { *************************************************************************/ #if defined(GLFW_EXPOSE_NATIVE_WIN32) +/*! @brief Returns the adapter device name of the specified monitor. + * + * @return The UTF-8 encoded adapter device name (for example `\\.\DISPLAY1`) + * of the specified monitor, or `NULL` if an [error](@ref error_handling) + * occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.1. + * + * @ingroup native + */ +GLFWAPI const char* glfwGetWin32Adapter(GLFWmonitor* monitor); + +/*! @brief Returns the display device name of the specified monitor. + * + * @return The UTF-8 encoded display device name (for example + * `\\.\DISPLAY1\Monitor0`) of the specified monitor, or `NULL` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.1. + * + * @ingroup native + */ +GLFWAPI const char* glfwGetWin32Monitor(GLFWmonitor* monitor); + /*! @brief Returns the `HWND` of the specified window. - * @return The `HWND` of the specified window. + * + * @return The `HWND` of the specified window, or `NULL` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI HWND glfwGetWin32Window(GLFWwindow* window); @@ -111,15 +158,48 @@ GLFWAPI HWND glfwGetWin32Window(GLFWwindow* window); #if defined(GLFW_EXPOSE_NATIVE_WGL) /*! @brief Returns the `HGLRC` of the specified window. - * @return The `HGLRC` of the specified window. + * + * @return The `HGLRC` of the specified window, or `NULL` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI HGLRC glfwGetWGLContext(GLFWwindow* window); #endif #if defined(GLFW_EXPOSE_NATIVE_COCOA) +/*! @brief Returns the `CGDirectDisplayID` of the specified monitor. + * + * @return The `CGDirectDisplayID` of the specified monitor, or + * `kCGNullDirectDisplay` if an [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.1. + * + * @ingroup native + */ +GLFWAPI CGDirectDisplayID glfwGetCocoaMonitor(GLFWmonitor* monitor); + /*! @brief Returns the `NSWindow` of the specified window. - * @return The `NSWindow` of the specified window. + * + * @return The `NSWindow` of the specified window, or `nil` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI id glfwGetCocoaWindow(GLFWwindow* window); @@ -127,7 +207,16 @@ GLFWAPI id glfwGetCocoaWindow(GLFWwindow* window); #if defined(GLFW_EXPOSE_NATIVE_NSGL) /*! @brief Returns the `NSOpenGLContext` of the specified window. - * @return The `NSOpenGLContext` of the specified window. + * + * @return The `NSOpenGLContext` of the specified window, or `nil` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI id glfwGetNSGLContext(GLFWwindow* window); @@ -135,12 +224,61 @@ GLFWAPI id glfwGetNSGLContext(GLFWwindow* window); #if defined(GLFW_EXPOSE_NATIVE_X11) /*! @brief Returns the `Display` used by GLFW. - * @return The `Display` used by GLFW. + * + * @return The `Display` used by GLFW, or `NULL` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI Display* glfwGetX11Display(void); + +/*! @brief Returns the `RRCrtc` of the specified monitor. + * + * @return The `RRCrtc` of the specified monitor, or `None` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.1. + * + * @ingroup native + */ +GLFWAPI RRCrtc glfwGetX11Adapter(GLFWmonitor* monitor); + +/*! @brief Returns the `RROutput` of the specified monitor. + * + * @return The `RROutput` of the specified monitor, or `None` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.1. + * + * @ingroup native + */ +GLFWAPI RROutput glfwGetX11Monitor(GLFWmonitor* monitor); + /*! @brief Returns the `Window` of the specified window. - * @return The `Window` of the specified window. + * + * @return The `Window` of the specified window, or `None` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI Window glfwGetX11Window(GLFWwindow* window); @@ -148,7 +286,16 @@ GLFWAPI Window glfwGetX11Window(GLFWwindow* window); #if defined(GLFW_EXPOSE_NATIVE_GLX) /*! @brief Returns the `GLXContext` of the specified window. - * @return The `GLXContext` of the specified window. + * + * @return The `GLXContext` of the specified window, or `NULL` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI GLXContext glfwGetGLXContext(GLFWwindow* window); @@ -156,17 +303,46 @@ GLFWAPI GLXContext glfwGetGLXContext(GLFWwindow* window); #if defined(GLFW_EXPOSE_NATIVE_EGL) /*! @brief Returns the `EGLDisplay` used by GLFW. - * @return The `EGLDisplay` used by GLFW. + * + * @return The `EGLDisplay` used by GLFW, or `EGL_NO_DISPLAY` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI EGLDisplay glfwGetEGLDisplay(void); + /*! @brief Returns the `EGLContext` of the specified window. - * @return The `EGLContext` of the specified window. + * + * @return The `EGLContext` of the specified window, or `EGL_NO_CONTEXT` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI EGLContext glfwGetEGLContext(GLFWwindow* window); + /*! @brief Returns the `EGLSurface` of the specified window. - * @return The `EGLSurface` of the specified window. + * + * @return The `EGLSurface` of the specified window, or `EGL_NO_SURFACE` if an + * [error](@ref error_handling) occurred. + * + * @par Thread Safety + * This function may be called from any thread. Access is not synchronized. + * + * @par History + * Added in GLFW 3.0. + * * @ingroup native */ GLFWAPI EGLSurface glfwGetEGLSurface(GLFWwindow* window); diff --git a/examples/opengl_example/glfw/lib-msvc100/glfw3.lib b/examples/opengl_example/glfw/lib-msvc100/glfw3.lib deleted file mode 100644 index 777b5b6cefdb2719f3ed19d69db00c92e30b5a83..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 121560 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 Od(k0V?c>UJAo#z??x9ft diff --git a/examples/opengl_example/glfw/lib-msvc110/glfw3.lib b/examples/opengl_example/glfw/lib-msvc110/glfw3.lib deleted file mode 100644 index c5c9cbae9dc033ef0cf48652aefe05ee324c5105..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 121720 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 aU_IiI0n~nyC9BpQgHHDSSoY@r*#8fJe)rq} diff --git a/examples/opengl_example/glfw/lib-vc2010-32/glfw3.lib b/examples/opengl_example/glfw/lib-vc2010-32/glfw3.lib new file mode 100644 index 0000000000000000000000000000000000000000..014458efed8cff308e876c22c2e4832c705079eb GIT binary patch literal 129546 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 L?xao99RL3TeG$IW literal 0 HcmV?d00001 diff --git a/examples/opengl_example/opengl_example.vcxproj b/examples/opengl_example/opengl_example.vcxproj index 6dc90454..c92d0acc 100644 --- a/examples/opengl_example/opengl_example.vcxproj +++ b/examples/opengl_example/opengl_example.vcxproj @@ -54,7 +54,7 @@ true - $(ProjectDir)\glfw\lib-msvc100;$(ProjectDir)\glew\lib\Release\Win32;%(AdditionalLibraryDirectories) + $(ProjectDir)\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) opengl32.lib;glfw3.lib;%(AdditionalDependencies) Console msvcrt.lib @@ -72,7 +72,7 @@ true true true - $(ProjectDir)\glfw\lib-msvc100;$(ProjectDir)\glew\lib\Release\Win32;%(AdditionalLibraryDirectories) + $(ProjectDir)\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) opengl32.lib;glfw3.lib;%(AdditionalDependencies) Console From ef957b7878e1ea3d477ab35abb49e0f1c767e91a Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 10:04:30 +0000 Subject: [PATCH 17/44] Examples: Moved GLFW/GL3W to a common libs/ folder --- examples/{opengl3_example => libs}/gl3w/GL/gl3w.c | 0 examples/{opengl3_example => libs}/gl3w/GL/gl3w.h | 0 .../{opengl3_example => libs}/gl3w/GL/glcorearb.h | 0 .../{opengl_example => libs}/glfw/COPYING.txt | 0 .../glfw/include/GLFW/glfw3.h | 0 .../glfw/include/GLFW/glfw3native.h | 0 .../glfw/lib-vc2010-32/glfw3.lib | Bin examples/opengl3_example/opengl3_example.vcxproj | 14 +++++++------- .../opengl3_example.vcxproj.filters | 6 +++--- examples/opengl_example/opengl_example.vcxproj | 8 ++++---- 10 files changed, 14 insertions(+), 14 deletions(-) rename examples/{opengl3_example => libs}/gl3w/GL/gl3w.c (100%) rename examples/{opengl3_example => libs}/gl3w/GL/gl3w.h (100%) rename examples/{opengl3_example => libs}/gl3w/GL/glcorearb.h (100%) rename examples/{opengl_example => libs}/glfw/COPYING.txt (100%) rename examples/{opengl_example => libs}/glfw/include/GLFW/glfw3.h (100%) rename examples/{opengl_example => libs}/glfw/include/GLFW/glfw3native.h (100%) rename examples/{opengl_example => libs}/glfw/lib-vc2010-32/glfw3.lib (100%) diff --git a/examples/opengl3_example/gl3w/GL/gl3w.c b/examples/libs/gl3w/GL/gl3w.c similarity index 100% rename from examples/opengl3_example/gl3w/GL/gl3w.c rename to examples/libs/gl3w/GL/gl3w.c diff --git a/examples/opengl3_example/gl3w/GL/gl3w.h b/examples/libs/gl3w/GL/gl3w.h similarity index 100% rename from examples/opengl3_example/gl3w/GL/gl3w.h rename to examples/libs/gl3w/GL/gl3w.h diff --git a/examples/opengl3_example/gl3w/GL/glcorearb.h b/examples/libs/gl3w/GL/glcorearb.h similarity index 100% rename from examples/opengl3_example/gl3w/GL/glcorearb.h rename to examples/libs/gl3w/GL/glcorearb.h diff --git a/examples/opengl_example/glfw/COPYING.txt b/examples/libs/glfw/COPYING.txt similarity index 100% rename from examples/opengl_example/glfw/COPYING.txt rename to examples/libs/glfw/COPYING.txt diff --git a/examples/opengl_example/glfw/include/GLFW/glfw3.h b/examples/libs/glfw/include/GLFW/glfw3.h similarity index 100% rename from examples/opengl_example/glfw/include/GLFW/glfw3.h rename to examples/libs/glfw/include/GLFW/glfw3.h diff --git a/examples/opengl_example/glfw/include/GLFW/glfw3native.h b/examples/libs/glfw/include/GLFW/glfw3native.h similarity index 100% rename from examples/opengl_example/glfw/include/GLFW/glfw3native.h rename to examples/libs/glfw/include/GLFW/glfw3native.h diff --git a/examples/opengl_example/glfw/lib-vc2010-32/glfw3.lib b/examples/libs/glfw/lib-vc2010-32/glfw3.lib similarity index 100% rename from examples/opengl_example/glfw/lib-vc2010-32/glfw3.lib rename to examples/libs/glfw/lib-vc2010-32/glfw3.lib diff --git a/examples/opengl3_example/opengl3_example.vcxproj b/examples/opengl3_example/opengl3_example.vcxproj index 3dd9afdf..936c688d 100644 --- a/examples/opengl3_example/opengl3_example.vcxproj +++ b/examples/opengl3_example/opengl3_example.vcxproj @@ -50,11 +50,11 @@ Level3 Disabled - $(ProjectDir)..\opengl_example\glfw\include;gl3w;..\..;%(AdditionalIncludeDirectories) + $(SolutionDir)\libs\glfw\include;$(SolutionDir)\libs\gl3w;..\..;%(AdditionalIncludeDirectories) true - $(ProjectDir)..\opengl_example\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) + $(SolutionDir)\libs\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) opengl32.lib;glfw3.lib;%(AdditionalDependencies) Console msvcrt.lib @@ -66,13 +66,13 @@ MaxSpeed true true - $(ProjectDir)..\opengl_example\glfw\include;gl3w;..\..;%(AdditionalIncludeDirectories) + $(SolutionDir)\libs\glfw\include;$(SolutionDir)\libs\gl3w;..\..;%(AdditionalIncludeDirectories) true true true - $(ProjectDir)..\opengl_example\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) + $(SolutionDir)\libs\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) opengl32.lib;glfw3.lib;%(AdditionalDependencies) Console @@ -81,15 +81,15 @@ - + - - + + diff --git a/examples/opengl3_example/opengl3_example.vcxproj.filters b/examples/opengl3_example/opengl3_example.vcxproj.filters index 1c00552d..544bd2e7 100644 --- a/examples/opengl3_example/opengl3_example.vcxproj.filters +++ b/examples/opengl3_example/opengl3_example.vcxproj.filters @@ -22,7 +22,7 @@ sources - + gl3w @@ -36,10 +36,10 @@ sources - + gl3w - + gl3w diff --git a/examples/opengl_example/opengl_example.vcxproj b/examples/opengl_example/opengl_example.vcxproj index c92d0acc..1d4dbc55 100644 --- a/examples/opengl_example/opengl_example.vcxproj +++ b/examples/opengl_example/opengl_example.vcxproj @@ -50,11 +50,11 @@ Level3 Disabled - $(ProjectDir)\glfw\include;..\..;%(AdditionalIncludeDirectories) + $(SolutionDir)\libs\glfw\include;..\..;%(AdditionalIncludeDirectories) true - $(ProjectDir)\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) + $(SolutionDir)\libs\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) opengl32.lib;glfw3.lib;%(AdditionalDependencies) Console msvcrt.lib @@ -66,13 +66,13 @@ MaxSpeed true true - $(ProjectDir)\glfw\include;..\..;%(AdditionalIncludeDirectories) + $(SolutionDir)\libs\glfw\include;..\..;%(AdditionalIncludeDirectories) true true true - $(ProjectDir)\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) + $(SolutionDir)\libs\glfw\lib-vc2010-32;%(AdditionalLibraryDirectories) opengl32.lib;glfw3.lib;%(AdditionalDependencies) Console From e2c4ba21bc3f800b8beb7e3ab53b3651cb68f6ca Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 10:09:53 +0000 Subject: [PATCH 18/44] Examples: Added 64-bit projects for MSVC --- .../directx11_example.vcxproj | 63 ++++++++++++++++ .../directx9_example/directx9_example.vcxproj | 63 ++++++++++++++++ examples/imgui_examples_msvc2010.sln | 18 +++++ examples/libs/glfw/lib-vc2010-64/glfw3.lib | Bin 0 -> 202450 bytes .../opengl3_example/opengl3_example.vcxproj | 68 ++++++++++++++++++ .../opengl_example/opengl_example.vcxproj | 68 ++++++++++++++++++ 6 files changed, 280 insertions(+) create mode 100644 examples/libs/glfw/lib-vc2010-64/glfw3.lib diff --git a/examples/directx11_example/directx11_example.vcxproj b/examples/directx11_example/directx11_example.vcxproj index beddb330..767cbcc2 100644 --- a/examples/directx11_example/directx11_example.vcxproj +++ b/examples/directx11_example/directx11_example.vcxproj @@ -5,10 +5,18 @@ Debug Win32 + + Debug + x64 + Release Win32 + + Release + x64 + {9F316E83-5AE5-4939-A723-305A94F48005} @@ -20,30 +28,55 @@ true Unicode + + Application + true + Unicode + Application false true Unicode + + Application + false + true + Unicode + + + + + + + $(ProjectDir)$(Configuration)\ $(ProjectDir)$(Configuration)\ + + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ $(ProjectDir)$(Configuration)\ + + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ + Level3 @@ -57,6 +90,19 @@ Console + + + Level3 + Disabled + ..\..;$(DXSDK_DIR)/Include;%(AdditionalIncludeDirectories) + + + true + d3d11.lib;d3dcompiler.lib;dxgi.lib;%(AdditionalDependencies) + $(DXSDK_DIR)/Lib/x64;%(AdditionalLibraryDirectories) + Console + + Level3 @@ -74,6 +120,23 @@ Console + + + Level3 + MaxSpeed + true + true + ..\..;$(DXSDK_DIR)/Include;%(AdditionalIncludeDirectories) + + + true + true + true + d3d11.lib;d3dcompiler.lib;dxgi.lib;imm32.lib;%(AdditionalDependencies) + $(DXSDK_DIR)/Lib/x64;%(AdditionalLibraryDirectories) + Console + + diff --git a/examples/directx9_example/directx9_example.vcxproj b/examples/directx9_example/directx9_example.vcxproj index 3b638b7b..54cc2100 100644 --- a/examples/directx9_example/directx9_example.vcxproj +++ b/examples/directx9_example/directx9_example.vcxproj @@ -5,10 +5,18 @@ Debug Win32 + + Debug + x64 + Release Win32 + + Release + x64 + {4165A294-21F2-44CA-9B38-E3F935ABADF5} @@ -20,30 +28,55 @@ true Unicode + + Application + true + Unicode + Application false true Unicode + + Application + false + true + Unicode + + + + + + + $(ProjectDir)$(Configuration)\ $(ProjectDir)$(Configuration)\ + + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ $(ProjectDir)$(Configuration)\ + + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ + Level3 @@ -57,6 +90,19 @@ Console + + + Level3 + Disabled + ..\..;$(DXSDK_DIR)Include;%(AdditionalIncludeDirectories) + + + true + $(DXSDK_DIR)Lib\x64;%(AdditionalLibraryDirectories) + d3d9.lib;d3dx9d.lib;dxerr.lib;dxguid.lib;%(AdditionalDependencies) + Console + + Level3 @@ -74,6 +120,23 @@ Console + + + Level3 + MaxSpeed + true + true + ..\..;$(DXSDK_DIR)Include;%(AdditionalIncludeDirectories) + + + true + true + true + $(DXSDK_DIR)Lib\x64;%(AdditionalLibraryDirectories) + d3d9.lib;d3dx9d.lib;dxerr.lib;dxguid.lib;%(AdditionalDependencies) + Console + + diff --git a/examples/imgui_examples_msvc2010.sln b/examples/imgui_examples_msvc2010.sln index 1ff20521..e8e0bcad 100644 --- a/examples/imgui_examples_msvc2010.sln +++ b/examples/imgui_examples_msvc2010.sln @@ -12,25 +12,43 @@ EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Win32 = Debug|Win32 + Debug|x64 = Debug|x64 Release|Win32 = Release|Win32 + Release|x64 = Release|x64 EndGlobalSection GlobalSection(ProjectConfigurationPlatforms) = postSolution {9CDA7840-B7A5-496D-A527-E95571496D18}.Debug|Win32.ActiveCfg = Debug|Win32 {9CDA7840-B7A5-496D-A527-E95571496D18}.Debug|Win32.Build.0 = Debug|Win32 + {9CDA7840-B7A5-496D-A527-E95571496D18}.Debug|x64.ActiveCfg = Debug|x64 + {9CDA7840-B7A5-496D-A527-E95571496D18}.Debug|x64.Build.0 = Debug|x64 {9CDA7840-B7A5-496D-A527-E95571496D18}.Release|Win32.ActiveCfg = Release|Win32 {9CDA7840-B7A5-496D-A527-E95571496D18}.Release|Win32.Build.0 = Release|Win32 + {9CDA7840-B7A5-496D-A527-E95571496D18}.Release|x64.ActiveCfg = Release|x64 + {9CDA7840-B7A5-496D-A527-E95571496D18}.Release|x64.Build.0 = Release|x64 {4165A294-21F2-44CA-9B38-E3F935ABADF5}.Debug|Win32.ActiveCfg = Debug|Win32 {4165A294-21F2-44CA-9B38-E3F935ABADF5}.Debug|Win32.Build.0 = Debug|Win32 + {4165A294-21F2-44CA-9B38-E3F935ABADF5}.Debug|x64.ActiveCfg = Debug|x64 + {4165A294-21F2-44CA-9B38-E3F935ABADF5}.Debug|x64.Build.0 = Debug|x64 {4165A294-21F2-44CA-9B38-E3F935ABADF5}.Release|Win32.ActiveCfg = Release|Win32 {4165A294-21F2-44CA-9B38-E3F935ABADF5}.Release|Win32.Build.0 = Release|Win32 + {4165A294-21F2-44CA-9B38-E3F935ABADF5}.Release|x64.ActiveCfg = Release|x64 + {4165A294-21F2-44CA-9B38-E3F935ABADF5}.Release|x64.Build.0 = Release|x64 {9F316E83-5AE5-4939-A723-305A94F48005}.Debug|Win32.ActiveCfg = Debug|Win32 {9F316E83-5AE5-4939-A723-305A94F48005}.Debug|Win32.Build.0 = Debug|Win32 + {9F316E83-5AE5-4939-A723-305A94F48005}.Debug|x64.ActiveCfg = Debug|x64 + {9F316E83-5AE5-4939-A723-305A94F48005}.Debug|x64.Build.0 = Debug|x64 {9F316E83-5AE5-4939-A723-305A94F48005}.Release|Win32.ActiveCfg = Release|Win32 {9F316E83-5AE5-4939-A723-305A94F48005}.Release|Win32.Build.0 = Release|Win32 + {9F316E83-5AE5-4939-A723-305A94F48005}.Release|x64.ActiveCfg = Release|x64 + {9F316E83-5AE5-4939-A723-305A94F48005}.Release|x64.Build.0 = Release|x64 {4A1FB5EA-22F5-42A8-AB92-1D2DF5D47FB9}.Debug|Win32.ActiveCfg = Debug|Win32 {4A1FB5EA-22F5-42A8-AB92-1D2DF5D47FB9}.Debug|Win32.Build.0 = Debug|Win32 + {4A1FB5EA-22F5-42A8-AB92-1D2DF5D47FB9}.Debug|x64.ActiveCfg = Debug|x64 + {4A1FB5EA-22F5-42A8-AB92-1D2DF5D47FB9}.Debug|x64.Build.0 = Debug|x64 {4A1FB5EA-22F5-42A8-AB92-1D2DF5D47FB9}.Release|Win32.ActiveCfg = Release|Win32 {4A1FB5EA-22F5-42A8-AB92-1D2DF5D47FB9}.Release|Win32.Build.0 = Release|Win32 + {4A1FB5EA-22F5-42A8-AB92-1D2DF5D47FB9}.Release|x64.ActiveCfg = Release|x64 + {4A1FB5EA-22F5-42A8-AB92-1D2DF5D47FB9}.Release|x64.Build.0 = Release|x64 EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE diff --git a/examples/libs/glfw/lib-vc2010-64/glfw3.lib b/examples/libs/glfw/lib-vc2010-64/glfw3.lib new file mode 100644 index 0000000000000000000000000000000000000000..8cdda39987fe55d400fa1def0738b4bdc8672d51 GIT binary patch literal 202450 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 Win32 + + Debug + x64 + Release Win32 + + Release + x64 + {4a1fb5ea-22f5-42a8-ab92-1d2df5d47fb9} @@ -20,32 +28,59 @@ true MultiByte + + Application + true + MultiByte + Application false true MultiByte + + Application + false + true + MultiByte + + + + + + + $(ProjectDir)$(Configuration)\ $(ProjectDir)$(Configuration)\ $(IncludePath) + + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ + $(IncludePath) + $(ProjectDir)$(Configuration)\ $(ProjectDir)$(Configuration)\ $(IncludePath) + + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ + $(IncludePath) + Level3 @@ -60,6 +95,20 @@ msvcrt.lib + + + Level3 + Disabled + $(SolutionDir)\libs\glfw\include;$(SolutionDir)\libs\gl3w;..\..;%(AdditionalIncludeDirectories) + + + true + $(SolutionDir)\libs\glfw\lib-vc2010-64;%(AdditionalLibraryDirectories) + opengl32.lib;glfw3.lib;%(AdditionalDependencies) + Console + msvcrt.lib + + Level3 @@ -79,6 +128,25 @@ + + + Level3 + MaxSpeed + true + true + $(SolutionDir)\libs\glfw\include;$(SolutionDir)\libs\gl3w;..\..;%(AdditionalIncludeDirectories) + + + true + true + true + $(SolutionDir)\libs\glfw\lib-vc2010-64;%(AdditionalLibraryDirectories) + opengl32.lib;glfw3.lib;%(AdditionalDependencies) + Console + + + + diff --git a/examples/opengl_example/opengl_example.vcxproj b/examples/opengl_example/opengl_example.vcxproj index 1d4dbc55..f39a929f 100644 --- a/examples/opengl_example/opengl_example.vcxproj +++ b/examples/opengl_example/opengl_example.vcxproj @@ -5,10 +5,18 @@ Debug Win32 + + Debug + x64 + Release Win32 + + Release + x64 + {9CDA7840-B7A5-496D-A527-E95571496D18} @@ -20,32 +28,59 @@ true MultiByte + + Application + true + MultiByte + Application false true MultiByte + + Application + false + true + MultiByte + + + + + + + $(ProjectDir)$(Configuration)\ $(ProjectDir)$(Configuration)\ $(IncludePath) + + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ + $(IncludePath) + $(ProjectDir)$(Configuration)\ $(ProjectDir)$(Configuration)\ $(IncludePath) + + $(ProjectDir)$(Configuration)\ + $(ProjectDir)$(Configuration)\ + $(IncludePath) + Level3 @@ -60,6 +95,20 @@ msvcrt.lib + + + Level3 + Disabled + $(SolutionDir)\libs\glfw\include;..\..;%(AdditionalIncludeDirectories) + + + true + $(SolutionDir)\libs\glfw\lib-vc2010-64;%(AdditionalLibraryDirectories) + opengl32.lib;glfw3.lib;%(AdditionalDependencies) + Console + msvcrt.lib + + Level3 @@ -79,6 +128,25 @@ + + + Level3 + MaxSpeed + true + true + $(SolutionDir)\libs\glfw\include;..\..;%(AdditionalIncludeDirectories) + + + true + true + true + $(SolutionDir)\libs\glfw\lib-vc2010-64;%(AdditionalLibraryDirectories) + opengl32.lib;glfw3.lib;%(AdditionalDependencies) + Console + + + + From 8e0f04638d3f9b253d9c8985e4edf978792fdbf7 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 10:24:35 +0000 Subject: [PATCH 19/44] Examples: Fix includes order to avoid conflicts between DXSDK_DIR and Windows SDK --- examples/directx11_example/directx11_example.vcxproj | 8 ++++---- examples/directx9_example/directx9_example.vcxproj | 8 ++++---- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/examples/directx11_example/directx11_example.vcxproj b/examples/directx11_example/directx11_example.vcxproj index 767cbcc2..556b0427 100644 --- a/examples/directx11_example/directx11_example.vcxproj +++ b/examples/directx11_example/directx11_example.vcxproj @@ -81,7 +81,7 @@ Level3 Disabled - ..\..;$(DXSDK_DIR)/Include;%(AdditionalIncludeDirectories) + ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; true @@ -94,7 +94,7 @@ Level3 Disabled - ..\..;$(DXSDK_DIR)/Include;%(AdditionalIncludeDirectories) + ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; true @@ -109,7 +109,7 @@ MaxSpeed true true - ..\..;$(DXSDK_DIR)/Include;%(AdditionalIncludeDirectories) + ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; true @@ -126,7 +126,7 @@ MaxSpeed true true - ..\..;$(DXSDK_DIR)/Include;%(AdditionalIncludeDirectories) + ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; true diff --git a/examples/directx9_example/directx9_example.vcxproj b/examples/directx9_example/directx9_example.vcxproj index 54cc2100..f4dd7ed3 100644 --- a/examples/directx9_example/directx9_example.vcxproj +++ b/examples/directx9_example/directx9_example.vcxproj @@ -81,7 +81,7 @@ Level3 Disabled - ..\..;$(DXSDK_DIR)Include;%(AdditionalIncludeDirectories) + ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; true @@ -94,7 +94,7 @@ Level3 Disabled - ..\..;$(DXSDK_DIR)Include;%(AdditionalIncludeDirectories) + ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; true @@ -109,7 +109,7 @@ MaxSpeed true true - ..\..;$(DXSDK_DIR)Include;%(AdditionalIncludeDirectories) + ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; true @@ -126,7 +126,7 @@ MaxSpeed true true - ..\..;$(DXSDK_DIR)Include;%(AdditionalIncludeDirectories) + ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; true From 60591dd7c6725b888d0a6f4740d43f8784ad98fc Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 10:30:07 +0000 Subject: [PATCH 20/44] Fixed warning and split bit of SliderFloat() into a private ParseFormat() function --- imgui.cpp | 36 ++++++++++++++++++++---------------- 1 file changed, 20 insertions(+), 16 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 36c3f0f5..3d09a330 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4714,6 +4714,25 @@ static bool SliderFloatAsInputText(const char* label, float* v, const ImGuiID& i return value_changed; } +// Parse display precision back from the display format string +static void ParseFormat(const char* fmt, int& decimal_precision) +{ + while ((fmt = strchr(fmt, '%')) != NULL) + { + fmt++; + if (fmt[0] == '%') { fmt++; continue; } // Ignore "%%" + while (*fmt >= '0' && *fmt <= '9') + fmt++; + if (*fmt == '.') + { + decimal_precision = atoi(fmt + 1); + if (decimal_precision < 0 || decimal_precision > 10) + decimal_precision = 3; + } + break; + } +} + // Use power!=1.0 for logarithmic sliders. // Adjust display_format to decorate the value with a prefix or a suffix. // "%.3f" 1.234 @@ -4732,23 +4751,8 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c if (!display_format) display_format = "%.3f"; - - // Parse display precision back from the display format string int decimal_precision = 3; - for (const char* p = display_format; p = strchr(p, '%'); ) - { - p++; - if (p[0] == '%') { p ++; continue; } // Ignore "%%" - while (*p >= '0' && *p <= '9') - p++; - if (*p == '.') - { - decimal_precision = atoi(p+1); - if (decimal_precision < 0 || decimal_precision > 10) - decimal_precision = 3; - } - break; - } + ParseFormat(display_format, decimal_precision); const ImVec2 label_size = CalcTextSize(label, NULL, true); const ImGuiAabb frame_bb(window->DC.CursorPos, window->DC.CursorPos + ImVec2(w, label_size.y) + style.FramePadding*2.0f); From 2b0acc128c6c4dd0f5fe15257442481bbf4e83ad Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 10:32:29 +0000 Subject: [PATCH 21/44] Examples: MSVC: increase warning level from /W3 to /W4 for DirectX examples --- examples/directx11_example/directx11_example.vcxproj | 8 ++++---- examples/directx11_example/imgui_impl_dx11.cpp | 2 +- examples/directx11_example/main.cpp | 3 +++ examples/directx9_example/directx9_example.vcxproj | 8 ++++---- examples/directx9_example/imgui_impl_dx9.cpp | 2 +- examples/directx9_example/main.cpp | 3 +++ 6 files changed, 16 insertions(+), 10 deletions(-) diff --git a/examples/directx11_example/directx11_example.vcxproj b/examples/directx11_example/directx11_example.vcxproj index 556b0427..0ad66c4f 100644 --- a/examples/directx11_example/directx11_example.vcxproj +++ b/examples/directx11_example/directx11_example.vcxproj @@ -79,7 +79,7 @@ - Level3 + Level4 Disabled ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; @@ -92,7 +92,7 @@ - Level3 + Level4 Disabled ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; @@ -105,7 +105,7 @@ - Level3 + Level4 MaxSpeed true true @@ -122,7 +122,7 @@ - Level3 + Level4 MaxSpeed true true diff --git a/examples/directx11_example/imgui_impl_dx11.cpp b/examples/directx11_example/imgui_impl_dx11.cpp index f6ee6b5d..4e209c87 100644 --- a/examples/directx11_example/imgui_impl_dx11.cpp +++ b/examples/directx11_example/imgui_impl_dx11.cpp @@ -146,7 +146,7 @@ static void ImGui_ImplDX11_RenderDrawLists(ImDrawList** const cmd_lists, int cmd g_pd3dDeviceContext->VSSetShader(NULL, NULL, 0); } -LRESULT ImGui_ImplDX11_WndProcHandler(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) +LRESULT ImGui_ImplDX11_WndProcHandler(HWND, UINT msg, WPARAM wParam, LPARAM lParam) { ImGuiIO& io = ImGui::GetIO(); switch (msg) diff --git a/examples/directx11_example/main.cpp b/examples/directx11_example/main.cpp index 571fe1da..3bf4924f 100644 --- a/examples/directx11_example/main.cpp +++ b/examples/directx11_example/main.cpp @@ -130,6 +130,9 @@ LRESULT WINAPI WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) int main(int argc, char** argv) { + (void)argc; + (void)argv; + // Create application window WNDCLASSEX wc = { sizeof(WNDCLASSEX), CS_CLASSDC, WndProc, 0L, 0L, GetModuleHandle(NULL), NULL, LoadCursor(NULL, IDC_ARROW), NULL, NULL, L"ImGui Example", NULL }; RegisterClassEx(&wc); diff --git a/examples/directx9_example/directx9_example.vcxproj b/examples/directx9_example/directx9_example.vcxproj index f4dd7ed3..beaf53ea 100644 --- a/examples/directx9_example/directx9_example.vcxproj +++ b/examples/directx9_example/directx9_example.vcxproj @@ -79,7 +79,7 @@ - Level3 + Level4 Disabled ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; @@ -92,7 +92,7 @@ - Level3 + Level4 Disabled ..\..;%(AdditionalIncludeDirectories);$(DXSDK_DIR)Include; @@ -105,7 +105,7 @@ - Level3 + Level4 MaxSpeed true true @@ -122,7 +122,7 @@ - Level3 + Level4 MaxSpeed true true diff --git a/examples/directx9_example/imgui_impl_dx9.cpp b/examples/directx9_example/imgui_impl_dx9.cpp index b64b70eb..f6f2a11a 100644 --- a/examples/directx9_example/imgui_impl_dx9.cpp +++ b/examples/directx9_example/imgui_impl_dx9.cpp @@ -110,7 +110,7 @@ static void ImGui_ImplDX9_RenderDrawLists(ImDrawList** const cmd_lists, int cmd_ } } -LRESULT ImGui_ImplDX9_WndProcHandler(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) +LRESULT ImGui_ImplDX9_WndProcHandler(HWND, UINT msg, WPARAM wParam, LPARAM lParam) { ImGuiIO& io = ImGui::GetIO(); switch (msg) diff --git a/examples/directx9_example/main.cpp b/examples/directx9_example/main.cpp index 5e0e58d6..07fbb7b4 100644 --- a/examples/directx9_example/main.cpp +++ b/examples/directx9_example/main.cpp @@ -43,6 +43,9 @@ LRESULT WINAPI WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) int main(int argc, char** argv) { + (void)argc; + (void)argv; + // Create application window WNDCLASSEX wc = { sizeof(WNDCLASSEX), CS_CLASSDC, WndProc, 0L, 0L, GetModuleHandle(NULL), NULL, LoadCursor(NULL, IDC_ARROW), NULL, NULL, L"ImGui Example", NULL }; RegisterClassEx(&wc); From 7c9bd7151248c2e03c9f4cf7f68d75df9fc5eff2 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 10:41:42 +0000 Subject: [PATCH 22/44] Examples: MSVC: increase warning level from /W3 to /W4 for OpenGL examples --- examples/directx11_example/main.cpp | 5 +---- examples/directx9_example/main.cpp | 5 +---- examples/libs/gl3w/GL/gl3w.c | 5 +++++ examples/opengl3_example/imgui_impl_glfw_gl3.cpp | 8 ++++---- examples/opengl3_example/main.cpp | 4 ++-- examples/opengl3_example/opengl3_example.vcxproj | 8 ++++---- examples/opengl_example/imgui_impl_glfw.cpp | 8 ++++---- examples/opengl_example/main.cpp | 4 ++-- examples/opengl_example/opengl_example.vcxproj | 8 ++++---- 9 files changed, 27 insertions(+), 28 deletions(-) diff --git a/examples/directx11_example/main.cpp b/examples/directx11_example/main.cpp index 3bf4924f..b051fd24 100644 --- a/examples/directx11_example/main.cpp +++ b/examples/directx11_example/main.cpp @@ -128,11 +128,8 @@ LRESULT WINAPI WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) return DefWindowProc(hWnd, msg, wParam, lParam); } -int main(int argc, char** argv) +int main(int, char**) { - (void)argc; - (void)argv; - // Create application window WNDCLASSEX wc = { sizeof(WNDCLASSEX), CS_CLASSDC, WndProc, 0L, 0L, GetModuleHandle(NULL), NULL, LoadCursor(NULL, IDC_ARROW), NULL, NULL, L"ImGui Example", NULL }; RegisterClassEx(&wc); diff --git a/examples/directx9_example/main.cpp b/examples/directx9_example/main.cpp index 07fbb7b4..9a01ba31 100644 --- a/examples/directx9_example/main.cpp +++ b/examples/directx9_example/main.cpp @@ -41,11 +41,8 @@ LRESULT WINAPI WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) return DefWindowProc(hWnd, msg, wParam, lParam); } -int main(int argc, char** argv) +int main(int, char**) { - (void)argc; - (void)argv; - // Create application window WNDCLASSEX wc = { sizeof(WNDCLASSEX), CS_CLASSDC, WndProc, 0L, 0L, GetModuleHandle(NULL), NULL, LoadCursor(NULL, IDC_ARROW), NULL, NULL, L"ImGui Example", NULL }; RegisterClassEx(&wc); diff --git a/examples/libs/gl3w/GL/gl3w.c b/examples/libs/gl3w/GL/gl3w.c index d66b972f..bea6e40e 100644 --- a/examples/libs/gl3w/GL/gl3w.c +++ b/examples/libs/gl3w/GL/gl3w.c @@ -1,5 +1,10 @@ #include +#ifdef _MSC_VER +#pragma warning (disable: 4055) // warning C4055: 'type cast' : from data pointer 'void *' to function pointer +#pragma warning (disable: 4152) // warning C4152: nonstandard extension, function/data pointer conversion in expression +#endif + #ifdef _WIN32 #define WIN32_LEAN_AND_MEAN 1 #include diff --git a/examples/opengl3_example/imgui_impl_glfw_gl3.cpp b/examples/opengl3_example/imgui_impl_glfw_gl3.cpp index fa669d9c..07bf5a57 100644 --- a/examples/opengl3_example/imgui_impl_glfw_gl3.cpp +++ b/examples/opengl3_example/imgui_impl_glfw_gl3.cpp @@ -123,18 +123,18 @@ static void ImGui_ImplGlfwGL3_SetClipboardText(const char* text) glfwSetClipboardString(g_Window, text); } -void ImGui_ImplGlfwGL3_MouseButtonCallback(GLFWwindow* window, int button, int action, int mods) +void ImGui_ImplGlfwGL3_MouseButtonCallback(GLFWwindow*, int button, int action, int /*mods*/) { if (action == GLFW_PRESS && button >= 0 && button < 3) g_MousePressed[button] = true; } -void ImGui_ImplGlfwGL3_ScrollCallback(GLFWwindow* window, double xoffset, double yoffset) +void ImGui_ImplGlfwGL3_ScrollCallback(GLFWwindow*, double /*xoffset*/, double yoffset) { g_MouseWheel += (float)yoffset; // Use fractional mouse wheel, 1.0 unit 5 lines. } -void ImGui_ImplGlfwGL3_KeyCallback(GLFWwindow* window, int key, int scancode, int action, int mods) +void ImGui_ImplGlfwGL3_KeyCallback(GLFWwindow*, int key, int, int action, int mods) { ImGuiIO& io = ImGui::GetIO(); if (action == GLFW_PRESS) @@ -146,7 +146,7 @@ void ImGui_ImplGlfwGL3_KeyCallback(GLFWwindow* window, int key, int scancode, in io.KeyAlt = (mods & GLFW_MOD_ALT) != 0; } -void ImGui_ImplGlfwGL3_CharCallback(GLFWwindow* window, unsigned int c) +void ImGui_ImplGlfwGL3_CharCallback(GLFWwindow*, unsigned int c) { ImGuiIO& io = ImGui::GetIO(); if (c > 0 && c < 0x10000) diff --git a/examples/opengl3_example/main.cpp b/examples/opengl3_example/main.cpp index 9651970b..dc88a7c6 100644 --- a/examples/opengl3_example/main.cpp +++ b/examples/opengl3_example/main.cpp @@ -8,10 +8,10 @@ static void error_callback(int error, const char* description) { - fprintf(stderr, "Error: %s\n", description); + fprintf(stderr, "Error %d: %s\n", error, description); } -int main(int argc, char** argv) +int main(int, char**) { // Setup window glfwSetErrorCallback(error_callback); diff --git a/examples/opengl3_example/opengl3_example.vcxproj b/examples/opengl3_example/opengl3_example.vcxproj index 2a2c0b4f..202c5f73 100644 --- a/examples/opengl3_example/opengl3_example.vcxproj +++ b/examples/opengl3_example/opengl3_example.vcxproj @@ -83,7 +83,7 @@ - Level3 + Level4 Disabled $(SolutionDir)\libs\glfw\include;$(SolutionDir)\libs\gl3w;..\..;%(AdditionalIncludeDirectories) @@ -97,7 +97,7 @@ - Level3 + Level4 Disabled $(SolutionDir)\libs\glfw\include;$(SolutionDir)\libs\gl3w;..\..;%(AdditionalIncludeDirectories) @@ -111,7 +111,7 @@ - Level3 + Level4 MaxSpeed true true @@ -130,7 +130,7 @@ - Level3 + Level4 MaxSpeed true true diff --git a/examples/opengl_example/imgui_impl_glfw.cpp b/examples/opengl_example/imgui_impl_glfw.cpp index a6b9cb40..00313e45 100644 --- a/examples/opengl_example/imgui_impl_glfw.cpp +++ b/examples/opengl_example/imgui_impl_glfw.cpp @@ -103,18 +103,18 @@ static void ImGui_ImplGlfw_SetClipboardText(const char* text) glfwSetClipboardString(g_Window, text); } -void ImGui_ImplGlfw_MouseButtonCallback(GLFWwindow* window, int button, int action, int mods) +void ImGui_ImplGlfw_MouseButtonCallback(GLFWwindow*, int button, int action, int /*mods*/) { if (action == GLFW_PRESS && button >= 0 && button < 3) g_MousePressed[button] = true; } -void ImGui_ImplGlfw_ScrollCallback(GLFWwindow* window, double xoffset, double yoffset) +void ImGui_ImplGlfw_ScrollCallback(GLFWwindow*, double /*xoffset*/, double yoffset) { g_MouseWheel += (float)yoffset; // Use fractional mouse wheel, 1.0 unit 5 lines. } -void ImGui_ImplGlFw_KeyCallback(GLFWwindow* window, int key, int scancode, int action, int mods) +void ImGui_ImplGlFw_KeyCallback(GLFWwindow*, int key, int, int action, int mods) { ImGuiIO& io = ImGui::GetIO(); if (action == GLFW_PRESS) @@ -126,7 +126,7 @@ void ImGui_ImplGlFw_KeyCallback(GLFWwindow* window, int key, int scancode, int a io.KeyAlt = (mods & GLFW_MOD_ALT) != 0; } -void ImGui_ImplGlfw_CharCallback(GLFWwindow* window, unsigned int c) +void ImGui_ImplGlfw_CharCallback(GLFWwindow*, unsigned int c) { ImGuiIO& io = ImGui::GetIO(); if (c > 0 && c < 0x10000) diff --git a/examples/opengl_example/main.cpp b/examples/opengl_example/main.cpp index c9d303d0..471f9a29 100644 --- a/examples/opengl_example/main.cpp +++ b/examples/opengl_example/main.cpp @@ -7,10 +7,10 @@ static void error_callback(int error, const char* description) { - fprintf(stderr, "Error: %s\n", description); + fprintf(stderr, "Error %d: %s\n", error, description); } -int main(int argc, char** argv) +int main(int, char**) { // Setup window glfwSetErrorCallback(error_callback); diff --git a/examples/opengl_example/opengl_example.vcxproj b/examples/opengl_example/opengl_example.vcxproj index f39a929f..718fedc5 100644 --- a/examples/opengl_example/opengl_example.vcxproj +++ b/examples/opengl_example/opengl_example.vcxproj @@ -83,7 +83,7 @@ - Level3 + Level4 Disabled $(SolutionDir)\libs\glfw\include;..\..;%(AdditionalIncludeDirectories) @@ -97,7 +97,7 @@ - Level3 + Level4 Disabled $(SolutionDir)\libs\glfw\include;..\..;%(AdditionalIncludeDirectories) @@ -111,7 +111,7 @@ - Level3 + Level4 MaxSpeed true true @@ -130,7 +130,7 @@ - Level3 + Level4 MaxSpeed true true From e97d02a294434d12419fc267cb9c350d752994bd Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 10:49:26 +0000 Subject: [PATCH 23/44] Examples: Fixed Clang warnings in DirectX examples --- examples/directx11_example/imgui_impl_dx11.cpp | 4 ++-- examples/directx9_example/imgui_impl_dx9.cpp | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/examples/directx11_example/imgui_impl_dx11.cpp b/examples/directx11_example/imgui_impl_dx11.cpp index 4e209c87..2502b98d 100644 --- a/examples/directx11_example/imgui_impl_dx11.cpp +++ b/examples/directx11_example/imgui_impl_dx11.cpp @@ -171,11 +171,11 @@ LRESULT ImGui_ImplDX11_WndProcHandler(HWND, UINT msg, WPARAM wParam, LPARAM lPar io.MousePos.y = (signed short)(lParam >> 16); return true; case WM_KEYDOWN: - if (wParam >= 0 && wParam < 256) + if (wParam < 256) io.KeysDown[wParam] = 1; return true; case WM_KEYUP: - if (wParam >= 0 && wParam < 256) + if (wParam < 256) io.KeysDown[wParam] = 0; return true; case WM_CHAR: diff --git a/examples/directx9_example/imgui_impl_dx9.cpp b/examples/directx9_example/imgui_impl_dx9.cpp index f6f2a11a..60302d4c 100644 --- a/examples/directx9_example/imgui_impl_dx9.cpp +++ b/examples/directx9_example/imgui_impl_dx9.cpp @@ -135,11 +135,11 @@ LRESULT ImGui_ImplDX9_WndProcHandler(HWND, UINT msg, WPARAM wParam, LPARAM lPara io.MousePos.y = (signed short)(lParam >> 16); return true; case WM_KEYDOWN: - if (wParam >= 0 && wParam < 256) + if (wParam < 256) io.KeysDown[wParam] = 1; return true; case WM_KEYUP: - if (wParam >= 0 && wParam < 256) + if (wParam < 256) io.KeysDown[wParam] = 0; return true; case WM_CHAR: From 55edd522c2788ba6461e6cd7e4cac499001f2ccb Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 11:24:53 +0000 Subject: [PATCH 24/44] Examples: Fixed Makefile --- examples/opengl3_example/Makefile | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/examples/opengl3_example/Makefile b/examples/opengl3_example/Makefile index 208d279f..4357e331 100644 --- a/examples/opengl3_example/Makefile +++ b/examples/opengl3_example/Makefile @@ -10,8 +10,9 @@ #CXX = g++ -OBJS = main.o imgui_impl_glfw_gl3.o gl3w/GL/gl3w.o +OBJS = main.o imgui_impl_glfw_gl3.o OBJS += ../../imgui.o +OBJS += ../libs/gl3w/GL/gl3w.o UNAME_S := $(shell uname -s) @@ -31,7 +32,7 @@ ifeq ($(UNAME_S), Darwin) #APPLE LIBS += -L/usr/local/lib LIBS += -lglfw3 - CXXFLAGS = -I../../ -Igl3w -I/usr/local/Cellar/glew/1.10.0/include -I/usr/local/include + CXXFLAGS = -I../../ -I../libs/gl3w -I/usr/local/Cellar/glew/1.10.0/include -I/usr/local/include CXXFLAGS += -Wall # CXXFLAGS += -D__APPLE__ CFLAGS = $(CXXFLAGS) From 06fbeed41a0423a134082d3db7e337a87d0491a5 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 11:29:09 +0000 Subject: [PATCH 25/44] Examples: Fixed Makefile --- examples/opengl3_example/Makefile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/opengl3_example/Makefile b/examples/opengl3_example/Makefile index 4357e331..cc7ec31e 100644 --- a/examples/opengl3_example/Makefile +++ b/examples/opengl3_example/Makefile @@ -21,7 +21,7 @@ ifeq ($(UNAME_S), Linux) #LINUX ECHO_MESSAGE = "Linux" LIBS = `pkg-config --static --libs glfw3` - CXXFLAGS = -I../../ -Igl3w `pkg-config --cflags glfw3` + CXXFLAGS = -I../../ -I../libs/gl3w `pkg-config --cflags glfw3` CXXFLAGS += -Wall CFLAGS = $(CXXFLAGS) endif From 93befadc0bd7008a06d8691a5470b4cc81c53f1a Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 23:33:12 +0000 Subject: [PATCH 26/44] Slider code tweaks --- imgui.cpp | 160 +++++++++++++++++++++++++++--------------------------- 1 file changed, 79 insertions(+), 81 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 3d09a330..2689f631 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4680,6 +4680,7 @@ static void ApplyNumericalTextInput(const char* buf, float *v) *v = op_v; } +// Create text input in place of a slider (when CTRL+Clicking on slider) static bool SliderFloatAsInputText(const char* label, float* v, const ImGuiID& id, int decimal_precision) { ImGuiState& g = *GImGui; @@ -4749,11 +4750,6 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c const ImGuiID id = window->GetID(label); const float w = ImGui::CalcItemWidth(); - if (!display_format) - display_format = "%.3f"; - int decimal_precision = 3; - ParseFormat(display_format, decimal_precision); - const ImVec2 label_size = CalcTextSize(label, NULL, true); const ImGuiAabb frame_bb(window->DC.CursorPos, window->DC.CursorPos + ImVec2(w, label_size.y) + style.FramePadding*2.0f); const ImGuiAabb slider_bb(frame_bb.Min + style.FramePadding, frame_bb.Max - style.FramePadding); @@ -4766,10 +4762,37 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c return false; } - const bool tab_focus_requested = window->FocusItemRegister(g.ActiveId == id); + const bool hovered = IsHovered(slider_bb, id); + if (hovered) + g.HoveredId = id; - const bool is_unbound = v_min == -FLT_MAX || v_min == FLT_MAX || v_max == -FLT_MAX || v_max == FLT_MAX; - const bool is_logarithmic = abs(power - 1.0f) > 0.0001f; + if (!display_format) + display_format = "%.3f"; + int decimal_precision = 3; + ParseFormat(display_format, decimal_precision); + + const bool tab_focus_requested = window->FocusItemRegister(g.ActiveId == id); + const bool is_unbound = (v_min == -FLT_MAX || v_min == FLT_MAX || v_max == -FLT_MAX || v_max == FLT_MAX); + + bool start_text_input = false; + if (tab_focus_requested || (hovered && g.IO.MouseClicked[0])) + { + SetActiveId(id); + FocusWindow(window); + + const bool is_ctrl_down = g.IO.KeyCtrl; + if (tab_focus_requested || is_ctrl_down || is_unbound) + { + start_text_input = true; + g.SliderAsInputTextId = 0; + } + } + + // Tabbing or CTRL-clicking through slider turns into an input box + if (start_text_input || (g.ActiveId == id && id == g.SliderAsInputTextId)) + return SliderFloatAsInputText(label, v, id, decimal_precision); + + ItemSize(bb); const float grab_size_in_units = 1.0f; // In 'v' units. Probably needs to be parametrized, based on a 'v_step' value? decimal precision? float grab_size_in_pixels; @@ -4799,95 +4822,70 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c } } - const bool hovered = IsHovered(slider_bb, id); - if (hovered) - g.HoveredId = id; - - bool start_text_input = false; - if (tab_focus_requested || (hovered && g.IO.MouseClicked[0])) - { - SetActiveId(id); - FocusWindow(window); - - const bool is_ctrl_down = g.IO.KeyCtrl; - if (tab_focus_requested || is_ctrl_down || is_unbound) - { - start_text_input = true; - g.SliderAsInputTextId = 0; - } - } - - // Tabbing or CTRL-clicking through slider turns into an input box - bool value_changed = false; - if (start_text_input || (g.ActiveId == id && id == g.SliderAsInputTextId)) - { - value_changed = SliderFloatAsInputText(label, v, id, decimal_precision); - return value_changed; - } - - ItemSize(bb); RenderFrame(frame_bb.Min, frame_bb.Max, window->Color(ImGuiCol_FrameBg), true, style.FrameRounding); - // Process clicking on the slider - if (g.ActiveId == id) + const bool is_logarithmic = abs(power - 1.0f) > 0.0001f; + bool value_changed = false; + if (!is_unbound) { - if (!is_unbound && g.IO.MouseDown[0]) + // Process clicking on the slider + if (g.ActiveId == id) { - const float normalized_pos = ImClamp((g.IO.MousePos.x - slider_effective_x1) / slider_effective_w, 0.0f, 1.0f); - - float new_value; - if (is_logarithmic) + if (g.IO.MouseDown[0]) { - // Account for logarithmic scale on both sides of the zero - if (normalized_pos < linear_zero_pos) + const float normalized_pos = ImClamp((g.IO.MousePos.x - slider_effective_x1) / slider_effective_w, 0.0f, 1.0f); + + float new_value; + if (is_logarithmic) { - // Negative: rescale to the negative range before powering - float a = 1.0f - (normalized_pos / linear_zero_pos); - a = powf(a, power); - new_value = ImLerp(ImMin(v_max,0.f), v_min, a); + // Account for logarithmic scale on both sides of the zero + if (normalized_pos < linear_zero_pos) + { + // Negative: rescale to the negative range before powering + float a = 1.0f - (normalized_pos / linear_zero_pos); + a = powf(a, power); + new_value = ImLerp(ImMin(v_max,0.f), v_min, a); + } + else + { + // Positive: rescale to the positive range before powering + float a; + if (fabsf(linear_zero_pos - 1.0f) > 1.e-6) + a = (normalized_pos - linear_zero_pos) / (1.0f - linear_zero_pos); + else + a = normalized_pos; + a = powf(a, power); + new_value = ImLerp(ImMax(v_min,0.0f), v_max, a); + } } else { - // Positive: rescale to the positive range before powering - float a; - if (fabsf(linear_zero_pos - 1.0f) > 1.e-6) - a = (normalized_pos - linear_zero_pos) / (1.0f - linear_zero_pos); - else - a = normalized_pos; - a = powf(a, power); - new_value = ImLerp(ImMax(v_min,0.0f), v_max, a); + // Linear slider + new_value = ImLerp(v_min, v_max, normalized_pos); + } + + // Round past decimal precision + // 0->1, 1->0.1, 2->0.01, etc. + // So when our value is 1.99999 with a precision of 0.001 we'll end up rounding to 2.0 + const float min_step = 1.0f / powf(10.0f, (float)decimal_precision); + const float remainder = fmodf(new_value, min_step); + if (remainder <= min_step*0.5f) + new_value -= remainder; + else + new_value += (min_step - remainder); + + if (*v != new_value) + { + *v = new_value; + value_changed = true; } } else { - // Linear slider - new_value = ImLerp(v_min, v_max, normalized_pos); - } - - // Round past decimal precision - // 0->1, 1->0.1, 2->0.01, etc. - // So when our value is 1.99999 with a precision of 0.001 we'll end up rounding to 2.0 - const float min_step = 1.0f / powf(10.0f, (float)decimal_precision); - const float remainder = fmodf(new_value, min_step); - if (remainder <= min_step*0.5f) - new_value -= remainder; - else - new_value += (min_step - remainder); - - if (*v != new_value) - { - *v = new_value; - value_changed = true; + SetActiveId(0); } } - else - { - SetActiveId(0); - } - } - if (!is_unbound) - { // Calculate slider grab positioning float grab_t; if (is_logarithmic) From ce87c69c38d3a60bedc6e6ef78bfdac1c4da5b97 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 23:44:32 +0000 Subject: [PATCH 27/44] Passing ImGuiID by copy (32-bits) --- imgui.cpp | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 2689f631..96a9dba2 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -420,7 +420,7 @@ struct ImGuiIniData; struct ImGuiState; struct ImGuiWindow; -static bool ButtonBehaviour(const ImGuiAabb& bb, const ImGuiID& id, bool* out_hovered, bool* out_held, bool allow_key_modifiers, bool repeat = false, bool pressed_on_click = false); +static bool ButtonBehaviour(const ImGuiAabb& bb, ImGuiID id, bool* out_hovered, bool* out_held, bool allow_key_modifiers, bool repeat = false, bool pressed_on_click = false); static void LogText(const ImVec2& ref_pos, const char* text, const char* text_end = NULL); static void RenderText(ImVec2 pos, const char* text, const char* text_end = NULL, bool hide_text_after_hash = true); @@ -1206,7 +1206,7 @@ static void SetActiveId(ImGuiID id) g.ActiveIdIsFocusedOnly = false; } -static void RegisterAliveId(const ImGuiID& id) +static void RegisterAliveId(ImGuiID id) { ImGuiState& g = *GImGui; if (g.ActiveId == id) @@ -4015,7 +4015,7 @@ void ImGui::LabelText(const char* label, const char* fmt, ...) va_end(args); } -static bool IsHovered(const ImGuiAabb& bb, const ImGuiID& id) +static bool IsHovered(const ImGuiAabb& bb, ImGuiID id) { ImGuiState& g = *GImGui; if (g.HoveredId == 0) @@ -4030,7 +4030,7 @@ static bool IsHovered(const ImGuiAabb& bb, const ImGuiID& id) return false; } -static bool ButtonBehaviour(const ImGuiAabb& bb, const ImGuiID& id, bool* out_hovered, bool* out_held, bool allow_key_modifiers, bool repeat, bool pressed_on_click) +static bool ButtonBehaviour(const ImGuiAabb& bb, ImGuiID id, bool* out_hovered, bool* out_held, bool allow_key_modifiers, bool repeat, bool pressed_on_click) { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); @@ -4681,7 +4681,7 @@ static void ApplyNumericalTextInput(const char* buf, float *v) } // Create text input in place of a slider (when CTRL+Clicking on slider) -static bool SliderFloatAsInputText(const char* label, float* v, const ImGuiID& id, int decimal_precision) +static bool SliderFloatAsInputText(const char* label, float* v, ImGuiID id, int decimal_precision) { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); From 1ef789bb689a3473947c944822b8f3fe6a7a6e4e Mon Sep 17 00:00:00 2001 From: ocornut Date: Sat, 14 Mar 2015 23:55:07 +0000 Subject: [PATCH 28/44] Slider code tweaks, split into a SliderBehaviour() function --- imgui.cpp | 150 +++++++++++++++++++++++++++++------------------------- 1 file changed, 81 insertions(+), 69 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 96a9dba2..bbf50fa3 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4734,80 +4734,34 @@ static void ParseFormat(const char* fmt, int& decimal_precision) } } -// Use power!=1.0 for logarithmic sliders. -// Adjust display_format to decorate the value with a prefix or a suffix. -// "%.3f" 1.234 -// "%5.2f secs" 01.23 secs -// "Gold: %.0f" Gold: 1 -bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, const char* display_format, float power) +static bool SliderBehaviour(const ImGuiAabb& frame_bb, const ImGuiAabb& slider_bb, ImGuiID id, float* v, float v_min, float v_max, float power, int decimal_precision) { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); - if (window->SkipItems) - return false; - const ImGuiStyle& style = g.Style; - const ImGuiID id = window->GetID(label); - const float w = ImGui::CalcItemWidth(); - const ImVec2 label_size = CalcTextSize(label, NULL, true); - const ImGuiAabb frame_bb(window->DC.CursorPos, window->DC.CursorPos + ImVec2(w, label_size.y) + style.FramePadding*2.0f); - const ImGuiAabb slider_bb(frame_bb.Min + style.FramePadding, frame_bb.Max - style.FramePadding); - const ImGuiAabb bb(frame_bb.Min, frame_bb.Max + ImVec2(label_size.x > 0.0f ? style.ItemInnerSpacing.x + label_size.x : 0.0f, 0.0f)); + // Draw frame + RenderFrame(frame_bb.Min, frame_bb.Max, window->Color(ImGuiCol_FrameBg), true, style.FrameRounding); - // NB- we don't call ItemSize() yet becausae we may turn into a text edit box later in the function - if (!ItemAdd(slider_bb, &id)) - { - ItemSize(bb); - return false; - } + const bool is_finite = (v_min != -FLT_MAX && v_min != FLT_MAX && v_max != -FLT_MAX && v_max != FLT_MAX); + const bool is_non_linear = abs(power - 1.0f) > 0.0001f; - const bool hovered = IsHovered(slider_bb, id); - if (hovered) - g.HoveredId = id; - - if (!display_format) - display_format = "%.3f"; - int decimal_precision = 3; - ParseFormat(display_format, decimal_precision); - - const bool tab_focus_requested = window->FocusItemRegister(g.ActiveId == id); - const bool is_unbound = (v_min == -FLT_MAX || v_min == FLT_MAX || v_max == -FLT_MAX || v_max == FLT_MAX); - - bool start_text_input = false; - if (tab_focus_requested || (hovered && g.IO.MouseClicked[0])) - { - SetActiveId(id); - FocusWindow(window); - - const bool is_ctrl_down = g.IO.KeyCtrl; - if (tab_focus_requested || is_ctrl_down || is_unbound) - { - start_text_input = true; - g.SliderAsInputTextId = 0; - } - } - - // Tabbing or CTRL-clicking through slider turns into an input box - if (start_text_input || (g.ActiveId == id && id == g.SliderAsInputTextId)) - return SliderFloatAsInputText(label, v, id, decimal_precision); - - ItemSize(bb); - - const float grab_size_in_units = 1.0f; // In 'v' units. Probably needs to be parametrized, based on a 'v_step' value? decimal precision? + const float slider_w = slider_bb.GetWidth(); float grab_size_in_pixels; - if (decimal_precision > 0 || is_unbound) + if (decimal_precision > 0 || !is_finite) grab_size_in_pixels = style.GrabMinSize; else - grab_size_in_pixels = ImMax(grab_size_in_units * (w / (v_max-v_min+1.0f)), style.GrabMinSize); // Integer sliders + grab_size_in_pixels = ImMax(1.0f * (slider_w / (v_max-v_min+1.0f)), style.GrabMinSize); // Integer sliders, if possible have the grab size represent 1 unit const float slider_effective_w = slider_bb.GetWidth() - grab_size_in_pixels; const float slider_effective_x1 = slider_bb.Min.x + grab_size_in_pixels*0.5f; const float slider_effective_x2 = slider_bb.Max.x - grab_size_in_pixels*0.5f; - // For logarithmic sliders that cross over sign boundary we want the exponential increase to be symmetric around 0.0f - float linear_zero_pos = 0.0f; // 0.0->1.0f - if (!is_unbound) + bool value_changed = false; + + if (is_finite) { + // For logarithmic sliders that cross over sign boundary we want the exponential increase to be symmetric around 0.0f + float linear_zero_pos = 0.0f; // 0.0->1.0f if (v_min * v_max < 0.0f) { // Different sign @@ -4820,23 +4774,16 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c // Same sign linear_zero_pos = v_min < 0.0f ? 1.0f : 0.0f; } - } - RenderFrame(frame_bb.Min, frame_bb.Max, window->Color(ImGuiCol_FrameBg), true, style.FrameRounding); - - const bool is_logarithmic = abs(power - 1.0f) > 0.0001f; - bool value_changed = false; - if (!is_unbound) - { // Process clicking on the slider if (g.ActiveId == id) { if (g.IO.MouseDown[0]) { const float normalized_pos = ImClamp((g.IO.MousePos.x - slider_effective_x1) / slider_effective_w, 0.0f, 1.0f); - + float new_value; - if (is_logarithmic) + if (is_non_linear) { // Account for logarithmic scale on both sides of the zero if (normalized_pos < linear_zero_pos) @@ -4888,7 +4835,7 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c // Calculate slider grab positioning float grab_t; - if (is_logarithmic) + if (is_non_linear) { float v_clamped = ImClamp(*v, v_min, v_max); if (v_clamped < 0.0f) @@ -4914,6 +4861,71 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c window->DrawList->AddRectFilled(grab_bb.Min, grab_bb.Max, window->Color(g.ActiveId == id ? ImGuiCol_SliderGrabActive : ImGuiCol_SliderGrab)); } + return value_changed; +} + +// Use power!=1.0 for logarithmic sliders. +// Adjust display_format to decorate the value with a prefix or a suffix. +// "%.3f" 1.234 +// "%5.2f secs" 01.23 secs +// "Gold: %.0f" Gold: 1 +bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, const char* display_format, float power) +{ + ImGuiState& g = *GImGui; + ImGuiWindow* window = GetCurrentWindow(); + if (window->SkipItems) + return false; + + const ImGuiStyle& style = g.Style; + const ImGuiID id = window->GetID(label); + const float w = ImGui::CalcItemWidth(); + + const ImVec2 label_size = CalcTextSize(label, NULL, true); + const ImGuiAabb frame_bb(window->DC.CursorPos, window->DC.CursorPos + ImVec2(w, label_size.y) + style.FramePadding*2.0f); + const ImGuiAabb slider_bb(frame_bb.Min + style.FramePadding, frame_bb.Max - style.FramePadding); + const ImGuiAabb bb(frame_bb.Min, frame_bb.Max + ImVec2(label_size.x > 0.0f ? style.ItemInnerSpacing.x + label_size.x : 0.0f, 0.0f)); + + // NB- we don't call ItemSize() yet becausae we may turn into a text edit box below + if (!ItemAdd(slider_bb, &id)) + { + ItemSize(bb); + return false; + } + + const bool hovered = IsHovered(slider_bb, id); + if (hovered) + g.HoveredId = id; + + if (!display_format) + display_format = "%.3f"; + int decimal_precision = 3; + ParseFormat(display_format, decimal_precision); + + const bool tab_focus_requested = window->FocusItemRegister(g.ActiveId == id); + const bool is_finite = (v_min != -FLT_MAX && v_min != FLT_MAX && v_max != -FLT_MAX && v_max != FLT_MAX); + + // Tabbing or CTRL-clicking through slider turns into an input box + bool start_text_input = false; + if (tab_focus_requested || (hovered && g.IO.MouseClicked[0])) + { + SetActiveId(id); + FocusWindow(window); + + const bool is_ctrl_down = g.IO.KeyCtrl; + if (tab_focus_requested || is_ctrl_down || !is_finite) + { + start_text_input = true; + g.SliderAsInputTextId = 0; + } + } + if (start_text_input || (g.ActiveId == id && id == g.SliderAsInputTextId)) + return SliderFloatAsInputText(label, v, id, decimal_precision); + + ItemSize(bb); + + // Actual slider behavior + render grab + bool value_changed = SliderBehaviour(frame_bb, slider_bb, id, v, v_min, v_max, power, decimal_precision); + // Display value using user-provided display format so user can add prefix/suffix/decorations to the value. char value_buf[64]; char* value_buf_end = value_buf + ImFormatString(value_buf, IM_ARRAYSIZE(value_buf), display_format, *v); From 98eeeab0255f1839962457582439b53823675bb3 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 01:23:56 +0000 Subject: [PATCH 29/44] Added ImGuiStyleVar_GrabMinSize enum value for PushStyleVar() --- imgui.cpp | 1 + imgui.h | 3 ++- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/imgui.cpp b/imgui.cpp index bbf50fa3..5d54e062 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -3382,6 +3382,7 @@ static float* GetStyleVarFloatAddr(ImGuiStyleVar idx) case ImGuiStyleVar_ChildWindowRounding: return &g.Style.ChildWindowRounding; case ImGuiStyleVar_FrameRounding: return &g.Style.FrameRounding; case ImGuiStyleVar_TreeNodeSpacing: return &g.Style.TreeNodeSpacing; + case ImGuiStyleVar_GrabMinSize: return &g.Style.GrabMinSize; } return NULL; } diff --git a/imgui.h b/imgui.h index 0ad58bf1..f3c51b6d 100644 --- a/imgui.h +++ b/imgui.h @@ -502,7 +502,8 @@ enum ImGuiStyleVar_ ImGuiStyleVar_FrameRounding, // float ImGuiStyleVar_ItemSpacing, // ImVec2 ImGuiStyleVar_ItemInnerSpacing, // ImVec2 - ImGuiStyleVar_TreeNodeSpacing // float + ImGuiStyleVar_TreeNodeSpacing, // float + ImGuiStyleVar_GrabMinSize // float }; // Enumeration for ColorEditMode() From 6da83cd5b7420feb80be05430f62673c0e2d2e9c Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 01:27:37 +0000 Subject: [PATCH 30/44] Plot: size provided to PlotHistogram(), PlotLines() include the padding (sorry users). --- imgui.cpp | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 5d54e062..5dbc408a 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -5074,11 +5074,11 @@ static void Plot(ImGuiPlotType plot_type, const char* label, float (*values_gett const ImVec2 label_size = ImGui::CalcTextSize(label, NULL, true); if (graph_size.x == 0.0f) - graph_size.x = ImGui::CalcItemWidth(); + graph_size.x = ImGui::CalcItemWidth() + (style.FramePadding.x * 2); if (graph_size.y == 0.0f) - graph_size.y = label_size.y; + graph_size.y = label_size.y + (style.FramePadding.y * 2); - const ImGuiAabb frame_bb(window->DC.CursorPos, window->DC.CursorPos + ImVec2(graph_size.x, graph_size.y) + style.FramePadding*2.0f); + const ImGuiAabb frame_bb(window->DC.CursorPos, window->DC.CursorPos + ImVec2(graph_size.x, graph_size.y)); const ImGuiAabb graph_bb(frame_bb.Min + style.FramePadding, frame_bb.Max - style.FramePadding); const ImGuiAabb bb(frame_bb.Min, frame_bb.Max + ImVec2(label_size.x > 0.0f ? style.ItemInnerSpacing.x + label_size.x : 0.0f, 0)); ItemSize(bb); @@ -9026,10 +9026,10 @@ void ImGui::ShowTestWindow(bool* opened) phase += 0.10f*values_offset; } } - ImGui::PlotLines("Frame Times", &values.front(), (int)values.size(), (int)values_offset, "avg 0.0", -1.0f, 1.0f, ImVec2(0,70)); + ImGui::PlotLines("Frame Times", &values.front(), (int)values.size(), (int)values_offset, "avg 0.0", -1.0f, 1.0f, ImVec2(0,80)); ImGui::SameLine(); ImGui::Checkbox("pause", &pause); - ImGui::PlotHistogram("Histogram", arr, IM_ARRAYSIZE(arr), 0, NULL, 0.0f, 1.0f, ImVec2(0,70)); + ImGui::PlotHistogram("Histogram", arr, IM_ARRAYSIZE(arr), 0, NULL, 0.0f, 1.0f, ImVec2(0,80)); } if (ImGui::CollapsingHeader("Horizontal Layout")) From ce8150ce6969aec0d85b74bfb52ad3f6ee1f1e80 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 01:36:29 +0000 Subject: [PATCH 31/44] Slider fix hovering bounding test excluding padding between outer frame and grab --- imgui.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 5dbc408a..dbc7b251 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4887,13 +4887,13 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c const ImGuiAabb bb(frame_bb.Min, frame_bb.Max + ImVec2(label_size.x > 0.0f ? style.ItemInnerSpacing.x + label_size.x : 0.0f, 0.0f)); // NB- we don't call ItemSize() yet becausae we may turn into a text edit box below - if (!ItemAdd(slider_bb, &id)) + if (!ItemAdd(frame_bb, &id)) { ItemSize(bb); return false; } - const bool hovered = IsHovered(slider_bb, id); + const bool hovered = IsHovered(frame_bb, id); if (hovered) g.HoveredId = id; From c1547dd79ed4533a73c8bf4d2c3b4dadaefc8030 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 01:47:12 +0000 Subject: [PATCH 32/44] Internal SliderBehaviour() function now supports vertical sliders --- imgui.cpp | 33 +++++++++++++++++++++------------ 1 file changed, 21 insertions(+), 12 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index dbc7b251..bbc67c94 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4735,7 +4735,7 @@ static void ParseFormat(const char* fmt, int& decimal_precision) } } -static bool SliderBehaviour(const ImGuiAabb& frame_bb, const ImGuiAabb& slider_bb, ImGuiID id, float* v, float v_min, float v_max, float power, int decimal_precision) +static bool SliderBehaviour(const ImGuiAabb& frame_bb, const ImGuiAabb& slider_bb, ImGuiID id, float* v, float v_min, float v_max, float power, int decimal_precision, bool horizontal) { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); @@ -4747,15 +4747,15 @@ static bool SliderBehaviour(const ImGuiAabb& frame_bb, const ImGuiAabb& slider_b const bool is_finite = (v_min != -FLT_MAX && v_min != FLT_MAX && v_max != -FLT_MAX && v_max != FLT_MAX); const bool is_non_linear = abs(power - 1.0f) > 0.0001f; - const float slider_w = slider_bb.GetWidth(); - float grab_size_in_pixels; + const float slider_sz = horizontal ? slider_bb.GetWidth() : slider_bb.GetHeight(); + float grab_sz; if (decimal_precision > 0 || !is_finite) - grab_size_in_pixels = style.GrabMinSize; + grab_sz = style.GrabMinSize; else - grab_size_in_pixels = ImMax(1.0f * (slider_w / (v_max-v_min+1.0f)), style.GrabMinSize); // Integer sliders, if possible have the grab size represent 1 unit - const float slider_effective_w = slider_bb.GetWidth() - grab_size_in_pixels; - const float slider_effective_x1 = slider_bb.Min.x + grab_size_in_pixels*0.5f; - const float slider_effective_x2 = slider_bb.Max.x - grab_size_in_pixels*0.5f; + grab_sz = ImMax(1.0f * (slider_sz / (v_max-v_min+1.0f)), style.GrabMinSize); // Integer sliders, if possible have the grab size represent 1 unit + const float slider_usable_sz = slider_sz - grab_sz; + const float slider_usable_pos_min = (horizontal ? slider_bb.Min.x : slider_bb.Min.y) + grab_sz*0.5f; + const float slider_usable_pos_max = (horizontal ? slider_bb.Max.x : slider_bb.Max.y) - grab_sz*0.5f; bool value_changed = false; @@ -4781,7 +4781,10 @@ static bool SliderBehaviour(const ImGuiAabb& frame_bb, const ImGuiAabb& slider_b { if (g.IO.MouseDown[0]) { - const float normalized_pos = ImClamp((g.IO.MousePos.x - slider_effective_x1) / slider_effective_w, 0.0f, 1.0f); + const float mouse_abs_pos = horizontal ? g.IO.MousePos.x : g.IO.MousePos.y; + float normalized_pos = ImClamp((mouse_abs_pos - slider_usable_pos_min) / slider_usable_sz, 0.0f, 1.0f); + if (!horizontal) + normalized_pos = 1.0f - normalized_pos; float new_value; if (is_non_linear) @@ -4857,8 +4860,14 @@ static bool SliderBehaviour(const ImGuiAabb& frame_bb, const ImGuiAabb& slider_b } // Draw - const float grab_x = ImLerp(slider_effective_x1, slider_effective_x2, grab_t); - const ImGuiAabb grab_bb(ImVec2(grab_x-grab_size_in_pixels*0.5f,frame_bb.Min.y+2.0f), ImVec2(grab_x+grab_size_in_pixels*0.5f,frame_bb.Max.y-2.0f)); + if (!horizontal) + grab_t = 1.0f - grab_t; + const float grab_pos = ImLerp(slider_usable_pos_min, slider_usable_pos_max, grab_t); + ImGuiAabb grab_bb; + if (horizontal) + grab_bb = ImGuiAabb(ImVec2(grab_pos - grab_sz*0.5f, frame_bb.Min.y + 2.0f), ImVec2(grab_pos + grab_sz*0.5f, frame_bb.Max.y - 2.0f)); + else + grab_bb = ImGuiAabb(ImVec2(frame_bb.Min.x + 2.0f, grab_pos - grab_sz*0.5f), ImVec2(frame_bb.Max.x - 2.0f, grab_pos + grab_sz*0.5f)); window->DrawList->AddRectFilled(grab_bb.Min, grab_bb.Max, window->Color(g.ActiveId == id ? ImGuiCol_SliderGrabActive : ImGuiCol_SliderGrab)); } @@ -4925,7 +4934,7 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c ItemSize(bb); // Actual slider behavior + render grab - bool value_changed = SliderBehaviour(frame_bb, slider_bb, id, v, v_min, v_max, power, decimal_precision); + bool value_changed = SliderBehaviour(frame_bb, slider_bb, id, v, v_min, v_max, power, decimal_precision, true); // Display value using user-provided display format so user can add prefix/suffix/decorations to the value. char value_buf[64]; From 7019cb10f7983b1e5fb40cf13d5ed4d3a0029493 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 01:48:59 +0000 Subject: [PATCH 33/44] "behaviour" -> "behavior" to behave like the greatest number --- imgui.cpp | 40 ++++++++++++++++++++-------------------- 1 file changed, 20 insertions(+), 20 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index bbc67c94..4a85f59f 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -420,7 +420,7 @@ struct ImGuiIniData; struct ImGuiState; struct ImGuiWindow; -static bool ButtonBehaviour(const ImGuiAabb& bb, ImGuiID id, bool* out_hovered, bool* out_held, bool allow_key_modifiers, bool repeat = false, bool pressed_on_click = false); +static bool ButtonBehavior(const ImGuiAabb& bb, ImGuiID id, bool* out_hovered, bool* out_held, bool allow_key_modifiers, bool repeat = false, bool pressed_on_click = false); static void LogText(const ImVec2& ref_pos, const char* text, const char* text_end = NULL); static void RenderText(ImVec2 pos, const char* text, const char* text_end = NULL, bool hide_text_after_hash = true); @@ -1912,7 +1912,7 @@ void ImGui::NewFrame() ImGui::Begin("Debug", NULL, ImVec2(400,400)); } -// NB: behaviour of ImGui after Shutdown() is not tested/guaranteed at the moment. This function is merely here to free heap allocations. +// NB: behavior of ImGui after Shutdown() is not tested/guaranteed at the moment. This function is merely here to free heap allocations. void ImGui::Shutdown() { ImGuiState& g = *GImGui; @@ -2980,7 +2980,7 @@ bool ImGui::Begin(const char* name, bool* p_opened, const ImVec2& size, float bg const ImGuiAabb resize_aabb(window->Aabb().GetBR()-ImVec2(18,18), window->Aabb().GetBR()); const ImGuiID resize_id = window->GetID("#RESIZE"); bool hovered, held; - ButtonBehaviour(resize_aabb, resize_id, &hovered, &held, true); + ButtonBehavior(resize_aabb, resize_id, &hovered, &held, true); resize_col = window->Color(held ? ImGuiCol_ResizeGripActive : hovered ? ImGuiCol_ResizeGripHovered : ImGuiCol_ResizeGrip); if (g.HoveredWindow == window && held && g.IO.MouseDoubleClicked[0]) @@ -3196,7 +3196,7 @@ static void Scrollbar(ImGuiWindow* window) bool held = false; bool hovered = false; const bool previously_held = (g.ActiveId == id); - ButtonBehaviour(bb, id, &hovered, &held, true); + ButtonBehavior(bb, id, &hovered, &held, true); const float scroll_max = ImMax(1.0f, window->SizeContents.y - window->Size.y); float scroll_ratio = ImSaturate(window->ScrollY / scroll_max); @@ -4031,7 +4031,7 @@ static bool IsHovered(const ImGuiAabb& bb, ImGuiID id) return false; } -static bool ButtonBehaviour(const ImGuiAabb& bb, ImGuiID id, bool* out_hovered, bool* out_held, bool allow_key_modifiers, bool repeat, bool pressed_on_click) +static bool ButtonBehavior(const ImGuiAabb& bb, ImGuiID id, bool* out_hovered, bool* out_held, bool allow_key_modifiers, bool repeat, bool pressed_on_click) { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); @@ -4102,7 +4102,7 @@ bool ImGui::Button(const char* label, const ImVec2& size_arg, bool repeat_when_h return false; bool hovered, held; - bool pressed = ButtonBehaviour(bb, id, &hovered, &held, true, repeat_when_held); + bool pressed = ButtonBehavior(bb, id, &hovered, &held, true, repeat_when_held); // Render const ImU32 col = window->Color((hovered && held) ? ImGuiCol_ButtonActive : hovered ? ImGuiCol_ButtonHovered : ImGuiCol_Button); @@ -4132,7 +4132,7 @@ bool ImGui::SmallButton(const char* label) return false; bool hovered, held; - bool pressed = ButtonBehaviour(bb, id, &hovered, &held, true); + bool pressed = ButtonBehavior(bb, id, &hovered, &held, true); // Render const ImU32 col = window->Color((hovered && held) ? ImGuiCol_ButtonActive : hovered ? ImGuiCol_ButtonHovered : ImGuiCol_Button); @@ -4157,7 +4157,7 @@ bool ImGui::InvisibleButton(const char* str_id, const ImVec2& size) return false; bool hovered, held; - bool pressed = ButtonBehaviour(bb, id, &hovered, &held, true); + bool pressed = ButtonBehavior(bb, id, &hovered, &held, true); return pressed; } @@ -4172,7 +4172,7 @@ static bool CloseWindowButton(bool* p_opened) const ImGuiAabb bb(window->Aabb().GetTR() + ImVec2(-3.0f-size,2.0f), window->Aabb().GetTR() + ImVec2(-3.0f,2.0f+size)); bool hovered, held; - bool pressed = ButtonBehaviour(bb, id, &hovered, &held, true); + bool pressed = ButtonBehavior(bb, id, &hovered, &held, true); // Render const ImU32 col = window->Color((held && hovered) ? ImGuiCol_CloseButtonActive : hovered ? ImGuiCol_CloseButtonHovered : ImGuiCol_CloseButton); @@ -4243,7 +4243,7 @@ bool ImGui::ImageButton(ImTextureID user_texture_id, const ImVec2& size, const I return false; bool hovered, held; - bool pressed = ButtonBehaviour(bb, id, &hovered, &held, true); + bool pressed = ButtonBehavior(bb, id, &hovered, &held, true); // Render const ImU32 col = window->Color((hovered && held) ? ImGuiCol_ButtonActive : hovered ? ImGuiCol_ButtonHovered : ImGuiCol_Button); @@ -4424,7 +4424,7 @@ bool ImGui::CollapsingHeader(const char* label, const char* str_id, bool display const ImGuiAabb text_bb(bb.Min, bb.Min + ImVec2(window->FontSize() + style.FramePadding.x*2*2,0) + label_size); ItemSize(ImVec2(text_bb.GetSize().x, bb.GetSize().y)); // NB: we don't provide our width so that it doesn't get feed back into AutoFit - // When logging is enabled, if automatically expand tree nodes (but *NOT* collapsing headers.. seems like sensible behaviour). + // When logging is enabled, if automatically expand tree nodes (but *NOT* collapsing headers.. seems like sensible behavior). // NB- If we are above max depth we still allow manually opened nodes to be logged. if (g.LogEnabled && !display_frame && window->DC.TreeDepth < g.LogAutoExpandMaxDepth) opened = true; @@ -4433,7 +4433,7 @@ bool ImGui::CollapsingHeader(const char* label, const char* str_id, bool display return opened; bool hovered, held; - bool pressed = ButtonBehaviour(display_frame ? bb : text_bb, id, &hovered, &held, false); + bool pressed = ButtonBehavior(display_frame ? bb : text_bb, id, &hovered, &held, false); if (pressed) { opened = !opened; @@ -4735,7 +4735,7 @@ static void ParseFormat(const char* fmt, int& decimal_precision) } } -static bool SliderBehaviour(const ImGuiAabb& frame_bb, const ImGuiAabb& slider_bb, ImGuiID id, float* v, float v_min, float v_max, float power, int decimal_precision, bool horizontal) +static bool SliderBehavior(const ImGuiAabb& frame_bb, const ImGuiAabb& slider_bb, ImGuiID id, float* v, float v_min, float v_max, float power, int decimal_precision, bool horizontal) { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); @@ -4934,7 +4934,7 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c ItemSize(bb); // Actual slider behavior + render grab - bool value_changed = SliderBehaviour(frame_bb, slider_bb, id, v, v_min, v_max, power, decimal_precision, true); + bool value_changed = SliderBehavior(frame_bb, slider_bb, id, v, v_min, v_max, power, decimal_precision, true); // Display value using user-provided display format so user can add prefix/suffix/decorations to the value. char value_buf[64]; @@ -5233,7 +5233,7 @@ bool ImGui::Checkbox(const char* label, bool* v) return false; bool hovered, held; - bool pressed = ButtonBehaviour(total_bb, id, &hovered, &held, true); + bool pressed = ButtonBehavior(total_bb, id, &hovered, &held, true); if (pressed) *v = !(*v); @@ -5296,7 +5296,7 @@ bool ImGui::RadioButton(const char* label, bool active) const float radius = check_bb.GetHeight() * 0.5f; bool hovered, held; - bool pressed = ButtonBehaviour(total_bb, id, &hovered, &held, true); + bool pressed = ButtonBehavior(total_bb, id, &hovered, &held, true); window->DrawList->AddCircleFilled(center, radius, window->Color((held && hovered) ? ImGuiCol_CheckActive : hovered ? ImGuiCol_CheckHovered : ImGuiCol_FrameBg), 16); if (active) @@ -6215,7 +6215,7 @@ bool ImGui::Selectable(const char* label, bool selected, const ImVec2& size_arg) return false; bool hovered, held; - bool pressed = ButtonBehaviour(bb_with_spacing, id, &hovered, &held, true, false, false); + bool pressed = ButtonBehavior(bb_with_spacing, id, &hovered, &held, true, false, false); // Render if (hovered || selected) @@ -6346,7 +6346,7 @@ bool ImGui::ColorButton(const ImVec4& col, bool small_height, bool outline_borde return false; bool hovered, held; - bool pressed = ButtonBehaviour(bb, id, &hovered, &held, true); + bool pressed = ButtonBehavior(bb, id, &hovered, &held, true); RenderFrame(bb.Min, bb.Max, window->Color(col), outline_border, style.FrameRounding); if (hovered) @@ -6622,7 +6622,7 @@ static bool ItemAdd(const ImGuiAabb& bb, const ImGuiID* id) // This is a sensible default, but widgets are free to override it after calling ItemAdd() const bool hovered = IsMouseHoveringBox(bb); - //const bool hovered = (g.ActiveId == 0 || (id && g.ActiveId == *id) || g.ActiveIdIsFocusedOnly) && IsMouseHoveringBox(bb); // matching the behaviour of IsHovered(), not always what the user wants? + //const bool hovered = (g.ActiveId == 0 || (id && g.ActiveId == *id) || g.ActiveIdIsFocusedOnly) && IsMouseHoveringBox(bb); // matching the behavior of IsHovered(), not always what the user wants? window->DC.LastItemHovered = hovered; return true; } @@ -6787,7 +6787,7 @@ void ImGui::Columns(int columns_count, const char* id, bool border) continue; bool hovered, held; - ButtonBehaviour(column_aabb, column_id, &hovered, &held, true); + ButtonBehavior(column_aabb, column_id, &hovered, &held, true); // Draw before resize so our items positioning are in sync with the line being drawn const ImU32 col = window->Color(held ? ImGuiCol_ColumnActive : hovered ? ImGuiCol_ColumnHovered : ImGuiCol_Column); From 1ac4f33736d5122f7dcb1a20433d22bd21378d77 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 01:58:14 +0000 Subject: [PATCH 34/44] Added VSliderFloat() --- imgui.cpp | 81 +++++++++++++++++++++++++++++++++++++++++++++++++++++++ imgui.h | 1 + 2 files changed, 82 insertions(+) diff --git a/imgui.cpp b/imgui.cpp index 4a85f59f..53b24a1a 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -4947,6 +4947,56 @@ bool ImGui::SliderFloat(const char* label, float* v, float v_min, float v_max, c return value_changed; } +bool ImGui::VSliderFloat(const char* label, const ImVec2& size, float* v, float v_min, float v_max, const char* display_format, float power) +{ + ImGuiState& g = *GImGui; + ImGuiWindow* window = GetCurrentWindow(); + if (window->SkipItems) + return false; + + const ImGuiStyle& style = g.Style; + const ImGuiID id = window->GetID(label); + + const ImVec2 label_size = CalcTextSize(label, NULL, true); + const ImGuiAabb frame_bb(window->DC.CursorPos, window->DC.CursorPos + size); + const ImGuiAabb slider_bb(frame_bb.Min + style.FramePadding, frame_bb.Max - style.FramePadding); + const ImGuiAabb bb(frame_bb.Min, frame_bb.Max + ImVec2(label_size.x > 0.0f ? style.ItemInnerSpacing.x + label_size.x : 0.0f, 0.0f)); + + ItemSize(bb); + if (!ItemAdd(frame_bb, &id)) + return false; + + const bool hovered = IsHovered(frame_bb, id); + if (hovered) + g.HoveredId = id; + + if (!display_format) + display_format = "%.3f"; + int decimal_precision = 3; + ParseFormat(display_format, decimal_precision); + + if (hovered && g.IO.MouseClicked[0]) + { + SetActiveId(id); + FocusWindow(window); + } + + // Actual slider behavior + render grab + bool value_changed = SliderBehavior(frame_bb, slider_bb, id, v, v_min, v_max, power, decimal_precision, false); + + // Display value using user-provided display format so user can add prefix/suffix/decorations to the value. + // For the vertical slider we allow centered text to overlap the frame padding + char value_buf[64]; + char* value_buf_end = value_buf + ImFormatString(value_buf, IM_ARRAYSIZE(value_buf), display_format, *v); + const ImVec2 value_text_size = CalcTextSize(value_buf, value_buf_end, true); + RenderTextClipped(ImVec2(ImMax(frame_bb.Min.x, slider_bb.GetCenter().x - value_text_size.x*0.5f), frame_bb.Min.y + style.FramePadding.y), value_buf, value_buf_end, &value_text_size, frame_bb.Max); + + if (label_size.x > 0.0f) + RenderText(ImVec2(frame_bb.Max.x + style.ItemInnerSpacing.x, slider_bb.Min.y), label); + + return value_changed; +} + bool ImGui::SliderAngle(const char* label, float* v, float v_degrees_min, float v_degrees_max) { float v_deg = *v * 360.0f / (2*PI); @@ -9012,6 +9062,37 @@ void ImGui::ShowTestWindow(bool* opened) //ImGui::PushItemWidth(-1); //ImGui::ListBox("##listbox2", &listbox_item_current2, listbox_items, IM_ARRAYSIZE(listbox_items), 4); //ImGui::PopItemWidth(); + + if (ImGui::TreeNode("Vertical Sliders")) + { + const int spacing = 4; + + ImGui::PushID("set1"); + static float values[4] = { 0.0f, 0.60f, 0.35f, 0.9f }; + for (size_t i = 0; i < IM_ARRAYSIZE(values); i++) + { + if (i > 0) ImGui::SameLine(0, spacing); + ImGui::PushID(i); + ImGui::VSliderFloat("##v", ImVec2(18,160), &values[i], 0.0f, 1.0f, ""); + ImGui::PopID(); + } + ImGui::PopID(); + + ImGui::SameLine(); + ImGui::PushID("set2"); + ImGui::PushStyleVar(ImGuiStyleVar_GrabMinSize, 40); + for (size_t i = 0; i < IM_ARRAYSIZE(values); i++) + { + if (i > 0) ImGui::SameLine(0, spacing); + ImGui::PushID(i); + ImGui::VSliderFloat("##v", ImVec2(40,160), &values[i], 0.0f, 1.0f, "%.2f"); + ImGui::PopID(); + } + ImGui::PopStyleVar(); + ImGui::PopID(); + + ImGui::TreePop(); + } } if (ImGui::CollapsingHeader("Graphs widgets")) diff --git a/imgui.h b/imgui.h index f3c51b6d..54b5a90d 100644 --- a/imgui.h +++ b/imgui.h @@ -283,6 +283,7 @@ namespace ImGui IMGUI_API bool SliderInt2(const char* label, int v[2], int v_min, int v_max, const char* display_format = "%.0f"); IMGUI_API bool SliderInt3(const char* label, int v[3], int v_min, int v_max, const char* display_format = "%.0f"); IMGUI_API bool SliderInt4(const char* label, int v[4], int v_min, int v_max, const char* display_format = "%.0f"); + IMGUI_API bool VSliderFloat(const char* label, const ImVec2& size, float* v, float v_min, float v_max, const char* display_format = "%.3f", float power = 1.0f); IMGUI_API void PlotLines(const char* label, const float* values, int values_count, int values_offset = 0, const char* overlay_text = NULL, float scale_min = FLT_MAX, float scale_max = FLT_MAX, ImVec2 graph_size = ImVec2(0,0), size_t stride = sizeof(float)); IMGUI_API void PlotLines(const char* label, float (*values_getter)(void* data, int idx), void* data, int values_count, int values_offset = 0, const char* overlay_text = NULL, float scale_min = FLT_MAX, float scale_max = FLT_MAX, ImVec2 graph_size = ImVec2(0,0)); IMGUI_API void PlotHistogram(const char* label, const float* values, int values_count, int values_offset = 0, const char* overlay_text = NULL, float scale_min = FLT_MAX, float scale_max = FLT_MAX, ImVec2 graph_size = ImVec2(0,0), size_t stride = sizeof(float)); From b443c7ebe981d5dd281009df49cb69d86e7a28d3 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 10:38:19 +0000 Subject: [PATCH 35/44] Tweak VSliderFloat() Not really happy about them --- imgui.cpp | 19 ++++++++++++++----- 1 file changed, 14 insertions(+), 5 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 53b24a1a..d988cc1f 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -269,6 +269,8 @@ - input number: holding [-]/[+] buttons could increase the step speed non-linearly (or user-controlled) - input number: use mouse wheel to step up/down - input number: non-decimal input. + - text: vertical alignment (e.g. for Text after a large widget) + - text: proper horizontal centering - layout: horizontal layout helper (github issue #97) - layout: more generic alignment state (left/right/centered) for single items? - layout: clean up the InputFloatN/SliderFloatN/ColorEdit4 layout code. item width should include frame padding. @@ -9067,28 +9069,35 @@ void ImGui::ShowTestWindow(bool* opened) { const int spacing = 4; + static float values[7] = { 0.0f, 0.60f, 0.35f, 0.9f, 0.70f, 0.20f, 0.0f }; ImGui::PushID("set1"); - static float values[4] = { 0.0f, 0.60f, 0.35f, 0.9f }; - for (size_t i = 0; i < IM_ARRAYSIZE(values); i++) + for (int i = 0; i < 7; i++) { if (i > 0) ImGui::SameLine(0, spacing); ImGui::PushID(i); + ImGui::PushStyleColor(ImGuiCol_FrameBg, ImColor::HSV(i/7.0f, 0.5f, 0.5f)); + ImGui::PushStyleColor(ImGuiCol_SliderGrab, ImColor::HSV(i/7.0f, 0.9f, 0.9f)); + ImGui::PushStyleColor(ImGuiCol_SliderGrabActive, ImColor::HSV(i/7.0f, 1.0f, 1.0f)); ImGui::VSliderFloat("##v", ImVec2(18,160), &values[i], 0.0f, 1.0f, ""); + if (ImGui::IsItemActive() || ImGui::IsItemHovered()) + ImGui::SetTooltip("%.3f", values[i]); + ImGui::PopStyleColor(3); ImGui::PopID(); } ImGui::PopID(); ImGui::SameLine(); + ImGui::PushID("set2"); - ImGui::PushStyleVar(ImGuiStyleVar_GrabMinSize, 40); - for (size_t i = 0; i < IM_ARRAYSIZE(values); i++) + for (size_t i = 0; i < 4; i++) { if (i > 0) ImGui::SameLine(0, spacing); ImGui::PushID(i); + ImGui::PushStyleVar(ImGuiStyleVar_GrabMinSize, 40); ImGui::VSliderFloat("##v", ImVec2(40,160), &values[i], 0.0f, 1.0f, "%.2f"); + ImGui::PopStyleVar(); ImGui::PopID(); } - ImGui::PopStyleVar(); ImGui::PopID(); ImGui::TreePop(); From 304d19a0d88c4408a8cb486c0b9e888706dc79c2 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 11:03:55 +0000 Subject: [PATCH 36/44] ShowTestWindow() added multi-components sliders in a sub-node + right align all widgets --- imgui.cpp | 39 +++++++++++++++++++++------------------ 1 file changed, 21 insertions(+), 18 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index d988cc1f..974877fa 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -8706,9 +8706,9 @@ void ImGui::ShowTestWindow(bool* opened) ImGui::End(); return; } - - ImGui::PushItemWidth(ImGui::GetWindowWidth() * 0.65f); // 2/3 of the space for widget and 1/3 for labels - //ImGui::PushItemWidth(-140); // Right align, keep 140 pixels for labels + + //ImGui::PushItemWidth(ImGui::GetWindowWidth() * 0.65f); // 2/3 of the space for widget and 1/3 for labels + ImGui::PushItemWidth(-140); // Right align, keep 140 pixels for labels ImGui::Text("ImGui says hello."); //ImGui::Text("MousePos (%g, %g)", ImGui::GetIO().MousePos.x, ImGui::GetIO().MousePos.y); @@ -9041,16 +9041,6 @@ void ImGui::ShowTestWindow(bool* opened) static float angle = 0.0f; ImGui::SliderAngle("angle", &angle); - static float vec4b[4] = { 0.10f, 0.20f, 0.30f, 0.40f }; - //ImGui::SliderFloat2("slider float2", vec4b, 0.0f, 1.0f); - ImGui::SliderFloat3("slider float3", vec4b, 0.0f, 1.0f); - //ImGui::SliderFloat4("slider float4", vec4b, 0.0f, 1.0f); - - //static int vec4i[4] = { 1, 5, 100, 255 }; - //ImGui::SliderInt2("slider int2", vec4i, 0, 255); - //ImGui::SliderInt3("slider int3", vec4i, 0, 255); - //ImGui::SliderInt4("slider int4", vec4i, 0, 255); - static float col1[3] = { 1.0f,0.0f,0.2f }; static float col2[4] = { 0.4f,0.7f,0.0f,0.5f }; ImGui::ColorEdit3("color 1", col1); @@ -9065,15 +9055,30 @@ void ImGui::ShowTestWindow(bool* opened) //ImGui::ListBox("##listbox2", &listbox_item_current2, listbox_items, IM_ARRAYSIZE(listbox_items), 4); //ImGui::PopItemWidth(); + if (ImGui::TreeNode("Multi-component Sliders")) + { + ImGui::TreePop(); // Intentionally undo the tree-node tab early on + + static float vec4b[4] = { 0.10f, 0.20f, 0.30f, 0.40f }; + ImGui::SliderFloat2("slider float2", vec4b, 0.0f, 1.0f); + ImGui::SliderFloat3("slider float3", vec4b, 0.0f, 1.0f); + ImGui::SliderFloat4("slider float4", vec4b, 0.0f, 1.0f); + + static int vec4i[4] = { 1, 5, 100, 255 }; + ImGui::SliderInt2("slider int2", vec4i, 0, 255); + ImGui::SliderInt3("slider int3", vec4i, 0, 255); + ImGui::SliderInt4("slider int4", vec4i, 0, 255); + } + if (ImGui::TreeNode("Vertical Sliders")) { - const int spacing = 4; + ImGui::TreePop(); // Intentionally undo the tree-node tab early on static float values[7] = { 0.0f, 0.60f, 0.35f, 0.9f, 0.70f, 0.20f, 0.0f }; ImGui::PushID("set1"); for (int i = 0; i < 7; i++) { - if (i > 0) ImGui::SameLine(0, spacing); + if (i > 0) ImGui::SameLine(0, 4); ImGui::PushID(i); ImGui::PushStyleColor(ImGuiCol_FrameBg, ImColor::HSV(i/7.0f, 0.5f, 0.5f)); ImGui::PushStyleColor(ImGuiCol_SliderGrab, ImColor::HSV(i/7.0f, 0.9f, 0.9f)); @@ -9091,7 +9096,7 @@ void ImGui::ShowTestWindow(bool* opened) ImGui::PushID("set2"); for (size_t i = 0; i < 4; i++) { - if (i > 0) ImGui::SameLine(0, spacing); + if (i > 0) ImGui::SameLine(0, 4); ImGui::PushID(i); ImGui::PushStyleVar(ImGuiStyleVar_GrabMinSize, 40); ImGui::VSliderFloat("##v", ImVec2(40,160), &values[i], 0.0f, 1.0f, "%.2f"); @@ -9099,8 +9104,6 @@ void ImGui::ShowTestWindow(bool* opened) ImGui::PopID(); } ImGui::PopID(); - - ImGui::TreePop(); } } From edde703f8f241f16743dbf83a9146184c257a4f3 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 11:10:48 +0000 Subject: [PATCH 37/44] Renamed style.TreeNodeSpacing to style.IndentSpacing, ImGuiStyleVar_TreeNodeSpacing to ImGuiStyleVar_IndentSpacing --- imgui.cpp | 15 ++++++++------- imgui.h | 4 ++-- 2 files changed, 10 insertions(+), 9 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 974877fa..c46c4b0f 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -129,6 +129,7 @@ Occasionally introducing changes that are breaking the API. The breakage are generally minor and easy to fix. Here is a change-log of API breaking changes, if you are using one of the functions listed, expect to have to fix some code. + - 2015/03/15 (1.36) - renamed style.TreeNodeSpacing to style.IndentSpacing, ImGuiStyleVar_TreeNodeSpacing to ImGuiStyleVar_IndentSpacing - 2015/03/13 (1.36) - renamed GetWindowIsFocused() to IsWindowFocused(). Kept inline redirection function. - 2015/03/08 (1.35) - renamed style.ScrollBarWidth to style.ScrollbarWidth - 2015/02/27 (1.34) - renamed OpenNextNode(bool) to SetNextTreeNodeOpened(bool, ImGuiSetCond). Kept inline redirection function. @@ -506,7 +507,7 @@ ImGuiStyle::ImGuiStyle() TouchExtraPadding = ImVec2(0,0); // Expand bounding box for touch-based system where touch position is not accurate enough (unnecessary for mouse inputs). Unfortunately we don't sort widgets so priority on overlap will always be given to the first widget running. So dont grow this too much! AutoFitPadding = ImVec2(8,8); // Extra space after auto-fit (double-clicking on resize grip) WindowFillAlphaDefault = 0.70f; // Default alpha of window background, if not specified in ImGui::Begin() - TreeNodeSpacing = 22.0f; // Horizontal spacing when entering a tree node + IndentSpacing = 22.0f; // Horizontal spacing when e.g. entering a tree node ColumnsMinSpacing = 6.0f; // Minimum horizontal spacing between two columns ScrollbarWidth = 16.0f; // Width of the vertical scrollbar GrabMinSize = 10.0f; // Minimum width/height of a slider or scrollbar grab @@ -3383,7 +3384,7 @@ static float* GetStyleVarFloatAddr(ImGuiStyleVar idx) case ImGuiStyleVar_WindowRounding: return &g.Style.WindowRounding; case ImGuiStyleVar_ChildWindowRounding: return &g.Style.ChildWindowRounding; case ImGuiStyleVar_FrameRounding: return &g.Style.FrameRounding; - case ImGuiStyleVar_TreeNodeSpacing: return &g.Style.TreeNodeSpacing; + case ImGuiStyleVar_IndentSpacing: return &g.Style.IndentSpacing; case ImGuiStyleVar_GrabMinSize: return &g.Style.GrabMinSize; } return NULL; @@ -4542,7 +4543,7 @@ bool ImGui::TreeNodeV(const char* str_id, const char* fmt, va_list args) str_id = fmt; ImGui::PushID(str_id); - const bool opened = ImGui::CollapsingHeader(buf, "", false); // do not add to the ID so that TreeNodeSetOpen can access + const bool opened = ImGui::CollapsingHeader(buf, "", false); ImGui::PopID(); if (opened) @@ -6894,7 +6895,7 @@ void ImGui::TreePush(const char* str_id) { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); - window->DC.ColumnsStartX += g.Style.TreeNodeSpacing; + window->DC.ColumnsStartX += g.Style.IndentSpacing; window->DC.CursorPos.x = window->Pos.x + window->DC.ColumnsStartX + window->DC.ColumnsOffsetX; window->DC.TreeDepth++; PushID(str_id ? str_id : "#TreePush"); @@ -6904,7 +6905,7 @@ void ImGui::TreePush(const void* ptr_id) { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); - window->DC.ColumnsStartX += g.Style.TreeNodeSpacing; + window->DC.ColumnsStartX += g.Style.IndentSpacing; window->DC.CursorPos.x = window->Pos.x + window->DC.ColumnsStartX + window->DC.ColumnsOffsetX; window->DC.TreeDepth++; PushID(ptr_id ? ptr_id : (const void*)"#TreePush"); @@ -6914,7 +6915,7 @@ void ImGui::TreePop() { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); - window->DC.ColumnsStartX -= g.Style.TreeNodeSpacing; + window->DC.ColumnsStartX -= g.Style.IndentSpacing; window->DC.CursorPos.x = window->Pos.x + window->DC.ColumnsStartX + window->DC.ColumnsOffsetX; window->DC.TreeDepth--; PopID(); @@ -8585,7 +8586,7 @@ void ImGui::ShowStyleEditor(ImGuiStyle* ref) ImGui::SliderFloat2("ItemSpacing", (float*)&style.ItemSpacing, 0.0f, 20.0f, "%.0f"); ImGui::SliderFloat2("ItemInnerSpacing", (float*)&style.ItemInnerSpacing, 0.0f, 20.0f, "%.0f"); ImGui::SliderFloat2("TouchExtraPadding", (float*)&style.TouchExtraPadding, 0.0f, 10.0f, "%.0f"); - ImGui::SliderFloat("TreeNodeSpacing", &style.TreeNodeSpacing, 0.0f, 20.0f, "%.0f"); + ImGui::SliderFloat("IndentSpacing", &style.IndentSpacing, 0.0f, 20.0f, "%.0f"); ImGui::SliderFloat("ScrollBarWidth", &style.ScrollbarWidth, 1.0f, 20.0f, "%.0f"); ImGui::SliderFloat("GrabMinSize", &style.GrabMinSize, 1.0f, 20.0f, "%.0f"); ImGui::TreePop(); diff --git a/imgui.h b/imgui.h index 54b5a90d..37c8c1e7 100644 --- a/imgui.h +++ b/imgui.h @@ -503,7 +503,7 @@ enum ImGuiStyleVar_ ImGuiStyleVar_FrameRounding, // float ImGuiStyleVar_ItemSpacing, // ImVec2 ImGuiStyleVar_ItemInnerSpacing, // ImVec2 - ImGuiStyleVar_TreeNodeSpacing, // float + ImGuiStyleVar_IndentSpacing, // float ImGuiStyleVar_GrabMinSize // float }; @@ -540,7 +540,7 @@ struct ImGuiStyle ImVec2 TouchExtraPadding; // Expand bounding box for touch-based system where touch position is not accurate enough (unnecessary for mouse inputs). Unfortunately we don't sort widgets so priority on overlap will always be given to the first widget running. So dont grow this too much! ImVec2 AutoFitPadding; // Extra space after auto-fit (double-clicking on resize grip) float WindowFillAlphaDefault; // Default alpha of window background, if not specified in ImGui::Begin() - float TreeNodeSpacing; // Horizontal spacing when entering a tree node + float IndentSpacing; // Horizontal indentation when e.g. entering a tree node float ColumnsMinSpacing; // Minimum horizontal spacing between two columns float ScrollbarWidth; // Width of the vertical scrollbar float GrabMinSize; // Minimum width/height of a slider or scrollbar grab From 44826972c05277fed62a09ed432a5289f0822fbf Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 11:19:10 +0000 Subject: [PATCH 38/44] Added Indent(), Unindent() --- imgui.cpp | 25 ++++++++++++++++++------- imgui.h | 4 +++- 2 files changed, 21 insertions(+), 8 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index c46c4b0f..0a4b2698 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -6891,32 +6891,43 @@ void ImGui::Columns(int columns_count, const char* id, bool border) } } -void ImGui::TreePush(const char* str_id) + +inline void ImGui::Indent() { ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); window->DC.ColumnsStartX += g.Style.IndentSpacing; window->DC.CursorPos.x = window->Pos.x + window->DC.ColumnsStartX + window->DC.ColumnsOffsetX; +} + +inline void ImGui::Unindent() +{ + ImGuiState& g = *GImGui; + ImGuiWindow* window = GetCurrentWindow(); + window->DC.ColumnsStartX -= g.Style.IndentSpacing; + window->DC.CursorPos.x = window->Pos.x + window->DC.ColumnsStartX + window->DC.ColumnsOffsetX; +} + +void ImGui::TreePush(const char* str_id) +{ + ImGuiWindow* window = GetCurrentWindow(); + ImGui::Indent(); window->DC.TreeDepth++; PushID(str_id ? str_id : "#TreePush"); } void ImGui::TreePush(const void* ptr_id) { - ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); - window->DC.ColumnsStartX += g.Style.IndentSpacing; - window->DC.CursorPos.x = window->Pos.x + window->DC.ColumnsStartX + window->DC.ColumnsOffsetX; + ImGui::Indent(); window->DC.TreeDepth++; PushID(ptr_id ? ptr_id : (const void*)"#TreePush"); } void ImGui::TreePop() { - ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); - window->DC.ColumnsStartX -= g.Style.IndentSpacing; - window->DC.CursorPos.x = window->Pos.x + window->DC.ColumnsStartX + window->DC.ColumnsOffsetX; + ImGui::Unindent(); window->DC.TreeDepth--; PopID(); } diff --git a/imgui.h b/imgui.h index 37c8c1e7..7b520d33 100644 --- a/imgui.h +++ b/imgui.h @@ -225,7 +225,9 @@ namespace ImGui // Layout IMGUI_API void Separator(); // horizontal line IMGUI_API void SameLine(int column_x = 0, int spacing_w = -1); // call between widgets to layout them horizontally - IMGUI_API void Spacing(); + IMGUI_API void Spacing(); // add vertical spacing + IMGUI_API void Indent(); // move content position toward the right by style.IndentSpacing pixels + IMGUI_API void Unindent(); // move content position back to the left (cancel Indent) IMGUI_API void Columns(int count = 1, const char* id = NULL, bool border=true); // setup number of columns IMGUI_API void NextColumn(); // next column IMGUI_API int GetColumnIndex(); // get current column index From 9c6c8a1ab9e9eff4c14b517ffb8128d8ce8f8548 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 11:21:59 +0000 Subject: [PATCH 39/44] ShowTestWindow() using Unindent() / Indent() instead of an explicit TreePop --- imgui.cpp | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 0a4b2698..e0203b45 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -9069,7 +9069,7 @@ void ImGui::ShowTestWindow(bool* opened) if (ImGui::TreeNode("Multi-component Sliders")) { - ImGui::TreePop(); // Intentionally undo the tree-node tab early on + ImGui::Unindent(); static float vec4b[4] = { 0.10f, 0.20f, 0.30f, 0.40f }; ImGui::SliderFloat2("slider float2", vec4b, 0.0f, 1.0f); @@ -9080,11 +9080,14 @@ void ImGui::ShowTestWindow(bool* opened) ImGui::SliderInt2("slider int2", vec4i, 0, 255); ImGui::SliderInt3("slider int3", vec4i, 0, 255); ImGui::SliderInt4("slider int4", vec4i, 0, 255); + + ImGui::Indent(); + ImGui::TreePop(); } if (ImGui::TreeNode("Vertical Sliders")) { - ImGui::TreePop(); // Intentionally undo the tree-node tab early on + ImGui::Unindent(); static float values[7] = { 0.0f, 0.60f, 0.35f, 0.9f, 0.70f, 0.20f, 0.0f }; ImGui::PushID("set1"); @@ -9116,6 +9119,9 @@ void ImGui::ShowTestWindow(bool* opened) ImGui::PopID(); } ImGui::PopID(); + + ImGui::Indent(); + ImGui::TreePop(); } } From 0fc9168439a3e21f8a0516d36f3c0a2bb004e995 Mon Sep 17 00:00:00 2001 From: ocornut Date: Sun, 15 Mar 2015 11:30:00 +0000 Subject: [PATCH 40/44] Added InputInt2(), InputInt3(), InputInt4() --- imgui.cpp | 70 ++++++++++++++++++++++++++++++++++++++++++++++++++----- imgui.h | 3 +++ 2 files changed, 67 insertions(+), 6 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index e0203b45..e8033cb4 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -6084,6 +6084,56 @@ bool ImGui::InputFloat4(const char* label, float v[4], int decimal_precision) return InputFloatN(label, v, 4, decimal_precision); } +static bool InputIntN(const char* label, int* v, int components) +{ + ImGuiState& g = *GImGui; + ImGuiWindow* window = GetCurrentWindow(); + if (window->SkipItems) + return false; + + const ImGuiStyle& style = g.Style; + const float w_full = ImGui::CalcItemWidth(); + const float w_item_one = ImMax(1.0f, (float)(int)((w_full - (style.FramePadding.x*2.0f + style.ItemInnerSpacing.x) * (components-1)) / (float)components)); + const float w_item_last = ImMax(1.0f, (float)(int)(w_full - (w_item_one + style.FramePadding.x*2.0f + style.ItemInnerSpacing.x) * (components-1))); + + bool value_changed = false; + ImGui::PushID(label); + ImGui::PushItemWidth(w_item_one); + for (int i = 0; i < components; i++) + { + ImGui::PushID(i); + if (i + 1 == components) + { + ImGui::PopItemWidth(); + ImGui::PushItemWidth(w_item_last); + } + value_changed |= ImGui::InputInt("##v", &v[i], 0, 0); + ImGui::SameLine(0, (int)style.ItemInnerSpacing.x); + ImGui::PopID(); + } + ImGui::PopItemWidth(); + ImGui::PopID(); + + ImGui::TextUnformatted(label, FindTextDisplayEnd(label)); + + return value_changed; +} + +bool ImGui::InputInt2(const char* label, int v[2]) +{ + return InputIntN(label, v, 2); +} + +bool ImGui::InputInt3(const char* label, int v[3]) +{ + return InputIntN(label, v, 3); +} + +bool ImGui::InputInt4(const char* label, int v[4]) +{ + return InputIntN(label, v, 4); +} + static bool Items_ArrayGetter(void* data, int idx, const char** out_text) { const char** items = (const char**)data; @@ -9067,18 +9117,26 @@ void ImGui::ShowTestWindow(bool* opened) //ImGui::ListBox("##listbox2", &listbox_item_current2, listbox_items, IM_ARRAYSIZE(listbox_items), 4); //ImGui::PopItemWidth(); - if (ImGui::TreeNode("Multi-component Sliders")) + if (ImGui::TreeNode("Multi-component Widgets")) { ImGui::Unindent(); - static float vec4b[4] = { 0.10f, 0.20f, 0.30f, 0.40f }; - ImGui::SliderFloat2("slider float2", vec4b, 0.0f, 1.0f); - ImGui::SliderFloat3("slider float3", vec4b, 0.0f, 1.0f); - ImGui::SliderFloat4("slider float4", vec4b, 0.0f, 1.0f); - + static float vec4f[4] = { 0.10f, 0.20f, 0.30f, 0.44f }; static int vec4i[4] = { 1, 5, 100, 255 }; + + ImGui::InputFloat2("input float2", vec4f); + ImGui::SliderFloat2("slider float2", vec4f, 0.0f, 1.0f); + ImGui::InputInt2("input int2", vec4i); ImGui::SliderInt2("slider int2", vec4i, 0, 255); + + ImGui::InputFloat3("input float3", vec4f); + ImGui::SliderFloat3("slider float3", vec4f, 0.0f, 1.0f); + ImGui::InputInt3("input int3", vec4i); ImGui::SliderInt3("slider int3", vec4i, 0, 255); + + ImGui::InputFloat4("input float4", vec4f); + ImGui::SliderFloat4("slider float4", vec4f, 0.0f, 1.0f); + ImGui::InputInt4("input int4", vec4i); ImGui::SliderInt4("slider int4", vec4i, 0, 255); ImGui::Indent(); diff --git a/imgui.h b/imgui.h index 7b520d33..53e66b70 100644 --- a/imgui.h +++ b/imgui.h @@ -300,6 +300,9 @@ namespace ImGui IMGUI_API bool InputFloat3(const char* label, float v[3], int decimal_precision = -1); IMGUI_API bool InputFloat4(const char* label, float v[4], int decimal_precision = -1); IMGUI_API bool InputInt(const char* label, int* v, int step = 1, int step_fast = 100, ImGuiInputTextFlags extra_flags = 0); + IMGUI_API bool InputInt2(const char* label, int v[2]); + IMGUI_API bool InputInt3(const char* label, int v[3]); + IMGUI_API bool InputInt4(const char* label, int v[4]); IMGUI_API bool Combo(const char* label, int* current_item, const char** items, int items_count, int height_in_items = -1); IMGUI_API bool Combo(const char* label, int* current_item, const char* items_separated_by_zeros, int height_in_items = -1); // separate items with \0, end item-list with \0\0 IMGUI_API bool Combo(const char* label, int* current_item, bool (*items_getter)(void* data, int idx, const char** out_text), void* data, int items_count, int height_in_items = -1); From d08ce58876740e19c121739363f7ac9df9a577ec Mon Sep 17 00:00:00 2001 From: ocornut Date: Mon, 16 Mar 2015 00:46:03 +0000 Subject: [PATCH 41/44] TreeNodeV() early out if SkipItems is set (avoid formatting string only to be discarded later) --- imgui.cpp | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/imgui.cpp b/imgui.cpp index e8033cb4..02635a45 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -3992,6 +3992,7 @@ void ImGui::LabelTextV(const char* label, const char* fmt, va_list args) ImGuiWindow* window = GetCurrentWindow(); if (window->SkipItems) return; + const ImGuiStyle& style = g.Style; const float w = ImGui::CalcItemWidth(); @@ -4536,6 +4537,10 @@ void ImGui::BulletText(const char* fmt, ...) // If returning 'true' the node is open and the user is responsible for calling TreePop bool ImGui::TreeNodeV(const char* str_id, const char* fmt, va_list args) { + ImGuiWindow* window = GetCurrentWindow(); + if (window->SkipItems) + return false; + static char buf[1024]; ImFormatStringV(buf, IM_ARRAYSIZE(buf), fmt, args); @@ -4564,6 +4569,10 @@ bool ImGui::TreeNode(const char* str_id, const char* fmt, ...) // If returning 'true' the node is open and the user is responsible for calling TreePop bool ImGui::TreeNodeV(const void* ptr_id, const char* fmt, va_list args) { + ImGuiWindow* window = GetCurrentWindow(); + if (window->SkipItems) + return false; + static char buf[1024]; ImFormatStringV(buf, IM_ARRAYSIZE(buf), fmt, args); From 733eedc4773f126f157c3a264790eca305a38dde Mon Sep 17 00:00:00 2001 From: ocornut Date: Mon, 16 Mar 2015 09:45:32 +0000 Subject: [PATCH 42/44] Moved static buffer into state --- imgui.cpp | 43 ++++++++++++++++++++++++++++++++++--------- 1 file changed, 34 insertions(+), 9 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 02635a45..31d9d7cd 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -1068,6 +1068,8 @@ struct ImGuiState float FramerateSecPerFrame[120]; // calculate estimate of framerate for user int FramerateSecPerFrameIdx; float FramerateSecPerFrameAccum; + bool TempBufferLocked; + char TempBuffer[1024*3+1]; // temporary text buffer ImGuiState() { @@ -1216,6 +1218,26 @@ static void RegisterAliveId(ImGuiID id) g.ActiveIdIsAlive = true; } +static char* TempBufferLock() +{ + ImGuiState& g = *GImGui; + IM_ASSERT(!g.TempBufferLocked); + g.TempBufferLocked = true; + return g.TempBuffer; +} + +static void TempBufferUnlock() +{ + ImGuiState& g = *GImGui; + IM_ASSERT(g.TempBufferLocked); + g.TempBufferLocked = false; +} + +static size_t TempBufferSize() +{ + return sizeof(GImGui->TempBuffer); +} + //----------------------------------------------------------------------------- // Helper: Key->value storage @@ -5805,7 +5827,6 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT bool value_changed = false; bool cancel_edit = false; bool enter_pressed = false; - static char text_tmp_utf8[IM_ARRAYSIZE(edit_state.InitialText)]; if (g.ActiveId == id) { // Edit in progress @@ -5878,8 +5899,10 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT { const int ib = edit_state.HasSelection() ? ImMin(edit_state.StbState.select_start, edit_state.StbState.select_end) : 0; const int ie = edit_state.HasSelection() ? ImMax(edit_state.StbState.select_start, edit_state.StbState.select_end) : (int)edit_state.CurLenW; - ImTextStrToUtf8(text_tmp_utf8, IM_ARRAYSIZE(text_tmp_utf8), edit_state.Text+ib, edit_state.Text+ie); - g.IO.SetClipboardTextFn(text_tmp_utf8); + char* tmp_buf = TempBufferLock(); + ImTextStrToUtf8(tmp_buf, TempBufferSize(), edit_state.Text+ib, edit_state.Text+ie); + g.IO.SetClipboardTextFn(tmp_buf); + TempBufferUnlock(); } if (cut) @@ -5931,7 +5954,8 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT // Note that as soon as we can focus into the input box, the in-widget value gets priority over any underlying modification of the input buffer // FIXME: We actually always render 'buf' in RenderTextScrolledClipped // FIXME-OPT: CPU waste to do this every time the widget is active, should mark dirty state from the stb_textedit callbacks - ImTextStrToUtf8(text_tmp_utf8, IM_ARRAYSIZE(text_tmp_utf8), edit_state.Text, NULL); + char* tmp_buf = TempBufferLock(); + ImTextStrToUtf8(tmp_buf, TempBufferSize(), edit_state.Text, NULL); // User callback if ((flags & (ImGuiInputTextFlags_CallbackCompletion | ImGuiInputTextFlags_CallbackHistory | ImGuiInputTextFlags_CallbackAlways)) != 0) @@ -5962,7 +5986,7 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT ImGuiTextEditCallbackData callback_data; callback_data.EventFlag = event_flag; callback_data.EventKey = event_key; - callback_data.Buf = text_tmp_utf8; + callback_data.Buf = tmp_buf; callback_data.BufSize = edit_state.BufSizeA; callback_data.BufDirty = false; callback_data.Flags = flags; @@ -5977,7 +6001,7 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT callback(&callback_data); // Read back what user may have modified - IM_ASSERT(callback_data.Buf == text_tmp_utf8); // Invalid to modify those fields + IM_ASSERT(callback_data.Buf == tmp_buf); // Invalid to modify those fields IM_ASSERT(callback_data.BufSize == edit_state.BufSizeA); IM_ASSERT(callback_data.Flags == flags); if (callback_data.CursorPos != utf8_cursor_pos) edit_state.StbState.cursor = ImTextCountCharsFromUtf8(callback_data.Buf, callback_data.Buf + callback_data.CursorPos); @@ -5985,17 +6009,18 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT if (callback_data.SelectionEnd != utf8_selection_end) edit_state.StbState.select_end = ImTextCountCharsFromUtf8(callback_data.Buf, callback_data.Buf + callback_data.SelectionEnd); if (callback_data.BufDirty) { - ImTextStrFromUtf8(edit_state.Text, IM_ARRAYSIZE(edit_state.Text), text_tmp_utf8, NULL); + ImTextStrFromUtf8(edit_state.Text, IM_ARRAYSIZE(edit_state.Text), tmp_buf, NULL); edit_state.CursorAnimReset(); } } } - if (strcmp(text_tmp_utf8, buf) != 0) + if (strcmp(tmp_buf, buf) != 0) { - ImFormatString(buf, buf_size, "%s", text_tmp_utf8); + ImFormatString(buf, buf_size, "%s", tmp_buf); value_changed = true; } + TempBufferUnlock(); } } From f4970d0e00836b61f2100e9db4820667e10024bb Mon Sep 17 00:00:00 2001 From: ocornut Date: Mon, 16 Mar 2015 09:54:28 +0000 Subject: [PATCH 43/44] Moved static buffers into state No recursion in any of those functions. Tho keeping a lock api may be preferable? --- imgui.cpp | 69 +++++++++++++++++-------------------------------------- 1 file changed, 21 insertions(+), 48 deletions(-) diff --git a/imgui.cpp b/imgui.cpp index 31d9d7cd..b2c8d11c 100644 --- a/imgui.cpp +++ b/imgui.cpp @@ -1068,7 +1068,6 @@ struct ImGuiState float FramerateSecPerFrame[120]; // calculate estimate of framerate for user int FramerateSecPerFrameIdx; float FramerateSecPerFrameAccum; - bool TempBufferLocked; char TempBuffer[1024*3+1]; // temporary text buffer ImGuiState() @@ -1218,26 +1217,6 @@ static void RegisterAliveId(ImGuiID id) g.ActiveIdIsAlive = true; } -static char* TempBufferLock() -{ - ImGuiState& g = *GImGui; - IM_ASSERT(!g.TempBufferLocked); - g.TempBufferLocked = true; - return g.TempBuffer; -} - -static void TempBufferUnlock() -{ - ImGuiState& g = *GImGui; - IM_ASSERT(g.TempBufferLocked); - g.TempBufferLocked = false; -} - -static size_t TempBufferSize() -{ - return sizeof(GImGui->TempBuffer); -} - //----------------------------------------------------------------------------- // Helper: Key->value storage @@ -3847,13 +3826,13 @@ ImGuiStorage* ImGui::GetStateStorage() void ImGui::TextV(const char* fmt, va_list args) { + ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); if (window->SkipItems) return; - static char buf[1024]; - const char* text_end = buf + ImFormatStringV(buf, IM_ARRAYSIZE(buf), fmt, args); - TextUnformatted(buf, text_end); + const char* text_end = g.TempBuffer + ImFormatStringV(g.TempBuffer, IM_ARRAYSIZE(g.TempBuffer), fmt, args); + TextUnformatted(g.TempBuffer, text_end); } void ImGui::Text(const char* fmt, ...) @@ -4018,9 +3997,8 @@ void ImGui::LabelTextV(const char* label, const char* fmt, va_list args) const ImGuiStyle& style = g.Style; const float w = ImGui::CalcItemWidth(); - static char buf[1024]; - const char* value_text_begin = &buf[0]; - const char* value_text_end = value_text_begin + ImFormatStringV(buf, IM_ARRAYSIZE(buf), fmt, args); + const char* value_text_begin = &g.TempBuffer[0]; + const char* value_text_end = value_text_begin + ImFormatStringV(g.TempBuffer, IM_ARRAYSIZE(g.TempBuffer), fmt, args); const ImVec2 label_size = CalcTextSize(label, NULL, true); const ImGuiAabb value_bb(window->DC.CursorPos, window->DC.CursorPos + ImVec2(w + style.FramePadding.x*2, label_size.y)); @@ -4530,9 +4508,8 @@ void ImGui::BulletTextV(const char* fmt, va_list args) if (window->SkipItems) return; - static char buf[1024]; - const char* text_begin = buf; - const char* text_end = text_begin + ImFormatStringV(buf, IM_ARRAYSIZE(buf), fmt, args); + const char* text_begin = g.TempBuffer; + const char* text_end = text_begin + ImFormatStringV(g.TempBuffer, IM_ARRAYSIZE(g.TempBuffer), fmt, args); const ImGuiStyle& style = g.Style; const float line_height = window->FontSize(); @@ -4559,18 +4536,18 @@ void ImGui::BulletText(const char* fmt, ...) // If returning 'true' the node is open and the user is responsible for calling TreePop bool ImGui::TreeNodeV(const char* str_id, const char* fmt, va_list args) { + ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); if (window->SkipItems) return false; - static char buf[1024]; - ImFormatStringV(buf, IM_ARRAYSIZE(buf), fmt, args); + ImFormatStringV(g.TempBuffer, IM_ARRAYSIZE(g.TempBuffer), fmt, args); if (!str_id || !str_id[0]) str_id = fmt; ImGui::PushID(str_id); - const bool opened = ImGui::CollapsingHeader(buf, "", false); + const bool opened = ImGui::CollapsingHeader(g.TempBuffer, "", false); ImGui::PopID(); if (opened) @@ -4591,18 +4568,18 @@ bool ImGui::TreeNode(const char* str_id, const char* fmt, ...) // If returning 'true' the node is open and the user is responsible for calling TreePop bool ImGui::TreeNodeV(const void* ptr_id, const char* fmt, va_list args) { + ImGuiState& g = *GImGui; ImGuiWindow* window = GetCurrentWindow(); if (window->SkipItems) return false; - static char buf[1024]; - ImFormatStringV(buf, IM_ARRAYSIZE(buf), fmt, args); + ImFormatStringV(g.TempBuffer, IM_ARRAYSIZE(g.TempBuffer), fmt, args); if (!ptr_id) ptr_id = fmt; ImGui::PushID(ptr_id); - const bool opened = ImGui::CollapsingHeader(buf, "", false); + const bool opened = ImGui::CollapsingHeader(g.TempBuffer, "", false); ImGui::PopID(); if (opened) @@ -5899,10 +5876,8 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT { const int ib = edit_state.HasSelection() ? ImMin(edit_state.StbState.select_start, edit_state.StbState.select_end) : 0; const int ie = edit_state.HasSelection() ? ImMax(edit_state.StbState.select_start, edit_state.StbState.select_end) : (int)edit_state.CurLenW; - char* tmp_buf = TempBufferLock(); - ImTextStrToUtf8(tmp_buf, TempBufferSize(), edit_state.Text+ib, edit_state.Text+ie); - g.IO.SetClipboardTextFn(tmp_buf); - TempBufferUnlock(); + ImTextStrToUtf8(g.TempBuffer, IM_ARRAYSIZE(g.TempBuffer), edit_state.Text+ib, edit_state.Text+ie); + g.IO.SetClipboardTextFn(g.TempBuffer); } if (cut) @@ -5954,8 +5929,7 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT // Note that as soon as we can focus into the input box, the in-widget value gets priority over any underlying modification of the input buffer // FIXME: We actually always render 'buf' in RenderTextScrolledClipped // FIXME-OPT: CPU waste to do this every time the widget is active, should mark dirty state from the stb_textedit callbacks - char* tmp_buf = TempBufferLock(); - ImTextStrToUtf8(tmp_buf, TempBufferSize(), edit_state.Text, NULL); + ImTextStrToUtf8(g.TempBuffer, IM_ARRAYSIZE(g.TempBuffer), edit_state.Text, NULL); // User callback if ((flags & (ImGuiInputTextFlags_CallbackCompletion | ImGuiInputTextFlags_CallbackHistory | ImGuiInputTextFlags_CallbackAlways)) != 0) @@ -5986,7 +5960,7 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT ImGuiTextEditCallbackData callback_data; callback_data.EventFlag = event_flag; callback_data.EventKey = event_key; - callback_data.Buf = tmp_buf; + callback_data.Buf = g.TempBuffer; callback_data.BufSize = edit_state.BufSizeA; callback_data.BufDirty = false; callback_data.Flags = flags; @@ -6001,7 +5975,7 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT callback(&callback_data); // Read back what user may have modified - IM_ASSERT(callback_data.Buf == tmp_buf); // Invalid to modify those fields + IM_ASSERT(callback_data.Buf == g.TempBuffer); // Invalid to modify those fields IM_ASSERT(callback_data.BufSize == edit_state.BufSizeA); IM_ASSERT(callback_data.Flags == flags); if (callback_data.CursorPos != utf8_cursor_pos) edit_state.StbState.cursor = ImTextCountCharsFromUtf8(callback_data.Buf, callback_data.Buf + callback_data.CursorPos); @@ -6009,18 +5983,17 @@ bool ImGui::InputText(const char* label, char* buf, size_t buf_size, ImGuiInputT if (callback_data.SelectionEnd != utf8_selection_end) edit_state.StbState.select_end = ImTextCountCharsFromUtf8(callback_data.Buf, callback_data.Buf + callback_data.SelectionEnd); if (callback_data.BufDirty) { - ImTextStrFromUtf8(edit_state.Text, IM_ARRAYSIZE(edit_state.Text), tmp_buf, NULL); + ImTextStrFromUtf8(edit_state.Text, IM_ARRAYSIZE(edit_state.Text), g.TempBuffer, NULL); edit_state.CursorAnimReset(); } } } - if (strcmp(tmp_buf, buf) != 0) + if (strcmp(g.TempBuffer, buf) != 0) { - ImFormatString(buf, buf_size, "%s", tmp_buf); + ImFormatString(buf, buf_size, "%s", g.TempBuffer); value_changed = true; } - TempBufferUnlock(); } } From d3e444dfd96338b4cfda56420b3ca9819e940c6a Mon Sep 17 00:00:00 2001 From: ocornut Date: Mon, 16 Mar 2015 10:02:10 +0000 Subject: [PATCH 44/44] Examples: DirectX9: fixed duplicate creation of vertex buffer. Size of static vertex buffer at top of the code. --- examples/directx11_example/imgui_impl_dx11.cpp | 3 ++- examples/directx9_example/imgui_impl_dx9.cpp | 6 ++---- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/examples/directx11_example/imgui_impl_dx11.cpp b/examples/directx11_example/imgui_impl_dx11.cpp index 2502b98d..ba8d3d8b 100644 --- a/examples/directx11_example/imgui_impl_dx11.cpp +++ b/examples/directx11_example/imgui_impl_dx11.cpp @@ -27,6 +27,7 @@ static ID3D11PixelShader* g_pPixelShader = NULL; static ID3D11SamplerState* g_pFontSampler = NULL; static ID3D11ShaderResourceView*g_pFontTextureView = NULL; static ID3D11BlendState* g_blendState = NULL; +static int VERTEX_BUFFER_SIZE = 30000; // TODO: Make vertex buffer smaller and grow dynamically as needed. struct CUSTOMVERTEX { @@ -358,7 +359,7 @@ bool ImGui_ImplDX11_CreateDeviceObjects() D3D11_BUFFER_DESC bufferDesc; memset(&bufferDesc, 0, sizeof(D3D11_BUFFER_DESC)); bufferDesc.Usage = D3D11_USAGE_DYNAMIC; - bufferDesc.ByteWidth = 100000 * sizeof(CUSTOMVERTEX); // Maybe we should handle that more dynamically? + bufferDesc.ByteWidth = VERTEX_BUFFER_SIZE * sizeof(CUSTOMVERTEX); bufferDesc.BindFlags = D3D11_BIND_VERTEX_BUFFER; bufferDesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE; bufferDesc.MiscFlags = 0; diff --git a/examples/directx9_example/imgui_impl_dx9.cpp b/examples/directx9_example/imgui_impl_dx9.cpp index 60302d4c..b726e50e 100644 --- a/examples/directx9_example/imgui_impl_dx9.cpp +++ b/examples/directx9_example/imgui_impl_dx9.cpp @@ -15,6 +15,7 @@ static INT64 g_Time = 0; static INT64 g_TicksPerSecond = 0; static LPDIRECT3DDEVICE9 g_pd3dDevice = NULL; static LPDIRECT3DVERTEXBUFFER9 g_pVB = NULL; +static int VERTEX_BUFFER_SIZE = 30000; // TODO: Make vertex buffer smaller and grow dynamically as needed. struct CUSTOMVERTEX { @@ -183,9 +184,6 @@ bool ImGui_ImplDX9_Init(void* hwnd, IDirect3DDevice9* device) io.RenderDrawListsFn = ImGui_ImplDX9_RenderDrawLists; io.ImeWindowHandle = g_hWnd; - if (g_pd3dDevice->CreateVertexBuffer(10000 * sizeof(CUSTOMVERTEX), D3DUSAGE_DYNAMIC | D3DUSAGE_WRITEONLY, D3DFVF_CUSTOMVERTEX, D3DPOOL_DEFAULT, &g_pVB, NULL) < 0) - return false; - return true; } @@ -232,7 +230,7 @@ bool ImGui_ImplDX9_CreateDeviceObjects() if (!g_pd3dDevice) return false; - if (g_pd3dDevice->CreateVertexBuffer(10000 * sizeof(CUSTOMVERTEX), D3DUSAGE_DYNAMIC | D3DUSAGE_WRITEONLY, D3DFVF_CUSTOMVERTEX, D3DPOOL_DEFAULT, &g_pVB, NULL) < 0) + if (g_pd3dDevice->CreateVertexBuffer(VERTEX_BUFFER_SIZE * sizeof(CUSTOMVERTEX), D3DUSAGE_DYNAMIC | D3DUSAGE_WRITEONLY, D3DFVF_CUSTOMVERTEX, D3DPOOL_DEFAULT, &g_pVB, NULL) < 0) return false; ImGui_ImplDX9_CreateFontsTexture();