Dimensions.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461
  1. using InABox.Core;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Collections.Immutable;
  5. using System.Linq.Expressions;
  6. namespace Comal.Classes
  7. {
  8. public abstract class Dimensions<TLink, TUnit> : EnclosedEntity, IDimensions
  9. where TLink : DimensionUnitLink<TUnit>, new()
  10. where TUnit : DimensionUnit, new()
  11. {
  12. [EditorSequence(1)]
  13. [RequiredColumn]
  14. [Caption("Sizing", IncludePath = false)]
  15. public abstract TLink Unit { get; set; }
  16. public IDimensionUnit GetUnit() => Unit;
  17. [DoubleEditor(Visible = Visible.Hidden)]
  18. [EditorSequence(2)]
  19. [Caption("Quantity", IncludePath = false)]
  20. [RequiredColumn]
  21. public abstract double Quantity { get; set; }
  22. [DoubleEditor(Visible = Visible.Hidden)]
  23. [EditorSequence(3)]
  24. [Caption("Length", IncludePath = false)]
  25. [RequiredColumn]
  26. public abstract double Length { get; set; }
  27. [DoubleEditor(Visible = Visible.Hidden)]
  28. [EditorSequence(4)]
  29. [Caption("Width", IncludePath = false)]
  30. [RequiredColumn]
  31. public abstract double Width { get; set; }
  32. [DoubleEditor(Visible = Visible.Hidden)]
  33. [EditorSequence(5)]
  34. [Caption("Height", IncludePath = false)]
  35. [RequiredColumn]
  36. public abstract double Height { get; set; }
  37. [DoubleEditor(Visible = Visible.Hidden)]
  38. [EditorSequence(6)]
  39. [Caption("Weight", IncludePath = false)]
  40. [RequiredColumn]
  41. public abstract double Weight { get; set; }
  42. [DoubleEditor(Visible = Visible.Optional, Editable = Editable.Hidden)]
  43. [Caption("Value", IncludePath = false)]
  44. [EditorSequence(7)]
  45. [RequiredColumn]
  46. public abstract double Value { get; set; }
  47. [TextBoxEditor(Visible = Visible.Default, Editable = Editable.Hidden)]
  48. [EditorSequence(8)]
  49. [Caption("Unit Size", IncludePath = false)]
  50. [RequiredColumn]
  51. public abstract String UnitSize { get; set; }
  52. IDimensionUnit IDimensions.Unit => Unit;
  53. protected override void Init()
  54. {
  55. base.Init();
  56. Unit.PropertyChanged += (s, e) =>
  57. {
  58. if (e.PropertyName == "ID")
  59. {
  60. DoPropertyChanged("Unit." + e.PropertyName, OriginalValues.GetValueOrDefault("Unit." + e.PropertyName), Unit.ID);
  61. }
  62. else if (e.PropertyName == "Formula")
  63. {
  64. DoPropertyChanged("Unit." + e.PropertyName, OriginalValues.GetValueOrDefault("Unit." + e.PropertyName), Unit.Formula);
  65. }
  66. else if (e.PropertyName == "Format")
  67. {
  68. DoPropertyChanged("Unit." + e.PropertyName, OriginalValues.GetValueOrDefault("Unit." + e.PropertyName), Unit.Format);
  69. }
  70. };
  71. }
  72. private bool bCalculating = false;
  73. private static Column<Dimensions<TLink, TUnit>> unitid = new Column<Dimensions<TLink, TUnit>>(x => x.Unit.ID);
  74. private static Column<Dimensions<TLink, TUnit>> quantity = new Column<Dimensions<TLink, TUnit>>(x => x.Quantity);
  75. private static Column<Dimensions<TLink, TUnit>> length = new Column<Dimensions<TLink, TUnit>>(x => x.Length);
  76. private static Column<Dimensions<TLink, TUnit>> width = new Column<Dimensions<TLink, TUnit>>(x => x.Width);
  77. private static Column<Dimensions<TLink, TUnit>> height = new Column<Dimensions<TLink, TUnit>>(x => x.Height);
  78. private static Column<Dimensions<TLink, TUnit>> weight = new Column<Dimensions<TLink, TUnit>>(x => x.Weight);
  79. private static Column<Dimensions<TLink, TUnit>> sizeformula = new Column<Dimensions<TLink, TUnit>>(x => x.Unit.Formula);
  80. private static Column<Dimensions<TLink, TUnit>> sizeformat = new Column<Dimensions<TLink, TUnit>>(x => x.Unit.Format);
  81. protected override void DoPropertyChanged(string name, object? before, object? after)
  82. {
  83. base.DoPropertyChanged(name, before, after);
  84. if (bCalculating)
  85. return;
  86. bCalculating = true;
  87. try
  88. {
  89. if (unitid.IsEqualTo(name))
  90. Calculate(Quantity, Length, Width, Height, Weight, Unit.Formula, Unit.Format);
  91. else if (quantity.IsEqualTo(name))
  92. Calculate(after, Length, Width, Height, Weight, Unit.Formula, Unit.Format);
  93. else if (length.IsEqualTo(name))
  94. Calculate(Quantity, after, Width, Height, Weight, Unit.Formula, Unit.Format);
  95. else if (width.IsEqualTo(name))
  96. Calculate(Quantity, Length, after, Height, Weight, Unit.Formula, Unit.Format);
  97. else if (height.IsEqualTo(name))
  98. Calculate(Quantity, Length, Width, after, Weight, Unit.Formula, Unit.Format);
  99. else if (weight.IsEqualTo(name))
  100. Calculate(Quantity, Length, Width, Height, after, Unit.Formula, Unit.Format);
  101. else if (sizeformula.IsEqualTo(name))
  102. Calculate(Quantity, Length, Width, Height, Weight, after as string, Unit.Format);
  103. else if (sizeformat.IsEqualTo(name))
  104. Calculate(Quantity, Length, Width, Height, Weight, Unit.Formula, after as string);
  105. }
  106. finally
  107. {
  108. bCalculating = false;
  109. }
  110. }
  111. public void Set(IDimensionUnit unit, double quantity, double length, double width, double height, double weight)
  112. {
  113. bCalculating = true;
  114. try
  115. {
  116. Unit.ID = unit.ID;
  117. Unit.HasQuantity = unit.HasQuantity;
  118. Unit.HasLength = unit.HasLength;
  119. Unit.HasWidth = unit.HasWidth;
  120. Unit.HasHeight = unit.HasHeight;
  121. Unit.HasWeight = unit.HasWeight;
  122. Unit.Code = unit.Code;
  123. Unit.Description = unit.Description;
  124. Unit.Formula = unit.Formula;
  125. Unit.Format = unit.Format;
  126. Quantity = quantity;
  127. Length = length;
  128. Width = width;
  129. Height = height;
  130. Weight = weight;
  131. Calculate(quantity, length, width, height, weight, unit.Formula, unit.Format);
  132. }
  133. finally
  134. {
  135. bCalculating = false;
  136. }
  137. }
  138. public void CalculateValueAndUnitSize()
  139. {
  140. Calculate(Quantity, Length, Width, Height, Weight, Unit.Formula, Unit.Format);
  141. }
  142. private void Calculate(object? quantity, object? length, object? width, object? height, object? weight, string? formula, string? format)
  143. {
  144. if (Evaluate<double>(formula, quantity, length, width, height, weight, out double value))
  145. Value = value;
  146. if (Evaluate<String>(format, quantity, length, width, height, weight, out string unitsize))
  147. UnitSize = unitsize;
  148. }
  149. private bool Evaluate<T>(string? formula, object? quantity, object? length, object? width, object? height, object? weight, out T result)
  150. {
  151. if (!String.IsNullOrWhiteSpace(formula))
  152. {
  153. var variables = new Dictionary<string, object?>()
  154. {
  155. { "Quantity", Convert.ToDouble(quantity) },
  156. { "Length", Convert.ToDouble(length) },
  157. { "Width", Convert.ToDouble(width) },
  158. { "Height", Convert.ToDouble(height) },
  159. { "Weight", Convert.ToDouble(weight) }
  160. };
  161. try
  162. {
  163. var expr = new CoreExpression(formula);
  164. var eval = expr.Evaluate(variables);
  165. result = (T)CoreUtils.ChangeType(eval, typeof(T));
  166. return true;
  167. }
  168. catch (Exception e)
  169. {
  170. Logger.Send(LogType.Information, "", String.Format("Error in Formula: [{0}] ({1})", formula, e.Message));
  171. result = default(T);
  172. return false;
  173. }
  174. }
  175. result = default(T);
  176. return true;
  177. }
  178. public void CopyFrom(IDimensions source, bool observing = false)
  179. {
  180. if (!observing)
  181. SetObserving(false);
  182. Unit.ID = source.GetUnit().ID;
  183. Unit.Synchronise(source.GetUnit());
  184. Quantity = source.Quantity;
  185. Length = source.Length;
  186. Width = source.Width;
  187. Height = source.Height;
  188. Weight = source.Weight;
  189. Value = source.Value;
  190. UnitSize = source.UnitSize;
  191. if (!observing)
  192. SetObserving(true);
  193. }
  194. public override string ToString()
  195. {
  196. var result = Value != 0 ? Math.Round(Value, 3).ToString() : "";
  197. result = !string.IsNullOrWhiteSpace(UnitSize) ? result + " " + UnitSize + ", " : "Empty unitsize";
  198. result = Quantity != 0 ? result + "Quantity: " + Quantity + ", " : result;
  199. result = Length != 0 ? result + "Length: " + Length + ", " : result;
  200. result = Width != 0 ? result + "Width: " + Width + ", " : result;
  201. result = Height != 0 ? result + "Height: " + Height + ", " : result;
  202. result = Weight != 0 ? result + "Weight: " + Weight : result;
  203. if (result.EndsWith(", "))
  204. result = result.Remove(result.Length - 2);
  205. return result;
  206. }
  207. public override bool Equals(object obj)
  208. {
  209. return obj is IDimensions dim
  210. && Unit.ID == dim.Unit.ID
  211. && Quantity.IsEffectivelyEqual(dim.Quantity)
  212. && Length.IsEffectivelyEqual(dim.Length)
  213. && Width.IsEffectivelyEqual(dim.Width)
  214. && Height.IsEffectivelyEqual(dim.Height)
  215. && Weight.IsEffectivelyEqual(dim.Weight);
  216. }
  217. public override int GetHashCode()
  218. {
  219. return HashCode.Combine(
  220. Unit.ID,
  221. Quantity,
  222. Length,
  223. Width,
  224. Height,
  225. Weight);
  226. }
  227. }
  228. public static class Dimensions
  229. {
  230. private static readonly Column<IDimensions> unitid = new Column<IDimensions>(x => x.Unit.ID);
  231. private static readonly Column<IDimensions> quantity = new Column<IDimensions>(x => x.Quantity);
  232. private static readonly Column<IDimensions> length = new Column<IDimensions>(x => x.Length);
  233. private static readonly Column<IDimensions> width = new Column<IDimensions>(x => x.Width);
  234. private static readonly Column<IDimensions> height = new Column<IDimensions>(x => x.Height);
  235. private static readonly Column<IDimensions> weight = new Column<IDimensions>(x => x.Weight);
  236. public static IEnumerable<Column<IDimensions>> GetFilterColumns()
  237. {
  238. yield return unitid;
  239. yield return quantity;
  240. yield return length;
  241. yield return width;
  242. yield return height;
  243. yield return weight;
  244. }
  245. public static Filter<T> DimensionEquals<T>(this Filter<T> filter, IDimensions dim)
  246. {
  247. if (!CoreUtils.TryFindMemberExpression(filter.Expression, out var mexp))
  248. {
  249. throw new ArgumentException("Filter expression is not a MemberExpression");
  250. }
  251. var prop = CoreUtils.GetFullPropertyName(mexp, ".") + ".";
  252. filter.Expression = CoreUtils.GetMemberExpression(typeof(T), prop + unitid.Property);
  253. filter.IsEqualTo(dim.Unit.ID);
  254. filter.And(prop + quantity.Property).IsEqualTo(dim.Quantity);
  255. filter.And(prop + length.Property).IsEqualTo(dim.Length);
  256. filter.And(prop + width.Property).IsEqualTo(dim.Width);
  257. filter.And(prop + height.Property).IsEqualTo(dim.Height);
  258. filter.And(prop + weight.Property).IsEqualTo(dim.Weight);
  259. return filter.Parent ?? filter;
  260. }
  261. public static IEnumerable<KeyValuePair<Expression<Func<T, object?>>, Expression<Func<U, object?>>>> GetLinks<T, U>()
  262. where T : IDimensioned
  263. where U : IDimensioned
  264. {
  265. return GetLinks<T, U>(x => x.Dimensions, x => x.Dimensions);
  266. }
  267. public static IEnumerable<KeyValuePair<Expression<Func<T, object?>>, Expression<Func<U, object?>>>> GetLinks<T, U>(
  268. Expression<Func<T, IDimensions>> tDimensions,
  269. Expression<Func<U, IDimensions>> uDimensions
  270. )
  271. {
  272. var tPrefix = CoreUtils.GetFullPropertyName(tDimensions, ".") + ".";
  273. var uPrefix = CoreUtils.GetFullPropertyName(uDimensions, ".") + ".";
  274. yield return new KeyValuePair<Expression<Func<T, object?>>, Expression<Func<U, object?>>>(
  275. CoreUtils.GetPropertyExpression<T>(tPrefix + unitid.Property), CoreUtils.GetPropertyExpression<U>(uPrefix + unitid.Property));
  276. yield return new KeyValuePair<Expression<Func<T, object?>>, Expression<Func<U, object?>>>(
  277. CoreUtils.GetPropertyExpression<T>(tPrefix + quantity.Property), CoreUtils.GetPropertyExpression<U>(uPrefix + quantity.Property));
  278. yield return new KeyValuePair<Expression<Func<T, object?>>, Expression<Func<U, object?>>>(
  279. CoreUtils.GetPropertyExpression<T>(tPrefix + length.Property), CoreUtils.GetPropertyExpression<U>(uPrefix + length.Property));
  280. yield return new KeyValuePair<Expression<Func<T, object?>>, Expression<Func<U, object?>>>(
  281. CoreUtils.GetPropertyExpression<T>(tPrefix + width.Property), CoreUtils.GetPropertyExpression<U>(uPrefix + width.Property));
  282. yield return new KeyValuePair<Expression<Func<T, object?>>, Expression<Func<U, object?>>>(
  283. CoreUtils.GetPropertyExpression<T>(tPrefix + height.Property), CoreUtils.GetPropertyExpression<U>(uPrefix + height.Property));
  284. yield return new KeyValuePair<Expression<Func<T, object?>>, Expression<Func<U, object?>>>(
  285. CoreUtils.GetPropertyExpression<T>(tPrefix + weight.Property), CoreUtils.GetPropertyExpression<U>(uPrefix + weight.Property));
  286. }
  287. public enum ColumnsType
  288. {
  289. Local,
  290. Data,
  291. All
  292. }
  293. public static int[] GetFilterColumnIndices<T>(Columns<T> columns, Expression<Func<T, IDimensions>> dimensions)
  294. {
  295. var dimCol = CoreUtils.GetFullPropertyName(dimensions, ".") + ".";
  296. return new int[]
  297. {
  298. columns.IndexOf(dimCol + unitid.Property),
  299. columns.IndexOf(dimCol + quantity.Property),
  300. columns.IndexOf(dimCol + length.Property),
  301. columns.IndexOf(dimCol + width.Property),
  302. columns.IndexOf(dimCol + height.Property),
  303. columns.IndexOf(dimCol + weight.Property)
  304. };
  305. }
  306. public static int[] GetFilterColumnIndices<T>(CoreTable table, Expression<Func<T, IDimensions>> dimensions)
  307. {
  308. var dimCol = CoreUtils.GetFullPropertyName(dimensions, ".") + ".";
  309. return new int[]
  310. {
  311. table.GetColumnIndex(dimCol + unitid.Property),
  312. table.GetColumnIndex(dimCol + quantity.Property),
  313. table.GetColumnIndex(dimCol + length.Property),
  314. table.GetColumnIndex(dimCol + width.Property),
  315. table.GetColumnIndex(dimCol + height.Property),
  316. table.GetColumnIndex(dimCol + weight.Property)
  317. };
  318. }
  319. public static TDim ToDimensions<TDim>(this CoreRow row, int[] columnIndices)
  320. where TDim : IDimensions, new()
  321. {
  322. var dimensions = new TDim();
  323. dimensions.Unit.ID = row.Get<Guid>(columnIndices[0]);
  324. dimensions.Quantity = row.Get<double>(columnIndices[1]);
  325. dimensions.Length = row.Get<double>(columnIndices[2]);
  326. dimensions.Width = row.Get<double>(columnIndices[3]);
  327. dimensions.Height = row.Get<double>(columnIndices[4]);
  328. dimensions.Weight = row.Get<double>(columnIndices[5]);
  329. return dimensions;
  330. }
  331. public static TDim ToDimensions<T, TDim>(this CoreRow row, Expression<Func<T, TDim>> dim)
  332. where TDim : IDimensions, new()
  333. {
  334. var dimensions = new TDim();
  335. var dimCol = CoreUtils.GetFullPropertyName(dim, ".") + ".";
  336. dimensions.Unit.ID = row.Get<Guid>(dimCol + unitid.Property);
  337. dimensions.Length = row.Get<double>(dimCol + length.Property);
  338. dimensions.Quantity = row.Get<double>(dimCol + quantity.Property);
  339. dimensions.Width = row.Get<double>(dimCol + width.Property);
  340. dimensions.Height = row.Get<double>(dimCol + height.Property);
  341. dimensions.Weight = row.Get<double>(dimCol + weight.Property);
  342. return dimensions;
  343. }
  344. public static Columns<T> LocalColumns<T>()
  345. where T : IDimensions
  346. {
  347. return Columns.None<T>().Add(x => x.Unit.ID)
  348. .Add(x => x.Quantity)
  349. .Add(x => x.Length)
  350. .Add(x => x.Width)
  351. .Add(x => x.Height)
  352. .Add(x => x.Weight)
  353. .Add(x => x.UnitSize)
  354. .Add(x => x.Value);
  355. }
  356. public static Columns<T> DataColumns<T>()
  357. where T : IDimensions
  358. {
  359. return Columns.None<T>().Add(x => x.Unit.ID)
  360. .Add(x => x.Unit.ID)
  361. .Add(x => x.Unit.Format)
  362. .Add(x => x.Unit.Formula)
  363. .Add(x => x.Unit.HasHeight)
  364. .Add(x => x.Unit.HasWeight)
  365. .Add(x => x.Unit.HasWidth)
  366. .Add(x => x.Unit.HasQuantity)
  367. .Add(x => x.Unit.HasLength)
  368. .Add(x => x.Quantity)
  369. .Add(x => x.Length)
  370. .Add(x => x.Width)
  371. .Add(x => x.Height)
  372. .Add(x => x.Weight)
  373. .Add(x => x.UnitSize)
  374. .Add(x => x.Value);
  375. }
  376. public static Columns<T> AllColumns<T>()
  377. where T : IDimensions
  378. {
  379. return Columns.None<T>().Add(x => x.Unit.ID)
  380. .Add(x => x.Unit.ID)
  381. .Add(x => x.Unit.Format)
  382. .Add(x => x.Unit.Formula)
  383. .Add(x => x.Unit.HasHeight)
  384. .Add(x => x.Unit.HasWeight)
  385. .Add(x => x.Unit.HasWidth)
  386. .Add(x => x.Unit.HasQuantity)
  387. .Add(x => x.Unit.HasLength)
  388. .Add(x => x.Unit.Code)
  389. .Add(x => x.Unit.Description)
  390. .Add(x => x.Quantity)
  391. .Add(x => x.Length)
  392. .Add(x => x.Width)
  393. .Add(x => x.Height)
  394. .Add(x => x.Weight)
  395. .Add(x => x.UnitSize)
  396. .Add(x => x.Value);
  397. }
  398. public static Columns<T> AddDimensionsColumns<T, TDim>(this Columns<T> columns, Expression<Func<T, TDim>> dim, ColumnsType type = ColumnsType.Local)
  399. where TDim : IDimensions
  400. {
  401. return columns.AddSubColumns(dim, type switch
  402. {
  403. ColumnsType.Data => DataColumns<TDim>(),
  404. ColumnsType.All => AllColumns<TDim>(),
  405. ColumnsType.Local => LocalColumns<TDim>(),
  406. _ => LocalColumns<TDim>()
  407. });
  408. }
  409. }
  410. }