You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
CrowdControl/Assets/Feel/MMFeedbacks/MMFeedbacks/Feedbacks/MMF_ImageTextureOffset.cs

211 lines
7.9 KiB
C#

using System;
using System.Collections;
using System.Collections.Generic;
using MoreMountains.Tools;
using UnityEngine;
using UnityEngine.UI;
namespace MoreMountains.Feedbacks
{
/// <summary>
/// This feedback will let you control the texture offset of a target UI Image over time
/// </summary>
[AddComponentMenu("")]
[FeedbackHelp("This feedback will let you control the texture offset of a target UI Image over time.")]
[FeedbackPath("UI/Image Texture Offset")]
public class MMF_ImageTextureOffset : MMF_Feedback
{
/// a static bool used to disable all feedbacks of this type at once
public static bool FeedbackTypeAuthorized = true;
/// sets the inspector color for this feedback
#if UNITY_EDITOR
public override Color FeedbackColor { get { return MMFeedbacksInspectorColors.UIColor; } }
public override bool EvaluateRequiresSetup() { return (TargetImage == null); }
public override string RequiredTargetText { get { return TargetImage != null ? TargetImage.name : ""; } }
public override string RequiresSetupText { get { return "This feedback requires that a TargetImage be set to be able to work properly. You can set one below."; } }
#endif
public override bool HasAutomatedTargetAcquisition => true;
protected override void AutomateTargetAcquisition() => TargetImage = FindAutomatedTarget<Image>();
/// the possible modes for this feedback
public enum Modes { OverTime, Instant }
//
public enum MaterialPropertyTypes { Main, TextureID }
[MMFInspectorGroup("Texture Scale", true, 63, true)]
/// the UI Image on which to change texture offset on
[Tooltip("the UI Image on which to change texture offset on")]
public Image TargetImage;
/// whether to target the main texture property, or one specified in MaterialPropertyName
[Tooltip("whether to target the main texture property, or one specified in MaterialPropertyName")]
public MaterialPropertyTypes MaterialPropertyType = MaterialPropertyTypes.Main;
/// the property name, for example _MainTex_ST, or _MainTex if you don't have UseMaterialPropertyBlocks set to true
[Tooltip("the property name, for example _MainTex_ST, or _MainTex if you don't have UseMaterialPropertyBlocks set to true")]
[MMEnumCondition("MaterialPropertyType", (int)MaterialPropertyTypes.TextureID)]
public string MaterialPropertyName = "_MainTex_ST";
/// whether the feedback should affect the material instantly or over a period of time
[Tooltip("whether the feedback should affect the material instantly or over a period of time")]
public Modes Mode = Modes.OverTime;
/// how long the material should change over time
[Tooltip("how long the material should change over time")]
[MMFEnumCondition("Mode", (int)Modes.OverTime)]
public float Duration = 0.2f;
/// whether or not the values should be relative
[Tooltip("whether or not the values should be relative")]
public bool RelativeValues = true;
/// if this is true, calling that feedback will trigger it, even if it's in progress. If it's false, it'll prevent any new Play until the current one is over
[Tooltip("if this is true, calling that feedback will trigger it, even if it's in progress. If it's false, it'll prevent any new Play until the current one is over")]
public bool AllowAdditivePlays = false;
[MMFInspectorGroup("Intensity", true, 65)]
/// the curve to tween the offset on
[Tooltip("the curve to tween the offset on")]
[MMFEnumCondition("Mode", (int)Modes.OverTime)]
public AnimationCurve OffsetCurve = new AnimationCurve(new Keyframe(0, 0), new Keyframe(0.3f, 1f), new Keyframe(1, 0));
/// the value to remap the offset curve's 0 to
[Tooltip("the value to remap the offset curve's 0 to")]
[MMFEnumCondition("Mode", (int)Modes.OverTime)]
public Vector2 RemapZero = Vector2.zero;
/// the value to remap the offset curve's 1 to
[Tooltip("the value to remap the offset curve's 1 to")]
[MMFEnumCondition("Mode", (int)Modes.OverTime)]
public Vector2 RemapOne = Vector2.one;
/// the value to move the intensity to in instant mode
[Tooltip("the value to move the intensity to in instant mode")]
[MMFEnumCondition("Mode", (int)Modes.Instant)]
public Vector2 InstantOffset;
protected Vector2 _initialValue;
protected Coroutine _coroutine;
protected Vector2 _newValue;
protected Material _material;
/// the duration of this feedback is the duration of the transition
public override float FeedbackDuration { get { return (Mode == Modes.Instant) ? 0f : ApplyTimeMultiplier(Duration); } set { Duration = value; } }
public override bool HasRandomness => true;
/// <summary>
/// On init we store our initial texture offset
/// </summary>
/// <param name="owner"></param>
protected override void CustomInitialization(MMF_Player owner)
{
base.CustomInitialization(owner);
_material = TargetImage.materialForRendering;
switch (MaterialPropertyType)
{
case MaterialPropertyTypes.Main:
_initialValue = _material.mainTextureOffset;
break;
case MaterialPropertyTypes.TextureID:
_initialValue = _material.GetTextureOffset(MaterialPropertyName);
break;
}
}
/// <summary>
/// On Play we initiate our offset change
/// </summary>
/// <param name="position"></param>
/// <param name="feedbacksIntensity"></param>
protected override void CustomPlayFeedback(Vector3 position, float feedbacksIntensity = 1.0f)
{
if (!Active || !FeedbackTypeAuthorized)
{
return;
}
float intensityMultiplier = ComputeIntensity(feedbacksIntensity, position);
switch (Mode)
{
case Modes.Instant:
ApplyValue(InstantOffset * intensityMultiplier);
break;
case Modes.OverTime:
if (!AllowAdditivePlays && (_coroutine != null))
{
return;
}
_coroutine = Owner.StartCoroutine(TransitionCo(intensityMultiplier));
break;
}
}
/// <summary>
/// This coroutine will modify the offset value over time
/// </summary>
/// <returns></returns>
protected virtual IEnumerator TransitionCo(float intensityMultiplier)
{
float journey = NormalPlayDirection ? 0f : FeedbackDuration;
IsPlaying = true;
while ((journey >= 0) && (journey <= FeedbackDuration) && (FeedbackDuration > 0))
{
float remappedTime = MMFeedbacksHelpers.Remap(journey, 0f, FeedbackDuration, 0f, 1f);
SetMaterialValues(remappedTime, intensityMultiplier);
journey += NormalPlayDirection ? FeedbackDeltaTime : -FeedbackDeltaTime;
yield return null;
}
SetMaterialValues(FinalNormalizedTime, intensityMultiplier);
IsPlaying = false;
_coroutine = null;
yield return null;
}
/// <summary>
/// Applies offset to the target material
/// </summary>
/// <param name="time"></param>
protected virtual void SetMaterialValues(float time, float intensityMultiplier)
{
_newValue.x = MMFeedbacksHelpers.Remap(OffsetCurve.Evaluate(time), 0f, 1f, RemapZero.x, RemapOne.x);
_newValue.y = MMFeedbacksHelpers.Remap(OffsetCurve.Evaluate(time), 0f, 1f, RemapZero.y, RemapOne.y);
if (RelativeValues)
{
_newValue += _initialValue;
}
ApplyValue(_newValue * intensityMultiplier);
}
/// <summary>
/// Applies the specified value to the material
/// </summary>
/// <param name="newValue"></param>
protected virtual void ApplyValue(Vector2 newValue)
{
switch (MaterialPropertyType)
{
case MaterialPropertyTypes.Main:
_material.mainTextureOffset = newValue;
break;
case MaterialPropertyTypes.TextureID:
_material.SetTextureOffset(MaterialPropertyName, newValue);
break;
}
}
/// <summary>
/// Stops this feedback
/// </summary>
/// <param name="position"></param>
/// <param name="feedbacksIntensity"></param>
protected override void CustomStopFeedback(Vector3 position, float feedbacksIntensity = 1)
{
if (!Active || !FeedbackTypeAuthorized || (_coroutine == null))
{
return;
}
base.CustomStopFeedback(position, feedbacksIntensity);
IsPlaying = false;
Owner.StopCoroutine(_coroutine);
_coroutine = null;
}
}
}