2014-01-29 19:09:30 -08:00
/ *
2014-02-15 18:06:03 -08:00
Copyright ( c ) 2014 , Lars Brubaker
2014-01-29 19:09:30 -08:00
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.Diagnostics ;
using System.Globalization ;
2014-03-16 10:23:56 -07:00
using System.IO ;
using System.Text ;
using System.Threading ;
2014-01-29 19:09:30 -08:00
using MatterHackers.Agg ;
using MatterHackers.Agg.UI ;
2014-06-11 14:52:58 -07:00
using MatterHackers.Localizations ;
2014-01-29 19:09:30 -08:00
using MatterHackers.MatterControl.DataStorage ;
2014-02-04 22:28:30 -08:00
using MatterHackers.MatterControl.PartPreviewWindow ;
2014-03-16 10:23:56 -07:00
using MatterHackers.MatterControl.PluginSystem ;
2014-06-11 14:52:58 -07:00
using MatterHackers.MatterControl.PrinterCommunication ;
2014-03-16 10:23:56 -07:00
using MatterHackers.MatterControl.PrintQueue ;
2014-05-06 18:25:16 -07:00
using MatterHackers.MatterControl.SettingsManagement ;
2014-06-11 14:52:58 -07:00
using MatterHackers.MatterControl.SlicerConfiguration ;
2014-03-16 10:23:56 -07:00
using MatterHackers.VectorMath ;
2014-10-12 08:18:24 -07:00
using MatterHackers.PolygonMesh.Processors ;
2014-11-29 08:20:10 -08:00
using MatterHackers.MatterControl.CreatorPlugins ;
2014-11-25 07:47:03 -08:00
using MatterHackers.Agg.PlatformAbstract ;
2014-01-29 19:09:30 -08:00
namespace MatterHackers.MatterControl
{
public class MatterControlApplication : SystemWindow
{
2014-02-04 22:28:30 -08:00
string [ ] commandLineArgs = null ;
2014-01-29 19:09:30 -08:00
bool firstDraw = true ;
2014-02-27 14:56:20 -08:00
bool ShowMemoryUsed = false ;
bool DoCGCollectEveryDraw = false ;
2014-03-30 17:52:33 -07:00
public bool RestartOnClose = false ;
2014-01-29 19:09:30 -08:00
2014-12-02 10:28:49 -08:00
static MatterControlApplication ( )
{
// Because fields on this class call localization methods and because those methods depend on the StaticData provider and because the field
// initializers run before the class constructor, we need to init the platform specific provider in the static constructor (or write a custom initializer method)
//
// Initialize a standard file system backed StaticData provider
StaticData . Instance = new MatterHackers . Agg . FileSystemStaticData ( ) ;
}
2014-02-10 14:11:51 -08:00
public MatterControlApplication ( double width , double height )
2014-01-29 19:09:30 -08:00
: base ( width , height )
{
2014-06-24 12:32:12 -07:00
CrashTracker . Reset ( ) ;
2014-02-26 11:05:58 -08:00
this . commandLineArgs = Environment . GetCommandLineArgs ( ) ;
2014-01-29 19:09:30 -08:00
Thread . CurrentThread . CurrentCulture = CultureInfo . InvariantCulture ;
2014-03-25 07:46:14 -07:00
foreach ( string command in commandLineArgs )
2014-02-26 11:05:58 -08:00
{
2014-03-25 07:46:14 -07:00
string commandUpper = command . ToUpper ( ) ;
switch ( commandUpper )
2014-02-26 11:05:58 -08:00
{
2014-02-27 14:56:20 -08:00
case "TEST" :
Testing . TestingDispatch testDispatch = new Testing . TestingDispatch ( ) ;
2014-08-11 10:23:30 -07:00
testDispatch . RunTests ( ) ;
2014-02-27 14:56:20 -08:00
return ;
2014-03-25 07:46:14 -07:00
case "CLEAR_CACHE" :
AboutPage . DeleteCacheData ( ) ;
break ;
2014-02-27 14:56:20 -08:00
case "SHOW_MEMORY" :
ShowMemoryUsed = true ;
break ;
case "DO_GC_COLLECT_EVERY_DRAW" :
ShowMemoryUsed = true ;
DoCGCollectEveryDraw = true ;
break ;
2014-02-26 11:05:58 -08:00
}
2014-10-12 08:18:24 -07:00
if ( MeshFileIo . ValidFileExtensions ( ) . Contains ( Path . GetExtension ( command ) . ToUpper ( ) ) )
{
// If we are the only instance running then do nothing.
// Else send these to the running instance so it can load them.
}
2014-02-26 11:05:58 -08:00
}
2014-02-04 10:47:57 -08:00
//WriteTestGCodeFile();
2014-03-02 15:28:47 -08:00
#if ! DEBUG
2014-01-29 19:09:30 -08:00
if ( File . Exists ( "RunUnitTests.txt" ) )
2014-03-02 15:28:47 -08:00
#endif
2014-01-29 19:09:30 -08:00
{
2014-07-28 13:48:28 -07:00
#if IS_WINDOWS_FORMS
2014-04-12 12:20:04 -07:00
Clipboard . SetSystemClipboardFunctions ( System . Windows . Forms . Clipboard . GetText , System . Windows . Forms . Clipboard . SetText , System . Windows . Forms . Clipboard . ContainsText ) ;
2014-07-28 13:48:28 -07:00
#endif
2014-01-29 19:09:30 -08:00
2014-04-07 18:28:47 -07:00
MatterHackers . PolygonMesh . UnitTests . UnitTests . Run ( ) ;
2014-03-31 13:42:18 -07:00
MatterHackers . RayTracer . UnitTests . Run ( ) ;
2014-01-29 19:09:30 -08:00
MatterHackers . Agg . Tests . UnitTests . Run ( ) ;
MatterHackers . VectorMath . Tests . UnitTests . Run ( ) ;
MatterHackers . Agg . UI . Tests . UnitTests . Run ( ) ;
// you can turn this on to debug some bounds issues
//GuiWidget.DebugBoundsUnderMouse = true;
}
GuiWidget . DefaultEnforceIntegerBounds = true ;
2014-09-23 18:02:07 -07:00
if ( ActiveTheme . Instance . DisplayMode = = ActiveTheme . ApplicationDisplayType . Touchscreen )
{
TextWidget . GlobalPointSizeScaleRatio = 1.3 ;
}
2014-08-30 16:03:28 -07:00
2014-09-19 19:17:12 -07:00
this . AddChild ( ApplicationController . Instance . MainView ) ;
2014-12-15 15:44:26 -08:00
this . MinimumSize = new Vector2 ( 570 , 600 ) ;
2014-01-29 19:09:30 -08:00
this . Padding = new BorderDouble ( 0 ) ; //To be re-enabled once native borders are turned off
#if false // this is to test freeing gcodefile memory
Button test = new Button ( "test" ) ;
test . Click + = ( sender , e ) = >
{
//MatterHackers.GCodeVisualizer.GCodeFile gcode = new GCodeVisualizer.GCodeFile();
//gcode.Load(@"C:\Users\lbrubaker\Downloads\drive assy.gcode");
SystemWindow window = new SystemWindow ( 100 , 100 ) ;
window . ShowAsSystemWindow ( ) ;
} ;
allControls . AddChild ( test ) ;
#endif
this . AnchorAll ( ) ;
UseOpenGL = true ;
2014-11-14 13:33:01 -08:00
string version = "1.2" ;
2014-05-22 16:55:28 -07:00
Title = "MatterControl {0}" . FormatWith ( version ) ;
if ( OemSettings . Instance . WindowTitleExtra ! = null & & OemSettings . Instance . WindowTitleExtra . Trim ( ) . Length > 0 )
{
Title = Title + " - {1}" . FormatWith ( version , OemSettings . Instance . WindowTitleExtra ) ;
}
2014-01-29 19:09:30 -08:00
UiThread . RunOnIdle ( CheckOnPrinter ) ;
2014-04-12 11:54:05 -07:00
string desktopPosition = ApplicationSettings . Instance . get ( "DesktopPosition" ) ;
if ( desktopPosition ! = null & & desktopPosition ! = "" )
{
string [ ] sizes = desktopPosition . Split ( ',' ) ;
2014-04-28 15:19:41 -07:00
//If the desktop position is less than -10,-10, override
int xpos = Math . Max ( int . Parse ( sizes [ 0 ] ) , - 10 ) ;
int ypos = Math . Max ( int . Parse ( sizes [ 1 ] ) , - 10 ) ;
DesktopPosition = new Point2D ( xpos , ypos ) ;
}
2014-12-16 12:37:57 -08:00
// make sure when we start up that we are showing the 3D view
UserSettings . Instance . Fields . EmbededViewShowingGCode = false ;
2014-01-29 19:09:30 -08:00
ShowAsSystemWindow ( ) ;
}
2014-02-04 10:47:57 -08:00
private static void WriteMove ( StringBuilder gcodeStringBuilder , Vector2 center )
{
gcodeStringBuilder . AppendLine ( "G1 X" + center . x . ToString ( ) + " Y" + center . y . ToString ( ) ) ;
}
2014-02-27 14:56:20 -08:00
2014-02-04 10:47:57 -08:00
public static void WriteTestGCodeFile ( )
{
StringBuilder gcodeStringBuilder = new StringBuilder ( ) ;
2014-03-17 14:41:36 -07:00
int loops = 15 ;
2014-11-10 13:00:14 -08:00
int steps = 200 ;
double radius = 50 ;
Vector2 center = new Vector2 ( 150 , 100 ) ;
2014-02-04 10:47:57 -08:00
gcodeStringBuilder . AppendLine ( "G28 ; home all axes" ) ;
gcodeStringBuilder . AppendLine ( "G90 ; use absolute coordinates" ) ;
gcodeStringBuilder . AppendLine ( "G21 ; set units to millimeters" ) ;
gcodeStringBuilder . AppendLine ( "G92 E0" ) ;
gcodeStringBuilder . AppendLine ( "G1 F7800.000" ) ;
2014-11-10 13:00:14 -08:00
gcodeStringBuilder . AppendLine ( "G1 Z" + ( 5 ) . ToString ( ) ) ;
2014-02-04 10:47:57 -08:00
WriteMove ( gcodeStringBuilder , center ) ;
for ( int loop = 0 ; loop < loops ; loop + + )
{
for ( int step = 0 ; step < steps ; step + + )
{
Vector2 nextPosition = new Vector2 ( radius , 0 ) ;
nextPosition . Rotate ( MathHelper . Tau / steps * step ) ;
WriteMove ( gcodeStringBuilder , center + nextPosition ) ;
}
}
gcodeStringBuilder . AppendLine ( "M84 ; disable motors" ) ;
System . IO . File . WriteAllText ( "PerformanceTest.gcode" , gcodeStringBuilder . ToString ( ) ) ;
}
2014-02-27 14:56:20 -08:00
2014-01-29 19:09:30 -08:00
void CheckOnPrinter ( object state )
{
2014-06-11 14:52:58 -07:00
PrinterConnectionAndCommunication . Instance . OnIdle ( ) ;
2014-01-29 19:09:30 -08:00
UiThread . RunOnIdle ( CheckOnPrinter ) ;
}
public override void OnParentChanged ( EventArgs e )
{
if ( File . Exists ( "RunUnitTests.txt" ) )
{
//DiagnosticWidget diagnosticView = new DiagnosticWidget(this);
}
base . OnParentChanged ( e ) ;
// now that we are all set up lets load our plugins and allow them their chance to set things up
FindAndInstantiatePlugins ( ) ;
}
private void FindAndInstantiatePlugins ( )
{
#if false
string pluginDirectory = Path . Combine ( ".." , ".." , ".." , "MatterControlPlugins" , "bin" ) ;
#if DEBUG
pluginDirectory = Path . Combine ( pluginDirectory , "Debug" ) ;
#else
pluginDirectory = Path . Combine ( pluginDirectory , "Release" ) ;
#endif
if ( ! Directory . Exists ( pluginDirectory ) )
{
string dataPath = DataStorage . ApplicationDataStorage . Instance . ApplicationUserDataPath ;
pluginDirectory = Path . Combine ( dataPath , "Plugins" ) ;
}
// TODO: this should look in a plugin folder rather than just the application directory (we probably want it in the user folder).
PluginFinder < MatterControlPlugin > pulginFinder = new PluginFinder < MatterControlPlugin > ( pluginDirectory ) ;
#else
PluginFinder < MatterControlPlugin > pulginFinder = new PluginFinder < MatterControlPlugin > ( ) ;
#endif
2014-11-25 07:47:03 -08:00
2014-01-29 19:09:30 -08:00
string oemName = ApplicationSettings . Instance . GetOEMName ( ) ;
foreach ( MatterControlPlugin plugin in pulginFinder . Plugins )
{
string pluginInfo = plugin . GetPluginInfoJSon ( ) ;
Dictionary < string , string > nameValuePairs = Newtonsoft . Json . JsonConvert . DeserializeObject < Dictionary < string , string > > ( pluginInfo ) ;
if ( nameValuePairs ! = null & & nameValuePairs . ContainsKey ( "OEM" ) )
{
if ( nameValuePairs [ "OEM" ] = = oemName )
{
plugin . Initialize ( this ) ;
}
}
else
{
plugin . Initialize ( this ) ;
}
}
}
Stopwatch totalDrawTime = new Stopwatch ( ) ;
int drawCount = 0 ;
2014-08-17 20:53:00 -07:00
Gaming . Game . DataViewGraph msGraph = new Gaming . Game . DataViewGraph ( new Vector2 ( 20 , 500 ) , 50 , 50 , 0 , 200 ) ;
2014-01-29 19:09:30 -08:00
public override void OnDraw ( Graphics2D graphics2D )
{
2014-03-02 15:28:47 -08:00
totalDrawTime . Restart ( ) ;
2014-07-20 17:49:49 -07:00
GuiWidget . DrawCount = 0 ;
2014-01-29 19:09:30 -08:00
base . OnDraw ( graphics2D ) ;
2014-03-02 15:28:47 -08:00
totalDrawTime . Stop ( ) ;
2014-01-29 19:09:30 -08:00
2014-02-27 14:56:20 -08:00
if ( ShowMemoryUsed )
2014-01-29 19:09:30 -08:00
{
2014-02-27 14:56:20 -08:00
long memory = GC . GetTotalMemory ( false ) ;
2014-07-20 17:49:49 -07:00
this . Title = "Allocated = {0:n0} : {1}ms, d{2} Size = {3}x{4}, onIdle = {5}, drawCount = {6}" . FormatWith ( memory , totalDrawTime . ElapsedMilliseconds , drawCount + + , this . Width , this . Height , UiThread . Count , GuiWidget . DrawCount ) ;
2014-02-27 14:56:20 -08:00
if ( DoCGCollectEveryDraw )
{
GC . Collect ( ) ;
}
2014-01-29 19:09:30 -08:00
}
2014-02-27 14:56:20 -08:00
2014-01-29 19:09:30 -08:00
if ( firstDraw )
{
2014-06-05 12:55:05 -07:00
UiThread . RunOnIdle ( DoAutoConnectIfRequired ) ;
2014-01-29 19:09:30 -08:00
firstDraw = false ;
2014-02-04 22:28:30 -08:00
foreach ( string arg in commandLineArgs )
{
2014-10-24 14:55:27 -07:00
string argExtension = Path . GetExtension ( arg ) . ToUpper ( ) ;
if ( argExtension . Length > 1
& & MeshFileIo . ValidFileExtensions ( ) . Contains ( argExtension ) )
2014-02-04 22:28:30 -08:00
{
2014-07-15 14:02:03 -07:00
QueueData . Instance . AddItem ( new PrintItemWrapper ( new DataStorage . PrintItem ( Path . GetFileName ( arg ) , Path . GetFullPath ( arg ) ) ) ) ;
2014-02-04 22:28:30 -08:00
}
}
2014-11-29 08:20:10 -08:00
#if false
foreach ( CreatorInformation creatorInfo in RegisteredCreators . Instance . Creators )
{
if ( creatorInfo . description . Contains ( "Image" ) )
{
creatorInfo . functionToLaunchCreator ( null , null ) ;
}
}
#endif
2014-01-29 19:09:30 -08:00
}
2014-08-17 20:53:00 -07:00
//msGraph.AddData("ms", totalDrawTime.ElapsedMilliseconds);
//msGraph.Draw(MatterHackers.Agg.Transform.Affine.NewIdentity(), graphics2D);
2014-01-29 19:09:30 -08:00
}
2014-06-05 12:55:05 -07:00
public void DoAutoConnectIfRequired ( object state )
{
ActivePrinterProfile . CheckForAndDoAutoConnect ( ) ;
}
2014-01-29 19:09:30 -08:00
public override void OnMouseMove ( MouseEventArgs mouseEvent )
{
if ( GuiWidget . DebugBoundsUnderMouse )
{
Invalidate ( ) ;
}
base . OnMouseMove ( mouseEvent ) ;
}
[STAThread]
2014-02-10 14:11:51 -08:00
public static void Main ( )
2014-01-29 19:09:30 -08:00
{
2014-07-15 14:02:03 -07:00
// Make sure we have the right woring directory as we assume everything relative to the executable.
Directory . SetCurrentDirectory ( Path . GetDirectoryName ( System . Reflection . Assembly . GetEntryAssembly ( ) . Location ) ) ;
2014-01-29 19:09:30 -08:00
Datastore . Instance . Initialize ( ) ;
// try and open our window matching the last size that we had for it.
string windowSize = ApplicationSettings . Instance . get ( "WindowSize" ) ;
2014-12-09 17:53:15 -08:00
int width = 1280 ;
int height = 720 ;
2014-02-27 14:56:20 -08:00
if ( windowSize ! = null & & windowSize ! = "" )
2014-01-29 19:09:30 -08:00
{
string [ ] sizes = windowSize . Split ( ',' ) ;
2014-12-09 17:53:15 -08:00
width = Math . Max ( int . Parse ( sizes [ 0 ] ) , 1280 ) ;
height = Math . Max ( int . Parse ( sizes [ 1 ] ) , 720 ) ;
2014-01-29 19:09:30 -08:00
}
2014-09-23 16:11:06 -07:00
new MatterControlApplication ( width , height ) ;
2014-01-29 19:09:30 -08:00
}
public override void OnClosed ( EventArgs e )
{
2014-06-11 14:52:58 -07:00
PrinterConnectionAndCommunication . Instance . Disable ( ) ;
2014-01-29 19:09:30 -08:00
//Close connection to the local datastore
Datastore . Instance . Exit ( ) ;
2014-06-11 14:52:58 -07:00
PrinterConnectionAndCommunication . Instance . HaltConnectionThread ( ) ;
2014-01-29 19:09:30 -08:00
SlicingQueue . Instance . ShutDownSlicingThread ( ) ;
2014-04-10 10:36:59 -07:00
if ( RestartOnClose )
{
string appPathAndFile = System . Reflection . Assembly . GetExecutingAssembly ( ) . Location ;
string pathToAppFolder = Path . GetDirectoryName ( appPathAndFile ) ;
ProcessStartInfo runAppLauncherStartInfo = new ProcessStartInfo ( ) ;
runAppLauncherStartInfo . Arguments = "\"{0}\" \"{1}\"" . FormatWith ( appPathAndFile , 1000 ) ;
runAppLauncherStartInfo . FileName = Path . Combine ( pathToAppFolder , "Launcher.exe" ) ;
runAppLauncherStartInfo . WindowStyle = ProcessWindowStyle . Hidden ;
runAppLauncherStartInfo . CreateNoWindow = true ;
Process . Start ( runAppLauncherStartInfo ) ;
}
2014-01-29 19:09:30 -08:00
base . OnClosed ( e ) ;
}
2014-06-10 09:25:38 -07:00
string unableToExitMessage = "Oops! You cannot exit while a print is active." . Localize ( ) ;
string unableToExitTitle = "Unable to Exit" . Localize ( ) ;
string savePartsSheetExitAnywayMessage = "You are currently saving a parts sheet, are you sure you want to exit?" . Localize ( ) ;
string confirmExit = "Confirm Exit" . Localize ( ) ;
2014-01-29 19:09:30 -08:00
public override void OnClosing ( out bool CancelClose )
{
2014-07-14 10:40:12 -07:00
// save the last size of the window so we can restore it next time.
ApplicationSettings . Instance . set ( "WindowSize" , string . Format ( "{0},{1}" , Width , Height ) ) ;
ApplicationSettings . Instance . set ( "DesktopPosition" , string . Format ( "{0},{1}" , DesktopPosition . x , DesktopPosition . y ) ) ;
2014-01-29 19:09:30 -08:00
//Save a snapshot of the prints in queue
2014-04-15 18:13:27 -07:00
QueueData . Instance . SaveDefaultQueue ( ) ;
2014-01-29 19:09:30 -08:00
2014-06-11 14:52:58 -07:00
if ( PrinterConnectionAndCommunication . Instance . PrinterIsPrinting )
2014-01-29 19:09:30 -08:00
{
2014-10-21 21:20:09 -07:00
StyledMessageBox . ShowMessageBox ( null , unableToExitMessage , unableToExitTitle ) ;
2014-01-29 19:09:30 -08:00
CancelClose = true ;
}
2014-02-27 14:56:20 -08:00
else if ( PartsSheet . IsSaving ( ) )
2014-01-29 19:09:30 -08:00
{
2014-10-21 21:20:09 -07:00
StyledMessageBox . ShowMessageBox ( onConfirmExit , savePartsSheetExitAnywayMessage , confirmExit , StyledMessageBox . MessageType . YES_NO ) ;
CancelClose = true ;
2014-01-29 19:09:30 -08:00
}
else
{
base . OnClosing ( out CancelClose ) ;
}
}
2014-10-21 21:20:09 -07:00
void onConfirmExit ( bool messageBoxResponse )
{
bool CancelClose ;
if ( messageBoxResponse )
{
base . OnClosing ( out CancelClose ) ;
}
}
2014-01-29 19:09:30 -08:00
}
}