123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 |
- using System;
- using System.Collections.Generic;
- using Controller.Bicycle;
- using Controller.Lean;
- using Study;
- using Sensors;
- using Sensors.ANT;
- using Sensors.Bluetooth;
- using Tracking;
- using System.Linq;
- using UnityEngine;
- // Declare outisde for public visibility
- //public enum SteeringMode { frontWheel, Leaning, HMD };
- namespace Controller
- {
- [Serializable]
- public struct FrontWheelTrackerConfig
- {
- public FrontWheelTracker frontWheelTracker;
- public float multiplicator;
- public float AdjustedRotation => frontWheelTracker.SteerRotation * multiplicator;
- }
- [Serializable]
- public struct HMDTrackerConfig
- {
- public HMDTracker cameraTracker;
- public float multiplicator;
- public float steeringOffset;
- public float AdjustedRotation => cameraTracker.SteerRotation * multiplicator;
- public float AdjustedLean => cameraTracker.LeanRotation * multiplicator;
- }
- [RequireComponent(typeof(IBicycleController))]
- public class SensorBikeController : MonoBehaviour
- {
- public PolarRotationMapping polarRotationMapping;
- public FrontWheelTrackerConfig frontWheelTrackerConfig;
- public HMDTrackerConfig hmdTrackerConfig;
- private float leanFactor;
- public bool steer = true;
- public bool accelerate = true;
- public bool lean = true;
- public Study.SteeringMode steeringSelection;
- private IBicycleController bicycleController;
- private bool isFrontWheelTrackerNotNull;
- private bool isHMDTrackerNotNull;
- private BikeSensorData sensorData;
- private Queue<float> previousLeanValues;
- public int framesToConsider = 10;
- public float tolerance = 5;
- private void Start()
- {
- isFrontWheelTrackerNotNull = frontWheelTrackerConfig.frontWheelTracker != null;
- isHMDTrackerNotNull = hmdTrackerConfig.cameraTracker != null;
- bicycleController = GetComponent<IBicycleController>();
- sensorData = BikeSensorData.Instance;
- //leanFactor = 90f / (polarRotationMapping.maxRight - polarRotationMapping.center);
- leanFactor = 45f / polarRotationMapping.maxRight;
- if (framesToConsider > 1) previousLeanValues = new Queue<float>(framesToConsider);
- // Sort of callibration for polar Sensor
- var polarData = sensorData.BleData;
- if (steeringSelection == Study.SteeringMode.Leaning && polarData != null)
- {
- Debug.Log("Calibration of Polar Sensor");
- polarRotationMapping.center = polarData.Value.Acc.y;
- }
- }
- private void Update()
- {
- Debug.Log("Bike Sensor Controller called");
- var speedData = sensorData.SpeedData;
- if (speedData != null && accelerate) SetSpeed(speedData.Value);
- if (steer) SetSteer();
- if (lean) SetLeaningAngle();
- }
- public void SetCondition(Study.SteeringMode mode)
- {
- Debug.Log("Setting Condition");
- steeringSelection = mode;
- }
- private void SetSteer()
- {
- switch (steeringSelection)
- {
- case Study.SteeringMode.frontWheel:
- if (isFrontWheelTrackerNotNull)
- {
- bicycleController.CurrentSteerAngle = frontWheelTrackerConfig.AdjustedRotation * 2f;
- }
- break;
- case Study.SteeringMode.Leaning:
- var polarData = sensorData.BleData;
- if (polarData != null)
- {
- Debug.Log("Updating Polar Data");
- Debug.Log("Polar Y: " + polarData.Value.Acc);
- var polarLean = CalculateLeanRotationMultiFrame(polarData);
- Debug.Log("Ploar Post-Processed: " + polarLean);
- bicycleController.CurrentSteerAngle = polarLean;
- // Activate below in case we also need the steering angle
- //bicycleController.CurrentSteerAngle = (polarData.Value.Acc.y - polarRotationMapping.center) * leanFactor;
- }
- break;
- case Study.SteeringMode.HMD:
- if (isHMDTrackerNotNull)
- {
- // We emprirically observed a right-drift
- // We subtract a constant to counteract this
- bicycleController.CurrentSteerAngle = hmdTrackerConfig.AdjustedRotation;
- }
- break;
- }
- Debug.Log("Updating Steering Angle to " + bicycleController.CurrentSteerAngle);
- }
- private void SetLeaningAngle()
- {
- //bicycleController.CurrentLeaningAngle =
- }
- private void SetSpeed(SpeedSensorData speedData)
- {
- bicycleController.CurrentSpeed = speedData.Speed;
- }
- // Helpers for Lean Steering
- private float CalculateLeanRotationSingleFrame(BleSensorData? polarData)
- {
- float steerUpdate = polarData.Value.Acc.y - polarRotationMapping.center;
- //steerUpdate = Math.Abs(steerUpdate - polarRotationMapping.center) < tolerance ? 0 : steerUpdate;
- return steerUpdate * leanFactor;
- }
- private float CalculateLeanRotationMultiFrame(BleSensorData? polarData)
- {
- var angleThisFrame = CalculateLeanRotationSingleFrame(polarData);
- if (previousLeanValues.Count > framesToConsider) previousLeanValues.Dequeue();
- previousLeanValues.Enqueue(angleThisFrame);
- // Use exponential moving average
- float alpha = 2f / (framesToConsider + 1);
- float steerUpdate = previousLeanValues.DefaultIfEmpty()
- .Aggregate((ema, nextQuote) => alpha * nextQuote + (1 - alpha) * ema);
- // Do Global Thresholding, to avoid noisy steering
- return Math.Abs(steerUpdate - polarRotationMapping.center) < tolerance ? 0 : steerUpdate;
- //return steerUpdate;
- }
- }
- }
|