LeaveRequestGrid.cs 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Threading.Tasks;
  5. using System.Windows;
  6. using System.Windows.Controls;
  7. using System.Windows.Media.Imaging;
  8. using Comal.Classes;
  9. using InABox.Clients;
  10. using InABox.Core;
  11. using InABox.DynamicGrid;
  12. using InABox.WPF;
  13. using Syncfusion.Linq;
  14. namespace PRSDesktop
  15. {
  16. public class LeaveRequestGrid : DynamicDataGrid<LeaveRequest>
  17. {
  18. //public bool FutureOnly = true;
  19. //private bool InProgressOnly = true;
  20. protected override void Init()
  21. {
  22. base.Init();
  23. HiddenColumns.Add(x=>x.OpenForms);
  24. HiddenColumns.Add(x=>x.FormCount);
  25. HiddenColumns.Add(x=>x.ApprovalSet.ID);
  26. HiddenColumns.Add(x=>x.OutstandingApprovals);
  27. ActionColumns.Add(new DynamicImageColumn(FormImage, FormAction) { Position = DynamicActionColumnPosition.End, ToolTip = FormToolTip});
  28. ActionColumns.Add(new DynamicMenuColumn(BuildMenu));
  29. }
  30. private Guid[]? availableApprovalSets = null;
  31. private void BuildMenu(DynamicMenuColumn column, CoreRow? row)
  32. {
  33. if (row is null) return;
  34. var menu = column.GetMenu();
  35. availableApprovalSets ??= Client.Query(
  36. new Filter<LeaveRequestApprovalSetEmployee>(x=>x.Employee.ID).IsEqualTo(App.EmployeeID),
  37. Columns.None<LeaveRequestApprovalSetEmployee>().Add(x=>x.ApprovalSet.ID)
  38. ).Rows
  39. .Select(r =>r.Get<LeaveRequestApprovalSetEmployee,Guid>(c=>c.ApprovalSet.ID))
  40. .Distinct()
  41. .ToArray();
  42. if (availableApprovalSets.Contains(row.Get<LeaveRequest, Guid>(x => x.ApprovalSet.ID)))
  43. {
  44. var approvals = row.Get<LeaveRequest, string>(x => x.OutstandingApprovals).Split(',');
  45. if (approvals.Contains(App.EmployeeCode))
  46. {
  47. menu.AddItem("Approve", null, () => UpdateLeaveRequest(row,LeaveRequestApprovalStatus.Approved));
  48. menu.AddItem("Reject", null, () => UpdateLeaveRequest(row,LeaveRequestApprovalStatus.Rejected, true));
  49. }
  50. else
  51. menu.AddItem("Reset", null, () => UpdateLeaveRequest(row,LeaveRequestApprovalStatus.NotYetApproved));
  52. }
  53. if (Security.IsAllowed<ManageLeaveRequestApprovals>())
  54. {
  55. if (menu.Items.Count > 0)
  56. menu.AddSeparator();
  57. var approvalSetItem = menu.AddItem("Approval Set", null, null);
  58. approvalSetItem.AddItem("Loading...", null, null, enabled: false);
  59. menu.AddItem("Manage Approvals", null, row, ManageApprovals_Click);
  60. var id = row.Get<LeaveRequest, Guid>(x => x.ID);
  61. var selectedApproval = row.Get<LeaveRequest, Guid>(x => x.ApprovalSet.ID);
  62. Task.Run(() =>
  63. {
  64. return Client.Query<LeaveRequestApprovalSet>(
  65. null,
  66. Columns.None<LeaveRequestApprovalSet>().Add(x => x.ID).Add(x => x.Code).Add(x => x.Description))
  67. .ToArray<LeaveRequestApprovalSet>();
  68. }).ContinueWith(approvals =>
  69. {
  70. approvalSetItem.Items.Clear();
  71. foreach(var approval in approvals.Result)
  72. {
  73. var item = approvalSetItem.AddItem($"{approval.Code}: {approval.Description}", null, (row, approval), ApprovalSet_Click);
  74. item.IsChecked = approval.ID == selectedApproval;
  75. }
  76. if (row.Get<LeaveRequest, Guid>(x => x.ApprovalSet.ID) != Guid.Empty)
  77. {
  78. if (approvalSetItem.Items.Count > 0)
  79. approvalSetItem.AddSeparator();
  80. approvalSetItem.AddItem("Clear Approval Set", null, row, ApprovalSetClear_Click);
  81. }
  82. }, TaskScheduler.FromCurrentSynchronizationContext());
  83. }
  84. }
  85. private void UpdateLeaveRequest(CoreRow row, LeaveRequestApprovalStatus status, bool requireNotes = false)
  86. {
  87. var note = "";
  88. if (requireNotes)
  89. {
  90. if (!TextBoxDialog.Execute("Reason for Rejection", ref note, false))
  91. return;
  92. }
  93. var requestid = row.Get<LeaveRequest, Guid>(x => x.ID);
  94. var approval = Client.Query(
  95. new Filter<LeaveRequestApproval>(x => x.LeaveRequest.ID).IsEqualTo(requestid)
  96. .And(x => x.Employee.ID).IsEqualTo(App.EmployeeID),
  97. Columns.None<LeaveRequestApproval>().Add(x=>x.ID).Add(x=>x.Status)
  98. ).Rows
  99. .FirstOrDefault()?
  100. .ToObject<LeaveRequestApproval>();
  101. if ((approval != null) && (approval.Status != status))
  102. {
  103. approval.Status = status;
  104. approval.Notes = note;
  105. Client.Save(approval,$"Status Updated to {status}");
  106. Refresh(false, true);
  107. }
  108. }
  109. private void ApprovalSetClear_Click(CoreRow row)
  110. {
  111. var bill = row.ToObject<LeaveRequest>();
  112. bill.ApprovalSet.ID = Guid.Empty;
  113. SaveItem(bill);
  114. Refresh(false, true);
  115. }
  116. private void ManageApprovals_Click(CoreRow row)
  117. {
  118. var request = row.ToObject<LeaveRequest>();
  119. var grid = new LeaveRequestApprovalGrid(request);
  120. var dlg = new DynamicContentDialog(grid, buttonsVisible: false)
  121. {
  122. Title = "Manage Approvals"
  123. };
  124. grid.Refresh(true, true);
  125. dlg.ShowDialog();
  126. request = LoadItem(row);
  127. UpdateRow(row, request);
  128. DoChanged();
  129. }
  130. private void ApprovalSet_Click((CoreRow row, LeaveRequestApprovalSet approvalSet) item)
  131. {
  132. var request = item.row.ToObject<LeaveRequest>();
  133. if (request.ApprovalSet.ID == item.approvalSet.ID) return;
  134. request.ApprovalSet.CopyFrom(item.approvalSet);
  135. SaveItem(request);
  136. Refresh(false, true);
  137. }
  138. private readonly BitmapImage _openforms = PRSDesktop.Resources.warning.AsBitmapImage();
  139. private readonly BitmapImage _completedforms = PRSDesktop.Resources.contract.AsBitmapImage();
  140. private BitmapImage? FormImage(CoreRow? row)
  141. {
  142. return row == null
  143. ? _completedforms
  144. : row?.Get<LeaveRequest, int>(x => x.OpenForms) != 0
  145. ? _openforms
  146. : row?.Get<LeaveRequest, int>(x => x.FormCount) != 0
  147. ? _completedforms
  148. : null;
  149. }
  150. private bool FormAction(CoreRow? row)
  151. {
  152. if (row is null)
  153. return false;
  154. var leaveRequestID = row.Get<LeaveRequest, Guid>(x => x.ID);
  155. var menu = new ContextMenu();
  156. DynamicGridUtils.PopulateFormMenu<LeaveRequestForm, LeaveRequest, LeaveRequestLink>(menu, leaveRequestID, () => row.ToObject<LeaveRequest>());
  157. menu.IsOpen = true;
  158. return false;
  159. }
  160. private FrameworkElement? FormToolTip(DynamicActionColumn column, CoreRow? row)
  161. {
  162. if (row == null)
  163. return column.TextToolTip("Leave Request Form Status?");
  164. int open = row.Get<LeaveRequest, int>(x => x.OpenForms);
  165. int total = row.Get<LeaveRequest, int>(x => x.FormCount);
  166. if (open != 0)
  167. return column.TextToolTip($"{open} incomplete forms");
  168. if (total != 0)
  169. return column.TextToolTip($"{total} forms");
  170. return null;
  171. }
  172. protected override void DoReconfigure(DynamicGridOptions options)
  173. {
  174. base.DoReconfigure(options);
  175. options.RecordCount = true;
  176. options.FilterRows = true;
  177. options.SelectColumns = true;
  178. }
  179. protected override Dictionary<string, object?> EditorValueChanged(IDynamicEditorForm editor, LeaveRequest[] items, string name, object value)
  180. {
  181. var result = base.EditorValueChanged(editor, items, name, value);
  182. if (name == "LeaveType.ID")
  183. {
  184. ReloadForms<LeaveRequest, LeaveRequestForm, ActivityForm>(editor, items.FirstOrDefault(), x => x.Activity.ID,
  185. value != null ? (Guid)value : Guid.Empty);
  186. }
  187. else if (name.Equals("EmployeeLink.ID"))
  188. {
  189. var activity = editor.FindEditor("LeaveType.ID") as ILookupEditorControl;
  190. if (activity != null)
  191. DefineLookups(activity, items);
  192. }
  193. return result;
  194. }
  195. protected override void AfterLoad(IDynamicEditorForm editor, LeaveRequest[] items)
  196. {
  197. base.AfterLoad(editor, items);
  198. if (items.First().ID == Guid.Empty)
  199. ReloadForms<LeaveRequest, LeaveRequestForm, ActivityForm>(editor, items.First(), x => x.Activity.ID,
  200. items.First().LeaveType.ID);
  201. }
  202. protected override BaseEditor? GetEditor(object item, DynamicGridColumn column)
  203. {
  204. var result = base.GetEditor(item, column);
  205. if (result == null)
  206. return null;
  207. if (!Security.IsAllowed<CanApproveLeaveRequests>() && (column.ColumnName.Equals("Approved") || column.ColumnName.Equals("Status")))
  208. {
  209. result = result.CloneEditor();
  210. result.Editable = Editable.Disabled;
  211. }
  212. return result;
  213. }
  214. }
  215. }