-
Notifications
You must be signed in to change notification settings - Fork 32
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add Fly camera + fix gimbal locking issue #30
base: develop
Are you sure you want to change the base?
Conversation
Hi Alex, Exposing the camera fly speed and mouse locking prefs via ViewportSettings/ImagingSettings is the way to go. I think the mouse locking can be the new behaviour, replacing the older one, as I don’t see why we would use the current behaviour over the locked mouse. So may be removing the option to select it and make it default is better The code interacting with the mouse should ideally go in the camera manipulator as you figured it out. But the camera rig Move function only deals with mouse positions deltas while your code needs to deal with key pressed. I had in mind originally to have several camera manipulators: one for the “arc ball, turntable, etc” and one for the fly and walk navigation, adding the ability to choose between them via a button, a bit like how you would do it in blender. In that case the new camera manipulator could call another Move function with "movements", up down, forward, back etc. What do you think ? Having the right mouse click to enable this camera manipulator is also cool. The camera pointing downwards is due to the fact that usd can have the space oriented differently, with Z or Y as up axis and UT is not alway handling it correctly. There is an option is the stage to specify the orientation and before loading the stage we don't know which one it will be. It’s on my list to deal with the different situations happening with the upAxis difference between stages. I’ll add a some comments on the code, overall looks great ! Cheers, |
Hmm, I didn't notice the camera jumping randomly until you pointed it out. I couldn't get it to reproduce consistently either, but with print debugging I can see the cause is due to random large spikes in the mouse delta coming into the Move function. I suspect it's related to this bug in GLFW, which other people report when using the same API to lock the mouse. It seems like a common solution is to always ignore the first delta after locking the mouse. I'll need to investigate a bit more to find the best place to implement that. I like the idea of having a dedicated manipulator for the fly camera, as it'll provide a cleaner way to implement more advanced features like input rebinding, support for more peripherals (gamepad, etc). One approach could be to rename Alternatively, the current |
Thanks Alex ! |
Alright, just had a 'fun' time doing a bunch of camera math :)
And now the ugly
I did at least confirm that disabling mouse capture fixes the random jumping issues. |
src/Editor.cpp
Outdated
@@ -838,7 +844,41 @@ float Editor::GetScaleUI() const { | |||
return _settings._uiScale; | |||
} | |||
|
|||
static int gSkippedFirstMouseEventAfterCapture = 0; | |||
GfVec2d Editor::GetMouseDelta() { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
since _mouseCapture
and gSkippedFirstMouseEventAfterCapture
are somewhat unique when the application run, they could go in the Viewport as static variables instead of the Editor. All the functions related to those variables could also move as static functions in Viewport. This will also remove the need for an editor singleton, the calls would look like Viewport::SetMouseCaptured(true);
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I moved the mouse capture API into Editor because it makes more sense to have it only in one place, since mouse "locking" is a global flag unassociated with any viewport/widget. Having it in viewport makes it seem like it's valid to call SetMouseCaptured(true)
simultaneously for multiple viewports. Doing so shouldn't break anything, but I think it's a confusing API. Additionally, other systems/widgets might want to lock the mouse too.
If you don't like Editor becoming a singleton, it could instead go in another singleton dedicated to global window manager/platform stuff. In the future, it might be expanded to expose things like tray icons, multiple windows, system notifications, etc.
...but actually think I should just remove all the mouse locking stuff for now and leave a stub, until I can find the time to properly track down and fix the mouse jumping thing. What do you think?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
btw, if it wasn't clear from my comment, the gSkippedFirstMouseEventAfterCapture
stuff doesn't really solve the problem. Making it skip 3 frames appears to solve it on my machine, but as I can't reliably reproduce the bug, I'm not 100% sure, and I feel like it's just sweeping the problem under the rug...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The mouse capture is really cool, if the problem is only happening when glfw hides the cursor, then I think it’s not a big deal if we don’t hide the cursor but keep the capture mechanism. This would avoid the workaround for sure.
Ok, I understand why you wanted to add the capture in the Editor. My point was more about trying to avoid storing the Editor pointer in a global variable, I think it is better to store the mouseCapture variable as static variable instead. In the Editor or Viewport then have static functions access it:
class Editor {
static void SetMouseCaptured(bool captured);
static bool GetMouseCaptured() { return _mouseCaptured; }
static bool _mouseCaptured;
};
This way you can access the mouse capture functions without the need of the GetInstance() function like this:
….
Editor::SetMouseCaptured(true);
…
Personally I would add mouseCapture in the Viewport since it’s only used there and refactor it if it is needed somewhere else later on, but I don’t mind if it is in the Editor as it also makes sense, your call :)
The code proposed here is not “technically” a singleton in the sense that a singleton would prevent the creation of objects to make sure there is only one possible in the application. Here it is still possible to create more Editor and GetInstance would point to one of them only. If we wanted to have a singleton we would have to rewrite the constructor and all the Editor function calls with GetInstance(), that’s a lot of refactoring for not additional feature or functionalities. Also I have in mind that we might need to have multiple editors living at the same time, possibly in a wasm port, so I tried to avoid making it a singleton since there is no compelling reason to do so.
Thanks for all the update, that’s really cool !
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I spent some time debugging the mouse issue more thoroughly, and I've at least managed to confirm why skipping 3 frames works. The TL;DR is that it has to do with the way the delta is calculated using cached mouse position from previous frames in imgui_impl_glfw
. I think there's a path to solve the problem by skipping less frames, but doing so would require quite a bit of refactoring and adding new APIs, and probably not worth the effort. FWIW I did a blind test with someone, and they didn't notice the initial 3 frame latency when controlling the camera.
I refactored the existing solution a bit to get rid of Editor::GetMouseDelta
to avoid further bloating the Editor class, and I got rid of GetInstance too as your idea for making the mouse capture static is certainly better. However, I left the SetMouseCaptured
API in Editor. This can change in the future if needed, but for now I think Editor::SetMouseCaptured
is simple enough.
The commit I just pushed contains the refactored mouse lock code, which didn't exhibit the jumping behavior in my local testing, but more testing from other people on other platforms would be needed. If it does still happen, tweaking gSkipCapturedMouseDelta = 2;
(in Editor::SetMouseCaptured
) to a higher value will likely fix it.
src/Editor.cpp
Outdated
} | ||
|
||
|
||
void Editor::SetMouseCaptured(bool set) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
bool set
I generally avoid to name variables with names which can be confused with other things like data structures. Ideally renaming set
to something else like mouseCaptured
or other would be great
src/Editor.cpp
Outdated
if(gSkippedFirstMouseEventAfterCapture > 0){ | ||
ImGuiIO& io = ImGui::GetIO(); | ||
if(std::abs(io.MouseDelta.x) > 0 || std::abs(io.MouseDelta.y) > 0){ | ||
printf("SKIPPED (%f, %f)\n", io.MouseDelta.x, io.MouseDelta.y); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
remove printf ?
float FlyCameraManipulator::InputForward() { | ||
if (ImGui::IsKeyDown(ImGuiKey_W) || ImGui::IsKeyDown(ImGuiKey_UpArrow)) | ||
return 1.0f; | ||
if (ImGui::IsKeyDown(ImGuiKey_S) || ImGui::IsKeyDown(ImGuiKey_DownArrow)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
would be better to have an else if
instead of ìf`since it allows to avoid testing more conditions and save some cpu work
float FlyCameraManipulator::InputRight() { | ||
if (ImGui::IsKeyDown(ImGuiKey_D) || ImGui::IsKeyDown(ImGuiKey_RightArrow)) | ||
return 1.0f; | ||
if (ImGui::IsKeyDown(ImGuiKey_A) || ImGui::IsKeyDown(ImGuiKey_LeftArrow)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
same as above if
->else if
Could you a formatting pass the parts of the code you added/modified with clang-format ? There is a rule file a the root of the project so clang-format will pickup the correct formatting. (Try not to format all the project otherwise there will be a lot of diffs :) ) |
removed Editor::GetMouseDelta and Editor::GetInstance cleaned up glfw mouse delta bug workaround
Hi @AlexRamallo , are you happy with your last changes ? To me it's all good, let me know and I'll merge. |
I decided to do some work on the viewport cameras. The main changes here are:
I exposed the camera fly speed and mouse locking prefs via ViewportSettings/ImagingSettings. Not sure if it's the best way to do that. Storing the camera speed in user prefs probably isn't very useful, but currently it doesn't seem like per-viewport settings get saved anywhere.
The fly mode controls are implemented in
CameraRig.cpp
using ImGui APIs, and so I ended up having to pull in theGui.h
header. This is probably not the best place to put it. It might be better to move that intoCameraManipulator.cpp
, and pass the inputs as member variables of CameraRig?Also, I noticed that the viewport cameras are initialized to be looking 'down'. If you load up the kitchen scene, the camera is on top and looking down at the roof, instead of looking into the kitchen as you'd expect. I wasn't sure why this was, so I didn't change it. The new orbit code correctly initializes at that same angle, but I can change it if desired.