EmployeeResourcePlanner.xaml.cs 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Data;
  4. using System.Globalization;
  5. using System.Linq;
  6. using System.Linq.Expressions;
  7. using System.Windows;
  8. using System.Windows.Controls;
  9. using System.Windows.Data;
  10. using System.Windows.Input;
  11. using System.Windows.Media;
  12. using Comal.Classes;
  13. using InABox.Clients;
  14. using InABox.Configuration;
  15. using InABox.Core;
  16. using InABox.DynamicGrid;
  17. using InABox.WPF;
  18. using net.sf.mpxj.mspdi.schema;
  19. using PRSDesktop.WidgetGroups;
  20. using Syncfusion.Data.Extensions;
  21. using Syncfusion.UI.Xaml.Grid;
  22. using Syncfusion.UI.Xaml.Grid.Helpers;
  23. using Syncfusion.Windows.Tools.Controls;
  24. using SelectionChangedEventArgs = System.Windows.Controls.SelectionChangedEventArgs;
  25. namespace PRSDesktop
  26. {
  27. public class EmployeeResourcePlannerValue
  28. {
  29. public Guid ID { get; set; }
  30. public Brush Background { get; set; }
  31. public Brush Foreground { get; set; }
  32. public String Text { get; set; }
  33. private String _color = "";
  34. public String Color
  35. {
  36. get { return _color; }
  37. set
  38. {
  39. _color = value;
  40. var color = String.IsNullOrWhiteSpace(value) ? Colors.Transparent : (Color)ColorConverter.ConvertFromString(value);
  41. Background = new SolidColorBrush(color);
  42. Foreground = new SolidColorBrush(ImageUtils.GetForegroundColor(color));
  43. }
  44. }
  45. }
  46. public partial class EmployeeResourcePlanner : UserControl
  47. {
  48. private enum Suppress
  49. {
  50. This
  51. }
  52. private EmployeeResourceModel[] _employees = new EmployeeResourceModel[] { };
  53. private StandardLeaveModel[] _standardleaves = new StandardLeaveModel[] { };
  54. private LeaveRequestModel[] _leaverequests = new LeaveRequestModel[] { };
  55. private JobModel[] _jobs = new JobModel[] { };
  56. private ActivityModel[] _activities = new ActivityModel[] { };
  57. private DynamicGridFilters _jobfilters = new DynamicGridFilters();
  58. public event LoadSettings<EmployeeResourcePlannerProperties> LoadSettings;
  59. public event SaveSettings<EmployeeResourcePlannerProperties> SaveSettings;
  60. private void DoLoadSettings()
  61. {
  62. Properties = LoadSettings?.Invoke(this) ?? new EmployeeResourcePlannerProperties();
  63. _jobfilters = new GlobalConfiguration<DynamicGridFilters>("Job").Load();
  64. }
  65. private void DoSaveSettings()
  66. {
  67. SaveSettings?.Invoke(this, Properties);
  68. }
  69. public EmployeeResourcePlannerProperties Properties { get; set; }
  70. public EmployeeResourcePlanner()
  71. {
  72. using (new EventSuppressor(Suppress.This))
  73. InitializeComponent();
  74. }
  75. private Filter<Job>? GetJobFilter()
  76. {
  77. var jobfilter = _jobfilters.FirstOrDefault(x => String.Equals(x.Name, Properties.JobFilter));
  78. return !String.IsNullOrWhiteSpace(jobfilter?.Filter)
  79. ? Serialization.Deserialize<Filter<Job>>(jobfilter.Filter)
  80. : LookupFactory.DefineFilter<Job>();
  81. }
  82. public void Setup()
  83. {
  84. using (new EventSuppressor(Suppress.This))
  85. {
  86. DoLoadSettings();
  87. EmployeeSelector.Setup();
  88. EmployeeSelector.Settings = Properties.EmployeeSettings;
  89. EmployeeSelector.Selection = Properties.EmployeeSelection;
  90. LeaveType.SelectedIndex = Properties.IncludeUnApprovedLeave ? 1 : 0;
  91. FromDate.DateTime = DateTime.Today;
  92. ToDate.DateTime = DateTime.Today.AddYears(1);
  93. MultiQuery query = new MultiQuery();
  94. query.Add<Job>(
  95. GetJobFilter(),
  96. JobModel.Columns,
  97. new SortOrder<Job>(x => x.JobNumber)
  98. );
  99. query.Add<StandardLeave>(
  100. null,
  101. StandardLeaveModel.Columns
  102. );
  103. query.Add<LeaveRequest>(
  104. new Filter<LeaveRequest>(x => x.Status).IsNotEqualTo(LeaveRequestStatus.Rejected),
  105. LeaveRequestModel.Columns
  106. );
  107. query.Add<Activity>(
  108. LookupFactory.DefineFilter<Activity>(),
  109. new Columns<Activity>(x => x.ID).Add(x => x.Code).Add(x => x.Description),
  110. new SortOrder<Activity>(x => x.Code)
  111. );
  112. query.Query();
  113. _jobs = query.Get<Job>().Rows.Select(r => new JobModel(r)).ToArray();
  114. _standardleaves = query.Get<StandardLeave>().Rows.Select(r=>new StandardLeaveModel(r)).ToArray();
  115. _leaverequests = query.Get<LeaveRequest>().Rows.Select(r => new LeaveRequestModel(r)).ToArray();
  116. _activities = query.Get<Activity>().Rows.Select(r => new ActivityModel(r)).ToArray();
  117. ActivityType.ItemsSource = _activities;
  118. ActivityType.SelectedValue = Properties.ActivityType;
  119. JobFilter.ItemsSource = _jobfilters;
  120. JobFilter.SelectedValue = _jobfilters.FirstOrDefault(x => String.Equals(x.Name, Properties.JobFilter));
  121. }
  122. }
  123. public void Shutdown()
  124. {
  125. }
  126. public void Refresh()
  127. {
  128. using (new WaitCursor())
  129. {
  130. _employees = EmployeeSelector.GetEmployeeData((row) => new EmployeeResourceModel(row));
  131. var empids = _employees.Select(x => x.ID).ToArray();
  132. DateTime fromdate = FromDate.DateTime.HasValue ? FromDate.DateTime.Value.Date : DateTime.Today;
  133. DateTime todate = ToDate.DateTime.HasValue ? ToDate.DateTime.Value.Date : DateTime.Today.AddYears(1);
  134. MultiQuery query = new MultiQuery();
  135. query.Add<Assignment>(
  136. new Filter<Assignment>(x => x.EmployeeLink.ID).InList(empids)
  137. .And(x => x.Date).IsGreaterThanOrEqualTo(fromdate)
  138. .And(x => x.Date).IsLessThanOrEqualTo(todate),
  139. AssignmentModel.Columns,
  140. new SortOrder<Assignment>(x => x.EmployeeLink.ID).ThenBy(x => x.Date).ThenBy(x => x.Booked.Duration, SortDirection.Descending)
  141. );
  142. query.Query();
  143. var assignments = query.Get<Assignment>().Rows.Select(r => new AssignmentModel(r)).ToArray();
  144. var data = new DataTable();
  145. data.Columns.Add("Date", typeof(DateTime));
  146. foreach (var employee in _employees)
  147. data.Columns.Add(employee.ID.ToString(), typeof(object));
  148. for (var curdate = fromdate; curdate <= todate; curdate = curdate.AddDays(1))
  149. {
  150. var leavevalue = GetStandardLeave(curdate, _standardleaves);
  151. var values = new List<object> { curdate };
  152. foreach (var employee in _employees)
  153. {
  154. var value = new EmployeeResourcePlannerValue();
  155. var bOK = CheckAssignments(employee, curdate, assignments, value);
  156. bOK = bOK || CheckRoster(employee, curdate, value);
  157. bOK = bOK || CheckStandardLeave(leavevalue, value);
  158. bOK = bOK || CheckLeaveRequest(employee, curdate, _leaverequests, value);
  159. values.Add(value);
  160. }
  161. data.Rows.Add(values.ToArray());
  162. }
  163. dataGrid.ItemsSource = data;
  164. }
  165. }
  166. private bool CheckAssignments(EmployeeResourceModel employee, DateTime curdate, AssignmentModel[] assignments, EmployeeResourcePlannerValue value)
  167. {
  168. var assignment = assignments.FirstOrDefault(x => (x.EmployeeID == employee.ID) && (x.Date == curdate.Date));
  169. if (assignment == null)
  170. return false;
  171. value.ID = assignment.ID;
  172. value.Text = (value.ID != Guid.Empty) ? assignment.JobNumber : "XX";
  173. value.Color = assignment.Color;
  174. return true;
  175. }
  176. private bool CheckRoster(EmployeeResourceModel employee, DateTime curdate, EmployeeResourcePlannerValue value)
  177. {
  178. value.Text = "";
  179. var roster = RosterUtils.GetRoster(employee.Roster, employee.Start, curdate);
  180. var color1 = Colors.LightGray;
  181. var color2 = Colors.LightGray;
  182. if (roster?.Enabled == true)
  183. {
  184. color1 = Colors.LightYellow;
  185. if (roster.Duration >= 5.0F)
  186. color2 = Colors.LightYellow;
  187. }
  188. value.Foreground = new SolidColorBrush(ImageUtils.GetForegroundColor(ImageUtils.MixColors(color1, 0.5, color2))) { Opacity = 0.8 };
  189. value.Background = new LinearGradientBrush(color1,color2,90.0F) { Opacity = 0.8 };
  190. return roster?.Enabled == false;
  191. }
  192. private EmployeeResourcePlannerValue? GetStandardLeave(DateTime curdate, StandardLeaveModel[] standardleaves)
  193. {
  194. var standardleave = standardleaves.FirstOrDefault(x =>
  195. (x.From <= curdate)
  196. && (x.To.Add(x.ToTime) > curdate)
  197. );
  198. return (standardleave != null)
  199. ? new EmployeeResourcePlannerValue() { Text = standardleave.Code, Color = standardleave.Color}
  200. : null;
  201. }
  202. private bool CheckStandardLeave(EmployeeResourcePlannerValue? leavevalue, EmployeeResourcePlannerValue value)
  203. {
  204. if (leavevalue == null)
  205. return false;
  206. value.Text = leavevalue.Text;
  207. value.Color = leavevalue.Color;
  208. return true;
  209. }
  210. private bool CheckLeaveRequest(EmployeeResourceModel employee, DateTime curdate, LeaveRequestModel[] leaverequests, EmployeeResourcePlannerValue value)
  211. {
  212. var leaverequest = leaverequests.FirstOrDefault(c =>
  213. (c.EmployeeID == employee.ID)
  214. && (c.From <= curdate)
  215. && (c.To.Add(c.ToTime) > curdate)
  216. && (Properties.IncludeUnApprovedLeave ? true : c.Status == LeaveRequestStatus.Approved));
  217. if (leaverequest == null)
  218. return false;
  219. value.Text = leaverequest.Code;
  220. value.Color = (leaverequest.Status == LeaveRequestStatus.Approved) ? leaverequest.Color : Colors.DimGray.ToString();
  221. return true;
  222. }
  223. #region AutoGenerate Columns / Styling
  224. private class EmployeeResourcePlannerBackgroundConverter : IValueConverter
  225. {
  226. public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
  227. {
  228. if (value is not EmployeeResourcePlannerValue val)
  229. return DependencyProperty.UnsetValue;
  230. return val.Background;
  231. }
  232. public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
  233. {
  234. throw new NotImplementedException();
  235. }
  236. }
  237. private class EmployeeResourcePlannerForegroundConverter : IValueConverter
  238. {
  239. public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
  240. {
  241. if (value is not EmployeeResourcePlannerValue val)
  242. return DependencyProperty.UnsetValue;
  243. return val.Foreground;
  244. }
  245. public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
  246. {
  247. throw new NotImplementedException();
  248. }
  249. }
  250. private class EmployeeResourcePlannerFontStyleConverter : IValueConverter
  251. {
  252. public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
  253. {
  254. if (value is not EmployeeResourcePlannerValue val)
  255. return DependencyProperty.UnsetValue;
  256. return FontStyles.Normal;
  257. }
  258. public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
  259. {
  260. throw new NotImplementedException();
  261. }
  262. }
  263. private class EmployeeResourcePlannerFontWeightConverter : IValueConverter
  264. {
  265. public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
  266. {
  267. if (value is not EmployeeResourcePlannerValue val)
  268. return DependencyProperty.UnsetValue;
  269. return FontWeights.Normal;
  270. }
  271. public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
  272. {
  273. throw new NotImplementedException();
  274. }
  275. }
  276. private class EmployeeResourcePlannerContentConverter : IValueConverter
  277. {
  278. public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
  279. {
  280. if (value is not EmployeeResourcePlannerValue val)
  281. return DependencyProperty.UnsetValue;
  282. return val.Text;
  283. }
  284. public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
  285. {
  286. throw new NotImplementedException();
  287. }
  288. }
  289. private void DataGrid_AutoGeneratingColumn(object? sender, AutoGeneratingColumnArgs e)
  290. {
  291. e.Column.TextAlignment = TextAlignment.Center;
  292. e.Column.HorizontalHeaderContentAlignment = HorizontalAlignment.Center;
  293. e.Column.ColumnSizer = GridLengthUnitType.None;
  294. var value = (e.Column.ValueBinding as Binding)!;
  295. if (value.Path.Path.Equals("Date"))
  296. {
  297. e.Column.Width = 80;
  298. e.Column.HeaderStyle = Resources["DateHeaderStyle"] as Style;
  299. e.Column.AllowFocus = false;
  300. }
  301. else
  302. {
  303. var style = new Style(typeof(GridCell));
  304. style.Setters.Add(new Setter(BackgroundProperty, new Binding(value.Path.Path) { Converter = new EmployeeResourcePlannerBackgroundConverter() }));
  305. style.Setters.Add(new Setter(ForegroundProperty, new Binding(value.Path.Path) { Converter = new EmployeeResourcePlannerForegroundConverter() }));
  306. style.Setters.Add(new Setter(FontStyleProperty, new Binding(value.Path.Path) { Converter = new EmployeeResourcePlannerFontStyleConverter() }));
  307. style.Setters.Add(new Setter(FontWeightProperty, new Binding(value.Path.Path) { Converter = new EmployeeResourcePlannerFontWeightConverter() }));
  308. e.Column.CellStyle = style;
  309. e.Column.Width = 55;
  310. e.Column.HeaderStyle = Resources["ContentHeaderStyle"] as Style;
  311. e.Column.HeaderText = (Guid.TryParse(value.Path.Path, out Guid id))
  312. ? _employees.FirstOrDefault(x => String.Equals(x.ID, id))?.Name ?? value.Path.Path
  313. : value.Path.Path;
  314. e.Column.DisplayBinding = new Binding { Path = new PropertyPath(e.Column.MappingName), Converter = new EmployeeResourcePlannerContentConverter() };
  315. //e.Column.ValueBinding = new Binding() { Path = new PropertyPath(e.Column.MappingName), Converter = new LeaveContentConverter() };
  316. //e.Column.UseBindingValue = true;
  317. e.Column.AllowFocus = true;
  318. }
  319. }
  320. #endregion
  321. private bool HasData()
  322. {
  323. foreach (var cell in dataGrid.GetSelectedCells())
  324. {
  325. if (!cell.IsDataRowCell)
  326. continue;
  327. var propertyCollection = dataGrid.View.GetPropertyAccessProvider();
  328. var cellValue = propertyCollection.GetValue(cell.RowData, cell.Column.MappingName);
  329. if (cellValue is EmployeeResourcePlannerValue val && val.ID != Guid.Empty)
  330. return true;
  331. }
  332. return false;
  333. }
  334. private bool HasData(GridCellInfo cell)
  335. {
  336. if (!cell.IsDataRowCell)
  337. return false;
  338. var propertyCollection = dataGrid.View.GetPropertyAccessProvider();
  339. var cellValue = propertyCollection.GetValue(cell.RowData, cell.Column.MappingName);
  340. return cellValue is EmployeeResourcePlannerValue val && val.ID != Guid.Empty;
  341. }
  342. private void DataGrid_ContextMenuOpening(object sender, ContextMenuEventArgs e)
  343. {
  344. var vc = dataGrid.GetVisualContainer();
  345. var p = Mouse.GetPosition(vc);
  346. var rci = vc.PointToCellRowColumnIndex(p);
  347. if (rci.RowIndex < 1 || rci.ColumnIndex < 1)
  348. {
  349. e.Handled = true;
  350. return;
  351. }
  352. dataGrid.ContextMenu.Items.Clear();
  353. var bAssign = !HasData(dataGrid.CurrentCellInfo);
  354. var bClear = HasData();
  355. if (bAssign)
  356. {
  357. foreach (var job in _jobs)
  358. {
  359. var assign = new MenuItem
  360. {
  361. Header = job.Name,
  362. Tag = job
  363. };
  364. assign.Click += AssignJobClick;
  365. dataGrid.ContextMenu.Items.Add(assign);
  366. }
  367. }
  368. if (bClear && bAssign)
  369. dataGrid.ContextMenu.Items.Add(new Separator());
  370. if (bClear)
  371. {
  372. var clear = new MenuItem { Header = "Clear Assignments" };
  373. clear.Click += ClearJobClick;
  374. dataGrid.ContextMenu.Items.Add(clear);
  375. }
  376. }
  377. private void GetSelectionData(out DateTime from, out DateTime to, out Guid[] employees, out Guid[] assignments)
  378. {
  379. var emps = new List<Guid>();
  380. var items = new List<Guid>();
  381. from = DateTime.MaxValue;
  382. to = DateTime.MinValue;
  383. foreach (var cell in dataGrid.GetSelectedCells())
  384. {
  385. var binding = (cell.Column.ValueBinding as Binding)!;
  386. if (Guid.TryParse(binding.Path.Path, out var emp))
  387. if (!emps.Contains(emp))
  388. emps.Add(emp);
  389. var row = (cell.RowData as DataRowView)!;
  390. var date = (DateTime)row.Row.ItemArray.First()!;
  391. if (date < from)
  392. from = date;
  393. if (date > to)
  394. to = date;
  395. Guid itemid = (row[binding.Path.Path] as EmployeeResourcePlannerValue).ID;
  396. if (itemid != Guid.Empty)
  397. items.Add(itemid);
  398. }
  399. employees = emps.ToArray();
  400. assignments = items.ToArray();
  401. }
  402. private void AssignJobClick(object sender, RoutedEventArgs e)
  403. {
  404. JobModel? job = (sender as MenuItem)?.Tag as JobModel;
  405. if (job == null)
  406. return;
  407. GetSelectionData(out var from, out var to, out var ids, out var assignments);
  408. var updates = new List<Assignment>();
  409. foreach (var id in ids)
  410. {
  411. for (DateTime curdate = from; curdate <= to; curdate = curdate.AddDays(1))
  412. {
  413. var employee = _employees.FirstOrDefault(x => x.ID == id);
  414. if (employee != null)
  415. {
  416. bool bAvail =
  417. (GetStandardLeave(curdate, _standardleaves) == null)
  418. && !CheckLeaveRequest(employee, curdate, _leaverequests, new EmployeeResourcePlannerValue());
  419. var roster = bAvail ? RosterUtils.GetRoster(employee.Roster, employee.Start, curdate) : null;
  420. if (roster?.Enabled == true)
  421. {
  422. var assign = new Assignment();
  423. assign.Date = curdate;
  424. assign.Booked.Start = roster.Start;
  425. assign.Booked.Finish = roster.Finish;
  426. assign.JobLink.ID = job.ID;
  427. assign.EmployeeLink.ID = id;
  428. assign.ActivityLink.ID = Properties.ActivityType;
  429. updates.Add(assign);
  430. }
  431. if (roster?.SplitShift == true)
  432. {
  433. var assign = new Assignment();
  434. assign.Date = curdate;
  435. assign.Booked.Start = roster.Start2;
  436. assign.Booked.Finish = roster.Finish2;
  437. assign.JobLink.ID = job.ID;
  438. assign.EmployeeLink.ID = id;
  439. assign.ActivityLink.ID = Properties.ActivityType;
  440. updates.Add(assign);
  441. }
  442. }
  443. }
  444. }
  445. if (updates.Any())
  446. {
  447. using (new WaitCursor())
  448. {
  449. new Client<Assignment>().Save(updates, "Assigned from Employee Resource Planner");
  450. Refresh();
  451. }
  452. }
  453. }
  454. private void ClearJobClick(object sender, RoutedEventArgs e)
  455. {
  456. GetSelectionData(out DateTime from, out DateTime to, out Guid[] ids, out Guid[] assignments);
  457. if (assignments.Any() && MessageBox.Show("Clear Assignments?", "Confirm", MessageBoxButton.YesNo) == MessageBoxResult.Yes)
  458. {
  459. var deletes = assignments.Select(x => new Assignment() { ID = x }).ToArray();
  460. using (new WaitCursor())
  461. {
  462. new Client<Assignment>().Delete(deletes, "Deleted from Employee Resource Planner");
  463. Refresh();
  464. }
  465. }
  466. }
  467. public void Heartbeat(TimeSpan time)
  468. {
  469. }
  470. private void _employees_OnSettingsChanged(object sender, EmployeeSelectorSettingsChangedArgs args)
  471. {
  472. Properties.EmployeeSettings = args.Settings;
  473. DoSaveSettings();
  474. }
  475. private void _employees_OnSelectionChanged(object sender, EmployeeSelectorSelectionChangedArgs args)
  476. {
  477. Properties.EmployeeSelection = args.Selection;
  478. DoSaveSettings();
  479. Refresh();
  480. }
  481. private void DateTimeChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
  482. {
  483. if (EventSuppressor.IsSet(Suppress.This))
  484. return;
  485. Refresh();
  486. }
  487. private void LeaveType_OnSelectionChanged(object sender, SelectionChangedEventArgs e)
  488. {
  489. if (EventSuppressor.IsSet(Suppress.This))
  490. return;
  491. Properties.IncludeUnApprovedLeave = LeaveType.SelectedIndex > 0;
  492. DoSaveSettings();
  493. Refresh();
  494. }
  495. private void ActivityType_OnSelectionChanged(object sender, SelectionChangedEventArgs e)
  496. {
  497. if (EventSuppressor.IsSet(Suppress.This))
  498. return;
  499. Properties.ActivityType = (Guid)(ActivityType.SelectedValue ?? Guid.Empty);
  500. DoSaveSettings();
  501. }
  502. private void JobFilter_OnSelectionChanged(object sender, SelectionChangedEventArgs e)
  503. {
  504. if (EventSuppressor.IsSet(Suppress.This))
  505. return;
  506. var sel = JobFilter.SelectedValue as DynamicGridFilter;
  507. Properties.JobFilter = sel?.Name ?? "";
  508. using (new WaitCursor())
  509. {
  510. DoSaveSettings();
  511. _jobs = new Client<Job>().Query(
  512. GetJobFilter(),
  513. JobModel.Columns,
  514. new SortOrder<Job>(x => x.JobNumber)
  515. ).Rows.Select(r => new JobModel(r)).ToArray();
  516. }
  517. }
  518. private void JobFilterButton_Click(object sender, RoutedEventArgs e)
  519. {
  520. var window = new DynamicGridFilterEditor(_jobfilters, typeof(Job));
  521. if (window.ShowDialog() == true)
  522. {
  523. new GlobalConfiguration<DynamicGridFilters>("Job").Save(_jobfilters);
  524. JobFilter.SelectedValue = _jobfilters.FirstOrDefault(x => String.Equals(x.Name, Properties.JobFilter));
  525. }
  526. }
  527. }
  528. }