/* Copyright (c) 2019, 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; using System.IO; using System.Linq; using System.Text; using MatterHackers.Agg; using MatterHackers.MatterControl.SlicerConfiguration.MappingClasses; using MatterHackers.DataConverters3D; using MatterHackers.VectorMath; using Newtonsoft.Json; using Newtonsoft.Json.Converters; using Newtonsoft.Json.Linq; namespace MatterHackers.MatterControl.SlicerConfiguration { public enum NamedSettingsLayers { MHBaseSettings, OEMSettings, Quality, Material, User, All } public enum BedShape { Rectangular, Circular } [JsonConverter(typeof(StringEnumConverter))] public enum LevelingSystem { Probe3Points, Probe7PointRadial, Probe13PointRadial, Probe100PointRadial, Probe3x3Mesh, Probe5x5Mesh, Probe10x10Mesh, ProbeCustom } public class PrinterSettings { private IObjectSlicer _slicer = null; [JsonIgnore] public IObjectSlicer Slicer { get { if (_slicer == null) { string userSlicer = this.GetValue(SettingsKey.slice_engine); if (SliceEngines.TryGetValue(userSlicer, out IObjectSlicer slicer)) { _slicer = slicer; } else { _slicer = SliceEngines.Values.First(); } } return _slicer; } set => _slicer = value; } public static Dictionary SliceEngines { get; } = new Dictionary(); // Latest version should be in the form of: // Year|month|day|versionForDay (to support multiple revisions on a given day) public static int LatestVersion { get; } = 201606271; public static Dictionary SettingsData { get; } public static SettingsLayout Layout { get; } public static event EventHandler AnyPrinterSettingChanged; public event EventHandler SettingChanged; private static HashSet knownSettings; private HashSet replacementTerms; private static HashSet ScaledSpeedFields = new HashSet() { SettingsKey.first_layer_speed, SettingsKey.external_perimeter_speed, SettingsKey.raft_print_speed, SettingsKey.infill_speed, SettingsKey.min_print_speed, SettingsKey.perimeter_speed, SettingsKey.retract_speed, SettingsKey.support_material_speed, SettingsKey.travel_speed, SettingsKey.load_filament_speed, }; /// /// Application level settings control MatterControl behaviors but aren't used or passed through to the slice engine. Putting settings /// in this list ensures they show up for all slice engines and the lack of a MappedSetting for the engine guarantees that it won't pass /// through into the slicer config file. /// public static readonly HashSet ApplicationLevelSettings = new HashSet() { SettingsKey.enable_fan, SettingsKey.extruder_wipe_temperature, SettingsKey.extruders_share_temperature, SettingsKey.slice_engine, SettingsKey.first_layer_bed_temperature, SettingsKey.g0, SettingsKey.layer_to_pause, SettingsKey.selector_ip_address, SettingsKey.solid_shell, SettingsKey.z_homes_to_max, SettingsKey.auto_connect, SettingsKey.auto_release_motors, SettingsKey.backup_firmware_before_update, SettingsKey.baud_rate, SettingsKey.bed_remove_part_temperature, SettingsKey.bed_shape, SettingsKey.bed_size, SettingsKey.bed_temperature, SettingsKey.before_toolchange_gcode, SettingsKey.before_toolchange_gcode_1, SettingsKey.before_toolchange_gcode_2, SettingsKey.before_toolchange_gcode_3, SettingsKey.toolchange_gcode, SettingsKey.toolchange_gcode_1, SettingsKey.toolchange_gcode_2, SettingsKey.toolchange_gcode_3, SettingsKey.build_height, SettingsKey.cancel_gcode, SettingsKey.com_port, SettingsKey.connect_gcode, SettingsKey.created_date, SettingsKey.emulate_endstops, SettingsKey.enable_line_splitting, SettingsKey.enable_network_printing, SettingsKey.enable_retractions, SettingsKey.enable_sailfish_communication, SettingsKey.filament_cost, SettingsKey.filament_density, SettingsKey.filament_has_been_loaded, SettingsKey.filament_1_has_been_loaded, SettingsKey.filament_runout_sensor, SettingsKey.firmware_type, SettingsKey.has_c_axis, SettingsKey.has_fan, SettingsKey.has_fan_per_extruder, SettingsKey.has_hardware_leveling, SettingsKey.has_heated_bed, SettingsKey.has_power_control, SettingsKey.has_sd_card_reader, SettingsKey.has_z_probe, SettingsKey.has_z_servo, SettingsKey.heat_extruder_before_homing, SettingsKey.inactive_cool_down, SettingsKey.include_firmware_updater, SettingsKey.insert_filament_markdown2, SettingsKey.insert_filament_1_markdown, SettingsKey.ip_address, SettingsKey.ip_port, SettingsKey.laser_speed_025, SettingsKey.laser_speed_100, SettingsKey.leveling_sample_points, SettingsKey.load_filament_length, SettingsKey.load_filament_speed, SettingsKey.make, SettingsKey.model, SettingsKey.t0_inset, SettingsKey.t1_inset, SettingsKey.number_of_first_layers, SettingsKey.extruder_offset, SettingsKey.pause_gcode, SettingsKey.print_center, SettingsKey.print_leveling_probe_start, SettingsKey.print_leveling_required_to_print, SettingsKey.print_leveling_solution, SettingsKey.print_time_estimate_multiplier, SettingsKey.printer_name, SettingsKey.probe_has_been_calibrated, SettingsKey.probe_offset, SettingsKey.probe_offset_sample_point, SettingsKey.progress_reporting, SettingsKey.read_regex, SettingsKey.recover_first_layer_speed, SettingsKey.recover_is_enabled, SettingsKey.recover_position_before_z_home, SettingsKey.resume_gcode, SettingsKey.running_clean_markdown2, SettingsKey.running_clean_1_markdown, SettingsKey.seconds_to_reheat, SettingsKey.send_with_checksum, SettingsKey.show_reset_connection, SettingsKey.sla_printer, SettingsKey.t1_extrusion_move_speed_multiplier, SettingsKey.temperature, SettingsKey.temperature1, SettingsKey.temperature2, SettingsKey.temperature3, SettingsKey.trim_filament_markdown, SettingsKey.unload_filament_length, SettingsKey.use_z_probe, SettingsKey.validate_layer_height, SettingsKey.write_regex, SettingsKey.xy_offsets_have_been_calibrated, SettingsKey.z_offset, SettingsKey.z_probe_samples, SettingsKey.z_servo_depolyed_angle, SettingsKey.z_servo_retracted_angle, }; /// /// Call the given function for every active tool /// /// The type for the given key /// The settings key to lookup per tool /// The function to call per active tool public void ForTools(string key, Action applyToEach) { string Tool(int toolIndex) { if (toolIndex == 0) { return key; } // If the setting has a described method for constructing the keys per tool, use it. var perToolName = SettingsData[key].PerToolName; if (perToolName != null) { return perToolName(toolIndex); } return $"{key}_{toolIndex}"; } var extruderCount = this.GetValue(SettingsKey.extruder_count); for (int i = 0; i < extruderCount; i++) { var toolKey = Tool(i); applyToEach(toolKey, this.GetValue(toolKey), i); } } public void OnSettingChanged(string slicerConfigName) { if (slicerConfigName == SettingsKey.t0_inset || slicerConfigName == SettingsKey.t1_inset || slicerConfigName == SettingsKey.bed_size || slicerConfigName == SettingsKey.print_center) { this.ResetHotendBounds(); } SettingChanged?.Invoke(this, new StringEventArgs(slicerConfigName)); AnyPrinterSettingChanged?.Invoke(this, new StringEventArgs(slicerConfigName)); } public event EventHandler PrintLevelingEnabledChanged; public event EventHandler MacrosChanged; public static PrinterSettings Empty { get; } public int DocumentVersion { get; set; } = LatestVersion; public string ID { get; set; } [JsonIgnore] public PrinterSettingsLayer QualityLayer { get; set; } [JsonIgnore] public PrinterSettingsLayer MaterialLayer { get; set; } public PrinterSettingsLayer StagedUserSettings { get; set; } = new PrinterSettingsLayer(); static PrinterSettings() { // Convert settings array into dictionary on initial load using settings key (SlicerConfigName) PrinterSettings.SettingsData = SliceSettingsFields.AllSettings().ToDictionary(s => s.SlicerConfigName); PrinterSettings.Layout = new SettingsLayout(); Empty = new PrinterSettings() { ID = "EmptyProfile" }; Empty.UserLayer[SettingsKey.printer_name] = "Empty Printer"; } public PrinterSettings() { this.Helpers = new SettingsHelpers(this); replacementTerms = new HashSet() { SettingsKey.first_layer_speed, SettingsKey.external_perimeter_speed, SettingsKey.raft_print_speed, SettingsKey.bed_remove_part_temperature, SettingsKey.bridge_fan_speed, SettingsKey.bridge_speed, SettingsKey.air_gap_speed, SettingsKey.extruder_wipe_temperature, SettingsKey.filament_diameter, SettingsKey.first_layer_bed_temperature, SettingsKey.first_layer_temperature, SettingsKey.max_fan_speed, SettingsKey.min_fan_speed, SettingsKey.retract_length, SettingsKey.temperature, SettingsKey.bed_temperature, SettingsKey.temperature1, SettingsKey.temperature2, SettingsKey.temperature3, SettingsKey.infill_speed, SettingsKey.min_print_speed, SettingsKey.perimeter_speed, SettingsKey.perimeter_acceleration, SettingsKey.default_acceleration, SettingsKey.retract_speed, SettingsKey.support_material_speed, SettingsKey.travel_speed, SettingsKey.load_filament_speed, SettingsKey.trim_filament_markdown, SettingsKey.insert_filament_markdown2, SettingsKey.insert_filament_1_markdown, SettingsKey.running_clean_markdown2, SettingsKey.running_clean_1_markdown, }; } public List Macros { get; set; } = new List(); /// /// Restore deactivated user overrides by iterating the active preset and removing/restoring matching items /// public void RestoreConflictingUserOverrides(PrinterSettingsLayer settingsLayer) { if (settingsLayer == null) { return; } foreach (var settingsKey in settingsLayer.Keys) { RestoreUserOverride(settingsLayer, settingsKey); } } public string ReplaceMacroValues(string gcodeWithMacros) { foreach (string replacementTerm in replacementTerms) { // first check if this setting is anywhere in the line if (gcodeWithMacros.Contains(replacementTerm)) { // Acquire the replacement value string value = this.ResolveValue(replacementTerm); if (ScaledSpeedFields.Contains(replacementTerm) && double.TryParse(value, out double doubleValue)) { doubleValue *= 60; value = $"{doubleValue:0.###}"; } // Use bed_temperature if the slice engine does not have first_layer_bed_temperature if (replacementTerm == SettingsKey.first_layer_bed_temperature && !this.Slicer.Exports.ContainsKey(SettingsKey.first_layer_bed_temperature)) { value = $"{this.GetValue(SettingsKey.bed_temperature)}"; } // braces then brackets replacement gcodeWithMacros = gcodeWithMacros.Replace("{" + replacementTerm + "}", value); gcodeWithMacros = gcodeWithMacros.Replace("[" + replacementTerm + "]", value); } } return gcodeWithMacros; } private void RestoreUserOverride(PrinterSettingsLayer settingsLayer, string settingsKey) { if (StagedUserSettings.TryGetValue(settingsKey, out string stagedUserOverride)) { StagedUserSettings.Remove(settingsKey); UserLayer[settingsKey] = stagedUserOverride; } } public void NotifyMacrosChanged() { this.MacrosChanged?.Invoke(this, null); } /// /// Move conflicting user overrides to the temporary staging area, allowing presets values to take effect /// public void DeactivateConflictingUserOverrides(PrinterSettingsLayer settingsLayer) { if (settingsLayer == null) { return; } foreach (var settingsKey in settingsLayer.Keys) { StashUserOverride(settingsLayer, settingsKey); } } /// /// Determines if a given field should be shown given its filter /// /// The view filter - order of precedence &, |, !, = /// An indicator if the field should be shown given the current filter public bool ParseShowString(string filter) { if (!string.IsNullOrEmpty(filter)) { string[] splitOnAnd = filter.Split('&'); foreach (var andGroup in splitOnAnd) { bool orResult = false; string[] splitOnOr = andGroup.Split('|'); foreach (var orGroup in splitOnOr) { var matchString = "1"; var orItem = orGroup; bool negate = orItem.StartsWith("!"); if (negate) { orItem = orItem.Substring(1); } string sliceSettingValue = ""; if (orItem.Contains("=")) { string[] splitOnEquals = orItem.Split('='); sliceSettingValue = this.GetValue(splitOnEquals[0]); matchString = splitOnEquals[1]; } else if (orItem.Contains(">")) { matchString = "no_match"; string[] splitOnGreater = orItem.Split('>'); sliceSettingValue = this.GetValue(splitOnGreater[0]); if (double.TryParse(sliceSettingValue, out double doubleValue)) { if (double.TryParse(splitOnGreater[1], out double greater)) { if (doubleValue > greater) { matchString = sliceSettingValue; } } } } else { sliceSettingValue = this.GetValue(orItem); } if ((!negate && sliceSettingValue == matchString) || (negate && sliceSettingValue != matchString)) { orResult = true; } } if (orResult == false) { return false; } } } return true; } /// /// Move conflicting user overrides to the temporary staging area, allowing presets values to take effect /// private void StashUserOverride(PrinterSettingsLayer settingsLayer, string settingsKey) { if (this.UserLayer.TryGetValue(settingsKey, out string userOverride)) { this.UserLayer.Remove(settingsKey); this.StagedUserSettings[settingsKey] = userOverride; } } public PrinterSettingsLayer OemLayer { get; set; } public void Merge(PrinterSettingsLayer destinationLayer, PrinterSettings settingsToImport, List rawSourceFilter, bool setLayerName) { var skipKeys = new HashSet { "layer_id", }; if (!setLayerName) { skipKeys.Add(SettingsKey.layer_name); } var destinationFilter = new List { OemLayer, BaseLayer, destinationLayer, }.Where(layer => layer != null); var sourceFilter = rawSourceFilter.Where(layer => layer != null); foreach (var keyName in PrinterSettings.KnownSettings) { if (settingsToImport.Contains(keyName)) { // Compare the value to import to the layer cascade value and only set if different string currentValue = this.GetValue(keyName, destinationFilter).Trim(); string importValue = settingsToImport.GetValue(keyName, sourceFilter).Trim(); if (!string.IsNullOrEmpty(importValue) && currentValue != importValue && !skipKeys.Contains(keyName)) { destinationLayer[keyName] = importValue; } } } if (setLayerName) { destinationLayer[SettingsKey.layer_name] = settingsToImport.GetValue(SettingsKey.layer_name, sourceFilter); } this.OnSettingChanged("na"); } public void CopyFrom(PrinterSettings printerSettings) { this.OemLayer = printerSettings.OemLayer; this.MaterialLayers = printerSettings.MaterialLayers; this.QualityLayers = printerSettings.QualityLayers; this.UserLayer = printerSettings.UserLayer; this.ID = printerSettings.ID; this.QualityLayer = GetQualityLayer(printerSettings.ActiveQualityKey); this.MaterialLayer = GetMaterialLayer(printerSettings.ActiveMaterialKey); this.StagedUserSettings = printerSettings.StagedUserSettings; this.Macros = printerSettings.Macros; } internal PrinterSettingsLayer GetMaterialLayer(string layerID) { if (string.IsNullOrEmpty(layerID)) { return null; } return MaterialLayers.Where(layer => layer.LayerID == layerID).FirstOrDefault(); } private PrinterSettingsLayer GetQualityLayer(string layerID) { return QualityLayers.Where(layer => layer.LayerID == layerID).FirstOrDefault(); } // Properties [JsonIgnore] public string ActiveQualityKey { get => GetValue(SettingsKey.active_quality_key); set { if (this.ActiveQualityKey != value) { SetValue(SettingsKey.active_quality_key, value); QualityLayer = GetQualityLayer(value); } } } [JsonIgnore] public string ActiveMaterialKey { get => GetValue(SettingsKey.active_material_key); set { if (this.ActiveMaterialKey != value) { SetValue(SettingsKey.active_material_key, value); MaterialLayer = GetMaterialLayer(value); } } } [JsonIgnore] public RectangleDouble BedBounds { get { var bedSize = this.GetValue(SettingsKey.bed_size); var printCenter = this.GetValue(SettingsKey.print_center); return new RectangleDouble(printCenter.X - bedSize.X / 2, printCenter.Y - bedSize.Y / 2, printCenter.X + bedSize.X / 2, printCenter.Y + bedSize.Y / 2); } } private void ResetHotendBounds() { var bounds = this.BedBounds; RectangleDouble GetHotendBounds(int index) { string settingsKey = index == 0 ? SettingsKey.t0_inset : SettingsKey.t1_inset; var inset = this.GetValue(settingsKey); return new RectangleDouble( bounds.Left + inset.X, bounds.Bottom + inset.Y, bounds.Right - inset.Z, bounds.Top - inset.W); } this.ToolBounds = new[] { GetHotendBounds(0), GetHotendBounds(1), }; } /// /// Gets the bounds that are accessible for a given hotend /// [JsonIgnore] public RectangleDouble[] ToolBounds { get; private set; } [JsonIgnore] public bool AutoSave { get; set; } = true; public string ComputeSHA1() { return ComputeSHA1(this.ToJson()); } public string ComputeSHA1(string json) { // SHA1 value is based on UTF8 encoded file contents using (var memoryStream = new MemoryStream(Encoding.UTF8.GetBytes(json))) { return HashGenerator.ComputeSHA1(memoryStream); } } /// /// User settings overrides /// public PrinterSettingsLayer UserLayer { get; private set; } = new PrinterSettingsLayer(); public static PrinterSettings LoadFile(string printerProfilePath, bool performMigrations = false) { if (performMigrations) { JObject jObject = null; try { jObject = JObject.Parse(File.ReadAllText(printerProfilePath)); } catch { return null; } int documentVersion = jObject?.GetValue("DocumentVersion")?.Value() ?? PrinterSettings.LatestVersion; if (documentVersion < PrinterSettings.LatestVersion) { printerProfilePath = ProfileMigrations.MigrateDocument(printerProfilePath, documentVersion); } } try { var settings = JsonConvert.DeserializeObject(File.ReadAllText(printerProfilePath), new PrinterSettingsConverter()); settings.OnDeserialize(); return settings; } catch { return null; } } private void OnDeserialize() { this.ResetHotendBounds(); } internal void OnPrintLevelingEnabledChanged(object s, EventArgs e) { PrintLevelingEnabledChanged?.Invoke(s, e); } public List MaterialLayers { get; private set; } = new List(); public List QualityLayers { get; private set; } = new List(); /// /// Gets the first matching value discovered while enumerating the settings layers /// public string GetValue(string sliceSetting, IEnumerable layerCascade = null) { if (layerCascade == null) { layerCascade = defaultLayerCascade; } foreach (PrinterSettingsLayer layer in layerCascade) { if (layer.TryGetValue(sliceSetting, out string value)) { return value; } } return ""; } public bool IsOverride(string sliceSetting) { var values = new List(); string firstBaseValue = null; foreach (PrinterSettingsLayer layer in defaultLayerCascade) { if (layer.TryGetValue(sliceSetting, out string value)) { if (layer == this.BaseLayer || layer == this.OemLayer) { firstBaseValue = value; break; } values.Add(value); } } string currentValue = values.FirstOrDefault(); string firstPresetValue = values.Skip(1).FirstOrDefault(); bool differsFromPreset = values.Count > 0 && firstPresetValue != null && firstPresetValue != currentValue; bool differsFromBase = currentValue != firstBaseValue; return currentValue != null && (differsFromPreset || differsFromBase); } public bool IsActive(string canonicalSettingsName) { return this.Slicer.Exports.ContainsKey(canonicalSettingsName) || PrinterSettings.ApplicationLevelSettings.Contains(canonicalSettingsName); } // Helper method to debug settings layers per setting public List<(string layerName, string currentValue)> GetLayerValues(string sliceSetting, IEnumerable layerCascade = null) { if (layerCascade == null) { layerCascade = defaultLayerCascade; } var results = new List<(string layerName, string currentValue)>(); foreach (PrinterSettingsLayer layer in layerCascade) { if (layer.TryGetValue(sliceSetting, out string value)) { string layerName = "User"; if (layer == this.BaseLayer) { layerName = "Base"; } else if (layer == this.OemLayer) { layerName = "Oem"; } else if (layer == this.MaterialLayer) { layerName = "Material"; } else if (layer == this.QualityLayer) { layerName = "Quality"; } results.Add((layerName, value)); } } return results; } public (string currentValue, string layerName) GetValueAndLayerName(string sliceSetting, IEnumerable layerCascade = null) { if (layerCascade == null) { layerCascade = defaultLayerCascade; } foreach (PrinterSettingsLayer layer in layerCascade) { string value; if (layer.TryGetValue(sliceSetting, out value)) { string layerName = "User"; if (layer == this.BaseLayer) { layerName = "Base"; } else if (layer == this.OemLayer) { layerName = "Oem"; } else if (layer == this.MaterialLayer) { layerName = "Material"; } else if (layer == this.QualityLayer) { layerName = "Quality"; } return (value, layerName); } } return ("", ""); } public bool Contains(string sliceSetting, IEnumerable layerCascade = null) { if (layerCascade == null) { layerCascade = defaultLayerCascade; } foreach (PrinterSettingsLayer layer in layerCascade) { if (layer.ContainsKey(sliceSetting)) { return true; } } return false; } private PrinterSettingsLayer _baseLayer; [JsonIgnore] public PrinterSettingsLayer BaseLayer { get { if (_baseLayer == null) { var settingsLayer = new PrinterSettingsLayer(); foreach (var settingsData in PrinterSettings.SettingsData.Values) { settingsLayer[settingsData.SlicerConfigName] = settingsData.DefaultValue; } _baseLayer = settingsLayer; } return _baseLayer; } } private IEnumerable defaultLayerCascade { get { if (this.UserLayer != null) { yield return this.UserLayer; } if (this.MaterialLayer != null) { yield return this.MaterialLayer; } if (this.QualityLayer != null) { yield return this.QualityLayer; } if (this.OemLayer != null) { yield return this.OemLayer; } yield return this.BaseLayer; } } public IEnumerable GetDefaultLayerCascade() => defaultLayerCascade; [JsonIgnore] public SettingsHelpers Helpers { get; set; } [JsonIgnore] public bool PrinterSelected => OemLayer?.Keys.Count > 0; internal void RunInTransaction(Action action) { // TODO: Implement RunInTransaction // Suspend writes action(this); // Commit } public void ClearUserOverrides() { var userOverrides = this.UserLayer.Keys.ToArray(); // Leave user layer items that have no Organizer definition and thus cannot be changed by the user var keysToRetain = new HashSet(userOverrides.Except(KnownSettings)); // Print leveling data has no SliceSettingsWidget editor but should be removed on 'Reset to Defaults' keysToRetain.Remove(SettingsKey.print_leveling_data); keysToRetain.Remove(SettingsKey.print_leveling_enabled); // Iterate all items that have .ShowAsOverride = false and conditionally add to the retention list foreach (var item in PrinterSettings.SettingsData.Values.Where(settingsItem => settingsItem.ShowAsOverride == false)) { switch (item.SlicerConfigName) { case SettingsKey.baud_rate: case SettingsKey.auto_connect: // Items *should* reset to defaults break; default: // Items should *not* reset to defaults keysToRetain.Add(item.SlicerConfigName); break; } } var keysToRemove = (from keyValue in this.UserLayer where !keysToRetain.Contains(keyValue.Key) select keyValue.Key).ToList(); foreach (string key in keysToRemove) { this.UserLayer.Remove(key); } } #region Migrate to LayeredProfile static Dictionary expectedMappingTypes = new Dictionary() { [SettingsKey.extruders_share_temperature] = typeof(int), [SettingsKey.extruders_share_temperature] = typeof(bool), [SettingsKey.has_heated_bed] = typeof(bool), [SettingsKey.nozzle_diameter] = typeof(double), [SettingsKey.bed_temperature] = typeof(double), }; void ValidateType(string settingsKey) { if (expectedMappingTypes.ContainsKey(settingsKey)) { if (expectedMappingTypes[settingsKey] != typeof(T)) { throw new Exception("You must request the correct type of this settingsKey."); } } if (settingsKey.Contains("%")) { if (typeof(T) != typeof(double) || typeof(T) != typeof(int)) { throw new Exception("To get processing of a % you must request the type as double."); } } } public string ResolveValue(string settingsKey) { string value = this.GetValue(settingsKey); if (SettingsData.TryGetValue(settingsKey, out SliceSettingData settingsData) && settingsData.Converter is ValueConverter resolver) { return resolver.Convert(value, this); } return value; } /// /// Returns the first matching value discovered while enumerating the settings layers /// /// The type to return /// The setting to look up /// The settings layers to look at in order /// The value of the setting cast to type public T GetValue(string settingsKey, IEnumerable layerCascade = null) { #if DEBUG ValidateType(settingsKey); #endif if (layerCascade == null) { layerCascade = defaultLayerCascade; } string settingsValue = null; foreach (PrinterSettingsLayer layer in layerCascade) { if (layer.TryGetValue(settingsKey, out settingsValue)) { break; } } if (settingsValue == null) { settingsValue = ""; } if (typeof(T) == typeof(string)) { // this way we can use the common pattern without error return (T)(object)settingsValue; } else if (typeof(T) == typeof(LevelingSystem)) { switch (settingsValue) { case "3 Point Plane": return (T)(object)LevelingSystem.Probe3Points; case "7 Point Disk": return (T)(object)LevelingSystem.Probe7PointRadial; case "13 Point Disk": return (T)(object)LevelingSystem.Probe13PointRadial; case "100 Point Disk": return (T)(object)LevelingSystem.Probe100PointRadial; case "3x3 Mesh": return (T)(object)LevelingSystem.Probe3x3Mesh; case "5x5 Mesh": return (T)(object)LevelingSystem.Probe5x5Mesh; case "10x10 Mesh": return (T)(object)LevelingSystem.Probe10x10Mesh; case "Custom Points": return (T)(object)LevelingSystem.ProbeCustom; default: #if DEBUG throw new NotImplementedException(); #else break; #endif } } else if (typeof(T) == typeof(bool)) { return (T)(object)(settingsValue == "1"); } else if (typeof(T) == typeof(int)) { int.TryParse(this.ResolveValue(settingsKey), out int result); return (T)(object)result; } else if (typeof(T) == typeof(Vector2)) { return (T)(object)Vector2.Parse(settingsValue); } else if (typeof(T) == typeof(Vector3)) { return (T)(object)Vector3.Parse(settingsValue); } else if (typeof(T) == typeof(Vector4)) { return (T)(object)Vector4.Parse(settingsValue); } else if (typeof(T) == typeof(double)) { double.TryParse(this.ResolveValue(settingsKey), out double doubleValue); return (T)(object)doubleValue; } else if (typeof(T) == typeof(BedShape)) { switch (GetValue(settingsKey)) { case "rectangular": return (T)(object)BedShape.Rectangular; case "circular": return (T)(object)BedShape.Circular; default: #if DEBUG throw new NotImplementedException("{0} is not a known bed_shape.".FormatWith(GetValue(SettingsKey.bed_shape))); #else return (T)(object)BedShape.Rectangular; #endif } } return (T)default(T); } /// /// Returns whether or not the setting is overridden by the active layer /// public bool SettingExistsInLayer(string sliceSetting, NamedSettingsLayers layer) { switch (layer) { case NamedSettingsLayers.Quality: return QualityLayer?.ContainsKey(sliceSetting) == true; case NamedSettingsLayers.Material: return MaterialLayer?.ContainsKey(sliceSetting) == true; case NamedSettingsLayers.User: return UserLayer?.ContainsKey(sliceSetting) == true; default: return false; } } public ulong GetGCodeCacheKey() { var bigStringForHashCode = new StringBuilder(); // Loop over all known settings foreach (var keyValue in PrinterSettings.SettingsData) { // Add key/value to accumulating string for hash if (keyValue.Value?.RebuildGCodeOnChange == true) { bigStringForHashCode.Append(keyValue.Key); bigStringForHashCode.Append(this.GetValue(keyValue.Key)); } } return bigStringForHashCode.ToString().GetLongHashCode(); } #endregion [JsonIgnore] public static HashSet KnownSettings { get { if (knownSettings == null) { knownSettings = LoadSettingsNamesFromPropertiesJson(); } return knownSettings; } } private static HashSet LoadSettingsNamesFromPropertiesJson() { return new HashSet(PrinterSettings.SettingsData.Keys); } public void SetValue(string settingsKey, string settingsValue, PrinterSettingsLayer layer = null) { // Stash user overrides if a non-user override is being set if (layer != null && layer != UserLayer) { StashUserOverride(layer, settingsKey); } else { // Remove any staged/conflicting user override, making this the new and active user override if (StagedUserSettings.ContainsKey(settingsKey)) { StagedUserSettings.Remove(settingsKey); } } var persistenceLayer = layer ?? UserLayer; // If the setting exists and is set to the requested value, exit without setting or saving if (persistenceLayer.TryGetValue(settingsKey, out string existingValue) && existingValue == settingsValue) { return; } // Otherwise, set and save persistenceLayer[settingsKey] = settingsValue; this.OnSettingChanged(settingsKey); } public string ToJson() { return JsonConvert.SerializeObject(this, Formatting.Indented); } public void ClearValue(string settingsKey, PrinterSettingsLayer layer = null) { var persistenceLayer = layer ?? UserLayer; if (persistenceLayer.ContainsKey(settingsKey)) { persistenceLayer.Remove(settingsKey); // Restore user overrides if a non-user override is being cleared if (layer != null && layer != UserLayer) { RestoreUserOverride(layer, settingsKey); } if (PrinterSettings.SettingsData.TryGetValue(settingsKey, out SliceSettingData settingData)) { if (settingData.DataEditType == SliceSettingData.DataEditTypes.CHECK_BOX) { string checkedKey = this.GetValue(settingsKey) ? "OnValue" : "OffValue"; // Linked settings should be updated in all cases (user clicked checkbox, user clicked clear) foreach (var setSettingsData in settingData.SetSettingsOnChange) { if (setSettingsData.TryGetValue(checkedKey, out string targetValue)) { if (this.GetValue(setSettingsData["TargetSetting"]) != targetValue) { this.SetValue(setSettingsData["TargetSetting"], targetValue); } } } } } this.OnSettingChanged(settingsKey); } } /// /// Provides a one-way import mechanism for ActiveMaterialKey from the retired MaterialSettingsKeys array /// private class PrinterSettingsConverter : JsonConverter { public override bool CanConvert(Type objectType) { return objectType == typeof(PrinterSettings); } public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer) { // Load object from reader JObject jObject = JObject.Load(reader); // Deserialize using default serializer var settings = jObject.ToObject(); // Look for and import retired MaterialSettingsKeys property if ActiveMaterialKey unset if (string.IsNullOrWhiteSpace(settings.ActiveMaterialKey) && jObject["MaterialSettingsKeys"] is JArray materialSettingsKeys && materialSettingsKeys.Count > 0) { string firstValue = materialSettingsKeys[0].Value(); settings.UserLayer[SettingsKey.active_material_key] = firstValue; } settings.QualityLayer = settings.GetQualityLayer(settings.ActiveQualityKey); if (!string.IsNullOrEmpty(settings.ActiveMaterialKey)) { settings.MaterialLayer = settings.GetMaterialLayer(settings.ActiveMaterialKey); } // Migrate deprecated OemLayer probe setting if (settings.OemLayer.ContainsKey("z_probe_z_offset")) { MigrateProbeOffset(settings.OemLayer); } // Migrate deprecated UserLayer probe setting if (settings.UserLayer.ContainsKey("z_probe_z_offset")) { MigrateProbeOffset(settings.UserLayer, settings.GetValue(SettingsKey.probe_offset)); } return settings; } private static void MigrateProbeOffset(PrinterSettingsLayer settingsLayer, Vector3 position = default(Vector3)) { if (settingsLayer.ContainsKey("z_probe_xy_offset") || settingsLayer.ContainsKey("z_probe_z_offset")) { if (double.TryParse(settingsLayer["z_probe_z_offset"], out double zOffset)) { // and negate it as it was stored in the opposite direction before position.Z = -zOffset; } if (settingsLayer.ContainsKey("z_probe_xy_offset")) { var probeXyOffset = settingsLayer["z_probe_xy_offset"]; if (!string.IsNullOrEmpty(probeXyOffset)) { var split = probeXyOffset.Split(','); if (split.Length == 2) { double.TryParse(split[0], out position.X); double.TryParse(split[1], out position.Y); } } } settingsLayer[SettingsKey.probe_offset] = $"{position.X},{position.Y},{position.Z}"; // clear it settingsLayer.Remove("z_probe_z_offset"); settingsLayer.Remove("z_probe_xy_offset"); } } public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer) { throw new NotImplementedException(); } public override bool CanRead => true; public override bool CanWrite => false; } } }