CSLA.NET 6.0.0
CSLA .NET is a software development framework that helps you build a reusable, maintainable object-oriented business layer for your app.
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Events Macros Pages
UndoableHandler.cs
Go to the documentation of this file.
1#if !NETFX_CORE && !IOS
2
3//-----------------------------------------------------------------------
4// <copyright file="UndoableHandler.cs" company="Marimer LLC">
5// Copyright (c) Marimer LLC. All rights reserved.
6// Website: https://cslanet.com
7// </copyright>
8// <summary>no summary</summary>
9//-----------------------------------------------------------------------
10using System;
11using System.Collections.Generic;
12using System.Reflection;
13#if NET5_0_OR_GREATER
14using System.Runtime.Loader;
15
16using Csla.Runtime;
17#endif
18using Csla.Reflection;
19
20namespace Csla.Core
21{
22 internal static class UndoableHandler
23 {
24#if NET5_0_OR_GREATER
25 private static readonly Dictionary<Type, Tuple<string, List<DynamicMemberHandle>>> _undoableFieldCache =
26 new Dictionary<Type, Tuple<string, List<DynamicMemberHandle>>>();
27#else
28 private static readonly Dictionary<Type, List<DynamicMemberHandle>> _undoableFieldCache = new Dictionary<Type, List<DynamicMemberHandle>>();
29#endif
30
31 public static List<DynamicMemberHandle> GetCachedFieldHandlers(Type type)
32 {
33 List<DynamicMemberHandle> handlers;
34
35#if NET5_0_OR_GREATER
36 var found = _undoableFieldCache.TryGetValue(type, out var handlersInfo);
37
38 handlers = handlersInfo?.Item2;
39
40 if (!found)
41 {
42
43 lock (_undoableFieldCache)
44 {
45 found = _undoableFieldCache.TryGetValue(type, out handlersInfo);
46
47 handlers = handlersInfo?.Item2;
48
49 if (!found)
50 {
51 var newHandlers = BuildHandlers(type);
52
53 var cacheInstance = AssemblyLoadContextManager.CreateCacheInstance(type, newHandlers, OnAssemblyLoadContextUnload);
54
55 _undoableFieldCache.Add(type, cacheInstance);
56
57 handlers = newHandlers;
58 }
59 }
60 }
61#else
62 if (!_undoableFieldCache.TryGetValue(type, out handlers))
63 {
64 var newHandlers = BuildHandlers(type);
65
66 lock (_undoableFieldCache) //ready to add, lock
67 {
68 if(!_undoableFieldCache.TryGetValue(type, out handlers))
69 {
70 _undoableFieldCache.Add(type, newHandlers);
71
72 handlers = newHandlers;
73 }
74 }
75 }
76#endif
77
78 return handlers;
79 }
80
81 private static List<DynamicMemberHandle> BuildHandlers(Type type)
82 {
83 var handlers = new List<DynamicMemberHandle>();
84 // get the list of fields in this type
85 var fields = type.GetFields(
86 BindingFlags.NonPublic |
87 BindingFlags.Instance |
88 BindingFlags.Public);
89
90 foreach (FieldInfo field in fields)
91 {
92 // make sure we process only our variables
93 if (field.DeclaringType == type)
94 {
95 // see if this field is marked as not undoable
96 if (!NotUndoableField(field))
97 {
98 // the field is undoable, so it needs to be processed.
99 handlers.Add(new DynamicMemberHandle(field));
100 }
101 }
102 }
103 return handlers;
104 }
105
106 private static bool NotUndoableField(FieldInfo field)
107 {
108 // see if this field is marked as not undoable or IsInitOnly (ie: readonly property)
109 return field.IsInitOnly || Attribute.IsDefined(field, typeof(NotUndoableAttribute));
110 }
111#if NET5_0_OR_GREATER
112
113 private static void OnAssemblyLoadContextUnload(AssemblyLoadContext context)
114 {
115 lock (_undoableFieldCache)
116 AssemblyLoadContextManager.RemoveFromCache(_undoableFieldCache, context);
117 }
118#endif
119 }
120}
121
122#endif