Update_7_34.cs 15 KB

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