123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626 |
- using System;
- using System.Collections.Generic;
- using System.ComponentModel;
- using System.Data;
- using System.Globalization;
- using System.Linq;
- using System.Linq.Expressions;
- using System.Windows;
- using System.Windows.Controls;
- using System.Windows.Data;
- using System.Windows.Input;
- using System.Windows.Media;
- using Comal.Classes;
- using InABox.Clients;
- using InABox.Configuration;
- using InABox.Core;
- using InABox.DynamicGrid;
- using InABox.WPF;
- using net.sf.mpxj.mspdi.schema;
- using PRSDesktop.WidgetGroups;
- using Syncfusion.Data.Extensions;
- using Syncfusion.UI.Xaml.Grid;
- using Syncfusion.UI.Xaml.Grid.Helpers;
- using Syncfusion.Windows.Tools.Controls;
- using SelectionChangedEventArgs = System.Windows.Controls.SelectionChangedEventArgs;
- namespace PRSDesktop
- {
- public class EmployeeResourcePlannerValue
- {
- public Guid ID { get; set; }
-
- public Brush Background { get; set; }
- public Brush Foreground { get; set; }
- public String Text { get; set; }
- private String _color = "";
- public String Color
- {
- get { return _color; }
- set
- {
- _color = value;
- var color = String.IsNullOrWhiteSpace(value) ? Colors.Transparent : (Color)ColorConverter.ConvertFromString(value);
- Background = new SolidColorBrush(color);
- Foreground = new SolidColorBrush(ImageUtils.GetForegroundColor(color));
- }
- }
- }
- public partial class EmployeeResourcePlanner : UserControl
- {
- private enum Suppress
- {
- This
- }
-
- private EmployeeResourceModel[] _employees = new EmployeeResourceModel[] { };
- private StandardLeaveModel[] _standardleaves = new StandardLeaveModel[] { };
- private LeaveRequestModel[] _leaverequests = new LeaveRequestModel[] { };
- private JobModel[] _jobs = new JobModel[] { };
- private ActivityModel[] _activities = new ActivityModel[] { };
- private CoreFilterDefinitions _jobfilters = new CoreFilterDefinitions();
-
- public event LoadSettings<EmployeeResourcePlannerProperties> LoadSettings;
- public event SaveSettings<EmployeeResourcePlannerProperties> SaveSettings;
- private void DoLoadSettings()
- {
- Properties = LoadSettings?.Invoke(this) ?? new EmployeeResourcePlannerProperties();
- _jobfilters = new GlobalConfiguration<CoreFilterDefinitions>("Job").Load();
- }
-
- private void DoSaveSettings()
- {
- SaveSettings?.Invoke(this, Properties);
- }
-
- public EmployeeResourcePlannerProperties Properties { get; set; }
- public EmployeeResourcePlanner()
- {
- using (new EventSuppressor(Suppress.This))
- InitializeComponent();
- }
- private Filter<Job>? GetJobFilter()
- {
- var jobfilter = _jobfilters.FirstOrDefault(x => String.Equals(x.Name, Properties.JobFilter));
- return !String.IsNullOrWhiteSpace(jobfilter?.Filter)
- ? Serialization.Deserialize<Filter<Job>>(jobfilter.Filter)
- : LookupFactory.DefineFilter<Job>();
- }
-
- public void Setup()
- {
- using (new EventSuppressor(Suppress.This))
- {
- DoLoadSettings();
- EmployeeSelector.Setup();
- EmployeeSelector.Settings = Properties.EmployeeSettings;
- EmployeeSelector.Selection = Properties.EmployeeSelection;
- LeaveType.SelectedIndex = Properties.IncludeUnApprovedLeave ? 1 : 0;
- FromDate.DateTime = DateTime.Today;
- ToDate.DateTime = DateTime.Today.AddYears(1);
-
- MultiQuery query = new MultiQuery();
-
- query.Add<Job>(
- GetJobFilter(),
- JobModel.Columns,
- new SortOrder<Job>(x => x.JobNumber)
- );
-
- query.Add<StandardLeave>(
- null,
- StandardLeaveModel.Columns
- );
-
- query.Add<LeaveRequest>(
- new Filter<LeaveRequest>(x => x.Status).IsNotEqualTo(LeaveRequestStatus.Rejected),
- LeaveRequestModel.Columns
- );
- query.Add<Activity>(
- LookupFactory.DefineFilter<Activity>(),
- new Columns<Activity>(x => x.ID).Add(x => x.Code).Add(x => x.Description),
- new SortOrder<Activity>(x => x.Code)
- );
- query.Query();
- _jobs = query.Get<Job>().Rows.Select(r => new JobModel(r)).ToArray();
- _standardleaves = query.Get<StandardLeave>().Rows.Select(r=>new StandardLeaveModel(r)).ToArray();
- _leaverequests = query.Get<LeaveRequest>().Rows.Select(r => new LeaveRequestModel(r)).ToArray();
- _activities = query.Get<Activity>().Rows.Select(r => new ActivityModel(r)).ToArray();
- ActivityType.ItemsSource = _activities;
- ActivityType.SelectedValue = Properties.ActivityType;
-
- JobFilter.ItemsSource = _jobfilters;
- JobFilter.SelectedValue = _jobfilters.FirstOrDefault(x => String.Equals(x.Name, Properties.JobFilter));
- }
- }
-
- public void Shutdown(CancelEventArgs? cancel)
- {
- }
- public void Refresh()
- {
- using (new WaitCursor())
- {
- _employees = EmployeeSelector.GetEmployeeData((row, rosters) => new EmployeeResourceModel(row, rosters));
- var empids = _employees.Select(x => x.ID).ToArray();
- DateTime fromdate = FromDate.DateTime.HasValue ? FromDate.DateTime.Value.Date : DateTime.Today;
- DateTime todate = ToDate.DateTime.HasValue ? ToDate.DateTime.Value.Date : DateTime.Today.AddYears(1);
- MultiQuery query = new MultiQuery();
-
- query.Add<Assignment>(
- new Filter<Assignment>(x => x.EmployeeLink.ID).InList(empids)
- .And(x => x.Date).IsGreaterThanOrEqualTo(fromdate)
- .And(x => x.Date).IsLessThanOrEqualTo(todate),
- AssignmentModel.Columns,
- new SortOrder<Assignment>(x => x.EmployeeLink.ID).ThenBy(x => x.Date).ThenBy(x => x.Booked.Duration, SortDirection.Descending)
- );
- query.Query();
- var assignments = query.Get<Assignment>().Rows.Select(r => new AssignmentModel(r)).ToArray();
- var data = new DataTable();
- data.Columns.Add("Date", typeof(DateTime));
- foreach (var employee in _employees)
- data.Columns.Add(employee.ID.ToString(), typeof(object));
- for (var curdate = fromdate; curdate <= todate; curdate = curdate.AddDays(1))
- {
- var leavevalue = GetStandardLeave(curdate, _standardleaves);
- var values = new List<object> { curdate };
- foreach (var employee in _employees)
- {
- var value = new EmployeeResourcePlannerValue();
- var bOK = CheckAssignments(employee, curdate, assignments, value);
- bOK = bOK || CheckRoster(employee, curdate, value);
- bOK = bOK || CheckStandardLeave(leavevalue, value);
- bOK = bOK || CheckLeaveRequest(employee, curdate, _leaverequests, value);
- values.Add(value);
- }
- data.Rows.Add(values.ToArray());
- }
- dataGrid.ItemsSource = data;
- }
- }
-
- private bool CheckAssignments(EmployeeResourceModel employee, DateTime curdate, AssignmentModel[] assignments, EmployeeResourcePlannerValue value)
- {
- var assignment = assignments.FirstOrDefault(x => (x.EmployeeID == employee.ID) && (x.Date == curdate.Date));
- if (assignment == null)
- return false;
- value.ID = assignment.ID;
- value.Text = (value.ID != Guid.Empty) ? assignment.JobNumber : "XX";
- value.Color = assignment.Color;
- return true;
- }
- private bool CheckRoster(EmployeeResourceModel employee, DateTime curdate, EmployeeResourcePlannerValue value)
- {
- value.Text = "";
- var roster = RosterUtils.GetRoster(employee.Roster, employee.Start, curdate);
- var color1 = Colors.LightGray;
- var color2 = Colors.LightGray;
- if (roster?.Enabled == true)
- {
- color1 = Colors.LightYellow;
- if (roster.Duration >= 5.0F)
- color2 = Colors.LightYellow;
- }
- value.Foreground = new SolidColorBrush(ImageUtils.GetForegroundColor(ImageUtils.MixColors(color1, 0.5, color2))) { Opacity = 0.8 };
- value.Background = new LinearGradientBrush(color1,color2,90.0F) { Opacity = 0.8 };
- return roster?.Enabled == false;
- }
-
- private EmployeeResourcePlannerValue? GetStandardLeave(DateTime curdate, StandardLeaveModel[] standardleaves)
- {
- var standardleave = standardleaves.FirstOrDefault(x =>
- (x.From <= curdate)
- && (x.To.Add(x.ToTime) > curdate)
- );
- return (standardleave != null)
- ? new EmployeeResourcePlannerValue() { Text = standardleave.Code, Color = standardleave.Color}
- : null;
- }
- private bool CheckStandardLeave(EmployeeResourcePlannerValue? leavevalue, EmployeeResourcePlannerValue value)
- {
- if (leavevalue == null)
- return false;
- value.Text = leavevalue.Text;
- value.Color = leavevalue.Color;
- return true;
- }
-
- private bool CheckLeaveRequest(EmployeeResourceModel employee, DateTime curdate, LeaveRequestModel[] leaverequests, EmployeeResourcePlannerValue value)
- {
- var leaverequest = leaverequests.FirstOrDefault(c =>
- (c.EmployeeID == employee.ID)
- && (c.From <= curdate)
- && (c.To.Add(c.ToTime) > curdate)
- && (Properties.IncludeUnApprovedLeave ? true : c.Status == LeaveRequestStatus.Approved));
- if (leaverequest == null)
- return false;
- value.Text = leaverequest.Code;
- value.Color = (leaverequest.Status == LeaveRequestStatus.Approved) ? leaverequest.Color : Colors.DimGray.ToString();
- return true;
- }
-
- #region AutoGenerate Columns / Styling
-
- private class EmployeeResourcePlannerBackgroundConverter : IValueConverter
- {
- public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
- {
- if (value is not EmployeeResourcePlannerValue val)
- return DependencyProperty.UnsetValue;
- return val.Background;
- }
- public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
- {
- throw new NotImplementedException();
- }
- }
- private class EmployeeResourcePlannerForegroundConverter : IValueConverter
- {
- public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
- {
- if (value is not EmployeeResourcePlannerValue val)
- return DependencyProperty.UnsetValue;
- return val.Foreground;
- }
- public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
- {
- throw new NotImplementedException();
- }
- }
- private class EmployeeResourcePlannerFontStyleConverter : IValueConverter
- {
- public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
- {
- if (value is not EmployeeResourcePlannerValue val)
- return DependencyProperty.UnsetValue;
- return FontStyles.Normal;
- }
- public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
- {
- throw new NotImplementedException();
- }
- }
- private class EmployeeResourcePlannerFontWeightConverter : IValueConverter
- {
- public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
- {
- if (value is not EmployeeResourcePlannerValue val)
- return DependencyProperty.UnsetValue;
- return FontWeights.Normal;
- }
- public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
- {
- throw new NotImplementedException();
- }
- }
- private class EmployeeResourcePlannerContentConverter : IValueConverter
- {
- public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
- {
- if (value is not EmployeeResourcePlannerValue val)
- return DependencyProperty.UnsetValue;
- return val.Text;
- }
- public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
- {
- throw new NotImplementedException();
- }
- }
-
- private void DataGrid_AutoGeneratingColumn(object? sender, AutoGeneratingColumnArgs e)
- {
- e.Column.TextAlignment = TextAlignment.Center;
- e.Column.HorizontalHeaderContentAlignment = HorizontalAlignment.Center;
- e.Column.ColumnSizer = GridLengthUnitType.None;
- var value = (e.Column.ValueBinding as Binding)!;
- if (value.Path.Path.Equals("Date"))
- {
- e.Column.Width = 80;
- e.Column.HeaderStyle = Resources["DateHeaderStyle"] as Style;
- e.Column.AllowFocus = false;
- }
- else
- {
- var style = new Style(typeof(GridCell));
- style.Setters.Add(new Setter(BackgroundProperty, new Binding(value.Path.Path) { Converter = new EmployeeResourcePlannerBackgroundConverter() }));
- style.Setters.Add(new Setter(ForegroundProperty, new Binding(value.Path.Path) { Converter = new EmployeeResourcePlannerForegroundConverter() }));
- style.Setters.Add(new Setter(FontStyleProperty, new Binding(value.Path.Path) { Converter = new EmployeeResourcePlannerFontStyleConverter() }));
- style.Setters.Add(new Setter(FontWeightProperty, new Binding(value.Path.Path) { Converter = new EmployeeResourcePlannerFontWeightConverter() }));
- e.Column.CellStyle = style;
- e.Column.Width = 55;
- e.Column.HeaderStyle = Resources["ContentHeaderStyle"] as Style;
- e.Column.HeaderText = (Guid.TryParse(value.Path.Path, out Guid id))
- ? _employees.FirstOrDefault(x => String.Equals(x.ID, id))?.Name ?? value.Path.Path
- : value.Path.Path;
- e.Column.DisplayBinding = new Binding { Path = new PropertyPath(e.Column.MappingName), Converter = new EmployeeResourcePlannerContentConverter() };
- //e.Column.ValueBinding = new Binding() { Path = new PropertyPath(e.Column.MappingName), Converter = new LeaveContentConverter() };
- //e.Column.UseBindingValue = true;
- e.Column.AllowFocus = true;
- }
- }
- #endregion
-
- private bool HasData()
- {
- foreach (var cell in dataGrid.GetSelectedCells())
- {
- if (!cell.IsDataRowCell)
- continue;
- var propertyCollection = dataGrid.View.GetPropertyAccessProvider();
- var cellValue = propertyCollection.GetValue(cell.RowData, cell.Column.MappingName);
- if (cellValue is EmployeeResourcePlannerValue val && val.ID != Guid.Empty)
- return true;
- }
- return false;
- }
-
- private bool HasData(GridCellInfo cell)
- {
- if (!cell.IsDataRowCell)
- return false;
- var propertyCollection = dataGrid.View.GetPropertyAccessProvider();
- var cellValue = propertyCollection.GetValue(cell.RowData, cell.Column.MappingName);
- return cellValue is EmployeeResourcePlannerValue val && val.ID != Guid.Empty;
- }
- private void DataGrid_ContextMenuOpening(object sender, ContextMenuEventArgs e)
- {
- var vc = dataGrid.GetVisualContainer();
- var p = Mouse.GetPosition(vc);
- var rci = vc.PointToCellRowColumnIndex(p);
- if (rci.RowIndex < 1 || rci.ColumnIndex < 1)
- {
- e.Handled = true;
- return;
- }
- dataGrid.ContextMenu.Items.Clear();
- var bAssign = !HasData(dataGrid.CurrentCellInfo);
- var bClear = HasData();
- if (bAssign)
- {
- foreach (var job in _jobs)
- {
- var assign = new MenuItem
- {
- Header = job.Name,
- Tag = job
- };
- assign.Click += AssignJobClick;
- dataGrid.ContextMenu.Items.Add(assign);
- }
- }
-
-
- if (bClear && bAssign)
- dataGrid.ContextMenu.Items.Add(new Separator());
-
-
- if (bClear)
- {
- var clear = new MenuItem { Header = "Clear Assignments" };
- clear.Click += ClearJobClick;
- dataGrid.ContextMenu.Items.Add(clear);
- }
-
- }
-
- private void GetSelectionData(out DateTime from, out DateTime to, out Guid[] employees, out Guid[] assignments)
- {
- var emps = new List<Guid>();
- var items = new List<Guid>();
- from = DateTime.MaxValue;
- to = DateTime.MinValue;
- foreach (var cell in dataGrid.GetSelectedCells())
- {
- var binding = (cell.Column.ValueBinding as Binding)!;
- if (Guid.TryParse(binding.Path.Path, out var emp))
- if (!emps.Contains(emp))
- emps.Add(emp);
- var row = (cell.RowData as DataRowView)!;
- var date = (DateTime)row.Row.ItemArray.First()!;
- if (date < from)
- from = date;
- if (date > to)
- to = date;
- Guid itemid = (row[binding.Path.Path] as EmployeeResourcePlannerValue).ID;
- if (itemid != Guid.Empty)
- items.Add(itemid);
- }
- employees = emps.ToArray();
- assignments = items.ToArray();
- }
- private void AssignJobClick(object sender, RoutedEventArgs e)
- {
- JobModel? job = (sender as MenuItem)?.Tag as JobModel;
- if (job == null)
- return;
- GetSelectionData(out var from, out var to, out var ids, out var assignments);
- var updates = new List<Assignment>();
- foreach (var id in ids)
- {
- for (DateTime curdate = from; curdate <= to; curdate = curdate.AddDays(1))
- {
- var employee = _employees.FirstOrDefault(x => x.ID == id);
- if (employee != null)
- {
- bool bAvail =
- (GetStandardLeave(curdate, _standardleaves) == null)
- && !CheckLeaveRequest(employee, curdate, _leaverequests, new EmployeeResourcePlannerValue());
-
- var roster = bAvail ? RosterUtils.GetRoster(employee.Roster, employee.Start, curdate) : null;
- if (roster?.Enabled == true)
- {
- var assign = new Assignment();
- assign.Date = curdate;
- assign.Booked.Start = roster.Start;
- assign.Booked.Finish = roster.Finish;
- assign.JobLink.ID = job.ID;
- assign.EmployeeLink.ID = id;
- assign.ActivityLink.ID = Properties.ActivityType;
- updates.Add(assign);
- }
- if (roster?.SplitShift == true)
- {
- var assign = new Assignment();
- assign.Date = curdate;
- assign.Booked.Start = roster.Start2;
- assign.Booked.Finish = roster.Finish2;
- assign.JobLink.ID = job.ID;
- assign.EmployeeLink.ID = id;
- assign.ActivityLink.ID = Properties.ActivityType;
- updates.Add(assign);
- }
- }
- }
- }
- if (updates.Any())
- {
- using (new WaitCursor())
- {
- new Client<Assignment>().Save(updates, "Assigned from Employee Resource Planner");
- Refresh();
- }
- }
- }
- private void ClearJobClick(object sender, RoutedEventArgs e)
- {
- GetSelectionData(out DateTime from, out DateTime to, out Guid[] ids, out Guid[] assignments);
- if (assignments.Any() && MessageBox.Show("Clear Assignments?", "Confirm", MessageBoxButton.YesNo) == MessageBoxResult.Yes)
- {
- var deletes = assignments.Select(x => new Assignment() { ID = x }).ToArray();
- using (new WaitCursor())
- {
- new Client<Assignment>().Delete(deletes, "Deleted from Employee Resource Planner");
- Refresh();
- }
- }
- }
-
- public void Heartbeat(TimeSpan time)
- {
- }
- private void _employees_OnSettingsChanged(object sender, EmployeeSelectorSettingsChangedArgs args)
- {
- Properties.EmployeeSettings = args.Settings;
- DoSaveSettings();
- }
- private void _employees_OnSelectionChanged(object sender, EmployeeSelectorSelectionChangedArgs args)
- {
- Properties.EmployeeSelection = args.Selection;
- DoSaveSettings();
- Refresh();
- }
- private void DateTimeChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
- {
- if (EventSuppressor.IsSet(Suppress.This))
- return;
- Refresh();
- }
- private void LeaveType_OnSelectionChanged(object sender, SelectionChangedEventArgs e)
- {
- if (EventSuppressor.IsSet(Suppress.This))
- return;
- Properties.IncludeUnApprovedLeave = LeaveType.SelectedIndex > 0;
- DoSaveSettings();
- Refresh();
- }
- private void ActivityType_OnSelectionChanged(object sender, SelectionChangedEventArgs e)
- {
- if (EventSuppressor.IsSet(Suppress.This))
- return;
- Properties.ActivityType = (Guid)(ActivityType.SelectedValue ?? Guid.Empty);
- DoSaveSettings();
- }
- private void JobFilter_OnSelectionChanged(object sender, SelectionChangedEventArgs e)
- {
- if (EventSuppressor.IsSet(Suppress.This))
- return;
- var sel = JobFilter.SelectedValue as CoreFilterDefinition;
- Properties.JobFilter = sel?.Name ?? "";
- using (new WaitCursor())
- {
- DoSaveSettings();
- _jobs = new Client<Job>().Query(
- GetJobFilter(),
- JobModel.Columns,
- new SortOrder<Job>(x => x.JobNumber)
- ).Rows.Select(r => new JobModel(r)).ToArray();
- }
- }
- private void JobFilterButton_Click(object sender, RoutedEventArgs e)
- {
- var window = new DynamicGridFilterEditor(_jobfilters, typeof(Job));
- if (window.ShowDialog() == true)
- {
- new GlobalConfiguration<CoreFilterDefinitions>("Job").Save(_jobfilters);
- JobFilter.SelectedValue = _jobfilters.FirstOrDefault(x => String.Equals(x.Name, Properties.JobFilter));
- }
- }
- }
-
- }
|