mattercontrol/MatterControlLib/DesignTools/PublicPropertyEditor.cs

1014 lines
31 KiB
C#
Raw Normal View History

/*
Copyright (c) 2018, 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.ComponentModel;
using System.Linq;
using System.Reflection;
using System.Threading;
using MatterHackers.Agg;
using MatterHackers.Agg.Image;
using MatterHackers.Agg.Platform;
using MatterHackers.Agg.UI;
using MatterHackers.DataConverters3D;
using MatterHackers.Localizations;
using MatterHackers.MatterControl.CustomWidgets;
using MatterHackers.MatterControl.DesignTools.EditableTypes;
2019-02-11 18:16:05 -08:00
using MatterHackers.MatterControl.DesignTools.Operations;
using MatterHackers.MatterControl.PartPreviewWindow;
using MatterHackers.MatterControl.PartPreviewWindow.View3D;
using MatterHackers.MatterControl.SlicerConfiguration;
using MatterHackers.VectorMath;
namespace MatterHackers.MatterControl.DesignTools
{
2018-01-31 13:15:29 -08:00
public class PublicPropertyEditor : IObject3DEditor
{
public string Name => "Property Editor";
2018-06-20 08:09:35 -07:00
public IEnumerable<Type> SupportedTypes() => new Type[] { typeof(IObject3D) };
2018-03-27 14:03:12 -07:00
private static readonly Type[] AllowedTypes =
2018-01-28 07:53:23 -08:00
{
typeof(double), typeof(int), typeof(char), typeof(string), typeof(bool),
typeof(Color),
typeof(Vector2), typeof(Vector3),
typeof(DirectionVector), typeof(DirectionAxis),
typeof(SelectedChildren),
typeof(ImageBuffer),
typeof(List<string>)
2018-01-28 07:53:23 -08:00
};
2018-01-31 14:34:10 -08:00
public const BindingFlags OwnedPropertiesOnly = BindingFlags.Public | BindingFlags.Instance | BindingFlags.DeclaredOnly;
2018-01-28 07:53:23 -08:00
public GuiWidget Create(IObject3D item, UndoBuffer undoBuffer, ThemeConfig theme)
{
var mainContainer = new FlowLayoutWidget(FlowDirection.TopToBottom)
{
HAnchor = HAnchor.Stretch
};
if (item != null)
{
var context = new PPEContext()
{
item = item
};
// CreateEditor
AddUnlockLinkIfRequired(context.item, mainContainer, theme);
2018-06-09 09:01:24 -07:00
GuiWidget scope = mainContainer;
2018-06-09 09:01:24 -07:00
// Create a field editor for each editable property detected via reflection
foreach (var property in GetEditablePropreties(context.item))
{
if (property.PropertyInfo.GetCustomAttributes(true).OfType<HideFromEditorAttribute>().Any())
{
continue;
}
// Create SectionWidget for SectionStartAttributes
if (property.PropertyInfo.GetCustomAttributes(true).OfType<SectionStartAttribute>().FirstOrDefault() is SectionStartAttribute sectionStart)
{
var column = new FlowLayoutWidget()
{
FlowDirection = FlowDirection.TopToBottom,
Padding = new BorderDouble(theme.DefaultContainerPadding).Clone(top: 0)
};
bool expanded = true;
var sectionState = item as ISectionState;
if (sectionState != null)
{
expanded = sectionState.GetSectionExpansion(sectionStart.Title);
}
var section = new SectionWidget(sectionStart.Title, column, theme, expanded: expanded);
theme.ApplyBoxStyle(section);
if (sectionState != null)
{
section.ExpandedChanged += (s, e) => sectionState.SectionExpansionChanged(sectionStart.Title, e);
}
mainContainer.AddChild(section);
scope = column;
}
// Create SectionWidget for SectionStartAttributes
if (property.PropertyInfo.GetCustomAttributes(true).OfType<SectionEndAttribute>().Any())
{
// Push scope back to mainContainer on
scope = mainContainer;
}
2018-06-09 10:28:06 -07:00
var editor = CreatePropertyEditor(property, undoBuffer, context, theme);
if (editor != null)
{
scope.AddChild(editor);
}
2018-06-09 09:01:24 -07:00
}
2021-01-18 21:48:31 -08:00
AddFunctionButtons(item, mainContainer, theme);
AddWebPageLinkIfRequired(context, mainContainer, theme);
2018-06-09 09:01:24 -07:00
// add in an Update button if applicable
2019-04-22 09:23:56 -07:00
if (context.item.GetType().GetCustomAttributes(typeof(ShowUpdateButtonAttribute), true).FirstOrDefault() is ShowUpdateButtonAttribute showUpdate)
2018-06-09 09:01:24 -07:00
{
2018-06-25 08:40:09 -07:00
var updateButton = new TextButton("Update".Localize(), theme)
{
Margin = 5,
BackgroundColor = theme.MinimalShade,
HAnchor = HAnchor.Right,
VAnchor = VAnchor.Absolute
};
2018-06-09 09:01:24 -07:00
updateButton.Click += (s, e) =>
{
context.item.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
2018-06-09 09:01:24 -07:00
};
mainContainer.AddChild(updateButton);
}
// Init with custom 'UpdateControls' hooks
(context.item as IPropertyGridModifier)?.UpdateControls(new PublicPropertyChange(context, "Update_Button"));
}
return mainContainer;
}
2021-01-18 21:48:31 -08:00
private void AddFunctionButtons(IObject3D item, FlowLayoutWidget mainContainer, ThemeConfig theme)
{
if (item is IEditorButtonProvider editorButtonProvider)
{
foreach (var editorButtonData in editorButtonProvider.GetEditorButtonsData())
{
var editorButton = new TextButton(editorButtonData.Name, theme)
{
Margin = 5,
ToolTipText = editorButtonData.HelpText,
BackgroundColor = theme.MinimalShade,
2021-01-18 21:48:31 -08:00
};
2021-02-24 18:17:14 -08:00
if (editorButtonData.PrimaryAction)
{
theme.ApplyPrimaryActionStyle(editorButton);
}
var row = new SettingsRow("".Localize(), null, editorButton, theme);
editorButtonData.SetStates(editorButton, row);
2021-01-18 21:48:31 -08:00
editorButton.Click += (s, e) =>
{
editorButtonData.Action?.Invoke();
};
mainContainer.AddChild(row);
2021-01-18 21:48:31 -08:00
}
}
}
private static GuiWidget CreateSettingsRow(EditableProperty property, UIField field, ThemeConfig theme)
{
return new SettingsRow(property.DisplayName.Localize(), property.Description, field.Content, theme);
}
private static FlowLayoutWidget CreateSettingsColumn(EditableProperty property, UIField field, bool fullWidth = false)
2018-07-02 14:50:51 -07:00
{
return CreateSettingsColumn(property.DisplayName.Localize(), field, property.Description, fullWidth: fullWidth);
2018-07-02 14:50:51 -07:00
}
private static FlowLayoutWidget CreateSettingsColumn(EditableProperty property)
{
return CreateSettingsColumn(property.DisplayName.Localize(), property.Description);
}
private static FlowLayoutWidget CreateSettingsColumn(string labelText, UIField field, string toolTipText = null, bool fullWidth = false)
2018-07-02 15:09:02 -07:00
{
var row = new FlowLayoutWidget()
{
HAnchor = HAnchor.Stretch
};
2020-10-10 16:18:14 -07:00
if (!fullWidth)
{
row.AddChild(new HorizontalSpacer());
}
2018-07-02 15:09:02 -07:00
row.AddChild(field.Content);
var column = CreateSettingsColumn(labelText, toolTipText);
2018-07-02 15:09:02 -07:00
column.AddChild(row);
2018-07-02 15:09:02 -07:00
return column;
}
private static FlowLayoutWidget CreateSettingsColumn(string labelText, string toolTipText = null)
{
var theme = AppContext.Theme;
var column = new FlowLayoutWidget(FlowDirection.TopToBottom)
{
HAnchor = HAnchor.Stretch,
Padding = new BorderDouble(9, 5, 5, 5), // Use hard-coded 9 pixel left margin to match SettingsRow
ToolTipText = toolTipText
};
var label = SettingsRow.CreateSettingsLabel(labelText, toolTipText, theme.TextColor);
label.VAnchor = VAnchor.Absolute;
label.HAnchor = HAnchor.Left;
column.AddChild(label);
return column;
}
public static IEnumerable<EditableProperty> GetEditablePropreties(IObject3D item)
2018-04-01 16:06:31 -07:00
{
return item.GetType().GetProperties(OwnedPropertiesOnly)
.Where(pi => (AllowedTypes.Contains(pi.PropertyType) || pi.PropertyType.IsEnum)
&& pi.GetGetMethod() != null
&& pi.GetSetMethod() != null)
.Select(p => new EditableProperty(p, item));
}
public static IEnumerable<EditableProperty> GetExecutableFunctions(IObject3D item)
{
BindingFlags buttonFunctionsOnly = BindingFlags.Public | BindingFlags.Instance | BindingFlags.DeclaredOnly;
return item.GetType().GetProperties(buttonFunctionsOnly)
.Where(pi => (AllowedTypes.Contains(pi.PropertyType) || pi.PropertyType.IsEnum)
2018-04-01 16:06:31 -07:00
&& pi.GetGetMethod() != null
&& pi.GetSetMethod() != null)
.Select(p => new EditableProperty(p, item));
}
2018-06-09 10:28:06 -07:00
public static GuiWidget CreatePropertyEditor(EditableProperty property, UndoBuffer undoBuffer, PPEContext context, ThemeConfig theme)
{
var object3D = property.Item;
2018-06-09 08:56:16 -07:00
var propertyGridModifier = property.Item as IPropertyGridModifier;
GuiWidget rowContainer = null;
// Get reflected property value once, then test for each case below
var propertyValue = property.Value;
void RegisterValueChanged(UIField field, Func<string, object> valueFromString, Func<object, string> valueToString = null)
{
field.ValueChanged += (s, e) =>
2018-01-21 14:38:36 -08:00
{
var newValue = field.Value;
var oldValue = property.Value.ToString();
if (valueToString != null)
{
oldValue = valueToString(property.Value);
}
2018-10-20 12:49:28 -07:00
2019-04-22 09:23:56 -07:00
// field.Content
if (undoBuffer != null)
2018-10-20 12:49:28 -07:00
{
undoBuffer.AddAndDo(new UndoRedoActions(() =>
{
property.SetValue(valueFromString(oldValue));
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
propertyGridModifier?.UpdateControls(new PublicPropertyChange(context, property.PropertyInfo.Name));
},
() =>
{
property.SetValue(valueFromString(newValue));
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
propertyGridModifier?.UpdateControls(new PublicPropertyChange(context, property.PropertyInfo.Name));
}));
}
else
2018-10-20 12:49:28 -07:00
{
property.SetValue(valueFromString(newValue));
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
2018-10-20 12:49:28 -07:00
propertyGridModifier?.UpdateControls(new PublicPropertyChange(context, property.PropertyInfo.Name));
}
};
}
2020-05-14 08:24:34 -07:00
var readOnly = property.PropertyInfo.GetCustomAttributes(true).OfType<ReadOnlyAttribute>().FirstOrDefault() != null;
// create a double editor
if (propertyValue is double doubleValue)
{
2019-11-26 19:47:05 -08:00
if (readOnly)
{
var valueField = new TextWidget(string.Format("{0:n}", doubleValue), textColor: theme.TextColor, pointSize: 10)
{
AutoExpandBoundsToText = true
};
2019-11-26 19:47:05 -08:00
rowContainer = new SettingsRow(property.DisplayName.Localize(),
property.Description,
2019-11-26 19:47:05 -08:00
valueField,
theme);
void RefreshField(object s, InvalidateArgs e)
{
if (e.InvalidateType.HasFlag(InvalidateType.DisplayValues))
{
double newValue = (double)property.Value;
valueField.Text = string.Format("{0:n}", newValue);
}
}
object3D.Invalidated += RefreshField;
valueField.Closed += (s, e) => object3D.Invalidated -= RefreshField;
2019-11-26 19:47:05 -08:00
}
else // normal edit row
{
2019-11-26 19:47:05 -08:00
var field = new DoubleField(theme);
field.Initialize(0);
field.DoubleValue = doubleValue;
field.ClearUndoHistory();
2019-11-26 19:47:05 -08:00
RegisterValueChanged(field, (valueString) => { return double.Parse(valueString); });
void RefreshField(object s, InvalidateArgs e)
{
2019-11-26 19:47:05 -08:00
if (e.InvalidateType.HasFlag(InvalidateType.DisplayValues))
{
2019-11-26 19:47:05 -08:00
double newValue = (double)property.Value;
if (newValue != field.DoubleValue)
{
field.DoubleValue = newValue;
}
}
}
2019-11-26 19:47:05 -08:00
object3D.Invalidated += RefreshField;
field.Content.Descendants<InternalTextEditWidget>().First().Name = property.DisplayName + " Edit";
2019-11-26 19:47:05 -08:00
field.Content.Closed += (s, e) => object3D.Invalidated -= RefreshField;
2019-11-26 19:47:05 -08:00
rowContainer = CreateSettingsRow(property, field, theme);
}
}
else if (propertyValue is Color color)
{
var field = new ColorField(theme, object3D.Color);
field.Initialize(0);
field.ValueChanged += (s, e) =>
{
property.SetValue(field.Color);
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
propertyGridModifier?.UpdateControls(new PublicPropertyChange(context, property.PropertyInfo.Name));
};
rowContainer = CreateSettingsRow(property, field, theme);
}
else if (propertyValue is Vector2 vector2)
{
var field = new Vector2Field(theme);
field.Initialize(0);
field.Vector2 = vector2;
field.ClearUndoHistory();
RegisterValueChanged(field,
2019-04-03 15:03:45 -07:00
(valueString) => Vector2.Parse(valueString),
(value) =>
{
var s = ((Vector2)value).ToString();
return s.Substring(1, s.Length - 2);
});
2018-07-02 14:50:51 -07:00
rowContainer = CreateSettingsColumn(property, field);
}
else if (propertyValue is Vector3 vector3)
{
var field = new Vector3Field(theme);
field.Initialize(0);
field.Vector3 = vector3;
field.ClearUndoHistory();
2019-04-03 15:03:45 -07:00
RegisterValueChanged(
field,
(valueString) => Vector3.Parse(valueString),
(value) =>
{
var s = ((Vector3)value).ToString();
return s.Substring(1, s.Length - 2);
});
2019-04-03 15:03:45 -07:00
2018-07-02 14:50:51 -07:00
rowContainer = CreateSettingsColumn(property, field);
}
else if (propertyValue is DirectionVector directionVector)
{
2018-07-12 09:22:28 -07:00
var field = new DirectionVectorField(theme);
field.Initialize(0);
field.SetValue(directionVector);
field.ClearUndoHistory();
field.ValueChanged += (s, e) =>
{
2018-06-09 07:38:27 -07:00
property.SetValue(field.DirectionVector);
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
propertyGridModifier?.UpdateControls(new PublicPropertyChange(context, property.PropertyInfo.Name));
2018-06-08 16:25:35 -07:00
};
rowContainer = CreateSettingsRow(property, field, theme);
}
else if (propertyValue is DirectionAxis directionAxis)
{
rowContainer = CreateSettingsColumn(property);
2018-07-12 09:22:28 -07:00
var field1 = new DirectionVectorField(theme);
field1.Initialize(0);
field1.ClearUndoHistory();
2019-06-11 14:05:35 -07:00
field1.SetValue(new DirectionVector()
{
Normal = directionAxis.Normal
});
rowContainer.AddChild(new SettingsRow("Axis".Localize(), null, field1.Content, theme));
2018-06-08 16:25:35 -07:00
// the direction axis
// the distance from the center of the part
// create a double editor
var field2 = new Vector3Field(theme);
field2.Initialize(0);
2018-07-02 14:50:51 -07:00
field2.Vector3 = directionAxis.Origin - property.Item.Children.First().GetAxisAlignedBoundingBox().Center;
field2.ClearUndoHistory();
var row2 = CreateSettingsColumn("Offset".Localize(), field2);
2018-06-08 16:20:57 -07:00
2018-06-08 21:08:56 -07:00
// update this when changed
2019-04-22 09:23:56 -07:00
void UpdateData(object s, InvalidateArgs e)
{
2018-07-02 14:50:51 -07:00
field2.Vector3 = ((DirectionAxis)property.Value).Origin - property.Item.Children.First().GetAxisAlignedBoundingBox().Center;
2019-04-22 09:23:56 -07:00
}
property.Item.Invalidated += UpdateData;
field2.Content.Closed += (s, e) =>
2018-06-08 16:25:35 -07:00
{
2019-04-22 09:23:56 -07:00
property.Item.Invalidated -= UpdateData;
2018-06-08 16:25:35 -07:00
};
// update functions
field1.ValueChanged += (s, e) =>
{
property.SetValue(new DirectionAxis()
{
Normal = field1.DirectionVector.Normal,
2018-07-02 14:50:51 -07:00
Origin = property.Item.Children.First().GetAxisAlignedBoundingBox().Center + field2.Vector3
});
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
propertyGridModifier?.UpdateControls(new PublicPropertyChange(context, property.PropertyInfo.Name));
};
field2.ValueChanged += (s, e) =>
{
property.SetValue(new DirectionAxis()
{
Normal = field1.DirectionVector.Normal,
2018-07-02 14:50:51 -07:00
Origin = property.Item.Children.First().GetAxisAlignedBoundingBox().Center + field2.Vector3
});
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
propertyGridModifier?.UpdateControls(new PublicPropertyChange(context, property.PropertyInfo.Name));
};
rowContainer.AddChild(row2);
}
else if (propertyValue is SelectedChildren childSelector)
{
var showAsList = property.PropertyInfo.GetCustomAttributes(true).OfType<ShowAsListAttribute>().FirstOrDefault() != null;
if (showAsList)
{
UIField field = new ChildrenSelectorListField(property, theme);
field.Initialize(0);
RegisterValueChanged(field,
(valueString) =>
{
var childrenSelector = new SelectedChildren();
foreach (var child in valueString.Split(','))
{
childrenSelector.Add(child);
}
2019-04-22 09:23:56 -07:00
return childrenSelector;
});
rowContainer = CreateSettingsRow(property, field, theme);
}
2018-11-29 13:41:24 -08:00
else // show the subtract editor for boolean subtract and subtract and replace
{
rowContainer = CreateSettingsColumn(property);
2019-02-11 18:16:05 -08:00
if (property.Item is OperationSourceContainerObject3D sourceContainer)
{
2020-10-16 16:25:11 -07:00
Action selected = null;
if (!(context.item.GetType().GetCustomAttributes(typeof(ShowUpdateButtonAttribute), true).FirstOrDefault() is ShowUpdateButtonAttribute showUpdate))
{
selected = () =>
{
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
};
}
rowContainer.AddChild(CreateSourceChildSelector(childSelector, sourceContainer, theme, selected));
2019-02-11 18:16:05 -08:00
}
else
{
rowContainer.AddChild(CreateSelector(childSelector, property.Item, theme));
}
}
}
else if (propertyValue is ImageBuffer imageBuffer)
{
rowContainer = CreateSettingsColumn(property);
2020-05-18 21:47:35 -07:00
rowContainer.AddChild(new ImageWidget(imageBuffer)
{
HAnchor = HAnchor.Left,
Margin = new BorderDouble(0, 3)
});
}
#if !__ANDROID__
else if (propertyValue is List<string> stringList)
{
2019-05-21 16:04:48 -07:00
var field = new SurfacedEditorsField(theme, property.Item);
field.Initialize(0);
field.ListValue = stringList;
field.ValueChanged += (s, e) =>
{
property.SetValue(field.ListValue);
};
rowContainer = CreateSettingsColumn(property, field);
rowContainer.Descendants<HorizontalSpacer>().FirstOrDefault()?.Close();
}
#endif
// create a int editor
else if (propertyValue is int intValue)
{
2020-02-19 22:07:09 -08:00
if (readOnly)
{
2020-05-23 11:11:09 -07:00
var valueField = new TextWidget(string.Format("{0:n0}", intValue),
2020-05-14 08:24:34 -07:00
textColor: theme.TextColor,
pointSize: 10)
{
AutoExpandBoundsToText = true
};
2020-05-14 08:24:34 -07:00
2020-02-19 22:07:09 -08:00
rowContainer = new SettingsRow(property.DisplayName.Localize(),
property.Description,
2020-02-19 22:07:09 -08:00
valueField,
theme);
void RefreshField(object s, InvalidateArgs e)
{
if (e.InvalidateType.HasFlag(InvalidateType.DisplayValues))
{
int newValue = (int)property.Value;
valueField.Text = string.Format("{0:n0}", newValue);
}
}
object3D.Invalidated += RefreshField;
valueField.Closed += (s, e) => object3D.Invalidated -= RefreshField;
2020-02-19 22:07:09 -08:00
}
else // normal edit row
{
2020-02-19 22:07:09 -08:00
var field = new IntField(theme);
field.Initialize(0);
field.IntValue = intValue;
field.ClearUndoHistory();
2020-02-19 22:07:09 -08:00
RegisterValueChanged(field, (valueString) => { return int.Parse(valueString); });
void RefreshField(object s, InvalidateArgs e)
{
2020-02-19 22:07:09 -08:00
if (e.InvalidateType.HasFlag(InvalidateType.DisplayValues))
{
2020-02-19 22:07:09 -08:00
int newValue = (int)property.Value;
if (newValue != field.IntValue)
{
field.IntValue = newValue;
}
}
}
2020-02-19 22:07:09 -08:00
object3D.Invalidated += RefreshField;
field.Content.Closed += (s, e) => object3D.Invalidated -= RefreshField;
2020-02-19 22:07:09 -08:00
rowContainer = CreateSettingsRow(property, field, theme);
}
}
else if (propertyValue is bool boolValue)
{
2019-04-22 09:23:56 -07:00
// create a bool editor
2018-04-12 17:45:50 -07:00
var field = new ToggleboxField(theme);
field.Initialize(0);
field.Checked = boolValue;
RegisterValueChanged(field,
(valueString) => { return valueString == "1"; },
2019-04-22 09:23:56 -07:00
(value) => { return ((bool)value) ? "1" : "0"; });
rowContainer = CreateSettingsRow(property, field, theme);
}
else if (propertyValue is string stringValue)
{
if (readOnly)
{
2020-05-18 21:47:35 -07:00
rowContainer = new WrappedTextWidget(stringValue,
textColor: theme.TextColor,
pointSize: 10)
{
Margin = 5
};
}
else // normal edit row
{
var multiLineEditAttribute = property.PropertyInfo.GetCustomAttributes(true).OfType<MultiLineEditAttribute>().FirstOrDefault();
if (multiLineEditAttribute != null)
{
// create a a multi-line string editor
var field = new MultilineStringField(theme);
field.Initialize(0);
field.SetValue(stringValue, false);
field.ClearUndoHistory();
field.Content.HAnchor = HAnchor.Stretch;
// field.Content.MinimumSize = new Vector2(0, 200 * GuiWidget.DeviceScale);
RegisterValueChanged(field, (valueString) => valueString);
rowContainer = CreateSettingsColumn(property, field, fullWidth: true);
}
else
{
// create a string editor
var field = new TextField(theme);
field.Initialize(0);
field.SetValue(stringValue, false);
field.ClearUndoHistory();
field.Content.HAnchor = HAnchor.Stretch;
RegisterValueChanged(field, (valueString) => valueString);
rowContainer = CreateSettingsRow(property, field, theme);
var label = rowContainer.Children.First();
var spacer = rowContainer.Children.OfType<HorizontalSpacer>().FirstOrDefault();
spacer.HAnchor = HAnchor.Absolute;
spacer.Width = Math.Max(0, 100 - label.Width);
}
}
}
else if (propertyValue is char charValue)
{
2019-04-22 09:23:56 -07:00
// create a char editor
var field = new CharField(theme);
2018-06-08 18:19:06 -07:00
field.Initialize(0);
field.SetValue(charValue.ToString(), false);
field.ClearUndoHistory();
2018-06-08 18:19:06 -07:00
field.ValueChanged += (s, e) =>
{
2018-06-09 07:38:27 -07:00
property.SetValue(Convert.ToChar(field.Value));
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
propertyGridModifier?.UpdateControls(new PublicPropertyChange(context, property.PropertyInfo.Name));
};
2018-06-08 18:19:06 -07:00
rowContainer = CreateSettingsRow(property, field, theme);
}
else if (property.PropertyType.IsEnum)
{
2019-04-22 09:23:56 -07:00
// create an enum editor
2018-06-09 07:14:38 -07:00
UIField field;
var enumDisplayAttribute = property.PropertyInfo.GetCustomAttributes(true).OfType<EnumDisplayAttribute>().FirstOrDefault();
var addToSettingsRow = true;
if (enumDisplayAttribute != null)
2018-06-09 07:14:38 -07:00
{
field = new EnumDisplayField(property, enumDisplayAttribute, theme)
{
2020-05-18 21:47:35 -07:00
InitialValue = propertyValue.ToString(),
};
if (enumDisplayAttribute.Mode == EnumDisplayAttribute.PresentationMode.Tabs)
{
addToSettingsRow = false;
}
2018-06-09 07:14:38 -07:00
}
else
{
2019-10-22 17:42:26 -07:00
if (property.PropertyType == typeof(NamedTypeFace))
{
field = new FontSelectorField(property, theme);
}
else
{
field = new EnumField(property, theme);
}
2018-06-09 07:14:38 -07:00
}
field.Initialize(0);
RegisterValueChanged(field,
(valueString) =>
{
return Enum.Parse(property.PropertyType, valueString);
});
field.ValueChanged += (s, e) =>
{
if (property.Value.ToString() != field.Value)
{
property.SetValue(Enum.Parse(property.PropertyType, field.Value));
object3D?.Invalidate(new InvalidateArgs(context.item, InvalidateType.Properties));
propertyGridModifier?.UpdateControls(new PublicPropertyChange(context, property.PropertyInfo.Name));
}
};
if (addToSettingsRow)
{
rowContainer = CreateSettingsRow(property, field, theme);
}
else
{
2020-05-14 08:24:34 -07:00
// field.Content.Margin = new BorderDouble(3, 0);
field.Content.HAnchor = HAnchor.Stretch;
rowContainer = field.Content;
}
}
else if (propertyValue is IObject3D item
2019-02-04 08:41:08 -08:00
&& ApplicationController.Instance.Extensions.GetEditorsForType(property.PropertyType)?.FirstOrDefault() is IObject3DEditor iObject3DEditor)
{
2019-04-22 09:23:56 -07:00
// Use known IObject3D editors
rowContainer = iObject3DEditor.Create(item, undoBuffer, theme);
}
// remember the row name and widget
context.editRows.Add(property.PropertyInfo.Name, rowContainer);
return rowContainer;
}
2020-10-16 16:25:11 -07:00
private static GuiWidget CreateSourceChildSelector(SelectedChildren childSelector, OperationSourceContainerObject3D sourceContainer, ThemeConfig theme, Action selectionChanged)
2019-02-11 18:16:05 -08:00
{
GuiWidget tabContainer = new FlowLayoutWidget(FlowDirection.TopToBottom)
{
Margin = new BorderDouble(0, 3, 0, 0),
};
2019-02-11 18:16:05 -08:00
var parentOfSubtractTargets = sourceContainer.SourceContainer.DescendantsAndSelfMultipleChildrenFirstOrSelf();
var sourceChildren = parentOfSubtractTargets.Children.ToList();
2019-02-11 18:16:05 -08:00
var objectChecks = new Dictionary<ICheckbox, IObject3D>();
var radioSiblings = new List<GuiWidget>();
for (int i = 0; i < sourceChildren.Count; i++)
{
var itemIndex = i;
var child = sourceChildren[itemIndex];
var rowContainer = new FlowLayoutWidget()
{
Padding = new BorderDouble(15, 0, 0, 3)
};
2019-02-11 18:16:05 -08:00
GuiWidget selectWidget;
if (sourceChildren.Count == 2)
{
var radioButton = new RadioButton(string.IsNullOrWhiteSpace(child.Name) ? $"{itemIndex}" : $"{child.Name}")
{
Checked = childSelector.Contains(child.ID),
TextColor = theme.TextColor,
Margin = 0,
2019-02-11 18:16:05 -08:00
};
radioSiblings.Add(radioButton);
radioButton.SiblingRadioButtonList = radioSiblings;
selectWidget = radioButton;
}
else
{
selectWidget = new CheckBox(string.IsNullOrWhiteSpace(child.Name) ? $"{itemIndex}" : $"{child.Name}")
{
Checked = childSelector.Contains(child.ID),
TextColor = theme.TextColor,
2019-02-11 18:16:05 -08:00
};
}
objectChecks.Add((ICheckbox)selectWidget, child);
rowContainer.AddChild(selectWidget);
var checkBox = selectWidget as ICheckbox;
checkBox.CheckedStateChanged += (s, e) =>
{
if (s is ICheckbox checkbox)
{
if (checkBox.Checked)
{
if (!childSelector.Contains(objectChecks[checkbox].ID))
2019-02-11 18:16:05 -08:00
{
childSelector.Add(objectChecks[checkbox].ID);
2019-02-11 18:16:05 -08:00
}
}
else
{
if (childSelector.Contains(objectChecks[checkbox].ID))
2019-02-11 18:16:05 -08:00
{
childSelector.Remove(objectChecks[checkbox].ID);
2019-02-11 18:16:05 -08:00
}
}
2020-10-16 16:25:11 -07:00
selectionChanged?.Invoke();
2019-02-11 18:16:05 -08:00
}
};
tabContainer.AddChild(rowContainer);
}
return tabContainer;
}
private static GuiWidget CreateSelector(SelectedChildren childSelector, IObject3D parent, ThemeConfig theme)
{
GuiWidget tabContainer = new FlowLayoutWidget(FlowDirection.TopToBottom);
void UpdateSelectColors(bool selectionChanged = false)
{
foreach (var child in parent.Children.ToList())
{
2018-06-20 08:09:35 -07:00
using (child.RebuildLock())
{
2018-06-20 08:09:35 -07:00
if (selectionChanged)
{
child.Visible = true;
}
}
}
}
tabContainer.Closed += (s, e) => UpdateSelectColors();
var children = parent.Children.ToList();
2019-04-22 09:23:56 -07:00
var objectChecks = new Dictionary<ICheckbox, IObject3D>();
2019-04-22 09:23:56 -07:00
var radioSiblings = new List<GuiWidget>();
for (int i = 0; i < children.Count; i++)
{
var itemIndex = i;
var child = children[itemIndex];
2019-04-22 09:23:56 -07:00
var rowContainer = new FlowLayoutWidget();
GuiWidget selectWidget;
if (children.Count == 2)
{
var radioButton = new RadioButton(string.IsNullOrWhiteSpace(child.Name) ? $"{itemIndex}" : $"{child.Name}")
{
Checked = childSelector.Contains(child.ID),
TextColor = theme.TextColor
};
radioSiblings.Add(radioButton);
radioButton.SiblingRadioButtonList = radioSiblings;
selectWidget = radioButton;
}
else
{
selectWidget = new CheckBox(string.IsNullOrWhiteSpace(child.Name) ? $"{itemIndex}" : $"{child.Name}")
{
Checked = childSelector.Contains(child.ID),
TextColor = theme.TextColor
};
}
objectChecks.Add((ICheckbox)selectWidget, child);
rowContainer.AddChild(selectWidget);
2019-04-22 09:23:56 -07:00
var checkBox = selectWidget as ICheckbox;
checkBox.CheckedStateChanged += (s, e) =>
{
if (s is ICheckbox checkbox)
{
if (checkBox.Checked)
{
if (!childSelector.Contains(objectChecks[checkbox].ID))
{
childSelector.Add(objectChecks[checkbox].ID);
}
}
else
{
if (childSelector.Contains(objectChecks[checkbox].ID))
{
childSelector.Remove(objectChecks[checkbox].ID);
}
}
2019-04-22 09:23:56 -07:00
if (parent is MeshWrapperObject3D meshWrapper)
{
2018-06-20 08:09:35 -07:00
using (meshWrapper.RebuildLock())
{
meshWrapper.ResetMeshWrapperMeshes(Object3DPropertyFlags.All, CancellationToken.None);
}
}
UpdateSelectColors(true);
}
};
tabContainer.AddChild(rowContainer);
UpdateSelectColors();
}
return tabContainer;
}
public static void AddUnlockLinkIfRequired(IObject3D item, GuiWidget editControlsContainer, ThemeConfig theme)
{
(string url, GuiWidget markdownWidget)? unlockdata = null;
if (item.GetType().GetCustomAttributes(typeof(RequiresPermissionsAttribute), true).FirstOrDefault() is RequiresPermissionsAttribute unlockLink
&& !ApplicationController.Instance.UserHasPermission(item))
{
unlockdata = ApplicationController.Instance.GetUnlockData?.Invoke(item, theme);
}
else if (!item.Persistable)
{
// find the first self or child that is not authorized
2020-09-13 21:20:59 -07:00
var permission = item.DescendantsAndSelf()
.Where(i => !i.Persistable && !ApplicationController.Instance.UserHasPermission(i));
2020-09-13 21:20:59 -07:00
if (permission.Any())
{
2020-09-13 21:20:59 -07:00
var unlockItem = permission.First();
unlockdata = ApplicationController.Instance.GetUnlockData?.Invoke(unlockItem, theme);
}
}
if (unlockdata != null && !string.IsNullOrEmpty(unlockdata.Value.url))
{
if (unlockdata.Value.markdownWidget != null)
{
unlockdata.Value.markdownWidget.VAnchor = VAnchor.Fit;
editControlsContainer.AddChild(unlockdata.Value.markdownWidget);
}
editControlsContainer.AddChild(GetUnlockRow(theme, unlockdata.Value.url));
}
}
public static GuiWidget GetUnlockRow(ThemeConfig theme, string url)
2018-12-31 12:12:05 -08:00
{
var detailsLink = new TextIconButton("Unlock".Localize(), StaticData.Instance.LoadIcon("locked.png", 16, 16, theme.InvertIcons), theme)
2018-12-31 12:12:05 -08:00
{
Margin = 5,
ToolTipText = "Visit MatterHackers.com to Purchase".Localize()
2018-12-31 12:12:05 -08:00
};
detailsLink.Click += (s, e) =>
{
2020-09-22 17:52:57 -07:00
ApplicationController.LaunchBrowser(url);
2018-12-31 12:12:05 -08:00
};
theme.ApplyPrimaryActionStyle(detailsLink);
return new SettingsRow("Demo Mode".Localize(), null, detailsLink, theme);
2018-12-31 12:12:05 -08:00
}
private void AddWebPageLinkIfRequired(PPEContext context, FlowLayoutWidget editControlsContainer, ThemeConfig theme)
{
2019-04-22 09:23:56 -07:00
if (context.item.GetType().GetCustomAttributes(typeof(WebPageLinkAttribute), true).FirstOrDefault() is WebPageLinkAttribute unlockLink)
{
var detailsLink = new TextIconButton(unlockLink.Name.Localize(), StaticData.Instance.LoadIcon("internet.png", 16, 16, theme.InvertIcons), theme)
{
BackgroundColor = theme.MinimalShade,
ToolTipText = unlockLink.Url,
};
detailsLink.Click += (s, e) =>
{
2020-09-22 17:52:57 -07:00
ApplicationController.LaunchBrowser(unlockLink.Url);
};
// website row
editControlsContainer.AddChild(new SettingsRow("Website".Localize(), null, detailsLink, theme));
}
}
}
}