Data Analysis of Eye Data with Motion Data in VR 3D coordinates

Hello!

I am working on a VR experiment using HTC Vive Pro Eye with the SRanipal SDK in Unity. The experiment involves a pedestrian that needs to make a decision on whether it is safe to cross the road based on different vehicle approaching speeds, crosswalk geometric designs, deceleration rates, etc. If he/she feels it is safe, then they should cross the road.

I can collect motion data (position and rotation of HMD in x, y, z) and eye data. I would like to know if there is a way to analyze the eye data with the motion data? For example, a 3D heatmap of the VR environment to show where the pedestrian was looking during the experiment? Or any other way to analyze the data of eye gaze.

I have attached an example of eye data I can collect and a picture of one scenario of my VR environment for reference.

If anyone can help me on this, it would be highly appreciated. Thank you! :slight_smile:

8363511–1101489–eyedata_P11_S1.txt (1.44 MB)

The gaze origin and direction are given in local space so you would need to use the camera’s Transform to TransformPoint() and TransformDirection() them into world space. Then use the Physics API to cast that ray into the scene.

From there you will have a series of gaze points which you can accumulate into a 3D texture. Sample that texture in the shader used for the environment to visualize the heatmap. Additionally, if you log the pedestrians head position, you could also use the point array (interpolate between the two nearest points in time to drive a simple distance calculation in a shader) to replay where they were looking at any point during the trial.

1 Like

Hi @Shane_Michael
Thank you so much for your response!
I was just wondering, is it still possible to transform the gaze origin and direction from local space to world space after the trials? The main reason I am asking is because the experiment is ongoing and I already collected eye data for 10 people. Or do I have to start again and implement the TransformPoint() and TransformDirection() operations into my Eye Tracking script?

As long as you were logging the HMD rotation/position data, then you can definitely do the math after the fact. Easiest would be to simply instantiate a dummy Transform, iterate through the points to set its position/rotation to match the head position, and use the TransformPoint/Direction functions on that Transform.

Hi @Shane_Michael
Is it possible to obtain a video recording of the ray casted by the Physics API into the scene for each run?
Also, I am beginner in Unity. I am unsure on how to use the TransformPoint() and TransformDirection() operations to convert the gaze origin and direction from local to world space. I am sorry… Can you please elaborate on how I can do that? I have attached my script that obtains raw eye data for your reference. Thank you. :slight_smile:

using System.Collections;
using System.Runtime.InteropServices;
using UnityEngine;
using System;
using System.IO;
using ViveSR.anipal.Eye;
using ViveSR.anipal;
using ViveSR;

/// <summary>
/// Example usage for eye tracking callback
/// Note: Callback runs on a separate thread to report at ~120hz.
/// Unity is not threadsafe and cannot call any UnityEngine api from within callback thread.
/// </summary>
public class EyeTrackingRecordData : MonoBehaviour
{
    // ********************************************************************************************************************
    //
    //  Define user ID information.
    //  - The developers can define the user ID format such as "ABC_001". The ID is used for the name of text file
    //    that records the measured eye movement data.
    //
    // ********************************************************************************************************************
    public static int UserID = 11; // Always change Participant Number for every participant
    public static int scenario = 1; // Always change Scenario Number for every scenario
    //public string UserID;       // Definte ID number such as 001, ABC001, etc.
    public string Path = Directory.GetCurrentDirectory();
    //string File_Path = Directory.GetCurrentDirectory() + "\\video_" + UserID + ".txt";
    public string File_Path = Directory.GetCurrentDirectory() + "P" + UserID + "_" + "S" + scenario + ".txt";


    // ********************************************************************************************************************
    //
    //  Parameters for time-related information.
    //
    // ********************************************************************************************************************
    public static int cnt_callback = 0;
    public int cnt_saccade = 0, Endbuffer = 3, SaccadeTimer = 30;
    float Timeout = 1.0f, InitialTimer = 0.0f;
    private static long SaccadeEndTime = 0;
    private static long MeasureTime, CurrentTime, MeasureEndTime = 0;
    private static float time_stamp;
    private static int frame;

    // ********************************************************************************************************************
    //
    //  Parameters for eye data.
    //
    // ********************************************************************************************************************
    private static EyeData_v2 eyeData = new EyeData_v2();
    public EyeParameter eye_parameter = new EyeParameter();
    public GazeRayParameter gaze = new GazeRayParameter();
    private static bool eye_callback_registered = false;
    private static UInt64 eye_valid_L, eye_valid_R;                 // The bits explaining the validity of eye data.
    private static float openness_L, openness_R;                    // The level of eye openness.
    private static float pupil_diameter_L, pupil_diameter_R;        // Diameter of pupil dilation.
    private static Vector2 pos_sensor_L, pos_sensor_R;              // Positions of pupils.
    private static Vector3 gaze_origin_L, gaze_origin_R;            // Position of gaze origin.
    private static Vector3 gaze_direct_L, gaze_direct_R;            // Direction of gaze ray.
    private static float frown_L, frown_R;                          // The level of user's frown.
    private static float squeeze_L, squeeze_R;                      // The level to show how the eye is closed tightly.
    private static float wide_L, wide_R;                            // The level to show how the eye is open widely.
    private static double gaze_sensitive;                           // The sensitive factor of gaze ray.
    private static float distance_C;                                // Distance from the central point of right and left eyes.
    private static bool distance_valid_C;                           // Validity of combined data of right and left eyes.
    public bool cal_need;                                           // Calibration judge.
    public bool result_cal;                                         // Result of calibration.
    private static int track_imp_cnt = 0;
    private static TrackingImprovement[] track_imp_item;

    //private static EyeData eyeData = new EyeData();
    //private static bool eye_callback_registered = false;

    //public Text uiText;
    private float updateSpeed = 0;
    private static float lastTime, currentTime;


    // ********************************************************************************************************************
    //
    //  Start is called before the first frame update. The Start() function is performed only one time.
    //
    // ********************************************************************************************************************
    void Start()
    {
        //File_Path = Directory.GetCurrentDirectory() + "\\Assets" + UserID + ".txt";
        InputUserID();                              // Check if the file with the same ID exists.
        //Invoke("SystemCheck", 0.5f);                // System check.
        //SRanipal_Eye_v2.LaunchEyeCalibration();     // Perform calibration for eye tracking.
        //Calibration();
        //TargetPosition();                           // Implement the targets on the VR view.
        //Invoke("Measurement", 0.5f);                // Start the measurement of ocular movements in a separate callback function. 
    }


    // ********************************************************************************************************************
    //
    //  Checks if the filename with the same user ID already exists. If so, you need to change the name of UserID.
    //
    // ********************************************************************************************************************
    void InputUserID()
    {
        Debug.Log(File_Path);

        if (File.Exists(File_Path))
        {
            Debug.Log("File with the same UserID already exists. Please change the UserID in the C# code.");

            //  When the same file name is found, we stop playing Unity.

            if (UnityEditor.EditorApplication.isPlaying)
            {
                UnityEditor.EditorApplication.isPlaying = false;
            }
        }
        else
        {
            Data_txt();
        }
    }


    // ********************************************************************************************************************
    //
    //  Check if the system works properly.
    //
    // ********************************************************************************************************************
    void SystemCheck()
    {
        if (SRanipal_Eye_API.GetEyeData_v2(ref eyeData) == ViveSR.Error.WORK)
        {
            Debug.Log("Device is working properly.");
        }

        if (SRanipal_Eye_API.GetEyeParameter(ref eye_parameter) == ViveSR.Error.WORK)
        {
            Debug.Log("Eye parameters are measured.");
        }

        //  Check again if the initialisation of eye tracking functions successfully. If not, we stop playing Unity.
        Error result_eye_init = SRanipal_API.Initial(SRanipal_Eye_v2.ANIPAL_TYPE_EYE_V2, IntPtr.Zero);

        if (result_eye_init == Error.WORK)
        {
            Debug.Log("[SRanipal] Initial Eye v2: " + result_eye_init);
        }
        else
        {
            Debug.LogError("[SRanipal] Initial Eye v2: " + result_eye_init);

            if (UnityEditor.EditorApplication.isPlaying)
            {
                UnityEditor.EditorApplication.isPlaying = false;    // Stops Unity editor.
            }
        }
    }

    // ********************************************************************************************************************
    //
    //  Calibration is performed if the calibration is necessary.
    //
    // ********************************************************************************************************************
    void Calibration()
    {
        SRanipal_Eye_API.IsUserNeedCalibration(ref cal_need);           // Check the calibration status. If needed, we perform the calibration.

        if (cal_need == true)
        {
            result_cal = SRanipal_Eye_v2.LaunchEyeCalibration();

            if (result_cal == true)
            {
                Debug.Log("Calibration is done successfully.");
            }

            else
            {
                Debug.Log("Calibration is failed.");
                if (UnityEditor.EditorApplication.isPlaying)
                {
                    UnityEditor.EditorApplication.isPlaying = false;    // Stops Unity editor if the calibration if failed.
                }
            }
        }

        if (cal_need == false)
        {
            Debug.Log("Calibration is not necessary");
        }
    }

    // ********************************************************************************************************************
    //
    //  Create a text file and header names of each column to store the measured data of eye movements.
    //
    // ********************************************************************************************************************
    void Data_txt()
    {
        string variable =
        "time(100ns)" + "," +
        "time_stamp(ms)" + "," +
        "frame" + "," +
        "eye_valid_L" + "," +
        "eye_valid_R" + "," +
        "openness_L" + "," +
        "openness_R" + "," +
        "pupil_diameter_L(mm)" + "," +
        "pupil_diameter_R(mm)" + "," +
        "pos_sensor_L.x" + "," +
        "pos_sensor_L.y" + "," +
        "pos_sensor_R.x" + "," +
        "pos_sensor_R.y" + "," +
        "gaze_origin_L.x(mm)" + "," +
        "gaze_origin_L.y(mm)" + "," +
        "gaze_origin_L.z(mm)" + "," +
        "gaze_origin_R.x(mm)" + "," +
        "gaze_origin_R.y(mm)" + "," +
        "gaze_origin_R.z(mm)" + "," +
        "gaze_direct_L.x" + "," +
        "gaze_direct_L.y" + "," +
        "gaze_direct_L.z" + "," +
        "gaze_direct_R.x" + "," +
        "gaze_direct_R.y" + "," +
        "gaze_direct_R.z" + "," +
        "gaze_sensitive" + "," +
        "frown_L" + "," +
        "frown_R" + "," +
        "squeeze_L" + "," +
        "squeeze_R" + "," +
        "wide_L" + "," +
        "wide_R" + "," +
        "distance_valid_C" + "," +
        "distance_C(mm)" + "," +
        "track_imp_cnt" +
        Environment.NewLine;

        File.AppendAllText("eyedata_" + "P" + UserID + "_" + "S" + scenario + ".txt", variable);
    }


    void Update()
    {
        if (SRanipal_Eye_Framework.Status != SRanipal_Eye_Framework.FrameworkStatus.WORKING) return;


        if (SRanipal_Eye_Framework.Instance.EnableEyeDataCallback == true && eye_callback_registered == false)
        {
            SRanipal_Eye_v2.WrapperRegisterEyeDataCallback(Marshal.GetFunctionPointerForDelegate((SRanipal_Eye_v2.CallbackBasic)EyeCallback));
            eye_callback_registered = true;
        }
        else if (SRanipal_Eye_Framework.Instance.EnableEyeDataCallback == false && eye_callback_registered == true)
        {
            SRanipal_Eye_v2.WrapperUnRegisterEyeDataCallback(Marshal.GetFunctionPointerForDelegate((SRanipal_Eye_v2.CallbackBasic)EyeCallback));
            eye_callback_registered = false;
        }

        float timeNow = Time.realtimeSinceStartup;
    }


    private void OnDisable()
    {
        Release();
    }

    void OnApplicationQuit()
    {
        Release();
    }

    /// <summary>
    /// Release callback thread when disabled or quit
    /// </summary>
    private static void Release()
    {
        if (eye_callback_registered == true)
        {
            SRanipal_Eye.WrapperUnRegisterEyeDataCallback(Marshal.GetFunctionPointerForDelegate((SRanipal_Eye_v2.CallbackBasic)EyeCallback));
            eye_callback_registered = false;
        }
    }

    /// <summary>
    /// Required class for IL2CPP scripting backend support
    /// </summary>
    internal class MonoPInvokeCallbackAttribute : System.Attribute
    {
        public MonoPInvokeCallbackAttribute() { }
    }

    /// <summary>
    /// Eye tracking data callback thread.
    /// Reports data at ~120hz
    /// MonoPInvokeCallback attribute required for IL2CPP scripting backend
    /// </summary>
    /// <param name="eye_data">Reference to latest eye_data</param>
    [MonoPInvokeCallback]
    private static void EyeCallback(ref EyeData_v2 eye_data)
    {
        EyeParameter eye_parameter = new EyeParameter();
        SRanipal_Eye_API.GetEyeParameter(ref eye_parameter);

        eyeData = eye_data;
        // do stuff with eyeData...

        //lastTime = currentTime;
        //currentTime = eyeData.timestamp;

        MeasureTime = DateTime.Now.Ticks;
        time_stamp = eyeData.timestamp;
        frame = eyeData.frame_sequence;
        eye_valid_L = eyeData.verbose_data.left.eye_data_validata_bit_mask;
        eye_valid_R = eyeData.verbose_data.right.eye_data_validata_bit_mask;
        openness_L = eyeData.verbose_data.left.eye_openness;
        openness_R = eyeData.verbose_data.right.eye_openness;
        pupil_diameter_L = eyeData.verbose_data.left.pupil_diameter_mm;
        pupil_diameter_R = eyeData.verbose_data.right.pupil_diameter_mm;
        pos_sensor_L = eyeData.verbose_data.left.pupil_position_in_sensor_area;
        pos_sensor_R = eyeData.verbose_data.right.pupil_position_in_sensor_area;
        gaze_origin_L = eyeData.verbose_data.left.gaze_origin_mm;
        gaze_origin_R = eyeData.verbose_data.right.gaze_origin_mm;
        gaze_direct_L = eyeData.verbose_data.left.gaze_direction_normalized;
        gaze_direct_R = eyeData.verbose_data.right.gaze_direction_normalized;
        gaze_sensitive = eye_parameter.gaze_ray_parameter.sensitive_factor;
        frown_L = eyeData.expression_data.left.eye_frown;
        frown_R = eyeData.expression_data.right.eye_frown;
        squeeze_L = eyeData.expression_data.left.eye_squeeze;
        squeeze_R = eyeData.expression_data.right.eye_squeeze;
        wide_L = eyeData.expression_data.left.eye_wide;
        wide_R = eyeData.expression_data.right.eye_wide;
        distance_valid_C = eyeData.verbose_data.combined.convergence_distance_validity;
        distance_C = eyeData.verbose_data.combined.convergence_distance_mm;
        track_imp_cnt = eyeData.verbose_data.tracking_improvements.count;
        //track_imp_item = eyeData.verbose_data.tracking_improvements.items;

        //  Convert the measured data to string data to write in a text file.
        string value =
            MeasureTime.ToString() + "," +
            time_stamp.ToString() + "," +
            frame.ToString() + "," +
            eye_valid_L.ToString() + "," +
            eye_valid_R.ToString() + "," +
            openness_L.ToString() + "," +
            openness_R.ToString() + "," +
            pupil_diameter_L.ToString() + "," +
            pupil_diameter_R.ToString() + "," +
            pos_sensor_L.x.ToString() + "," +
            pos_sensor_L.y.ToString() + "," +
            pos_sensor_R.x.ToString() + "," +
            pos_sensor_R.y.ToString() + "," +
            gaze_origin_L.x.ToString() + "," +
            gaze_origin_L.y.ToString() + "," +
            gaze_origin_L.z.ToString() + "," +
            gaze_origin_R.x.ToString() + "," +
            gaze_origin_R.y.ToString() + "," +
            gaze_origin_R.z.ToString() + "," +
            gaze_direct_L.x.ToString() + "," +
            gaze_direct_L.y.ToString() + "," +
            gaze_direct_L.z.ToString() + "," +
            gaze_direct_R.x.ToString() + "," +
            gaze_direct_R.y.ToString() + "," +
            gaze_direct_R.z.ToString() + "," +
            gaze_sensitive.ToString() + "," +
            frown_L.ToString() + "," +
            frown_R.ToString() + "," +
            squeeze_L.ToString() + "," +
            squeeze_R.ToString() + "," +
            wide_L.ToString() + "," +
            wide_R.ToString() + "," +
            distance_valid_C.ToString() + "," +
            distance_C.ToString() + "," +
            track_imp_cnt.ToString() +
            //track_imp_item.ToString() +
            Environment.NewLine;

        File.AppendAllText("eyedata_" + "P" + UserID + "_" + "S" + scenario + ".txt", value);


    }
}

I don’t have time to look through code, but I found a reference here of visualizing eye rays.

You may have to modify it to use the data from your logs instead of reading the data from the device in real-time, and then attach it to an arbitrary model to represent the head (instead of the actual camera).

From there you can drop a camera elsewhere in your scene, and screen capture it as it plays through the data.

Hi @Shane_Michael !
I finally managed to get the 3D world space coordinates of Eye Gaze. Thank you so much for your suggestions. I am just a bit confused on how to create the 3D texture (heatmap) of the gaze points I collected. Can you please elaborate on how to achieve this since I am new to Unity… Looking forward to your response! :slight_smile:

For your reference: An example of Eye data I can collect in the 3D world space coordinates can be seen in the below image.

I would calculate the bounding volume for your points, and then create a 3D texture with enough resolution to sufficiently capture the data. Any modern GPU has enough memory and performance that you can likely afford to round up, but 3D textures can really balloon in size. I might start with 128 x 64 x 256 given your data, and bump it up if it looks too blocky. It’s useful to use a single channel texture (i.e. only red) to save memory.

From there it’s fairly straightforward to loop through each point, and for each texel in the 3D texture calculate the distant, apply some falloff function, if you want, and accumulate it in an array. Once you’d added up all the contributions, normalize it (i.e. divide by the max value), and assign the values to each of the texels in the 3D texture.

From there, inside your shader, you use the world position of each fragment to sample the 3D texture at the correct place (i.e. with respect to the bounding volume of your 3D texture) and use that to index into a color gradient lookup table (your classic heat map) to generate the heatmap output. Add it to a lit shader as emission channel or output the value directly in an unlit shader. This is easier if you are using URP/HDRP and can use Shader Graph, but with the built-in pipeline, you’d need to write a surface shader.

Another option is to put all your points in an array that is passed into a shader, and have your shader accumulate the distances at runtime to generate the heatmap. This wouldn’t scale well as you get more and more points, but requires less precomputation and might be less work to set up. Would still require getting comfortable with Shader Graph, though.

Hello
I am working on a project similar to the above using a HTC Vive Pro Eye and SRanipal SDK for a behavioral study.
I am trying the save the eye tracking data to a csv file and be able to visualize the gaze data using a heatmap.

I saw the script that you had shared and the link to the gaze rays. I wanted to ask if you might be willing to share some information on how you resolved the transformation to world space and if you were able to visualize the eye rays.

Any help would be much appreciated!

I would like to ask when you convert from local to global, why left and right eye need to multiply -1 on x value, but the combined eye did not do that. I get this information from the GazeRaySample code, But I don’t think that’s right? I couldn’t understand why the combined eye don’t need -x?