using Microsoft.Win32; using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.ComponentModel; using System.IO; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Windows.Data; using Tango.BL; using Tango.BL.Builders; using Tango.BL.Calibration; using Tango.BL.Entities; using Tango.Core.Commands; using Tango.MachineStudio.Common; using Tango.MachineStudio.Common.Notifications; using Tango.MachineStudio.RML.Contracts; using Tango.MachineStudio.RML.Models; using Tango.MachineStudio.RML.Views; using Tango.PMR.ColorLab; using System.Data.Entity; using Tango.Core.ExtensionMethods; using Tango.MachineStudio.Common.Authentication; using Tango.BL.ActionLogs; using Tango.BL.DTO; using Tango.BL.Enumerations; namespace Tango.MachineStudio.RML.ViewModels { public class MainViewVM : StudioViewModel { private INotificationProvider _notification; private IAuthenticationProvider _authentication; private IActionLogManager _actionLogManager; private RmlDTO _rmlBeforeSave; private ObservablesContext _rmls_context; private ObservablesContext _active_context; private ObservableCollection _rmls; public ObservableCollection Rmls { get { return _rmls; } set { _rmls = value; RaisePropertyChangedAuto(); } } private ObservableCollection _materials; public ObservableCollection Materials { get { return _materials; } set { _materials = value; RaisePropertyChangedAuto(); } } private ObservableCollection _purposes; public ObservableCollection Purposes { get { return _purposes; } set { _purposes = value; RaisePropertyChangedAuto(); } } private ObservableCollection _conditions; public ObservableCollection Conditions { get { return _conditions; } set { _conditions = value; RaisePropertyChangedAuto(); } } private ObservableCollection _linearMassDensityUnits; public ObservableCollection LinearMassDensityUnits { get { return _linearMassDensityUnits; } set { _linearMassDensityUnits = value; RaisePropertyChangedAuto(); } } private ObservableCollection _fiberShapes; public ObservableCollection FiberShapes { get { return _fiberShapes; } set { _fiberShapes = value; RaisePropertyChangedAuto(); } } private ObservableCollection _fiberSynths; public ObservableCollection FiberSynths { get { return _fiberSynths; } set { _fiberSynths = value; RaisePropertyChangedAuto(); } } private ObservableCollection _spoolTypes; public ObservableCollection SpoolTypes { get { return _spoolTypes; } set { _spoolTypes = value; RaisePropertyChangedAuto(); } } private Rml _selectedRML; public Rml SelectedRML { get { return _selectedRML; } set { _selectedRML = value; RaisePropertyChangedAuto(); InvalidateRelayCommands(); } } private Rml _activeRML; public Rml ActiveRML { get { return _activeRML; } set { _activeRML = value; RaisePropertyChangedAuto(); } } private CalibrationDataViewVM _calibrationDataViewVM; public CalibrationDataViewVM CalibrationDataViewVM { get { return _calibrationDataViewVM; } set { _calibrationDataViewVM = value; RaisePropertyChangedAuto(); } } private ObservableCollection _liquidTypesRmls; public ObservableCollection LiquidTypesRmls { get { return _liquidTypesRmls; } set { _liquidTypesRmls = value; RaisePropertyChangedAuto(); } } private ProcessParametersTablesGroup _activeProcessParametersGroup; public ProcessParametersTablesGroup ActiveProcessParametersGroup { get { return _activeProcessParametersGroup; } set { _activeProcessParametersGroup = value; RaisePropertyChangedAuto(); } } private ICollectionView _activeProcessParametersTableView; public ICollectionView ActiveProcessParametersTableView { get { return _activeProcessParametersTableView; } set { _activeProcessParametersTableView = value; RaisePropertyChangedAuto(); } } private ObservableCollection _ccts; public ObservableCollection CCTS { get { return _ccts; } set { _ccts = value; RaisePropertyChangedAuto(); } } private CctModel _selectedCCT; public CctModel SelectedCCT { get { return _selectedCCT; } set { _selectedCCT = value; RaisePropertyChangedAuto(); OnSelectedCCTChanged(); InvalidateRelayCommands(); } } private ColorConversionViewVM _colorConversionViewVM; public ColorConversionViewVM ColorConversionViewVM { get { return _colorConversionViewVM; } set { _colorConversionViewVM = value; RaisePropertyChangedAuto(); } } private RmlsSpool _selectedSpool; public RmlsSpool SelectedSpool { get { return _selectedSpool; } set { _selectedSpool = value; RaisePropertyChangedAuto(); InvalidateRelayCommands(); } } /// /// Gets or sets the manage RML command. /// public RelayCommand ManageRmlCommand { get; set; } /// /// Gets or sets the add RML command. /// public RelayCommand AddRmlCommand { get; set; } /// /// Gets or sets the remove RML command. /// public RelayCommand RemoveRmlCommand { get; set; } public RelayCommand ImportForwardDataCommand { get; set; } public RelayCommand ExportForwardDataCommand { get; set; } public RelayCommand AddProcessParametersTableCommand { get; set; } public RelayCommand RemoveProcessParametersTableCommand { get; set; } public RelayCommand AddLiquidFactorCommand { get; set; } public RelayCommand RemoveLiquidFactorCommand { get; set; } public RelayCommand CreateCalibrationDataExcelTemplateCommand { get; set; } /// /// Gets or sets the back to RMLS command. /// public RelayCommand BackToRmlsCommand { get; set; } public RelayCommand SaveCommand { get; set; } public RelayCommand CloneRmlCommand { get; set; } public RelayCommand ExportRMLFileCommand { get; set; } public RelayCommand ImportRMLFileCommand { get; set; } /// /// Gets or sets the add spool command. /// public RelayCommand AddSpoolCommand { get; set; } /// /// Gets or sets the remove spool command. /// public RelayCommand RemoveSpoolCommand { get; set; } public MainViewVM(INotificationProvider notificationProvider, IAuthenticationProvider authentication, IActionLogManager actionLogManager) { _notification = notificationProvider; _authentication = authentication; _actionLogManager = actionLogManager; ManageRmlCommand = new RelayCommand(() => LoadActiveRML(SelectedRML.Guid), () => SelectedRML != null); RemoveRmlCommand = new RelayCommand(RemoveSelectedRml, () => SelectedRML != null); CloneRmlCommand = new RelayCommand(CloneSelectedRml, () => SelectedRML != null); AddRmlCommand = new RelayCommand(AddNewRml); BackToRmlsCommand = new RelayCommand(BackToRmls, () => IsFree); AddProcessParametersTableCommand = new RelayCommand(AddProcessParametersTable, () => IsFree); RemoveProcessParametersTableCommand = new RelayCommand(RemoveProcessParametersTable, () => IsFree); AddLiquidFactorCommand = new RelayCommand(AddLiquidFactor, () => IsFree); RemoveLiquidFactorCommand = new RelayCommand(RemoveLiquidFactor, () => IsFree); CreateCalibrationDataExcelTemplateCommand = new RelayCommand(CreateCalibrationDataExcelTemplate); SaveCommand = new RelayCommand(Save, () => IsFree); ImportForwardDataCommand = new RelayCommand(ImportCCTData, () => ActiveRML != null && IsFree); ExportForwardDataCommand = new RelayCommand(ExportCCTData, () => ActiveRML != null && SelectedCCT != null && IsFree); ExportRMLFileCommand = new RelayCommand(ExportRmlFile, () => SelectedRML != null && IsFree); ImportRMLFileCommand = new RelayCommand(ImportRmlFile, () => IsFree); AddSpoolCommand = new RelayCommand(AddNewSpool); RemoveSpoolCommand = new RelayCommand(RemoveSpool, () => SelectedSpool != null); } public override void OnApplicationReady() { LoadRmls(); } private async void LoadRmls() { if (_rmls_context != null) _rmls_context.Dispose(); _rmls_context = ObservablesContext.CreateDefault(); Rmls = await new RmlsCollectionBuilder(_rmls_context).SetAll().WithLiquidFactors().WithMediaProperties().BuildAsync(); //Load CCT file names... var ccts = await _rmls_context.Ccts.Select(x => new { x.Guid, x.FileName }).ToListAsync(); foreach (var rml in Rmls) { var cct = ccts.SingleOrDefault(x => x.Guid == rml.CctGuid); if (cct != null) { rml.Cct = new Cct() { Guid = cct.Guid, FileName = cct.FileName, }; } } } private async void LoadActiveRML(String guid) { using (_notification.PushTaskItem("Loading RML...")) { try { IsFree = false; if (_active_context != null) { _active_context.Dispose(); } _active_context = ObservablesContext.CreateDefault(); CCTS = _active_context.Ccts .Select(x => new CctModel() { Guid = x.Guid, FileName = x.FileName, }).ToObservableCollection(); CCTS.Where(x => String.IsNullOrWhiteSpace(x.FileName)).ToList().ForEach(x => x.FileName = x.Guid); LoadRmlProperties(); ActiveRML = await new RmlBuilder(_active_context) .Set(guid) .WithActiveParametersGroup() .WithLiquidFactors() .WithCCT() .WithSpools() .BuildAsync(); if (ActiveRML.Cct != null) { SelectedCCT = CCTS.SingleOrDefault(x => x.Guid == ActiveRML.Cct.Guid); } if (ActiveRML.ProcessParametersTablesGroups.ToList().Count == 0) { if (!_notification.ShowQuestion("Could not find any process group for the selected RML. Would you like to create one?")) { _notification.ShowError("Cannot load an RML with no process group."); IsFree = true; return; } else { ProcessParametersTablesGroup group = new ProcessParametersTablesGroup(); group.Name = "Active Group"; group.Active = true; group.ProcessParametersTables.Add(new ProcessParametersTable() { Name = "Process Table 1", }); group.Rml = ActiveRML; _active_context.ProcessParametersTablesGroups.Add(group); _active_context.ProcessParametersTables.Add(group.ProcessParametersTables[0]); await _active_context.SaveChangesAsync(); LoadActiveRML(ActiveRML.Guid); return; } } ActiveProcessParametersGroup = ActiveRML.ProcessParametersTablesGroups.ToList().FirstOrDefault(); ActiveProcessParametersTableView = CollectionViewSource.GetDefaultView(ActiveProcessParametersGroup.ProcessParametersTables); ActiveProcessParametersTableView.SortDescriptions.Add(new SortDescription(nameof(ProcessParametersTable.TableIndex), ListSortDirection.Ascending)); CalibrationDataViewVM = new CalibrationDataViewVM(_notification); LiquidTypesRmls = ActiveRML.LiquidTypesRmls; foreach (var liquidTypeRml in LiquidTypesRmls) { CalibrationDataVM catVM = new CalibrationDataVM(liquidTypeRml.LiquidType); if (liquidTypeRml.DefaultCatData != null) { catVM.CalibrationPoints = liquidTypeRml.GetCalibrationData().CalibrationPoints.Select(x => new CalibrationDataPointVM(x.X, x.Y)).ToObservableCollection(); } CalibrationDataViewVM.LiquidsCalibrationData.Add(catVM); } ColorConversionViewVM = new ColorConversionViewVM(_notification) { RML = ActiveRML, CCT = SelectedCCT, LiquidsCalibrationData = CalibrationDataViewVM.LiquidsCalibrationData, LiquidTypesRmls = LiquidTypesRmls, }; _rmlBeforeSave = RmlDTO.FromObservable(ActiveRML); View.NavigateTo(RmlNavigationView.RmlView); InvalidateRelayCommands(); IsFree = true; } catch (Exception ex) { LogManager.Log($"Error loading RML '{ActiveRML.Name}'..."); _notification.ShowError($"Error loading the selected thread.\n{ex.FlattenMessage()}"); } finally { IsFree = true; } } } private async void OnSelectedCCTChanged() { if (SelectedCCT != null && !SelectedCCT.IsNew) { using (_notification.PushTaskItem("Loading CCT data...")) { IsFree = false; var cct = await _active_context.Ccts.SingleOrDefaultAsync(x => x.Guid == SelectedCCT.Guid); if (cct != null) { SelectedCCT.Data = cct.Data; ColorConversionViewVM.CCT = SelectedCCT; } IsFree = true; } } } private void LoadRmlProperties() { Materials = _active_context.MediaMaterials.ToObservableCollection(); Purposes = _active_context.MediaPurposes.ToObservableCollection(); Conditions = _active_context.MediaConditions.ToObservableCollection(); LinearMassDensityUnits = _active_context.LinearMassDensityUnits.ToObservableCollection(); FiberShapes = _active_context.FiberShapes.ToObservableCollection(); FiberSynths = _active_context.FiberSynths.ToObservableCollection(); SpoolTypes = _active_context.SpoolTypes.ToObservableCollection(); } private async void AddNewRml() { var name = _notification.ShowTextInput("Please enter RML name", "Name"); if (!String.IsNullOrWhiteSpace(name)) { if (Rmls.ToList().Exists(x => x.Name == name)) { _notification.ShowError("The specified RML name already exists. Please select a different name."); return; } using (_notification.PushTaskItem("Creating new RML...")) { IsFree = false; if (_active_context != null) { _active_context.Dispose(); } _active_context = ObservablesContext.CreateDefault(); LoadRmlProperties(); Rml rml = new Rml(); rml.Name = name; rml.DisplayName = name; rml.QualificationDate = DateTime.UtcNow; rml.Manufacturer = "Twine"; rml.Code = Rmls.Max(x => x.Code) + 1; rml.MediaMaterial = Materials.FirstOrDefault(); rml.MediaPurpose = Purposes.FirstOrDefault(); rml.MediaCondition = Conditions.FirstOrDefault(); rml.LinearMassDensityUnit = LinearMassDensityUnits.FirstOrDefault(); rml.FiberShape = FiberShapes.FirstOrDefault(); rml.FiberSynth = FiberSynths.FirstOrDefault(); ProcessParametersTablesGroup group = new ProcessParametersTablesGroup(); group.Name = "Active Group"; group.Active = true; group.ProcessParametersTables.Add(new ProcessParametersTable() { Name = "Process Table 1", }); group.Rml = rml; _active_context.ProcessParametersTablesGroups.Add(group); _active_context.ProcessParametersTables.Add(group.ProcessParametersTables[0]); _active_context.Rmls.Add(rml); await _active_context.SaveChangesAsync(); _actionLogManager.InsertLog(BL.Enumerations.ActionLogType.RmlCreated, _authentication.CurrentUser, rml.Name, rml, "Rml created using Machine Studio."); LoadActiveRML(rml.Guid); IsFree = true; } } } private async void RemoveSelectedRml() { if (_notification.ShowQuestion("Removing the selected thread will result in the loss of all related process parameters and default calibration data. Are you sure you want to delete the selected RML?")) { IsFree = false; using (_notification.PushTaskItem("Removing RML...")) { try { var rml_jobs = await _rmls_context.Jobs.Where(x => x.RmlGuid == SelectedRML.Guid).Include(x => x.Machine).OrderBy(x => x.Machine.SerialNumber).ToListAsync(); if (rml_jobs.Count > 0) { _notification.ShowError($"The following jobs must be removed or change thread type before the selected thread can be deleted:\n{String.Join("\n", rml_jobs.Select(x => $"{x.Machine.SerialNumber} => {x.Name}"))}"); return; } await SelectedRML.DeleteCascadeAsync(_rmls_context); _actionLogManager.InsertLog(BL.Enumerations.ActionLogType.RmlDeleted, _authentication.CurrentUser, SelectedRML.Name, SelectedRML, "RML deleted from Machine Studio."); LoadRmls(); } catch (Exception ex) { LogManager.Log(ex, $"Error removing selected RML {SelectedRML.Name}."); _notification.ShowError($"An error occurred while trying to remove the selected RML.\n{ex.Message}"); } } IsFree = true; } } private async void CloneSelectedRml() { String name = _notification.ShowTextInput("Enter thread name", "thread name"); if (!String.IsNullOrWhiteSpace(name)) { using (_notification.PushTaskItem("Cloning thread...")) { try { IsFree = false; using (var context = ObservablesContext.CreateDefault()) { var rml = await new RmlBuilder(context).Set(SelectedRML.Guid).WithActiveParametersGroup().WithLiquidFactors().WithSpools().BuildAsync(); Rml cloned = new Rml(); rml.MapPropertiesTo(cloned, MappingFlags.NoReferenceTypes); cloned.Code = Rmls.Max(x => x.Code) + 1; cloned.Guid = Guid.NewGuid().ToString(); cloned.ID = 0; cloned.Name = name; cloned.DisplayName = name; cloned.RmlQualificationLevel = RmlQualifications.Provisional; cloned.QualificationDate = DateTime.UtcNow; ProcessParametersTablesGroup group = new ProcessParametersTablesGroup(); group.Name = rml.GetActiveProcessGroup().Name; group.Active = true; foreach (var p in rml.GetActiveProcessGroup().ProcessParametersTables) { var pc = new ProcessParametersTable(); p.MapPrimitivesTo(pc); pc.Name = p.Name; group.ProcessParametersTables.Add(pc); } cloned.ProcessParametersTablesGroups.Add(group); foreach (var liquidFactor in rml.LiquidTypesRmls) { LiquidTypesRml l = new LiquidTypesRml(); l.DefaultCatData = liquidFactor.DefaultCatData; l.LiquidType = liquidFactor.LiquidType; l.MaxNlPerCm = liquidFactor.MaxNlPerCm; cloned.LiquidTypesRmls.Add(l); } foreach (var spool in rml.RmlsSpools) { RmlsSpool s = new RmlsSpool(); spool.MapPropertiesTo(s, MappingFlags.ValueTypesOnly); s.RmlGuid = cloned.Guid; s.SpoolTypeGuid = spool.SpoolTypeGuid; cloned.RmlsSpools.Add(s); } context.Rmls.Add(cloned); await context.SaveChangesAsync(); _actionLogManager.InsertLog(BL.Enumerations.ActionLogType.RmlCreated, _authentication.CurrentUser, cloned.Name, cloned, "RML cloned from Machine Studio."); } LoadRmls(); } catch (Exception ex) { LogManager.Log(ex, "Error cloning thread."); _notification.ShowError($"An error occurred while trying to clone the selected thread\n{ex.Message}"); } finally { IsFree = true; } } } } private void AddProcessParametersTable() { var name = _notification.ShowTextInput("Enter table name", "Name"); if (!String.IsNullOrWhiteSpace(name)) { _active_context.ProcessParametersTables.Add(new ProcessParametersTable() { ProcessParametersTablesGroup = ActiveProcessParametersGroup, Name = name, TableIndex = ActiveProcessParametersGroup.ProcessParametersTables.Max(x => x.TableIndex) + 1, }); } } public void OnProcessParametersTableDropped(ProcessParametersTable dragged, ProcessParametersTable dropped) { if (dragged.TableIndex > dropped.TableIndex) { dragged.TableIndex = dropped.TableIndex - 1; } else { dragged.TableIndex = dropped.TableIndex + 1; } int index = 0; foreach (var table in ActiveProcessParametersGroup.ProcessParametersTables.OrderBy(x => x.TableIndex)) { table.TableIndex = index++; } ActiveProcessParametersTableView.Refresh(); } private void RemoveLiquidFactor(LiquidTypesRml liquidFactor) { if (_notification.ShowQuestion("Removing this liquid factor will remove the liquid type association with the RML and will drop the calibration data. Are you sure?")) { var catVM = CalibrationDataViewVM.LiquidsCalibrationData.SingleOrDefault(x => x.LiquidType == liquidFactor.LiquidType); CalibrationDataViewVM.LiquidsCalibrationData.Remove(catVM); _active_context.LiquidTypesRmls.Remove(liquidFactor); } } private void AddLiquidFactor() { AddLiquidFactorViewVM vm = new AddLiquidFactorViewVM(_active_context); _notification.ShowModalDialog(vm, (_) => { if (LiquidTypesRmls.ToList().Exists(x => x.LiquidType == vm.SelectedLiquidType)) { _notification.ShowError("The selected liquid type is already associated with this RML."); return; } LiquidTypesRml liquidFactor = new LiquidTypesRml() { LiquidType = vm.SelectedLiquidType, Rml = ActiveRML, }; _active_context.LiquidTypesRmls.Add(liquidFactor); CalibrationDataVM catVM = new CalibrationDataVM(liquidFactor.LiquidType); CalibrationDataViewVM.LiquidsCalibrationData.Add(catVM); }, () => { }); } private void RemoveProcessParametersTable(ProcessParametersTable processParametersTable) { if (ActiveProcessParametersGroup.ProcessParametersTables.Count == 1) { _notification.ShowError("The process group must contain at least one table."); return; } if (_notification.ShowQuestion("Are you sure you want to remove this process parameters table?")) { _active_context.ProcessParametersTables.Remove(processParametersTable); } } private void CreateCalibrationDataExcelTemplate() { SaveFileDialog dlg = new SaveFileDialog(); try { dlg.Title = $"Create excel template file"; dlg.Filter = "Excel Files|*.xlsx"; dlg.DefaultExt = ".xlsx"; dlg.FileName = "Calibration File Template"; if (dlg.ShowDialog().Value) { CalibrationHelper.CreateCalibrationDataExcelTemplate(dlg.FileName); } } catch (Exception ex) { LogManager.Log(ex, "Error generating excel calibration template file " + dlg.FileName); _notification.ShowError("An error occurred while trying to generate the calibration file."); } } private async void Save() { IsFree = false; try { using (_notification.PushTaskItem("Saving RML...")) { foreach (var calDataVM in CalibrationDataViewVM.LiquidsCalibrationData) { var liquidTypeRml = LiquidTypesRmls.SingleOrDefault(x => x.LiquidType == calDataVM.LiquidType); CalibrationData calData = new CalibrationData(); calData.LiquidType = (PMR.ColorLab.LiquidType)liquidTypeRml.LiquidType.Code; calData.CalibrationPoints.AddRange(calDataVM.CalibrationPoints.Select(x => new CalibrationPoint() { X = x.X, Y = x.Y })); liquidTypeRml.PutCalibrationData(calData); } ActiveRML.LastUpdated = DateTime.UtcNow; if (_rmlBeforeSave.QualificationLevel != ActiveRML.QualificationLevel) { ActiveRML.QualificationDate = DateTime.UtcNow; } if (SelectedCCT != null) { if (SelectedCCT.IsNew) { Cct cct = new Cct(); cct.Guid = SelectedCCT.Guid; cct.FileName = SelectedCCT.FileName; cct.Data = SelectedCCT.Data; ActiveRML.Cct = cct; SelectedCCT.IsNew = false; } else { ActiveRML.CctGuid = SelectedCCT.Guid; } } var rmlAfter = RmlDTO.FromObservable(ActiveRML); await _active_context.SaveChangesAsync(); _actionLogManager.InsertLog(BL.Enumerations.ActionLogType.RmlSaved, _authentication.CurrentUser, _rmlBeforeSave.Name, _rmlBeforeSave, rmlAfter, "RML saved using Machine Studio."); _rmlBeforeSave = rmlAfter; LoadActiveRML(ActiveRML.Guid); } } catch (Exception ex) { LogManager.Log(ex, $"Error saving RML {ActiveRML.Name}"); _notification.ShowError($"An error occurred while trying to save the current RML.\n{ex.FlattenMessage()}"); } finally { IsFree = true; } } private void BackToRmls() { View.NavigateTo(RmlNavigationView.RmlsView); LoadRmls(); } #region Import / Export Color Conversion Data private void ImportCCTData() { String file = GetCCTFileOpen(); if (file != null) { CctModel cctModel = new CctModel(); cctModel.Guid = Guid.NewGuid().ToString(); cctModel.IsNew = true; cctModel.FileName = Path.GetFileName(file); cctModel.Data = File.ReadAllBytes(file); CCTS.Insert(0, cctModel); SelectedCCT = cctModel; ColorConversionViewVM.CCT = SelectedCCT; } } private void ExportCCTData() { if (SelectedCCT != null) { String file = GetCCTFileSave(ActiveRML.Cct.FileName); if (file != null) { using (_notification.PushTaskItem("Exporting CCT file...")) { try { IsFree = false; if (SelectedCCT.IsNew) { File.WriteAllBytes(file, SelectedCCT.Data); } else { var cct = _active_context.Ccts.SingleOrDefault(x => x.Guid == SelectedCCT.Guid); if (cct != null) { File.WriteAllBytes(file, cct.Data); } } } catch (Exception ex) { LogManager.Log(ex, "Error exporting CCT file."); _notification.ShowError($"An error occurred while trying to export the CCT file.\n{ex.Message}"); } finally { IsFree = true; } } } } } private String GetCCTFileOpen() { OpenFileDialog dlg = new OpenFileDialog(); dlg.Title = "Select color adjustment file"; dlg.Filter = "Color Conversion Table|*.cct"; if (dlg.ShowDialogCenter()) { return dlg.FileName; } return null; } private String GetCCTFileSave(String fileName) { SaveFileDialog dlg = new SaveFileDialog(); dlg.Title = "Select color adjustment file"; dlg.Filter = "Color Conversion Table|*.cct"; dlg.FileName = fileName; dlg.DefaultExt = ".cct"; if (dlg.ShowDialogCenter()) { return dlg.FileName; } return null; } #endregion #region RML Import / Export private async void ImportRmlFile() { OpenFileDialog dlg = new OpenFileDialog(); dlg.Title = "Import Thread Files"; dlg.Filter = "Twine Thread Files|*.rml"; dlg.Multiselect = true; if (dlg.ShowDialog().Value) { using (_notification.PushTaskItem($"Importing thread files...")) { try { IsFree = false; LogManager.Log($"Importing thread files..."); using (ObservablesContext db = ObservablesContext.CreateDefault()) { foreach (var file in dlg.FileNames) { var json = File.ReadAllText(file); var rmlFile = await Rml.FromRmlFile(db, json); db.Rmls.Add(rmlFile); _actionLogManager.InsertLog(BL.Enumerations.ActionLogType.RmlImported, _authentication.CurrentUser, rmlFile.Name, rmlFile, "RML imported from Machine Studio."); } await db.SaveChangesAsync(); } IsFree = true; _notification.ShowInfo($"Threads imported successfully."); LoadRmls(); } catch (Exception ex) { LogManager.Log(ex, "Error importing thread file."); _notification.ShowError($"An error occurred while trying to import the selected thread file.\n{ex.FlattenMessage()}"); } finally { IsFree = true; } } } } private async void ExportRmlFile() { SaveFileDialog dlg = new SaveFileDialog(); dlg.Title = "Export Thread File"; dlg.Filter = "Twine Thread Files|*.rml"; dlg.DefaultExt = ".rml"; dlg.FileName = SelectedRML.Name; if (dlg.ShowDialog().Value) { using (_notification.PushTaskItem($"Exporting Thread '{SelectedRML.Name}'...")) { try { LogManager.Log($"Exporting Thread file {SelectedRML.Name}"); using (ObservablesContext db = ObservablesContext.CreateDefault()) { var rmlJsonFile = await SelectedRML.ToRmlFile(db); File.WriteAllText(dlg.FileName, rmlJsonFile); } } catch (Exception ex) { LogManager.Log(ex, "Error exporting Thread file."); _notification.ShowError($"An error occurred while trying to export thread '{SelectedRML.Name}'.\n{ex.FlattenMessage()}"); } } } } #endregion #region Spools private void AddNewSpool() { _active_context.RmlsSpools.Add(new RmlsSpool() { Rml = ActiveRML, }); } private void RemoveSpool() { if (SelectedSpool != null) { _active_context.RmlsSpools.Remove(SelectedSpool); ActiveRML.RmlsSpools.Remove(SelectedSpool); } } #endregion } }