Skip to content
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

Timeline arrow key navigation #89

Open
wants to merge 6 commits into
base: main
Choose a base branch
from

Conversation

ThomasWilshaw
Copy link
Contributor

I mad a start on tackling #73 here and I think I made some decent progress. This currently adds support for using the left and right arrow keys to move around the timeline and change the selected item. It's a little clunky when moving off the edge of the visible workarea and I'd love some input on improving that. At the moment it jumps forwards or backwards by the size of the newly selected item.

I'm not entirely sure I've implemented the keyboard control in the most elegant way or put the code in the correct place in the code base, imeadiate mode GUIs are still new to me. The other thing to note is it was necessary to disable the built in keyboard navigation to stop it conflicting

with the new controls in odd ways. Is that okay? I think the way we're using a table here is not exactly how it was intended to be used.

Fixes #73

2024-11-29.17-11-26.mp4

Add inital support for left and right arrow key navigation in the
timeline. It was necessary to turn off the default navigation as it was
interacting badly.

Signed-off-by: Thomas Wilshaw <[email protected]>
@ThomasWilshaw ThomasWilshaw marked this pull request as draft November 29, 2024 17:40
@ThomasWilshaw
Copy link
Contributor Author

I'm marking this as ready because it does work. However, I feel it's a bit clunky and could be polished so any feedback would be great.

2024-12-02.19-02-48.mp4

@ThomasWilshaw ThomasWilshaw marked this pull request as ready for review December 2, 2024 19:06
Copy link
Member

@jminor jminor left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This looks great! I just tried it out and it is so much better than before.

I left a couple of comments about checking for null to avoid crashing in some edge cases.

A couple of additions that would be helpful (and could be added later):

  • Scrolling up/down when the selection is off the top/bottom.
  • Stepping from the bottom video track to the top audio track and vice-versa.

// Left Arrow
if (ImGui::IsKeyPressed(ImGuiKey::ImGuiKey_LeftArrow)) {
std::string selected_type = appState.selected_object->schema_name();
if (selected_type == "Clip" || selected_type == "Gap" || selected_type == "Transition") {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of checking the schema_name, it would be safer and shorter to check if the dynamic_cast to Composable returned a value. If it was non-null, then you can find the parent. For safety you might also want to check that parent() was non-null, since the top-level stack won't have a parent. That will cover all the generalized cases - though the left/right navigation is really assuming the parent is a Track (vs a Stack).

So maybe the logic could be:

  • Try dynamic_cast to Composable.
  • Get the parent()
  • Check that the parent is not null
  • Check that the parent is a Track

if (selected_type == "Clip" || selected_type == "Gap") {
auto child = dynamic_cast<otio::Item*>(appState.selected_object);
start_time = child->range_in_parent().start_time();
} else if (selected_type == "Transition") {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You could avoid this special case by using parent->range_of_child(child) which works on all Composables.

}

// Get the parent object and track stack
auto parent = dynamic_cast<otio::Composable*>(appState.selected_object)->parent();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There's already a parent variable on line 1574.


// Get the parent object and track stack
auto parent = dynamic_cast<otio::Composable*>(appState.selected_object)->parent();
auto tracks = dynamic_cast<otio::Stack*>(parent->parent());
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You'll need to check that tracks was non-null. If you have an OTIO with nested compositions, the parent's parent might not be a Stack. At the moment, I'm not sure you could end up selecting such an item in Raven, but we will allow for it eventually.

if (track == parent) {
// Down Arrow and Video or Up Arrow and Audio
if ((ImGui::IsKeyPressed(ImGuiKey::ImGuiKey_DownArrow) && selected_track_type == "Video") ||
(ImGui::IsKeyPressed(ImGuiKey::ImGuiKey_UpArrow) && selected_track_type == "Audio")) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oof, sorry about this track order hassle. I often wish we had designed Timeline to have separate lists for video and audio (or other) tracks. I suppose Raven could make separate lists when it first loads? Anyways, the solution you have here is fine.

otio::Track* next_track = dynamic_cast<otio::Track*>(next_it);

// Only iterate over tracks of the same kind
if(next_track->kind() != selected_track_type){
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you need to check for next_track being null here?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Arrow keys should navigate by selection
2 participants