2019-08-17 04:20:14 +00:00
|
|
|
using System;
|
|
|
|
using UnityEngine;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
|
|
namespace Restock
|
|
|
|
{
|
|
|
|
public class ModuleRestockHeatEffects : PartModule
|
|
|
|
{
|
2019-08-17 04:26:01 +00:00
|
|
|
// enable the heat glow emissive
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public bool enableHeatEmissive = false;
|
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// what shader property to modify. must be a color.
|
|
|
|
[KSPField] public string shaderProperty = "_EmissiveColor";
|
2019-08-17 04:20:14 +00:00
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// animation curve for the red channel
|
2019-12-03 20:41:33 +00:00
|
|
|
[KSPField] public FloatCurve redCurve = new FloatCurve();
|
2019-08-17 04:20:14 +00:00
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// animation curve for the green channel
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public FloatCurve greenCurve = new FloatCurve();
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// animation curve for the blue channel
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public FloatCurve blueCurve = new FloatCurve();
|
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// animation curve for the alpha channel
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public FloatCurve alphaCurve = new FloatCurve();
|
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// draper point, the temperature in Kelvin where materials start glowing
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public double draperPoint = 798.0;
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// temperature where the animation is at its maximum
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public double lerpMax = double.NaN;
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// temperature where the animation is at its minimum, added with draperPoint
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public double lerpMin = 0.0;
|
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// use the part's core temperature? (overrides useSkinTemp)
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public bool useCoreTemp = false;
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// use the part's skin temperature?
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public bool useSkinTemp = false;
|
|
|
|
|
2019-08-17 04:26:01 +00:00
|
|
|
// should the module disable the stock blackbody glow effect on the included renderers?
|
2019-08-17 04:20:14 +00:00
|
|
|
[KSPField] public bool disableBlackbody = false;
|
2019-12-03 20:41:33 +00:00
|
|
|
|
|
|
|
public List<Renderer> renderers = new List<Renderer>();
|
2024-08-15 06:43:45 +00:00
|
|
|
public List<string> excludedRendererNames = new List<string>();
|
2019-08-17 04:20:14 +00:00
|
|
|
|
|
|
|
private readonly string _shaderBlackbody = "_TemperatureColor";
|
|
|
|
|
|
|
|
private ModuleCoreHeat _coreHeatModule = null;
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2019-08-17 04:20:14 +00:00
|
|
|
private int _shaderPropertyID;
|
|
|
|
|
|
|
|
private int _shaderBlackbodyID;
|
|
|
|
|
|
|
|
private double _lerpRange;
|
|
|
|
|
2019-12-03 20:41:33 +00:00
|
|
|
private Color _emissiveColor;
|
|
|
|
private MaterialPropertyBlock _propertyBlock;
|
2019-08-17 04:20:14 +00:00
|
|
|
|
|
|
|
public void Start()
|
|
|
|
{
|
|
|
|
if (base.vessel == null) return;
|
2019-12-03 20:41:33 +00:00
|
|
|
|
|
|
|
_emissiveColor = new Color();
|
|
|
|
_propertyBlock = new MaterialPropertyBlock();
|
|
|
|
|
2019-08-17 04:20:14 +00:00
|
|
|
if (enableHeatEmissive)
|
|
|
|
{
|
|
|
|
if (useCoreTemp)
|
|
|
|
{
|
|
|
|
_coreHeatModule = base.part.FindModuleImplementing<ModuleCoreHeat>();
|
|
|
|
if (_coreHeatModule == null)
|
|
|
|
{
|
|
|
|
this.LogError("Part has no Core Heat module, skipping");
|
|
|
|
useCoreTemp = false;
|
|
|
|
}
|
|
|
|
}
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2019-08-17 04:20:14 +00:00
|
|
|
if (double.IsNaN(lerpMax))
|
|
|
|
{
|
|
|
|
if (useCoreTemp)
|
|
|
|
{
|
|
|
|
lerpMax = _coreHeatModule.CoreShutdownTemp;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
lerpMax = useSkinTemp ? part.skinMaxTemp : part.maxTemp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_lerpRange = lerpMax - lerpMin - draperPoint;
|
|
|
|
|
|
|
|
_shaderPropertyID = Shader.PropertyToID(shaderProperty);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (disableBlackbody)
|
|
|
|
{
|
|
|
|
_shaderBlackbodyID = Shader.PropertyToID(_shaderBlackbody);
|
|
|
|
}
|
|
|
|
}
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2019-08-17 04:20:14 +00:00
|
|
|
public override void OnLoad(ConfigNode node)
|
|
|
|
{
|
|
|
|
if (HighLogic.LoadedSceneIsEditor || HighLogic.LoadedSceneIsFlight) return;
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2019-08-17 04:20:14 +00:00
|
|
|
if (node.HasValue("excludedRenderer"))
|
|
|
|
{
|
2024-08-15 06:43:45 +00:00
|
|
|
excludedRendererNames = new List<string>(node.GetValues("excludedRenderer"));
|
2019-08-17 04:20:14 +00:00
|
|
|
}
|
2024-08-15 06:43:45 +00:00
|
|
|
|
|
|
|
FindRenderers();
|
2019-08-17 04:20:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void LateUpdate()
|
|
|
|
{
|
|
|
|
if (!HighLogic.LoadedSceneIsFlight) return;
|
2019-12-03 20:41:33 +00:00
|
|
|
if (renderers == null) return;
|
2019-11-05 00:55:36 +00:00
|
|
|
//when switching to the flight scene LateUpdate gets called AFTER OnLoad for some reason
|
|
|
|
// so renderers should hopefully only be null for one frame
|
2019-08-17 04:20:14 +00:00
|
|
|
|
|
|
|
if (enableHeatEmissive)
|
|
|
|
{
|
|
|
|
var temp = 0.0;
|
|
|
|
if (useCoreTemp)
|
|
|
|
{
|
|
|
|
temp = _coreHeatModule.CoreTemperature;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
temp = useSkinTemp ? base.part.skinTemperature : base.part.temperature;
|
|
|
|
}
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2024-08-15 06:43:45 +00:00
|
|
|
var temp2 = (float)((temp - draperPoint) / _lerpRange);
|
2019-08-17 04:20:14 +00:00
|
|
|
temp2 = Mathf.Clamp01(temp2);
|
2019-12-03 20:41:33 +00:00
|
|
|
|
2019-08-17 04:20:14 +00:00
|
|
|
_emissiveColor.r = redCurve.Evaluate(temp2);
|
|
|
|
_emissiveColor.g = greenCurve.Evaluate(temp2);
|
|
|
|
_emissiveColor.b = blueCurve.Evaluate(temp2);
|
|
|
|
_emissiveColor.a = alphaCurve.Evaluate(temp2);
|
|
|
|
|
|
|
|
_propertyBlock.SetColor(_shaderPropertyID, _emissiveColor);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (disableBlackbody)
|
|
|
|
{
|
|
|
|
_propertyBlock.SetColor(_shaderBlackbodyID, Color.black);
|
|
|
|
}
|
2019-08-31 18:28:37 +00:00
|
|
|
|
2024-08-15 06:43:45 +00:00
|
|
|
try
|
|
|
|
{
|
|
|
|
UpdateRenderers(_propertyBlock);
|
|
|
|
} catch(NullReferenceException)
|
|
|
|
{
|
|
|
|
// if any renderers are null, rebuild renderer list
|
|
|
|
// any bonus renderers will just have to be ignored I guess
|
|
|
|
FindRenderers();
|
|
|
|
UpdateRenderers(_propertyBlock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void UpdateRenderers(MaterialPropertyBlock mpb)
|
|
|
|
{
|
2019-08-31 18:28:37 +00:00
|
|
|
for (var i = 0; i < renderers.Count; i++)
|
2019-08-17 04:20:14 +00:00
|
|
|
{
|
2024-08-15 06:43:45 +00:00
|
|
|
renderers[i].SetPropertyBlock(mpb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void FindRenderers()
|
|
|
|
{
|
|
|
|
renderers= part.FindModelComponents<Renderer>();
|
|
|
|
|
|
|
|
renderers.RemoveAll(renderer => renderer == null);
|
|
|
|
|
|
|
|
if( excludedRendererNames.Count != 0)
|
|
|
|
{
|
|
|
|
renderers.RemoveAll(renderer => excludedRendererNames.Contains(renderer.name));
|
2019-08-17 04:20:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|