【问题标题】:DataGridView custom column accessing design time parameters at run timeDataGridView 自定义列在运行时访问设计时参数
【发布时间】:2013-07-17 23:23:39
【问题描述】:

我一直在玩一些示例代码,它为 DataGridView 实现了一个 datepicker 列。

我苦苦挣扎的地方是如何在我想将 Format 设置为 DateFormat 的 CalendarEditingControl 中适当地访问设计时间参数(感谢 Bradley Smith 提供了关于覆盖 Clone 方法的提示)。

我在下面提供了完整的示例,如果您只是扫描评论,看看我想在哪里使用 DateFormat: //*********** 我'我想在此处将 Format 设置为 DataGridViewCalendarColumn.DateFormat 中提供的设计时间值 ***********//

谢谢

马特

使用系统; 使用 System.Windows.Forms;

命名空间 MyApplication.Components { 公共类 DataGridViewCalendarColumn : DataGridViewColumn { 公共 DataGridViewCalendarColumn() :基础(新日历单元()) { }

    public override DataGridViewCell CellTemplate
    {
        get
        {
            return base.CellTemplate;
        }
        set
        {
            // Ensure that the cell used for the template is a CalendarCell. 
            if (value != null &&
                !value.GetType().IsAssignableFrom(typeof(CalendarCell)))
            {
                throw new InvalidCastException("Must be a CalendarCell");
            }
            base.CellTemplate = value;

        }
    }

    public DateTimePickerFormat DateFormat { get; set; }

    public override object Clone()
    {
        DataGridViewCalendarColumn clone = (DataGridViewCalendarColumn)base.Clone();
        clone.DateFormat = this.DateFormat;
        return clone;
    }

}

public class CalendarCell : DataGridViewTextBoxCell
{

    public CalendarCell(): base()
    {
        // Use the short date format. 
        this.Style.Format = "d";
    }

    public override void InitializeEditingControl(int rowIndex, object initialFormattedValue, DataGridViewCellStyle dataGridViewCellStyle)
    {
        // Set the value of the editing control to the current cell value. 
        base.InitializeEditingControl(rowIndex, initialFormattedValue,
            dataGridViewCellStyle);
        CalendarEditingControl ctl =
            DataGridView.EditingControl as CalendarEditingControl;
        // Use the default row value when Value property is null. 
        if (this.Value == null)
        {
            ctl.Value = (DateTime)this.DefaultNewRowValue;
        }
        else
        {
            ctl.Value = (DateTime)this.Value;
        }
    }

    public override Type EditType
    {
        get
        {
            // Return the type of the editing control that CalendarCell uses. 
            return typeof(CalendarEditingControl);
        }
    }

    public override Type ValueType
    {
        get
        {
            // Return the type of the value that CalendarCell contains. 

            return typeof(DateTime);
        }
    }

    public override object DefaultNewRowValue
    {
        get
        {
            // Use the current date and time as the default value. 
            return DateTime.Now;
        }
    }


}

class CalendarEditingControl : DateTimePicker, IDataGridViewEditingControl
{
    DataGridView dataGridView;
    private bool valueChanged = false;
    int rowIndex;

    public CalendarEditingControl()
    {
        //*********************** I'd like to set the Format to the design time value provided inDataGridViewCalendarColumn.DateFormat here ***********************//
        //(instead of hard coding it to DateTimePickerFormat.Short)
        this.Format = DateTimePickerFormat.Short;
    }

    // Implements the IDataGridViewEditingControl.EditingControlFormattedValue  
    // property. 
    public object EditingControlFormattedValue
    {
        get
        {
            return this.Value.ToShortDateString();
        }
        set
        {
            if (value is String)
            {
                try
                {
                    // This will throw an exception of the string is  
                    // null, empty, or not in the format of a date. 
                    this.Value = DateTime.Parse((String)value);
                }
                catch
                {
                    // In the case of an exception, just use the  
                    // default value so we're not left with a null 
                    // value. 
                    this.Value = DateTime.Now;
                }
            }
        }
    }

    // Implements the  
    // IDataGridViewEditingControl.GetEditingControlFormattedValue method. 
    public object GetEditingControlFormattedValue(
        DataGridViewDataErrorContexts context)
    {
        return EditingControlFormattedValue;
    }

    // Implements the  
    // IDataGridViewEditingControl.ApplyCellStyleToEditingControl method. 
    public void ApplyCellStyleToEditingControl(
        DataGridViewCellStyle dataGridViewCellStyle)
    {
        this.Font = dataGridViewCellStyle.Font;
        this.CalendarForeColor = dataGridViewCellStyle.ForeColor;
        this.CalendarMonthBackground = dataGridViewCellStyle.BackColor;
    }

    // Implements the IDataGridViewEditingControl.EditingControlRowIndex  
    // property. 
    public int EditingControlRowIndex
    {
        get
        {
            return rowIndex;
        }
        set
        {
            rowIndex = value;
        }
    }

    // Implements the IDataGridViewEditingControl.EditingControlWantsInputKey  
    // method. 
    public bool EditingControlWantsInputKey(
        Keys key, bool dataGridViewWantsInputKey)
    {
        // Let the DateTimePicker handle the keys listed. 
        switch (key & Keys.KeyCode)
        {
            case Keys.Left:
            case Keys.Up:
            case Keys.Down:
            case Keys.Right:
            case Keys.Home:
            case Keys.End:
            case Keys.PageDown:
            case Keys.PageUp:
                return true;
            default:
                return !dataGridViewWantsInputKey;
        }
    }

    // Implements the IDataGridViewEditingControl.PrepareEditingControlForEdit  
    // method. 
    public void PrepareEditingControlForEdit(bool selectAll)
    {
        // No preparation needs to be done.
    }

    // Implements the IDataGridViewEditingControl 
    // .RepositionEditingControlOnValueChange property. 
    public bool RepositionEditingControlOnValueChange
    {
        get
        {
            return false;
        }
    }

    // Implements the IDataGridViewEditingControl 
    // .EditingControlDataGridView property. 
    public DataGridView EditingControlDataGridView
    {
        get
        {
            return dataGridView;
        }
        set
        {
            dataGridView = value;
        }
    }

    // Implements the IDataGridViewEditingControl 
    // .EditingControlValueChanged property. 
    public bool EditingControlValueChanged
    {
        get
        {
            return valueChanged;
        }
        set
        {
            valueChanged = value;
        }
    }

    // Implements the IDataGridViewEditingControl 
    // .EditingPanelCursor property. 
    public Cursor EditingPanelCursor
    {
        get
        {
            return base.Cursor;
        }
    }

    protected override void OnValueChanged(EventArgs eventargs)
    {
        // Notify the DataGridView that the contents of the cell 
        // have changed.
        valueChanged = true;
        this.EditingControlDataGridView.NotifyCurrentCellDirty(true);
        base.OnValueChanged(eventargs);
    }
}

}

【问题讨论】:

    标签: c# datagridview custom-component


    【解决方案1】:

    您需要做的就是将列中的DateFormat 属性应用到编辑控件上。在InitializeEditingControl 方法中,添加以下行:

    ctl.Format = (OwningColumn as DataGridViewCalendarColumn).DateFormat;
    

    这应该可以解决问题。

    【讨论】:

    • 谢谢布拉德利,太好了。最后一个障碍是我在多个列中使用自定义列,例如开始和结束日期,并添加了一个额外的 DefaultDate 属性。正在发生的事情是,即使将两列设置为不同的值,它们也会获得相同的值。我需要管理一组值还是有更优雅的方式?
    • @mattpm 你是指相同的 DateFormat 值还是相同的 DateTime 值?
    • 我的意思是相同的 DateTime 值。我已经通过移动代码 ctl.DefaultDate = (OwningColumn as DataGridViewCalendarColumn).DefaultDate;行前 base.InitializeEditingControl(rowIndex, initialFormattedValue, dataGridViewCellStyle);
    【解决方案2】:

    在 WinForms(以及 WebForms 和 WPF...)中,您无法访问构造函数中的设计时值。相反,框架会调用您的无参数构造函数,然后在您的对象上使用适当的属性设置器来完成初始化。

    在构造函数中为对象的属性设置默认值是明智之举,以防设计者-用户在设计时没有指定所有属性值应该是什么。所以你的构造函数是好的。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2016-09-26
      • 1970-01-01
      • 2012-11-22
      • 1970-01-01
      • 1970-01-01
      • 2011-10-09
      相关资源
      最近更新 更多