TasksByUserControl.xaml.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870
  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.Master != null)
  259. filter = filter.And(c => c.Kanban.JobLink.ID).IsEqualTo(Host.Master.ID);
  260. if (!Host.KanbanSettings.UserSettings.IncludeCompleted)
  261. filter = filter.And(new Filter<KanbanSubscriber>(x => x.Kanban.Completed).IsEqualTo(DateTime.MinValue));
  262. var emps = Employees.Where(x => Host.KanbanSettings.UserSettings.SelectedEmployees.Contains(x.Key));
  263. filter = filter.And(c => c.Employee.ID).InList(emps.Select(x => x.Key).ToArray());
  264. if (!Host.KanbanSettings.UserSettings.IncludeObserved)
  265. {
  266. if (Host.KanbanSettings.UserSettings.IncludeManaged)
  267. filter = filter.And(new Filter<KanbanSubscriber>(x => x.Manager).IsEqualTo(true).Or(x => x.Assignee).IsEqualTo(true));
  268. else
  269. filter = filter.And(x => x.Assignee).IsEqualTo(true);
  270. }
  271. if (additional is not null)
  272. {
  273. return additional.And(filter);
  274. }
  275. else
  276. {
  277. return filter;
  278. }
  279. }
  280. private void ReloadColumns()
  281. {
  282. Model.SectionHeaders.SupressNotification = true;
  283. Model.SectionHeaders.Clear();
  284. var emps = Host.KanbanSettings.UserSettings.SelectedEmployees.OrderBy(x => Employees[x].Name).ToArray();
  285. foreach (var employeeID in emps)
  286. {
  287. if (Employees.TryGetValue(employeeID, out var employee))
  288. {
  289. Model.SectionHeaders.Add(new TasksByUserEmployeeHeader(employeeID, employee.Name, employee.Image ?? anonymous, Model));
  290. }
  291. }
  292. Model.SectionHeaders.SupressNotification = false;
  293. Model.Categories.SupressNotification = true;
  294. var oldCategories = Model.Categories.ToDictionary(x => x.Status);
  295. Model.Categories.Clear();
  296. foreach (var status in _statusOrder)
  297. {
  298. if (status == KanbanStatus.Complete && !Host.KanbanSettings.UserSettings.IncludeCompleted)
  299. {
  300. continue;
  301. }
  302. var newCategory = new TasksByUserCategory(status, oldCategories.GetValueOrDefault(status)?.Collapsed ?? false);
  303. foreach (var employeeID in emps)
  304. {
  305. if (Employees.TryGetValue(employeeID, out var employee))
  306. {
  307. var cat = new TasksByUserEmployeeCategory(employeeID, status);
  308. newCategory.EmployeeCategoryDictionary[employeeID] = cat;
  309. var header = Model.SectionHeaders.First(x => x.EmployeeID == employeeID);
  310. cat.Tasks.CollectionChanged += (s, e) =>
  311. {
  312. header.UpdateTasks();
  313. };
  314. }
  315. }
  316. Model.Categories.Add(newCategory);
  317. }
  318. Model.Categories.SupressNotification = false;
  319. }
  320. private static readonly KanbanStatus[] _statusOrder = new[]
  321. {
  322. KanbanStatus.Open,
  323. KanbanStatus.InProgress,
  324. KanbanStatus.Waiting,
  325. KanbanStatus.Complete
  326. };
  327. private Columns<Kanban> GetKanbanColumns()
  328. {
  329. return new Columns<Kanban>(
  330. x => x.ID,
  331. x => x.DueDate,
  332. x => x.Completed,
  333. x => x.Description,
  334. x => x.Summary,
  335. x => x.Status,
  336. x => x.EmployeeLink.ID,
  337. x => x.EmployeeLink.Name,
  338. x => x.ManagerLink.ID,
  339. x => x.ManagerLink.Name,
  340. x => x.Notes,
  341. x => x.Title,
  342. x => x.JobLink.ID,
  343. x => x.JobLink.JobNumber,
  344. x => x.JobLink.Name,
  345. x => x.Type.ID,
  346. x => x.Type.Code,
  347. x => x.Number,
  348. x => x.Attachments,
  349. x => x.Locked,
  350. x => x.EstimatedTime);
  351. }
  352. private IEnumerable<KanbanSubscriber> LoadSubscribers(Filter<KanbanSubscriber>? filter = null)
  353. {
  354. filter = GetKanbanSubscriberFilter(filter);
  355. var kanbanFilter = new Filter<Kanban>(x => x.ID).InQuery(filter, x => x.Kanban.ID);
  356. var buttonFilter = FilterButton.GetFilter();
  357. if(buttonFilter is not null)
  358. {
  359. kanbanFilter.And(buttonFilter);
  360. }
  361. var results = Client.QueryMultiple(
  362. new KeyedQueryDef<KanbanSubscriber>(
  363. filter,
  364. new Columns<KanbanSubscriber>(x => x.ID, x => x.Employee.ID, x => x.Kanban.ID),
  365. new SortOrder<KanbanSubscriber>(x => x.Kanban.DueDate) { Direction = SortDirection.Ascending }),
  366. new KeyedQueryDef<Kanban>(
  367. kanbanFilter,
  368. GetKanbanColumns()));
  369. var kanbans = results.GetObjects<Kanban>().ToDictionary(x => x.ID);
  370. return results.GetObjects<KanbanSubscriber>().Select(x =>
  371. {
  372. if (kanbans.TryGetValue(x.Kanban.ID, out var kanban))
  373. {
  374. x.Kanban.Synchronise(kanban);
  375. return x;
  376. }
  377. else
  378. {
  379. return null;
  380. }
  381. }).NotNull();
  382. }
  383. public void Refresh()
  384. {
  385. using (new WaitCursor())
  386. {
  387. var models = CreateModels(LoadSubscribers()).ToList();
  388. ReloadColumns();
  389. AllTasks = models.OrderBy(x => x.DueDate).ToList();
  390. FilterKanbans();
  391. }
  392. }
  393. /// <summary>
  394. /// Take the full list of kanbans loaded from the database, and filter based on the search UI elements, filtering into the columns.
  395. /// </summary>
  396. private void FilterKanbans()
  397. {
  398. IEnumerable<TaskModel> filtered = AllTasks;
  399. if (TaskType.SelectedItem is KanbanType kanbanType)
  400. {
  401. filtered = filtered.Where(x => x.Type.ID == kanbanType.ID);
  402. }
  403. if (!string.IsNullOrWhiteSpace(Search.Text))
  404. {
  405. var searches = Search.Text.Split();
  406. filtered = filtered.Where(x => x.Search(searches));
  407. }
  408. var categoryMap = Model.Categories.ToDictionary(x => x.Status, x => x.EmployeeCategoryDictionary);
  409. foreach(var category in Model.Categories)
  410. {
  411. foreach(var empCat in category.EmployeeCategories)
  412. {
  413. empCat.Tasks.Clear();
  414. }
  415. }
  416. SelectedTasks.Clear();
  417. foreach (var task in filtered)
  418. {
  419. if(categoryMap.TryGetValue(task.Status, out var categoryDict))
  420. {
  421. if (categoryDict.TryGetValue(task.EmployeeCategory, out var employeeCategory))
  422. {
  423. employeeCategory.Tasks.Add(task);
  424. if (task.Checked)
  425. {
  426. SelectedTasks.Add(task);
  427. }
  428. }
  429. }
  430. }
  431. }
  432. private IEnumerable<TaskModel> CreateModels(IEnumerable<KanbanSubscriber> subscribers)
  433. {
  434. foreach(var subscriber in subscribers)
  435. {
  436. var kanban = subscriber.Kanban;
  437. var model = new TaskModel
  438. {
  439. Title = kanban.Title,
  440. ID = kanban.ID,
  441. Description = kanban.Summary ?? "",
  442. EmployeeCategory = subscriber.Employee.ID,
  443. Status = kanban.Status,
  444. Attachments = kanban.Attachments > 0,
  445. DueDate = kanban.DueDate,
  446. CompletedDate = kanban.Completed,
  447. Locked = kanban.Locked,
  448. EstimatedTime = kanban.EstimatedTime,
  449. EmployeeID = kanban.EmployeeLink.ID,
  450. ManagerID = kanban.ManagerLink.ID,
  451. JobID = kanban.JobLink.ID,
  452. JobNumber = kanban.JobLink.JobNumber?.Trim() ?? "",
  453. JobName = kanban.JobLink.Name,
  454. Type = new KanbanType
  455. {
  456. ID = kanban.Type.ID,
  457. Code = kanban.Type.Code
  458. },
  459. Number = kanban.Number,
  460. Checked = SelectedTasks.Any(x => x.ID == kanban.ID)
  461. };
  462. var colour = subscriber.Employee.ID == kanban.EmployeeLink.ID
  463. ? TaskModel.KanbanColor(
  464. kanban.DueDate,
  465. kanban.Completed)
  466. : subscriber.Employee.ID == kanban.ManagerLink.ID
  467. ? Color.Silver
  468. : Color.Plum;
  469. if (kanban.Locked)
  470. {
  471. colour = colour.MixColors(0.5F, Color.White);
  472. }
  473. model.Color = System.Windows.Media.Color.FromArgb(colour.A, colour.R, colour.G, colour.B);
  474. var notes = new List<List<string>> { new() };
  475. var kanbanNotes = kanban.Notes;
  476. if (kanbanNotes != null)
  477. {
  478. foreach (var line in kanbanNotes)
  479. {
  480. if (line == "===================================")
  481. {
  482. notes.Add(new());
  483. }
  484. else
  485. {
  486. notes.Last().Add(line);
  487. }
  488. }
  489. }
  490. model.Notes = string.Join("\n===================================\n", notes.Reverse<List<string>>().Select(x => string.Join('\n', x)));
  491. SetTaskModelAssignedTo(model, kanban, subscriber.Employee.ID);
  492. yield return model;
  493. }
  494. }
  495. private static void SetTaskModelAssignedTo(TaskModel model, IKanban kanban, Guid subscriberID)
  496. {
  497. var employeeString = kanban.EmployeeLink.ID == subscriberID
  498. ? ""
  499. : kanban.EmployeeLink.ID == Guid.Empty
  500. ? " to (Unallocated)"
  501. : " to " + kanban.EmployeeLink.Name;
  502. var managerString = kanban.ManagerLink.ID == Guid.Empty || kanban.ManagerLink.ID == subscriberID
  503. ? ""
  504. : " by " + kanban.ManagerLink.Name;
  505. model.AssignedTo = !string.IsNullOrEmpty(employeeString) || !managerString.IsNullOrWhiteSpace()
  506. ? $"Assigned{employeeString}{managerString}"
  507. : "";
  508. }
  509. #endregion
  510. #region Kanban
  511. private readonly List<TaskModel> SelectedTasks = new();
  512. private List<TaskModel> AllTasks { get; set; } = new();
  513. private void DoEdit(TaskModel task)
  514. {
  515. var result = Host.EditReferences(new[] { task });
  516. if (result)
  517. {
  518. Refresh();
  519. }
  520. }
  521. private void EditTask_Executed(object sender, ExecutedRoutedEventArgs e)
  522. {
  523. if (e.Parameter is not TaskModel model) return;
  524. DoEdit(model);
  525. }
  526. private void OpenTaskMenu_Executed(object sender, ExecutedRoutedEventArgs e)
  527. {
  528. if (e.Parameter is not KanbanResources.OpenTaskMenuCommandArgs args) return;
  529. Host.PopulateMenu(this, args.Model, args.Menu);
  530. }
  531. private void SelectTask_Executed(object sender, ExecutedRoutedEventArgs e)
  532. {
  533. if (e.Parameter is not TaskModel model) return;
  534. if (!SelectedTasks.Remove(model))
  535. {
  536. SelectedTasks.Add(model);
  537. }
  538. }
  539. private void CommandBinding_CanExecute(object sender, CanExecuteRoutedEventArgs e)
  540. {
  541. e.CanExecute = true;
  542. }
  543. private void ItemsControl_DragOver(object sender, DragEventArgs e)
  544. {
  545. if (sender is not FrameworkElement element || element.Tag is not TasksByUserEmployeeCategory category) return;
  546. e.Effects = DragDropEffects.None;
  547. if (e.Data.GetDataPresent(typeof(TaskModel)))
  548. {
  549. var model = e.Data.GetData(typeof(TaskModel)) as TaskModel;
  550. if (model is not null
  551. && (model.Status != category.Status || model.EmployeeCategory != category.EmployeeID)
  552. && !SelectedTasks.Any(x => x.Locked)
  553. && SelectedTasks.Concat(CoreUtils.One(model)).Any(x => x.IsAssignee))
  554. {
  555. e.Effects = DragDropEffects.Move;
  556. }
  557. }
  558. }
  559. private void ItemsControl_Drop(object sender, DragEventArgs e)
  560. {
  561. if (sender is not FrameworkElement element || element.Tag is not TasksByUserEmployeeCategory category) return;
  562. if (e.Data.GetDataPresent(typeof(TaskModel)))
  563. {
  564. var models = SelectedModels(e.Data.GetData(typeof(TaskModel)) as TaskModel)
  565. .Where(x => (!x.Status.Equals(category.Status) || x.EmployeeID != category.EmployeeID) && x.IsAssignee)
  566. .ToList();
  567. if (!models.Any())
  568. {
  569. return;
  570. }
  571. var changingCategory = models.Any(x => !x.Status.Equals(category.Status));
  572. var completing = changingCategory && category.Status == KanbanStatus.Complete;
  573. var completed = DateTime.Now;
  574. if (completing)
  575. {
  576. if (!MessageWindow.ShowYesNo($"Are you sure you want to complete the selected tasks?", "Confirm Completion"))
  577. return;
  578. }
  579. var kanbans = Host.LoadKanbans(models, new Columns<Kanban>(x => x.ID, x => x.EmployeeLink.ID, x => x.Private, x => x.Number, x => x.Status));
  580. var subscribers = new ClientKanbanSubscriberSet(kanbans.Select(x => x.ID));
  581. foreach(var kanban in kanbans)
  582. {
  583. if (!kanban.Private)
  584. {
  585. kanban.EmployeeLink.ID = category.EmployeeID;
  586. subscribers.EnsureAssignee(kanban.ID, kanban.EmployeeLink.ID);
  587. kanban.Status = category.Status;
  588. if (completing)
  589. {
  590. kanban.Completed = completed;
  591. }
  592. }
  593. else
  594. {
  595. MessageBox.Show($"Cannot change assignee for task {kanban.Number} because it is private.");
  596. models.RemoveAll(x => x.ID == kanban.ID);
  597. }
  598. }
  599. Client.Save(kanbans.Where(x => x.IsChanged()), $"Task Employee Updated");
  600. subscribers.Save(true);
  601. var kanbanIDs = models.Select(x => x.ID).ToArray();
  602. AllTasks.RemoveAll(x => kanbanIDs.Contains(x.ID));
  603. AllTasks.AddRange(CreateModels(LoadSubscribers(new Filter<KanbanSubscriber>(x => x.Kanban.ID).InList(kanbanIDs))));
  604. AllTasks.Sort((x, y) => x.DueDate.CompareTo(y.DueDate));
  605. FilterKanbans();
  606. }
  607. }
  608. #endregion
  609. #region ITaskControl
  610. public ITaskHost Host { get; set; }
  611. public KanbanViewType KanbanViewType => KanbanViewType.User;
  612. public bool IsReady { get; set; }
  613. public string SectionName => "Tasks By User";
  614. public DataModel DataModel(Selection selection)
  615. {
  616. var ids = SelectedModels().Select(x => x.ID).ToArray();
  617. return new AutoDataModel<Kanban>(new Filter<Kanban>(x => x.ID).InList(ids));
  618. }
  619. public IEnumerable<TaskModel> SelectedModels(TaskModel? sender = null)
  620. {
  621. if (sender is null)
  622. {
  623. return SelectedTasks;
  624. }
  625. else
  626. {
  627. var result = SelectedTasks.ToList();
  628. if (!result.Contains(sender))
  629. {
  630. result.Add(sender);
  631. }
  632. return result;
  633. }
  634. }
  635. #endregion
  636. #region Settings
  637. private void SaveSettings()
  638. {
  639. Host.KanbanSettings.UserSettings.AnchorWidth = SplitPanel.AnchorWidth;
  640. Host.KanbanSettings.UserSettings.TeamsHeight = SelectedTeams.ActualHeight;
  641. var teams = SelectedTeams.SelectedItems.Select(x => ((KeyValuePair<Guid, string>)x).Key);
  642. Host.KanbanSettings.UserSettings.SelectedTeams = teams.ToArray();
  643. var emps = SelectedEmployees.SelectedItems.Select(x => ((KeyValuePair<Guid, EmployeeModel>)x).Key);
  644. emps = emps.Where(e => TeamEmployees.Any(t => t.Contains(e)));
  645. Host.KanbanSettings.UserSettings.SelectedEmployees = emps.ToArray();
  646. Host.KanbanSettings.UserSettings.IncludeCompleted = IncludeCompleted.IsChecked == true;
  647. Host.KanbanSettings.UserSettings.IncludeObserved = IncludeObserved.IsChecked == true;
  648. Host.KanbanSettings.UserSettings.IncludeManaged = IncludeManaged.IsChecked == true;
  649. Host.SaveSettings();
  650. }
  651. private void SplitPanel_OnChanged(object sender, DynamicSplitPanelSettings e)
  652. {
  653. if (!IsReady || Equals(Host.KanbanSettings.UserSettings.AnchorWidth, e.AnchorWidth))
  654. return;
  655. SaveSettings();
  656. }
  657. private void SelectedTeams_ItemChecked(object sender, ItemCheckedEventArgs e)
  658. {
  659. if (!IsReady)
  660. return;
  661. PopulateEmployees();
  662. SaveSettings();
  663. Refresh();
  664. }
  665. private void SelectedTeams_SizeChanged(object sender, SizeChangedEventArgs e)
  666. {
  667. if (!IsReady || Equals(Host.KanbanSettings.UserSettings.TeamsHeight, SelectedTeams.ActualHeight))
  668. return;
  669. SaveSettings();
  670. }
  671. private void SelectedEmployees_ItemChecked(object sender, ItemCheckedEventArgs e)
  672. {
  673. if (!IsReady || bPopulating || sender != SelectedEmployees)
  674. return;
  675. SaveSettings();
  676. Refresh();
  677. }
  678. private void ViewType_SelectionChanged(object sender, SelectionChangedEventArgs e)
  679. {
  680. if (!IsReady)
  681. return;
  682. Mode = ViewType.SelectedIndex switch
  683. {
  684. 0 => KanbanViewMode.Full,
  685. 1 => KanbanViewMode.Compact,
  686. _ => KanbanViewMode.Full
  687. };
  688. if (IsReady)
  689. {
  690. Host.KanbanSettings.UserSettings.CompactView = Mode == KanbanViewMode.Compact;
  691. Host.SaveSettings();
  692. }
  693. }
  694. #endregion
  695. private void Export_Click(object sender, RoutedEventArgs e)
  696. {
  697. var form = new DynamicExportForm(typeof(Kanban), GetKanbanColumns().ColumnNames());
  698. if (form.ShowDialog() != true)
  699. return;
  700. var export = Client.Query(
  701. new Filter<Kanban>(x => x.ID).InQuery(GetKanbanSubscriberFilter(), x => x.Kanban.ID),
  702. new Columns<Kanban>(form.Fields),
  703. LookupFactory.DefineSort<Kanban>()
  704. );
  705. var employee = "Tasks for " + string.Join(',', Employees.Where(x => Host.KanbanSettings.UserSettings.SelectedEmployees.Contains(x.Key)).Select(x => x.Value.Name));
  706. ExcelExporter.DoExport<Kanban>(
  707. export,
  708. string.Format(
  709. "{0} ({1:dd-MMM-yy})",
  710. employee,
  711. DateTime.Today
  712. )
  713. );
  714. }
  715. private void FoldButton_Click(object sender, RoutedEventArgs e)
  716. {
  717. if (sender is not FrameworkElement element || element.Tag is not TasksByUserCategory category) return;
  718. category.Collapsed = !category.Collapsed;
  719. }
  720. }