TaskPanel.xaml.cs 52 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using System.Threading.Tasks;
  6. using System.Windows;
  7. using System.Windows.Controls;
  8. using Comal.Classes;
  9. using InABox.Clients;
  10. using InABox.Configuration;
  11. using InABox.Core;
  12. using InABox.DynamicGrid;
  13. using InABox.WPF;
  14. using Syncfusion.Pdf.Graphics;
  15. using Syncfusion.Pdf;
  16. using System.Drawing;
  17. using System.ComponentModel;
  18. using InABox.Wpf;
  19. namespace PRSDesktop
  20. {
  21. public class TaskPanelProperties : BaseObject, IGlobalConfigurationSettings
  22. {
  23. [CheckBoxEditor(ToolTip = "Require that all tasks are given a task type.")]
  24. public bool RequireTaskTypes { get; set; } = false;
  25. }
  26. public class TaskPanelFilterButton : FilterButton<Kanban>
  27. {
  28. public TaskPanelFilterButton() : base(
  29. new GlobalConfiguration<CoreFilterDefinitions>(nameof(Kanban)),
  30. new UserConfiguration<CoreFilterDefinitions>(nameof(Kanban)))
  31. {
  32. }
  33. public static Filter<Kanban> ConvertFilterToKanbanFilter(Filter<Kanban>? kanbanFilter)
  34. {
  35. if (kanbanFilter is null)
  36. {
  37. return new Filter<Kanban>().All();
  38. }
  39. else if (CoreUtils.TryFindMemberExpression(kanbanFilter.Expression, out var mexp))
  40. {
  41. var prop = CoreUtils.GetFullPropertyName(mexp, ".");
  42. var filter = new Filter<Kanban>(prop)
  43. {
  44. Operator = kanbanFilter.Operator,
  45. Value = kanbanFilter.Value
  46. };
  47. filter.Ands.AddRange(kanbanFilter.Ands.Select(ConvertFilterToKanbanFilter));
  48. filter.Ors.AddRange(kanbanFilter.Ors.Select(ConvertFilterToKanbanFilter));
  49. return filter;
  50. }
  51. else
  52. {
  53. return new Filter<Kanban>().None();
  54. }
  55. }
  56. public static Filter<KanbanSubscriber> ConvertFilterToSubscriberFilter(Filter<IKanban>? kanbanFilter)
  57. {
  58. if (kanbanFilter is null)
  59. {
  60. return new Filter<KanbanSubscriber>().All();
  61. }
  62. else if (CoreUtils.TryFindMemberExpression(kanbanFilter.Expression, out var mexp))
  63. {
  64. var prop = CoreUtils.GetFullPropertyName(mexp, ".");
  65. var filter = new Filter<KanbanSubscriber>(nameof(KanbanSubscriber.Kanban) + "." + prop)
  66. {
  67. Operator = kanbanFilter.Operator,
  68. Value = kanbanFilter.Value
  69. };
  70. filter.Ands.AddRange(kanbanFilter.Ands.Select(ConvertFilterToSubscriberFilter));
  71. filter.Ors.AddRange(kanbanFilter.Ors.Select(ConvertFilterToSubscriberFilter));
  72. return filter;
  73. }
  74. else
  75. {
  76. return new Filter<KanbanSubscriber>().None();
  77. }
  78. }
  79. }
  80. /// <summary>
  81. /// Interaction logic for TaskPanel.xaml
  82. /// </summary>
  83. public partial class TaskPanel : UserControl, IPanel<Kanban>, ITaskHost, IMasterDetailControl<Job>, IPropertiesPanel<TaskPanelProperties, CanConfigureTasksPanel>
  84. {
  85. private bool _bTabChanging;
  86. private KanbanType[] kanbanTypes = null!; // Initialized in Setup()
  87. public IList<KanbanType> KanbanTypes => kanbanTypes;
  88. public Job? Master { get; set; }
  89. public TaskPanel()
  90. {
  91. InitializeComponent();
  92. foreach (TabItem tab in TaskPanels.Items)
  93. {
  94. var panel = (tab.Content as ITaskControl)!;
  95. _viewmap[panel.KanbanViewType] = tab;
  96. panel.Host = this;
  97. }
  98. }
  99. #region Menu
  100. private void CompleteTask(ITaskControl control, RoutedEventArgs e, DateTime completed)
  101. {
  102. if (MessageBox.Show($"Are you sure you want to complete the selected tasks?", "Confirm Completion",
  103. MessageBoxButton.YesNo) != MessageBoxResult.Yes)
  104. return;
  105. var tasks = (((FrameworkElement)e.Source).Tag as IEnumerable<TaskModel>)!;
  106. Progress.ShowModal("Completing Tasks", progress =>
  107. {
  108. var kanbans = LoadKanbans(tasks, new Columns<Kanban>(x => x.ID, x => x.Completed, x => x.Status));
  109. foreach (var kanban in kanbans)
  110. {
  111. kanban.Completed = completed;
  112. kanban.Status = KanbanStatus.Complete;
  113. }
  114. Client.Save(kanbans, $"Kanban Marked as Complete");
  115. });
  116. control.Refresh();
  117. }
  118. private void AddChangeStatusButton(ITaskControl control, TaskModel[] models, MenuItem menu, string header, KanbanStatus status)
  119. {
  120. menu.AddItem(header, null, Tuple.Create(control, models, status), ChangeStatus_Click);
  121. }
  122. private void ChangeStatus_Click(Tuple<ITaskControl, TaskModel[], KanbanStatus> obj)
  123. {
  124. var (control, tasks, status) = obj;
  125. if (MessageBox.Show($"Are you sure you want to mark the selected tasks as {status}?", "Confirm Change Status",
  126. MessageBoxButton.YesNo) != MessageBoxResult.Yes)
  127. return;
  128. Progress.ShowModal("Changing Status", progress =>
  129. {
  130. var kanbans = LoadKanbans(tasks, new Columns<Kanban>(x => x.ID, x => x.Completed, x => x.Status));
  131. foreach (var kanban in kanbans)
  132. {
  133. if (status == KanbanStatus.Complete)
  134. {
  135. kanban.Completed = DateTime.Now;
  136. }
  137. kanban.Status = status;
  138. }
  139. new Client<Kanban>().Save(kanbans, $"Kanban Marked as {status}");
  140. });
  141. control.Refresh();
  142. }
  143. public bool CanChangeTasks(IEnumerable<TaskModel> models)
  144. {
  145. foreach (var task in models)
  146. {
  147. if (!App.EmployeeID.Equals(task.ManagerID) && !App.EmployeeID.Equals(task.EmployeeID))
  148. {
  149. // If you can change others tasks, IsFullControl is true - but we don't check at the beginning of the function
  150. // to save checking security tokens every time.
  151. return Security.IsAllowed<CanChangeOthersTasks>();
  152. }
  153. }
  154. return true;
  155. }
  156. public void PopulateMenu(ITaskControl control, TaskModel task, ContextMenu menu)
  157. {
  158. menu.Items.Clear();
  159. var models = control.SelectedModels(task).ToArray();
  160. var references = GetReferences(models);
  161. var bLinks = references.Any(x => x.ReferenceType() != null);
  162. var referencetypes = references.Select(x => x.ReferenceType()).Distinct().ToArray();
  163. var bSingle = models.Length == 1;
  164. var canChange = CanChangeTasks(models);
  165. var edit = new MenuItem
  166. {
  167. Tag = models,
  168. Header = referencetypes.SingleOrDefault() == typeof(Requisition)
  169. ? "Edit Requisition Details"
  170. : referencetypes.SingleOrDefault() == typeof(Setout)
  171. ? "Edit Setout Details"
  172. : referencetypes.SingleOrDefault() == typeof(Delivery)
  173. ? "Edit Delivery Details"
  174. : referencetypes.SingleOrDefault() == typeof(PurchaseOrder)
  175. ? "Edit Order Details"
  176. : "Edit Task" + (bSingle ? "" : "s")
  177. };
  178. edit.Click += (o, e) =>
  179. {
  180. var tasks = (((MenuItem)e.Source).Tag as IEnumerable<TaskModel>)!;
  181. if (EditReferences(tasks))
  182. control.Refresh();
  183. e.Handled = true;
  184. };
  185. edit.IsEnabled = referencetypes.Length == 1;
  186. menu.Items.Add(edit);
  187. if (!bLinks && models.Length == 1)
  188. {
  189. var digitalForms = new MenuItem { Header = "Digital Forms" };
  190. var model = models.First();
  191. DynamicGridUtils.PopulateFormMenu<KanbanForm, Kanban, KanbanLink>(
  192. digitalForms,
  193. model.ID,
  194. () => new Client<Kanban>().Load(new Filter<Kanban>(x => x.ID).IsEqualTo(model.ID)).First(),
  195. model.EmployeeID == App.EmployeeID);
  196. menu.Items.Add(digitalForms);
  197. }
  198. if (!models.Any(x => !x.CompletedDate.IsEmpty()) && !bLinks)
  199. {
  200. menu.Items.Add(new Separator());
  201. var job = new MenuItem
  202. {
  203. Tag = models,
  204. Header = "Link to Job"
  205. };
  206. job.SubmenuOpened += (o, e) => CreateJobSubMenu(control, job, models);
  207. menu.Items.Add(job);
  208. if (bSingle)
  209. {
  210. menu.AddItem("Create Setout from Task", null, models.First(), task =>
  211. {
  212. if (MessageBox.Show("This will convert this task into a Setout.\n\nDo you wish to continue?", "Confirmation", MessageBoxButton.YesNo) != MessageBoxResult.Yes)
  213. return;
  214. ManufacturingTemplate? template = new Client<ManufacturingTemplate>()
  215. .Load(new Filter<ManufacturingTemplate>(x => x.Code).IsEqualTo("PRS")).FirstOrDefault();
  216. if (template == null)
  217. {
  218. MessageBox.Show("[Pressing] Template does not exist!");
  219. return;
  220. }
  221. string? setoutNumber = null;
  222. Kanban? kanban = null;
  223. ManufacturingTemplateStage[] tstages = Array.Empty<ManufacturingTemplateStage>();
  224. Progress.ShowModal("Creating Setout", (progress) =>
  225. {
  226. var kanbanFilter = new Filter<Kanban>(x => x.ID).IsEqualTo(task.ID);
  227. var tables = Client.QueryMultiple(new Dictionary<string, IQueryDef>
  228. {
  229. { "ManufacturingTemplateStage", new QueryDef<ManufacturingTemplateStage>(
  230. new Filter<ManufacturingTemplateStage>(x => x.Template.ID).IsEqualTo(template.ID),
  231. null,
  232. new SortOrder<ManufacturingTemplateStage>(x => x.Sequence)) },
  233. { "Kanban", new QueryDef<Kanban>(
  234. kanbanFilter,
  235. null,
  236. null) },
  237. { "Setout", new QueryDef<Setout>(
  238. new Filter<Setout>(x => x.JobLink.ID)
  239. .InQuery(new SubQuery<Kanban>(kanbanFilter, new Column<Kanban>(x => x.JobLink.ID))),
  240. new Columns<Setout>(x => x.JobLink.JobNumber, x => x.Number),
  241. null) }
  242. });
  243. tstages = tables["ManufacturingTemplateStage"].Rows
  244. .Select(x => x.ToObject<ManufacturingTemplateStage>()).ToArray();
  245. kanban = tables["Kanban"].Rows.FirstOrDefault()?.ToObject<Kanban>();
  246. if (kanban == null)
  247. {
  248. MessageBox.Show("Task does not exist!");
  249. return;
  250. }
  251. progress.Report("Creating Setouts");
  252. CoreTable setouts = tables["Setout"];
  253. int ireq = 0;
  254. string sreq = "";
  255. while (true)
  256. {
  257. ireq++;
  258. sreq = string.Format("{0}-{1:yyMMdd}-{2}", kanban.JobLink.JobNumber, DateTime.Now, ireq);
  259. if (!setouts.Rows.Any(r => sreq.Equals(r.Get<Setout, String>(c => c.Number))))
  260. break;
  261. }
  262. setoutNumber = sreq;
  263. });
  264. if (setoutNumber == null || kanban == null)
  265. {
  266. return;
  267. }
  268. var result = CreateSetout(
  269. task,
  270. s =>
  271. {
  272. s.Number = setoutNumber;
  273. s.JobLink.ID = task.JobID;
  274. var notes = kanban.Notes.ToList();
  275. var description = kanban.Summary;
  276. if (string.IsNullOrWhiteSpace(description))
  277. {
  278. description = CoreUtils.StripHTML(kanban.Description);
  279. }
  280. if (!string.IsNullOrWhiteSpace(description))
  281. {
  282. notes.Insert(0, description);
  283. }
  284. s.Description = string.Join("\n==========================================\n", notes);
  285. }
  286. );
  287. if (result != null)
  288. {
  289. Progress.ShowModal("Creating Manufacturing Packet", progress =>
  290. {
  291. ManufacturingPacket packet = new ManufacturingPacket()
  292. {
  293. Serial = template.Code,
  294. Title = kanban.Title,
  295. Quantity = 1,
  296. BarcodeQty = 1,
  297. DueDate = kanban.DueDate
  298. };
  299. packet.ManufacturingTemplateLink.ID = template.ID;
  300. packet.ManufacturingTemplateLink.Code = template.Code;
  301. packet.ManufacturingTemplateLink.Factory.ID = template.Factory.ID;
  302. packet.SetoutLink.ID = result.ID;
  303. new Client<ManufacturingPacket>().Save(packet, "Created from Task");
  304. DoLink<ManufacturingPacketKanban, ManufacturingPacket, ManufacturingPacketLink>(task, packet.ID);
  305. List<ManufacturingPacketStage> pstages = new List<ManufacturingPacketStage>();
  306. foreach (var tstage in tstages)
  307. {
  308. var pstage = new ManufacturingPacketStage()
  309. {
  310. Time = tstage.Time,
  311. Sequence = tstage.Sequence,
  312. SequenceType = tstage.SequenceType,
  313. Started = DateTime.MinValue,
  314. PercentageComplete = 0.0F,
  315. Completed = DateTime.MinValue,
  316. QualityChecks = tstage.QualityChecks,
  317. QualityStatus = QualityStatus.NotChecked,
  318. QualityNotes = "",
  319. };
  320. pstage.Parent.ID = packet.ID;
  321. pstage.ManufacturingSectionLink.ID = tstage.Section.ID;
  322. pstage.ManufacturingSectionLink.Name = tstage.Section.Name;
  323. pstages.Add(pstage);
  324. }
  325. new Client<ManufacturingPacketStage>().Save(pstages, "Created from Task", (_, __) => { });
  326. progress.Report("Processing Documents");
  327. List<SetoutDocument> _setoutdocuments = new List<SetoutDocument>();
  328. List<KanbanDocument> _kanbandocuments = new List<KanbanDocument>();
  329. KanbanDocument[] docrefs = new Client<KanbanDocument>()
  330. .Load(new Filter<KanbanDocument>(x => x.EntityLink.ID).IsEqualTo(kanban.ID));
  331. foreach (var docref in docrefs)
  332. {
  333. // Convert the document to a PDF
  334. var docid = ProcessKanbanDocument(docref);
  335. var newdoc = new SetoutDocument();
  336. newdoc.EntityLink.ID = result.ID;
  337. newdoc.DocumentLink.ID = docid;
  338. _setoutdocuments.Add(newdoc);
  339. if (docid != docref.DocumentLink.ID)
  340. {
  341. docref.DocumentLink.ID = docid;
  342. _kanbandocuments.Add(docref);
  343. }
  344. }
  345. new Client<SetoutDocument>().Save(_setoutdocuments, "Converted from Task", (_, __) => { });
  346. new Client<KanbanDocument>().Save(_kanbandocuments, "Converted to PDF", (_, __) => { });
  347. progress.Report("Updating Task");
  348. kanban.Title = kanban.Title + " (" + result.Number + ")";
  349. new Client<Kanban>().Save(kanban, "Converting Kanban to Setout");
  350. });
  351. control.Refresh();
  352. }
  353. });
  354. menu.AddItem("Create Requisition from Task", null, models, tasks =>
  355. {
  356. var taskModel = tasks.First();
  357. var kanbanTable = new Client<Kanban>().Query(new Filter<Kanban>(x => x.ID).IsEqualTo(taskModel.ID));
  358. var kanban = kanbanTable.Rows.First().ToObject<Kanban>();
  359. var result = CreateRequisition(
  360. taskModel,
  361. r =>
  362. {
  363. r.RequestedBy.ID = kanban.ManagerLink.ID;
  364. r.Employee.ID = Guid.Empty;
  365. r.Title = kanban.Title;
  366. r.Request = string.IsNullOrWhiteSpace(kanban.Summary)
  367. ? String.IsNullOrWhiteSpace(kanban.Description)
  368. ? String.Join("\n", kanban.Notes)
  369. : CoreUtils.StripHTML(kanban.Description)
  370. : kanban.Summary;
  371. r.Notes = kanban.Notes;
  372. r.Due = kanban.DueDate;
  373. r.JobLink.ID = taskModel.JobID;
  374. }
  375. );
  376. if (result != null)
  377. {
  378. Progress.ShowModal("Updating Documents", progress =>
  379. {
  380. progress.Report("Updating Documents");
  381. List<RequisitionDocument> requiDocuments = new();
  382. KanbanDocument[] kanbanDocuments = new Client<KanbanDocument>()
  383. .Load(new Filter<KanbanDocument>(x => x.EntityLink.ID).IsEqualTo(kanban.ID));
  384. foreach (var document in kanbanDocuments)
  385. {
  386. var newdoc = new RequisitionDocument();
  387. newdoc.EntityLink.ID = result.ID;
  388. newdoc.DocumentLink.ID = document.DocumentLink.ID;
  389. requiDocuments.Add(newdoc);
  390. }
  391. new Client<RequisitionDocument>().Save(requiDocuments, "Converted from Task", (_, __) => { });
  392. /*RequisitionKanban link = new();
  393. link.Entity.ID = result.ID;
  394. link.Kanban.ID = kanban.ID;
  395. new Client<RequisitionKanban>().Save(link, "Converting Task -> Requisition", (_, __) => { });*/
  396. progress.Report("Updating Task");
  397. kanban.Status = KanbanStatus.Open;
  398. kanban.Completed = DateTime.MinValue;
  399. kanban.Title += $" (Requi #{result.Number})";
  400. new Client<Kanban>().Save(kanban, "Converted to Requisition", (_, __) => { });
  401. });
  402. MessageBox.Show(String.Format("Created Requisition {0}", result.Number));
  403. control.Refresh();
  404. }
  405. });
  406. menu.AddItem("Create Delivery from Task", null, models, tasks =>
  407. {
  408. var result = CreateDelivery(
  409. tasks.First(),
  410. d =>
  411. {
  412. // Post-Process Requi Here
  413. }
  414. );
  415. if (result != null)
  416. control.Refresh();
  417. });
  418. menu.AddItem("Create Purchase Order from Task", null, models, tasks =>
  419. {
  420. var result = CreateOrder(
  421. tasks.First(),
  422. p =>
  423. {
  424. // Post-Process Requi Here
  425. }
  426. );
  427. if (result != null)
  428. control.Refresh();
  429. });
  430. }
  431. }
  432. if (!bLinks && canChange)
  433. {
  434. menu.Items.Add(new Separator());
  435. var changeStatus = new MenuItem { Header = "Change Status" };
  436. AddChangeStatusButton(control, models, changeStatus, "Open", KanbanStatus.Open);
  437. AddChangeStatusButton(control, models, changeStatus, "In Progress", KanbanStatus.InProgress);
  438. AddChangeStatusButton(control, models, changeStatus, "Waiting", KanbanStatus.Waiting);
  439. if (models.Any(x => x.CompletedDate.IsEmpty()))
  440. {
  441. var complete = new MenuItem
  442. {
  443. Tag = models,
  444. Header = models.Length > 1 ? "Complete Tasks" : "Complete Task"
  445. };
  446. complete.Click += (o, e) =>
  447. {
  448. CompleteTask(control, e, DateTime.Now);
  449. };
  450. menu.Items.Add(complete);
  451. if (Security.IsAllowed<CanSetKanbanCompletedDate>())
  452. {
  453. var completeDate = new MenuItem
  454. {
  455. Tag = models,
  456. Header = "Set Completed Date"
  457. };
  458. var dateItem = new MenuItem();
  459. var dateCalendar = new System.Windows.Controls.Calendar { SelectedDate = DateTime.MinValue };
  460. dateCalendar.Tag = models;
  461. dateCalendar.SelectedDatesChanged += (o, e) =>
  462. {
  463. if (e.Source is not System.Windows.Controls.Calendar calendar) return;
  464. menu.IsOpen = false;
  465. var selectedDate = calendar.SelectedDate ?? DateTime.Now;
  466. CompleteTask(control, e, selectedDate);
  467. };
  468. dateItem.Header = dateCalendar;
  469. dateItem.Style = Resources["calendarItem"] as Style;
  470. completeDate.Items.Add(dateItem);
  471. menu.Items.Add(completeDate);
  472. }
  473. }
  474. else
  475. {
  476. menu.AddItem(models.Length > 1 ? "Archive Tasks" : "Archive Task", null, models, tasks =>
  477. {
  478. if (MessageBox.Show("Are you sure you want to remove the selected tasks from the list?", "Confirm removal",
  479. MessageBoxButton.YesNo) != MessageBoxResult.Yes)
  480. return;
  481. Progress.ShowModal("Closing Kanbans", progress =>
  482. {
  483. var kanbans = LoadKanbans(tasks, new Columns<Kanban>(x => x.ID, x => x.Closed));
  484. foreach (var kanban in kanbans)
  485. kanban.Closed = DateTime.Now;
  486. new Client<Kanban>().Save(kanbans, "Kanban Marked as Closed");
  487. });
  488. control.Refresh();
  489. });
  490. }
  491. menu.Items.Add(changeStatus);
  492. var changeType = new MenuItem { Header = "Change Task Type", Tag = models };
  493. foreach(var type in KanbanTypes)
  494. {
  495. changeType.AddItem($"{type.Code}: {type.Description}", null, type, type =>
  496. {
  497. Progress.ShowModal("Changing Task Type", progress =>
  498. {
  499. var kanbans = LoadKanbans(models, new Columns<Kanban>(x => x.ID, x => x.Type.ID));
  500. foreach (var kanban in kanbans)
  501. {
  502. kanban.Type.ID = type.ID;
  503. }
  504. new Client<Kanban>().Save(kanbans, $"Kanban Task Type changed to {type}");
  505. });
  506. control.Refresh();
  507. });
  508. }
  509. menu.Items.Add(changeType);
  510. var changeDueDate = new MenuItem { Header = "Change Due Date" };
  511. var calendarItem = new MenuItem();
  512. var calendar = new System.Windows.Controls.Calendar { SelectedDate = models.Length == 1 ? models[0].DueDate : DateTime.Today };
  513. calendar.Tag = models;
  514. calendar.SelectedDatesChanged += (o, e) =>
  515. {
  516. if (e.Source is not System.Windows.Controls.Calendar calendar) return;
  517. var selectedDate = calendar.SelectedDate ?? DateTime.Now;
  518. var models = (calendar.Tag as IList<TaskModel>)!;
  519. Progress.ShowModal("Changing Due Date", progress =>
  520. {
  521. var kanbans = LoadKanbans(models, new Columns<Kanban>(x => x.ID, x => x.DueDate));
  522. foreach (var kanban in kanbans)
  523. {
  524. kanban.DueDate = selectedDate;
  525. }
  526. new Client<Kanban>().Save(kanbans, $"Kanban Due Date changed to {selectedDate:dd MMM yyyy}");
  527. });
  528. control.Refresh();
  529. menu.IsOpen = false;
  530. };
  531. calendarItem.Header = calendar;
  532. calendarItem.Style = Resources["calendarItem"] as Style;
  533. changeDueDate.Items.Add(calendarItem);
  534. menu.Items.Add(changeDueDate);
  535. }
  536. }
  537. /// <summary>
  538. /// Takes a <see cref="KanbanDocument"/>, and if it is a .txt or an image (".png", ".jpg", ".jpeg" or ".bmp"), converts to a PDF
  539. /// with the content of the document, saving a new document with extension changed to ".pdf".
  540. /// </summary>
  541. /// <param name="docref">The original document.</param>
  542. /// <returns>
  543. /// The ID of the new <see cref="Document"/> or,
  544. /// if not one of the given types, the original document ID.
  545. /// </returns>
  546. private static Guid ProcessKanbanDocument(KanbanDocument docref)
  547. {
  548. var result = docref.DocumentLink.ID;
  549. var ext = System.IO.Path.GetExtension(docref.DocumentLink.FileName).ToLower();
  550. if (ext.EndsWith("txt"))
  551. {
  552. var doc = new Client<Document>().Load(new Filter<Document>(x => x.ID).IsEqualTo(docref.DocumentLink.ID)).FirstOrDefault();
  553. if (doc is null)
  554. {
  555. Logger.Send(LogType.Error, "", $"Document {docref.DocumentLink.ID} does not exist!");
  556. return docref.DocumentLink.ID;
  557. }
  558. PdfDocument pdf = new PdfDocument();
  559. PdfPage page = pdf.Pages.Add();
  560. PdfGraphics graphics = page.Graphics;
  561. PdfFont font = new PdfStandardFont(PdfFontFamily.Courier, 12);
  562. String text = System.Text.Encoding.UTF8.GetString(doc.Data);
  563. graphics.DrawString(text, font, PdfBrushes.Black, new PointF(0, 0));
  564. MemoryStream ms = new MemoryStream();
  565. pdf.Save(ms);
  566. pdf.Close(true);
  567. byte[] data = ms.ToArray();
  568. var newdoc = new Document()
  569. {
  570. Data = data,
  571. FileName = System.IO.Path.ChangeExtension(docref.DocumentLink.FileName, "pdf"),
  572. CRC = CoreUtils.CalculateCRC(data),
  573. TimeStamp = DateTime.Now,
  574. };
  575. new Client<Document>().Save(newdoc, "Converted from Text");
  576. return newdoc.ID;
  577. }
  578. else if (ext.EndsWith("png") || ext.EndsWith("bmp") || ext.EndsWith("jpg") || ext.EndsWith("jpeg"))
  579. {
  580. var doc = new Client<Document>().Load(new Filter<Document>(x => x.ID).IsEqualTo(docref.DocumentLink.ID)).FirstOrDefault();
  581. if (doc is null)
  582. {
  583. Logger.Send(LogType.Error, "", $"Document {docref.DocumentLink.ID} does not exist!");
  584. return docref.DocumentLink.ID;
  585. }
  586. PdfBitmap image = new PdfBitmap(new MemoryStream(doc.Data));
  587. PdfDocument pdf = new PdfDocument();
  588. pdf.PageSettings.Orientation = image.Height > image.Width ? PdfPageOrientation.Portrait : PdfPageOrientation.Landscape;
  589. pdf.PageSettings.Size = new SizeF(image.Width, image.Height);
  590. PdfPage page = pdf.Pages.Add();
  591. PdfGraphics graphics = page.Graphics;
  592. graphics.DrawImage(image, 0.0F, 0.0F);
  593. MemoryStream ms = new MemoryStream();
  594. pdf.Save(ms);
  595. pdf.Close(true);
  596. byte[] data = ms.ToArray();
  597. var newdoc = new Document()
  598. {
  599. Data = data,
  600. FileName = System.IO.Path.ChangeExtension(docref.DocumentLink.FileName, "pdf"),
  601. CRC = CoreUtils.CalculateCRC(data),
  602. TimeStamp = DateTime.Now,
  603. };
  604. new Client<Document>().Save(newdoc, "Converted from Image");
  605. return newdoc.ID;
  606. }
  607. return result;
  608. }
  609. private void CreateJobSubMenu(ITaskControl control, MenuItem job, IEnumerable<TaskModel> tasks)
  610. {
  611. job.Items.Clear();
  612. job.Items.Add(new MenuItem { Header = "Loading...", IsEnabled = false });
  613. using (new WaitCursor())
  614. {
  615. job.Items.Clear();
  616. var jobs = new Client<Job>().Query(
  617. LookupFactory.DefineFilter<Job>(),
  618. LookupFactory.DefineColumns<Job>(),
  619. LookupFactory.DefineSort<Job>()
  620. );
  621. foreach (var row in jobs.Rows)
  622. {
  623. var jobNumber = row.Get<Job, string>(x => x.JobNumber);
  624. var jobName = row.Get<Job, string>(x => x.Name);
  625. job.AddItem($"{jobNumber}: {jobName}", null, tasks, tasks =>
  626. {
  627. using (new WaitCursor())
  628. {
  629. var kanbans = LoadKanbans(tasks, new Columns<Kanban>(x => x.ID, x => x.JobLink.ID));
  630. foreach (var kanban in kanbans)
  631. kanban.JobLink.ID = row.Get<Job, Guid>(x => x.ID);
  632. new Client<Kanban>().Save(kanbans, "Updated Job Number");
  633. control.Refresh();
  634. }
  635. });
  636. }
  637. }
  638. }
  639. #endregion
  640. private void TaskPanels_SelectionChanged(object sender, SelectionChangedEventArgs e)
  641. {
  642. if (!IsReady)
  643. return;
  644. if (e.Source is not TabControl)
  645. return;
  646. if (_bTabChanging)
  647. return;
  648. try
  649. {
  650. _bTabChanging = true;
  651. var panel = GetCurrentPanel();
  652. if(panel is not null)
  653. {
  654. KanbanSettings.ViewType = panel.KanbanViewType;
  655. new UserConfiguration<KanbanSettings>().Save(KanbanSettings);
  656. panel.Refresh();
  657. }
  658. }
  659. finally
  660. {
  661. _bTabChanging = false;
  662. }
  663. }
  664. #region Get/Save Settings
  665. private KanbanSettings? _settings;
  666. public KanbanSettings KanbanSettings
  667. {
  668. get
  669. {
  670. _settings ??= new UserConfiguration<KanbanSettings>().Load();
  671. return _settings;
  672. }
  673. }
  674. public void SaveSettings()
  675. {
  676. if(_settings != null)
  677. new UserConfiguration<KanbanSettings>().Save(_settings);
  678. }
  679. #endregion
  680. #region IPanel Stuff
  681. public event DataModelUpdateEvent? OnUpdateDataModel;
  682. public bool IsReady { get; set; }
  683. public void CreateToolbarButtons(IPanelHost host)
  684. {
  685. host.CreatePanelAction(
  686. new PanelAction
  687. {
  688. Caption = "New Task",
  689. OnExecute = a => {
  690. if(CreateKanban(k => { }) != null)
  691. {
  692. Refresh();
  693. }
  694. },
  695. Image = PRSDesktop.Resources.add
  696. }
  697. );
  698. if (Security.CanView<KanbanType>())
  699. {
  700. host.CreateSetupAction(
  701. new PanelAction
  702. {
  703. Caption = "Task Types",
  704. Image = PRSDesktop.Resources.kanbantype,
  705. OnExecute = a =>
  706. {
  707. var list = new MasterList(typeof(KanbanType));
  708. list.ShowDialog();
  709. }
  710. });
  711. }
  712. }
  713. public Dictionary<string, object[]> Selected()
  714. {
  715. return new Dictionary<string, object[]>();
  716. }
  717. public void Heartbeat(TimeSpan time)
  718. {
  719. }
  720. private readonly Dictionary<KanbanViewType, TabItem> _viewmap = new();
  721. private readonly List<ITaskControl> _initialized = new();
  722. private ITaskControl GetCurrentPanel()
  723. {
  724. var result = (TaskPanels.SelectedContent as ITaskControl)!;
  725. if (result == null)
  726. result = (TaskPanels.Items[0] as DynamicTabItem)?.Content as ITaskControl;
  727. try
  728. {
  729. //if (result != null)
  730. if (!_initialized.Contains(result))
  731. {
  732. result.Setup();
  733. result.IsReady = true;
  734. _initialized.Add(result);
  735. }
  736. }
  737. catch (Exception e)
  738. {
  739. Logger.Send(LogType.Error, "", $"Error in TaskPanel.GetCurrentPanel: {CoreUtils.FormatException(e)}");
  740. }
  741. return result;
  742. }
  743. public void Setup()
  744. {
  745. _settings = new UserConfiguration<KanbanSettings>().Load();
  746. TaskPanels.SelectedItem = _viewmap[_settings.ViewType];
  747. kanbanTypes = new Client<KanbanType>()
  748. .Query(new Filter<KanbanType>(x => x.Hidden).IsEqualTo(false), new Columns<KanbanType>(x => x.ID, x => x.Code, x => x.Description))
  749. .Rows.Select(x => x.ToObject<KanbanType>()).ToArray();
  750. }
  751. public void Shutdown(CancelEventArgs? cancel)
  752. {
  753. }
  754. public void Refresh()
  755. {
  756. if (Master == null)
  757. {
  758. if (Equals(TaskPanels.SelectedItem, TasksPlannerTabItem))
  759. TaskPanels.SelectedItem = _viewmap[KanbanViewType.Status];
  760. TasksPlannerTabItem.Visibility = Visibility.Collapsed;
  761. }
  762. else
  763. TasksPlannerTabItem.Visibility = Visibility.Visible;
  764. GetCurrentPanel()?.Refresh();
  765. }
  766. public string SectionName => GetCurrentPanel().SectionName;
  767. public TaskPanelProperties Properties { get; set; }
  768. public DataModel DataModel(Selection selection)
  769. {
  770. return GetCurrentPanel().DataModel(selection);
  771. //return new AutoDataModel<Kanban>(new Filter<Kanban>(x => x.ID).IsEqualTo(Guid.Empty));
  772. }
  773. #endregion
  774. #region CRUD Functionality
  775. private TEntity? DoCreate<TEntity>(Action<TEntity> customise)
  776. where TEntity : Entity, IRemotable, IPersistent, new()
  777. {
  778. var result = new TEntity();
  779. customise?.Invoke(result);
  780. if (DoEdit(new[] { result }, null))
  781. return result;
  782. return null;
  783. }
  784. private readonly Dictionary<Type, IDynamicGrid> _grids = new();
  785. private readonly List<Tuple<Guid, Entity>> _entitycache = new();
  786. private DynamicDataGrid<TEntity> GetGrid<TEntity>() where TEntity : Entity, IRemotable, IPersistent, new()
  787. {
  788. if(!_grids.TryGetValue(typeof(TEntity), out var grid))
  789. {
  790. grid = (DynamicGridUtils.CreateDynamicGrid(typeof(DynamicDataGrid<>), typeof(TEntity)) as DynamicDataGrid<TEntity>)!;
  791. _grids[typeof(TEntity)] = grid;
  792. if (typeof(TEntity) == typeof(Kanban))
  793. {
  794. CustomiseKanbanGrid((grid as DynamicDataGrid<Kanban>)!);
  795. }
  796. }
  797. return (grid as DynamicDataGrid<TEntity>)!;
  798. }
  799. private IEnumerable<TEntity> DoLoad<TEntity>(IEnumerable<TaskModel> models, Columns<TEntity> columns)
  800. where TEntity : Entity, IRemotable, IPersistent, new()
  801. {
  802. var result = new List<TEntity>();
  803. var load = new List<Guid>();
  804. foreach (var model in models)
  805. {
  806. var entity = _entitycache.FirstOrDefault(x => Equals(x.Item1, model.ID) && x.Item2 is TEntity) as TEntity;
  807. if (entity is not null)
  808. result.Add(entity);
  809. else
  810. load.Add(model.ID);
  811. }
  812. if (load.Any())
  813. {
  814. var entities = new Client<TEntity>()
  815. .Query(new Filter<TEntity>(x => x.ID).InList(load.ToArray()), columns)
  816. .Rows.Select(x => x.ToObject<TEntity>()).ToList();
  817. foreach (var entity in entities)
  818. _entitycache.Add(new Tuple<Guid, Entity>(entity.ID, entity));
  819. result.AddRange(entities);
  820. }
  821. return result;
  822. }
  823. private IEnumerable<TEntity> DoLoad<TEntityKanban, TEntity, TLink>(IEnumerable<TaskModel> models, Columns<TEntity> columns)
  824. where TEntityKanban : EntityKanban<TEntity, TLink>, new()
  825. where TEntity : Entity, IRemotable, IPersistent, new()
  826. where TLink : IEntityLink<TEntity>, new()
  827. {
  828. var result = DoLoad(models, columns);
  829. if (!result.Any())
  830. foreach (var model in models)
  831. {
  832. result = new Client<TEntity>().Load(
  833. new Filter<TEntity>(x => x.ID).InQuery(new Filter<TEntityKanban>(x => x.Kanban.ID).IsEqualTo(model.ID),
  834. x => x.Entity.ID));
  835. foreach (var r in result)
  836. _entitycache.Add(new Tuple<Guid, Entity>(model.ID, r));
  837. }
  838. return result;
  839. }
  840. private void DoCache<TEntity>(Guid kanbanid, TEntity entity) where TEntity : Entity
  841. {
  842. if (!_entitycache.Any(x => Equals(x.Item1, kanbanid) && x.Item2 is TEntity && Equals(x.Item2.ID, entity.ID)))
  843. _entitycache.Add(new Tuple<Guid, Entity>(kanbanid, entity));
  844. }
  845. private bool DoEdit<TEntity>(IEnumerable<TEntity> entities, Action<TEntity>? action = null)
  846. where TEntity : Entity, IRemotable, IPersistent, new()
  847. {
  848. if (entities == null || !entities.Any())
  849. return false;
  850. foreach (var entity in entities)
  851. action?.Invoke(entity);
  852. return GetGrid<TEntity>().EditItems(entities.ToArray());
  853. }
  854. private void DoLink<TEntityKanban, TEntity, TLink>(TaskModel model, Guid entityid)
  855. where TEntityKanban : EntityKanban<TEntity, TLink>, new()
  856. where TEntity : Entity, IRemotable, IPersistent, new()
  857. where TLink : IEntityLink<TEntity>, new()
  858. {
  859. var linktask = Task.Run(() =>
  860. {
  861. var link = new TEntityKanban();
  862. link.Kanban.ID = model.ID;
  863. link.Entity.ID = entityid;
  864. new Client<TEntityKanban>().Save(link, "");
  865. });
  866. var kanbantask = Task.Run(() =>
  867. {
  868. var kanban = LoadKanbans(new[] { model }, new Columns<Kanban>(x => x.ID, x => x.Locked)).FirstOrDefault();
  869. if (kanban is not null)
  870. {
  871. kanban.Locked = true;
  872. new Client<Kanban>().Save(kanban, "Locked because of linked " + typeof(TEntity).EntityName().Split('.').Last());
  873. }
  874. });
  875. Task.WaitAll(linktask, kanbantask);
  876. }
  877. private static void DoDelete<TEntity>(IList<TEntity> entities, string auditnote)
  878. where TEntity : Entity, IRemotable, IPersistent, new()
  879. {
  880. new Client<TEntity>().Delete(entities, auditnote);
  881. }
  882. public Kanban? CreateKanban(Action<Kanban> customise)
  883. {
  884. var result = DoCreate<Kanban>(
  885. kanban =>
  886. {
  887. kanban.Title = "New Task";
  888. kanban.Description = "";
  889. kanban.Status = KanbanStatus.Open;
  890. kanban.DueDate = DateTime.Today;
  891. kanban.Private = false;
  892. kanban.JobLink.ID = Master?.ID ?? Guid.Empty;
  893. kanban.JobLink.Synchronise(Master ?? new Job());
  894. kanban.EmployeeLink.ID = App.EmployeeID;
  895. kanban.ManagerLink.ID = App.EmployeeID;
  896. customise?.Invoke(kanban);
  897. });
  898. if (result != null)
  899. DoCache(result.ID, result);
  900. return result;
  901. }
  902. public IEnumerable<Kanban> LoadKanbans(IEnumerable<TaskModel> models, Columns<Kanban> columns)
  903. {
  904. columns.Add(x => x.ID);
  905. columns.Add(x => x.Number);
  906. columns.Add(x => x.Title);
  907. columns.Add(x => x.Notes);
  908. columns.Add(x => x.Summary);
  909. columns.Add(x => x.Completed);
  910. columns.Add(x => x.DueDate);
  911. columns.Add(x => x.ManagerLink.ID);
  912. columns.Add(x => x.EmployeeLink.ID);
  913. return DoLoad(models, columns);
  914. }
  915. public void OnValidateKanban(object sender, Kanban[] items, List<string> errors)
  916. {
  917. if (Properties.RequireTaskTypes && items.Any(x => x.Type.ID == Guid.Empty))
  918. {
  919. errors.Add("[Task Type] may not be blank!");
  920. }
  921. }
  922. public void CustomiseKanbanGrid(DynamicDataGrid<Kanban> grid)
  923. {
  924. grid.OnValidate += OnValidateKanban;
  925. }
  926. public bool EditKanbans(IEnumerable<TaskModel> models, Action<Kanban>? customise = null)
  927. {
  928. var entities = LoadKanbans(models, GetGrid<Kanban>().LoadEditorColumns());
  929. return DoEdit(entities, customise);
  930. }
  931. public void DeleteKanbans(IEnumerable<TaskModel> models, string auditnote)
  932. {
  933. var kanbans = models.Select(x => new Kanban { ID = x.ID }).ToList();
  934. DoDelete(kanbans, auditnote);
  935. }
  936. public Requisition? CreateRequisition(TaskModel model, Action<Requisition>? customise)
  937. {
  938. var result = DoCreate<Requisition>(
  939. requi =>
  940. {
  941. requi.JobLink.ID = Master?.ID ?? Guid.Empty;
  942. requi.JobLink.Synchronise(Master ?? new Job());
  943. customise?.Invoke(requi);
  944. });
  945. if (result != null)
  946. {
  947. DoCache(model.ID, result);
  948. DoLink<RequisitionKanban, Requisition, RequisitionLink>(model, result.ID);
  949. }
  950. return result;
  951. }
  952. public bool EditRequisitions(IEnumerable<TaskModel> models, Action<Requisition>? customise = null)
  953. {
  954. var requis = DoLoad<RequisitionKanban, Requisition, RequisitionLink>(models, GetGrid<Requisition>().LoadEditorColumns());
  955. if (requis.Any())
  956. return DoEdit(requis, customise);
  957. return false;
  958. }
  959. public Setout? CreateSetout(TaskModel model, Action<Setout> customise)
  960. {
  961. var result = DoCreate<Setout>(
  962. setout =>
  963. {
  964. setout.JobLink.ID = Master?.ID ?? Guid.Empty;
  965. setout.JobLink.Synchronise(Master ?? new Job());
  966. customise?.Invoke(setout);
  967. });
  968. if (result != null)
  969. {
  970. DoCache(model.ID, result);
  971. //DoLink<SetoutKanban, Setout, SetoutLink>(model, result.ID);
  972. }
  973. return result;
  974. }
  975. public bool EditSetouts(IEnumerable<TaskModel> models, Action<Setout>? customise = null)
  976. {
  977. var setouts = DoLoad<SetoutKanban, Setout, SetoutLink>(models, GetGrid<Setout>().LoadEditorColumns());
  978. if (setouts.Any())
  979. return DoEdit(setouts, customise);
  980. return false;
  981. }
  982. public Delivery? CreateDelivery(TaskModel model, Action<Delivery> customise)
  983. {
  984. var result = DoCreate<Delivery>(
  985. delivery =>
  986. {
  987. delivery.Job.ID = Master?.ID ?? Guid.Empty;
  988. delivery.Job.Synchronise(Master ?? new Job());
  989. customise?.Invoke(delivery);
  990. });
  991. if (result != null)
  992. {
  993. DoCache(model.ID, result);
  994. DoLink<DeliveryKanban, Delivery, DeliveryLink>(model, result.ID);
  995. }
  996. return result;
  997. }
  998. public bool EditDeliveries(IEnumerable<TaskModel> models, Action<Delivery>? customise = null)
  999. {
  1000. var deliveries = DoLoad<DeliveryKanban, Delivery, DeliveryLink>(models, GetGrid<Delivery>().LoadEditorColumns());
  1001. if (deliveries.Any())
  1002. return DoEdit(deliveries, customise);
  1003. return false;
  1004. }
  1005. public PurchaseOrder? CreateOrder(TaskModel model, Action<PurchaseOrder> customise)
  1006. {
  1007. var result = DoCreate<PurchaseOrder>(
  1008. order => { customise?.Invoke(order); });
  1009. if (result != null)
  1010. {
  1011. DoCache(model.ID, result);
  1012. DoLink<PurchaseOrderKanban, PurchaseOrder, PurchaseOrderLink>(model, result.ID);
  1013. }
  1014. return result;
  1015. }
  1016. public bool EditPurchaseOrders(IEnumerable<TaskModel> models, Action<PurchaseOrder>? customise = null)
  1017. {
  1018. var orders = DoLoad<PurchaseOrderKanban, PurchaseOrder, PurchaseOrderLink>(models, GetGrid<PurchaseOrder>().LoadEditorColumns());
  1019. if (orders.Any())
  1020. return DoEdit(orders, customise);
  1021. return false;
  1022. }
  1023. #endregion
  1024. #region EntityReferences
  1025. private static void AddQuery<TEntityKanban, TEntity, TLink>(MultiQuery query, Guid[] taskids)
  1026. where TEntityKanban : EntityKanban<TEntity, TLink>, new()
  1027. where TEntity : Entity
  1028. where TLink : IEntityLink<TEntity>, new()
  1029. {
  1030. query.Add(
  1031. new Filter<TEntityKanban>(x => x.Kanban.ID).InList(taskids),
  1032. new Columns<TEntityKanban>(x => x.Entity.ID).Add(x => x.Kanban.ID)
  1033. );
  1034. }
  1035. private static Guid[] ExtractIDs<TEntityKanban, TEntity, TLink>(MultiQuery query)
  1036. where TEntityKanban : EntityKanban<TEntity, TLink>, new()
  1037. where TEntity : Entity
  1038. where TLink : IEntityLink<TEntity>, new()
  1039. {
  1040. var lookup = query.Get<TEntityKanban>().ToLookup<TEntityKanban, Guid, Guid>(x => x.Kanban.ID, x => x.Entity.ID);
  1041. return query.Get<TEntityKanban>().ExtractValues<TEntityKanban, Guid>(x => x.Entity.ID).ToArray();
  1042. }
  1043. public KanbanReferences[] GetReferences(IEnumerable<TaskModel> models)
  1044. {
  1045. var result = new List<KanbanReferences>();
  1046. var ids = models.Select(x => x.ID).ToArray();
  1047. var query = new MultiQuery();
  1048. AddQuery<RequisitionKanban, Requisition, RequisitionLink>(query, ids);
  1049. AddQuery<SetoutKanban, Setout, SetoutLink>(query, ids);
  1050. AddQuery<DeliveryKanban, Delivery, DeliveryLink>(query, ids);
  1051. AddQuery<PurchaseOrderKanban, PurchaseOrder, PurchaseOrderLink>(query, ids);
  1052. query.Query();
  1053. var requis = query.Get<RequisitionKanban>().ToLookup<RequisitionKanban, Guid, Guid>(x => x.Kanban.ID, x => x.Entity.ID);
  1054. var setouts = query.Get<SetoutKanban>().ToLookup<SetoutKanban, Guid, Guid>(x => x.Kanban.ID, x => x.Entity.ID);
  1055. var deliveries = query.Get<DeliveryKanban>().ToLookup<DeliveryKanban, Guid, Guid>(x => x.Kanban.ID, x => x.Entity.ID);
  1056. var orders = query.Get<PurchaseOrderKanban>().ToLookup<PurchaseOrderKanban, Guid, Guid>(x => x.Kanban.ID, x => x.Entity.ID);
  1057. foreach (var id in ids)
  1058. {
  1059. var references = new KanbanReferences
  1060. {
  1061. Kanban = id,
  1062. Requisitions = requis.Contains(id) ? requis[id].ToArray() : Array.Empty<Guid>(),
  1063. Setouts = setouts.Contains(id) ? setouts[id].ToArray() : Array.Empty<Guid>(),
  1064. Deliveries = deliveries.Contains(id) ? deliveries[id].ToArray() : Array.Empty<Guid>(),
  1065. Orders = orders.Contains(id) ? orders[id].ToArray() : Array.Empty<Guid>()
  1066. };
  1067. result.Add(references);
  1068. }
  1069. return result.ToArray();
  1070. }
  1071. public bool EditReferences(IEnumerable<TaskModel> models)
  1072. {
  1073. var result = false;
  1074. var refs = GetReferences(models).First();
  1075. if (refs.ReferenceType() == typeof(Requisition))
  1076. result = EditRequisitions(
  1077. models,
  1078. requi =>
  1079. {
  1080. requi.Notes = Utility.ProcessNotes(requi.Notes, requi.Request);
  1081. requi.Request = "";
  1082. }
  1083. );
  1084. else if (refs.ReferenceType() == typeof(Setout))
  1085. result = EditSetouts(models);
  1086. else if (refs.ReferenceType() == typeof(Delivery))
  1087. result = EditDeliveries(models);
  1088. else if (refs.ReferenceType() == typeof(PurchaseOrder))
  1089. result = EditPurchaseOrders(models);
  1090. else
  1091. result = EditKanbans(models);
  1092. return result;
  1093. }
  1094. #endregion
  1095. }
  1096. }