File: System\Windows\Forms\Controls\PropertyGrid\PropertyGridInternal\PropertyDescriptorGridEntry.cs
Web Access
Project: src\src\System.Windows.Forms\src\System.Windows.Forms.csproj (System.Windows.Forms)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.ComponentModel;
using System.ComponentModel.Design;
using System.Drawing;
using System.Drawing.Design;
 
namespace System.Windows.Forms.PropertyGridInternal;
 
internal partial class PropertyDescriptorGridEntry : GridEntry
{
    private TypeConverter? _exceptionConverter;
    private UITypeEditor? _exceptionEditor;
    private bool _isSerializeContentsProperty;
    private bool? _parensAroundName;
    private IPropertyValueUIService? _propertyValueUIService;
    protected IEventBindingService? _eventBindings;
    private bool _propertyValueUIServiceChecked;
    private PropertyValueUIItem[]? _propertyValueUIItems;
    private Rectangle[]? _uiItemRects;
    private bool _readOnlyVerified;
    private bool _forceRenderReadOnly;
    private string? _helpKeyword;
    private string? _toolTipText;
    private readonly bool _hide;
 
    private const int LogicalImageSize = 8;
    private static int s_imageSize = LogicalImageSize;
    private static bool s_isScalingInitialized;
 
    private static IEventBindingService? s_targetBindingService;
    private static IComponent? s_targetComponent;
    private static EventDescriptor? s_targetEventdesc;
 
    internal PropertyDescriptorGridEntry(
        PropertyGrid ownerGrid,
        GridEntry parent,
        PropertyDescriptor propertyDescriptor,
        bool hide)
        : base(ownerGrid, parent)
    {
        _hide = hide;
        PropertyDescriptor = propertyDescriptor;
        Initialize(propertyDescriptor);
    }
 
    public override bool AllowMerge
        => PropertyDescriptor.GetAttribute<MergablePropertyAttribute>()?.IsDefaultAttribute() ?? true;
 
    protected override AttributeCollection Attributes => PropertyDescriptor.Attributes;
 
    public override string? HelpKeyword
    {
        get
        {
            if (_helpKeyword is not null)
            {
                return _helpKeyword;
            }
 
            object? owner = GetValueOwner();
            if (owner is null)
            {
                return null;
            }
 
            if (PropertyDescriptor.TryGetAttribute(out HelpKeywordAttribute? helpAttribute) &&
                !helpAttribute.IsDefaultAttribute())
            {
                return helpAttribute.HelpKeyword;
            }
            else if (this is ImmutablePropertyDescriptorGridEntry)
            {
                _helpKeyword = PropertyName;
 
                GridEntry entry = this;
 
                while (entry.ParentGridEntry is not null)
                {
                    entry = entry.ParentGridEntry;
 
                    // For value classes, the equality will never work, so just try the type equality.
                    if (entry.PropertyValue == owner
                        || (owner.GetType().IsValueType && owner.GetType() == entry.PropertyValue?.GetType()))
                    {
                        _helpKeyword = $"{entry.PropertyName}.{_helpKeyword}";
                        break;
                    }
                }
            }
            else
            {
                string? typeName;
 
                Type? componentType = PropertyDescriptor.ComponentType;
 
                if (componentType is not null && componentType.IsCOMObject)
                {
                    typeName = TypeDescriptor.GetClassName(owner);
                }
                else
                {
                    // Make sure this property is declared on a class that is related to the component we're
                    // looking at. If it's not, it could be a shadow property so we need to try to find the
                    // real property.
                    Type ownerType = owner.GetType();
                    if (componentType is not null && (!componentType.IsPublic || !componentType.IsAssignableFrom(ownerType)))
                    {
                        PropertyDescriptor? componentProperty = TypeDescriptor.GetProperties(ownerType)[PropertyName!];
                        componentType = componentProperty?.ComponentType;
                    }
 
                    typeName = componentType is null ? TypeDescriptor.GetClassName(owner) : componentType.FullName;
                }
 
                _helpKeyword = $"{typeName}.{PropertyDescriptor.Name}";
            }
 
            return _helpKeyword;
        }
    }
 
    internal override string? LabelToolTipText => _toolTipText ?? base.LabelToolTipText;
 
    internal override bool Enumerable => base.Enumerable && !IsPropertyReadOnly;
 
    internal virtual bool IsPropertyReadOnly => PropertyDescriptor.IsReadOnly;
 
    public override bool IsValueEditable => _exceptionConverter is null && !IsPropertyReadOnly && base.IsValueEditable;
 
    public override bool NeedsDropDownButton => base.NeedsDropDownButton && !IsPropertyReadOnly;
 
    internal bool ParensAroundName
    {
        get
        {
            if (!_parensAroundName.HasValue)
            {
                _parensAroundName = PropertyDescriptor.GetAttribute<ParenthesizePropertyNameAttribute>()?.NeedParenthesis ?? false;
            }
 
            return _parensAroundName.Value;
        }
    }
 
    public override string PropertyCategory
    {
        get
        {
            string? category = PropertyDescriptor.Category;
            if (category is null || category.Length == 0)
            {
                category = base.PropertyCategory;
            }
 
            return category;
        }
    }
 
    public sealed override PropertyDescriptor PropertyDescriptor { get; }
 
    public override string? PropertyDescription => PropertyDescriptor.Description;
 
    public override string? PropertyLabel
    {
        get
        {
            string? label = PropertyDescriptor.DisplayName;
            if (ParensAroundName)
            {
                label = $"({label})";
            }
 
            return label;
        }
    }
 
    /// <summary>
    ///  Returns non-localized name of this property.
    /// </summary>
    public override string? PropertyName => PropertyDescriptor.Name;
 
    public override Type? PropertyType => PropertyDescriptor.PropertyType;
 
    /// <summary>
    ///  Gets or sets the value for the property that is represented by this GridEntry.
    /// </summary>
    public override object? PropertyValue
    {
        get
        {
            try
            {
                object? value = GetPropertyValue(GetValueOwner());
 
                if (_exceptionConverter is not null)
                {
                    // Undo the exception converter.
                    ClearFlags();
                    _exceptionConverter = null;
                    _exceptionEditor = null;
                }
 
                return value;
            }
            catch (Exception e)
            {
                if (_exceptionConverter is null)
                {
                    // Clear the flags.
                    ClearFlags();
                    _exceptionConverter = new ExceptionConverter();
                    _exceptionEditor = new ExceptionEditor();
                }
 
                return e;
            }
        }
        set
        {
            SetPropertyValue(GetValueOwner(), value, reset: false, undoText: null);
        }
    }
 
    private IPropertyValueUIService? PropertyValueUIService
    {
        get
        {
            if (!_propertyValueUIServiceChecked && _propertyValueUIService is null)
            {
                _propertyValueUIService = (IPropertyValueUIService?)GetService(typeof(IPropertyValueUIService));
                _propertyValueUIServiceChecked = true;
            }
 
            return _propertyValueUIService;
        }
    }
 
    public override bool ShouldRenderReadOnly
    {
        get
        {
            if (base.ForceReadOnly || _forceRenderReadOnly)
            {
                return true;
            }
 
            // If read only editable is set, make sure it's valid.
            if (PropertyDescriptor.IsReadOnly && !_readOnlyVerified && GetFlagSet(Flags.ReadOnlyEditable))
            {
                Type? propertyType = PropertyType;
 
                if (propertyType is not null && (propertyType.IsArray || propertyType.IsValueType || propertyType.IsPrimitive))
                {
                    SetFlag(Flags.ReadOnlyEditable, false);
                    SetFlag(Flags.RenderReadOnly, true);
                    _forceRenderReadOnly = true;
                }
            }
 
            _readOnlyVerified = true;
 
            return base.ShouldRenderReadOnly && !_isSerializeContentsProperty && !NeedsModalEditorButton;
        }
    }
 
    internal override TypeConverter TypeConverter
    {
        get
        {
            if (_exceptionConverter is not null)
            {
                return _exceptionConverter;
            }
 
            _typeConverter ??= PropertyDescriptor.Converter;
 
            return base.TypeConverter;
        }
    }
 
    internal override UITypeEditor? UITypeEditor
    {
        get
        {
            if (_exceptionEditor is not null)
            {
                return _exceptionEditor;
            }
 
            Editor = (UITypeEditor?)PropertyDescriptor.GetEditor(typeof(UITypeEditor));
 
            return base.UITypeEditor;
        }
    }
 
    internal override void EditPropertyValue(PropertyGridView gridView)
    {
        base.EditPropertyValue(gridView);
 
        if (!IsValueEditable)
        {
            if (PropertyDescriptor.TryGetAttribute(out RefreshPropertiesAttribute? refreshAttribute) &&
                !refreshAttribute.RefreshProperties.Equals(RefreshProperties.None))
            {
                OwnerGridView?.Refresh(fullRefresh: refreshAttribute.Equals(RefreshPropertiesAttribute.All));
            }
        }
    }
 
    internal override Point GetLabelToolTipLocation(int mouseX, int mouseY)
    {
        if (_propertyValueUIItems is not null
            && _uiItemRects is not null
            && OwnerGridView is not null)
        {
            for (int i = 0; i < _propertyValueUIItems.Length; i++)
            {
                if (_uiItemRects[i].Contains(mouseX, OwnerGridView.GridEntryHeight / 2))
                {
                    _toolTipText = _propertyValueUIItems[i].ToolTip;
                    return new Point(mouseX, mouseY);
                }
            }
        }
 
        _toolTipText = null;
        return base.GetLabelToolTipLocation(mouseX, mouseY);
    }
 
    private object? GetPropertyValue(object? owner)
    {
        if (owner is ICustomTypeDescriptor descriptor)
        {
            owner = descriptor.GetPropertyOwner(PropertyDescriptor);
        }
 
        return PropertyDescriptor.GetValue(owner);
    }
 
    protected void Initialize(PropertyDescriptor propertyDescriptor)
    {
        _isSerializeContentsProperty = propertyDescriptor.SerializationVisibility == DesignerSerializationVisibility.Content;
 
        if (!_hide && IsPropertyReadOnly)
        {
            SetFlag(Flags.TextEditable, false);
        }
 
        if (_isSerializeContentsProperty && TypeConverter.GetPropertiesSupported())
        {
            SetFlag(Flags.Expandable, true);
        }
    }
 
    protected virtual void NotifyParentsOfChanges(GridEntry? entry)
    {
        // See if we need to notify the parent(s) up the chain.
        while (entry is PropertyDescriptorGridEntry propertyEntry &&
            propertyEntry.PropertyDescriptor.Attributes.Contains(NotifyParentPropertyAttribute.Yes))
        {
            // Find the next parent property with a different value owner.
            object? owner = entry.GetValueOwner();
 
            // When owner is an instance of a value type we can't use == in the following while condition.
            bool isValueType = owner?.GetType().IsValueType ?? false;
 
            // Find the next property descriptor with a different parent.
            while (entry is not PropertyDescriptorGridEntry
                || isValueType ? owner!.Equals(entry.GetValueOwner()) : owner == entry.GetValueOwner())
            {
                entry = entry.ParentGridEntry;
                if (entry is null)
                {
                    break;
                }
            }
 
            // Fire the change on the owner.
            if (entry is not null)
            {
                owner = entry.GetValueOwner();
 
                if (owner is not null)
                {
                    ComponentChangeService?.OnComponentChanging(owner, entry.PropertyDescriptor);
                    ComponentChangeService?.OnComponentChanged(owner, entry.PropertyDescriptor);
                }
 
                // Clear the value so it paints correctly next time.
                entry.ClearCachedValues(clearChildren: false);
                OwnerGridView?.InvalidateGridEntryValue(entry);
            }
        }
    }
 
    protected override bool SendNotification(object? owner, Notify type)
    {
        if (owner is ICustomTypeDescriptor descriptor)
        {
            owner = descriptor.GetPropertyOwner(PropertyDescriptor);
        }
 
        switch (type)
        {
            case Notify.Reset:
 
                SetPropertyValue(owner, value: null, reset: true, undoText: string.Format(SR.PropertyGridResetValue, PropertyName));
                if (_propertyValueUIItems is not null)
                {
                    for (int i = 0; i < _propertyValueUIItems.Length; i++)
                    {
                        _propertyValueUIItems[i].Reset();
                    }
                }
 
                _propertyValueUIItems = null;
                return false;
            case Notify.CanReset:
                try
                {
                    return PropertyDescriptor.CanResetValue(owner!)
                        || (_propertyValueUIItems is not null && _propertyValueUIItems.Length > 0);
                }
                catch
                {
                    if (_exceptionConverter is null)
                    {
                        ClearFlags();
                        _exceptionConverter = new ExceptionConverter();
                        _exceptionEditor = new ExceptionEditor();
                    }
 
                    return false;
                }
 
            case Notify.ShouldPersist:
                try
                {
                    return PropertyDescriptor.ShouldSerializeValue(owner!);
                }
                catch
                {
                    if (_exceptionConverter is null)
                    {
                        ClearFlags();
                        _exceptionConverter = new ExceptionConverter();
                        _exceptionEditor = new ExceptionEditor();
                    }
 
                    return false;
                }
 
            case Notify.DoubleClick:
            case Notify.Return:
                _eventBindings ??= this.GetService<IEventBindingService>();
 
                if (_eventBindings?.GetEvent(PropertyDescriptor) is not null)
                {
                    return ViewEvent(owner, newHandler: null, eventDescriptor: null, alwaysNavigate: true);
                }
 
                break;
        }
 
        return false;
    }
 
    public override void OnComponentChanged()
    {
        base.OnComponentChanged();
 
        // If we got this it means someone called ITypeDescriptorContext.OnComponentChanged.
        // We need to echo that change up the inheritance in case the owner object isn't a sited component.
        NotifyParentsOfChanges(this);
    }
 
    public override bool OnMouseClick(int x, int y, int count, MouseButtons button)
    {
        if (_propertyValueUIItems is not null
            && _uiItemRects is not null
            && count == 2
            && ((button & MouseButtons.Left) == MouseButtons.Left)
            && OwnerGridView is not null)
        {
            for (int i = 0; i < _propertyValueUIItems.Length; i++)
            {
                if (_uiItemRects[i].Contains(x, OwnerGridView.GridEntryHeight / 2))
                {
                    _propertyValueUIItems[i].InvokeHandler(this, PropertyDescriptor, _propertyValueUIItems[i]);
                    return true;
                }
            }
        }
 
        return base.OnMouseClick(x, y, count, button);
    }
 
    public override void PaintLabel(
        Graphics g,
        Rectangle rect,
        Rectangle clipRect,
        bool selected,
        bool paintFullLabel)
    {
        base.PaintLabel(g, rect, clipRect, selected, paintFullLabel);
 
        IPropertyValueUIService? uiService = PropertyValueUIService;
 
        if (uiService is null)
        {
            return;
        }
 
        _propertyValueUIItems = uiService.GetPropertyUIValueItems(this, PropertyDescriptor);
 
        if (_propertyValueUIItems is null)
        {
            return;
        }
 
        if (_uiItemRects is null || _uiItemRects.Length != _propertyValueUIItems.Length)
        {
            _uiItemRects = new Rectangle[_propertyValueUIItems.Length];
        }
 
        if (!s_isScalingInitialized)
        {
            s_imageSize = ScaleHelper.ScaleToInitialSystemDpi(LogicalImageSize);
            s_isScalingInitialized = true;
        }
 
        for (int i = 0; i < _propertyValueUIItems.Length; i++)
        {
            _uiItemRects[i] = new Rectangle(
                rect.Right - ((s_imageSize + 1) * (i + 1)),
                (rect.Height - s_imageSize) / 2,
                s_imageSize,
                s_imageSize);
            g.DrawImage(_propertyValueUIItems[i].Image, _uiItemRects[i]);
        }
 
        if (OwnerGridView is { } ownerGridView)
        {
            ownerGridView.LabelPaintMargin = (s_imageSize + 1) * _propertyValueUIItems.Length;
        }
    }
 
    private object? SetPropertyValue(
        object? owner,
        object? value,
        bool reset,
        string? undoText)
    {
        DesignerTransaction? transaction = null;
        try
        {
            object? oldValue = GetPropertyValue(owner);
 
            if (value is not null && value.Equals(oldValue))
            {
                return value;
            }
 
            ClearCachedValues();
 
            IDesignerHost? host = DesignerHost;
 
            transaction = host?.CreateTransaction(undoText ?? string.Format(SR.PropertyGridSetValue, PropertyDescriptor.Name));
 
            // Usually IComponent things are sited and this notification will be fired automatically by
            // the PropertyDescriptor. However, for non-IComponent sub objects or sub objects that are
            // non-sited sub components, we need to manually fire the notification.
 
            bool needChangeNotify = owner is not IComponent component || component.Site is null;
 
            if (needChangeNotify)
            {
                try
                {
                    if (owner is null)
                    {
                        throw new InvalidOperationException();
                    }
 
                    ComponentChangeService?.OnComponentChanging(owner, PropertyDescriptor);
                }
                catch (CheckoutException coEx)
                {
                    if (coEx == CheckoutException.Canceled)
                    {
                        return oldValue;
                    }
 
                    throw;
                }
            }
 
            bool wasExpanded = InternalExpanded;
            int childCount = -1;
            if (wasExpanded)
            {
                childCount = ChildCount;
            }
 
            // See if we need to refresh the property browser.
            var refresh = PropertyDescriptor.GetAttribute<RefreshPropertiesAttribute>();
            bool needsRefresh = wasExpanded || (refresh is not null && !refresh.RefreshProperties.Equals(RefreshProperties.None));
 
            if (needsRefresh)
            {
                DisposeChildren();
            }
 
            // Determine if this is an event being created, and if so, navigate to the event code
 
            EventDescriptor? eventDescriptor = null;
 
            // This is possibly an event. Check it out.
            if (owner is not null && value is string)
            {
                _eventBindings ??= this.GetService<IEventBindingService>();
                eventDescriptor = _eventBindings?.GetEvent(PropertyDescriptor);
 
                // For a merged set of properties, the event binding service won't
                // find an event. So, we ask the type descriptor directly.
                if (eventDescriptor is null)
                {
                    // If we have a merged property descriptor, pull out one of the elements.
                    object? eventObj = owner;
 
                    if (PropertyDescriptor is MergePropertyDescriptor && owner is Array objArray)
                    {
                        if (objArray.Length > 0)
                        {
                            eventObj = objArray.GetValue(0);
                        }
                    }
 
                    // TypeDescriptor will return an empty collection if component is null.
                    eventDescriptor = TypeDescriptor.GetEvents(eventObj!)[PropertyDescriptor.Name];
                }
            }
 
            bool setSuccessful = false;
            try
            {
                if (reset)
                {
                    if (owner is null)
                    {
                        throw new InvalidOperationException();
                    }
 
                    PropertyDescriptor.ResetValue(owner);
                }
                else if (eventDescriptor is not null)
                {
                    ViewEvent(owner, (string?)value, eventDescriptor, false);
                }
                else
                {
                    // Not an event
                    SetPropertyValueCore(owner, value);
                }
 
                setSuccessful = true;
 
                // Now notify the change service that the change was successful.
                if (needChangeNotify)
                {
                    // We have already checked if owner is null.
                    ComponentChangeService?.OnComponentChanged(owner!, PropertyDescriptor, oldValue: null, value);
                }
 
                NotifyParentsOfChanges(this);
            }
            finally
            {
                if (needsRefresh && OwnerGridView is not null)
                {
                    RecreateChildren(childCount);
                    if (setSuccessful)
                    {
                        OwnerGridView.Refresh(refresh is not null && refresh.Equals(RefreshPropertiesAttribute.All));
                    }
                }
            }
        }
        catch (CheckoutException checkoutEx)
        {
            transaction?.Cancel();
            transaction = null;
 
            if (checkoutEx == CheckoutException.Canceled)
            {
                return null;
            }
 
            throw;
        }
        catch
        {
            transaction?.Cancel();
            transaction = null;
 
            throw;
        }
        finally
        {
            transaction?.Commit();
        }
 
        return owner;
    }
 
    protected void SetPropertyValueCore(object? owner, object? newValue)
    {
        // Store the current cursor and set it to the HourGlass.
        Cursor? oldCursor = Cursor.Current;
        try
        {
            Cursor.Current = Cursors.WaitCursor;
 
            object? realOwner = owner;
 
            if (realOwner is ICustomTypeDescriptor descriptor)
            {
                realOwner = descriptor.GetPropertyOwner(PropertyDescriptor);
            }
 
            // Check the type of the object we are modifying. If it's a value type or an array,
            // we need to modify the object and push the value back up to the parent.
 
            bool treatAsValueType = false;
 
            if (ParentGridEntry is not null)
            {
                Type? propertyType = ParentGridEntry.PropertyType;
                if (propertyType is not null)
                {
                    treatAsValueType = propertyType.IsValueType || propertyType.IsArray;
                }
            }
 
            if (realOwner is not null)
            {
                PropertyDescriptor.SetValue(realOwner, newValue);
 
                // Since the value that we modified may not be stored by the parent property, we need to push this
                // value back into the parent. An example here is Size or Location, which return Point objects that
                // are unconnected to the object they relate to. So we modify the Point object and push it back
                // into the object we got it from.
 
                if (treatAsValueType)
                {
                    GridEntry? parent = ParentGridEntry;
                    if (parent is not null && parent.IsValueEditable)
                    {
                        parent.PropertyValue = owner;
                    }
                }
            }
        }
        finally
        {
            // Flip back to the old cursor.
            Cursor.Current = oldCursor;
        }
    }
 
    /// <summary>
    ///  Navigates code to the given event.
    /// </summary>
    protected bool ViewEvent(
        object? owner,
        string? newHandler,
        EventDescriptor? eventDescriptor,
        bool alwaysNavigate)
    {
        object? value = GetPropertyValue(owner);
 
        string? handler = value as string;
 
        if (handler is null && value is not null && TypeConverter is not null && TypeConverter.CanConvertTo(typeof(string)))
        {
            handler = TypeConverter.ConvertToString(value);
        }
 
        if (newHandler is null && !string.IsNullOrEmpty(handler))
        {
            newHandler = handler;
        }
        else if (handler == newHandler && !string.IsNullOrEmpty(newHandler))
        {
            return true;
        }
 
        var component = owner as IComponent;
 
        if (component is null && PropertyDescriptor is MergePropertyDescriptor)
        {
            // It's possible that multiple objects are selected, and we're trying to create an event for each of them.
            if (owner is Array array && array.Length > 0)
            {
                component = array.GetValue(0) as IComponent;
            }
        }
 
        if (component is null)
        {
            return false;
        }
 
        if (PropertyDescriptor.IsReadOnly)
        {
            return false;
        }
 
        if (eventDescriptor is null)
        {
            _eventBindings ??= this.GetService<IEventBindingService>();
            eventDescriptor = _eventBindings?.GetEvent(PropertyDescriptor);
        }
 
        IDesignerHost? host = DesignerHost;
        DesignerTransaction? transaction = null;
 
        try
        {
            // This check can cause exceptions if the event has unreferenced dependencies, which we want to catch.
            // This must be done before the transaction is started or the commit/cancel will also throw.
            if (eventDescriptor is null)
            {
                throw new InvalidOperationException();
            }
 
            if (eventDescriptor.EventType is null)
            {
                return false;
            }
 
            if (host is not null)
            {
                string componentName = component.Site?.Name ?? component.GetType().Name;
                transaction = host.CreateTransaction(string.Format(
                    SR.WindowsFormsSetEvent,
                    $"{componentName}.{PropertyName}"));
            }
 
            _eventBindings ??= component.Site?.GetService<IEventBindingService>();
 
            newHandler ??= _eventBindings?.CreateUniqueMethodName(component, eventDescriptor);
 
            if (newHandler is not null)
            {
                // Now walk through all the matching methods to see if this one exists.
                // If it doesn't we'll want to show code.
                if (_eventBindings is not null)
                {
                    bool methodExists = false;
                    foreach (string methodName in _eventBindings.GetCompatibleMethods(eventDescriptor))
                    {
                        if (newHandler == methodName)
                        {
                            methodExists = true;
                            break;
                        }
                    }
 
                    if (!methodExists)
                    {
                        alwaysNavigate = true;
                    }
                }
 
                try
                {
                    PropertyDescriptor.SetValue(owner, newHandler);
                }
                catch (InvalidOperationException ex)
                {
                    transaction?.Cancel();
                    transaction = null;
 
                    OwnerGridView?.ShowInvalidMessage(newHandler, ex);
 
                    return false;
                }
            }
 
            if (alwaysNavigate && _eventBindings is not null)
            {
                s_targetBindingService = _eventBindings;
                s_targetComponent = component;
                s_targetEventdesc = eventDescriptor;
                Application.Idle += ShowCodeIdle;
            }
        }
        catch
        {
            transaction?.Cancel();
            transaction = null;
            throw;
        }
        finally
        {
            transaction?.Commit();
        }
 
        return true;
    }
 
    /// <summary>
    ///  Displays the user code for the given event. This will return true if the user
    ///  code could be displayed, or false otherwise.
    /// </summary>
    private static void ShowCodeIdle(object? sender, EventArgs e)
    {
        Application.Idle -= ShowCodeIdle;
        if (s_targetBindingService is not null)
        {
            // We set s_targetComponent and s_targetEventdesc with s_targetBindingService,
            // so we don't expect them to be null here.
            s_targetBindingService.ShowCode(s_targetComponent!, s_targetEventdesc!);
            s_targetBindingService = null;
            s_targetComponent = null;
            s_targetEventdesc = null;
        }
    }
 
    protected override GridEntryAccessibleObject GetAccessibilityObject() =>
        new PropertyDescriptorGridEntryAccessibleObject(this);
}