2015-05-07 19:57:33 -07:00
/ *
Copyright ( c ) 2014 , Lars Brubaker
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 .
* /
2016-09-20 09:39:57 -07:00
using System ;
using System.Collections.Generic ;
using System.Diagnostics ;
using System.IO ;
using System.Linq ;
using System.Reflection ;
2016-11-05 10:56:52 -07:00
using System.Threading ;
2016-10-25 06:17:37 -07:00
using System.Threading.Tasks ;
2016-09-20 09:39:57 -07:00
using MatterHackers.Agg ;
2015-05-07 19:57:33 -07:00
using MatterHackers.Agg.Image ;
2015-09-04 11:28:01 -07:00
using MatterHackers.Agg.PlatformAbstract ;
2015-05-26 11:59:56 -07:00
using MatterHackers.Agg.UI ;
2015-08-18 13:01:29 -07:00
using MatterHackers.GuiAutomation ;
2015-08-25 11:56:21 -07:00
using MatterHackers.MatterControl.DataStorage ;
2017-06-02 21:20:56 -07:00
using MatterHackers.MatterControl.PrinterCommunication ;
2016-11-04 10:23:29 -07:00
using MatterHackers.MatterControl.SlicerConfiguration ;
2017-01-10 12:24:01 -08:00
using MatterHackers.PrinterEmulator ;
2015-10-22 18:23:22 -07:00
using Newtonsoft.Json ;
2016-10-19 13:27:27 -07:00
using Newtonsoft.Json.Converters ;
2015-08-22 16:14:31 -07:00
using NUnit.Framework ;
2015-05-07 19:57:33 -07:00
2016-05-11 09:13:56 -07:00
namespace MatterHackers.MatterControl.Tests.Automation
2015-05-07 19:57:33 -07:00
{
2016-05-11 09:13:56 -07:00
[TestFixture, Category("MatterControl.UI.Automation")]
2015-09-01 10:26:14 -07:00
public static class MatterControlUtilities
2015-05-07 19:57:33 -07:00
{
private static bool saveImagesForDebug = true ;
2016-11-05 10:56:52 -07:00
private static event EventHandler unregisterEvents ;
2015-12-22 11:34:04 -08:00
private static int testID = 0 ;
private static string runName = DateTime . Now . ToString ( "yyyy-MM-ddTHH-mm-ss" ) ;
2016-12-01 13:25:17 -08:00
public static void RemoveAllFromQueue ( this AutomationRunner testRunner )
2015-08-18 13:01:29 -07:00
{
2016-12-01 13:25:17 -08:00
testRunner . ClickByName ( "Queue... Menu" , 2 ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( 1 ) ;
2016-12-01 13:25:17 -08:00
testRunner . ClickByName ( " Remove All Menu Item" , 2 ) ;
2015-08-20 10:24:28 -07:00
}
2016-01-13 17:02:41 -08:00
public static void CreateDownloadsSubFolder ( )
{
2016-09-30 16:03:57 -07:00
Directory . CreateDirectory ( PathToDownloadsSubFolder ) ;
2016-01-13 17:02:41 -08:00
}
public static string PathToDownloadsSubFolder
2015-12-29 18:26:00 -08:00
{
get
{
2016-08-22 15:42:52 -07:00
return Path . Combine ( Environment . GetFolderPath ( Environment . SpecialFolder . UserProfile ) , "Downloads" , "-Temporary" ) ;
2015-12-29 18:26:00 -08:00
}
}
2016-09-30 16:03:57 -07:00
public static void DeleteDownloadsSubFolder ( )
2016-01-13 17:02:41 -08:00
{
2016-09-30 16:03:57 -07:00
Directory . Delete ( PathToDownloadsSubFolder , true ) ;
2016-01-13 17:02:41 -08:00
}
2016-09-19 09:38:20 -07:00
public static void SignOut ( AutomationRunner testRunner )
{
testRunner . ClickByName ( "User Options Menu" , 2 ) ;
testRunner . ClickByName ( "Sign Out Menu Item" , 2 ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( . 5 ) ;
2016-09-23 14:16:39 -07:00
2016-11-05 10:56:52 -07:00
// Rather than waiting a fixed amount of time, we wait for the ReloadAll to complete before returning
testRunner . WaitForReloadAll ( ( ) = > testRunner . ClickByName ( "Yes Button" ) ) ;
}
public static void WaitForReloadAll ( this AutomationRunner testRunner , Action reloadAllAction )
{
// Wire up a block and release mechanism to wait until the sign in process has completed
AutoResetEvent resetEvent = new AutoResetEvent ( false ) ;
ApplicationController . Instance . DoneReloadingAll . RegisterEvent ( ( s , e ) = > resetEvent . Set ( ) , ref unregisterEvents ) ;
// Start the procedure that begins a ReloadAll event in MatterControl
reloadAllAction ( ) ;
2016-12-06 18:52:25 -08:00
// Wait up to 10 seconds for the DoneReloadingAll event
resetEvent . WaitOne ( 10 * 1000 ) ;
2016-11-05 10:56:52 -07:00
// Remove our DoneReloadingAll listener
unregisterEvents ( null , null ) ;
// Wait for any post DoneReloadingAll code to finish up and return
2017-02-01 10:12:31 -08:00
testRunner . Delay ( . 2 ) ;
2016-09-19 09:38:20 -07:00
}
2016-01-13 17:02:41 -08:00
2015-12-31 12:57:00 -08:00
public static string PathToExportGcodeFolder
{
2016-09-22 08:52:15 -07:00
get { return TestContext . CurrentContext . ResolveProjectPath ( 4 , "Tests" , "TestData" , "ExportedGcode" , runName ) ; }
2015-12-31 12:57:00 -08:00
}
2016-01-13 17:02:41 -08:00
public static string GetTestItemPath ( string queueItemToLoad )
2015-10-23 09:57:26 -07:00
{
2016-09-22 08:52:15 -07:00
return TestContext . CurrentContext . ResolveProjectPath ( 4 , "Tests" , "TestData" , "QueueItems" , queueItemToLoad ) ;
2015-10-23 09:57:26 -07:00
}
2017-03-07 17:52:44 -08:00
public static void CloseMatterControlViaMenu ( this AutomationRunner testRunner )
2015-08-20 10:24:28 -07:00
{
SystemWindow mcWindowLocal = MatterControlApplication . Instance ;
2016-09-20 09:39:57 -07:00
testRunner . ClickByName ( "File Menu" , 5 ) ;
testRunner . ClickByName ( "Exit Menu Item" , 5 ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( . 2 ) ;
2015-08-20 10:24:28 -07:00
if ( mcWindowLocal . Parent ! = null )
{
mcWindowLocal . CloseOnIdle ( ) ;
}
2015-08-18 13:01:29 -07:00
}
2015-08-20 18:42:34 -07:00
2016-07-29 16:44:00 -07:00
public enum PrepAction
{
2016-08-29 17:13:45 -07:00
CloseSignInAndPrinterSelect ,
2016-07-29 16:44:00 -07:00
} ;
2017-03-15 16:17:06 -07:00
public static void Select3DPart ( this AutomationRunner testRunner , string partNameToSelect )
{
if ( testRunner . NameExists ( "3D View Edit" ) )
{
testRunner . ClickByName ( "3D View Edit" ) ;
}
testRunner . DragDropByName ( "centerPartPreviewAndControls" , "centerPartPreviewAndControls" , offsetDrop : new Agg . Point2D ( 10 , 15 ) , mouseButtons : MouseButtons . Right ) ;
testRunner . Delay ( 1 ) ;
testRunner . ClickByName ( partNameToSelect ) ;
}
2016-10-26 08:35:51 -07:00
public static void CloseSignInAndPrinterSelect ( this AutomationRunner testRunner , PrepAction preAction = PrepAction . CloseSignInAndPrinterSelect )
2016-07-29 16:44:00 -07:00
{
2016-10-26 08:35:51 -07:00
// Non-MCCentral builds won't have the plugin. Reduce the wait time for these cases
if ( testRunner . WaitForName ( "Connection Wizard Skip Sign In Button" , 0.5 ) )
2016-07-29 16:44:00 -07:00
{
2016-10-26 08:35:51 -07:00
testRunner . ClickByName ( "Connection Wizard Skip Sign In Button" ) ;
2016-07-29 16:44:00 -07:00
}
2016-10-26 08:35:51 -07:00
2016-12-01 13:25:17 -08:00
if ( testRunner . WaitForName ( "Cancel Wizard Button" , 1 ) )
{
testRunner . ClickByName ( "Cancel Wizard Button" ) ;
}
2016-07-29 16:44:00 -07:00
}
2017-05-21 16:29:00 -07:00
public static void ChangeToQueueContainer ( this AutomationRunner testRunner )
{
testRunner . NavigateToFolder ( "Print Queue Row Item Collection" ) ;
}
2016-10-26 07:27:16 -07:00
public class PrintEmulatorProcess : Process
{
protected override void Dispose ( bool disposing )
{
2016-12-07 13:21:53 -08:00
try
{
this . Kill ( ) ;
}
catch { }
2016-10-26 07:27:16 -07:00
base . Dispose ( disposing ) ;
}
}
2017-06-02 19:57:15 -07:00
public static Emulator LaunchAndConnectToPrinterEmulator ( this AutomationRunner testRunner , string make = "Airwolf 3D" , string model = "HD" , bool runSlow = false )
2016-09-14 15:50:24 -07:00
{
2016-09-20 09:39:57 -07:00
// Load the TestEnv config
var config = TestAutomationConfig . Load ( ) ;
2016-09-19 14:46:47 -07:00
2016-09-20 09:39:57 -07:00
// Create the printer
2017-06-04 09:01:56 -07:00
testRunner . AddAndSelectPrinter ( make , model ) ;
2016-09-19 14:46:47 -07:00
2017-06-02 19:57:15 -07:00
var emulator = new Emulator ( ) ;
2016-09-23 10:52:58 -07:00
2017-01-10 12:24:01 -08:00
emulator . PortName = config . Printer ;
emulator . RunSlow = runSlow ;
2016-09-14 15:50:24 -07:00
2017-01-10 12:24:01 -08:00
emulator . Startup ( ) ;
2016-09-14 15:50:24 -07:00
2016-09-19 14:46:47 -07:00
// edit the com port
2017-01-03 10:45:16 -08:00
SystemWindow containingWindow ;
var editButton = testRunner . GetWidgetByName ( "Edit Printer Button" , out containingWindow ) ;
2017-01-06 16:54:34 -08:00
2017-02-01 10:12:31 -08:00
testRunner . Delay ( ( ) = > editButton . Enabled , 5 ) ; // Wait until the edit button is ready to click it. Ensures the printer is loaded.
2016-12-12 17:33:02 -08:00
testRunner . ClickByName ( "Edit Printer Button" , 3 ) ;
2016-09-19 14:46:47 -07:00
2016-12-12 17:33:02 -08:00
testRunner . ClickByName ( "Serial Port Dropdown" , 3 ) ;
2016-09-19 14:46:47 -07:00
2016-12-19 11:59:49 -08:00
testRunner . ClickByName ( config . MCPort + " Menu Item" , 5 ) ;
2016-09-19 14:46:47 -07:00
testRunner . ClickByName ( "Cancel Wizard Button" ) ;
// connect to the created printer
testRunner . ClickByName ( "Connect to printer button" , 2 ) ;
2016-12-12 17:33:02 -08:00
testRunner . WaitForName ( "Disconnect from printer button" , 5 ) ;
2016-09-19 14:46:47 -07:00
2017-01-10 17:20:24 -08:00
return emulator ;
2017-01-10 13:43:12 -08:00
}
2016-09-14 15:50:24 -07:00
2017-02-01 10:20:50 -08:00
public static void CancelPrint ( this AutomationRunner testRunner )
{
testRunner . ClickByName ( "Cancel Print Button" ) ;
if ( testRunner . WaitForName ( "Yes Button" , 1 ) )
{
testRunner . ClickByName ( "Yes Button" ) ;
}
}
2016-03-03 16:57:43 -08:00
public static bool CompareExpectedSliceSettingValueWithActualVaue ( string sliceSetting , string expectedValue )
{
2016-09-23 13:34:57 -07:00
string fullPath = TestContext . CurrentContext . ResolveProjectPath ( 4 , "Tests" , "temp" , runName , "Test0" , "data" , "gcode" ) ;
2016-03-03 16:57:43 -08:00
2016-11-04 10:23:29 -07:00
foreach ( string iniPath in Directory . GetFiles ( fullPath , "*.ini" ) )
2016-03-03 16:57:43 -08:00
{
2016-11-04 10:23:29 -07:00
var settings = PrinterSettingsLayer . LoadFromIni ( iniPath ) ;
string currentValue ;
2016-03-03 16:57:43 -08:00
2016-11-04 10:23:29 -07:00
if ( settings . TryGetValue ( sliceSetting , out currentValue ) )
{
return currentValue . Trim ( ) = = expectedValue ;
2016-03-03 16:57:43 -08:00
}
}
return false ;
}
2016-09-08 14:20:39 -07:00
public static void DeleteSelectedPrinter ( AutomationRunner testRunner )
{
// delete printer
testRunner . ClickByName ( "Edit Printer Button" , 5 ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( . 5 ) ;
2016-11-05 11:12:20 -07:00
2016-09-08 14:20:39 -07:00
testRunner . ClickByName ( "Delete Printer Button" , 5 ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( . 5 ) ;
2016-11-05 11:12:20 -07:00
testRunner . WaitForReloadAll ( ( ) = > testRunner . ClickByName ( "Yes Button" , 5 ) ) ;
2016-09-08 14:20:39 -07:00
}
2017-06-04 09:01:56 -07:00
public static void AddAndSelectPrinter ( this AutomationRunner testRunner , string make , string model )
2016-01-13 10:26:17 -08:00
{
2017-06-04 16:16:07 -07:00
// If SelectMake is not visible and the ConnectionWizard is, click Skip
if ( ! testRunner . NameExists ( "Select Make" )
& & testRunner . WaitForName ( "Connection Wizard Skip Sign In Button" , 1 ) )
{
testRunner . ClickByName ( "Connection Wizard Skip Sign In Button" ) ;
}
2017-06-03 08:20:09 -07:00
if ( ! testRunner . WaitForName ( "Select Make" , 1 ) )
2016-12-12 17:33:02 -08:00
{
2017-06-03 08:20:09 -07:00
// TODO: The overflow menu needs to always be on screen and when there's not enough room siblings should be removed from the actions bar and pushed into the overflow menu, rather than the menu clipping from the screen
2017-06-04 16:16:07 -07:00
testRunner . OpenPrintersDropdown ( ) ;
2016-12-12 17:33:02 -08:00
testRunner . ClickByName ( "Add New Printer... Menu Item" , 5 , delayBeforeReturn : . 5 ) ;
}
2016-11-04 10:23:29 -07:00
2016-12-12 17:33:02 -08:00
testRunner . ClickByName ( "Select Make" , 5 ) ;
2016-09-26 15:59:29 -07:00
testRunner . Type ( make ) ;
2016-10-17 11:04:39 -07:00
testRunner . Type ( "{Enter}" ) ;
2016-01-13 10:26:17 -08:00
2016-12-12 17:33:02 -08:00
testRunner . ClickByName ( "Select Model" , 5 ) ;
testRunner . Type ( model ) ;
testRunner . Type ( "{Enter}" ) ;
2016-08-29 17:13:45 -07:00
2016-11-15 15:02:34 -08:00
// An unpredictable period of time will pass between Clicking Save, everything reloading and us returning to the caller.
// Block until ReloadAll has completed then close and return to the caller, at which point hopefully everything is reloaded.
WaitForReloadAll ( testRunner , ( ) = > testRunner . ClickByName ( "Save & Continue Button" , 2 ) ) ;
2016-08-29 17:13:45 -07:00
2016-12-12 17:33:02 -08:00
testRunner . ClickByName ( "Cancel Wizard Button" , 5 ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( 1 ) ;
2016-01-13 10:26:17 -08:00
}
2017-06-04 16:16:07 -07:00
public static void OpenPrintersDropdown ( this AutomationRunner testRunner )
{
testRunner . ClickByName ( "Printer Overflow Menu" ) ;
testRunner . ClickByName ( "Printers... Menu" ) ;
}
public static void ClosePrintersDropdown ( this AutomationRunner testRunner )
{
testRunner . ClickByName ( "Printer Overflow Menu" ) ;
// If a sub menu is open the first click will close it but not the main menu. Second click as needed
if ( testRunner . WaitForName ( "Printers... Menu" , 0.5 ) )
{
testRunner . ClickByName ( "Printer Overflow Menu" ) ;
}
}
2015-09-01 10:26:14 -07:00
private static void OutputImage ( ImageBuffer imageToOutput , string fileName )
2015-05-07 19:57:33 -07:00
{
if ( saveImagesForDebug )
{
ImageTgaIO . Save ( imageToOutput , fileName ) ;
}
}
2015-09-01 10:26:14 -07:00
private static void OutputImage ( GuiWidget widgetToOutput , string fileName )
2015-05-07 19:57:33 -07:00
{
if ( saveImagesForDebug )
{
OutputImage ( widgetToOutput . BackBuffer , fileName ) ;
}
}
2015-09-01 10:26:14 -07:00
private static void OutputImages ( GuiWidget control , GuiWidget test )
2015-05-07 19:57:33 -07:00
{
OutputImage ( control , "image-control.tga" ) ;
OutputImage ( test , "image-test.tga" ) ;
}
2015-08-25 11:56:21 -07:00
2015-12-22 11:34:04 -08:00
/// <summary>
/// Overrides the AppData location, ensuring each test starts with a fresh MatterControl database.
/// </summary>
2016-09-20 09:39:57 -07:00
public static void OverrideAppDataLocation ( string matterControlDirectory )
2015-08-25 11:56:21 -07:00
{
2016-09-20 14:32:22 -07:00
string tempFolderPath = Path . Combine ( matterControlDirectory , "Tests" , "temp" , runName , $"Test{testID++}" ) ;
2016-09-20 09:39:57 -07:00
ApplicationDataStorage . Instance . OverrideAppDataLocation ( tempFolderPath ) ;
2015-08-25 11:56:21 -07:00
}
2015-10-22 18:23:22 -07:00
public static void AddItemsToQueue ( string queueItemFolderToLoad )
{
2016-09-23 14:16:39 -07:00
// Default location of mcp file
2015-10-22 18:23:22 -07:00
string mcpPath = Path . Combine ( ApplicationDataStorage . ApplicationUserDataPath , "data" , "default.mcp" ) ;
Directory . CreateDirectory ( Path . GetDirectoryName ( mcpPath ) ) ;
if ( ! File . Exists ( mcpPath ) )
{
File . WriteAllText ( mcpPath , JsonConvert . SerializeObject ( new ManifestFile ( )
2016-09-20 14:32:22 -07:00
{
ProjectFiles = new System . Collections . Generic . List < PrintItem > ( )
} , Formatting . Indented ) ) ;
2015-10-22 18:23:22 -07:00
}
var queueItemData = JsonConvert . DeserializeObject < ManifestFile > ( File . ReadAllText ( mcpPath ) ) ;
string queueData = Path . Combine ( ApplicationDataStorage . ApplicationUserDataPath , "data" , "testitems" ) ;
2016-09-23 14:16:39 -07:00
// Create empty TestParts folder
2015-10-22 18:23:22 -07:00
Directory . CreateDirectory ( queueData ) ;
2016-09-22 08:52:15 -07:00
string queueItemsDirectory = TestContext . CurrentContext . ResolveProjectPath ( 5 , "MatterControl" , "Tests" , "TestData" , "QueueItems" , queueItemFolderToLoad ) ;
2015-12-22 11:34:04 -08:00
2016-09-22 08:52:15 -07:00
foreach ( string file in Directory . GetFiles ( queueItemsDirectory ) )
2015-10-22 18:23:22 -07:00
{
string newFilePath = Path . Combine ( queueData , Path . GetFileName ( file ) ) ;
File . Copy ( file , newFilePath , true ) ;
queueItemData . ProjectFiles . Add ( new PrintItem ( )
2016-09-22 08:52:15 -07:00
{
FileLocation = newFilePath ,
Name = Path . GetFileNameWithoutExtension ( file ) ,
DateAdded = DateTime . Now
} ) ;
2015-10-22 18:23:22 -07:00
}
File . WriteAllText ( mcpPath , JsonConvert . SerializeObject ( queueItemData , Formatting . Indented ) ) ;
Assert . IsTrue ( queueItemData ! = null & & queueItemData . ProjectFiles . Count > 0 ) ;
}
2016-12-01 13:25:17 -08:00
public static void NavigateToFolder ( this AutomationRunner testRunner , string libraryRowItemName )
2015-09-01 16:03:29 -07:00
{
2016-12-01 13:25:17 -08:00
testRunner . ClickByName ( libraryRowItemName ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( . 5 ) ;
2017-05-20 17:47:36 -07:00
testRunner . DoubleClickByName ( libraryRowItemName ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( . 5 ) ;
2015-09-01 16:03:29 -07:00
}
2017-06-03 19:08:14 -07:00
public static void AddDefaultFileToBedplate ( this AutomationRunner testRunner , string containerName = "Calibration Parts Row Item Collection" , string partName = "Row Item Calibration - Box.stl" )
2017-06-02 19:40:10 -07:00
{
testRunner . ClickByName ( "Library Tab" ) ;
testRunner . NavigateToFolder ( containerName ) ;
2017-06-03 19:08:14 -07:00
testRunner . ClickByName ( partName ) ;
2017-06-02 19:40:10 -07:00
2017-06-04 08:57:17 -07:00
testRunner . AddSelectedItemToBedplate ( ) ;
2017-06-05 09:27:30 -07:00
testRunner . Delay ( 1 ) ;
2017-06-04 08:57:17 -07:00
}
2017-06-06 18:19:02 -07:00
public static void SaveBedplateToFolder ( this AutomationRunner testRunner , string newFileName , string folderName )
{
testRunner . ClickByName ( "Save As Menu" ) ;
testRunner . ClickByName ( "Save As Menu Item" ) ;
testRunner . Delay ( 1 ) ;
testRunner . Type ( newFileName ) ;
testRunner . NavigateToFolder ( folderName ) ;
testRunner . ClickByName ( "Save As Save Button" ) ;
// Give the SaveAs window time to close before returning to the caller
testRunner . Delay ( 2 ) ;
}
2017-06-04 08:57:17 -07:00
public static void AddSelectedItemToBedplate ( this AutomationRunner testRunner )
{
2017-06-03 19:08:14 -07:00
testRunner . ClickByName ( "Print Library Overflow Menu" ) ;
2017-06-03 13:29:36 -07:00
testRunner . ClickByName ( "Add to Plate Menu Item" ) ;
2017-06-02 19:40:10 -07:00
}
2017-06-02 21:20:56 -07:00
public static void WaitForPrintFinished ( this AutomationRunner testRunner )
{
2017-06-05 09:27:30 -07:00
testRunner . Delay ( ( ) = > PrinterConnectionAndCommunication . Instance . CommunicationState = = PrinterConnectionAndCommunication . CommunicationStates . FinishedPrint , 500 ) ;
2017-06-02 21:20:56 -07:00
}
2016-10-25 06:17:37 -07:00
public static async Task RunTest (
AutomationTest testMethod ,
2016-09-20 14:32:22 -07:00
string staticDataPathOverride = null ,
double maxTimeToRun = 60 ,
2016-08-02 09:47:26 -07:00
QueueTemplate queueItemFolderToAdd = QueueTemplate . None ,
2016-10-07 13:49:01 -07:00
int overrideWidth = - 1 ,
int overrideHeight = - 1 ,
string defaultTestImages = null )
2015-09-04 11:28:01 -07:00
{
2016-10-17 11:02:40 -07:00
// Walk back a step in the stack and output the callers name
StackTrace st = new StackTrace ( false ) ;
Debug . WriteLine ( "\r\n ***** Running automation test: {0} {1} " , st . GetFrames ( ) . Skip ( 1 ) . First ( ) . GetMethod ( ) . Name , DateTime . Now ) ;
2015-09-04 11:28:01 -07:00
if ( staticDataPathOverride = = null )
{
2016-09-20 09:39:57 -07:00
// Popping one directory above MatterControl, then back down into MatterControl ensures this works in MCCentral as well and MatterControl
staticDataPathOverride = TestContext . CurrentContext . ResolveProjectPath ( 5 , "MatterControl" , "StaticData" ) ;
2015-09-04 11:28:01 -07:00
}
2015-12-22 11:34:04 -08:00
2016-10-17 11:02:40 -07:00
#if DEBUG
string outputDirectory = "Debug" ;
#else
string outputDirectory = "Release" ;
#endif
2016-10-19 14:17:09 -07:00
Environment . CurrentDirectory = TestContext . CurrentContext . ResolveProjectPath ( 5 , "MatterControl" , "bin" , outputDirectory ) ;
2016-10-17 11:02:40 -07:00
2015-09-04 11:28:01 -07:00
#if ! __ANDROID__
// Set the static data to point to the directory of MatterControl
2016-09-20 09:39:57 -07:00
StaticData . Instance = new FileSystemStaticData ( staticDataPathOverride ) ;
2015-09-04 11:28:01 -07:00
#endif
2016-09-20 09:39:57 -07:00
// Popping one directory above MatterControl, then back down into MatterControl ensures this works in MCCentral as well and MatterControl
MatterControlUtilities . OverrideAppDataLocation ( TestContext . CurrentContext . ResolveProjectPath ( 5 , "MatterControl" ) ) ;
2015-10-22 18:23:22 -07:00
2015-12-22 11:34:04 -08:00
if ( queueItemFolderToAdd ! = QueueTemplate . None )
2015-10-22 18:23:22 -07:00
{
2017-06-04 17:39:30 -07:00
MatterControlUtilities . AddItemsToQueue ( queueItemFolderToAdd . ToString ( ) ) ;
2015-10-22 18:23:22 -07:00
}
2015-09-04 11:28:01 -07:00
2016-10-19 11:49:52 -07:00
if ( defaultTestImages = = null )
{
defaultTestImages = TestContext . CurrentContext . ResolveProjectPath ( 4 , "Tests" , "TestData" , "TestImages" ) ;
}
2016-12-02 10:52:33 -08:00
UserSettings . Instance . set ( UserSettingsKey . ThumbnailRenderingMode , "orthographic" ) ;
2016-12-07 15:24:48 -08:00
//GL.HardwareAvailable = false;
2016-12-02 10:52:33 -08:00
MatterControlApplication matterControlWindow = MatterControlApplication . CreateInstance ( overrideWidth , overrideHeight ) ;
2016-10-19 13:27:27 -07:00
var config = TestAutomationConfig . Load ( ) ;
2016-12-13 10:20:12 -08:00
// Extract mouse speed from config
AutomationRunner . TimeToMoveMouse = config . TimeToMoveMouse ;
2016-10-25 06:17:37 -07:00
await AutomationRunner . ShowWindowAndExecuteTests ( matterControlWindow , testMethod , maxTimeToRun , defaultTestImages , config . AutomationInputType ) ;
2015-09-04 11:28:01 -07:00
}
2016-08-01 13:53:28 -07:00
public static void LibraryAddSelectionToQueue ( AutomationRunner testRunner )
{
2017-06-03 13:29:36 -07:00
testRunner . ClickByName ( "Print Library Overflow Menu" ) ;
2016-08-01 13:53:28 -07:00
testRunner . ClickByName ( "Add to Queue Menu Item" , 1 ) ;
}
2016-08-01 14:22:16 -07:00
public static void LibraryEditSelectedItem ( AutomationRunner testRunner )
{
testRunner . ClickByName ( "Edit Menu Item" , 1 ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( 1 ) ; // wait for the new window to open
2016-08-01 14:22:16 -07:00
}
2016-08-01 17:21:31 -07:00
2017-06-03 13:29:36 -07:00
public static void LibraryRenameSelectedItem ( this AutomationRunner testRunner )
2016-08-01 17:21:31 -07:00
{
2017-06-03 13:29:36 -07:00
testRunner . ClickByName ( "Print Library Overflow Menu" ) ;
2016-08-01 17:21:31 -07:00
testRunner . ClickByName ( "Rename Menu Item" , 1 ) ;
}
2017-06-06 18:52:13 -07:00
public static void LibraryRemoveSelectedItem ( this AutomationRunner testRunner )
2016-08-01 17:21:31 -07:00
{
2017-06-03 13:29:36 -07:00
testRunner . ClickByName ( "Print Library Overflow Menu" ) ;
2016-08-01 17:21:31 -07:00
testRunner . ClickByName ( "Remove Menu Item" , 1 ) ;
}
2017-06-03 13:29:36 -07:00
2016-09-20 09:39:57 -07:00
public static string ResolveProjectPath ( this TestContext context , int stepsToProjectRoot , params string [ ] relativePathSteps )
{
string assemblyPath = Path . GetDirectoryName ( Assembly . GetExecutingAssembly ( ) . Location ) ;
var allPathSteps = new List < string > { assemblyPath } ;
allPathSteps . AddRange ( Enumerable . Repeat ( ".." , stepsToProjectRoot ) ) ;
if ( relativePathSteps . Any ( ) )
{
allPathSteps . AddRange ( relativePathSteps ) ;
}
return Path . GetFullPath ( Path . Combine ( allPathSteps . ToArray ( ) ) ) ;
}
2016-09-21 15:34:53 -07:00
/// <summary>
/// Set the working directory to the location of the executing assembly. This is essentially the Nunit2 behavior
/// </summary>
/// <param name="context"></param>
public static void SetCompatibleWorkingDirectory ( this TestContext context )
{
Environment . CurrentDirectory = Path . GetDirectoryName ( Assembly . GetExecutingAssembly ( ) . Location ) ;
}
2016-09-20 14:32:22 -07:00
2017-06-02 19:39:29 -07:00
public static void SwitchToAdvancedSliceSettings ( this AutomationRunner testRunner )
2016-09-20 14:32:22 -07:00
{
2017-06-02 19:39:29 -07:00
// Switch to Slice Settings Tab
testRunner . ClickByName ( "Slice Settings Tab" ) ;
2017-06-09 20:12:25 -07:00
// Show the overflow menu
testRunner . ClickByName ( "Slice Settings Overflow Menu" ) ;
2017-06-02 19:39:29 -07:00
// Change to Advanced view
2016-12-20 17:44:27 -08:00
testRunner . ClickByName ( "User Level Dropdown" ) ;
testRunner . ClickByName ( "Advanced Menu Item" ) ;
2017-02-01 10:12:31 -08:00
testRunner . Delay ( . 5 ) ;
2016-09-20 14:32:22 -07:00
}
2017-06-04 08:22:04 -07:00
/// <summary>
/// Adds the given asset names to the local library and validates the result
/// </summary>
/// <param name="testRunner"></param>
/// <param name="assetNames">The test assets to add to the library</param>
public static void AddTestAssetsToLibrary ( this AutomationRunner testRunner , params string [ ] assetNames )
{
// Switch to the Local Library tab
testRunner . ClickByName ( "Library Tab" ) ;
testRunner . NavigateToFolder ( "Local Library Row Item Collection" ) ;
// Assert that the requested items are not currently in the list
foreach ( string assetName in assetNames )
{
string friendlyName = Path . GetFileNameWithoutExtension ( assetName ) ;
Assert . IsFalse ( testRunner . WaitForName ( $"Row Item {friendlyName}" , 1 ) , $"{friendlyName} part should not exist at test start" ) ;
}
// Generate the full, quoted paths for the requested assets
string fullQuotedAssetPaths = string . Join ( " " , assetNames . Select ( name = > $"\" { MatterControlUtilities . GetTestItemPath ( name ) } \ "" ) ) ;
// Add Library item
testRunner . ClickByName ( "Library Add Button" ) ;
testRunner . Delay ( 2 ) ;
testRunner . Type ( fullQuotedAssetPaths ) ;
testRunner . Delay ( 1 ) ;
testRunner . Type ( "{Enter}" ) ;
// Assert that the requested items are not currently in the list
foreach ( string assetName in assetNames )
{
string friendlyName = Path . GetFileNameWithoutExtension ( assetName ) ;
Assert . IsTrue ( testRunner . WaitForName ( $"Row Item {friendlyName}" , 2 ) , $"{friendlyName} part should exist after adding" ) ;
}
}
2017-06-06 18:42:44 -07:00
/// <summary>
/// Control clicks each specified item
/// </summary>
/// <param name="testRunner"></param>
/// <param name="widgetNames">The widgets to click</param>
public static void SelectListItems ( this AutomationRunner testRunner , params string [ ] widgetNames )
{
// Control click all items
Keyboard . SetKeyDownState ( Keys . ControlKey , down : true ) ;
foreach ( var widgetName in widgetNames )
{
testRunner . ClickByName ( widgetName ) ;
}
Keyboard . SetKeyDownState ( Keys . ControlKey , down : false ) ;
}
2016-09-20 14:32:22 -07:00
}
2015-12-22 11:34:04 -08:00
/// <summary>
/// Represents a queue template folder on disk (located at Tests/TestData/QueueItems) that should be synced into the default
/// queue during test init. The enum name and folder name *must* be the same in order to function
/// </summary>
public enum QueueTemplate
{
None ,
Three_Queue_Items
}
2016-09-20 09:39:57 -07:00
public class TestAutomationConfig
{
private static readonly string configPath = Path . Combine ( Environment . GetFolderPath ( Environment . SpecialFolder . UserProfile ) , "MHTest.config" ) ;
/// <summary>
/// The ClientToken used by tests to emulate an external client
/// </summary>
public string TestEnvClientToken { get ; set ; }
/// <summary>
/// The serial port that MatterControl will communicate with for the Com0Com connection
/// </summary>
public string MCPort { get ; set ; }
/// <summary>
/// The serial port that Python will communicate with to emulate printer firmware
/// </summary>
public string Printer { get ; set ; }
2016-10-19 13:27:27 -07:00
[JsonConverter(typeof(StringEnumConverter))]
public AutomationRunner . InputType AutomationInputType { get ; set ; } = AutomationRunner . InputType . Native ;
2016-12-13 10:20:12 -08:00
/// <summary>
/// The number of seconds to move the mouse when going to a new position.
/// </summary>
public double TimeToMoveMouse { get ; set ; } = . 5 ;
2016-09-20 09:39:57 -07:00
public static TestAutomationConfig Load ( )
{
TestAutomationConfig config = null ;
if ( ! File . Exists ( configPath ) )
{
config = new TestAutomationConfig ( ) ;
config . Save ( ) ;
}
else
{
config = JsonConvert . DeserializeObject < TestAutomationConfig > ( File . ReadAllText ( configPath ) ) ;
}
// if no com port set, issue instructions on how to set it
if ( string . IsNullOrEmpty ( config . MCPort ) | | string . IsNullOrEmpty ( config . Printer ) )
{
throw new Exception ( "You must set the port and printer in: " + configPath ) ;
}
return config ;
}
/// <summary>
/// Persist the current settings to the 'MHTest.config' in the user profile - %userprofile%\MHTest.config
/// </summary>
public void Save ( )
{
File . WriteAllText ( configPath , JsonConvert . SerializeObject ( this , Formatting . Indented ) ) ;
}
}
2015-05-07 19:57:33 -07:00
}