chr
2026-04-05 fe750b791d5b517cc4e9bc8e99a9a75139a0cfba
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
using System;
using System.Reflection;
namespace OpenTap
{
    /// <summary>
    /// Marks a factory attribute, for now this can be internal.
    /// </summary>
    interface IFactoryAttribute
    {
        public string FactoryMethodName { get; }
    }
 
    /// <summary> Specifies that another member method can be used to create a value for a given property.</summary>
    public class FactoryAttribute : Attribute, IFactoryAttribute
    {
        /// <summary> The name of the method that can be used to create a value. This can be a private method.</summary>
        public string FactoryMethodName { get; }
        
        /// <summary> Creates an instance of FactoryAttribute. </summary>
        /// <param name="factoryMethodName"></param>
        public FactoryAttribute(string factoryMethodName)
        {
            FactoryMethodName = factoryMethodName;
        }
 
        /// <summary> Creates an object by calling that target member. </summary>
        internal static object Create(object ownerObj, IFactoryAttribute factoryAttribute)
        {
            var type = ownerObj.GetType();
            var fac = type.GetMethod(factoryAttribute.FactoryMethodName,
                BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic);
            // This is a bug, but showing this message in the log is much nicer than a null reference exception.
            if (fac == null)
                throw new Exception(
                    $"Factory method '{factoryAttribute.FactoryMethodName}' not found on object of type '{type.FullName}'");
            return fac.Invoke(ownerObj, []);
        }
 
        // Recursively search for the source of the parameter to try constructing this object
        internal static bool TryCreateFromMember(IParameterMemberData member, IFactoryAttribute factoryAttribute, out object obj)
        {
            obj = null;
            if (member is IParameterMemberData pmd)
            {
                foreach (var p in pmd.ParameterizedMembers)
                {
                    // If this is a concrete member, we should be able to use the source object as a factory source.
                    if (p.Member is MemberData)
                    {
                        obj = Create(p.Source, factoryAttribute);
                        return true;
                    }
 
                    // Otherwise, try recursing further
                    if (p.Member is IParameterMemberData pmd2)
                    {
                        if (TryCreateFromMember(pmd2, factoryAttribute, out obj))
                            return true;
                    }
                }
            }
 
            return false;
        }
    }
 
    /// <summary> Specifies that another member method can be used to create an element value for a given list property.</summary>
    public class ElementFactoryAttribute : Attribute, IFactoryAttribute
    {
        /// <summary> The name of the method that can be used to create a value. This can be a private method.</summary>
        public string FactoryMethodName { get; }
 
        /// <summary> Creates an instance of ElementFactoryAttribute. </summary>
        public ElementFactoryAttribute(string factoryMethodName)
        {
            FactoryMethodName = factoryMethodName;
        }
    }
}