/* Copyright (c) 2019, Lars Brubaker, John Lewin All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. The views and conclusions contained in the software and documentation are those of the authors and should not be interpreted as representing official policies, either expressed or implied, of the FreeBSD Project. */ using System; using System.Collections.Generic; using System.IO; using System.Threading; using System.Threading.Tasks; using MatterHackers.Agg; using MatterHackers.Agg.Platform; using MatterHackers.Agg.UI; using MatterHackers.DataConverters3D; using MatterHackers.MatterControl.PartPreviewWindow; using MatterHackers.PolygonMesh; using MatterHackers.PolygonMesh.Processors; using MatterHackers.VectorMath; using org.mariuszgromada.math.mxparser; namespace MatterHackers.MatterControl.DesignTools { [HideChildrenFromTreeView] [HideMeterialAndColor] [WebPageLink("Documentation", "Open", "https://matterhackers.com/support/mattercontrol-variable-support")] [MarkDownDescription("[BETA] - Experimental support for variables and equations with a sheets like interface.")] public class SheetObject3D : Object3D, IObject3DControlsProvider { public SheetData SheetData { get; set; } public static async Task Create() { var item = new SheetObject3D { SheetData = new SheetData(5, 5) }; await item.Rebuild(); return item; } public SheetObject3D() { Mesh border; using (Stream stlStream = StaticData.Instance.OpenStream(Path.Combine("Stls", "sheet_border.stl"))) { border = StlProcessing.Load(stlStream, CancellationToken.None); } this.Children.Add(new Object3D() { Mesh = border, Color = new Color("#9D9D9D") }); Mesh boxes; using (Stream stlStream = StaticData.Instance.OpenStream(Path.Combine("Stls", "sheet_boxes.stl"))) { boxes = StlProcessing.Load(stlStream, CancellationToken.None); } this.Children.Add(new Object3D() { Mesh = boxes, Color = new Color("#117c43") }); var aabb = border.GetAxisAlignedBoundingBox(); this.Matrix *= Matrix4X4.CreateScale(20 / aabb.XSize); } public override bool Persistable => false; public override void OnInvalidate(InvalidateArgs invalidateType) { if (invalidateType.InvalidateType.HasFlag(InvalidateType.SheetUpdated) && invalidateType.Source == this) { using (RebuildLock()) { // update the table info SheetData.Recalculate(); // send a message to all our siblings and their children SendInvalidateToAll(); } } else { base.OnInvalidate(invalidateType); } } private void SendInvalidateToAll() { var updatedItems = new HashSet(); updatedItems.Add(this); foreach (var sibling in this.Parent.Children) { SendInvalidateRecursive(sibling, updatedItems); } } private void SendInvalidateRecursive(IObject3D item, HashSet updatedItems) { if (updatedItems.Contains(item)) { return; } // process depth first foreach(var child in item.Children) { SendInvalidateRecursive(child, updatedItems); } // and send the invalidate RunningInterval runningInterval = null; void RebuildWhenUnlocked() { if (!item.RebuildLocked) { updatedItems.Add(item); UiThread.ClearInterval(runningInterval); item.Invalidate(new InvalidateArgs(item, InvalidateType.SheetUpdated)); } } if (!item.RebuildLocked) { updatedItems.Add(item); item.Invalidate(new InvalidateArgs(item, InvalidateType.SheetUpdated)); } else { // we need to get back to the user requested change when not locked runningInterval = UiThread.SetInterval(RebuildWhenUnlocked, .2); } } public static T EvaluateExpression(IObject3D owner, string inputExpression) { // check if the expression is not an equation (does not start with "=") if (inputExpression.Length > 0 && inputExpression[0] != '=') { if (typeof(T) == typeof(string)) { return (T)(object)inputExpression; } // not an equation so try to parse it directly if (double.TryParse(inputExpression, out var result)) { if (typeof(T) == typeof(double)) { return (T)(object)result; } if (typeof(T) == typeof(int)) { return (T)(object)(int)Math.Round(result); } } else { if (typeof(T) == typeof(double)) { return (T)(object)0.0; } if (typeof(T) == typeof(int)) { return (T)(object)0; } } } if (inputExpression.Length > 0 && inputExpression[0] == '=') { inputExpression = inputExpression.Substring(1); } // look through all the parents foreach (var parent in owner.Parents()) { // then each child of any give parent foreach (var sibling in parent.Children) { // if it is a sheet if (sibling != owner && sibling is SheetObject3D sheet) { // try to manage the cell into the correct data type string value = sheet.SheetData.EvaluateExpression(inputExpression); return CastResult(value, inputExpression); } } } // could not find a sheet, try to evaluate the expression directly var evaluator = new Expression(inputExpression); return CastResult(evaluator.calculate().ToString(), inputExpression); } public static T CastResult(string value, string inputExpression) { if (typeof(T) == typeof(string)) { // if parsing the equation resulted in NaN as the output if (value == "NaN") { // return the actual expression return (T)(object)inputExpression; } // get the value of the cell return (T)(object)value; } if (typeof(T) == typeof(double)) { if (double.TryParse(value, out double doubleValue) && !double.IsNaN(doubleValue) && !double.IsInfinity(doubleValue)) { return (T)(object)doubleValue; } // else return an error return (T)(object).1; } if (typeof(T) == typeof(int)) { if (double.TryParse(value, out double doubleValue) && !double.IsNaN(doubleValue) && !double.IsInfinity(doubleValue)) { return (T)(object)(int)Math.Round(doubleValue); } // else return an error return (T)(object)1; } return (T)(object)default(T); } public void AddObject3DControls(Object3DControlsLayer object3DControlsLayer) { } } }