Add naive implementation of multi-projection

This commit is contained in:
Andrew Cassidy 2020-10-05 22:42:01 -07:00
parent ef90f73288
commit 596675ad8d
5 changed files with 64 additions and 46 deletions

View File

@ -79,7 +79,7 @@ namespace ConformalDecals {
UI_FloatRange()] UI_FloatRange()]
public float wear = 100; public float wear = 100;
[KSPField(isPersistant = true)] public bool projectMultiple; // reserved for future features. do not modify [KSPField(isPersistant = true)] public bool projectMultiple = true;
[KSPField] public MaterialPropertyCollection materialProperties; [KSPField] public MaterialPropertyCollection materialProperties;
@ -215,13 +215,13 @@ namespace ConformalDecals {
foreach (var keyword in _decalMaterial.shaderKeywords) { foreach (var keyword in _decalMaterial.shaderKeywords) {
this.Log($"keyword: {keyword}"); this.Log($"keyword: {keyword}");
} }
if (HighLogic.LoadedSceneIsEditor) { if (HighLogic.LoadedSceneIsEditor) {
UpdateTweakables(); UpdateTweakables();
} }
if (HighLogic.LoadedSceneIsGame) { if (HighLogic.LoadedSceneIsGame) {
UpdateScale(); UpdateProjection();
} }
else { else {
scale = defaultScale; scale = defaultScale;
@ -237,7 +237,7 @@ namespace ConformalDecals {
/// <inheritdoc /> /// <inheritdoc />
public override void OnIconCreate() { public override void OnIconCreate() {
UpdateScale(); UpdateProjection();
} }
/// <inheritdoc /> /// <inheritdoc />
@ -304,11 +304,11 @@ namespace ConformalDecals {
protected void OnSizeTweakEvent(BaseField field, object obj) { protected void OnSizeTweakEvent(BaseField field, object obj) {
// scale or depth values have been changed, so update scale // scale or depth values have been changed, so update scale
// and update projection matrices if attached // and update projection matrices if attached
UpdateScale(); UpdateProjection();
foreach (var counterpart in part.symmetryCounterparts) { foreach (var counterpart in part.symmetryCounterparts) {
var decal = counterpart.GetComponent<ModuleConformalDecal>(); var decal = counterpart.GetComponent<ModuleConformalDecal>();
decal.UpdateScale(); decal.UpdateProjection();
} }
} }
@ -331,7 +331,7 @@ namespace ConformalDecals {
protected void OnVariantApplied(Part eventPart, PartVariant variant) { protected void OnVariantApplied(Part eventPart, PartVariant variant) {
if (_isAttached && eventPart == part.parent) { if (_isAttached && eventPart == part.parent) {
UpdateTargets(); UpdateProjection();
} }
} }
@ -346,7 +346,7 @@ namespace ConformalDecals {
break; break;
case ConstructionEventType.PartOffsetting: case ConstructionEventType.PartOffsetting:
case ConstructionEventType.PartRotating: case ConstructionEventType.PartRotating:
UpdateScale(); UpdateProjection();
break; break;
} }
} }
@ -377,8 +377,7 @@ namespace ConformalDecals {
Camera.onPreCull += Render; Camera.onPreCull += Render;
UpdateMaterials(); UpdateMaterials();
UpdateTargets(); UpdateProjection();
UpdateScale();
} }
protected virtual void OnDetach() { protected virtual void OnDetach() {
@ -394,10 +393,57 @@ namespace ConformalDecals {
Camera.onPreCull -= Render; Camera.onPreCull -= Render;
UpdateMaterials(); UpdateMaterials();
UpdateScale(); UpdateProjection();
} }
protected void UpdateScale() { protected void UpdateProjection() {
// Update projection targets
if (_targets == null) {
_targets = new List<ProjectionTarget>();
}
else {
_targets.Clear();
}
if (_isAttached) {
IEnumerable<Part> targetParts;
if (projectMultiple) {
if (HighLogic.LoadedSceneIsFlight) {
targetParts = part.vessel.parts;
}
else {
targetParts = EditorLogic.fetch.ship.parts;
}
}
else {
targetParts = new[] {part.parent};
}
foreach (var targetPart in targetParts) {
if (targetPart.GetComponent<ModuleConformalDecal>() != null) continue; // skip other decals
foreach (var targetRenderer in targetPart.FindModelComponents<MeshRenderer>()) {
// skip disabled renderers
if (targetRenderer.gameObject.activeInHierarchy == false) continue;
// skip blacklisted shaders
if (DecalConfig.IsBlacklisted(targetRenderer.material.shader)) continue;
var meshFilter = targetRenderer.GetComponent<MeshFilter>();
if (meshFilter == null) continue; // object has a meshRenderer with no filter, invalid
var mesh = meshFilter.sharedMesh;
if (mesh == null) continue; // object has a null mesh, invalid
// create new ProjectionTarget to represent the renderer
var target = new ProjectionTarget(targetPart, targetRenderer, mesh);
// add the target to the list
_targets.Add(target);
}
}
}
// Update projection matrix
scale = Mathf.Max(0.01f, scale); scale = Mathf.Max(0.01f, scale);
depth = Mathf.Max(0.01f, depth); depth = Mathf.Max(0.01f, depth);
var aspectRatio = materialProperties.AspectRatio; var aspectRatio = materialProperties.AspectRatio;
@ -468,36 +514,6 @@ namespace ConformalDecals {
if (!_isAttached) decalFrontTransform.GetComponent<MeshRenderer>().material = _previewMaterial; if (!_isAttached) decalFrontTransform.GetComponent<MeshRenderer>().material = _previewMaterial;
} }
protected void UpdateTargets() {
if (_targets == null) {
_targets = new List<ProjectionTarget>();
}
else {
_targets.Clear();
}
// find all valid renderers
var renderers = part.parent.FindModelComponents<MeshRenderer>();
foreach (var renderer in renderers) {
// skip disabled renderers
if (renderer.gameObject.activeInHierarchy == false) continue;
// skip blacklisted shaders
if (DecalConfig.IsBlacklisted(renderer.material.shader)) continue;
var meshFilter = renderer.GetComponent<MeshFilter>();
if (meshFilter == null) continue; // object has a meshRenderer with no filter, invalid
var mesh = meshFilter.mesh;
if (mesh == null) continue; // object has a null mesh, invalid
// create new ProjectionTarget to represent the renderer
var target = new ProjectionTarget(renderer, mesh);
// add the target to the list
_targets.Add(target);
}
}
protected virtual void UpdateTweakables() { protected virtual void UpdateTweakables() {
// setup tweakable fields // setup tweakable fields
var scaleField = Fields[nameof(scale)]; var scaleField = Fields[nameof(scale)];

View File

@ -95,7 +95,7 @@ namespace ConformalDecals {
materialProperties.AddOrGetTextureProperty("_Decal", true).TextureUrl = newFlagUrl; materialProperties.AddOrGetTextureProperty("_Decal", true).TextureUrl = newFlagUrl;
UpdateMaterials(); UpdateMaterials();
UpdateScale(); UpdateProjection();
} }
private void SetFlagSymmetryCounterparts(string newFlagUrl) { private void SetFlagSymmetryCounterparts(string newFlagUrl) {

View File

@ -271,7 +271,7 @@ namespace ConformalDecals {
_decalTextureProperty.SetTile(output.Window); _decalTextureProperty.SetTile(output.Window);
UpdateMaterials(); UpdateMaterials();
UpdateScale(); UpdateProjection();
} }
protected override void UpdateMaterials() { protected override void UpdateMaterials() {

View File

@ -5,6 +5,7 @@ namespace ConformalDecals {
public class ProjectionTarget { public class ProjectionTarget {
// Target object data // Target object data
public readonly Transform target; public readonly Transform target;
public readonly Part targetPart;
private readonly Renderer _targetRenderer; private readonly Renderer _targetRenderer;
private readonly Mesh _targetMesh; private readonly Mesh _targetMesh;
@ -13,8 +14,9 @@ namespace ConformalDecals {
// property block // property block
private readonly MaterialPropertyBlock _decalMPB; private readonly MaterialPropertyBlock _decalMPB;
public ProjectionTarget(MeshRenderer targetRenderer, Mesh targetMesh) { public ProjectionTarget(Part targetPart, MeshRenderer targetRenderer, Mesh targetMesh) {
target = targetRenderer.transform; this.targetPart = targetPart;
this.target = targetRenderer.transform;
_targetRenderer = targetRenderer; _targetRenderer = targetRenderer;
_targetMesh = targetMesh; _targetMesh = targetMesh;
_decalMPB = new MaterialPropertyBlock(); _decalMPB = new MaterialPropertyBlock();