It’s common in VR to be able to interact with the world simply by looking at it. This is called gaze based interaction. From menus to game mechanics, gaze based interaction is a powerful VR tool.
In this quick tutorial, we’re going to cover an implementation of gaze based selection. Here’s what a selection will look like:
We’ll be creating the interface that allows us to:
This lesson assumes you’ve gone through the Materials lesson and are comfortable with updating materials dynamically.
Before we jump right into it, let’s review what we’re starting with.
BP_GazePlayer is the character pawn we’re going to be adding our selection logic to. It’s already set up with a reticle and a Dynamic Material Instance variable for the reticle called ReticleDMI!
We should point out a few things that are different about this setup:
You may notice that the reticle is not implemented with a trace. Instead, it uses a Spring Arm component.
We used the Spring Arm component in Lesson 4 to add a smooth lerp effect to the 2.5D camera. We use the Spring Arm here for the same reason.
The Spring Arm also works exactly as your trace reticle works: the reticle is attached to the end and extends as far as it can until it hits something. This is in fact the core purpose the SpringArm. Here are all the details we’ve modified (marked by the grey arrows):
Hover over any of the details in the editor to get a description of what they do.
Finally, in the Event Graph we call a function called Update Spring Arm on Event Tick. Open the function and you’ll see that it ensures the SpringArm is rooted at the exact position of the player’s head.
This solves a problem where the HMD and spring arm are not in sync due to translational movement, causing the reticle to be a bit off center.
Attached to our SpringArm is a reticle made with a MaterialBillboard. This reticle uses the M_ProgressReticle material which is set up to show progress with a radial selection bar. It utilizes a Progress scalar variable that goes from 0 to 1. Since we need to dynamically update this material, the BP_GazePlayer has a ReticleDMI variable that is set in the ConstructionScript.
Notice the new Create Dynamic Billboard Material which is a VARlabs utility node meant to take a MaterialBillboard and return a dynamic version of its material.
That’s it!
Many people prefer the smooth lerp of this reticle. Consider using this setup moving forward. Of course, you can change the amount of lag by updating the SpringArm’s Camera Rotation Lag Speed setting.
It’s time to set up the foundation of our selection interaction. We’ll do this with the goal of creating our hover on/off script. By the end of this section, we’ll have something that looks like:
This type of interaction, one that we expect to happen all over the place, is perfect for a Blueprint Interface (BPI).
Let’s start by adding messages for Hover On and Hover Off. We’ll use these to try and communicate with Actors in the world from our BP_GazePlayer.
Before we jump into the BP_GazePlayer interaction, let’s set up a button that implements our new Hover On and Hover Off events.
Currently, the BP_Button is a static mesh and a Dynamic Material Instance variable named DMI. Let’s add our BPI and its hover events:
Now we can add the interface events in the Event Graph:
Great! Now we can start scripting our player to fire these events when appropriate:
In order to communicate with Actors in front of our player on gaze, we need to send out a trace every Tick.
Challenge:
Back to basics: set up a Line Trace By Channel that traces from the player’s eyes to 10000cm out in front the of the player every frame.
Great! To send the Hover On and Hover Off messages, we need to know when we’ve hovered over a new Actor. To do that, we have to store the Actor we traced last tick, then check if it is the same Actor we just traced.
We just set up a Branch that tells us whether or not we are tracing the same Actor as we did the previous frame. The Sequence is used to set our Last Trace Actor variable at the very end of the tick. All of this will help us hover on and off other Actors.
In this case, we’re not using the LineTraceByChannel > Return Value with a Branch because it is unnecessary. If our trace doesn’t hit anything, the hit’s Hit Actor will return NULL.
Make sure you understand what’s going on with our Last Trace Actor variable and the Branch. Take your time to trace the execution path if it’s confusing you.
Challenge:
Send the Hover On and Hover Off messages to Hit Actor and Last Trace Actor at the correct time.
Hint: Find the execution path that defines when we have hovered over a new Actor. At that point, send the on/off messages to the proper Actors.
Awesome! Go ahead and test it out. When you look at the button in the world, your strings should print “hover on” and “hover off” appropriately.
Rather than print strings, let’s use our DMI to change the color of our button.
Challenge:
Use the DMI variable to change the color of the button when it is hovered on and off. You’ll want to change the “Color” parameter of the material which can be done with a Set Vector Parameter Value node.
To get access to the Set Vector Parameter Value node, you must first drag in a reference to the DMI variable, then pull a wire out of it to access its context sensitive menu.
Tada!
Let’s get our reticle to show the progress of our selection! To do that, we’ll need a new variable:
Challenge:
Add 0.01 to the HoverProgress variable if we’re hovering over the same Actor as last tick. If we’re not, set HoverProgress back to 0.
To test that this is working, use a Print String node at the end of the tick to print the value of HoverProgress
When you test it, you’ll see numbers increasing from 0 printed to the screen. Whenever you hover over a new object, the count will restart from 0:
Great! Rather than print the numbers, let’s use our M_ProgressReticle’s scalar “progress” variable.
Challenge:
Use the ReticleDMI variable to update the “progress” value of the reticle. You’ll want to change the “Progress” parameter of the material which can be done with a Set Scalar Parameter Value node.
To get access to the Set Scalar Parameter Value node, you must first drag in a reference to the ReticleDMI variable, then pull a wire out of it to access its context sensitive menu.
Update the progress in the same place you called Print String
The reticle should look like this when you’re done:
Very nice!
Let’s address the issue of our reticle trying to select everything in the world. It only makes sense for our reticle to show progress when it is hovering something that is actually selectable. Let’s implement a BPI message that returns a boolean value to inform the player if they’re hovering a selectable Actor:
Now we have to go implement our function in BP_Button.
You won’t see the new function unless you Compiled your BPI_GazeInteraction.
If you are unable to double click and open the Progress function, try closing and reopening the BP_Button window.
Now our BP_Button class has implemented the Progress message/function. With this done, we can go back to our player and send that message to ensure we’re hovering over a selectable button.
Challenge:
Use the new Progress message and a Branch to check if you should add 0.01 to HoverProgress
If the hovered Actor is not Selectable, set HoverProgress back to 0.
When finished, your reticle should only show progress when hovering over a BP_Button.
Excellent! Now all we have to do is select our button when HoverProgress reaches 1. This calls for another BPI message.
Let’s go ahead and implement this message in BP_Button.
Great! Let’s fire this event from BP_GazePlayer.
Challenge:
When HoverProgress reaches 1, send the Select message to the button.
Selection will look like this on completion:
Okay, so you’ve noticed how once HoverProgress reaches 1, we fire Select over and over again. Let’s fix this on the BP_Button side by making it unselectable for a short amount of time.
First, let’s add a boolean variable Selectable that we can switch from true to false.
Now we have to return this variable in our Progress function rather than always return true.
Try it out! Now you can only select each button once.
Great stuff! To finalize the button, let’s add an animation of it moving up and down, then reset Selectable back to true so we can continue to press it.
Challenge:
Use a Timeline to animate the button up and down once Select is called. Once the animation is over, set Selectable back to true so we can continue to press the button.
Your finished product should look like:
WoooOOOOoOooOOOOoOO!!!!
After a bit of set up, you now have a very extendable and reusable gaze based interaction interface. With the same (or slightly altered) BP_GazePlayer you can easily create all types of variants of the BP_Button to make your own unique form of gaze based selection.
If you want to continue using the M_ProgressReticle, make sure to make a Material Instance out of it to see all the ways you can customize it.
Check out the Lessons > Lesson7 > Examples folder for more ideas on how to extend this form of interaction and selection.