TasksByUserControl.xaml.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875
  1. using System;
  2. using System.Collections.Generic;
  3. using System.ComponentModel;
  4. using System.Drawing;
  5. using System.Globalization;
  6. using System.Linq;
  7. using System.Reflection;
  8. using System.Runtime.CompilerServices;
  9. using System.Windows;
  10. using System.Windows.Controls;
  11. using System.Windows.Data;
  12. using System.Windows.Input;
  13. using System.Windows.Media.Imaging;
  14. using AvalonDock.Layout;
  15. using com.sun.org.apache.bcel.@internal.generic;
  16. using Comal.Classes;
  17. using InABox.Clients;
  18. using InABox.Core;
  19. using InABox.DynamicGrid;
  20. using InABox.Wpf;
  21. using InABox.WPF;
  22. using Syncfusion.UI.Xaml.Kanban;
  23. using Syncfusion.Windows.Tools.Controls;
  24. using static com.sun.tools.javac.code.Symbol;
  25. namespace PRSDesktop;
  26. public class TasksByUserEmployeeHeader : INotifyPropertyChanged
  27. {
  28. public Guid EmployeeID { get; set; }
  29. public string Name { get; set; }
  30. public BitmapImage Image { get; set; }
  31. public int NumTasks { get => Tasks.Count(); }
  32. public double NumHours { get => Tasks.Sum(x => x.EstimatedTime.TotalHours); }
  33. public IEnumerable<TaskModel> Tasks => Model.Categories
  34. .SelectMany(x => x.EmployeeCategoryDictionary.GetValueOrDefault(EmployeeID)?.Tasks ?? Enumerable.Empty<TaskModel>());
  35. private TasksByUserModel Model;
  36. public event PropertyChangedEventHandler? PropertyChanged;
  37. // Create the OnPropertyChanged method to raise the event
  38. // The calling member's name will be used as the parameter.
  39. protected void OnPropertyChanged([CallerMemberName] string? name = null)
  40. {
  41. PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(name));
  42. if (name.Equals(nameof(Tasks)))
  43. {
  44. OnPropertyChanged(nameof(NumTasks));
  45. OnPropertyChanged(nameof(NumHours));
  46. }
  47. }
  48. public TasksByUserEmployeeHeader(Guid employeeID, string name, BitmapImage image, TasksByUserModel model)
  49. {
  50. EmployeeID = employeeID;
  51. Name = name;
  52. Image = image;
  53. Model = model;
  54. }
  55. public void UpdateTasks()
  56. {
  57. OnPropertyChanged(nameof(Tasks));
  58. }
  59. }
  60. public class TasksByUserEmployeeCategory
  61. {
  62. public Guid EmployeeID { get; set; }
  63. public KanbanStatus Status { get; set; }
  64. public SuspendableObservableCollection<TaskModel> Tasks { get; set; } = new();
  65. public TasksByUserEmployeeCategory(Guid employeeID, KanbanStatus status)
  66. {
  67. EmployeeID = employeeID;
  68. Status = status;
  69. }
  70. }
  71. public class TasksByUserCategory : INotifyPropertyChanged
  72. {
  73. private bool _collapsed;
  74. public bool Collapsed
  75. {
  76. get => _collapsed;
  77. set
  78. {
  79. _collapsed = value;
  80. OnPropertyChanged();
  81. }
  82. }
  83. public KanbanStatus Status { get; set; }
  84. public IEnumerable<TasksByUserEmployeeCategory> EmployeeCategories => EmployeeCategoryDictionary.Values;
  85. public Dictionary<Guid, TasksByUserEmployeeCategory> EmployeeCategoryDictionary { get; set; } = new();
  86. public TasksByUserCategory(KanbanStatus status, bool collapsed)
  87. {
  88. Status = status;
  89. Collapsed = collapsed;
  90. }
  91. public event PropertyChangedEventHandler? PropertyChanged;
  92. // Create the OnPropertyChanged method to raise the event
  93. // The calling member's name will be used as the parameter.
  94. protected void OnPropertyChanged([CallerMemberName] string? name = null)
  95. {
  96. PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(name));
  97. }
  98. }
  99. public class TasksByUserModel
  100. {
  101. public SuspendableObservableCollection<TasksByUserEmployeeHeader> SectionHeaders { get; set; } = new();
  102. public SuspendableObservableCollection<TasksByUserCategory> Categories { get; set; } = new();
  103. }
  104. public partial class TasksByUserControl : UserControl, INotifyPropertyChanged, ITaskControl
  105. {
  106. private static readonly BitmapImage anonymous = PRSDesktop.Resources.anonymous.AsBitmapImage();
  107. public TasksByUserModel Model { get; set; } = new();
  108. private ILookup<Guid, Guid> TeamEmployees;
  109. private Dictionary<Guid, EmployeeModel> Employees;
  110. private bool bPopulating;
  111. private KanbanViewMode _mode;
  112. public KanbanViewMode Mode
  113. {
  114. get => _mode;
  115. set
  116. {
  117. _mode = value;
  118. OnPropertyChanged();
  119. }
  120. }
  121. public TasksByUserControl()
  122. {
  123. InitializeComponent();
  124. }
  125. #region INotifyPropertyChanged
  126. public event PropertyChangedEventHandler? PropertyChanged;
  127. // Create the OnPropertyChanged method to raise the event
  128. // The calling member's name will be used as the parameter.
  129. protected void OnPropertyChanged([CallerMemberName] string? name = null)
  130. {
  131. PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(name));
  132. }
  133. #endregion
  134. #region Setup
  135. private void LoadEmployees()
  136. {
  137. var empfilter = LookupFactory.DefineFilter<Employee>();
  138. var results = Client.QueryMultiple(
  139. new KeyedQueryDef<Employee>(
  140. LookupFactory.DefineFilter<Employee>(),
  141. new Columns<Employee>(x => x.ID)
  142. .Add(x => x.Name)
  143. .Add(x => x.Thumbnail.ID),
  144. new SortOrder<Employee>(x => x.Name)),
  145. new KeyedQueryDef<Team>(
  146. LookupFactory.DefineFilter<Team>(),
  147. new Columns<Team>(x => x.ID)
  148. .Add(x => x.Name),
  149. new SortOrder<Team>(x => x.Name)),
  150. new KeyedQueryDef<EmployeeTeam>(
  151. LookupFactory.DefineFilter<EmployeeTeam>(),
  152. new Columns<EmployeeTeam>(x => x.EmployeeLink.ID)
  153. .Add(x => x.TeamLink.ID)));
  154. TeamEmployees = results.Get<EmployeeTeam>().ToLookup<EmployeeTeam, Guid, Guid>(x => x.TeamLink.ID, x => x.EmployeeLink.ID);
  155. Employees = results.GetObjects<Employee>().ToDictionary(
  156. x => x.ID,
  157. x => new EmployeeModel(x.ID, x.Name, x.Thumbnail.ID, null));
  158. var teams = results.GetObjects<Team>().ToDictionary(x => x.ID, x => x.Name);
  159. SelectedTeams.ItemsSource = teams;
  160. foreach (var team in Host.KanbanSettings.UserSettings.SelectedTeams)
  161. {
  162. SelectedTeams.SelectedItems.Add(teams.FirstOrDefault(x => x.Key == team));
  163. }
  164. }
  165. private void PopulateEmployees()
  166. {
  167. bPopulating = true;
  168. try
  169. {
  170. var availableemployees = new List<Guid>();
  171. foreach (var team in SelectedTeams.SelectedItems.Select(v => (KeyValuePair<Guid, string>)v))
  172. availableemployees.AddRange(TeamEmployees[team.Key].Where(x => !availableemployees.Contains(x)));
  173. SelectedEmployees.ItemsSource = Employees.Where(x => availableemployees.Contains(x.Key));
  174. SelectedEmployees.SelectedItems.Clear();
  175. foreach (var employee in Host.KanbanSettings.UserSettings.SelectedEmployees.Where(availableemployees.Contains))
  176. SelectedEmployees.SelectedItems.Add(Employees.FirstOrDefault(x => Equals(x.Key, employee)));
  177. }
  178. catch (Exception e)
  179. {
  180. }
  181. bPopulating = false;
  182. }
  183. private void SetupToolbar()
  184. {
  185. IncludeCompleted.Visibility = Security.IsAllowed<CanHideTaskCompletedColumn>() ? Visibility.Visible : Visibility.Collapsed;
  186. IncludeCompleted.IsChecked = IncludeCompleted.Visibility == Visibility.Visible ? Host.KanbanSettings.UserSettings.IncludeCompleted : true;
  187. IncludeObserved.IsChecked = Host.KanbanSettings.UserSettings.IncludeObserved;
  188. IncludeManaged.IsChecked = Host.KanbanSettings.UserSettings.IncludeManaged;
  189. ViewType.SelectedIndex = Host.KanbanSettings.UserSettings.CompactView ? 1 : 0;
  190. }
  191. private void PopulateKanbanTypes()
  192. {
  193. TaskType.Items.Add("");
  194. foreach (var kanbanType in Host.KanbanTypes)
  195. {
  196. TaskType.Items.Add(kanbanType);
  197. }
  198. }
  199. public void Setup()
  200. {
  201. SetupToolbar();
  202. SplitPanel.AnchorWidth = Host.KanbanSettings.UserSettings.AnchorWidth;
  203. TeamsRow.Height = new GridLength(Host.KanbanSettings.UserSettings.TeamsHeight);
  204. LoadEmployees();
  205. PopulateEmployees();
  206. Mode = Host.KanbanSettings.UserSettings.CompactView ? KanbanViewMode.Compact : KanbanViewMode.Full;
  207. PopulateKanbanTypes();
  208. }
  209. #endregion
  210. #region Filters
  211. private void TaskPanelFilterButton_OnFilterRefresh()
  212. {
  213. Refresh();
  214. }
  215. private void IncludeCompleted_Checked(object sender, RoutedEventArgs e)
  216. {
  217. if (!IsReady)
  218. return;
  219. SaveSettings();
  220. Refresh();
  221. }
  222. private void IncludeObserved_Checked(object sender, RoutedEventArgs e)
  223. {
  224. if (!IsReady)
  225. return;
  226. SaveSettings();
  227. Refresh();
  228. }
  229. private void IncludeManaged_Checked(object sender, RoutedEventArgs e)
  230. {
  231. if (!IsReady)
  232. return;
  233. SaveSettings();
  234. Refresh();
  235. }
  236. private void TaskType_SelectionChanged(object sender, SelectionChangedEventArgs e)
  237. {
  238. if (!IsReady)
  239. return;
  240. FilterKanbans();
  241. }
  242. private void Search_KeyUp(object sender, KeyEventArgs e)
  243. {
  244. FilterKanbans();
  245. }
  246. #endregion
  247. #region Refresh
  248. private Filter<KanbanSubscriber> GetKanbanSubscriberFilter(Filter<KanbanSubscriber>? additional = null)
  249. {
  250. var filter = new Filter<KanbanSubscriber>(c => c.Kanban.Closed).IsEqualTo(DateTime.MinValue)
  251. .And(x => x.Kanban.Locked).IsEqualTo(false);
  252. var privateFilter = new Filter<KanbanSubscriber>(x => x.Kanban.Private).IsEqualTo(false);
  253. if (App.EmployeeID != Guid.Empty)
  254. {
  255. privateFilter = privateFilter.Or(x => x.Employee.ID).IsEqualTo(App.EmployeeID);
  256. }
  257. filter.And(privateFilter);
  258. if (Host.Job != null)
  259. {
  260. if (Host.Job.ID != Guid.Empty)
  261. filter = filter.And(c => c.Kanban.JobLink.ID).IsEqualTo(Host.Job.ID);
  262. else
  263. filter = filter.And(c => c.Kanban.JobLink.ID).None();
  264. }
  265. if (!Host.KanbanSettings.UserSettings.IncludeCompleted)
  266. filter = filter.And(new Filter<KanbanSubscriber>(x => x.Kanban.Completed).IsEqualTo(DateTime.MinValue));
  267. var emps = Employees.Where(x => Host.KanbanSettings.UserSettings.SelectedEmployees.Contains(x.Key));
  268. filter = filter.And(c => c.Employee.ID).InList(emps.Select(x => x.Key).ToArray());
  269. if (!Host.KanbanSettings.UserSettings.IncludeObserved)
  270. {
  271. if (Host.KanbanSettings.UserSettings.IncludeManaged)
  272. filter = filter.And(new Filter<KanbanSubscriber>(x => x.Manager).IsEqualTo(true).Or(x => x.Assignee).IsEqualTo(true));
  273. else
  274. filter = filter.And(x => x.Assignee).IsEqualTo(true);
  275. }
  276. if (additional is not null)
  277. {
  278. return additional.And(filter);
  279. }
  280. else
  281. {
  282. return filter;
  283. }
  284. }
  285. private void ReloadColumns()
  286. {
  287. Model.SectionHeaders.SupressNotification = true;
  288. Model.SectionHeaders.Clear();
  289. var emps = Host.KanbanSettings.UserSettings.SelectedEmployees.OrderBy(x => Employees[x].Name).ToArray();
  290. foreach (var employeeID in emps)
  291. {
  292. if (Employees.TryGetValue(employeeID, out var employee))
  293. {
  294. Model.SectionHeaders.Add(new TasksByUserEmployeeHeader(employeeID, employee.Name, employee.Image ?? anonymous, Model));
  295. }
  296. }
  297. Model.SectionHeaders.SupressNotification = false;
  298. Model.Categories.SupressNotification = true;
  299. var oldCategories = Model.Categories.ToDictionary(x => x.Status);
  300. Model.Categories.Clear();
  301. foreach (var status in _statusOrder)
  302. {
  303. if (status == KanbanStatus.Complete && !Host.KanbanSettings.UserSettings.IncludeCompleted)
  304. {
  305. continue;
  306. }
  307. var newCategory = new TasksByUserCategory(status, oldCategories.GetValueOrDefault(status)?.Collapsed ?? false);
  308. foreach (var employeeID in emps)
  309. {
  310. if (Employees.TryGetValue(employeeID, out var employee))
  311. {
  312. var cat = new TasksByUserEmployeeCategory(employeeID, status);
  313. newCategory.EmployeeCategoryDictionary[employeeID] = cat;
  314. var header = Model.SectionHeaders.First(x => x.EmployeeID == employeeID);
  315. cat.Tasks.CollectionChanged += (s, e) =>
  316. {
  317. header.UpdateTasks();
  318. };
  319. }
  320. }
  321. Model.Categories.Add(newCategory);
  322. }
  323. Model.Categories.SupressNotification = false;
  324. }
  325. private static readonly KanbanStatus[] _statusOrder = new[]
  326. {
  327. KanbanStatus.Open,
  328. KanbanStatus.InProgress,
  329. KanbanStatus.Waiting,
  330. KanbanStatus.Complete
  331. };
  332. private Columns<Kanban> GetKanbanColumns()
  333. {
  334. return new Columns<Kanban>(
  335. x => x.ID,
  336. x => x.DueDate,
  337. x => x.Completed,
  338. x => x.Description,
  339. x => x.Summary,
  340. x => x.Status,
  341. x => x.EmployeeLink.ID,
  342. x => x.EmployeeLink.Name,
  343. x => x.ManagerLink.ID,
  344. x => x.ManagerLink.Name,
  345. x => x.Notes,
  346. x => x.Title,
  347. x => x.JobLink.ID,
  348. x => x.JobLink.JobNumber,
  349. x => x.JobLink.Name,
  350. x => x.Type.ID,
  351. x => x.Type.Code,
  352. x => x.Number,
  353. x => x.Attachments,
  354. x => x.Locked,
  355. x => x.EstimatedTime);
  356. }
  357. private IEnumerable<KanbanSubscriber> LoadSubscribers(Filter<KanbanSubscriber>? filter = null)
  358. {
  359. filter = GetKanbanSubscriberFilter(filter);
  360. var kanbanFilter = new Filter<Kanban>(x => x.ID).InQuery(filter, x => x.Kanban.ID);
  361. var buttonFilter = FilterButton.GetFilter();
  362. if(buttonFilter is not null)
  363. {
  364. kanbanFilter.And(buttonFilter);
  365. }
  366. var results = Client.QueryMultiple(
  367. new KeyedQueryDef<KanbanSubscriber>(
  368. filter,
  369. new Columns<KanbanSubscriber>(x => x.ID, x => x.Employee.ID, x => x.Kanban.ID),
  370. new SortOrder<KanbanSubscriber>(x => x.Kanban.DueDate) { Direction = SortDirection.Ascending }),
  371. new KeyedQueryDef<Kanban>(
  372. kanbanFilter,
  373. GetKanbanColumns()));
  374. var kanbans = results.GetObjects<Kanban>().ToDictionary(x => x.ID);
  375. return results.GetObjects<KanbanSubscriber>().Select(x =>
  376. {
  377. if (kanbans.TryGetValue(x.Kanban.ID, out var kanban))
  378. {
  379. x.Kanban.Synchronise(kanban);
  380. return x;
  381. }
  382. else
  383. {
  384. return null;
  385. }
  386. }).NotNull();
  387. }
  388. public void Refresh()
  389. {
  390. using (new WaitCursor())
  391. {
  392. var models = CreateModels(LoadSubscribers()).ToList();
  393. ReloadColumns();
  394. AllTasks = models.OrderBy(x => x.DueDate).ToList();
  395. FilterKanbans();
  396. }
  397. }
  398. /// <summary>
  399. /// Take the full list of kanbans loaded from the database, and filter based on the search UI elements, filtering into the columns.
  400. /// </summary>
  401. private void FilterKanbans()
  402. {
  403. IEnumerable<TaskModel> filtered = AllTasks;
  404. if (TaskType.SelectedItem is KanbanType kanbanType)
  405. {
  406. filtered = filtered.Where(x => x.Type.ID == kanbanType.ID);
  407. }
  408. if (!string.IsNullOrWhiteSpace(Search.Text))
  409. {
  410. var searches = Search.Text.Split();
  411. filtered = filtered.Where(x => x.Search(searches));
  412. }
  413. var categoryMap = Model.Categories.ToDictionary(x => x.Status, x => x.EmployeeCategoryDictionary);
  414. foreach(var category in Model.Categories)
  415. {
  416. foreach(var empCat in category.EmployeeCategories)
  417. {
  418. empCat.Tasks.Clear();
  419. }
  420. }
  421. SelectedTasks.Clear();
  422. foreach (var task in filtered)
  423. {
  424. if(categoryMap.TryGetValue(task.Status, out var categoryDict))
  425. {
  426. if (categoryDict.TryGetValue(task.EmployeeCategory, out var employeeCategory))
  427. {
  428. employeeCategory.Tasks.Add(task);
  429. if (task.Checked)
  430. {
  431. SelectedTasks.Add(task);
  432. }
  433. }
  434. }
  435. }
  436. }
  437. private IEnumerable<TaskModel> CreateModels(IEnumerable<KanbanSubscriber> subscribers)
  438. {
  439. foreach(var subscriber in subscribers)
  440. {
  441. var kanban = subscriber.Kanban;
  442. var model = new TaskModel
  443. {
  444. Title = kanban.Title,
  445. ID = kanban.ID,
  446. Description = kanban.Summary ?? "",
  447. EmployeeCategory = subscriber.Employee.ID,
  448. Status = kanban.Status,
  449. Attachments = kanban.Attachments > 0,
  450. DueDate = kanban.DueDate,
  451. CompletedDate = kanban.Completed,
  452. Locked = kanban.Locked,
  453. EstimatedTime = kanban.EstimatedTime,
  454. EmployeeID = kanban.EmployeeLink.ID,
  455. ManagerID = kanban.ManagerLink.ID,
  456. JobID = kanban.JobLink.ID,
  457. JobNumber = kanban.JobLink.JobNumber?.Trim() ?? "",
  458. JobName = kanban.JobLink.Name,
  459. Type = new KanbanType
  460. {
  461. ID = kanban.Type.ID,
  462. Code = kanban.Type.Code
  463. },
  464. Number = kanban.Number,
  465. Checked = SelectedTasks.Any(x => x.ID == kanban.ID)
  466. };
  467. var colour = subscriber.Employee.ID == kanban.EmployeeLink.ID
  468. ? TaskModel.KanbanColor(
  469. kanban.DueDate,
  470. kanban.Completed)
  471. : subscriber.Employee.ID == kanban.ManagerLink.ID
  472. ? Color.Silver
  473. : Color.Plum;
  474. if (kanban.Locked)
  475. {
  476. colour = colour.MixColors(0.5F, Color.White);
  477. }
  478. model.Color = System.Windows.Media.Color.FromArgb(colour.A, colour.R, colour.G, colour.B);
  479. var notes = new List<List<string>> { new() };
  480. var kanbanNotes = kanban.Notes;
  481. if (kanbanNotes != null)
  482. {
  483. foreach (var line in kanbanNotes)
  484. {
  485. if (line == "===================================")
  486. {
  487. notes.Add(new());
  488. }
  489. else
  490. {
  491. notes.Last().Add(line);
  492. }
  493. }
  494. }
  495. model.Notes = string.Join("\n===================================\n", notes.Reverse<List<string>>().Select(x => string.Join('\n', x)));
  496. SetTaskModelAssignedTo(model, kanban, subscriber.Employee.ID);
  497. yield return model;
  498. }
  499. }
  500. private static void SetTaskModelAssignedTo(TaskModel model, IKanban kanban, Guid subscriberID)
  501. {
  502. var employeeString = kanban.EmployeeLink.ID == subscriberID
  503. ? ""
  504. : kanban.EmployeeLink.ID == Guid.Empty
  505. ? " to (Unallocated)"
  506. : " to " + kanban.EmployeeLink.Name;
  507. var managerString = kanban.ManagerLink.ID == Guid.Empty || kanban.ManagerLink.ID == subscriberID
  508. ? ""
  509. : " by " + kanban.ManagerLink.Name;
  510. model.AssignedTo = !string.IsNullOrEmpty(employeeString) || !managerString.IsNullOrWhiteSpace()
  511. ? $"Assigned{employeeString}{managerString}"
  512. : "";
  513. }
  514. #endregion
  515. #region Kanban
  516. private readonly List<TaskModel> SelectedTasks = new();
  517. private List<TaskModel> AllTasks { get; set; } = new();
  518. private void DoEdit(TaskModel task)
  519. {
  520. var result = Host.EditReferences(new[] { task });
  521. if (result)
  522. {
  523. Refresh();
  524. }
  525. }
  526. private void EditTask_Executed(object sender, ExecutedRoutedEventArgs e)
  527. {
  528. if (e.Parameter is not TaskModel model) return;
  529. DoEdit(model);
  530. }
  531. private void OpenTaskMenu_Executed(object sender, ExecutedRoutedEventArgs e)
  532. {
  533. if (e.Parameter is not KanbanResources.OpenTaskMenuCommandArgs args) return;
  534. Host.PopulateMenu(this, args.Model, args.Menu);
  535. }
  536. private void SelectTask_Executed(object sender, ExecutedRoutedEventArgs e)
  537. {
  538. if (e.Parameter is not TaskModel model) return;
  539. if (!SelectedTasks.Remove(model))
  540. {
  541. SelectedTasks.Add(model);
  542. }
  543. }
  544. private void CommandBinding_CanExecute(object sender, CanExecuteRoutedEventArgs e)
  545. {
  546. e.CanExecute = true;
  547. }
  548. private void ItemsControl_DragOver(object sender, DragEventArgs e)
  549. {
  550. if (sender is not FrameworkElement element || element.Tag is not TasksByUserEmployeeCategory category) return;
  551. e.Effects = DragDropEffects.None;
  552. if (e.Data.GetDataPresent(typeof(TaskModel)))
  553. {
  554. var model = e.Data.GetData(typeof(TaskModel)) as TaskModel;
  555. if (model is not null
  556. && (model.Status != category.Status || model.EmployeeCategory != category.EmployeeID)
  557. && !SelectedTasks.Any(x => x.Locked)
  558. && SelectedTasks.Concat(CoreUtils.One(model)).Any(x => x.IsAssignee))
  559. {
  560. e.Effects = DragDropEffects.Move;
  561. }
  562. }
  563. }
  564. private void ItemsControl_Drop(object sender, DragEventArgs e)
  565. {
  566. if (sender is not FrameworkElement element || element.Tag is not TasksByUserEmployeeCategory category) return;
  567. if (e.Data.GetDataPresent(typeof(TaskModel)))
  568. {
  569. var models = SelectedModels(e.Data.GetData(typeof(TaskModel)) as TaskModel)
  570. .Where(x => (!x.Status.Equals(category.Status) || x.EmployeeID != category.EmployeeID) && x.IsAssignee)
  571. .ToList();
  572. if (!models.Any())
  573. {
  574. return;
  575. }
  576. var changingCategory = models.Any(x => !x.Status.Equals(category.Status));
  577. var completing = changingCategory && category.Status == KanbanStatus.Complete;
  578. var completed = DateTime.Now;
  579. if (completing)
  580. {
  581. if (!MessageWindow.ShowYesNo($"Are you sure you want to complete the selected tasks?", "Confirm Completion"))
  582. return;
  583. }
  584. var kanbans = Host.LoadKanbans(models, new Columns<Kanban>(x => x.ID, x => x.EmployeeLink.ID, x => x.Private, x => x.Number, x => x.Status));
  585. var subscribers = new ClientKanbanSubscriberSet(kanbans.Select(x => x.ID));
  586. foreach(var kanban in kanbans)
  587. {
  588. if (!kanban.Private)
  589. {
  590. kanban.EmployeeLink.ID = category.EmployeeID;
  591. subscribers.EnsureAssignee(kanban.ID, kanban.EmployeeLink.ID);
  592. kanban.Status = category.Status;
  593. if (completing)
  594. {
  595. kanban.Completed = completed;
  596. }
  597. }
  598. else
  599. {
  600. MessageBox.Show($"Cannot change assignee for task {kanban.Number} because it is private.");
  601. models.RemoveAll(x => x.ID == kanban.ID);
  602. }
  603. }
  604. Client.Save(kanbans.Where(x => x.IsChanged()), $"Task Employee Updated");
  605. subscribers.Save(true);
  606. var kanbanIDs = models.Select(x => x.ID).ToArray();
  607. AllTasks.RemoveAll(x => kanbanIDs.Contains(x.ID));
  608. AllTasks.AddRange(CreateModels(LoadSubscribers(new Filter<KanbanSubscriber>(x => x.Kanban.ID).InList(kanbanIDs))));
  609. AllTasks.Sort((x, y) => x.DueDate.CompareTo(y.DueDate));
  610. FilterKanbans();
  611. }
  612. }
  613. #endregion
  614. #region ITaskControl
  615. public ITaskHost Host { get; set; }
  616. public KanbanViewType KanbanViewType => KanbanViewType.User;
  617. public bool IsReady { get; set; }
  618. public string SectionName => "Tasks By User";
  619. public DataModel DataModel(Selection selection)
  620. {
  621. var ids = SelectedModels().Select(x => x.ID).ToArray();
  622. return new AutoDataModel<Kanban>(new Filter<Kanban>(x => x.ID).InList(ids));
  623. }
  624. public IEnumerable<TaskModel> SelectedModels(TaskModel? sender = null)
  625. {
  626. if (sender is null)
  627. {
  628. return SelectedTasks;
  629. }
  630. else
  631. {
  632. var result = SelectedTasks.ToList();
  633. if (!result.Contains(sender))
  634. {
  635. result.Add(sender);
  636. }
  637. return result;
  638. }
  639. }
  640. #endregion
  641. #region Settings
  642. private void SaveSettings()
  643. {
  644. Host.KanbanSettings.UserSettings.AnchorWidth = SplitPanel.AnchorWidth;
  645. Host.KanbanSettings.UserSettings.TeamsHeight = SelectedTeams.ActualHeight;
  646. var teams = SelectedTeams.SelectedItems.Select(x => ((KeyValuePair<Guid, string>)x).Key);
  647. Host.KanbanSettings.UserSettings.SelectedTeams = teams.ToArray();
  648. var emps = SelectedEmployees.SelectedItems.Select(x => ((KeyValuePair<Guid, EmployeeModel>)x).Key);
  649. emps = emps.Where(e => TeamEmployees.Any(t => t.Contains(e)));
  650. Host.KanbanSettings.UserSettings.SelectedEmployees = emps.ToArray();
  651. Host.KanbanSettings.UserSettings.IncludeCompleted = IncludeCompleted.IsChecked == true;
  652. Host.KanbanSettings.UserSettings.IncludeObserved = IncludeObserved.IsChecked == true;
  653. Host.KanbanSettings.UserSettings.IncludeManaged = IncludeManaged.IsChecked == true;
  654. Host.SaveSettings();
  655. }
  656. private void SplitPanel_OnChanged(object sender, DynamicSplitPanelSettings e)
  657. {
  658. if (!IsReady || Equals(Host.KanbanSettings.UserSettings.AnchorWidth, e.AnchorWidth))
  659. return;
  660. SaveSettings();
  661. }
  662. private void SelectedTeams_ItemChecked(object sender, ItemCheckedEventArgs e)
  663. {
  664. if (!IsReady)
  665. return;
  666. PopulateEmployees();
  667. SaveSettings();
  668. Refresh();
  669. }
  670. private void SelectedTeams_SizeChanged(object sender, SizeChangedEventArgs e)
  671. {
  672. if (!IsReady || Equals(Host.KanbanSettings.UserSettings.TeamsHeight, SelectedTeams.ActualHeight))
  673. return;
  674. SaveSettings();
  675. }
  676. private void SelectedEmployees_ItemChecked(object sender, ItemCheckedEventArgs e)
  677. {
  678. if (!IsReady || bPopulating || sender != SelectedEmployees)
  679. return;
  680. SaveSettings();
  681. Refresh();
  682. }
  683. private void ViewType_SelectionChanged(object sender, SelectionChangedEventArgs e)
  684. {
  685. if (!IsReady)
  686. return;
  687. Mode = ViewType.SelectedIndex switch
  688. {
  689. 0 => KanbanViewMode.Full,
  690. 1 => KanbanViewMode.Compact,
  691. _ => KanbanViewMode.Full
  692. };
  693. if (IsReady)
  694. {
  695. Host.KanbanSettings.UserSettings.CompactView = Mode == KanbanViewMode.Compact;
  696. Host.SaveSettings();
  697. }
  698. }
  699. #endregion
  700. private void Export_Click(object sender, RoutedEventArgs e)
  701. {
  702. var form = new DynamicExportForm(typeof(Kanban), GetKanbanColumns().ColumnNames());
  703. if (form.ShowDialog() != true)
  704. return;
  705. var export = Client.Query(
  706. new Filter<Kanban>(x => x.ID).InQuery(GetKanbanSubscriberFilter(), x => x.Kanban.ID),
  707. new Columns<Kanban>(form.Fields),
  708. LookupFactory.DefineSort<Kanban>()
  709. );
  710. var employee = "Tasks for " + string.Join(',', Employees.Where(x => Host.KanbanSettings.UserSettings.SelectedEmployees.Contains(x.Key)).Select(x => x.Value.Name));
  711. ExcelExporter.DoExport<Kanban>(
  712. export,
  713. string.Format(
  714. "{0} ({1:dd-MMM-yy})",
  715. employee,
  716. DateTime.Today
  717. )
  718. );
  719. }
  720. private void FoldButton_Click(object sender, RoutedEventArgs e)
  721. {
  722. if (sender is not FrameworkElement element || element.Tag is not TasksByUserCategory category) return;
  723. category.Collapsed = !category.Collapsed;
  724. }
  725. }