Working on new 2D primitives and operations

This commit is contained in:
LarsBrubaker 2021-07-07 21:07:30 -07:00
parent 2bb944ba6f
commit 818a7fb8d7
13 changed files with 576 additions and 52 deletions

View file

@ -711,7 +711,7 @@ namespace MatterHackers.MatterControl
};
}
private static bool BooleanCandidate(IObject3D selectedItem, bool includePaths = true)
private static bool BooleanCandidate(IObject3D selectedItem, bool includePaths)
{
if (selectedItem != null)
{
@ -722,7 +722,6 @@ namespace MatterHackers.MatterControl
return true;
}
includePaths = false;
// path items
if (includePaths
&& selectedItem.VisiblePaths().Count() > 1
@ -790,6 +789,7 @@ namespace MatterHackers.MatterControl
PinchOperation(),
TwistOperation(),
PlaneCutOperation(),
FindSliceOperation(),
HollowOutOperation(),
}
},
@ -931,7 +931,7 @@ namespace MatterHackers.MatterControl
{
if (sceneContext.Scene.SelectedItem.VisiblePaths().Count() > 1)
{
new MergePathObject3D("Combine".Localize(), true).WrapSelectedItemAndSelect(sceneContext.Scene);
new MergePathObject3D("Combine".Localize(), ClipperLib.ClipType.ctUnion).WrapSelectedItemAndSelect(sceneContext.Scene);
}
else
{
@ -940,7 +940,7 @@ namespace MatterHackers.MatterControl
},
Icon = (theme) => StaticData.Instance.LoadIcon("combine.png", 16, 16).SetToColor(theme.TextColor).SetPreMultiply(),
HelpTextResolver = () => "*At least 2 parts must be selected*".Localize(),
IsEnabled = (sceneContext) => BooleanCandidate(sceneContext.Scene.SelectedItem),
IsEnabled = (sceneContext) => BooleanCandidate(sceneContext.Scene.SelectedItem, true),
};
}
@ -1133,7 +1133,7 @@ namespace MatterHackers.MatterControl
{
if (sceneContext.Scene.SelectedItem.VisiblePaths().Count() > 1)
{
new MergePathObject3D("Intersect".Localize(), false).WrapSelectedItemAndSelect(sceneContext.Scene);
new MergePathObject3D("Intersect".Localize(), ClipperLib.ClipType.ctIntersection).WrapSelectedItemAndSelect(sceneContext.Scene);
}
else
{
@ -1142,7 +1142,7 @@ namespace MatterHackers.MatterControl
},
Icon = (theme) => StaticData.Instance.LoadIcon("intersect.png", 16, 16),
HelpTextResolver = () => "*At least 2 parts must be selected*".Localize(),
IsEnabled = (sceneContext) => BooleanCandidate(sceneContext.Scene.SelectedItem),
IsEnabled = (sceneContext) => BooleanCandidate(sceneContext.Scene.SelectedItem, true),
};
}
@ -1243,6 +1243,24 @@ namespace MatterHackers.MatterControl
};
}
private static SceneOperation FindSliceOperation()
{
return new SceneOperation("Find Slice")
{
OperationType = typeof(IObject3D),
ResultType = typeof(PlaneCutObject3D),
TitleResolver = () => "Find Slice".Localize(),
Action = (sceneContext) =>
{
var cut = new FindSliceObject3D();
cut.WrapSelectedItemAndSelect(sceneContext.Scene);
},
Icon = (theme) => StaticData.Instance.LoadIcon("plane_cut.png", 16, 16).SetToColor(theme.TextColor),
HelpTextResolver = () => "*At least 1 part must be selected*".Localize(),
IsEnabled = (sceneContext) => IsMeshObject(sceneContext.Scene.SelectedItem),
};
}
private static SceneOperation RadialArrayOperation()
{
return new SceneOperation("Radial Array")
@ -1369,7 +1387,7 @@ namespace MatterHackers.MatterControl
},
Icon = (theme) => StaticData.Instance.LoadIcon("subtract.png", 16, 16).SetPreMultiply(),
HelpTextResolver = () => "*At least 2 parts must be selected*".Localize(),
IsEnabled = (sceneContext) => BooleanCandidate(sceneContext.Scene.SelectedItem),
IsEnabled = (sceneContext) => BooleanCandidate(sceneContext.Scene.SelectedItem, true),
};
}

View file

@ -77,8 +77,9 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
var distance = Distance.Value(this);
var count = Count.Value(this);
// add in all the array items
for (int i = 0; i < Math.Max(count, 1); i++)
// add in all the array items
for (int i = 0; i < Math.Max(count, 1); i++)
{
var next = arrayItem.Clone();
next.Matrix = arrayItem.Matrix * Matrix4X4.CreateTranslation(Direction.Normal.GetNormal() * distance * i);

View file

@ -0,0 +1,198 @@
/*
Copyright (c) 2018, 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.Collections.Generic;
using System.Threading.Tasks;
using MatterHackers.Agg.UI;
using MatterHackers.Agg.VertexSource;
using MatterHackers.DataConverters3D;
using MatterHackers.Localizations;
using MatterHackers.MatterControl.DesignTools.Operations;
using MatterHackers.PolygonMesh;
using MatterHackers.PolygonMesh.Csg;
using MatterHackers.VectorMath;
using Polygons = System.Collections.Generic.List<System.Collections.Generic.List<ClipperLib.IntPoint>>;
namespace MatterHackers.MatterControl.DesignTools
{
public class FindSliceObject3D : OperationSourceContainerObject3D, IPathObject, IPropertyGridModifier
{
public FindSliceObject3D()
{
Name = "Find Slice".Localize();
}
public double SliceHeight { get; set; } = 10;
public IVertexSource VertexSource { get; set; } = new VertexStorage();
private double cutMargin = .01;
public (Mesh mesh, Polygons polygons) Cut(IObject3D item)
{
var mesh = new Mesh(item.Mesh.Vertices, item.Mesh.Faces);
var itemMatrix = item.WorldMatrix(this);
mesh.Transform(itemMatrix);
// calculate and add the PWN face from the loops
var cutPlane = new Plane(Vector3.UnitZ, new Vector3(0, 0, SliceHeight));
var slice = SliceLayer.CreateSlice(mesh, cutPlane);
// copy every face that is on or below the cut plane
// cut the faces at the cut plane
mesh.Split(new Plane(Vector3.UnitZ, SliceHeight), cutMargin, cleanAndMerge: false);
// remove every face above the cut plane
RemoveFacesAboveCut(mesh);
slice.Vertices().TriangulateFaces(null, mesh, SliceHeight);
mesh.Transform(itemMatrix.Inverted);
return (mesh, slice);
}
public override void Flatten(UndoBuffer undoBuffer)
{
var newPathObject = new PathObject3D()
{
VertexSource = new VertexStorage(this.VertexSource)
};
base.Flatten(undoBuffer, new IObject3D[] { newPathObject });
}
private void RemoveFacesAboveCut(Mesh mesh)
{
var newVertices = new List<Vector3Float>();
var newFaces = new List<Face>();
var facesToRemove = new HashSet<int>();
var cutRemove = SliceHeight - cutMargin;
for (int i = 0; i < mesh.Faces.Count; i++)
{
var face = mesh.Faces[i];
if (mesh.Vertices[face.v0].Z >= cutRemove
&& mesh.Vertices[face.v1].Z >= cutRemove
&& mesh.Vertices[face.v2].Z >= cutRemove)
{
// record the face for removal
facesToRemove.Add(i);
}
}
// make a new list of all the faces we are keeping
var keptFaces = new List<Face>();
for (int i = 0; i < mesh.Faces.Count; i++)
{
if (!facesToRemove.Contains(i))
{
keptFaces.Add(mesh.Faces[i]);
}
}
var vertexCount = mesh.Vertices.Count;
// add the new vertices
mesh.Vertices.AddRange(newVertices);
// add the new faces (have to make the vertex indices to the new vertices
foreach (var newFace in newFaces)
{
Face faceNewIndices = newFace;
faceNewIndices.v0 += vertexCount;
faceNewIndices.v1 += vertexCount;
faceNewIndices.v2 += vertexCount;
keptFaces.Add(faceNewIndices);
}
mesh.Faces = new FaceList(keptFaces);
}
public override Task Rebuild()
{
this.DebugDepth("Rebuild");
var rebuildLocks = this.RebuilLockAll();
var valuesChanged = false;
return TaskBuilder(
"Find Slice".Localize(),
(reporter, cancellationToken) =>
{
var polygons = new Polygons();
VertexSource = polygons.PolygonToPathStorage();
var newChildren = new List<Object3D>();
foreach (var sourceItem in SourceContainer.VisibleMeshes())
{
var meshPolygons = Cut(sourceItem);
polygons = polygons.CreateUnion(meshPolygons.polygons);
var newMesh = new Object3D()
{
Mesh = meshPolygons.mesh,
OwnerID = sourceItem.ID
};
newMesh.CopyProperties(sourceItem, Object3DPropertyFlags.All);
newChildren.Add(newMesh);
}
VertexSource = polygons.PolygonToPathStorage();
RemoveAllButSource();
SourceContainer.Visible = false;
foreach (var child in newChildren)
{
this.Children.Add(child);
}
UiThread.RunOnIdle(() =>
{
rebuildLocks.Dispose();
if (valuesChanged)
{
Invalidate(InvalidateType.DisplayValues);
}
Parent?.Invalidate(new InvalidateArgs(this, InvalidateType.Children));
});
return Task.CompletedTask;
});
}
public void UpdateControls(PublicPropertyChange change)
{
}
}
}

View file

@ -40,6 +40,7 @@ using MatterHackers.Agg.Transform;
using MatterHackers.Agg.UI;
using MatterHackers.Agg.VertexSource;
using MatterHackers.DataConverters3D;
using MatterHackers.DataConverters3D.UndoCommands;
using MatterHackers.Localizations;
using MatterHackers.MarchingSquares;
using MatterHackers.MatterControl.DesignTools.Operations;
@ -148,7 +149,8 @@ namespace MatterHackers.MatterControl.DesignTools
private void UpdateHistogramDisplay()
{
if (_histogramRawCache != null)
if (_histogramRawCache != null
&& _histogramDisplayCache != null)
{
var graphics2D = _histogramDisplayCache.NewGraphics2D();
graphics2D.Clear(Color.Transparent);
@ -206,6 +208,13 @@ namespace MatterHackers.MatterControl.DesignTools
this.DrawPath();
}
public override bool CanFlatten => true;
public override void Flatten(UndoBuffer undoBuffer)
{
this.FlattenToPathObject(undoBuffer);
}
public void GenerateMarchingSquaresAndLines(Action<double, string> progressReporter, ImageBuffer image, IThresholdFunction thresholdFunction)
{
if (image != null)

View file

@ -27,15 +27,36 @@ of the authors and should not be interpreted as representing official policies,
either expressed or implied, of the FreeBSD Project.
*/
using ClipperLib;
using MatterHackers.Agg.UI;
using MatterHackers.Agg.VertexSource;
using MatterHackers.DataConverters3D;
using MatterHackers.MatterControl.DesignTools.Operations;
using MatterHackers.MatterControl.PartPreviewWindow;
using Newtonsoft.Json;
using System.Collections.Generic;
namespace MatterHackers.MatterControl.DesignTools
{
public class PathObject3D : Object3D, IPathObject
public class PathObject3D : Object3D, IPathObject, ISelectedEditorDraw
{
public IVertexSource VertexSource { get; set; } = new VertexStorage();
[JsonIgnore]
private IVertexSource _vertexSource = new VertexStorage();
public IVertexSource VertexSource
{
get => _vertexSource;
set
{
_vertexSource = value;
// set the mesh to show the path
this.Mesh = this.VertexSource.Extrude(Constants.PathPolygonsHeight);
}
}
public void DrawEditor(Object3DControlsLayer layer, List<Object3DView> transparentMeshes, DrawEventArgs e)
{
this.DrawPath();
}
}
}

View file

@ -35,6 +35,7 @@ using System.Text;
using ClipperLib;
using MatterHackers.Agg;
using MatterHackers.Agg.Transform;
using MatterHackers.Agg.UI;
using MatterHackers.Agg.VertexSource;
using MatterHackers.DataConverters2D;
using MatterHackers.DataConverters3D;
@ -103,6 +104,32 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
return 0;
}
public static void FlattenToPathObject(this IObject3D item, UndoBuffer undoBuffer)
{
if (item is IPathObject pathObject)
{
using (item.RebuildLock())
{
var newPathObject = new PathObject3D();
newPathObject.VertexSource = new VertexStorage(pathObject.VertexSource);
// and replace us with the children
var replaceCommand = new ReplaceCommand(new[] { item }, new[] { newPathObject });
if (undoBuffer != null)
{
undoBuffer.AddAndDo(replaceCommand);
}
else
{
replaceCommand.Do();
}
newPathObject.MakeNameNonColliding();
}
}
}
public static void DrawPath(this IObject3D item)
{
if (item is IPathObject pathObject)

View file

@ -39,13 +39,15 @@ using MatterHackers.Agg.VertexSource;
using MatterHackers.DataConverters3D;
using MatterHackers.DataConverters3D.UndoCommands;
using MatterHackers.Localizations;
using MatterHackers.MatterControl.PartPreviewWindow;
using MatterHackers.Plugins.EditorTools;
using MatterHackers.PolygonMesh;
using MatterHackers.VectorMath;
using Newtonsoft.Json;
namespace MatterHackers.MatterControl.DesignTools.Operations
{
public class LinearExtrudeObject3D : Object3D
public class LinearExtrudeObject3D : Object3D, IObject3DControlsProvider
#if DEBUG
, IPropertyGridModifier
#endif
@ -84,6 +86,24 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
}
}
public void AddObject3DControls(Object3DControlsLayer object3DControlsLayer)
{
double getHeight() => Height.Value(this);
void setHeight(double height) => Height = height;
object3DControlsLayer.Object3DControls.Add(new ScaleHeightControl(object3DControlsLayer,
null,
null,
null,
null,
getHeight,
setHeight,
null,
null));
object3DControlsLayer.AddControls(ControlTypes.ScaleMatrixXY);
object3DControlsLayer.AddControls(ControlTypes.MoveInZ);
object3DControlsLayer.AddControls(ControlTypes.RotateXYZ);
}
public override void Flatten(UndoBuffer undoBuffer)
{
if (Mesh == null)
@ -138,6 +158,7 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
public override Task Rebuild()
{
this.DebugDepth("Rebuild");
var rebuildLock = RebuildLock();
bool valuesChanged = false;
@ -149,8 +170,7 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
var bevelInset = BevelInset.ClampIfNotCalculated(this, 0, Math.Min(aabb.XSize /2, aabb.YSize / 2), ref valuesChanged);
#endif
var rebuildLock = RebuildLock();
// now create a long running task to process the image
// now create a long running task to do the extrusion
return ApplicationController.Instance.Tasks.Execute(
"Linear Extrude".Localize(),
null,
@ -188,6 +208,7 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
}
Parent?.Invalidate(new InvalidateArgs(this, InvalidateType.Mesh));
});
return Task.CompletedTask;
});
}

View file

@ -44,13 +44,13 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
{
public class MergePathObject3D : OperationSourceContainerObject3D, IPathObject, ISelectedEditorDraw, IObject3DControlsProvider
{
private bool union;
private ClipperLib.ClipType clipType;
private string operationName;
public MergePathObject3D(string name, bool union)
public MergePathObject3D(string name, ClipperLib.ClipType clipType)
{
this.operationName = name;
this.union = union;
this.clipType = clipType;
Name = name;
}
@ -61,6 +61,13 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
this.DrawPath();
}
public override bool CanFlatten => true;
public override void Flatten(UndoBuffer undoBuffer)
{
this.FlattenToPathObject(undoBuffer);
}
public void AddObject3DControls(Object3DControlsLayer object3DControlsLayer)
{
object3DControlsLayer.AddControls(ControlTypes.Standard2D);
@ -135,14 +142,7 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
{
var itemVertexSource = pathItem.VertexSource.Transform(item.Matrix);
if (union)
{
resultsVertexSource = resultsVertexSource.Union(itemVertexSource);
}
else
{
resultsVertexSource = resultsVertexSource.MergePaths(itemVertexSource, ClipperLib.ClipType.ctIntersection);
}
resultsVertexSource = resultsVertexSource.MergePaths(itemVertexSource, clipType);
percentCompleted += amountPerOperation;
progressStatus.Progress0To1 = percentCompleted;

View file

@ -0,0 +1,132 @@
/*
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.Collections.Generic;
using System.Threading.Tasks;
using MatterHackers.Agg.UI;
using MatterHackers.Agg.VertexSource;
using MatterHackers.DataConverters3D;
using MatterHackers.Localizations;
using MatterHackers.MatterControl.DesignTools.Operations;
using MatterHackers.MatterControl.PartPreviewWindow;
using MatterHackers.PolygonMesh;
using Newtonsoft.Json;
namespace MatterHackers.MatterControl.DesignTools
{
public class BoxPathObject3D : PrimitiveObject3D, IPathObject, IObject3DControlsProvider, ISelectedEditorDraw
{
public BoxPathObject3D()
{
Name = "Box".Localize();
Color = Operations.Object3DExtensions.PrimitiveColors["Cube"];
}
public override string ThumbnailName => "Box";
[JsonIgnore]
private IVertexSource _vertexSource = new VertexStorage();
public IVertexSource VertexSource
{
get => _vertexSource;
set
{
_vertexSource = value;
// set the mesh to show the path
this.Mesh = this.VertexSource.Extrude(Constants.PathPolygonsHeight);
}
}
public void DrawEditor(Object3DControlsLayer layer, List<Object3DView> transparentMeshes, DrawEventArgs e)
{
this.DrawPath();
}
/// <summary>
/// This is the actual serialized with that can use expressions
/// </summary>
[MaxDecimalPlaces(2)]
public DoubleOrExpression Width { get; set; } = 20;
[MaxDecimalPlaces(2)]
public DoubleOrExpression Depth { get; set; } = 20;
public static async Task<BoxPathObject3D> Create()
{
var item = new BoxPathObject3D();
await item.Rebuild();
return item;
}
public void AddObject3DControls(Object3DControlsLayer object3DControlsLayer)
{
object3DControlsLayer.AddControls(ControlTypes.MoveInZ);
object3DControlsLayer.AddWidthDepthControls(this, Width, Depth, null);
object3DControlsLayer.AddControls(ControlTypes.MoveInZ);
object3DControlsLayer.AddControls(ControlTypes.RotateXYZ);
}
public override async void OnInvalidate(InvalidateArgs invalidateArgs)
{
if ((invalidateArgs.InvalidateType.HasFlag(InvalidateType.Properties) && invalidateArgs.Source == this))
{
await Rebuild();
}
else if (SheetObject3D.NeedsRebuild(this, invalidateArgs))
{
await Rebuild();
}
else
{
base.OnInvalidate(invalidateArgs);
}
}
public override Task Rebuild()
{
this.DebugDepth("Rebuild");
using (RebuildLock())
{
using (new CenterAndHeightMaintainer(this))
{
var width = Width.Value(this);
var depth = Depth.Value(this);
VertexSource = new RoundedRect(-width / 2, -depth / 2, width / 2, depth / 2, 0);
}
}
Parent?.Invalidate(new InvalidateArgs(this, InvalidateType.Mesh));
return Task.CompletedTask;
}
}
}

View file

@ -115,10 +115,15 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
}
public override void Flatten(UndoBuffer undoBuffer)
{
Flatten(undoBuffer, null);
}
protected void Flatten(UndoBuffer undoBuffer, IEnumerable<IObject3D> extraItems)
{
using (RebuildLock())
{
List<IObject3D> newChildren = new List<IObject3D>();
var newChildren = new List<IObject3D>();
// push our matrix into a copy of our children
foreach (var child in this.Children)
{
@ -210,7 +215,7 @@ namespace MatterHackers.MatterControl.DesignTools.Operations
{
using (RebuildLock())
{
List<IObject3D> newChildren = new List<IObject3D>();
var newChildren = new List<IObject3D>();
// push our matrix into a copy of our children
foreach (var child in this.SourceContainer.Children)
{

View file

@ -29,6 +29,7 @@ either expressed or implied, of the FreeBSD Project.
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Linq;
using System.Threading;
@ -274,6 +275,10 @@ namespace MatterHackers.MatterControl.DesignTools
// could not find a sheet, try to evaluate the expression directly
var evaluator = new Expression(inputExpression.ToLower());
if(evaluator.checkSyntax())
{
Debug.WriteLine(evaluator.getErrorMessage());
}
return CastResult<T>(evaluator.calculate().ToString(), inputExpression);
}
@ -282,7 +287,7 @@ namespace MatterHackers.MatterControl.DesignTools
/// </summary>
/// <param name="item">The item to start the search from</param>
/// <returns></returns>
public static SheetObject3D Find(IObject3D item)
private static SheetObject3D Find(IObject3D item)
{
// look through all the parents
foreach (var parent in item.Parents())

View file

@ -31,12 +31,10 @@ using System;
using System.Collections.Generic;
using System.IO;
using System.Threading.Tasks;
using MatterHackers.Agg;
using MatterHackers.Agg.Platform;
using MatterHackers.DataConverters3D;
using MatterHackers.Localizations;
using MatterHackers.MatterControl.DesignTools;
using MatterHackers.MatterControl.SlicerConfiguration;
namespace MatterHackers.MatterControl.Library
{
@ -62,60 +60,60 @@ namespace MatterHackers.MatterControl.Library
new GeneratorItem(
() => "Cube".Localize(),
async () => await CubeObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Pyramid".Localize(),
async () => await PyramidObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Wedge".Localize(),
async () => await WedgeObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Half Wedge".Localize(),
async () => await HalfWedgeObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Text".Localize(),
async () => await TextObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Cylinder".Localize(),
async () => await CylinderObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Cone".Localize(),
async () => await ConeObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Half Cylinder".Localize(),
async () => await HalfCylinderObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Torus".Localize(),
async () => await TorusObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Ring".Localize(),
async () => await RingObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Sphere".Localize(),
async () => await SphereObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Half Sphere".Localize(),
async () => await HalfSphereObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
#if DEBUG
new GeneratorItem(
() => "SCAD Script".Localize(),
async () => await OpenScadScriptObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "MarchingSquares".Localize(),
async () => await MarchingSquaresObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
#endif
new GeneratorItem(
() => "Image Converter".Localize(),
@ -140,21 +138,21 @@ namespace MatterHackers.MatterControl.Library
var constructedComponent = tempScene.SelectedItem;
tempScene.Children.Remove(constructedComponent);
return Task.FromResult<IObject3D>(constructedComponent);
return Task.FromResult(constructedComponent);
})
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Measure Tool".Localize(),
async () => await MeasureToolObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Description".Localize(),
async () => await DescriptionObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Variable Sheet".Localize(),
async () => await SheetObject3D.Create())
{ DateCreated = new System.DateTime(index++) },
{ DateCreated = new DateTime(index++) },
};
string title = "Primitive Shapes".Localize();
@ -164,6 +162,80 @@ namespace MatterHackers.MatterControl.Library
item.Category = title;
Items.Add(item);
}
this.ChildContainers.Add(
new DynamicContainerLink(
() => "Primitives 2D".Localize(),
StaticData.Instance.LoadIcon(Path.Combine("Library", "folder.png")),
StaticData.Instance.LoadIcon(Path.Combine("Library", "primitives_library_icon.png")),
() => new Primitives2DContainer())
{
IsReadOnly = true
});
}
}
public class Primitives2DContainer : LibraryContainer
{
public Primitives2DContainer()
{
Name = "Primitives 2D".Localize();
DefaultSort = new LibrarySortBehavior()
{
SortKey = SortKey.ModifiedDate,
Ascending = true,
};
}
public override void Load()
{
var library = ApplicationController.Instance.Library;
long index = DateTime.Now.Ticks;
var libraryItems = new List<GeneratorItem>()
{
new GeneratorItem(
() => "Box".Localize(),
async () => await BoxPathObject3D.Create())
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Triangle".Localize(),
async () => await PyramidObject3D.Create())
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Trapezoid".Localize(),
async () => await WedgeObject3D.Create())
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Text".Localize(),
async () => await TextPathObject3D.Create())
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Oval".Localize(),
async () => await CylinderObject3D.Create())
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Star".Localize(),
async () => await ConeObject3D.Create())
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Ring".Localize(),
async () => await RingObject3D.Create())
{ DateCreated = new DateTime(index++) },
new GeneratorItem(
() => "Circle".Localize(),
async () => await SphereObject3D.Create())
{ DateCreated = new DateTime(index++) },
};
string title = "2D Shapes".Localize();
foreach (var item in libraryItems)
{
item.Category = title;
Items.Add(item);
}
}
}
}

View file

@ -89,6 +89,13 @@ namespace MatterHackers.MatterControl.PartPreviewWindow.View3D
}
}
public override bool CanFlatten => true;
public override void Flatten(UndoBuffer undoBuffer)
{
this.FlattenToPathObject(undoBuffer);
}
public override Task Rebuild()
{
this.DebugDepth("Rebuild");
@ -112,7 +119,15 @@ namespace MatterHackers.MatterControl.PartPreviewWindow.View3D
}
// set the mesh to show the path
this.Mesh = this.VertexSource.Extrude(Constants.PathPolygonsHeight);
var extrudeMesh = this.VertexSource.Extrude(Constants.PathPolygonsHeight);
if(extrudeMesh.Vertices.Count() > 5)
{
this.Mesh = extrudeMesh;
}
else
{
this.Mesh = null;
}
UiThread.RunOnIdle(() =>
{