153 lines
No EOL
4.8 KiB
C#
153 lines
No EOL
4.8 KiB
C#
/*
|
|
Copyright (c) 2016, 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;
|
|
|
|
namespace MatterHackers.MatterControl.PrinterCommunication.Io
|
|
{
|
|
public class MaxLengthStream : GCodeStreamProxy
|
|
{
|
|
// 20 instruction per second
|
|
private double maxSecondsPerSegment = 1.0 / 20.0;
|
|
private List<PrinterMove> movesToSend = new List<PrinterMove>();
|
|
|
|
public MaxLengthStream(PrinterConfig printer, GCodeStream internalStream, double maxSegmentLength)
|
|
: base(printer, internalStream)
|
|
{
|
|
this.MaxSegmentLength = maxSegmentLength;
|
|
}
|
|
|
|
PrinterMove lastDestination = PrinterMove.Unknown;
|
|
public double MaxSegmentLength { get; set; }
|
|
|
|
public override string DebugInfo
|
|
{
|
|
get
|
|
{
|
|
return $"Last Destination = {lastDestination}";
|
|
}
|
|
}
|
|
|
|
public void Cancel()
|
|
{
|
|
lock (movesToSend)
|
|
{
|
|
movesToSend.Clear();
|
|
}
|
|
}
|
|
|
|
public override string ReadLine()
|
|
{
|
|
if (movesToSend.Count == 0)
|
|
{
|
|
string lineToSend = base.ReadLine();
|
|
|
|
if (lineToSend != null
|
|
&& lineToSend.EndsWith("; NO_PROCESSING"))
|
|
{
|
|
return lineToSend;
|
|
}
|
|
|
|
if (lineToSend != null
|
|
&& LineIsMovement(lineToSend))
|
|
{
|
|
PrinterMove currentDestination = GetPosition(lineToSend, lastDestination);
|
|
|
|
if (currentDestination.FullyKnown)
|
|
{
|
|
PrinterMove deltaToDestination = currentDestination - lastDestination;
|
|
deltaToDestination.feedRate = 0; // remove the changing of the federate (we'll set it initially)
|
|
double lengthSquared = Math.Max(deltaToDestination.LengthSquared, deltaToDestination.extrusion * deltaToDestination.extrusion);
|
|
if (lengthSquared > MaxSegmentLength * MaxSegmentLength)
|
|
{
|
|
// create the line segments to send
|
|
double length = Math.Sqrt(lengthSquared);
|
|
int numSegmentsToCutInto = (int)Math.Ceiling(length / MaxSegmentLength);
|
|
|
|
// segments = (((mm/min) / (60s/min))mm/s / s/segment)segments*mm / mm
|
|
double maxSegmentsCanTransmit = 1 / (((currentDestination.feedRate / 60) * maxSecondsPerSegment) / length);
|
|
|
|
int numSegmentsToSend = Math.Max(1, Math.Min(numSegmentsToCutInto, (int)maxSegmentsCanTransmit));
|
|
|
|
if (numSegmentsToSend > 1)
|
|
{
|
|
PrinterMove deltaForSegment = deltaToDestination / numSegmentsToSend;
|
|
PrinterMove nextPoint = lastDestination + deltaForSegment;
|
|
nextPoint.feedRate = currentDestination.feedRate;
|
|
for (int i = 0; i < numSegmentsToSend; i++)
|
|
{
|
|
lock (movesToSend)
|
|
{
|
|
movesToSend.Add(nextPoint);
|
|
}
|
|
nextPoint += deltaForSegment;
|
|
}
|
|
|
|
// send the first one
|
|
PrinterMove positionToSend = movesToSend[0];
|
|
lock (movesToSend)
|
|
{
|
|
movesToSend.RemoveAt(0);
|
|
}
|
|
|
|
string altredLineToSend = CreateMovementLine(positionToSend, lastDestination);
|
|
lastDestination = positionToSend;
|
|
return altredLineToSend;
|
|
}
|
|
}
|
|
}
|
|
|
|
lastDestination = currentDestination;
|
|
}
|
|
return lineToSend;
|
|
}
|
|
else
|
|
{
|
|
PrinterMove positionToSend = movesToSend[0];
|
|
lock (movesToSend)
|
|
{
|
|
movesToSend.RemoveAt(0);
|
|
}
|
|
|
|
string lineToSend = CreateMovementLine(positionToSend, lastDestination);
|
|
|
|
lastDestination = positionToSend;
|
|
|
|
return lineToSend;
|
|
}
|
|
}
|
|
|
|
public override void SetPrinterPosition(PrinterMove position)
|
|
{
|
|
this.lastDestination.CopyKnowSettings(position);
|
|
internalStream.SetPrinterPosition(lastDestination);
|
|
}
|
|
}
|
|
} |