DFUtils.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. namespace InABox.Core
  6. {
  7. public interface IEntityFormUtils
  8. {
  9. public abstract bool CanEditForm(IDigitalFormInstance form, Entity entity);
  10. public abstract Entity NewEntity(DigitalForm form);
  11. public abstract void OnSave(IDigitalFormInstance form, Entity entity);
  12. public abstract IColumns? RequiredEntityColumns();
  13. }
  14. public abstract class EntityFormUtils<TForm, TEntity, TEntityLink> : IEntityFormUtils
  15. where TForm : BaseEntityForm<TEntity, TEntityLink, TForm>
  16. where TEntity : Entity, new()
  17. where TEntityLink : EntityLink<TEntity>, new()
  18. {
  19. public abstract bool CanEditForm(TForm form, TEntity entity);
  20. public abstract TEntity NewEntity(DigitalForm form);
  21. public abstract void OnSave(TForm form, TEntity entity);
  22. public abstract Columns<TEntity>? RequiredEntityColumns();
  23. void IEntityFormUtils.OnSave(IDigitalFormInstance form, Entity entity) => OnSave((TForm)form, (TEntity)entity);
  24. bool IEntityFormUtils.CanEditForm(IDigitalFormInstance form, Entity entity) => CanEditForm((form as TForm)!, (entity as TEntity)!);
  25. Entity IEntityFormUtils.NewEntity(DigitalForm form) => NewEntity(form);
  26. IColumns? IEntityFormUtils.RequiredEntityColumns() => RequiredEntityColumns();
  27. }
  28. public class DelegateEntityFormUtils<TForm, TEntity, TEntityLink> : EntityFormUtils<TForm, TEntity, TEntityLink>
  29. where TForm : BaseEntityForm<TEntity, TEntityLink, TForm>
  30. where TEntity : Entity, new()
  31. where TEntityLink : EntityLink<TEntity>, new()
  32. {
  33. public delegate bool CanEditEvent(TForm form, TEntity entity);
  34. public delegate TEntity NewEntityEvent(DigitalForm form);
  35. public delegate void OnSaveEvent(TForm form, TEntity entity);
  36. public delegate Columns<TEntity> RequiredColumnsEvent();
  37. public CanEditEvent OnCanEdit;
  38. public NewEntityEvent? OnNewEntity;
  39. public OnSaveEvent? SaveEvent;
  40. public RequiredColumnsEvent? RequiredColumns;
  41. public DelegateEntityFormUtils(CanEditEvent canEditForm, NewEntityEvent? onNewEntity = null, OnSaveEvent? onSave = null, RequiredColumnsEvent? requiredColumns = null)
  42. {
  43. OnCanEdit = canEditForm;
  44. OnNewEntity = onNewEntity;
  45. SaveEvent = onSave;
  46. RequiredColumns = requiredColumns;
  47. }
  48. public override bool CanEditForm(TForm form, TEntity entity) => OnCanEdit(form, entity);
  49. public override TEntity NewEntity(DigitalForm form) => OnNewEntity?.Invoke(form) ?? new TEntity();
  50. public override void OnSave(TForm form, TEntity entity) => SaveEvent?.Invoke(form, entity);
  51. public override Columns<TEntity>? RequiredEntityColumns() => RequiredColumns?.Invoke();
  52. }
  53. public static class DFUtils
  54. {
  55. private static List<Type>? _fieldTypes;
  56. public static List<Type> GetFieldTypes(bool includeObsolete = false)
  57. {
  58. _fieldTypes ??= CoreUtils.Entities.Where(x => x.IsClass && !x.IsGenericType && x.HasInterface<IDFLayoutFormField>()).ToList();
  59. return includeObsolete ? _fieldTypes : _fieldTypes.Where(x => !x.HasInterface<IDFLayoutObsoleteField>()).ToList();
  60. }
  61. public static Type? GetFieldType(string typeName)
  62. {
  63. var type = GetFieldTypes(true).Where(x => x.FullName == typeName).FirstOrDefault();
  64. if(type != null) return type;
  65. if(Enum.TryParse(typeName, out DigitalFormVariableType variableType))
  66. {
  67. // For the old types
  68. switch (variableType)
  69. {
  70. case DigitalFormVariableType.String:
  71. return typeof(DFLayoutStringField);
  72. case DigitalFormVariableType.Boolean:
  73. return typeof(DFLayoutBooleanField);
  74. case DigitalFormVariableType.Integer:
  75. return typeof(DFLayoutIntegerField);
  76. case DigitalFormVariableType.Double:
  77. return typeof(DFLayoutDoubleField);
  78. case DigitalFormVariableType.Date:
  79. return typeof(DFLayoutDateField);
  80. case DigitalFormVariableType.Time:
  81. return typeof(DFLayoutTimeField);
  82. case DigitalFormVariableType.Choice:
  83. return typeof(DFLayoutOptionField);
  84. case DigitalFormVariableType.Lookup:
  85. return typeof(DFLayoutLookupField);
  86. case DigitalFormVariableType.SignaturePad:
  87. return typeof(DFLayoutSignaturePad);
  88. case DigitalFormVariableType.EmbeddedImage:
  89. return typeof(DFLayoutEmbeddedImage);
  90. case DigitalFormVariableType.MultiImage:
  91. return typeof(DFLayoutMultiImage);
  92. }
  93. }
  94. return null;
  95. }
  96. // Indexed by 'From'
  97. private static Dictionary<Type, List<Tuple<Type, Type>>>? _fieldConverters;
  98. private static Dictionary<Type, List<Tuple<Type, Type>>> FieldConverters
  99. {
  100. get
  101. {
  102. if(_fieldConverters is null)
  103. {
  104. var fieldTypes = GetFieldTypes().Select(x =>
  105. {
  106. var propsType = DigitalFormVariable.GetPropertiesType(x);
  107. if (propsType is null) return null;
  108. return new Tuple<Type, Type>(propsType, x);
  109. }).NotNull().ToDictionary(x => x.Item1, x => x.Item2);
  110. _fieldConverters = CoreUtils.Entities.Where(x => x.HasInterface<IDFLayoutFieldConverter>() && !x.IsGenericType && !x.IsAbstract)
  111. .Select(x =>
  112. {
  113. var converter = x.GetInterfaceDefinition(typeof(IDFLayoutFieldConverter<,,,>));
  114. if (converter is null) return null;
  115. var from = converter.GenericTypeArguments[0];
  116. var to = converter.GenericTypeArguments[2];
  117. if(!fieldTypes.TryGetValue(from, out var fromFieldType)
  118. || !fieldTypes.TryGetValue(to, out var toFieldType))
  119. {
  120. return null;
  121. }
  122. return new Tuple<Type, Type, Type>(fromFieldType, toFieldType, x);
  123. })
  124. .NotNull()
  125. .GroupByDictionary(x => x.Item1, x => new Tuple<Type, Type>(x.Item2, x.Item3));
  126. }
  127. return _fieldConverters;
  128. }
  129. }
  130. public static IEnumerable<(Type to, Type converter)> GetFieldConverters(Type from)
  131. {
  132. if (FieldConverters.TryGetValue(from, out var list))
  133. {
  134. return list.Select(x => (x.Item1, x.Item2));
  135. }
  136. else
  137. {
  138. return Enumerable.Empty<(Type, Type)>();
  139. }
  140. }
  141. private static Dictionary<string, Tuple<Type, Type>>? _formInstanceTypes;
  142. private static Dictionary<string, Tuple<Type, Type>> FormInstanceTypes
  143. {
  144. get
  145. {
  146. _formInstanceTypes ??= CoreUtils.Entities.Where(x => x.HasInterface<IDigitalFormInstance>())
  147. .Select(x =>
  148. {
  149. var inter = x.GetInterfaceDefinition(typeof(IDigitalFormInstance<>));
  150. if (inter != null)
  151. {
  152. var link = inter.GenericTypeArguments[0];
  153. var entityLinkDef = link.GetSuperclassDefinition(typeof(EntityLink<>));
  154. if (entityLinkDef != null)
  155. {
  156. var entityType = entityLinkDef.GenericTypeArguments[0];
  157. return new Tuple<string, Type, Type>(entityType.Name, x, entityType);
  158. }
  159. }
  160. return null;
  161. }).Where(x => x != null).ToDictionary(x => x!.Item1, x => new Tuple<Type, Type>(x!.Item2, x!.Item3));
  162. return _formInstanceTypes;
  163. }
  164. }
  165. public static IEnumerable<Type> GetFormInstanceTypes()
  166. {
  167. return FormInstanceTypes.Select(x => x.Value.Item1);
  168. }
  169. public static Type? GetFormInstanceType(string appliesTo)
  170. {
  171. if (FormInstanceTypes.TryGetValue(appliesTo, out var result))
  172. {
  173. return result.Item1;
  174. }
  175. return null;
  176. }
  177. private static Dictionary<Type, IEntityFormUtils> _formUtils = new Dictionary<Type, IEntityFormUtils>();
  178. public static void AddFormUtils<TForm, TEntity, TEntityLink>(
  179. DelegateEntityFormUtils<TForm, TEntity, TEntityLink>.CanEditEvent editFormFunc,
  180. DelegateEntityFormUtils<TForm, TEntity, TEntityLink>.NewEntityEvent? newEntityFunc = null,
  181. DelegateEntityFormUtils<TForm, TEntity, TEntityLink>.OnSaveEvent? beforeSaveFunc = null,
  182. DelegateEntityFormUtils<TForm, TEntity, TEntityLink>.RequiredColumnsEvent? requiredColumnsFunc = null)
  183. where TForm : BaseEntityForm<TEntity, TEntityLink, TForm>
  184. where TEntity : Entity, new()
  185. where TEntityLink : EntityLink<TEntity>, new()
  186. {
  187. _formUtils.Add(typeof(TForm), new DelegateEntityFormUtils<TForm, TEntity, TEntityLink>(editFormFunc, newEntityFunc, beforeSaveFunc, requiredColumnsFunc));
  188. }
  189. public static void AddFormUtils<TForm, TEntity, TEntityLink>(EntityFormUtils<TForm, TEntity, TEntityLink> formUtils)
  190. where TForm : BaseEntityForm<TEntity, TEntityLink, TForm>
  191. where TEntity : Entity, new()
  192. where TEntityLink : EntityLink<TEntity>, new()
  193. {
  194. _formUtils.Add(typeof(TForm), formUtils);
  195. }
  196. public static bool CanEditForm(Type TForm, IDigitalFormInstance Form, Entity Entity)
  197. {
  198. if(_formUtils.TryGetValue(TForm, out var utils))
  199. {
  200. return utils.CanEditForm(Form, Entity);
  201. }
  202. return false;
  203. }
  204. public static bool CanEditForm<TForm, TEntity, TEntityLink>(TForm Form, TEntity Entity)
  205. where TForm : BaseEntityForm<TEntity, TEntityLink, TForm>
  206. where TEntity : Entity, new()
  207. where TEntityLink : EntityLink<TEntity>, new()
  208. {
  209. return CanEditForm(typeof(TForm), Form, Entity);
  210. }
  211. public static Entity NewEntity(Type TForm, Type TEntity, DigitalForm form)
  212. {
  213. if (_formUtils.TryGetValue(TForm, out var utils))
  214. {
  215. return utils.NewEntity(form);
  216. }
  217. return (Activator.CreateInstance(TEntity) as Entity)!;
  218. }
  219. public static TEntity NewEntity<TForm, TEntity, TEntityLink>(DigitalForm form)
  220. where TForm : BaseEntityForm<TEntity, TEntityLink, TForm>
  221. where TEntity : Entity, new()
  222. where TEntityLink : EntityLink<TEntity>, new()
  223. {
  224. return (TEntity)NewEntity(typeof(TForm), typeof(TEntity), form);
  225. }
  226. public static void OnSave(Type TForm, IDigitalFormInstance form, Entity entity)
  227. {
  228. if (_formUtils.TryGetValue(TForm, out var utils))
  229. {
  230. utils.OnSave(form, entity);
  231. }
  232. }
  233. public static void OnSave<TForm, TEntity, TEntityLink>(TForm form, TEntity entity)
  234. where TForm : BaseEntityForm<TEntity, TEntityLink, TForm>
  235. where TEntity : Entity, new()
  236. where TEntityLink : EntityLink<TEntity>, new()
  237. {
  238. OnSave(typeof(TForm), form, entity);
  239. }
  240. public static IColumns? RequiredEntityColumns(Type TForm)
  241. {
  242. if(_formUtils.TryGetValue(TForm, out var utils))
  243. {
  244. return utils.RequiredEntityColumns();
  245. }
  246. return null;
  247. }
  248. public static Columns<TEntity>? RequiredEntityColumns<TForm, TEntity, TEntityLink>()
  249. where TForm : BaseEntityForm<TEntity, TEntityLink, TForm>
  250. where TEntity : Entity, new()
  251. where TEntityLink : EntityLink<TEntity>, new()
  252. {
  253. return RequiredEntityColumns(typeof(TForm)) as Columns<TEntity>;
  254. }
  255. public static Type? FormEntityType(DigitalForm form)
  256. {
  257. return FormEntityType(form.AppliesTo);
  258. }
  259. public static Type? FormEntityType(string appliesTo)
  260. {
  261. return CoreUtils.Entities.FirstOrDefault(x => string.Equals(x.Name, appliesTo));
  262. }
  263. public static Type FormEntityLinkType(Type TForm)
  264. {
  265. var formInstanceType = TForm.GetInterfaceDefinition(typeof(IDigitalFormInstance<>))
  266. ?? throw new Exception("Form does not implement IDigitalFormInstance<>");
  267. return formInstanceType.GenericTypeArguments[0];
  268. }
  269. public static Type FormEntityType(Type TForm)
  270. {
  271. var linkType = FormEntityLinkType(TForm);
  272. var linkInterface = linkType.GetInterfaceDefinition(typeof(IEntityLink<>));
  273. return linkInterface!.GenericTypeArguments[0];
  274. }
  275. /*public static IEntityLink GetParentLink(IDigitalFormInstance formInstance)
  276. {
  277. var parentlink = CoreUtils.HasProperty(formInstance.GetType(), "Parent")
  278. ? CoreUtils.GetPropertyValue(formInstance, "Parent") as IEntityLink
  279. : null;
  280. }*/
  281. public static IColumns RequiredEntityColumns(Type TForm, Type TEntity, IEnumerable<DigitalFormVariable> variables)
  282. {
  283. var entityColumns = LookupFactory.RequiredColumns(TEntity);
  284. foreach(var variable in variables)
  285. {
  286. var property = variable.GetProperties().Property;
  287. if (!property.IsNullOrWhiteSpace())
  288. {
  289. entityColumns.Add(property);
  290. }
  291. }
  292. var others = RequiredEntityColumns(TForm);
  293. if(others != null)
  294. {
  295. foreach(var column in others)
  296. {
  297. entityColumns.Add(column);
  298. }
  299. }
  300. return entityColumns;
  301. }
  302. public static Columns<TEntity> EntityColumns<TForm, TEntity, TEntityLink>(IEnumerable<DigitalFormVariable> variables)
  303. where TForm : IDigitalFormInstance<TEntityLink>
  304. where TEntity : Entity, new()
  305. where TEntityLink : EntityLink<TEntity>
  306. => (RequiredEntityColumns(typeof(TForm), typeof(TEntity), variables) as Columns<TEntity>)!;
  307. }
  308. }