TimeSheetLeaveRequestGrid.cs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Threading;
  5. using System.Windows.Media.Imaging;
  6. using Comal.Classes;
  7. using InABox.Clients;
  8. using InABox.Core;
  9. using InABox.DynamicGrid;
  10. using InABox.WPF;
  11. using PRS.Shared;
  12. namespace PRSDesktop
  13. {
  14. public class TimeSheetLeaveRequestGrid : DynamicDataGrid<LeaveRequest>
  15. {
  16. public DateTime From { get; set; }
  17. public DateTime To { get; set; }
  18. public TimeSheetLeaveRequestGrid() : base()
  19. {
  20. ColumnsTag = "TimeSheetLeaveRequest";
  21. ActionColumns.Add(new DynamicImageColumn(SelectedImage, SelectedAction));
  22. HiddenColumns.Add(x => x.EmployeeLink.ID);
  23. HiddenColumns.Add(x => x.LeaveType.ID);
  24. HiddenColumns.Add(x => x.From);
  25. HiddenColumns.Add(x => x.FromTime);
  26. HiddenColumns.Add(x => x.To);
  27. HiddenColumns.Add(x => x.ToTime);
  28. HiddenColumns.Add(x => x.Notes);
  29. }
  30. protected override void DoReconfigure(DynamicGridOptions options)
  31. {
  32. base.DoReconfigure(options);
  33. options.Clear();
  34. options.SelectColumns = true;
  35. }
  36. public override DynamicGridColumns GenerateColumns()
  37. {
  38. var columns = new DynamicGridColumns();
  39. columns.Add<LeaveRequest>(x => x.EmployeeLink.Name, 0, "Employee", "", Alignment.MiddleLeft);
  40. columns.Add<LeaveRequest>(x => x.LeaveType.Description, 200, "Type", "", Alignment.MiddleCenter);
  41. columns.Add<LeaveRequest>(x => x.From, 80, "", "dd MMM yy", Alignment.MiddleCenter);
  42. columns.Add<LeaveRequest>(x => x.To, 80, "", "dd MMM yy", Alignment.MiddleCenter);
  43. return columns;
  44. }
  45. private BitmapImage tick = PRSDesktop.Resources.tick.AsBitmapImage();
  46. public HashSet<Guid> SelectedIDs = new HashSet<Guid>();
  47. private bool SelectedAction(CoreRow? arg)
  48. {
  49. if (arg == null)
  50. {
  51. foreach (var row in Data.Rows)
  52. ToggleRow(row);
  53. }
  54. else
  55. ToggleRow(arg);
  56. return false;
  57. }
  58. private void ToggleRow(CoreRow row)
  59. {
  60. Guid id = row.Get<LeaveRequest, Guid>(x => x.ID);
  61. if (!SelectedIDs.Add(id))
  62. {
  63. SelectedIDs.Remove(id);
  64. }
  65. InvalidateRow(row);
  66. }
  67. private BitmapImage? SelectedImage(CoreRow? arg)
  68. {
  69. return arg == null
  70. ? tick
  71. : SelectedIDs.Contains(arg.Get<LeaveRequest,Guid>(x=>x.ID))
  72. ? tick
  73. : null;
  74. }
  75. protected override void Reload(
  76. Filters<LeaveRequest> criteria, Columns<LeaveRequest> columns, ref SortOrder<LeaveRequest>? sort,
  77. CancellationToken token, Action<CoreTable?, Exception?> action)
  78. {
  79. criteria.Add(new InABox.Core.Filter<LeaveRequest>(x => x.From).IsLessThanOrEqualTo(To)
  80. .And(x => x.To).IsGreaterThanOrEqualTo(From)
  81. .And(x=>x.Status).IsEqualTo(LeaveRequestStatus.Approved)
  82. .And(x =>x.Processed).IsEqualTo(DateTime.MinValue));
  83. base.Reload(criteria, columns, ref sort, token, action);
  84. }
  85. public void Process(IProgress<String> progress)
  86. {
  87. if (SelectedIDs.Count == 0)
  88. return;
  89. var employeeids = Data.ToDictionary<LeaveRequest, Guid, Guid>(x => x.ID, x => x.EmployeeLink.ID)
  90. .Where(x => SelectedIDs.Contains(x.Key))
  91. .Select(x=>x.Value)
  92. .Distinct()
  93. .ToArray();
  94. progress.Report("Loading Leave Requests");
  95. var employees = new Client<Employee>().Query(
  96. new Filter<Employee>(x=>x.ID).InList(employeeids),
  97. Columns.None<Employee>().Add(x => x.ID)
  98. .Add(x=>x.Name)
  99. .Add(x=>x.Roster)
  100. .Add(x=>x.RosterStart)
  101. .Add(x => x.StartDate)
  102. .Add(x => x.FinishDate)
  103. ).Rows.Select(x=>x.ToObject<Employee>()).ToArray();
  104. var rosters = employees.ToDictionary<Employee, Guid, String>(x => x.ID, x => x.Roster);
  105. int i = 0;
  106. List<LeaveRequest> leaves = new List<LeaveRequest>();
  107. foreach (var row in Data.Rows)
  108. {
  109. if (SelectedIDs.Contains(row.Get<LeaveRequest, Guid>(x => x.ID)))
  110. {
  111. var leave = row.ToObject<LeaveRequest>();
  112. var employee = employees.FirstOrDefault(x => x.ID == leave.EmployeeLink.ID);
  113. if (employee != null)
  114. {
  115. var roster = !String.IsNullOrWhiteSpace(rosters[employee.ID])
  116. ? Serialization.Deserialize<List<EmployeeRosterItem>>(rosters[employee.ID])?.OrderBy(x => x.Day).ToArray()
  117. : null;
  118. progress.Report($"Processing Leave Request {((double)i * 100.0F / (double)SelectedIDs.Count):F2}");
  119. var timesheets = RosterUtils.CreateLeaveTimesheets(
  120. employee,
  121. roster,
  122. leave.From,
  123. leave.FromTime,
  124. leave.To,
  125. leave.ToTime,
  126. leave.LeaveType.ID,
  127. (timesheet) =>
  128. {
  129. timesheet.Notes = leave.Notes;
  130. timesheet.LeaveRequestLink.ID = leave.ID;
  131. }
  132. );
  133. new Client<TimeSheet>().Save(timesheets, "Updated from TimeSheet leave Processor");
  134. }
  135. i++;
  136. leave.Processed = DateTime.Now;
  137. leaves.Add(leave);
  138. }
  139. }
  140. if (leaves.Any())
  141. {
  142. progress.Report("Saving Leave Requests");
  143. new Client<LeaveRequest>().Save(leaves, "Updated from TimeSheet leave Processor");
  144. }
  145. }
  146. }
  147. }