C # Назначение свойства по умолчанию для класса и оператора = - PullRequest
6 голосов
/ 11 января 2009

Задача 1:

У меня есть простое приложение winforms, и я хочу, чтобы DataBind мое свойство Person.Name в текстовое поле. Имя имеет тип StringField. Первоначально я определил свойство Name как String. Привязка данных отлично работает на типах значений, таких как String. Мне бы хотелось, чтобы свойство StringField.Value было свойством по умолчанию для StringField. Я хочу видеть значение StringField.Value в textBox, а не в тексте "FieldApp.StringField".

Задача 2:

Я хотел бы иметь возможность назначить строку для StringField с помощью operator =. Это назначение приведет к установке члена StringField.Value.

Можно ли это сделать?

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;

namespace FieldApp
{
    public class StringField
    {
        public string Value { get; set; }    
    }

    public class Person
    {

        //private String _Name;
        //public String Name
        //{
        //    get { return _Name; }
        //    set { _Name = value; }
        //}

        //public Person(string name)
        //{
        //    Name = name;
        //}

        private StringField _Name;
        public StringField Name
        {
            get { return _Name; }
            set { _Name = value; }
        }

        public Person(string name)
        {
            Name = new StringField();
            Name.Value = name;
        }
    }

    public partial class FieldAppForm : Form
    {
        Person person = new Person("steve");

        public FieldAppForm()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            //our form contains a button1 and textBox1

            //this compiles
            person.Name.Value = "steve";

            //this does not. Is there anyway to accomplish this?
            person.Name = "steve";

            //steve appears in the textbox 
            textBox1.DataBindings.Add("Text", person, "Name.Value");

            //FieldApp.StringField appears in the textbox 
            textBox1.DataBindings.Add("Text", person, "Name");
        }
    }
}

Ответы [ 5 ]

10 голосов
/ 11 января 2009

Вы можете создать перегрузку оператора implisit. Затем вы можете создать Stringfield из таких строк:

StringField field = "value of new object";
string value=(string)field;

Знайте, что это создает новый объект StringField. Я бы не советовал вам делать это.

[System.Diagnostics.DebuggerDisplay("{Value}")]
public class StringField
{
    public string Value { get; set; }
    public static implicit operator StringField(string s)
    {
        return new StringField { Value = s };
    }

    public static explicit operator string(StringField f)
    {
        return f.Value;
    }
    public override string ToString()
    {
        return Value;
    }
}
2 голосов
/ 11 января 2009

Повторная привязка данных, для некоторых целей привязки (PropertyGrid, DataGridView и т. Д.) Вы можете сделать это с помощью TypeConverter (см. Ниже). К сожалению, это не похоже на работу с TextBox, поэтому я думаю, что ваш лучший вариант - просто добавить свойство shim (как уже предлагалось):

string NameString
{
   get { return Name.Value; }
   set { Name.Value = value; } // or new blah...
}

(и привязка к NameString)

В прошлом я использовал пользовательские реализации PropertyDescriptor, чтобы обойти это, но это не стоило просто для этого.

В любом случае, пример TypeConverter (работает с PropertyGrid и DataGridView):

[TypeConverter(typeof(StringFieldConverter))]
public class StringField
{
    public StringField() : this("") { }
    public StringField(string value) { Value = value; }
    public string Value { get; private set; }
}

class StringFieldConverter : TypeConverter
{
    public override bool CanConvertFrom(
        ITypeDescriptorContext context, Type sourceType)
    {
        return sourceType == typeof(string)
            || base.CanConvertFrom(context, sourceType);
    }
    public override object ConvertFrom(
        ITypeDescriptorContext context,
        System.Globalization.CultureInfo culture,
        object value)
    {
        string s = value as string;
        if (s != null) return new StringField(s);
        return base.ConvertFrom(context, culture, value);
    }
    public override bool CanConvertTo(
        ITypeDescriptorContext context, Type destinationType)
    {
        return destinationType == typeof(string)
            || base.CanConvertTo(context, destinationType);
    }
    public override object ConvertTo(
        ITypeDescriptorContext context,
        System.Globalization.CultureInfo culture,
        object value, Type destinationType)
    {
        if (destinationType == typeof(string) && value != null
            && value is StringField)
        {
            return ((StringField)value).Value;
        }
        return base.ConvertTo(context, culture, value, destinationType);
    }
}
1 голос
/ 11 января 2009

Вы можете реализовать назначение, предоставив оператор преобразования. Учитывая характер вашего класса, вы также должны переопределить методы Object:

public class StringField {
  public string Value { get; set; }
  public static implicit operator StringField(string value) {
    StringField sf = new StringField();
    sf.Value = value;
    return sf;
  }
  public override string ToString() {
    return Value;
  }
  public override bool Equals(object obj) {
    if (obj == null || !(obj is StringField)) return false;
    return 0 == string.Compare(Value, (obj as StringField).Value);
  }
  public override int GetHashCode() {
    return Value.GetHashCode();
  }
}
0 голосов
/ 11 января 2009

Оператор присваивания не может быть переопределен в C #. Тем не менее, вы можете иметь свойство для преобразования типов и выставлять его классу

0 голосов
/ 11 января 2009

Вы можете скрыть StringField, сопоставив свойство Name с полем Name.Value внутри вашего класса.

, чтобы вы могли определить свойство Name следующим образом:

string Name 
{
   get { return _name.Value; }
   set { _name.Value = value; }
}

Здесь _name - ваша переменная StringField.

...