471 lines
No EOL
15 KiB
C#
471 lines
No EOL
15 KiB
C#
/*
|
|
Copyright (c) 2014, Kevin Pope
|
|
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 MatterHackers.Agg;
|
|
using MatterHackers.Agg.Image;
|
|
using MatterHackers.Agg.ImageProcessing;
|
|
using MatterHackers.Agg.PlatformAbstract;
|
|
using MatterHackers.Agg.UI;
|
|
using MatterHackers.Localizations;
|
|
using MatterHackers.MatterControl.CustomWidgets;
|
|
using MatterHackers.MatterControl.PrinterCommunication;
|
|
using MatterHackers.MatterControl.PrintQueue;
|
|
using MatterHackers.MatterControl.SlicerConfiguration;
|
|
using MatterHackers.VectorMath;
|
|
using System;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
|
|
namespace MatterHackers.MatterControl.ActionBar
|
|
{
|
|
public class PrintStatusRow : FlowLayoutWidget
|
|
{
|
|
private static FlowLayoutWidget iconContainer;
|
|
|
|
private TextWidget activePrintInfo;
|
|
|
|
private TextWidget activePrintLabel;
|
|
|
|
private TextWidget activePrintName;
|
|
|
|
private PartThumbnailWidget activePrintPreviewImage;
|
|
|
|
private TextWidget activePrintStatus;
|
|
|
|
private TemperatureWidgetBase bedTemperatureWidget;
|
|
|
|
private TemperatureWidgetBase extruderTemperatureWidget;
|
|
|
|
private QueueDataView queueDataView;
|
|
|
|
private TextInfo textInfo = new CultureInfo("en-US", false).TextInfo;
|
|
|
|
public PrintStatusRow(QueueDataView queueDataView)
|
|
{
|
|
Initialize();
|
|
|
|
this.HAnchor = HAnchor.ParentLeftRight;
|
|
|
|
this.queueDataView = queueDataView;
|
|
|
|
AddChildElements();
|
|
AddHandlers();
|
|
|
|
onActivePrintItemChanged(null, null);
|
|
|
|
if (privateAddIconToPrintStatusRow != null)
|
|
{
|
|
privateAddIconToPrintStatusRow(iconContainer);
|
|
}
|
|
}
|
|
|
|
public static event Action<GuiWidget> AddIconToPrintStatusRow
|
|
{
|
|
add
|
|
{
|
|
privateAddIconToPrintStatusRow += value;
|
|
// and call it right away
|
|
value(iconContainer);
|
|
}
|
|
|
|
remove
|
|
{
|
|
privateAddIconToPrintStatusRow -= value;
|
|
}
|
|
}
|
|
|
|
private static event Action<GuiWidget> privateAddIconToPrintStatusRow;
|
|
|
|
private event EventHandler unregisterEvents;
|
|
private string ActivePrintStatusText
|
|
{
|
|
set
|
|
{
|
|
if (activePrintStatus.Text != value)
|
|
{
|
|
activePrintStatus.Text = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
public override void OnClosed(EventArgs e)
|
|
{
|
|
if (activePrintPreviewImage.ItemWrapper != null)
|
|
{
|
|
activePrintPreviewImage.ItemWrapper.SlicingOutputMessage -= PrintItem_SlicingOutputMessage;
|
|
}
|
|
if (unregisterEvents != null)
|
|
{
|
|
unregisterEvents(this, null);
|
|
}
|
|
base.OnClosed(e);
|
|
}
|
|
|
|
public override void OnDraw(Graphics2D graphics2D)
|
|
{
|
|
base.OnDraw(graphics2D);
|
|
}
|
|
|
|
protected void AddHandlers()
|
|
{
|
|
PrinterConnectionAndCommunication.Instance.ActivePrintItemChanged.RegisterEvent(onPrintItemChanged, ref unregisterEvents);
|
|
PrinterConnectionAndCommunication.Instance.CommunicationStateChanged.RegisterEvent(onStateChanged, ref unregisterEvents);
|
|
PrinterConnectionAndCommunication.Instance.WroteLine.RegisterEvent(Instance_WroteLine, ref unregisterEvents);
|
|
PrinterConnectionAndCommunication.Instance.ActivePrintItemChanged.RegisterEvent(onActivePrintItemChanged, ref unregisterEvents);
|
|
}
|
|
|
|
protected void Initialize()
|
|
{
|
|
UiThread.RunOnIdle(OnIdle);
|
|
this.Margin = new BorderDouble(6, 3, 6, 6);
|
|
}
|
|
|
|
protected void onPrintItemChanged(object sender, EventArgs e)
|
|
{
|
|
UpdatePrintItemName();
|
|
UpdatePrintStatus();
|
|
}
|
|
|
|
private void AddChildElements()
|
|
{
|
|
activePrintPreviewImage = new PartThumbnailWidget(null, "part_icon_transparent_100x100.png", "building_thumbnail_100x100.png", PartThumbnailWidget.ImageSizes.Size115x115);
|
|
activePrintPreviewImage.VAnchor = VAnchor.ParentTop;
|
|
activePrintPreviewImage.Padding = new BorderDouble(0);
|
|
activePrintPreviewImage.HoverBackgroundColor = new RGBA_Bytes();
|
|
activePrintPreviewImage.BorderWidth = 3;
|
|
|
|
FlowLayoutWidget temperatureWidgets = new FlowLayoutWidget(FlowDirection.TopToBottom);
|
|
{
|
|
extruderTemperatureWidget = new TemperatureWidgetExtruder();
|
|
temperatureWidgets.AddChild(extruderTemperatureWidget);
|
|
|
|
bedTemperatureWidget = new TemperatureWidgetBed();
|
|
if (ActiveSliceSettings.Instance.HasHeatedBed())
|
|
{
|
|
temperatureWidgets.AddChild(bedTemperatureWidget);
|
|
}
|
|
}
|
|
temperatureWidgets.VAnchor |= VAnchor.ParentTop;
|
|
temperatureWidgets.Margin = new BorderDouble(left: 6);
|
|
|
|
FlowLayoutWidget printStatusContainer = CreateActivePrinterInfoWidget();
|
|
printStatusContainer.VAnchor |= VAnchor.ParentTop;
|
|
|
|
iconContainer = new FlowLayoutWidget(FlowDirection.TopToBottom);
|
|
iconContainer.Name = "PrintStatusRow.IconContainer";
|
|
iconContainer.VAnchor |= VAnchor.ParentTop;
|
|
iconContainer.Margin = new BorderDouble(top: 3);
|
|
iconContainer.AddChild(GetAutoLevelIndicator());
|
|
|
|
this.AddChild(activePrintPreviewImage);
|
|
this.AddChild(printStatusContainer);
|
|
this.AddChild(iconContainer);
|
|
this.AddChild(temperatureWidgets);
|
|
|
|
UpdatePrintStatus();
|
|
UpdatePrintItemName();
|
|
}
|
|
|
|
private FlowLayoutWidget CreateActivePrinterInfoWidget()
|
|
{
|
|
FlowLayoutWidget container = new FlowLayoutWidget(FlowDirection.TopToBottom);
|
|
container.Margin = new BorderDouble(6, 0, 6, 0);
|
|
container.HAnchor = HAnchor.ParentLeftRight;
|
|
container.VAnchor |= VAnchor.ParentTop;
|
|
|
|
FlowLayoutWidget topRow = new FlowLayoutWidget();
|
|
topRow.Name = "PrintStatusRow.ActivePrinterInfo.TopRow";
|
|
topRow.HAnchor = HAnchor.ParentLeftRight;
|
|
|
|
string nextPrintLabel = LocalizedString.Get("Next Print");
|
|
string nextPrintLabelFull = string.Format("{0}:", nextPrintLabel);
|
|
activePrintLabel = getPrintStatusLabel(nextPrintLabelFull, pointSize: 11);
|
|
activePrintLabel.VAnchor = VAnchor.ParentTop;
|
|
|
|
topRow.AddChild(activePrintLabel);
|
|
|
|
activePrintName = getPrintStatusLabel("this is the biggest name we will allow", pointSize: 14);
|
|
activePrintName.AutoExpandBoundsToText = false;
|
|
activePrintStatus = getPrintStatusLabel("this is the biggest label we will allow - bigger", pointSize: 11);
|
|
activePrintStatus.AutoExpandBoundsToText = false;
|
|
activePrintStatus.Text = "";
|
|
activePrintStatus.Margin = new BorderDouble(top: 3);
|
|
|
|
activePrintInfo = getPrintStatusLabel("", pointSize: 11);
|
|
activePrintInfo.AutoExpandBoundsToText = true;
|
|
|
|
PrintActionRow printActionRow = new PrintActionRow(queueDataView);
|
|
|
|
container.AddChild(topRow);
|
|
container.AddChild(activePrintName);
|
|
container.AddChild(activePrintStatus);
|
|
//container.AddChild(activePrintInfo);
|
|
container.AddChild(printActionRow);
|
|
|
|
return container;
|
|
}
|
|
|
|
private Button GetAutoLevelIndicator()
|
|
{
|
|
ImageButtonFactory imageButtonFactory = new ImageButtonFactory();
|
|
imageButtonFactory.InvertImageColor = false;
|
|
ImageBuffer levelingImage = StaticData.Instance.LoadIcon("leveling_32x32.png", 16, 16).InvertLightness();
|
|
Button autoLevelButton = imageButtonFactory.Generate(levelingImage, levelingImage);
|
|
autoLevelButton.Margin = new Agg.BorderDouble(top: 3);
|
|
autoLevelButton.ToolTipText = "Print leveling is enabled.".Localize();
|
|
autoLevelButton.Visible = ActiveSliceSettings.Instance.DoPrintLeveling();
|
|
|
|
ActiveSliceSettings.ActivePrinterChanged.RegisterEvent((sender, e) =>
|
|
{
|
|
autoLevelButton.Visible = ActiveSliceSettings.Instance.DoPrintLeveling();
|
|
}, ref unregisterEvents);
|
|
|
|
ActiveSliceSettings.Instance.DoPrintLevelingChanged.RegisterEvent((sender, e) =>
|
|
{
|
|
autoLevelButton.Visible = ActiveSliceSettings.Instance.DoPrintLeveling();
|
|
}, ref unregisterEvents);
|
|
|
|
return autoLevelButton;
|
|
}
|
|
|
|
private string getConnectionMessage()
|
|
{
|
|
if (ActiveSliceSettings.Instance == null)
|
|
{
|
|
return LocalizedString.Get("Press 'Connect' to select a printer.");
|
|
}
|
|
else
|
|
{
|
|
switch (PrinterConnectionAndCommunication.Instance.CommunicationState)
|
|
{
|
|
case PrinterConnectionAndCommunication.CommunicationStates.Disconnected:
|
|
return LocalizedString.Get("Not connected. Press 'Connect' to enable printing.");
|
|
|
|
case PrinterConnectionAndCommunication.CommunicationStates.AttemptingToConnect:
|
|
string attemptToConnect = LocalizedString.Get("Attempting to Connect");
|
|
string attemptToConnectFull = string.Format("{0}...", attemptToConnect);
|
|
return attemptToConnectFull;
|
|
|
|
case PrinterConnectionAndCommunication.CommunicationStates.ConnectionLost:
|
|
case PrinterConnectionAndCommunication.CommunicationStates.FailedToConnect:
|
|
return LocalizedString.Get("Unable to communicate with printer.");
|
|
|
|
default:
|
|
return "";
|
|
}
|
|
}
|
|
}
|
|
|
|
private TextWidget getPrintStatusLabel(string text, int pointSize)
|
|
{
|
|
TextWidget widget = new TextWidget(text, pointSize: pointSize);
|
|
widget.TextColor = RGBA_Bytes.White;
|
|
widget.AutoExpandBoundsToText = true;
|
|
widget.MinimumSize = new Vector2(widget.Width, widget.Height);
|
|
return widget;
|
|
}
|
|
|
|
private void Instance_WroteLine(object sender, EventArgs e)
|
|
{
|
|
UpdatePrintStatus();
|
|
}
|
|
|
|
private void onActivePrintItemChanged(object sender, EventArgs e)
|
|
{
|
|
// first we have to remove any link to an old part (the part currently in the view)
|
|
if (activePrintPreviewImage.ItemWrapper != null)
|
|
{
|
|
activePrintPreviewImage.ItemWrapper.SlicingOutputMessage -= PrintItem_SlicingOutputMessage;
|
|
}
|
|
|
|
activePrintPreviewImage.ItemWrapper = PrinterConnectionAndCommunication.Instance.ActivePrintItem;
|
|
|
|
// then hook up our new part
|
|
if (activePrintPreviewImage.ItemWrapper != null)
|
|
{
|
|
activePrintPreviewImage.ItemWrapper.SlicingOutputMessage += PrintItem_SlicingOutputMessage;
|
|
}
|
|
|
|
activePrintPreviewImage.Invalidate();
|
|
}
|
|
|
|
private void OnIdle()
|
|
{
|
|
if (PrinterConnectionAndCommunication.Instance.PrinterIsPrinting)
|
|
{
|
|
UpdatePrintStatus();
|
|
}
|
|
|
|
if (!HasBeenClosed)
|
|
{
|
|
UiThread.RunOnIdle(OnIdle, 1);
|
|
}
|
|
}
|
|
|
|
private void onStateChanged(object sender, EventArgs e)
|
|
{
|
|
UpdatePrintStatus();
|
|
}
|
|
|
|
private void PrintItem_SlicingOutputMessage(object sender, EventArgs e)
|
|
{
|
|
StringEventArgs message = e as StringEventArgs;
|
|
ActivePrintStatusText = message.Data;
|
|
}
|
|
private void SetVisibleStatus()
|
|
{
|
|
if (ActiveSliceSettings.Instance != null)
|
|
{
|
|
if (ActiveSliceSettings.Instance.HasHeatedBed())
|
|
{
|
|
bedTemperatureWidget.Visible = true;
|
|
}
|
|
else
|
|
{
|
|
bedTemperatureWidget.Visible = false;
|
|
}
|
|
}
|
|
}
|
|
private void UpdatePrintItemName()
|
|
{
|
|
if (PrinterConnectionAndCommunication.Instance.ActivePrintItem != null)
|
|
{
|
|
string labelName = textInfo.ToTitleCase(PrinterConnectionAndCommunication.Instance.ActivePrintItem.Name);
|
|
labelName = labelName.Replace('_', ' ');
|
|
this.activePrintName.Text = labelName;
|
|
}
|
|
else
|
|
{
|
|
this.activePrintName.Text = LocalizedString.Get("No items in the print queue");
|
|
}
|
|
}
|
|
|
|
private void UpdatePrintStatus()
|
|
{
|
|
if (PrinterConnectionAndCommunication.Instance.ActivePrintItem != null)
|
|
{
|
|
int totalSecondsInPrint = PrinterConnectionAndCommunication.Instance.TotalSecondsInPrint;
|
|
|
|
int totalHoursInPrint = (int)(totalSecondsInPrint / (60 * 60));
|
|
int totalMinutesInPrint = (int)(totalSecondsInPrint / 60 - totalHoursInPrint * 60);
|
|
totalSecondsInPrint = totalSecondsInPrint % 60;
|
|
|
|
string totalTimeLabel = LocalizedString.Get("Est. Print Time");
|
|
string calculatingLabel = LocalizedString.Get("Calculating...");
|
|
string totalPrintTimeText;
|
|
|
|
if (totalSecondsInPrint > 0)
|
|
{
|
|
if (totalHoursInPrint > 0)
|
|
{
|
|
totalPrintTimeText = string.Format("{3} {0}h {1:00}m {2:00}s",
|
|
totalHoursInPrint,
|
|
totalMinutesInPrint,
|
|
totalSecondsInPrint,
|
|
totalTimeLabel);
|
|
}
|
|
else
|
|
{
|
|
totalPrintTimeText = string.Format("{2} {0}m {1:00}s",
|
|
totalMinutesInPrint,
|
|
totalSecondsInPrint,
|
|
totalTimeLabel);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (totalSecondsInPrint < 0)
|
|
{
|
|
totalPrintTimeText = string.Format("{0}", LocalizedString.Get("Streaming GCode..."));
|
|
}
|
|
else
|
|
{
|
|
totalPrintTimeText = string.Format("{0}: {1}", totalTimeLabel, calculatingLabel);
|
|
}
|
|
}
|
|
|
|
//GC.WaitForFullGCComplete();
|
|
|
|
string printPercentRemainingText;
|
|
string printPercentCompleteText = LocalizedString.Get("complete");
|
|
printPercentRemainingText = string.Format("{0:0.0}% {1}", PrinterConnectionAndCommunication.Instance.PercentComplete, printPercentCompleteText);
|
|
|
|
switch (PrinterConnectionAndCommunication.Instance.CommunicationState)
|
|
{
|
|
case PrinterConnectionAndCommunication.CommunicationStates.PreparingToPrint:
|
|
string preparingPrintLabel = LocalizedString.Get("Preparing To Print");
|
|
string preparingPrintLabelFull = string.Format("{0}:", preparingPrintLabel);
|
|
activePrintLabel.Text = preparingPrintLabelFull;
|
|
//ActivePrintStatusText = ""; // set by slicer
|
|
activePrintInfo.Text = "";
|
|
break;
|
|
|
|
case PrinterConnectionAndCommunication.CommunicationStates.Printing:
|
|
{
|
|
activePrintLabel.Text = PrinterConnectionAndCommunication.Instance.PrintingStateString;
|
|
ActivePrintStatusText = totalPrintTimeText;
|
|
}
|
|
break;
|
|
|
|
case PrinterConnectionAndCommunication.CommunicationStates.Paused:
|
|
{
|
|
string activePrintLabelText = LocalizedString.Get("Printing Paused");
|
|
string activePrintLabelTextFull = string.Format("{0}:", activePrintLabelText);
|
|
activePrintLabel.Text = activePrintLabelTextFull;
|
|
ActivePrintStatusText = totalPrintTimeText;
|
|
}
|
|
break;
|
|
|
|
case PrinterConnectionAndCommunication.CommunicationStates.FinishedPrint:
|
|
string donePrintingText = LocalizedString.Get("Done Printing");
|
|
string donePrintingTextFull = string.Format("{0}:", donePrintingText);
|
|
activePrintLabel.Text = donePrintingTextFull;
|
|
ActivePrintStatusText = totalPrintTimeText;
|
|
break;
|
|
|
|
default:
|
|
string nextPrintLabelActive = LocalizedString.Get("Next Print");
|
|
string nextPrintLabelActiveFull = string.Format("{0}: ", nextPrintLabelActive);
|
|
|
|
activePrintLabel.Text = nextPrintLabelActiveFull;
|
|
ActivePrintStatusText = getConnectionMessage();
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
string nextPrintLabel = LocalizedString.Get("Next Print");
|
|
string nextPrintLabelFull = string.Format("{0}:", nextPrintLabel);
|
|
|
|
activePrintLabel.Text = nextPrintLabelFull;
|
|
ActivePrintStatusText = string.Format(LocalizedString.Get("Press 'Add' to choose an item to print"));
|
|
}
|
|
}
|
|
}
|
|
} |