Update_7_34.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385
  1. using InABox.Clients;
  2. using InABox.Core;
  3. using InABox.Database;
  4. using InABox.DynamicGrid;
  5. using InABox.WPF;
  6. using Microsoft.CodeAnalysis.CSharp.Syntax;
  7. using Newtonsoft.Json.Linq;
  8. using NPOI.POIFS.FileSystem;
  9. using PRSClasses;
  10. using Syncfusion.Data.Extensions;
  11. using Syncfusion.DocIO.DLS;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.Drawing;
  15. using System.Globalization;
  16. using System.Linq;
  17. using System.Text;
  18. using System.Threading.Tasks;
  19. using System.Windows.Interop;
  20. using System.Windows.Media.Imaging;
  21. using System.Windows.Navigation;
  22. namespace PRS.Shared
  23. {
  24. public class Update_7_34 : DatabaseUpdateScript
  25. {
  26. public override VersionNumber Version => new VersionNumber(7, 34);
  27. public override bool Update()
  28. {
  29. Logger.Send(LogType.Information, "", "Updating form data");
  30. var fnc = typeof(Update_7_34).GetMethod("UpdateForm", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic)!;
  31. FormUpdater.UpdateAllForms((_, vars) => vars.Count == 0, UpdateForm);
  32. Logger.Send(LogType.Information, "", "Updating form data complete");
  33. return true;
  34. }
  35. private Guid SaveDocument(byte[] data)
  36. {
  37. var id = Guid.NewGuid();
  38. var document = new Document
  39. {
  40. FileName = $"{id}.formdocument",
  41. Data = data
  42. };
  43. DbFactory.Provider.Save(document);
  44. return id;
  45. }
  46. private DFLayoutEmbeddedMediaValue ConvertImage(byte[] data)
  47. {
  48. var thumbnail = ImageUtils.BitmapImageFromBytes(data)?.Resize(200, 200).ToArray<BmpBitmapEncoder>();
  49. return new DFLayoutEmbeddedMediaValue
  50. {
  51. ID = SaveDocument(data),
  52. Thumbnail = thumbnail,
  53. Data = data
  54. };
  55. }
  56. private DFLayoutEmbeddedMediaValue ConvertVideo(byte[] data)
  57. {
  58. var thumbnail = ImageUtils.BitmapImageFromBytes(data)?.Resize(200, 200).ToArray<BmpBitmapEncoder>();
  59. return new DFLayoutEmbeddedMediaValue
  60. {
  61. ID = SaveDocument(data),
  62. Thumbnail = new Bitmap(256, 256).WatermarkImage("Video Data", Color.Gray).AsBitmapImage().ToArray<BmpBitmapEncoder>(),
  63. Data = data
  64. };
  65. }
  66. private DFLayoutLookupValue ConvertLookup(DigitalFormVariable variable, Guid lookupID)
  67. {
  68. if(lookupID == Guid.Empty)
  69. {
  70. return new DFLayoutLookupValue();
  71. }
  72. var properties = (variable.GetProperties() as DFLayoutLookupFieldProperties)!;
  73. if (!CoreUtils.TryGetEntity(properties.LookupType, out var type))
  74. {
  75. Logger.Send(LogType.Error, "", $"Invalid lookup type {properties.LookupType}");
  76. return new DFLayoutLookupValue { ID = lookupID };
  77. }
  78. var client = ClientFactory.CreateClient(type);
  79. var columns = LookupFactory.DefineColumns(type);
  80. foreach (var property in properties.AdditionalPropertiesList)
  81. {
  82. columns.Add(property);
  83. }
  84. var filter = Filter.Create(type, "ID").IsEqualTo(lookupID);
  85. var row = client.Query(
  86. filter,
  87. columns,
  88. LookupFactory.DefineSort(type)
  89. ).Rows.FirstOrDefault();
  90. if (row is null)
  91. {
  92. return new DFLayoutLookupValue { ID = lookupID };
  93. }
  94. var value = new DFLayoutLookupValue
  95. {
  96. ID = lookupID,
  97. Values = row.ToDictionary(new[] { "ID" })
  98. };
  99. value.Text = LookupFactory.FormatLookup(type, value.Values, Enumerable.Empty<string>());
  100. return value;
  101. }
  102. private object? Deserialize(DigitalFormVariable variable, Type fieldType, DFLoadStorage values)
  103. {
  104. var value = values.GetValue(variable.Code);
  105. if (fieldType == typeof(DFLayoutBooleanField))
  106. {
  107. if (value is bool b)
  108. {
  109. return b;
  110. }
  111. else if (bool.TryParse(value as string, out var result))
  112. {
  113. return result;
  114. }
  115. else if (value is string str)
  116. {
  117. var properties = (variable.GetProperties() as DFLayoutBooleanFieldProperties)!;
  118. if (str == properties.TrueValue)
  119. return true;
  120. if (str == properties.FalseValue)
  121. return false;
  122. throw new Exception($"Invalid boolean value {str}");
  123. }
  124. return variable.Deserialize(values.GetEntry(variable.Code));
  125. }
  126. else if(fieldType == typeof(DFLayoutDateField))
  127. {
  128. if (value is DateTime date)
  129. return date;
  130. if (DateTime.TryParseExact(value as string, "dd-MM-yyyy", CultureInfo.InvariantCulture, DateTimeStyles.None, out var result))
  131. return result;
  132. if (DateTime.TryParse(value as string, out result))
  133. return result;
  134. return variable.Deserialize(values.GetEntry(variable.Code));
  135. }
  136. else if (fieldType == typeof(DFLayoutDoubleField))
  137. {
  138. if (value is double d)
  139. return d;
  140. if (double.TryParse(value as string, out var result))
  141. return result;
  142. if(value is not null)
  143. {
  144. throw new Exception($"Invalid double '{value}'");
  145. }
  146. return variable.Deserialize(values.GetEntry(variable.Code));
  147. }
  148. else if (fieldType == typeof(DFLayoutEmbeddedImage))
  149. {
  150. if (value is byte[] b)
  151. return ConvertImage(b);
  152. else if(value is string str)
  153. {
  154. if (Guid.TryParse(str, out var id))
  155. return variable.Deserialize(values.GetEntry(variable.Code));
  156. try
  157. {
  158. var tuple = Serialization.Deserialize<Tuple<Guid, byte[]>>(str, true);
  159. if(tuple != null)
  160. {
  161. return new DFLayoutEmbeddedMediaValue
  162. {
  163. ID = tuple.Item1,
  164. Thumbnail = tuple.Item2,
  165. };
  166. }
  167. }
  168. catch
  169. {
  170. }
  171. try
  172. {
  173. return ConvertImage(System.Convert.FromBase64String(str));
  174. }
  175. catch(Exception e)
  176. {
  177. throw new Exception($"Error in image data; invalid Base-64: {e.Message}");
  178. }
  179. }
  180. return variable.Deserialize(values.GetEntry(variable.Code));
  181. }
  182. else if (fieldType == typeof(DFLayoutIntegerField))
  183. {
  184. if (value is null)
  185. return variable.Deserialize(values.GetEntry(variable.Code));
  186. if (value.GetType().IsNumeric())
  187. {
  188. return System.Convert.ToInt32(value);
  189. }
  190. if (int.TryParse(value as string, out var result))
  191. return result;
  192. if(value != null)
  193. {
  194. throw new Exception($"Invalid integer '{value}'");
  195. }
  196. return variable.Deserialize(values.GetEntry(variable.Code));
  197. }
  198. else if(fieldType == typeof(DFLayoutLookupField))
  199. {
  200. if(value is string str && Guid.TryParse(str, out var id))
  201. {
  202. return ConvertLookup(variable, id);
  203. }
  204. return variable.Deserialize(values.GetEntry(variable.Code));
  205. }
  206. else if(fieldType == typeof(DFLayoutMultiImage))
  207. {
  208. string[]? imgs;
  209. try
  210. {
  211. imgs = values.GetValue<string[]?>(variable.Code);
  212. }
  213. catch
  214. {
  215. var data = values.GetValue<byte[]>(variable.Code);
  216. if(data != null)
  217. {
  218. var newValueList = new DFLayoutEmbeddedMediaValues();
  219. newValueList.Add(ConvertImage(data));
  220. return newValueList;
  221. }
  222. return variable.Deserialize(values.GetEntry(variable.Code));
  223. }
  224. if (imgs is null)
  225. {
  226. return variable.Deserialize(values.GetEntry(variable.Code));
  227. }
  228. var valueList = new DFLayoutEmbeddedMediaValues();
  229. foreach (string s in imgs)
  230. {
  231. if (!s.IsNullOrWhiteSpace())
  232. {
  233. try
  234. {
  235. var externaldata = Serialization.Deserialize<(Guid, byte[])>(s, strict: true);
  236. if (externaldata.Item1 != Guid.Empty)
  237. {
  238. valueList.Add(new DFLayoutEmbeddedMediaValue()
  239. {
  240. ID = externaldata.Item1,
  241. Thumbnail = externaldata.Item2
  242. });
  243. }
  244. }
  245. catch
  246. {
  247. try
  248. {
  249. var data = System.Convert.FromBase64String(s);
  250. valueList.Add(ConvertImage(data));
  251. }
  252. catch
  253. {
  254. throw new Exception($"Could not convert multi-image");
  255. }
  256. }
  257. }
  258. }
  259. return valueList;
  260. }
  261. else if(fieldType == typeof(DFLayoutMultiSignaturePad))
  262. {
  263. return variable.Deserialize(values.GetEntry(variable.Code));
  264. }
  265. else if(fieldType == typeof(DFLayoutOptionField))
  266. {
  267. return variable.Deserialize(values.GetEntry(variable.Code));
  268. }
  269. else if (fieldType == typeof(DFLayoutSignaturePad))
  270. {
  271. return variable.Deserialize(values.GetEntry(variable.Code));
  272. }
  273. else if (fieldType == typeof(DFLayoutStringField))
  274. {
  275. return variable.Deserialize(values.GetEntry(variable.Code));
  276. }
  277. else if (fieldType == typeof(DFLayoutTimeField))
  278. {
  279. if (value is TimeSpan time)
  280. return time;
  281. if (TimeSpan.TryParseExact(value as string, "c", CultureInfo.InvariantCulture, TimeSpanStyles.None, out var result))
  282. return result;
  283. if (TimeSpan.TryParse(value as string, out result))
  284. return result;
  285. return variable.Deserialize(values.GetEntry(variable.Code));
  286. }
  287. else if (fieldType == typeof(DFLayoutVideoField))
  288. {
  289. if (value is byte[] b)
  290. return ConvertVideo(b);
  291. else if (value is string str)
  292. {
  293. if (Guid.TryParse(str, out var id))
  294. return variable.Deserialize(values.GetEntry(variable.Code));
  295. try
  296. {
  297. var tuple = Serialization.Deserialize<Tuple<Guid, byte[]>>(str, true);
  298. if (tuple != null)
  299. {
  300. return new DFLayoutEmbeddedMediaValue
  301. {
  302. ID = tuple.Item1,
  303. Thumbnail = tuple.Item2,
  304. };
  305. }
  306. }
  307. catch
  308. {
  309. }
  310. try
  311. {
  312. return ConvertVideo(System.Convert.FromBase64String(str));
  313. }
  314. catch (Exception e)
  315. {
  316. throw new Exception($"Error in video data; invalid Base-64: {e.Message}");
  317. }
  318. }
  319. return variable.Deserialize(values.GetEntry(variable.Code));
  320. }
  321. else if(fieldType == typeof(DFLayoutAddTaskField))
  322. {
  323. return value?.ToString();
  324. }
  325. else
  326. {
  327. throw new Exception($"Unhandled variable type {fieldType}");
  328. }
  329. }
  330. private bool UpdateForm(Type formType, IDigitalFormInstance instance, DigitalForm form, IList<DigitalFormVariable> variables)
  331. {
  332. var values = DigitalForm.DeserializeFormData(instance);
  333. if(values is null)
  334. {
  335. return false;
  336. }
  337. var save = new DFSaveStorage();
  338. foreach(var variable in variables)
  339. {
  340. try
  341. {
  342. var value = Deserialize(variable, variable.FieldType(), values);
  343. variable.Serialize(save.GetEntry(variable.Code), value);
  344. }
  345. catch(Exception e)
  346. {
  347. Logger.Send(LogType.Error, "", $"Error in data for {variable.Code} ({instance.ID}): {e.Message}");
  348. save.FormData[variable.Code] = values.GetValue(variable.Code);
  349. foreach(var (k, v) in values.GetEntry(variable.Code).SubItems())
  350. {
  351. save.FormData[$"{variable.Code}.{k}"] = v;
  352. }
  353. }
  354. }
  355. foreach(var (k, v) in values.Items())
  356. {
  357. if(!save.FormData.ContainsKey(k) && !k.Contains('.') && !variables.Any(x => x.Code == k))
  358. {
  359. save.FormData[k] = v;
  360. }
  361. }
  362. DigitalForm.SerializeFormData(instance, variables, save);
  363. return instance.IsChanged();
  364. }
  365. }
  366. }