323 lines
13 KiB
C#
323 lines
13 KiB
C#
// Copyright (c) Microsoft Corporation.
|
|
// Licensed under the MIT License.
|
|
|
|
using Microsoft.MixedReality.Toolkit.Input;
|
|
using System.Collections.Generic;
|
|
using System.Runtime.CompilerServices;
|
|
using UnityEngine;
|
|
|
|
namespace Microsoft.MixedReality.Toolkit.UI.BoundsControl
|
|
{
|
|
/// <summary>
|
|
/// ProximityEffect scales and switches out materials for registered objects whenever a pointer is in proximity.
|
|
/// Scaling is done on three different stages: far / medium and close proximity whereas material switching
|
|
/// will only be done on close proximity.
|
|
/// </summary>
|
|
|
|
public class ProximityEffect
|
|
{
|
|
private ProximityEffectConfiguration config;
|
|
internal ProximityEffect(ProximityEffectConfiguration configuration)
|
|
{
|
|
Debug.Assert(configuration != null, "Can't create ProximityEffect without valid configuration");
|
|
config = configuration;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Internal state tracking for proximity of a object
|
|
/// </summary>
|
|
internal enum ProximityState
|
|
{
|
|
FullsizeNoProximity = 0,
|
|
MediumProximity,
|
|
CloseProximity
|
|
}
|
|
|
|
/// <summary>
|
|
/// Container for object references and states
|
|
/// </summary>
|
|
private class ObjectProximityInfo
|
|
{
|
|
public Transform ScaledObject;
|
|
public Renderer ObjectVisualRenderer;
|
|
public ProximityState ProximityState = ProximityState.FullsizeNoProximity;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Dictionary that maps proximity object provider to list of objects that have proximity scaling applied
|
|
/// </summary>
|
|
|
|
private Dictionary<IProximityEffectObjectProvider, List<ObjectProximityInfo>> registeredObjects = new Dictionary<IProximityEffectObjectProvider, List<ObjectProximityInfo>>();
|
|
|
|
|
|
private HashSet<IMixedRealityPointer> proximityPointers = new HashSet<IMixedRealityPointer>();
|
|
private List<Vector3> proximityPoints = new List<Vector3>();
|
|
|
|
#region public methods
|
|
/// <summary>
|
|
/// register objects for proximity effect via a <see cref="IProximityEffectObjectProvider"/>
|
|
/// </summary>
|
|
public void RegisterObjectProvider(IProximityEffectObjectProvider provider)
|
|
{
|
|
registeredObjects.Add(provider, FetchObjects(provider));
|
|
// subscribe to object changes
|
|
provider.ProximityObjectsChanged.AddListener(RefreshObjects);
|
|
}
|
|
|
|
private List<ObjectProximityInfo> FetchObjects(IProximityEffectObjectProvider provider)
|
|
{
|
|
var registeredObjects = new List<ObjectProximityInfo>();
|
|
provider.ForEachProximityObject(proximityObject =>
|
|
{
|
|
registeredObjects.Add(new ObjectProximityInfo()
|
|
{
|
|
ScaledObject = proximityObject,
|
|
ObjectVisualRenderer = proximityObject.gameObject.GetComponentInChildren<Renderer>()
|
|
});
|
|
});
|
|
|
|
return registeredObjects;
|
|
}
|
|
|
|
private void RefreshObjects(IProximityEffectObjectProvider provider)
|
|
{
|
|
// refetch object list
|
|
registeredObjects[provider] = FetchObjects(provider);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Clears all registered objects in the proximity effect
|
|
/// </summary>
|
|
public void ClearObjects()
|
|
{
|
|
if (registeredObjects != null)
|
|
{
|
|
foreach (var keyValuePair in registeredObjects)
|
|
{
|
|
keyValuePair.Key.ProximityObjectsChanged.RemoveListener(RefreshObjects);
|
|
}
|
|
registeredObjects.Clear();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Resets all objects that had a proximity effect applied. This will reset them to their default size and reset to the base material
|
|
/// </summary>
|
|
public void ResetProximityScale()
|
|
{
|
|
if (config.ProximityEffectActive == false)
|
|
{
|
|
return;
|
|
}
|
|
|
|
foreach (var keyValuePair in registeredObjects)
|
|
{
|
|
foreach (var item in keyValuePair.Value)
|
|
{
|
|
if (item.ProximityState != ProximityState.FullsizeNoProximity)
|
|
{
|
|
item.ProximityState = ProximityState.FullsizeNoProximity;
|
|
|
|
if (item.ObjectVisualRenderer)
|
|
{
|
|
item.ObjectVisualRenderer.material = keyValuePair.Key.GetBaseMaterial();
|
|
}
|
|
|
|
ScaleObject(item.ProximityState, item.ScaledObject, keyValuePair.Key.GetObjectSize());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Updates proximity effect and its registered objects.
|
|
/// Highlights and scales objects in proximity according to the pointer distance
|
|
/// </summary>
|
|
/// <param name="boundsCenter">gameobject position the proximity effect is attached to</param>
|
|
/// <param name="boundsExtents">extents of the gameobject the proximity effect is attached to</param>
|
|
public void UpdateScaling(Vector3 boundsCenter, Vector3 boundsExtents)
|
|
{
|
|
// early out if effect is disabled
|
|
if (config.ProximityEffectActive == false || !IsAnyRegisteredObjectVisible())
|
|
{
|
|
return;
|
|
}
|
|
|
|
proximityPointers.Clear();
|
|
proximityPoints.Clear();
|
|
|
|
HashSet<IMixedRealityInputSource> inputSources = CoreServices.InputSystem?.DetectedInputSources;
|
|
|
|
if (inputSources == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Find all valid pointers
|
|
foreach (IMixedRealityInputSource inputSource in inputSources)
|
|
{
|
|
foreach (IMixedRealityPointer pointer in inputSource.Pointers)
|
|
{
|
|
// don't use IsInteractionEnabled for near pointers as the pointers might have a different radius when deciding
|
|
// if they can interact with a near-by object - we might still want to show proximity scaling even if
|
|
// eg. grab pointer decides it's too far away to actually perform the interaction
|
|
if (pointer.IsActive && !proximityPointers.Contains(pointer))
|
|
{
|
|
proximityPointers.Add(pointer);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Get the max radius possible of our current bounds and extent the range to include proximity scaled objects. This is done by adjusting the original bounds to include the ObjectMediumProximity range in x, y and z axis
|
|
float squareMaxLength = boundsExtents.sqrMagnitude + (3 * config.ObjectMediumProximity * config.ObjectMediumProximity);
|
|
|
|
// Grab points within sphere of influence from valid pointers
|
|
foreach (var pointer in proximityPointers)
|
|
{
|
|
if (IsPointWithinBounds(boundsCenter, pointer.Position, squareMaxLength))
|
|
{
|
|
proximityPoints.Add(pointer.Position);
|
|
}
|
|
|
|
if (pointer.Result?.CurrentPointerTarget != null)
|
|
{
|
|
Vector3? point = pointer.Result?.Details.Point;
|
|
if (point.HasValue && IsPointWithinBounds(boundsCenter, pointer.Result.Details.Point, squareMaxLength))
|
|
{
|
|
proximityPoints.Add(pointer.Result.Details.Point);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Loop through all objects and find closest one
|
|
Transform closestObject = null;
|
|
float closestDistanceSqr = float.MaxValue;
|
|
foreach (var point in proximityPoints)
|
|
{
|
|
foreach (var keyValuePair in registeredObjects)
|
|
{
|
|
foreach (var item in keyValuePair.Value)
|
|
{
|
|
// If object can't be visible, skip calculations
|
|
if (!keyValuePair.Key.IsActive)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
// Perform comparison on sqr distance since sqrt() operation is expensive in Vector3.Distance()
|
|
float sqrDistance = (item.ScaledObject.transform.position - point).sqrMagnitude;
|
|
if (sqrDistance < closestDistanceSqr)
|
|
{
|
|
closestObject = item.ScaledObject;
|
|
closestDistanceSqr = sqrDistance;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Loop through all objects and update visual state based on closest point
|
|
foreach (var keyValuePair in registeredObjects)
|
|
{
|
|
foreach (var item in keyValuePair.Value)
|
|
{
|
|
ProximityState newState = (closestObject == item.ScaledObject) ? GetProximityState(closestDistanceSqr) : ProximityState.FullsizeNoProximity;
|
|
IProximityEffectObjectProvider provider = keyValuePair.Key;
|
|
|
|
// Only apply updates if object is in a new state or closest object needs to lerp scaling
|
|
if (item.ProximityState != newState)
|
|
{
|
|
// Update and save new state
|
|
item.ProximityState = newState;
|
|
|
|
if (item.ObjectVisualRenderer)
|
|
{
|
|
item.ObjectVisualRenderer.material = newState == ProximityState.CloseProximity ? provider.GetHighlightedMaterial() : provider.GetBaseMaterial();
|
|
}
|
|
}
|
|
|
|
ScaleObject(newState, item.ScaledObject, provider.GetObjectSize(), true);
|
|
}
|
|
}
|
|
}
|
|
|
|
#endregion public methods
|
|
|
|
#region private methods
|
|
|
|
private bool IsAnyRegisteredObjectVisible()
|
|
{
|
|
foreach (var keyValuePair in registeredObjects)
|
|
{
|
|
if (keyValuePair.Key.IsActive)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private void ScaleObject(ProximityState state, Transform scaleVisual, float objectSize, bool lerp = false)
|
|
{
|
|
float targetScale = 1.0f, weight = 0.0f;
|
|
|
|
switch (state)
|
|
{
|
|
case ProximityState.FullsizeNoProximity:
|
|
targetScale = config.FarScale;
|
|
weight = lerp ? config.FarGrowRate : 1.0f;
|
|
break;
|
|
case ProximityState.MediumProximity:
|
|
targetScale = config.MediumScale;
|
|
weight = lerp ? config.MediumGrowRate : 1.0f;
|
|
break;
|
|
case ProximityState.CloseProximity:
|
|
targetScale = config.CloseScale;
|
|
weight = lerp ? config.CloseGrowRate : 1.0f;
|
|
break;
|
|
}
|
|
|
|
float maxScaleAxis = VisualUtils.GetMaxComponent(scaleVisual.localScale);
|
|
float newLocalScale = (maxScaleAxis * (1.0f - weight)) + (objectSize * targetScale * weight);
|
|
scaleVisual.localScale = new Vector3(newLocalScale, newLocalScale, newLocalScale);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Determine if passed point is within sphere of radius around this GameObject
|
|
/// To avoid function overhead, request compiler to inline this function since repeatedly called every Update() for every pointer position and result
|
|
/// </summary>
|
|
/// <param name="point">world space position</param>
|
|
/// <param name="radiusSqr">radius of sphere in distance squared for faster comparison</param>
|
|
/// <returns>true if point is within sphere</returns>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
private bool IsPointWithinBounds(Vector3 position, Vector3 point, float radiusSqr)
|
|
{
|
|
return (position - point).sqrMagnitude <= radiusSqr;
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Get the ProximityState value based on the distanced provided
|
|
/// </summary>
|
|
/// <param name="sqrDistance">distance squared in proximity in meters</param>
|
|
/// <returns>ProximityState for given distance</returns>
|
|
private ProximityState GetProximityState(float sqrDistance)
|
|
{
|
|
if (sqrDistance <= config.ObjectCloseProximity * config.ObjectCloseProximity)
|
|
{
|
|
return ProximityState.CloseProximity;
|
|
}
|
|
else if (sqrDistance <= config.ObjectMediumProximity * config.ObjectMediumProximity)
|
|
{
|
|
return ProximityState.MediumProximity;
|
|
}
|
|
else
|
|
{
|
|
return ProximityState.FullsizeNoProximity;
|
|
}
|
|
}
|
|
|
|
#endregion private methods
|
|
}
|
|
}
|