|
// 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.Serialization;
using System.Globalization;
namespace System.Windows.Forms;
/// <summary>
/// A TypeConverter for LinkLabel.Link objects. Access this class through the TypeDescriptor.
/// </summary>
public class LinkConverter : TypeConverter
{
/// <summary>
/// Determines if this converter can convert an object in the given source
/// type to the native type of the converter.
/// </summary>
public override bool CanConvertFrom(ITypeDescriptorContext? context, Type sourceType)
{
if (sourceType == typeof(string))
{
return true;
}
return base.CanConvertFrom(context, sourceType);
}
/// <summary>
/// Gets a value indicating whether this converter can convert an object to the given
/// destination type using the context.
/// </summary>
public override bool CanConvertTo(ITypeDescriptorContext? context, Type? destinationType)
{
if (destinationType == typeof(InstanceDescriptor) || destinationType == typeof(string))
{
return true;
}
return base.CanConvertTo(context, destinationType);
}
/// <summary>
/// Converts the given object to the converter's native type.
/// </summary>
public override object? ConvertFrom(ITypeDescriptorContext? context, CultureInfo? culture, object value)
{
if (value is string valueStr)
{
ReadOnlySpan<char> text = valueStr.AsSpan().Trim();
if (text.IsEmpty)
{
return null;
}
// Parse 2 integer values.
culture ??= CultureInfo.CurrentCulture;
Span<int> values = stackalloc int[2];
if (!TypeConverterHelper.TryParseAsSpan(context, culture, text, values))
{
throw new ArgumentException(
string.Format(
SR.TextParseFailedFormat,
valueStr,
"Start, Length"));
}
return new LinkLabel.Link(values[0], values[1]);
}
return base.ConvertFrom(context, culture, value);
}
/// <summary>
/// Converts the given object to another type. The most common types to convert
/// are to and from a string object. The default implementation will make a call
/// to ToString on the object if the object is valid and if the destination
/// type is string. If this cannot convert to the destination type, this will
/// throw a NotSupportedException.
/// </summary>
public override object? ConvertTo(ITypeDescriptorContext? context, CultureInfo? culture, object? value, Type destinationType)
{
if (value is LinkLabel.Link link)
{
if (destinationType == typeof(string))
{
culture ??= CultureInfo.CurrentCulture;
string sep = culture.TextInfo.ListSeparator + " ";
TypeConverter intConverter = TypeDescriptor.GetConverter(typeof(int));
string?[] args =
[
intConverter.ConvertToString(context, culture, link.Start),
intConverter.ConvertToString(context, culture, link.Length)
];
return string.Join(sep, args);
}
if (destinationType == typeof(InstanceDescriptor))
{
if (link.LinkData is null)
{
return new InstanceDescriptor(
typeof(LinkLabel.Link).GetConstructor([typeof(int), typeof(int)]),
new object[] { link.Start, link.Length },
true);
}
return new InstanceDescriptor(
typeof(LinkLabel.Link).GetConstructor([typeof(int), typeof(int), typeof(object)]),
new object[] { link.Start, link.Length, link.LinkData },
true);
}
}
return base.ConvertTo(context, culture, value, destinationType);
}
}
|