winforms 动态设置PropertyGrid中属性的只读特性

dgiusagp  于 2022-11-16  发布在  其他
关注(0)|答案(7)|浏览(298)

我有一个PropertyGrid,用来显示helper类中的属性。我将helper类分配给PropertyGrid,如下所示:

myPropertyGrid.SelectedObject = mySettingsHelper;

在helper类中,我在设计时将ReadOnlyAttribute赋值为:

[DisplayName("DisplayExA"),
Description("DescriptionExA"),
ReadOnlyAttribute(true)]
public string PropertyA { get; set; }

[DisplayName("DisplayExB"),
Description("DescriptionExB"),
ReadOnlyAttribute(false)]
public string PropertyB { get; set; }

[DisplayName("DisplayExC"),
Description("DescriptionExC"),
ReadOnlyAttribute(true)]
public string PropertyC { get; set; }

但是现在我需要能够在运行时动态地更改单个属性的这个属性。根据某些条件,这些属性中的一些可能需要是只读的或非只读的。我如何在运行时动态地进行更改呢?

编辑:

我尝试了下面的代码,但这会为对象的每个示例设置ReadOnly属性!我想对每个对象都这样做。有时一个对象的PropertyA可能是只读的,而另一个对象的PropertyA可能不是只读的。

public static class PropertyReadOnlyHelper
{
    public static  void SetReadOnly(object container, string name, bool value)
    {
        try
        {
            PropertyDescriptor descriptor = TypeDescriptor.GetProperties(container.GetType())[name];
            ReadOnlyAttribute attribute = (ReadOnlyAttribute)descriptor.Attributes[typeof(ReadOnlyAttribute)];
            FieldInfo fieldToChange = attribute.GetType().GetField("isReadOnly",
                                                System.Reflection.BindingFlags.NonPublic |
                                                System.Reflection.BindingFlags.Instance);
            fieldToChange.SetValue(attribute, value);
        }
        catch { }
    }
}
rbpvctlc

rbpvctlc1#

我能够使用这篇CodeProject文章中的库来完成我所需要的(只读属性的对象级赋值)。令人高兴的是,它使我能够仍然使用.NET PropertyGrid,而只是使用自定义属性来处理动态设置。

k5ifujac

k5ifujac2#

使用反射获取ReadOnlyAttribute类的示例引用,然后切换该示例上的IsReadOnly属性。最后,如果需要,通过将其SelectedObjects设置为null并重置它,重新选择PropertyGrid中的项目。您可能也可以使用PropertyGrid RefreshTabs方法来完成此操作,我不确定。
编辑:
不幸的是,IsReadOnly属性本身是只读的...在这种情况下,我们必须使用反射来更改IsReadOnly属性的支持字段的值。

w6mmgewl

w6mmgewl3#

添加只读

TextBoxID.Attributes.Add("readonly","true");

移除只读

TextBoxID.Attributes.Remove("readonly");
bbuxkriu

bbuxkriu4#

在PropertyGrid中动态设置属性的browsablereadonly属性通常需要同时进行,而且它们是类似的工作
经过几次修改后,***Reza Aghaei***关于“Hide some properties in PropertyGrid at run-time“的伟大回答也适用于操作readonly属性。

public class CustomObjectWrapper : CustomTypeDescriptor
{
    public object WrappedObject { get; private set; }
    public List<string> BrowsableProperties { get; private set; }
    public List<string> ReadonlyProperties { get; private set; }

    public CustomObjectWrapper(object o)
        : base(TypeDescriptor.GetProvider(o).GetTypeDescriptor(o))
    {
        WrappedObject = o;
        BrowsableProperties = new List<string>() { "Text", "BackColor" };
        ReadonlyProperties = new List<string>() { "Font" };
    }
    public override PropertyDescriptorCollection GetProperties()
    {
        return this.GetProperties(new Attribute[] { });
    }
    public override PropertyDescriptorCollection GetProperties(Attribute[] attributes)
    {
        List<PropertyDescriptor> result = new List<PropertyDescriptor>();

        IEnumerable<PropertyDescriptor> properties = base.GetProperties(attributes).Cast<PropertyDescriptor>()
            .Where(p => BrowsableProperties.Contains(p.Name));//unbrowsable filtering

        foreach (var p in properties)
        {
            PropertyDescriptor resultPropertyDescriptor = null;

            //handle being readonly 
            if (ReadonlyProperties.Contains(p.Name))
            {
                List<Attribute> atts = p.Attributes.Cast<Attribute>().ToList();
                atts.RemoveAll(a => a.GetType().Equals(typeof(ReadOnlyAttribute)));//remove any readonly attribute
                atts.Add(new ReadOnlyAttribute(true));//add "readonly=true" attribute

                resultPropertyDescriptor = TypeDescriptor.CreateProperty(WrappedObject.GetType(), p, atts.ToArray());
            }
            else
            {
                resultPropertyDescriptor = TypeDescriptor.CreateProperty(WrappedObject.GetType(), p, p.Attributes.Cast<Attribute>().ToArray());
            }

            if (resultPropertyDescriptor != null)
                result.Add(resultPropertyDescriptor);

        }

        return new PropertyDescriptorCollection(result.ToArray());
    }
}

用途:

propertyGrid1.SelectedObject = new CustomObjectWrapper(myobject);
yptwkmov

yptwkmov5#

Please try the code below.

  
[CategoryAttribute("2. LINE"), DisplayNameAttribute("Spline Line Tension"),
 DescriptionAttribute("Chart's Spline Line Tension "), ReadOnlyAttribute(false)]
public float _PG_SplineTension
{
    get
    {
        bool lbReadyOnly = true;
        SetPropertyReadOnly("_PG_SplineTension", lbReadyOnly);
        return this.cfSplineTension;
   }
    set { this.cfSplineTension = value; }
}


private void SetPropertyReadOnly(string lsProperty, bool lbIsReadOnly)
{
    PropertyDescriptor descriptor = TypeDescriptor.GetProperties(this.GetType())[lsProperty];
    ReadOnlyAttribute attribute = (ReadOnlyAttribute)

    descriptor.Attributes[typeof(ReadOnlyAttribute)];
    FieldInfo fieldToChange = attribute.GetType().GetField("isReadOnly",
        System.Reflection.BindingFlags.NonPublic |
        System.Reflection.BindingFlags.Instance);
    fieldToChange.SetValue(attribute, lbIsReadOnly);
}
tkqqtvp1

tkqqtvp16#

下面的代码是一个很好的例子:

private void SetReadonly ( object o, bool value )
{
  foreach ( PropertyInfo property in o.GetType().GetProperties() )
    if ( property.GetCustomAttribute<ReadOnlyAttribute>() != null )
    {
      Attribute readOnly = TypeDescriptor.GetProperties( o.GetType() )[property.Name].Attributes[typeof( ReadOnlyAttribute )];
      readOnly.GetType().GetField( nameof( ReadOnlyAttribute.IsReadOnly ), BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.IgnoreCase ).SetValue( readOnly, value );
    }
}
xcitsw88

xcitsw887#

查看此页面:
https://www.codeproject.com/Articles/152945/Enabling-disabling-properties-at-runtime-in-the-Pr
引自上述帖子:
静态地定义类的每个属性的ReadOnly属性为你想要的值是很重要的。2否则,在运行时改变属性会错误地修改类的每个属性的属性。
在运行时使用反射修改目标属性的“ReadOnly”特性以实现您的目标。设置一个属性应用于所有属性的问题是因为您需要显式地将同一PropertyGrid对象中的所有属性设置为ReadOnly特性以避免该问题。

[RefreshProperties(System.ComponentModel.RefreshProperties.All)]
[ReadOnly(false)]
public string Country
{
  get { return mCountry; }
  set
      {
      mCountry = value;
      PropertyDescriptor descriptor = TypeDescriptor.GetProperties(this.GetType())["State"];
      ReadOnlyAttribute attribute = (ReadOnlyAttribute)
                                    descriptor.Attributes[typeof(ReadOnlyAttribute)];
      FieldInfo fieldToChange = attribute.GetType().GetField("isReadOnly", 
                                       System.Reflection.BindingFlags.NonPublic | 
                                       System.Reflection.BindingFlags.Instance);
      fieldToChange.SetValue(attribute, mCountry != "U.S.");
      }
}

[ReadOnly(true)]
public string State
{
   get { return mState; }
   set { mState = value; }
}

相关问题