FactoryPanel.xaml.cs 115 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.ObjectModel;
  4. using System.Drawing;
  5. using System.Linq;
  6. using System.Text;
  7. using System.Threading.Tasks;
  8. using System.Windows;
  9. using System.Windows.Controls;
  10. using System.Windows.Controls.Primitives;
  11. using System.Windows.Input;
  12. using System.Windows.Media;
  13. using System.Windows.Media.Imaging;
  14. using Comal.Classes;
  15. using InABox.Clients;
  16. using InABox.Configuration;
  17. using InABox.Core;
  18. using InABox.DynamicGrid;
  19. using InABox.Reports;
  20. using InABox.Core.Reports;
  21. using InABox.Wpf.Reports;
  22. using InABox.WPF;
  23. using Motorola.Snapi;
  24. using Motorola.Snapi.Constants.Enums;
  25. using Motorola.Snapi.EventArguments;
  26. using org.apache.commons.io.comparator;
  27. using BarcodeType = Comal.Classes.BarcodeType;
  28. using Color = System.Drawing.Color;
  29. using Image = System.Windows.Controls.Image;
  30. using InABox.Wpf;
  31. using System.ComponentModel;
  32. namespace PRSDesktop
  33. {
  34. /// <summary>
  35. /// Interaction logic for FactoryPanel.xaml
  36. /// </summary>
  37. public partial class FactoryPanel : UserControl, IPanel<ManufacturingPacket>
  38. {
  39. private Button _currentButton;
  40. private readonly BitmapImage barcode = PRSDesktop.Resources.barcode.AsBitmapImage();
  41. private readonly BitmapImage disabled = PRSDesktop.Resources.disabled.AsBitmapImage();
  42. //private Document document = null;
  43. private SetoutDocument[] documents = { };
  44. private readonly BitmapImage grouped = PRSDesktop.Resources.grouped.AsBitmapImage();
  45. private readonly string NEARLYDUE_COLOR = "Orange";
  46. private readonly string NOTYETDUE_COLOR = "PaleGreen";
  47. private readonly string OVERDUE_COLOR = "Salmon";
  48. private PDFEditorControl PDFEditor;
  49. private readonly string PRIORITY_COLOR = "Red";
  50. private readonly string QA_COLOR = "Silver";
  51. private QAGrid QAGrid;
  52. public List<IMotorolaBarcodeScanner> Scanners = new();
  53. private readonly string SELECTED_COLOR = "Yellow";
  54. private FactoryFloorLocalSettings settings;
  55. private readonly string SHARED_COLOR = "Lime";
  56. private readonly BitmapImage speechbubble = PRSDesktop.Resources.speechbubble.AsBitmapImage();
  57. private readonly BitmapImage starred = PRSDesktop.Resources.report.AsBitmapImage(48, 48);
  58. private readonly string TREATED_COLOR = "DarkOrchid";
  59. public FactoryPanel()
  60. {
  61. InitializeComponent();
  62. var CanConfigure = Security.IsAllowed<CanConfigureFactoryFloor>();
  63. Section.IsEnabled = CanConfigure;
  64. Station.IsEnabled = CanConfigure;
  65. PendingVisible = Security.IsAllowed<CanManagePendingPackets>();
  66. Kanbans = new ObservableCollection<ManufacturingKanban>();
  67. RackContents.ItemsSource = rackcontents;
  68. }
  69. public bool IsReady { get; set; }
  70. public void Setup()
  71. {
  72. settings = new LocalConfiguration<FactoryFloorLocalSettings>().Load();
  73. if (settings.Section.Equals(CoreUtils.FullGuid) && settings.Station.Equals(-1))
  74. {
  75. var user = new UserConfiguration<FactoryFloorSettings>().Load();
  76. settings.Section = user.Section;
  77. settings.Station = user.Station;
  78. settings.LineColor = ColorTranslator.ToHtml(Color.Red);
  79. settings.FontSize = 16;
  80. new LocalConfiguration<FactoryFloorLocalSettings>().Save(settings);
  81. }
  82. var setups = Client.QueryMultiple(
  83. new KeyedQueryDef<ManufacturingFactory>(),
  84. new KeyedQueryDef<ManufacturingSection>(
  85. new Filter<ManufacturingSection>(x => x.Hidden).IsEqualTo(false),
  86. null,
  87. new SortOrder<ManufacturingSection>(x => x.Factory.Sequence).ThenBy(x => x.Sequence)),
  88. new KeyedQueryDef<ManufacturingTrolley>(),
  89. new KeyedQueryDef<Employee>(
  90. LookupFactory.DefineFilter<Employee>(),
  91. new Columns<Employee>(x => x.ID, x => x.Name, x => x.UserLink.ID),
  92. new SortOrder<Employee>(x => x.Name)),
  93. new KeyedQueryDef<Shipment>(
  94. LookupFactory.DefineFilter<Shipment>(),
  95. new Columns<Shipment>(x => x.ID, x => x.Code, x => x.Description, x => x.BarCode),
  96. null));
  97. Factories = setups[nameof(ManufacturingFactory)].Rows.Select(x => x.ToObject<ManufacturingFactory>()).ToArray();
  98. Sections = setups[nameof(ManufacturingSection)].Rows.Select(x => x.ToObject<ManufacturingSection>()).ToArray();
  99. Trolleys = setups[nameof(ManufacturingTrolley)].Rows.Select(x => x.ToObject<ManufacturingTrolley>()).ToArray();
  100. Employees = setups[nameof(Employee)].ToDictionary<Employee, Guid, string>(x => x.ID, x => x.Name);
  101. var myRow = setups[nameof(Employee)].Rows.FirstOrDefault(r => r.Get<Employee, Guid>(c => c.UserLink.ID).Equals(ClientFactory.UserGuid));
  102. myID = myRow != null ? myRow.Get<Employee, Guid>(c => c.ID) : Guid.Empty;
  103. myName = myRow != null ? myRow.Get<Employee, string>(c => c.Name) : "(Unknown Employee)";
  104. Shipments = setups[nameof(Shipment)];
  105. CurrentSection = Sections.FirstOrDefault(x => x.ID.Equals(settings.Section))
  106. ?? Sections.FirstOrDefault();
  107. var iSection = 0;
  108. var iCount = 0;
  109. foreach (var section in Sections)
  110. {
  111. if (section == CurrentSection)
  112. {
  113. iSection = Section.Items.Count;
  114. iCount = section.Stations; //Sections[id].Item2;
  115. }
  116. Section.Items.Add(string.Format("{0}: {1}", section.Factory.Name, section.Name));
  117. }
  118. Section.SelectedIndex = iSection;
  119. Station.Items.Clear();
  120. for (var i = 1; i <= (CurrentSection != null ? CurrentSection.Stations : 0); i++)
  121. Station.Items.Add(string.Format("Station #{0}", i));
  122. CurrentStation = settings.Station + 1;
  123. if (CurrentStation < 1)
  124. CurrentStation = 1;
  125. if (CurrentStation > iCount)
  126. CurrentStation = iCount;
  127. Station.SelectedIndex = CurrentStation - 1;
  128. SetupScanner();
  129. }
  130. public void Shutdown(CancelEventArgs? cancel)
  131. {
  132. ShutdownScanner();
  133. //UpdateTimeTracking(true);
  134. LoadDrawing(null);
  135. }
  136. public void CreateToolbarButtons(IPanelHost host)
  137. {
  138. ManufacturingSetupActions.Standard(host);
  139. //if (Security.IsAllowed<CanViewManufacturingSelfAssessment>())
  140. // host.CreatePanelAction(new PanelAction() { Caption = "Self Assessment", Image = PRSDesktop.Resources.star, OnExecute = DoSelfAssessment });
  141. host.CreatePanelAction(new PanelAction { Caption = "Lost Time", Image = PRSDesktop.Resources.smiley, OnExecute = DoSelectLostTime });
  142. host.CreatePanelAction(new PanelAction
  143. { Caption = "Treament PO", Image = PRSDesktop.Resources.purchase, OnExecute = DoCreatePurchaseOrder });
  144. host.CreatePanelAction(new PanelAction
  145. { Caption = "Treatment Delivery", Image = PRSDesktop.Resources.barcode, OnExecute = DoScanDeliveryItems });
  146. host.CreatePanelAction(new PanelAction { Caption = "Select Rack", Image = PRSDesktop.Resources.forklift, OnExecute = DoSelectRack });
  147. host.CreatePanelAction(new PanelAction { Caption = "Close Rack", Image = PRSDesktop.Resources.forklift, OnExecute = DoCloseRack });
  148. host.CreatePanelAction(new PanelAction { Caption = "Scan Barcode", Image = PRSDesktop.Resources.product, OnExecute = DoScanBarcode });
  149. }
  150. public Dictionary<string, object[]> Selected()
  151. {
  152. var rows = Packets.Rows.Where(p => Kanbans.Any(k => k.Checked && p.Get<ManufacturingPacket, Guid>(c => c.ID).ToString().Equals(k.ID)));
  153. var pkts = rows.ToArray().Select(r => r.ToObject<ManufacturingPacket>());
  154. return new Dictionary<string, object[]> { { typeof(ManufacturingPacket).EntityName(), pkts.ToArray() } };
  155. }
  156. public string SectionName => "Factory";
  157. public DataModel DataModel(Selection selection)
  158. {
  159. var rows = (Packets == null) || (selection == Selection.None)
  160. ? new CoreRow[] { }
  161. : selection == Selection.Selected
  162. ? Packets.Rows.Where(p => Kanbans.Any(k => k.Checked && p.Get<ManufacturingPacket, Guid>(c => c.ID).ToString().Equals(k.ID)))
  163. : Packets.Rows;
  164. var ids = rows.Any() ? rows.Select(r => r.Get<ManufacturingPacket, Guid>(x => x.ID)).ToArray() : new[] { CoreUtils.FullGuid };
  165. return new ManufacturingPacketDataModel(new Filter<ManufacturingPacket>(x => x.ID).InList(ids));
  166. }
  167. public event DataModelUpdateEvent? OnUpdateDataModel;
  168. public void Heartbeat(TimeSpan time)
  169. {
  170. // Security Descriptor should be disabled for non-manufacturing staff
  171. if (!Security.IsAllowed<CanTrackManufacturingHistory>() || CurrentSection == null)
  172. return;
  173. // If Lost Time is Active, let's record that
  174. if (LostTime != null)
  175. {
  176. var history = new ManufacturingHistory();
  177. history.Date = DateTime.Today;
  178. history.Employee.ID = myID;
  179. history.Packet.ID = Guid.Empty;
  180. history.LostTime.ID = LostTime.ID;
  181. history.Description = LostTime.Description;
  182. //history.Setout.ID = CheckedPackets[id].Item3;
  183. //history.Job.ID = CheckedPackets[id].Item4;
  184. history.Section.ID = CurrentSection.ID;
  185. history.Station = CurrentStation;
  186. history.Activity.ID = LostTime.Activity.ID;
  187. history.WorkDuration = new TimeSpan(time.Ticks);
  188. history.QACompleted = 0;
  189. history.WorkCompleted = 0;
  190. history.Window = time;
  191. new Client<ManufacturingHistory>().Save(history, "", (o, e) => { });
  192. }
  193. else
  194. {
  195. var CheckedPackets = new Dictionary<Guid, Tuple<DateTime, DateTime, Guid, Guid, string>>();
  196. foreach (var kanban in Kanbans.Where(x => x.Checked))
  197. {
  198. var packet = KanbanToPacket(kanban);
  199. if (packet != null)
  200. {
  201. var stagerow = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(packet.ID)
  202. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID)
  203. .Equals(settings.Section));
  204. if (stagerow != null)
  205. {
  206. var stage = stagerow.ToObject<ManufacturingPacketStage>();
  207. if (stage.Station != 0)
  208. CheckedPackets[packet.ID] = new Tuple<DateTime, DateTime, Guid, Guid, string>(stage.Started, stage.Completed,
  209. packet.SetoutLink.ID, packet.SetoutLink.JobLink.ID, packet.Serial);
  210. }
  211. }
  212. }
  213. if (!CheckedPackets.Any())
  214. CheckedPackets[Guid.Empty] =
  215. new Tuple<DateTime, DateTime, Guid, Guid, string>(DateTime.MinValue, DateTime.MinValue, Guid.Empty, Guid.Empty, "No Packet");
  216. var updates = new List<ManufacturingHistory>();
  217. var slice = time.Ticks / CheckedPackets.Count;
  218. foreach (var id in CheckedPackets.Keys)
  219. {
  220. var history = new ManufacturingHistory();
  221. history.Date = DateTime.Today;
  222. history.Employee.ID = myID;
  223. history.Packet.ID = id;
  224. history.Description = CheckedPackets[id].Item5;
  225. //history.Setout.ID = CheckedPackets[id].Item3;
  226. //history.Job.ID = CheckedPackets[id].Item4;
  227. history.Section.ID = CurrentSection.ID;
  228. history.Activity.ID = CurrentSection.Activity.ID;
  229. history.Station = CurrentStation;
  230. if (CheckedPackets[id].Item1.Equals(DateTime.MinValue))
  231. history.QADuration = new TimeSpan(slice);
  232. else
  233. history.WorkDuration = new TimeSpan(slice);
  234. history.QACompleted = CheckedPackets[id].Item1.Equals(DateTime.MinValue) ? 0 : 0;
  235. history.WorkCompleted = CheckedPackets[id].Item2.Equals(DateTime.MinValue) ? 0 : 1;
  236. history.Window = time;
  237. updates.Add(history);
  238. }
  239. if (updates.Any())
  240. new Client<ManufacturingHistory>().Save(updates, "", (o, e) => { });
  241. }
  242. }
  243. private void ShutdownScanner()
  244. {
  245. try
  246. {
  247. foreach (var scanner in Scanners) scanner.Actions.ToggleLed(LedMode.GreenOff);
  248. BarcodeScannerManager.Instance.DataReceived -= Instance_DataReceived;
  249. BarcodeScannerManager.Instance.Close();
  250. }
  251. catch (Exception e)
  252. {
  253. MessageBox.Show("Error Shutting down Scanner!\n\n" + e.Message);
  254. }
  255. }
  256. private void SetupScanner()
  257. {
  258. Scanners.Clear();
  259. BarcodeScannerManager.Instance.Open();
  260. BarcodeScannerManager.Instance.RegisterForEvents(EventType.Barcode, EventType.Pnp, EventType.Image, EventType.Other, EventType.Rmd);
  261. BarcodeScannerManager.Instance.GetDevices();
  262. foreach (var scanner in BarcodeScannerManager.Instance.GetDevices())
  263. {
  264. Scanners.Add(scanner);
  265. scanner.Actions.ToggleLed(LedMode.RedOn);
  266. scanner.Actions.SoundBeeper(BeepPattern.FastWarble);
  267. }
  268. BarcodeScannerManager.Instance.DataReceived += Instance_DataReceived;
  269. }
  270. private void Instance_DataReceived(object? sender, BarcodeScanEventArgs e)
  271. {
  272. Dispatcher.Invoke(() => { ProcessCode(Scanners[(int)e.ScannerId], e.Data); });
  273. }
  274. private void ProcessCode(IMotorolaBarcodeScanner scanner, string code)
  275. {
  276. try
  277. {
  278. var isGuid = Guid.TryParse(code, out var guid);
  279. if (isGuid)
  280. {
  281. if (Sections.Any(x => x.ID.Equals(guid)))
  282. {
  283. IsReady = false;
  284. CurrentSection = Sections.First(x => x.ID.Equals(guid));
  285. Section.SelectedIndex = Array.IndexOf(Sections, CurrentSection);
  286. IsReady = true;
  287. Progress.Show("Changing Sections");
  288. DoRefresh(true);
  289. Progress.Close();
  290. scanner?.Actions.SoundBeeper(BeepPattern.FastWarble);
  291. }
  292. else if (Trolleys.Any(x => x.ID.Equals(guid)))
  293. {
  294. var trolley = Trolleys.First().Code;
  295. var updates = new List<ManufacturingPacket>();
  296. foreach (var kanban in Kanbans.Where(x => x.Checked))
  297. {
  298. var packet = KanbanToPacket(kanban);
  299. packet.Trolleys = trolley;
  300. updates.Add(packet);
  301. var pktrow = Packets.Rows.FirstOrDefault(r => r.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(Guid.Parse(kanban.ID)));
  302. var stagerow = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(packet.ID)
  303. && r.Get<ManufacturingPacketStage, Guid>(
  304. c => c.ManufacturingSectionLink.ID)
  305. .Equals(settings.Section));
  306. pktrow.Set<ManufacturingPacket, string>(x => x.Trolleys, packet.Trolleys);
  307. LoadModel(kanban, pktrow, stagerow, true);
  308. UpdateSelectedKanban(false);
  309. }
  310. if (updates.Any())
  311. new Client<ManufacturingPacket>().Save(updates, "Set Trolley to " + trolley, (o, e) => { });
  312. scanner?.Actions.SoundBeeper(BeepPattern.LowHigh);
  313. }
  314. else
  315. {
  316. scanner?.Actions.SoundBeeper(BeepPattern.FourLowShort);
  317. }
  318. }
  319. else
  320. {
  321. var shiprow = Shipments.Rows.FirstOrDefault(r => r.Get<Shipment, string>(c => c.BarCode).Equals(code));
  322. if (shiprow != null)
  323. {
  324. if (string.Equals(code, rackbarcode))
  325. {
  326. RackPanel.Visibility = Visibility.Collapsed;
  327. RackContents.ItemsSource = null;
  328. rackid = Guid.Empty;
  329. rackbarcode = "";
  330. scanner?.Actions.SoundBeeper(BeepPattern.ThreeLowShort);
  331. }
  332. else
  333. {
  334. RackContents.ItemsSource = null;
  335. rackid = shiprow.Get<Shipment, Guid>(c => c.ID);
  336. rackbarcode = code;
  337. var rows = DeliveryItems.Rows.Where(r => r.Get<DeliveryItem, Guid>(c => c.ShipmentLink.ID).Equals(rackid)).ToArray();
  338. rackcontents.Clear();
  339. rackcontents.AddRange(rows.Select(x => x.ToObject<DeliveryItem>()));
  340. RackContents.ItemsSource = rackcontents;
  341. RackName.Content = string.Format("Rack {0}", shiprow.Get<Shipment, string>(c => c.Code));
  342. RackCount.Content = rows.Length.ToString();
  343. RackPanel.Visibility = Visibility.Visible;
  344. scanner?.Actions.SoundBeeper(BeepPattern.ThreeHighShort);
  345. }
  346. }
  347. else
  348. {
  349. if (RackPanel.Visibility == Visibility.Visible)
  350. {
  351. var row = DeliveryItems.Rows.FirstOrDefault(r => r.Get<DeliveryItem, string>(c => c.Barcode).Equals(code));
  352. if (row != null)
  353. {
  354. var delitem = rackcontents.FirstOrDefault(x => string.Equals(x.Barcode, code));
  355. if (delitem != null)
  356. {
  357. rackcontents.Remove(delitem);
  358. delitem.ShipmentLink.ID = Guid.Empty;
  359. new Client<DeliveryItem>().Save(delitem, "Item Removed From Rack", (o, e) => { });
  360. row.Set<DeliveryItem, Guid>(x => x.ShipmentLink.ID, Guid.Empty);
  361. scanner?.Actions.SoundBeeper(BeepPattern.HighLow);
  362. }
  363. else
  364. {
  365. delitem = row.ToObject<DeliveryItem>();
  366. delitem.ShipmentLink.ID = rackid;
  367. rackcontents.Add(delitem);
  368. new Client<DeliveryItem>().Save(delitem, "Item Added to " + RackName.Content, (o, e) => { });
  369. row.Set<DeliveryItem, Guid>(x => x.ShipmentLink.ID, rackid);
  370. if (Kanbans.Any(x => string.Equals(x.ID, delitem.ManufacturingPacketLink.ID.ToString())))
  371. ReloadPackets(true);
  372. scanner?.Actions.SoundBeeper(BeepPattern.LowHigh);
  373. }
  374. RackContents.ItemsSource = null;
  375. RackContents.ItemsSource = rackcontents;
  376. RackCount.Content = rackcontents.Count.ToString();
  377. }
  378. }
  379. else
  380. {
  381. var id = new Client<DeliveryItem>().Query(
  382. new Filter<DeliveryItem>(x => x.Barcode).IsEqualTo(code),
  383. new Columns<DeliveryItem>(x => x.ManufacturingPacketLink.ID)
  384. ).Rows.FirstOrDefault()?.Get<DeliveryItem, Guid>(x => x.ManufacturingPacketLink.ID);
  385. if (!id.HasValue)
  386. {
  387. scanner?.Actions.SoundBeeper(BeepPattern.FourLowShort);
  388. return;
  389. }
  390. var kanban = Kanbans.FirstOrDefault(x => string.Equals(x.ID, id.Value.ToString()));
  391. if (kanban == null)
  392. {
  393. // Error - packet not visible from this station
  394. scanner?.Actions.SoundBeeper(BeepPattern.FourLowShort);
  395. return;
  396. }
  397. var stage = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(id)
  398. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID)
  399. .Equals(settings.Section))?.ToObject<ManufacturingPacketStage>();
  400. if (stage == null)
  401. scanner?.Actions.SoundBeeper(BeepPattern.FourHighShort);
  402. if (stage.Station == 0) AddPacketToCurrentWorkload(stage);
  403. if (kanban != CurrentKanban)
  404. {
  405. CurrentKanban = kanban;
  406. UpdateSelectedKanban(false);
  407. }
  408. scanner?.Actions.SoundBeeper(BeepPattern.LowHigh);
  409. }
  410. }
  411. }
  412. }
  413. catch (Exception)
  414. {
  415. scanner?.Actions.SoundBeeper(BeepPattern.FourLowShort);
  416. }
  417. }
  418. private void LoadDrawing(Guid? id)
  419. {
  420. if (PDFEditor != null)
  421. {
  422. PDFEditor.Document = null;
  423. // Unload PDF Annotations
  424. PDFEditor = null;
  425. }
  426. else if (QAGrid != null)
  427. {
  428. Editor.Content = null;
  429. // Unload QA Answers
  430. QAGrid = null;
  431. }
  432. if (id.HasValue)
  433. {
  434. var pktid = CurrentKanban != null ? Guid.Parse(CurrentKanban.ID) : Guid.Empty;
  435. var row = Packets.Rows.FirstOrDefault(r => r.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(pktid));
  436. var stagerow = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(pktid)
  437. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID)
  438. .Equals(settings.Section));
  439. var packet = row.ToObject<ManufacturingPacket>();
  440. if (id.Value == Guid.Empty)
  441. {
  442. QAGrid = new QAGrid();
  443. var qadata = stagerow != null ? stagerow.Get<ManufacturingPacketStage, string>(c => c.FormData) : "";
  444. var values = string.IsNullOrWhiteSpace(qadata)
  445. ? new Dictionary<Guid, object>()
  446. : Serialization.Deserialize<Dictionary<Guid, object>>(qadata);
  447. if (CurrentSection != null && row != null)
  448. {
  449. var genquestions = LoadQAQuestions(packet, true, false);
  450. QAGrid.LoadChecks("Global Checks for " + CurrentSection.Name, genquestions, values);
  451. //var specquestions = LoadQAQuestions(packet, false, true);
  452. //QAGrid.LoadChecks(String.Format("Specific Checks for {0} [{1}] template", packet.ManufacturingTemplateLink.Name, packet.ManufacturingTemplateLink.Code), specquestions, values);
  453. }
  454. QAGrid.OnChanged += QAGridChanged;
  455. Editor.Content = QAGrid;
  456. }
  457. else
  458. {
  459. PDFEditor = new PDFEditorControl();
  460. PDFEditor.LineColor = settings.LineColor;
  461. PDFEditor.TextSize = settings.FontSize;
  462. //PDFEditor.PrintAllowed = Security.IsAllowed<CanPrintFactoryFloorDrawings>();
  463. PDFEditor.SaveAllowed = Security.IsAllowed<CanSaveFactoryFloorDrawings>();
  464. var document = documents.FirstOrDefault(x => x.DocumentLink.ID.Equals(id));
  465. PDFEditor.Watermark = packet.WaterMark;
  466. PDFEditor.Document = document;
  467. PDFEditor.PDFSettingsChanged += PDFEditorSettingsChanged;
  468. Editor.Content = PDFEditor;
  469. }
  470. }
  471. }
  472. private void DoSelectRack(PanelAction obj)
  473. {
  474. var dlg = new MultiSelectDialog<Shipment>(
  475. null,
  476. new Columns<Shipment>(x => x.ID, x => x.Code, x => x.Description, x => x.BarCode),
  477. false
  478. );
  479. if (dlg.ShowDialog())
  480. {
  481. var id = dlg.IDs().FirstOrDefault();
  482. var barcode = dlg.Data().Rows.FirstOrDefault(r => r.Get<Shipment, Guid>(c => c.ID).Equals(id))?.Get<Shipment, string>(x => x.BarCode);
  483. ProcessCode(Scanners.FirstOrDefault(), barcode);
  484. }
  485. }
  486. private void DoCloseRack(PanelAction obj)
  487. {
  488. if (!string.IsNullOrWhiteSpace(rackbarcode))
  489. ProcessCode(Scanners.FirstOrDefault(), rackbarcode);
  490. }
  491. private void DoScanBarcode(PanelAction obj)
  492. {
  493. var dlg = new MultiSelectDialog<DeliveryItem>(
  494. new Filter<DeliveryItem>(x => x.DeliveredDate).IsEqualTo(DateTime.MinValue)
  495. .And(x => x.ManufacturingPacketLink).LinkValid(),
  496. new Columns<DeliveryItem>(x => x.ID, x => x.Barcode, x => x.Description),
  497. false
  498. );
  499. if (dlg.ShowDialog())
  500. {
  501. var id = dlg.IDs().FirstOrDefault();
  502. var barcode = dlg.Data().Rows.FirstOrDefault(r => r.Get<DeliveryItem, Guid>(c => c.ID).Equals(id))
  503. ?.Get<DeliveryItem, string>(x => x.Barcode);
  504. ProcessCode(Scanners.FirstOrDefault(), barcode);
  505. }
  506. }
  507. private void DoScanDeliveryItems(PanelAction obj)
  508. {
  509. new DeliveryBuilder(Guid.Empty, Guid.Empty).ShowDialog();
  510. }
  511. private void DoCreatePurchaseOrder(PanelAction obj)
  512. {
  513. //Guid section = CurrentSection != null ? CurrentSection.ID : CoreUtils.FullGuid;
  514. //var stages = Stages.Rows.Where(row => row.Get<ManufacturingPacketStage, Guid>(col => col.ManufacturingSectionLink.ID).Equals(section) && row.Get<ManufacturingPacketStage, int>(col => col.Station).Equals(CurrentStation));
  515. //var ids = stages.Select(row => row.Get<ManufacturingPacketStage, Guid>(col => col.ManufacturingPacketLink.ID)).ToArray();
  516. var ids = Kanbans.Where(x => !x.ColorKey.Equals(QA_COLOR)).Select(x => Guid.Parse(x.ID)).ToArray();
  517. var treatments = new Client<ManufacturingTreatment>().Query(new Filter<ManufacturingTreatment>(x => x.Packet.ID).InList(ids));
  518. var window = new ManufacturingTreatmentWindow(treatments);
  519. if (window.ShowDialog() != true)
  520. return;
  521. Progress.Show("Creating Purchase Order");
  522. var order = new PurchaseOrder();
  523. //Supplier supplier = new Client<Supplier>().Query(
  524. // new Filter<Supplier>(x => x.ID).IsEqualTo(window.SupplierID),
  525. // new Columns<Supplier>(x=>x.ID
  526. //).Rows.FirstOrDefault()?.ToObject<Supplier>());
  527. order.SupplierLink.ID = window.SupplierID;
  528. order.SupplierLink.Name = window.SupplierName; //supplier != null ? supplier.Name : "Unknown Supplier";
  529. order.Description = string.Format("Materials Processing Request raised by {0} from Factory Floor", Employees[myID]);
  530. order.RaisedBy.ID = myID;
  531. order.IssuedBy.ID = myID;
  532. order.IssuedDate = DateTime.Today;
  533. order.DueDate = DateTime.Today.AddDays(7);
  534. new Client<PurchaseOrder>().Save(order, "Created Treatment Purchase Order");
  535. Progress.SetMessage("Processing Order");
  536. var orderitems = new List<PurchaseOrderItem>();
  537. var packetupdates = new List<ManufacturingPacket>();
  538. var query = new MultiQuery();
  539. query.Add(
  540. new Filter<SupplierProduct>(x => x.Product.ID).IsEqualTo(window.ProductID).And(x => x.SupplierLink.ID)
  541. .IsEqualTo(window.SupplierID),
  542. new Columns<SupplierProduct>(x => x.Job.ID).Add(x => x.CostPrice).Add(x => x.Product.NettCost)
  543. );
  544. query.Add(
  545. new Filter<Product>(x => x.ID).IsEqualTo(window.ProductID),
  546. new Columns<Product>(x => x.NettCost)
  547. );
  548. query.Query();
  549. var supprods = query.Get<SupplierProduct>().Rows.Select(x => x.ToObject<SupplierProduct>()).ToArray();
  550. var supprice = supprods.FirstOrDefault(x => Equals(x.Job.ID, Guid.Empty));
  551. var prodprice = query.Get<Product>().Rows.FirstOrDefault()?.ToObject<Product>();
  552. var stdcost = supprice != null
  553. ? supprice.CostPrice
  554. : prodprice != null
  555. ? prodprice.NettCost
  556. : 0.0F;
  557. foreach (var row in window.Selected)
  558. {
  559. var treatment = row.ToObject<ManufacturingTreatment>();
  560. var packet = Packets.Rows.First(p => p.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(treatment.Packet.ID))
  561. .ToObject<ManufacturingPacket>();
  562. packetupdates.Add(packet);
  563. var item = new PurchaseOrderItem();
  564. item.PurchaseOrderLink.ID = order.ID;
  565. item.Job.ID = treatment.Packet.SetoutLink.JobLink.ID;
  566. item.Packet.ID = packet.ID;
  567. item.Product.ID = window.ProductID;
  568. item.Qty = packet.BarcodeQty;
  569. item.Dimensions.Length = treatment.Parameter == 0.0F ? 1.0F : treatment.Parameter;
  570. var jobprice = supprods.FirstOrDefault(x => x.Job.ID.Equals(item.Job.ID));
  571. item.Cost = (jobprice != null ? jobprice.CostPrice : stdcost) * treatment.Parameter;
  572. var description = new List<string>();
  573. description.Add(string.Format("{0} x {1} - {2}", packet.BarcodeQty, packet.Serial, packet.Title));
  574. var dimensions = new List<string>();
  575. if (packet.Height > 0.0F)
  576. dimensions.Add(string.Format("H:{0:F2}mm", packet.Height));
  577. if (packet.Width > 0.0F)
  578. dimensions.Add(string.Format("W:{0:F2}mm", packet.Width));
  579. if (packet.Length > 0.0F)
  580. dimensions.Add(string.Format("L:{0:F2}mm", packet.Length));
  581. dimensions.Add(string.Format("Param:{0:F4}", treatment.Parameter));
  582. if (dimensions.Any())
  583. description.Add(string.Format("Dimensions: {0}", string.Join(" ", dimensions)));
  584. item.Description = string.Join("\n", description);
  585. orderitems.Add(item);
  586. }
  587. new Client<PurchaseOrderItem>().Save(orderitems, "Created by Factory Floor Purchase");
  588. Progress.SetMessage("Updating Packets");
  589. foreach (var orderitem in orderitems)
  590. {
  591. var packet = packetupdates.FirstOrDefault(x => x.ID.Equals(orderitem.Packet.ID));
  592. if (packet != null)
  593. packet.OrderItem.ID = orderitem.ID;
  594. }
  595. new Client<ManufacturingPacket>().Save(packetupdates.Where(x => x.IsChanged()), "");
  596. Progress.SetMessage("Creating Delivery");
  597. var delivery = new Delivery();
  598. delivery.Date = DateTime.Today;
  599. delivery.Due = DateTime.Today;
  600. delivery.Employee.ID = myID;
  601. var sb = new StringBuilder();
  602. sb.Append("Serial # ");
  603. sb.AppendLine();
  604. if (Employees.ContainsKey(myID))
  605. sb.AppendLine();
  606. delivery.Notes = string.Format("Delivery of Items for processing to {0}\nOrder #{1} (raised on {2:dd MMM yy} by {3})\nItems: {4}",
  607. order.SupplierLink.Name,
  608. order.PONumber,
  609. delivery.Date,
  610. myName,
  611. string.Join(", ",
  612. orderitems.Select(x =>
  613. string.Format("{0}{1}", x.Description.Split('\n').FirstOrDefault(), x.Qty > 1 ? " (x" + x.Qty.ToString("F0") + ")" : "")))
  614. );
  615. new Client<Delivery>().Save(delivery, "Created by Factory Floor Purchase");
  616. Progress.Close();
  617. PrintDeliveryBarcode(delivery, order);
  618. PrintOrderItemBarcodes(delivery, order);
  619. SendPurchaseNotification(order.PONumber);
  620. MessageBox.Show("All Done");
  621. new DeliveryBuilder(delivery.ID, order.ID).ShowDialog();
  622. Refresh();
  623. }
  624. private void SendPurchaseNotification(string PONumber)
  625. {
  626. var updates = new List<Notification>();
  627. var roles = new Client<EmployeeRole>().Query(new Filter<EmployeeRole>(x => x.RoleLink.Code).IsEqualTo("PURCHASES"));
  628. foreach (var role in roles.Rows)
  629. {
  630. var notification = new Notification
  631. {
  632. Title = string.Format("Treatment PO #{0} has been raised", PONumber),
  633. Description = "The above Purchase Order has been created, and is ready to be checked and issued to the relevant supplier."
  634. };
  635. notification.Sender.ID = myID;
  636. notification.Employee.ID = role.Get<EmployeeRole, Guid>(x => x.EmployeeLink.ID);
  637. updates.Add(notification);
  638. }
  639. new Client<Notification>().Save(updates, "Sent Notification");
  640. }
  641. private void PrintDeliveryBarcode(Delivery delivery, PurchaseOrder order)
  642. {
  643. var model = new ManufacturingTreatmentDataModel(order, delivery);
  644. var templatename = "Print Treatment Delivery Bar Code";
  645. var sectionName = "Treatment Delivery Bar Code";
  646. var template = new Client<ReportTemplate>()
  647. .Load(
  648. new Filter<ReportTemplate>(x => x.Name).IsEqualTo(templatename)
  649. .And(x => x.DataModel).IsEqualTo(model.Name)
  650. .And(x => x.Section).IsEqualTo(sectionName)
  651. ).FirstOrDefault();
  652. if (template == null)
  653. {
  654. template = new ReportTemplate
  655. {
  656. DataModel = model.Name,
  657. Section = sectionName,
  658. Name = templatename
  659. };
  660. new Client<ReportTemplate>().Save(template, "Auto Created Report Template");
  661. }
  662. ReportUtils.PreviewReport(template, model, !Security.IsAllowed<CanDesignReports>(), Security.IsAllowed<CanDesignReports>());
  663. }
  664. private void PrintOrderItemBarcodes(Delivery delivery, PurchaseOrder order)
  665. {
  666. var model = new ManufacturingTreatmentDataModel(order, delivery);
  667. var templatename = "Print Treatment Item Bar Codes";
  668. var sectionName = "Treatment Item Bar Code";
  669. var template = new Client<ReportTemplate>()
  670. .Load(
  671. new Filter<ReportTemplate>(x => x.Name).IsEqualTo(templatename)
  672. .And(x => x.DataModel).IsEqualTo(model.Name)
  673. .And(x => x.Section).IsEqualTo(sectionName)
  674. ).FirstOrDefault();
  675. if (template == null)
  676. {
  677. template = new ReportTemplate
  678. {
  679. DataModel = model.Name,
  680. Section = sectionName,
  681. Name = templatename
  682. };
  683. new Client<ReportTemplate>().Save(template, "Auto Created Report Template");
  684. }
  685. ReportUtils.PreviewReport(template, model, !Security.IsAllowed<CanDesignReports>(), Security.IsAllowed<CanDesignReports>());
  686. }
  687. private CoreTable POItemTable(IEnumerable<PurchaseOrderItem> items)
  688. {
  689. var result = new CoreTable();
  690. result.LoadColumns(typeof(PurchaseOrderItem));
  691. result.LoadRows(items);
  692. return result;
  693. }
  694. private void DoSelectLostTime(PanelAction obj)
  695. {
  696. var chooser = new FactoryLostTimeChooser();
  697. if (chooser.ShowDialog() == true)
  698. {
  699. LostTime = chooser.SelectedLostTime;
  700. LostTimeDescription.Content = LostTime.Description;
  701. LostTimeActive.Visibility = Visibility.Visible;
  702. }
  703. }
  704. private void CancelLostTime_Click(object sender, RoutedEventArgs e)
  705. {
  706. LostTime = null;
  707. LostTimeActive.Visibility = Visibility.Collapsed;
  708. }
  709. //private void Current_DispatcherUnhandledException(object sender, DispatcherUnhandledExceptionEventArgs e)
  710. //{
  711. // MessageBox.Show("Exception!\n\n" + e.Exception.Message + "\n\n" + e.Exception.StackTrace);
  712. // e.Handled = true;
  713. //}
  714. private void Section_SelectionChanged(object sender, SelectionChangedEventArgs e)
  715. {
  716. if (!IsReady)
  717. return;
  718. CurrentSection = Sections[Section.SelectedIndex];
  719. //Kanban.Columns.Clear();
  720. //Kanban.Columns.Add(new KanbanColumn() { Title = CurrentSection, Categories = CurrentSection.ID.ToString() });
  721. //Kanban.Columns[0].AllowDrag = false;
  722. //Kanban.Columns[0].PreviewMouseWheel += CardBorder_PreviewMouseWheel;
  723. IsReady = false;
  724. Station.Items.Clear();
  725. for (var i = 1; i <= CurrentSection.Stations; i++)
  726. Station.Items.Add(string.Format("Station #{0}", i));
  727. IsReady = true;
  728. Station.SelectedIndex = 0;
  729. }
  730. private void Station_SelectionChanged(object sender, SelectionChangedEventArgs e)
  731. {
  732. if (!IsReady)
  733. return;
  734. SelectSectionAndStation(CurrentSection.ID, Station.SelectedIndex + 1);
  735. }
  736. private void SelectSectionAndStation(Guid section, int station)
  737. {
  738. CurrentKanban = null;
  739. CurrentStation = station;
  740. settings.Section = section;
  741. settings.Station = CurrentStation - 1;
  742. new LocalConfiguration<FactoryFloorLocalSettings>().Save(settings);
  743. DoRefresh(true);
  744. }
  745. private void LoadKanban()
  746. {
  747. using (new WaitCursor())
  748. {
  749. if (CurrentKanban == null)
  750. {
  751. ButtonStack.Children.Clear();
  752. LoadDrawing(null);
  753. }
  754. else if (true) // Kanban has changed - how do we figure this out
  755. {
  756. var packet = KanbanToPacket(CurrentKanban);
  757. //SetoutStage stage = packet.GetCurrentStage();
  758. var row = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(packet.ID)
  759. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID)
  760. .Equals(settings.Section));
  761. var station = row != null ? row.Get<ManufacturingPacketStage, int>(c => c.Station) : 0;
  762. var quality = row != null ? row.Get<ManufacturingPacketStage, QualityStatus>(c => c.QualityStatus) : QualityStatus.NotChecked;
  763. var percentage = row != null ? row.Get<ManufacturingPacketStage, double>(c => c.PercentageComplete) : 0.00F;
  764. var Pending = station == 0; //packet.StageLink.Station == 0;
  765. // Set the Proper Button Set for this type of Packet
  766. MfgRow.Height = Pending ? new GridLength(00) : new GridLength(50);
  767. foreach (var btn in ButtonGrid.FindVisualChildren<Button>())
  768. btn.IsEnabled = CurrentKanban != null;
  769. var idle = CompleteButton.Background;
  770. var current = new SolidColorBrush(Colors.SteelBlue);
  771. Update100Button.Background = percentage >= 100.0F ? current : idle;
  772. Update075Button.Background = percentage >= 75.0F ? current : idle;
  773. Update050Button.Background = percentage >= 50.0F ? current : idle;
  774. Update025Button.Background = percentage >= 25.0F ? current : idle;
  775. Update000Button.Background = percentage >= 00.0F ? current : idle;
  776. Button firstbutton = null;
  777. //LoadDrawing(Guid.Empty);
  778. ButtonStack.Children.Clear();
  779. // If we are quality checking, add the QA Check button
  780. if (!Pending)
  781. {
  782. var qabtn = new Button();
  783. qabtn.Content = "Quality Checks";
  784. qabtn.Tag = null;
  785. qabtn.Style = Resources["UnselectedButton"] as Style;
  786. qabtn.Click += ViewQualityChecks;
  787. ButtonStack.Children.Add(qabtn);
  788. if (packet.Drawing.IsValid())
  789. {
  790. var pdfbtn = new Button();
  791. pdfbtn.Content = packet.Drawing.FileName.ToLower();
  792. pdfbtn.Tag = packet.Drawing;
  793. pdfbtn.Style = Resources["UnselectedButton"] as Style;
  794. pdfbtn.Click += ViewDrawing;
  795. ButtonStack.Children.Add(pdfbtn);
  796. firstbutton = pdfbtn;
  797. }
  798. // List all available PDFs for this Setout
  799. documents = new Client<SetoutDocument>().Load(
  800. new Filter<SetoutDocument>(x => x.EntityLink.ID).IsEqualTo(CurrentKanban.SetoutID));
  801. foreach (var document in documents)
  802. {
  803. // If this document is actually the cutting group document,
  804. // redirect any annotations to the cutting group id rather than the setout
  805. if (document.DocumentLink.ID == packet.SetoutLink.Group.OptimizationDocument.ID)
  806. document.ID = packet.SetoutLink.Group.ID;
  807. if (document.DocumentLink != null && document.DocumentLink.FileName != null)
  808. if (document.DocumentLink.FileName.ToLower().EndsWith(".pdf") ||
  809. document.DocumentLink.FileName.ToLower().EndsWith(".png") ||
  810. document.DocumentLink.FileName.ToLower().EndsWith(".bmp") ||
  811. document.DocumentLink.FileName.ToLower().EndsWith(".jpg"))
  812. {
  813. var pdfbtn = new Button();
  814. pdfbtn.Content = document.DocumentLink.FileName.ToLower() +
  815. (!document.Superceded.IsEmpty() ? " (SUPERCEDED)" : "");
  816. pdfbtn.Tag = document.DocumentLink;
  817. pdfbtn.Style = Resources["UnselectedButton"] as Style;
  818. pdfbtn.Click += ViewDrawing;
  819. ButtonStack.Children.Add(pdfbtn);
  820. if (document.Superceded.IsEmpty())
  821. if ((PDFEditor != null && PDFEditor.Document.ID.Equals(document.ID)) ||
  822. firstbutton == null)
  823. firstbutton = pdfbtn;
  824. }
  825. }
  826. if (firstbutton == null)
  827. firstbutton = qabtn;
  828. }
  829. if (firstbutton != null)
  830. //SetCurrentButton(firstbutton);
  831. firstbutton.RaiseEvent(new RoutedEventArgs(ButtonBase.ClickEvent));
  832. else
  833. LoadDrawing(null);
  834. } // Kanban has not changed - do nothing
  835. }
  836. }
  837. private void SetCurrentButton(Button button)
  838. {
  839. if (_currentButton != null)
  840. _currentButton.Style = Resources["UnselectedButton"] as Style;
  841. _currentButton = button;
  842. button.Style = Resources["SelectedButton"] as Style;
  843. }
  844. private void SwimLaneSelected(object sender, MouseButtonEventArgs e)
  845. {
  846. }
  847. private void CardSelected(object sender, MouseButtonEventArgs e)
  848. {
  849. var border = (Border)sender;
  850. CurrentKanban = (ManufacturingKanban)border.Tag;
  851. UpdateSelectedKanban();
  852. }
  853. private bool CheckedPacketsChanged(Dictionary<Guid, Tuple<DateTime, DateTime>> lastCheckedPackets,
  854. Dictionary<Guid, Tuple<DateTime, DateTime>> newCheckedPackets)
  855. {
  856. if (lastCheckedPackets.Count != newCheckedPackets.Count)
  857. return true;
  858. foreach (var key in lastCheckedPackets.Keys)
  859. {
  860. if (!newCheckedPackets.ContainsKey(key))
  861. return true;
  862. if (!newCheckedPackets[key].Item1.Equals(lastCheckedPackets[key].Item1))
  863. return true;
  864. if (!newCheckedPackets[key].Item2.Equals(lastCheckedPackets[key].Item2))
  865. return true;
  866. }
  867. return false;
  868. }
  869. private void UpdateSelectedKanban(bool uncheckothers = true)
  870. {
  871. if (CurrentKanban != null)
  872. {
  873. var others = Kanbans.Where(x => x.Checked && x.Assignee != CurrentKanban.Assignee).ToArray();
  874. foreach (var other in others)
  875. other.Checked = false;
  876. others = Kanbans.Where(x => x.Checked).ToArray();
  877. if (others.Length == 1)
  878. others[0].Checked = false;
  879. others = Kanbans.Where(x => x.IsSelected).ToArray();
  880. foreach (var other in others)
  881. {
  882. other.IsSelected = false;
  883. var pkt = KanbanToPacket(other);
  884. if (pkt != null)
  885. //var stg = Stages.FirstOrDefault(x => x.ManufacturingPacketLink.ID.Equals(pkt.ID));
  886. other.SelectedColor = other.ColorKey;
  887. else
  888. other.SelectedColor = other.ColorKey;
  889. }
  890. CurrentKanban.Checked = true;
  891. CurrentKanban.IsSelected = true;
  892. CurrentKanban.SelectedColor = SELECTED_COLOR;
  893. }
  894. //UpdateTimeTracking();
  895. Kanban.ItemsSource = null;
  896. Kanban.ItemsSource = Kanbans.OrderBy(x => x.Assignee).ThenByDescending(x => x.Tags.Length).ThenBy(x => x.DueDate).ThenBy(x => x.JobName);
  897. LoadKanban();
  898. }
  899. private void CardChecked(object sender, RoutedEventArgs e)
  900. {
  901. //UpdateTimeTracking();
  902. try
  903. {
  904. var chk = sender as CheckBox;
  905. var border = (chk.Parent as Grid).Parent as Border;
  906. var kanban = border.Tag as ManufacturingKanban;
  907. kanban.Checked = chk.IsChecked == true;
  908. if (kanban.Checked)
  909. foreach (var other in Kanbans.Where(x => x.Assignee != kanban.Assignee))
  910. other.Checked = false;
  911. }
  912. catch (Exception err)
  913. {
  914. Logger.Send(LogType.Error, "", string.Format("*** Unknown Error: {0}\n{1}", err.Message, err.StackTrace));
  915. }
  916. //Kanban.ItemsSource = null;
  917. //Kanban.ItemsSource = Kanbans.OrderBy(x => x.Assignee).ThenByDescending(x => x.Tags.Length).ThenBy(x => x.DueDate).ThenBy(x => x.JobName);
  918. }
  919. private void CompleteButton_Click(object sender, RoutedEventArgs e)
  920. {
  921. var packets = new List<ManufacturingPacket>();
  922. foreach (var kanban in Kanbans.Where(x => x.Checked || x.IsSelected))
  923. {
  924. var packet = KanbanToPacket(kanban);
  925. if (packet.StageLink.SectionID != CurrentSection.ID)
  926. {
  927. MessageBox.Show("You cannot complete a distributed packet that is not yet progressed to your station!");
  928. return;
  929. }
  930. var qacomplete = true;
  931. var stage = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(packet.ID)
  932. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID)
  933. .Equals(settings.Section));
  934. if (stage != null)
  935. {
  936. var qadata = stage.Get<ManufacturingPacketStage, string>(c => c.FormData);
  937. var questions = LoadQAQuestions(packet, true, true);
  938. if (questions != null)
  939. {
  940. var answers = string.IsNullOrWhiteSpace(qadata)
  941. ? new Dictionary<Guid, object>()
  942. : Serialization.Deserialize<Dictionary<Guid, object>>(qadata);
  943. foreach (var question in questions)
  944. {
  945. var parameters = question.ParseParameters();
  946. var bAnswerRequired = question.Answer != QAAnswer.Comment &&
  947. (!parameters.ContainsKey("Default") || string.IsNullOrWhiteSpace(parameters["Default"]));
  948. if (bAnswerRequired && (!answers.ContainsKey(question.ID) || answers[question.ID] == null))
  949. {
  950. qacomplete = false;
  951. break;
  952. }
  953. }
  954. }
  955. }
  956. if (!qacomplete)
  957. {
  958. MessageBox.Show("You must complete the required QA checks before completing these items!");
  959. return;
  960. }
  961. packets.Add(packet);
  962. }
  963. Progress.Show("Scanning Packets");
  964. //Filter<ManufacturingPacketStage> stgflt = new Filter<ManufacturingPacketStage>(x => x.ID).IsEqualTo(CoreUtils.FullGuid);
  965. //foreach (var pkt in packets)
  966. // stgflt = stgflt.Or(x => x.ManufacturingPacketLink.ID).IsEqualTo(pkt.ID);
  967. //ManufacturingPacketStage[] stgs = new Client<ManufacturingPacketStage>().Load(stgflt, new SortOrder<ManufacturingPacketStage>(x => x.Sequence));
  968. var stage_updates = new List<ManufacturingPacketStage>();
  969. for (var i = 0; i < packets.Count; i++)
  970. {
  971. var packet = packets[i];
  972. var stagerows = Stages.Rows.Where(r => r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(packet.ID));
  973. var stages = stagerows.Select(r => r.ToObject<ManufacturingPacketStage>());
  974. //var stages = stgs.Where(x => x.ManufacturingPacketLink.ID.Equals(packet.ID));
  975. long sequence = 0;
  976. // Send the update to the server
  977. var stage = stages.FirstOrDefault(x => x.ID.Equals(packet.StageLink.ID));
  978. if (stage != null)
  979. {
  980. stage.Completed = DateTime.Now;
  981. if(stage.Form.ID == Guid.Empty)
  982. {
  983. stage.Form.ID = CurrentSection.QAForm.ID;
  984. }
  985. if(stage.Form.ID != Guid.Empty)
  986. {
  987. stage.FormCompleted = DateTime.Now;
  988. stage.FormCompletedBy.ID = ClientFactory.UserGuid;
  989. }
  990. stage.PercentageComplete = 100.0F;
  991. if (!stage_updates.Contains(stage))
  992. stage_updates.Add(stage);
  993. sequence = stage.Sequence;
  994. }
  995. // Also update the local copy (hacky hacky)
  996. var stagerow = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(c => c.ID).Equals(packet.StageLink.ID));
  997. if (stagerow != null)
  998. {
  999. stagerow.Set<ManufacturingPacketStage, DateTime>(x => x.Completed, DateTime.Now);
  1000. stagerow.Set<ManufacturingPacketStage, DateTime>(x => x.FormCompleted, DateTime.Now);
  1001. stagerow.Set<ManufacturingPacketStage, Guid>(x => x.FormCompletedBy.ID, ClientFactory.UserGuid);
  1002. stagerow.Set<ManufacturingPacketStage, double>(x => x.PercentageComplete, 100.0F);
  1003. }
  1004. // Update the pointer to the next stage
  1005. // But only if it's at your station already
  1006. // This takes into account distributed packets that may not be completed elsewhere
  1007. if (packet.StageLink.SectionID == CurrentSection.ID)
  1008. {
  1009. stage = stages.Where(x => x.Sequence > sequence).FirstOrDefault();
  1010. // If this is a distributed packet, then it may already have data allocated to it
  1011. // We on;y want to reset the next stage if it's not distributed
  1012. if (!packet.Distributed)
  1013. if (stage != null)
  1014. {
  1015. stage.QualityStatus = QualityStatus.NotChecked;
  1016. stage.QualityNotes = "";
  1017. stage.Station = 0;
  1018. stage.Started = DateTime.MinValue;
  1019. stage.Completed = DateTime.MinValue;
  1020. stage.PercentageComplete = 0.0F;
  1021. if (!stage_updates.Contains(stage))
  1022. stage_updates.Add(stage);
  1023. }
  1024. }
  1025. packet.StageLink.ID = stage == null ? CoreUtils.FullGuid : stage.ID;
  1026. packet.Issued = !packet.StageLink.IsValid() ? DateTime.MinValue : packet.Issued.IsEmpty() ? DateTime.Now : packet.Issued;
  1027. packet.Completed = packet.StageLink.ID.Equals(CoreUtils.FullGuid)
  1028. ? packet.Completed.IsEmpty() ? DateTime.Now : packet.Completed
  1029. : DateTime.MinValue;
  1030. }
  1031. Progress.SetMessage("Progressing Items");
  1032. new Client<ManufacturingPacketStage>().Save(stage_updates, "ManufacturingPacket Progressed from Factory Floor");
  1033. new Client<ManufacturingPacket>().Save(packets, "ManufacturingPacket Progressed From Factory Floor");
  1034. //UpdateTimeTracking();
  1035. Heartbeat(new TimeSpan(0L));
  1036. CurrentKanban = null;
  1037. Refresh();
  1038. Progress.Close();
  1039. }
  1040. private void UpdateButtonClick(object sender, RoutedEventArgs e)
  1041. {
  1042. Progress.Show("");
  1043. double percent = 0.0F;
  1044. if (double.TryParse(((Button)sender).Tag.ToString(), out percent))
  1045. {
  1046. var selected = new List<ManufacturingPacket>();
  1047. var stages = new List<ManufacturingPacketStage>();
  1048. foreach (var kanban in Kanbans.Where(x => x.Checked || x.IsSelected))
  1049. {
  1050. selected.Add(KanbanToPacket(kanban));
  1051. var pktid = Guid.Parse(kanban.ID);
  1052. var stagerows = Stages.Rows.Where(r =>
  1053. r.Get<ManufacturingPacketStage, Guid>(c => c.SectionID).Equals(CurrentSection.ID) &&
  1054. r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(pktid));
  1055. stages.AddRange(stagerows.Select(r => r.ToObject<ManufacturingPacketStage>()));
  1056. }
  1057. var updates = new List<ManufacturingPacketStage>();
  1058. foreach (var stage in stages)
  1059. {
  1060. var bChanged = false;
  1061. if (stage.Started == DateTime.MinValue)
  1062. {
  1063. stage.Started = DateTime.Now;
  1064. bChanged = true;
  1065. }
  1066. if (stage.PercentageComplete != percent)
  1067. {
  1068. stage.PercentageComplete = percent;
  1069. bChanged = true;
  1070. }
  1071. if (stage.Station != -1 && stage.Station != CurrentStation)
  1072. {
  1073. stage.Station = CurrentStation;
  1074. bChanged = true;
  1075. }
  1076. if (bChanged)
  1077. updates.Add(stage);
  1078. }
  1079. new Client<ManufacturingPacketStage>().Save(updates, "Progress Updated from Factory Floor");
  1080. DoRefresh(true);
  1081. }
  1082. Progress.Close();
  1083. }
  1084. private void TearOffButton_Click(object sender, RoutedEventArgs e)
  1085. {
  1086. var doc = PDFEditor.Document;
  1087. var window = new ThemableWindow();
  1088. var editor = new PDFEditorControl();
  1089. editor.SaveAllowed = Security.IsAllowed<CanSaveFactoryFloorDrawings>();
  1090. editor.Document = doc;
  1091. window.Content = editor;
  1092. window.Show();
  1093. }
  1094. private void BarcodesButton_Click(object sender, RoutedEventArgs e)
  1095. {
  1096. //List<ManufacturingPacket> updates = new List<ManufacturingPacket>();
  1097. var checkedpackets =
  1098. Packets.Rows.Where(p => Kanbans.Any(k => k.Checked && p.Get<ManufacturingPacket, Guid>(c => c.ID).ToString().Equals(k.ID)));
  1099. if (!checkedpackets.Any())
  1100. {
  1101. MessageBox.Show("Please select a packet before continuing");
  1102. return;
  1103. }
  1104. Progress.Show("");
  1105. Filter<ManufacturingPacket> pktflt = null;
  1106. foreach (var row in checkedpackets)
  1107. pktflt = pktflt == null
  1108. ? new Filter<ManufacturingPacket>(x => x.ID).IsEqualTo(row.Get<ManufacturingPacket, Guid>(c => c.ID))
  1109. : pktflt.Or(x => x.ID).IsEqualTo(row.Get<ManufacturingPacket, Guid>(c => c.ID));
  1110. var pkts = new Client<ManufacturingPacket>().Load(pktflt);
  1111. foreach (var pkt in pkts)
  1112. //var pkt = row.ToObject<ManufacturingPacket>();
  1113. if (pkt.BarcodePrinted == DateTime.MinValue)
  1114. pkt.BarcodePrinted = DateTime.Now;
  1115. //updates.Add(pkt);
  1116. //if (updates.Any())
  1117. //{
  1118. Progress.SetMessage("Creating Barcodes");
  1119. new Client<ManufacturingPacket>().Save(pkts, "Delivery Barcodes created");
  1120. //}
  1121. Filter<DeliveryItem> filter = null;
  1122. foreach (var row in checkedpackets)
  1123. {
  1124. var pktid = row.Get<ManufacturingPacket, Guid>(c => c.ID);
  1125. if (filter == null)
  1126. filter = new Filter<DeliveryItem>(x => x.ManufacturingPacketLink.ID).IsEqualTo(pktid);
  1127. else
  1128. filter = filter.Or(x => x.ManufacturingPacketLink.ID).IsEqualTo(pktid);
  1129. }
  1130. var dataenvironment = new Dictionary<Type, CoreTable>();
  1131. var model = new DeliveryItemDataModel(filter);
  1132. var Group = Packets.Rows.Select(r => r.Get<ManufacturingPacket, string>(c => c.Group)).FirstOrDefault();
  1133. var BaseReportName = "Print Bar Codes";
  1134. var sectionName = "Delivery Items";
  1135. var repflt = new Filter<ReportTemplate>(x => x.Name).IsEqualTo(BaseReportName);
  1136. var DesiredReportName = BaseReportName + (!string.IsNullOrWhiteSpace(Group) ? " - " + Group : "");
  1137. if (DesiredReportName != BaseReportName)
  1138. repflt = repflt.Or(x => x.Name).IsEqualTo(DesiredReportName);
  1139. var repflt2 = new Filter<ReportTemplate>(x => x.DataModel).IsEqualTo(model.Name).And(x => x.Section).IsEqualTo(sectionName);
  1140. repflt2.Ands.Add(repflt);
  1141. var templates = new Client<ReportTemplate>().Load(repflt2);
  1142. var report = templates.FirstOrDefault(x => x.Name == DesiredReportName);
  1143. if (report == null)
  1144. report = templates.FirstOrDefault(x => x.Name == BaseReportName);
  1145. if (report == null)
  1146. {
  1147. Progress.Close();
  1148. MessageBox.Show("Ready To Go -> [Print Bar Codes] report does not exist!");
  1149. Refresh();
  1150. }
  1151. Progress.Close();
  1152. ReportUtils.PreviewReport(report, model, false, Security.IsAllowed<CanDesignReports>());
  1153. Refresh();
  1154. }
  1155. private void ViewDrawing(object sender, RoutedEventArgs e)
  1156. {
  1157. var button = (Button)sender;
  1158. SetCurrentButton(button);
  1159. var doc = button.Tag as IEntityLink;
  1160. LoadDrawing(doc.ID);
  1161. }
  1162. private void Search_KeyUp(object sender, KeyEventArgs e)
  1163. {
  1164. SearchFilter = Search.Text;
  1165. if (e.Key == Key.Enter || string.IsNullOrEmpty(Search.Text))
  1166. {
  1167. ClearAllSelections();
  1168. ReloadPackets(true);
  1169. }
  1170. }
  1171. //private void SwimlaneCheckBox_Checked(object sender, RoutedEventArgs e)
  1172. //{
  1173. // CheckBox chk = sender as CheckBox;
  1174. // String tag = chk.Tag as String;
  1175. // foreach (var packet in Kanbans)
  1176. // packet.Checked = (packet.Assignee == tag) ? !packet.Checked : false;
  1177. // Kanban.ItemsSource = null;
  1178. // Kanban.ItemsSource = Kanbans.OrderBy(x => x.Assignee).ThenByDescending(x => x.Tags.Length).ThenBy(x => x.DueDate).ThenBy(x => x.JobName);
  1179. //}
  1180. //private void CardBorder_PreviewMouseWheel(object sender, MouseWheelEventArgs e)
  1181. //{
  1182. // e.Handled = true;
  1183. // var eventArg = new MouseWheelEventArgs(e.MouseDevice, e.Timestamp, e.Delta);
  1184. // eventArg.RoutedEvent = UIElement.MouseWheelEvent;
  1185. // eventArg.Source = sender;
  1186. // var tgts = FindVisualChildren<UIElement>(Kanban.Columns[0]);
  1187. // foreach (var tgt in tgts)
  1188. // tgt.RaiseEvent(eventArg);
  1189. //}
  1190. private void PacketMenu_Opened(object sender, RoutedEventArgs e)
  1191. {
  1192. var menu = sender as ContextMenu;
  1193. var setcurrent = menu.Items[0] as MenuItem;
  1194. var setpending = menu.Items[1] as MenuItem;
  1195. var actionseparator = menu.Items[2] as Separator;
  1196. var settrolley = menu.Items[3] as MenuItem;
  1197. var cleartrolley = menu.Items[4] as MenuItem;
  1198. var trolleyseparator = menu.Items[5] as Separator;
  1199. var setpriority = menu.Items[6] as MenuItem;
  1200. var clearpriority = menu.Items[7] as MenuItem;
  1201. var editissues = menu.Items[8] as MenuItem;
  1202. var sethold = menu.Items[9] as MenuItem;
  1203. var clearhold = menu.Items[10] as MenuItem;
  1204. var setdistributed = menu.Items[11] as MenuItem;
  1205. var cleardistributed = menu.Items[12] as MenuItem;
  1206. var distseparator = menu.Items[13] as Separator;
  1207. var setshared = menu.Items[14] as MenuItem;
  1208. var clearshared = menu.Items[15] as MenuItem;
  1209. var movepacket = menu.Items[16] as MenuItem;
  1210. var model = menu.Tag as ManufacturingKanban;
  1211. if (!model.Checked)
  1212. {
  1213. foreach (var kanban in Kanbans.Where(x => x.Checked).ToArray())
  1214. kanban.Checked = false;
  1215. model.Checked = true;
  1216. //Kanban.ItemsSource = null;
  1217. //Kanban.ItemsSource = Kanbans.OrderBy(x => x.Assignee).ThenByDescending(x => x.Tags.Length).ThenBy(x => x.DueDate).ThenBy(x => x.JobName);
  1218. }
  1219. var checkedkanbans = Kanbans.Where(k => k.Checked).Select(x => Guid.Parse(x.ID));
  1220. var checkedpackets = Packets.Rows.Where(r => checkedkanbans.Contains(r.Get<ManufacturingPacket, Guid>(c => c.ID)));
  1221. var checkedstages = Stages.Rows.Where(r => checkedkanbans.Contains(r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID))
  1222. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID)
  1223. .Equals(settings.Section));
  1224. // Only if we right-click on a checked packet??
  1225. if (checkedpackets.Any() && checkedstages.Any())
  1226. {
  1227. var iStation = checkedstages.First().Get<ManufacturingPacketStage, int>(c => c.Station);
  1228. setcurrent.Visibility = iStation == 0 ? Visibility.Visible : Visibility.Collapsed;
  1229. setpending.Visibility = iStation > 0 ? Visibility.Visible : Visibility.Collapsed;
  1230. actionseparator.Visibility = Visibility.Collapsed;
  1231. settrolley.Visibility = Visibility.Visible;
  1232. cleartrolley.Visibility =
  1233. checkedpackets.Select(r => r.Get<ManufacturingPacket, string>(x => x.Trolleys)).Any(x => !string.IsNullOrWhiteSpace(x))
  1234. ? Visibility.Visible
  1235. : Visibility.Collapsed;
  1236. setpriority.Visibility = Visibility.Collapsed; // (packet != null) && packet.Priority ? Visibility.Collapsed : Visibility.Visible;
  1237. clearpriority.Visibility = Visibility.Collapsed; // (packet != null) && packet.Priority ? Visibility.Visible : Visibility.Collapsed;
  1238. editissues.Visibility = Visibility.Visible;
  1239. sethold.Visibility =
  1240. false //Security.IsAllowed<CanManagePacketHolds>() && checkedpackets.Any(r => !r.Get<ManufacturingPacket, bool>(c => c.OnHold))
  1241. ? Visibility.Visible
  1242. : Visibility.Collapsed;
  1243. clearhold.Visibility =
  1244. false //Security.IsAllowed<CanManagePacketHolds>() && checkedpackets.Any(r => r.Get<ManufacturingPacket, bool>(c => c.OnHold))
  1245. ? Visibility.Visible
  1246. : Visibility.Collapsed;
  1247. setdistributed.Visibility = Security.IsAllowed<CanDistributePackets>() &&
  1248. checkedpackets.Any(r => r.Get<ManufacturingPacket, bool>(c => c.Distributed) == false)
  1249. ? Visibility.Visible
  1250. : Visibility.Collapsed;
  1251. cleardistributed.Visibility = Security.IsAllowed<CanDistributePackets>() &&
  1252. checkedpackets.Any(r => r.Get<ManufacturingPacket, bool>(c => c.Distributed))
  1253. ? Visibility.Visible
  1254. : Visibility.Collapsed;
  1255. distseparator.Visibility = setdistributed.Visibility == Visibility.Visible || cleardistributed.Visibility == Visibility.Visible
  1256. ? Visibility.Visible
  1257. : Visibility.Collapsed;
  1258. setshared.Visibility = iStation > 0 ? Visibility.Visible : Visibility.Collapsed;
  1259. clearshared.Visibility = iStation < 0 ? Visibility.Visible : Visibility.Collapsed;
  1260. movepacket.Visibility = iStation > 0 ? Visibility.Visible : Visibility.Collapsed;
  1261. setcurrent.Tag = checkedstages;
  1262. setpending.Tag = checkedstages;
  1263. setpriority.Tag = checkedpackets;
  1264. clearpriority.Tag = checkedpackets;
  1265. editissues.Tag = checkedpackets;
  1266. sethold.Tag = checkedpackets;
  1267. clearhold.Tag = checkedpackets;
  1268. setdistributed.Tag = checkedpackets;
  1269. cleardistributed.Tag = checkedpackets;
  1270. setshared.Tag = checkedstages;
  1271. clearshared.Tag = checkedstages;
  1272. movepacket.Tag = checkedstages;
  1273. if (movepacket.Visibility == Visibility.Visible)
  1274. {
  1275. movepacket.Items.Clear();
  1276. foreach (var station in Station.Items)
  1277. {
  1278. var sub = new MenuItem { Header = station, Tag = checkedstages };
  1279. sub.Click += MovePacketClick;
  1280. movepacket.Items.Add(sub);
  1281. sub.Visibility = movepacket.Items.Count == CurrentStation ? Visibility.Collapsed : Visibility.Visible;
  1282. }
  1283. }
  1284. settrolley.Tag = Kanbans.Where(k => k.Checked);
  1285. cleartrolley.Tag = Kanbans.Where(k => k.Checked);
  1286. //if (Trolleys == null)
  1287. //{
  1288. // Trolleys = new Client<ManufacturingTrolley>().Query(
  1289. // LookupFactory.DefineFilter<ManufacturingTrolley>(),
  1290. // LookupFactory.DefineColumns<ManufacturingTrolley>(),
  1291. // LookupFactory.DefineSort<ManufacturingTrolley>()
  1292. // );
  1293. //}
  1294. //if (Trolleys.Rows.Any())
  1295. //{
  1296. // settrolley.Visibility = Visibility.Visible;
  1297. // settrolley.Items.Clear();
  1298. // foreach (var row in Trolleys.Rows)
  1299. // {
  1300. // MenuItem sub = new MenuItem() { Header = row.Get<ManufacturingTrolley,String>(x=>x.Code) + ": "+ row.Get<ManufacturingTrolley, String>(x => x.Description), Tag = row.Get<ManufacturingTrolley,Guid>(x=>x.ID) };
  1301. // sub.Click += SetTrolleyClick;
  1302. // settrolley.Items.Add(sub);
  1303. // }
  1304. //}
  1305. //else
  1306. // settrolley.Visibility = Visibility.Collapsed;
  1307. }
  1308. // Disable everything
  1309. }
  1310. private void SetPacketToTrolley(string trolley, params ManufacturingPacket[] packets)
  1311. {
  1312. var updates = new List<ManufacturingPacket>();
  1313. foreach (var packet in packets)
  1314. if (string.Equals(packet.Trolleys, trolley))
  1315. {
  1316. packet.Trolleys = trolley;
  1317. updates.Add(packet);
  1318. }
  1319. if (updates.Any())
  1320. using (new WaitCursor())
  1321. {
  1322. new Client<ManufacturingPacket>().Save(updates, string.Format("Moved Packet to Trolley {0}", trolley), (o, e) => { });
  1323. foreach (var update in updates)
  1324. {
  1325. var pktrow = Packets.Rows.FirstOrDefault(r => r.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(update.ID));
  1326. var stagerow = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(update.ID)
  1327. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID)
  1328. .Equals(settings.Section));
  1329. //ManufacturingPacketStage stage = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(x => x.Parent.ID).Equals(update.ID))?.ToObject<ManufacturingPacketStage>();
  1330. var kanban = Kanbans.FirstOrDefault(x => x.ID.Equals(update.ID.ToString()));
  1331. LoadModel(kanban, pktrow, stagerow, true);
  1332. }
  1333. }
  1334. }
  1335. private void SetTrolleyClick(object sender, RoutedEventArgs e)
  1336. {
  1337. var kanbans = (sender as MenuItem).Tag as IEnumerable<ManufacturingKanban>;
  1338. List<string> trolleyPrePopulate = new List<string>();
  1339. foreach (ManufacturingKanban kanban in kanbans)
  1340. {
  1341. if (!string.IsNullOrWhiteSpace(kanban.Trolleys))
  1342. {
  1343. if (kanban.Trolleys.Contains(','))
  1344. {
  1345. var splitTrolleys = kanban.Trolleys.Split(',');
  1346. foreach (string s in splitTrolleys)
  1347. {
  1348. if (!trolleyPrePopulate.Contains(s))
  1349. {
  1350. trolleyPrePopulate.Add(s);
  1351. }
  1352. }
  1353. }
  1354. else
  1355. {
  1356. if (!trolleyPrePopulate.Contains(kanban.Trolleys))
  1357. {
  1358. trolleyPrePopulate.Add(kanban.Trolleys);
  1359. }
  1360. }
  1361. }
  1362. }
  1363. var dlg = new MultiSelectWindow(typeof(ManufacturingTrolley), new Guid[] { }, trolleyPrePopulate);
  1364. if (dlg.ShowDialog() == true)
  1365. using (new WaitCursor())
  1366. {
  1367. var trolleys = new List<string>();
  1368. foreach (ManufacturingTrolley value in dlg.Values)
  1369. trolleys.Add(value.Code);
  1370. var updates = new List<ManufacturingPacket>();
  1371. foreach (var kanban in kanbans)
  1372. {
  1373. var pktrow = Packets.Rows.FirstOrDefault(r => r.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(Guid.Parse(kanban.ID)));
  1374. var stagerow = Stages.Rows.FirstOrDefault(r =>
  1375. r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(Guid.Parse(kanban.ID))
  1376. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID).Equals(settings.Section));
  1377. var packet = pktrow.ToObject<ManufacturingPacket>();
  1378. packet.Trolleys = string.Join(",", trolleys);
  1379. updates.Add(packet);
  1380. pktrow.Set<ManufacturingPacket, string>(x => x.Trolleys, packet.Trolleys);
  1381. LoadModel(kanban, pktrow, stagerow, true);
  1382. }
  1383. if (updates.Any())
  1384. {
  1385. new Client<ManufacturingPacket>().Save(updates, "Set Trolley to " + (sender as MenuItem).Header, (objects, error) => { });
  1386. var src = Kanban.ItemsSource;
  1387. Kanban.ItemsSource = null;
  1388. Kanban.ItemsSource = src;
  1389. }
  1390. }
  1391. //Refresh();
  1392. }
  1393. private void ClearTrolleyClick(object sender, RoutedEventArgs e)
  1394. {
  1395. var kanbans = (sender as MenuItem).Tag as IEnumerable<ManufacturingKanban>;
  1396. using (new WaitCursor())
  1397. {
  1398. var updates = new List<ManufacturingPacket>();
  1399. foreach (var kanban in kanbans)
  1400. {
  1401. var pktrow = Packets.Rows.FirstOrDefault(r => r.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(Guid.Parse(kanban.ID)));
  1402. var stagerow = Stages.Rows.FirstOrDefault(r =>
  1403. r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(Guid.Parse(kanban.ID))
  1404. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID)
  1405. .Equals(settings.Section));
  1406. var packet = pktrow.ToObject<ManufacturingPacket>();
  1407. packet.Trolleys = "";
  1408. updates.Add(packet);
  1409. pktrow.Set<ManufacturingPacket, string>(x => x.Trolleys, packet.Trolleys);
  1410. LoadModel(kanban, pktrow, stagerow, true);
  1411. }
  1412. if (updates.Any())
  1413. {
  1414. new Client<ManufacturingPacket>().Save(updates, "Removed From Trolley", (objects, error) => { });
  1415. var src = Kanban.ItemsSource;
  1416. Kanban.ItemsSource = null;
  1417. Kanban.ItemsSource = src;
  1418. }
  1419. }
  1420. }
  1421. private void SetCurrent_Click(object sender, RoutedEventArgs e)
  1422. {
  1423. var rows = (sender as MenuItem).Tag as IEnumerable<CoreRow>;
  1424. var stages = rows.Select(x => x.ToObject<ManufacturingPacketStage>()).ToArray();
  1425. AddPacketToCurrentWorkload(stages);
  1426. CurrentKanban = Kanbans.FirstOrDefault(x => x.ID.Equals(stages.First().Parent.ID.ToString()));
  1427. UpdateSelectedKanban(false);
  1428. }
  1429. private void AddPacketToCurrentWorkload(params ManufacturingPacketStage[] stages)
  1430. {
  1431. var updates = new List<ManufacturingPacketStage>();
  1432. foreach (var stage in stages)
  1433. if (stage.Station == 0)
  1434. {
  1435. stage.Station = CurrentStation;
  1436. stage.Started = DateTime.Now;
  1437. updates.Add(stage);
  1438. }
  1439. if (updates.Any())
  1440. using (new WaitCursor())
  1441. {
  1442. new Client<ManufacturingPacketStage>().Save(updates, string.Format("Moved Packet to Station {0}", CurrentStation), (o, e) => { });
  1443. foreach (var update in updates)
  1444. {
  1445. var pktrow = Packets.Rows.FirstOrDefault(r => r.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(update.Parent.ID));
  1446. var stagerow = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(x => x.ID).Equals(update.ID));
  1447. Stages.LoadRow(stagerow, update);
  1448. var kanban = Kanbans.FirstOrDefault(x => x.ID.Equals(update.Parent.ID.ToString()));
  1449. LoadModel(kanban, pktrow, stagerow, true);
  1450. }
  1451. }
  1452. }
  1453. private void SetPending_Click(object sender, RoutedEventArgs e)
  1454. {
  1455. var updates = new List<ManufacturingPacketStage>();
  1456. var rows = (sender as MenuItem).Tag as IEnumerable<CoreRow>;
  1457. foreach (var row in rows)
  1458. {
  1459. var stage = row.ToObject<ManufacturingPacketStage>();
  1460. if (stage.Station != 0)
  1461. {
  1462. stage.Station = 0;
  1463. stage.Started = DateTime.MinValue;
  1464. updates.Add(stage);
  1465. }
  1466. }
  1467. if (updates.Any())
  1468. using (new WaitCursor())
  1469. {
  1470. new Client<ManufacturingPacketStage>().Save(updates, string.Format("Removed Packet from Station {0}", CurrentStation));
  1471. foreach (var update in updates)
  1472. {
  1473. var pktrow = Packets.Rows.FirstOrDefault(r => r.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(update.Parent.ID));
  1474. var stagerow = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(x => x.ID).Equals(update.ID));
  1475. Stages.LoadRow(stagerow, update);
  1476. var kanban = Kanbans.FirstOrDefault(x => x.ID.Equals(update.Parent.ID.ToString()));
  1477. LoadModel(kanban, pktrow, stagerow, true);
  1478. }
  1479. CurrentKanban = Kanbans.FirstOrDefault(x => x.ID.Equals(updates.First().Parent.ID.ToString()));
  1480. UpdateSelectedKanban();
  1481. }
  1482. }
  1483. private void UpdateFlag(object sender, string property, bool value)
  1484. {
  1485. var updates = new List<ManufacturingPacket>();
  1486. var rows = (sender as MenuItem).Tag as IEnumerable<CoreRow>;
  1487. foreach (var row in rows)
  1488. {
  1489. var packet = row.ToObject<ManufacturingPacket>();
  1490. CoreUtils.SetPropertyValue(packet, property, value);
  1491. updates.Add(packet);
  1492. }
  1493. if (updates.Any())
  1494. using (new WaitCursor())
  1495. {
  1496. new Client<ManufacturingPacket>().Save(updates, property + " Flag " + (value ? "Set" : "Cleared"));
  1497. Refresh();
  1498. }
  1499. }
  1500. private void SetPriority_Click(object sender, RoutedEventArgs e)
  1501. {
  1502. UpdateFlag(sender, "Priority", true);
  1503. }
  1504. private void ClearPriority_Click(object sender, RoutedEventArgs e)
  1505. {
  1506. UpdateFlag(sender, "Priority", false);
  1507. }
  1508. private void EditIssues_Click(object sender, RoutedEventArgs e)
  1509. {
  1510. var updates = new List<ManufacturingPacket>();
  1511. var rows = (sender as MenuItem).Tag as IEnumerable<CoreRow>;
  1512. var pkts = rows.Select(x => x.ToObject<ManufacturingPacket>()).ToArray();
  1513. if (new DynamicIssuesEditor(pkts, true).ShowDialog() == true)
  1514. {
  1515. Progress.ShowModal("Updating Issues", progress => { new Client<ManufacturingPacket>().Save(pkts, "Updated Issues"); });
  1516. Refresh();
  1517. }
  1518. // String[] issues = rows.Select(r => r.Get<ManufacturingPacket, String>(c => c.Issues)).Distinct().ToArray();
  1519. // if (issues.Length > 1)
  1520. // {
  1521. // if (MessageBox.Show(
  1522. // "Multiple Issues found! If you continue, these may be lost.\n\nAre you sure you wish to continue?",
  1523. // "Multiple Issues",
  1524. // MessageBoxButton.YesNo) != MessageBoxResult.Yes)
  1525. // return;
  1526. // }
  1527. //
  1528. // string issue = issues.Length > 0 ? issues.First() : "";
  1529. // if (!TextBoxDialog.Execute("Edit Issues", ref issue))
  1530. // return;
  1531. //
  1532. // foreach (var row in rows)
  1533. // {
  1534. // ManufacturingPacket packet = row.ToObject<ManufacturingPacket>();
  1535. // packet.Issues = issue;
  1536. // updates.Add(packet);
  1537. // }
  1538. // if (updates.Any())
  1539. // {
  1540. // using (new WaitCursor())
  1541. // {
  1542. // new Client<ManufacturingPacket>().Save(updates, "Updated Issues");
  1543. // Refresh();
  1544. // }
  1545. // }
  1546. }
  1547. private void SetHold_Click(object sender, RoutedEventArgs e)
  1548. {
  1549. if (MessageBox.Show(
  1550. "Are you sure you wish to set this packet on hold?",
  1551. "Set Hold",
  1552. MessageBoxButton.YesNo) != MessageBoxResult.Yes)
  1553. return;
  1554. UpdateFlag(sender, "OnHold", true);
  1555. }
  1556. private void ClearHold_Click(object sender, RoutedEventArgs e)
  1557. {
  1558. if (MessageBox.Show(
  1559. "Are you sure you wish to release this packet from hold?",
  1560. "Set Hold",
  1561. MessageBoxButton.YesNo) != MessageBoxResult.Yes)
  1562. return;
  1563. UpdateFlag(sender, "OnHold", false);
  1564. }
  1565. private void SetDistributed_Click(object sender, RoutedEventArgs e)
  1566. {
  1567. UpdateFlag(sender, "Distributed", true);
  1568. }
  1569. private void ClearDistributed_Click(object sender, RoutedEventArgs e)
  1570. {
  1571. UpdateFlag(sender, "Distributed", false);
  1572. }
  1573. private void SetShared_Click(object sender, RoutedEventArgs e)
  1574. {
  1575. var updates = new List<ManufacturingPacketStage>();
  1576. var rows = (sender as MenuItem).Tag as IEnumerable<CoreRow>;
  1577. foreach (var row in rows)
  1578. {
  1579. var stage = row.ToObject<ManufacturingPacketStage>();
  1580. if (stage.Station > 0)
  1581. {
  1582. stage.Station = -1;
  1583. updates.Add(stage);
  1584. }
  1585. }
  1586. if (updates.Any())
  1587. using (new WaitCursor())
  1588. {
  1589. new Client<ManufacturingPacketStage>().Save(updates, "Set Packet as Shared");
  1590. Refresh();
  1591. }
  1592. }
  1593. private void ClearShared_Click(object sender, RoutedEventArgs e)
  1594. {
  1595. var updates = new List<ManufacturingPacketStage>();
  1596. var rows = (sender as MenuItem).Tag as IEnumerable<CoreRow>;
  1597. foreach (var row in rows)
  1598. {
  1599. var stage = row.ToObject<ManufacturingPacketStage>();
  1600. if (stage.Station == -1)
  1601. {
  1602. stage.Station = CurrentStation;
  1603. updates.Add(stage);
  1604. }
  1605. }
  1606. if (updates.Any())
  1607. using (new WaitCursor())
  1608. {
  1609. new Client<ManufacturingPacketStage>().Save(updates, "Cleared Shared flag from Packet");
  1610. Refresh();
  1611. }
  1612. }
  1613. private void MovePacketClick(object sender, RoutedEventArgs e)
  1614. {
  1615. var item = sender as MenuItem;
  1616. var station = (item.Parent as MenuItem).Items.IndexOf(item) + 1;
  1617. var updates = new List<ManufacturingPacketStage>();
  1618. var rows = (sender as MenuItem).Tag as IEnumerable<CoreRow>;
  1619. foreach (var row in rows)
  1620. {
  1621. var stage = row.ToObject<ManufacturingPacketStage>();
  1622. stage.Station = station;
  1623. updates.Add(stage);
  1624. }
  1625. if (updates.Any())
  1626. using (new WaitCursor())
  1627. {
  1628. new Client<ManufacturingPacketStage>().Save(updates, "Moved Packet to " + item.Header);
  1629. Refresh();
  1630. }
  1631. }
  1632. public void PDFEditorSettingsChanged(object sender, string linecolor, int fontsize)
  1633. {
  1634. settings.LineColor = linecolor;
  1635. settings.FontSize = fontsize;
  1636. new LocalConfiguration<FactoryFloorLocalSettings>().Save(settings);
  1637. }
  1638. #region Employee Stuff
  1639. private Dictionary<Guid, string> Employees = new() { { Guid.Empty, "" } };
  1640. private Guid myID = Guid.Empty;
  1641. private string myName = "";
  1642. #endregion
  1643. #region ManufacturingPacket Stuff
  1644. //FactorySetup settings = new GlobalConfiguration<FactorySetup>().Load();
  1645. //private Dictionary<Guid, Tuple<String,int>> Sections = new Dictionary<Guid, Tuple<String,int>>();
  1646. private ManufacturingFactory[] Factories; // new ManufacturingFactory[] { };
  1647. private ManufacturingSection[] Sections; // new ManufacturingSection[] { };
  1648. private ManufacturingTrolley[] Trolleys; // new ManufacturingTrolley[] { };
  1649. private CoreTable TemplateStages;
  1650. //private ManufacturingTemplateStage[] TemplateStages = null;
  1651. private CoreTable Questions;
  1652. //private IList<CoreRow> AllStages = new CoreRow[] { };
  1653. private CoreTable Packets;
  1654. private CoreTable Stages;
  1655. /// <remarks>
  1656. /// <see langword="null"/> if there are no <see cref="ManufacturingSection"/>s in the system.
  1657. /// </remarks>
  1658. private ManufacturingSection? CurrentSection;
  1659. private int CurrentStation;
  1660. private readonly bool PendingVisible = true;
  1661. private ManufacturingLostTime LostTime;
  1662. #endregion
  1663. #region Stuff to Do with Kanbans
  1664. public ObservableCollection<ManufacturingKanban> Kanbans { get; set; }
  1665. private string CurrentKanbanID = "";
  1666. private ManufacturingKanban CurrentKanban
  1667. {
  1668. get { return Kanbans.FirstOrDefault(x => x.ID == CurrentKanbanID); }
  1669. set => CurrentKanbanID = value != null ? value.ID : "";
  1670. }
  1671. private string SearchFilter = "";
  1672. //Border LastCard = null;
  1673. //Border[] CheckedPackets
  1674. //{
  1675. // get {
  1676. // List<Border> packets = new List<Border>();
  1677. // var cards = Kanban.Columns[0].Cards.Where(x => ((ManufacturingKanban)x.Content).Checked == true);
  1678. // foreach (var card in cards)
  1679. // {
  1680. // var prop = card.GetType().GetProperty("TemplateChild", BindingFlags.NonPublic | BindingFlags.Instance);
  1681. // var val = prop.GetValue(card) as DependencyObject;
  1682. // var border = FindVisualChildren<Border>(val).FirstOrDefault();
  1683. // packets.Add(border);
  1684. // }
  1685. // return packets.ToArray();
  1686. // }
  1687. //}
  1688. private void ClearAllSelections()
  1689. {
  1690. foreach (var kanban in Kanbans.Where(x => x.Checked))
  1691. kanban.Checked = false;
  1692. CurrentKanban = null;
  1693. //var cards = Kanban.Columns[0].Cards.Where(x => ((ManufacturingKanban)x.Content).Checked == true).ToArray();
  1694. //foreach (var card in cards)
  1695. // ((ManufacturingKanban)card.Content).Checked = false;
  1696. //CurrentKanban = null;
  1697. UpdateSelectedKanban();
  1698. }
  1699. private ManufacturingKanban GetKanban(Border border)
  1700. {
  1701. return border.Tag as ManufacturingKanban;
  1702. }
  1703. //private Border GetBorder(ManufacturingKanban kanban)
  1704. //{
  1705. // var card = Kanban.Columns[0].Cards.FirstOrDefault(x => ((ManufacturingKanban)x.Content) == kanban);
  1706. // var prop = card.GetType().GetProperty("TemplateChild", BindingFlags.NonPublic | BindingFlags.Instance);
  1707. // var val = prop.GetValue(card) as DependencyObject;
  1708. // return FindVisualChildren<Border>(val).FirstOrDefault();
  1709. //}
  1710. //private void CheckKanban(Border card, bool ischecked)
  1711. //{
  1712. // var chk = card.FindName("IsChecked") as CheckBox;
  1713. // chk.IsChecked = ischecked;
  1714. // card.ApplyTemplate();
  1715. //}
  1716. private void SelectKanban(Border card, bool isselected)
  1717. {
  1718. var img = card.FindName("IsCurrent") as Image;
  1719. img.Source = isselected ? starred : null;
  1720. }
  1721. private ManufacturingPacket KanbanToPacket(ManufacturingKanban kanban)
  1722. {
  1723. if (Packets == null)
  1724. return null;
  1725. var row = Packets.Rows.FirstOrDefault(r => r.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(Guid.Parse(kanban.ID)));
  1726. if (row == null)
  1727. return null;
  1728. return row.ToObject<ManufacturingPacket>();
  1729. }
  1730. #endregion
  1731. #region Shipping Rack Stuff
  1732. private CoreTable Shipments;
  1733. private CoreTable DeliveryItems;
  1734. private readonly ObservableList<DeliveryItem> rackcontents = new();
  1735. private Guid rackid = Guid.Empty;
  1736. private string rackbarcode = "";
  1737. #endregion
  1738. #region Refresh / Reload
  1739. private string GetQualityStatus(QualityStatus status)
  1740. {
  1741. if (status == QualityStatus.Passed)
  1742. return "PASSED";
  1743. if (status == QualityStatus.Skipped)
  1744. return "SKIPPED";
  1745. if (status == QualityStatus.PassedWithIssues)
  1746. return "ISSUES";
  1747. if (status == QualityStatus.Failed)
  1748. return "FAILED";
  1749. return "QA";
  1750. }
  1751. private BitmapImage GetBarCode(CoreRow row)
  1752. {
  1753. var type = row.Get<ManufacturingPacket, BarcodeType>(c => c.BarcodeType);
  1754. if (!row.Get<ManufacturingPacket, DateTime>(c => c.BarcodePrinted).IsEmpty())
  1755. return type == BarcodeType.Grouped ? grouped : barcode;
  1756. if (type == BarcodeType.None)
  1757. return disabled;
  1758. return null;
  1759. }
  1760. private void CreateKanban(CoreRow pktrow, CoreRow stagerow, bool IsChecked)
  1761. {
  1762. var id = pktrow.Get<ManufacturingPacket, Guid>(c => c.ID);
  1763. //var stagerow = Stages.Rows.FirstOrDefault(r=>r.Get<ManufacturingPacketStage,Guid>(c=>c.Parent.ID).Equals(id));
  1764. if (stagerow == null)
  1765. return;
  1766. var station = stagerow.Get<ManufacturingPacketStage, int>(c => c.Station);
  1767. //var stage = stagerow.ToObject<ManufacturingPacketStage>();
  1768. if ((PendingVisible && station.Equals(0)) || station.Equals(-1) || station.Equals(CurrentStation))
  1769. {
  1770. var model = new ManufacturingKanban();
  1771. LoadModel(model, pktrow, stagerow, IsChecked);
  1772. Kanbans.Add(model);
  1773. }
  1774. }
  1775. private void LoadModel(ManufacturingKanban model, CoreRow packet, CoreRow stage, bool IsChecked)
  1776. {
  1777. var station = stage.Get<ManufacturingPacketStage, int>(c => c.Station);
  1778. var Pending = station == 0;
  1779. var quality = stage.Get<ManufacturingPacketStage, QualityStatus>(c => c.QualityStatus);
  1780. var percentage = stage.Get<ManufacturingPacketStage, double>(c => c.PercentageComplete);
  1781. var packetid = packet.Get<ManufacturingPacket, Guid>(c => c.ID);
  1782. var title = packet.Get<ManufacturingPacket, string>(c => c.Title);
  1783. var watermark = packet.Get<ManufacturingPacket, string>(c => c.WaterMark);
  1784. var issues = packet.Get<ManufacturingPacket, string>(c => c.Issues);
  1785. var trolleys = packet.Get<ManufacturingPacket, string>(c => c.Trolleys);
  1786. var serial = packet.Get<ManufacturingPacket, string>(c => c.Serial);
  1787. var quantity = packet.Get<ManufacturingPacket, int>(c => c.Quantity);
  1788. var barcodes = packet.Get<ManufacturingPacket, int>(c => c.BarcodeQty);
  1789. var setoutnumber = packet.Get<ManufacturingPacket, string>(c => c.SetoutLink.Number);
  1790. var jobname = packet.Get<ManufacturingPacket, string>(c => c.SetoutLink.JobLink.Name);
  1791. var duedate = packet.Get<ManufacturingPacket, DateTime>(c => c.DueDate);
  1792. var location = packet.Get<ManufacturingPacket, string>(c => c.Location);
  1793. var setoutlocation = packet.Get<ManufacturingPacket, string>(c => c.SetoutLink.Location);
  1794. var suppliercode = packet.Get<ManufacturingPacket, string>(c => c.OrderItem.PurchaseOrderLink.SupplierLink.Code);
  1795. var ponumber = packet.Get<ManufacturingPacket, string>(c => c.OrderItem.PurchaseOrderLink.PONumber);
  1796. var receiveddate = packet.Get<ManufacturingPacket, DateTime>(c => c.OrderItem.ReceivedDate);
  1797. var receivedreference = packet.Get<ManufacturingPacket, string>(c => c.OrderItem.ReceivedReference);
  1798. var priority = packet.Get<ManufacturingPacket, bool>(c => c.Priority);
  1799. var estimateddate = packet.Get<ManufacturingPacket, DateTime>(c => c.EstimatedDate);
  1800. var distributed = packet.Get<ManufacturingPacket, bool>(c => c.Distributed);
  1801. var packetsection = packet.Get<ManufacturingPacket, string>(c => c.StageLink.Section);
  1802. var groupid = packet.Get<ManufacturingPacket, Guid>(c => c.SetoutLink.Group.ID);
  1803. var groupname = packet.Get<ManufacturingPacket, string>(c => c.SetoutLink.Group.Name);
  1804. model.TemplateID = packet.Get<ManufacturingPacket, Guid>(c => c.ManufacturingTemplateLink.ID);
  1805. model.ID = packetid.ToString();
  1806. model.Title = string.Format(
  1807. "{0}: {1}{2}",
  1808. serial,
  1809. quantity != barcodes ? string.Format("{0} x ", quantity) : "",
  1810. title
  1811. );
  1812. if (!string.IsNullOrWhiteSpace(watermark))
  1813. model.Title = "[" + watermark + "] " + model.Title;
  1814. model.Quantity = barcodes;
  1815. model.JobName = string.Format("{0}: {1}", setoutnumber, jobname);
  1816. model.DueDate = duedate;
  1817. model.GroupID = groupid;
  1818. var descrip = new List<string>
  1819. {
  1820. string.IsNullOrEmpty(location) ? setoutlocation : location
  1821. };
  1822. if (packet.IsEntityLinkValid<ManufacturingPacket, PurchaseOrderItemLink>(x => x.OrderItem))
  1823. descrip.Add(string.Format("{0} ({1}) RCVD {2:dd MMM yy} {3}",
  1824. suppliercode,
  1825. ponumber,
  1826. receiveddate,
  1827. string.IsNullOrWhiteSpace(receivedreference) ? "" : ": " + receivedreference
  1828. ));
  1829. model.Description = string.Join("\n", descrip);
  1830. model.Image = GetBarCode(packet);
  1831. //model.IsCurrent = packet.ID.ToString() == CurrentKanbanID ? starred : null;
  1832. model.IsSelected = packet.Get<ManufacturingPacket, Guid>(c => c.ID).ToString() == CurrentKanbanID;
  1833. model.Tags = priority
  1834. ? new[] { "PRIORITY" }
  1835. : new string[] { };
  1836. model.Category = CurrentSection != null
  1837. ? CurrentSection.ID.ToString()
  1838. : ""; // packet.StageLink.SectionID.ToString();
  1839. model.ColorKey = packet.IsEntityLinkValid<ManufacturingPacket, PurchaseOrderItemLink>(x => x.OrderItem)
  1840. ? TREATED_COLOR
  1841. : priority
  1842. ? PRIORITY_COLOR
  1843. : !Pending
  1844. ? GetColor(duedate.IsEmpty()
  1845. ? DateTime.Today
  1846. : duedate, estimateddate.IsEmpty()
  1847. ? DateTime.Today
  1848. : estimateddate)
  1849. : QA_COLOR;
  1850. model.SelectedColor = packet.Get<ManufacturingPacket, Guid>(c => c.ID).ToString() == CurrentKanbanID
  1851. ? SELECTED_COLOR
  1852. : model.ColorKey;
  1853. model.SharedColor = station.Equals(-1)
  1854. ? SHARED_COLOR
  1855. : model.ColorKey;
  1856. model.Checked = IsChecked;
  1857. model.SetoutID = packet.Get<ManufacturingPacket, Guid>(c => c.SetoutLink.ID);
  1858. model.Assignee = Pending
  1859. ? "Not Yet Started"
  1860. : "Current Workload";
  1861. model.Status = !Entity.IsEntityLinkValid<ManufacturingPacket, ManufacturingPacketStageLink>(x => x.StageLink, packet)
  1862. ? " "
  1863. : Pending
  1864. ? "PENDING"
  1865. : string.Format("{0:F0}%", percentage);
  1866. try
  1867. {
  1868. model.Flags = distributed
  1869. ? !packet.Get<ManufacturingPacket, Guid>(c => c.StageLink.SectionID).Equals(CurrentSection.ID)
  1870. ? string.IsNullOrEmpty(packetsection)
  1871. ? ""
  1872. : packetsection.ToUpper().Trim()
  1873. : "DISTRIB"
  1874. : "";
  1875. }
  1876. catch (Exception e)
  1877. {
  1878. Logger.Send(LogType.Error, "", string.Format("*** Unknown Error: {0}\n{1}", e.Message, e.StackTrace));
  1879. }
  1880. model.Issues = issues;
  1881. model.IssuesImage = string.IsNullOrWhiteSpace(issues)
  1882. ? null
  1883. : speechbubble;
  1884. model.Trolleys = trolleys;
  1885. }
  1886. private string GetColor(DateTime duedate, DateTime estdate)
  1887. {
  1888. var color = NOTYETDUE_COLOR;
  1889. if (duedate < estdate)
  1890. color = OVERDUE_COLOR;
  1891. else if (duedate < estdate.AddDays(7))
  1892. color = NEARLYDUE_COLOR;
  1893. return color;
  1894. }
  1895. private void ReloadPackets(bool reloaddata)
  1896. {
  1897. using var profiler = new Profiler(true);
  1898. using (new WaitCursor())
  1899. {
  1900. if (reloaddata)
  1901. {
  1902. var sectionid = CurrentSection != null ? CurrentSection.ID : CoreUtils.FullGuid;
  1903. var stageflt = new Filter<ManufacturingPacketStage>(x => x.Completed).IsEqualTo(DateTime.MinValue)
  1904. .Or(x => x.ManufacturingSectionLink.ID).IsEqualTo(CurrentSection.ID);
  1905. stageflt.Ands.Add(
  1906. new Filter<ManufacturingPacketStage>(x => x.Parent.StageLink.SectionID).IsEqualTo(sectionid)
  1907. .Or(x => x.Parent.Distributed).IsEqualTo(true)
  1908. );
  1909. var stageQuery = new KeyedQueryDef<ManufacturingPacketStage>(
  1910. stageflt,
  1911. new Columns<ManufacturingPacketStage>(x => x.ID)
  1912. .Add(x => x.Parent.ID)
  1913. .Add(x => x.Station)
  1914. .Add(x => x.QualityStatus)
  1915. .Add(x => x.PercentageComplete)
  1916. .Add(x => x.Time)
  1917. .Add(x => x.Started)
  1918. .Add(x => x.FormData)
  1919. .Add(x => x.Completed)
  1920. .Add(x => x.FormCompleted)
  1921. .Add(x => x.FormCompletedBy.ID)
  1922. .Add(x => x.Sequence)
  1923. .Add(x => x.QualityNotes)
  1924. .Add(x => x.SectionID)
  1925. .Add(x => x.ManufacturingSectionLink.ID));
  1926. var pktfilter = new Filter<ManufacturingPacket>(x => x.Completed).IsEqualTo(DateTime.MinValue)
  1927. .And(x => x.Archived).IsEqualTo(DateTime.MinValue)
  1928. .And(x => x.OnHold).IsEqualTo(false)
  1929. .And(new Filter<ManufacturingPacket>(x => x.OrderItem).NotLinkValid().Or(x => x.OrderItem.ReceivedDate)
  1930. .IsNotEqualTo(DateTime.MinValue))
  1931. .And(new Filter<ManufacturingPacket>(x => x.StageLink.SectionID).IsEqualTo(sectionid).Or(x => x.Distributed).IsEqualTo(true));
  1932. if (!string.IsNullOrWhiteSpace(SearchFilter))
  1933. pktfilter = pktfilter.TextSearch(
  1934. SearchFilter,
  1935. x => x.SetoutLink.JobLink.JobNumber,
  1936. x => x.SetoutLink.JobLink.Name,
  1937. x => x.SetoutLink.Number,
  1938. x => x.SetoutLink.Location,
  1939. //x => x.SetoutLink.Reference,
  1940. x => x.Title,
  1941. x => x.Serial,
  1942. x => x.Trolleys,
  1943. x => x.Location
  1944. );
  1945. var pktcolumns = new Columns<ManufacturingPacket>(x => x.ID)
  1946. .Add(x => x.Serial)
  1947. .Add(x => x.Title)
  1948. .Add(x => x.Quantity)
  1949. .Add(x => x.BarcodeQty)
  1950. .Add(x => x.SetoutLink.ID)
  1951. .Add(x => x.SetoutLink.Number)
  1952. .Add(x => x.SetoutLink.JobLink.JobNumber)
  1953. .Add(x => x.SetoutLink.JobLink.Name)
  1954. .Add(x => x.SetoutLink.Group.ID)
  1955. .Add(x => x.SetoutLink.Group.Name)
  1956. .Add(x => x.SetoutLink.Group.Job.JobNumber)
  1957. .Add(x => x.SetoutLink.Group.OptimizationDocument.ID)
  1958. .Add(x => x.DueDate)
  1959. .Add(x => x.SetoutLink.Location)
  1960. //.Add(x => x.SetoutLink.Reference)
  1961. .Add(x => x.Priority)
  1962. .Add(x => x.EstimatedDate)
  1963. .Add(x => x.Distributed)
  1964. .Add(x => x.StageLink.ID)
  1965. .Add(x => x.StageLink.SectionID)
  1966. .Add(x => x.StageLink.Section)
  1967. .Add(x => x.StageLink.Deleted)
  1968. .Add(x => x.BarcodePrinted)
  1969. .Add(x => x.BarcodeType)
  1970. .Add(x => x.ManufacturingTemplateLink.ID)
  1971. .Add(x => x.ManufacturingTemplateLink.Name)
  1972. .Add(x => x.ManufacturingTemplateLink.Code)
  1973. //.Add(x => x.CustomAttributes)
  1974. .Add(x => x.Trolleys)
  1975. .Add(x => x.Location)
  1976. .Add(x => x.Group)
  1977. .Add(x => x.Height)
  1978. .Add(x => x.Width)
  1979. .Add(x => x.Length)
  1980. .Add(x => x.OrderItem.ID)
  1981. .Add(x => x.OrderItem.Deleted)
  1982. .Add(x => x.OrderItem.PurchaseOrderLink.SupplierLink.Code)
  1983. .Add(x => x.OrderItem.PurchaseOrderLink.PONumber)
  1984. .Add(x => x.OrderItem.Consignment.EstimatedWarehouseArrival)
  1985. .Add(x => x.OrderItem.ReceivedReference)
  1986. //.Add(x => x.Zone.Code)
  1987. //.Add(x => x.Level.Code)
  1988. .Add(x => x.OrderItem.ReceivedDate)
  1989. .Add(x => x.WaterMark)
  1990. .Add(x => x.Issues)
  1991. .Add(x => x.OnHold)
  1992. .Add(DatabaseSchema.Properties(typeof(ManufacturingPacket))
  1993. .Where(x => x is CustomProperty)
  1994. .Select(x => x.Name).ToArray()
  1995. );
  1996. var pktQuery = new KeyedQueryDef<ManufacturingPacket>(
  1997. pktfilter,
  1998. pktcolumns,
  1999. new SortOrder<ManufacturingPacket>(x => x.Priority, SortDirection.Descending)
  2000. .ThenBy(x => x.SetoutLink.Number));
  2001. var results = Client.QueryMultiple(
  2002. stageQuery,
  2003. pktQuery,
  2004. new KeyedQueryDef<DeliveryItem>(
  2005. new Filter<DeliveryItem>(x => x.DeliveredDate).IsEqualTo(DateTime.MinValue)
  2006. .And(x => x.ManufacturingPacketLink).LinkValid(),
  2007. new Columns<DeliveryItem>(
  2008. x => x.ID,
  2009. x => x.Barcode,
  2010. x => x.ManufacturingPacketLink.ID,
  2011. x => x.ManufacturingPacketLink.Serial,
  2012. x => x.Description,
  2013. x => x.ShipmentLink.ID)));
  2014. Stages = results.Get<ManufacturingPacketStage>();
  2015. Packets = results.Get<ManufacturingPacket>();
  2016. DeliveryItems = results.Get<DeliveryItem>();
  2017. }
  2018. var checks = Kanbans.Where(x => x.Checked).Select(x => x.ID).ToArray();
  2019. Kanbans.Clear();
  2020. var stages = Stages.Rows
  2021. .Select(x => new Tuple<Guid, Guid>(
  2022. x.Get<ManufacturingPacketStage, Guid>(x => x.Parent.ID),
  2023. x.Get<ManufacturingPacketStage, Guid>(x => x.ManufacturingSectionLink.ID)
  2024. )).ToList();
  2025. foreach (var pktrow in Packets.Rows)
  2026. {
  2027. var id = pktrow.Get<ManufacturingPacket, Guid>(c => c.ID);
  2028. var distributed = pktrow.Get<ManufacturingPacket, bool>(c => c.Distributed);
  2029. var stageIndex = stages.FindIndex(x => x.Item1 == id && x.Item2 == settings.Section);
  2030. var stageRow = stageIndex >= 0 ? Stages.Rows[stageIndex] : null;
  2031. //this was commented out previously - leading to the distributed packet issue. Uncommented and tested to work by Nick. Was there a reason for commenting this out?
  2032. bool bOK = true;
  2033. if (distributed)
  2034. bOK = stageRow?.Get<ManufacturingPacketStage, DateTime>(c => c.Completed).IsEmpty() == true;
  2035. if (bOK)
  2036. {
  2037. CreateKanban(pktrow, stageRow!, checks.Contains(id.ToString()));
  2038. }
  2039. }
  2040. var sorted = Kanbans.OrderBy(x => x.Assignee).ThenByDescending(x => x.Tags.Length).ThenBy(x => x.DueDate).ThenBy(x => x.JobName);
  2041. if (CurrentKanban != null && !Kanbans.Contains(CurrentKanban))
  2042. CurrentKanban = sorted.FirstOrDefault();
  2043. Kanban.ItemsSource = null;
  2044. Kanban.ItemsSource =
  2045. sorted; //Kanbans.OrderBy(x => x.Assignee).ThenByDescending(x => x.Tags.Length).ThenBy(x => x.DueDate).ThenBy(x => x.JobName);
  2046. UpdateSelectedKanban();
  2047. }
  2048. }
  2049. private void DoRefresh(bool reloaddata)
  2050. {
  2051. if (!IsReady || Sections == null || !Sections.Any())
  2052. return;
  2053. ReloadPackets(reloaddata);
  2054. //LoadKanban();
  2055. }
  2056. public void Refresh()
  2057. {
  2058. DoRefresh(true);
  2059. }
  2060. #endregion
  2061. #region QAChecks
  2062. private IEnumerable<QAQuestion> LoadQAQuestions(ManufacturingPacket packet, bool section, bool template)
  2063. {
  2064. if (Questions == null)
  2065. Questions = new Client<QAQuestion>().Query();
  2066. if (TemplateStages == null)
  2067. TemplateStages = new Client<ManufacturingTemplateStage>().Query();
  2068. var tstage = TemplateStages.Rows.FirstOrDefault(r =>
  2069. r.Get<ManufacturingTemplateStage, Guid>(c => c.Template.ID).Equals(packet.ManufacturingTemplateLink.ID) &&
  2070. r.Get<ManufacturingTemplateStage, Guid>(c => c.Section.ID).Equals(CurrentSection.ID));
  2071. if (tstage == null)
  2072. return null;
  2073. IEnumerable<CoreRow> rows = null;
  2074. if (section && !template)
  2075. rows = Questions.Rows.Where(r => r.Get<QAQuestion, Guid>(c => c.QAForm.ID).Equals(CurrentSection.QAForm.ID));
  2076. if (!section && template)
  2077. rows = Questions.Rows.Where(r =>
  2078. r.Get<QAQuestion, Guid>(c => c.QAForm.ID).Equals(tstage.Get<ManufacturingTemplateStage, Guid>(x => x.QAForm.ID)));
  2079. if (section && template)
  2080. rows = Questions.Rows.Where(r =>
  2081. r.Get<QAQuestion, Guid>(c => c.QAForm.ID).Equals(CurrentSection.QAForm.ID) || r.Get<QAQuestion, Guid>(c => c.QAForm.ID)
  2082. .Equals(tstage.Get<ManufacturingTemplateStage, Guid>(x => x.QAForm.ID)));
  2083. var result = rows.Where(r => r.Get<QAQuestion, bool>(c => c.QAForm.Active).Equals(true)).Select(r => r.ToObject<QAQuestion>());
  2084. return result;
  2085. }
  2086. private void ViewQualityChecks(object sender, RoutedEventArgs e)
  2087. {
  2088. SetCurrentButton(sender as Button);
  2089. LoadDrawing(Guid.Empty);
  2090. }
  2091. private void QAGridChanged(object sender, Dictionary<Guid, object> values)
  2092. {
  2093. SaveQAData(values);
  2094. }
  2095. private void SaveQAData(Dictionary<Guid, object> values)
  2096. {
  2097. var updates = new List<ManufacturingPacketStage>();
  2098. var cks = Kanbans.Where(x => x.Checked);
  2099. foreach (var ck in cks)
  2100. {
  2101. //var row = Packets.Rows.FirstOrDefault(r => r.Get<ManufacturingPacket, Guid>(c => c.ID).Equals(ck.ID));
  2102. var stagerow = Stages.Rows.FirstOrDefault(r => r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(Guid.Parse(ck.ID))
  2103. && r.Get<ManufacturingPacketStage, Guid>(c => c.ManufacturingSectionLink.ID)
  2104. .Equals(settings.Section));
  2105. if (stagerow != null)
  2106. {
  2107. var stage = stagerow.ToObject<ManufacturingPacketStage>();
  2108. stage.Form.ID = CurrentSection.QAForm.ID;
  2109. stage.FormData = Serialization.Serialize(values);
  2110. stagerow.Set<ManufacturingPacketStage, string>(x => x.FormData, stage.FormData);
  2111. updates.Add(stage);
  2112. }
  2113. }
  2114. if (updates.Any())
  2115. Task.Run(() => { new Client<ManufacturingPacketStage>().Save(updates, "", (o, e) => { }); });
  2116. }
  2117. private void UpdateQuality(QualityStatus status, string note = null)
  2118. {
  2119. var selected = new List<ManufacturingPacket>();
  2120. var stages = new List<ManufacturingPacketStage>();
  2121. foreach (var kanban in Kanbans.Where(x => x.Checked || x.IsSelected))
  2122. {
  2123. selected.Add(KanbanToPacket(kanban));
  2124. var pktid = Guid.Parse(kanban.ID);
  2125. var stagerows = Stages.Rows.Where(r =>
  2126. r.Get<ManufacturingPacketStage, Guid>(c => c.SectionID).Equals(CurrentSection.ID) &&
  2127. r.Get<ManufacturingPacketStage, Guid>(c => c.Parent.ID).Equals(pktid));
  2128. stages.AddRange(stagerows.Select(r => r.ToObject<ManufacturingPacketStage>()));
  2129. }
  2130. if (note == null)
  2131. {
  2132. var form = new NotesForm(Employees) { Caption = "Please detail the quality issues identified with these item(s)" };
  2133. if (form.ShowDialog() != true) return;
  2134. note = form.Text;
  2135. if (form.EmployeeID != Guid.Empty)
  2136. {
  2137. var task = new Kanban();
  2138. task.DueDate = DateTime.Today;
  2139. task.Status = KanbanStatus.Open;
  2140. task.EmployeeLink.ID = form.EmployeeID;
  2141. task.ManagerLink.ID = myID;
  2142. task.Title = "QA Issues : " + CurrentSection.Factory.Name + ": " + CurrentSection.Name;
  2143. //task.JobLink.ID = packet.JobLink.ID;
  2144. var details = new List<string>();
  2145. details.Add(form.Text);
  2146. details.Add("");
  2147. details.Add("The following packets are affected by this issue:");
  2148. foreach (var sel in selected)
  2149. details.Add(string.Format("- {0}: {1} ({2})", sel.SetoutLink.Number, sel.Title,
  2150. string.IsNullOrEmpty(sel.Location) ? sel.SetoutLink.Location : sel.Location));
  2151. task.Notes = new[] { string.Join("\n", details) };
  2152. new Client<Kanban>().Save(task, "Created task from Factory Floor");
  2153. note = note + "\n" + "** Task Created for " + task.EmployeeLink.Name;
  2154. }
  2155. }
  2156. foreach (var stage in stages)
  2157. {
  2158. stage.Station = 0;
  2159. stage.QualityStatus = status;
  2160. if (!string.IsNullOrWhiteSpace(note))
  2161. {
  2162. if (!string.IsNullOrWhiteSpace(stage.QualityNotes))
  2163. stage.QualityNotes = string.Format("{0}\n\n{1}", stage.QualityNotes, note);
  2164. else
  2165. stage.QualityNotes = note;
  2166. }
  2167. }
  2168. Progress.Show("");
  2169. new Client<ManufacturingPacketStage>().Save(stages, note);
  2170. DoRefresh(true);
  2171. Progress.Close();
  2172. }
  2173. #endregion
  2174. private void RequestMaterials_Click(object sender, RoutedEventArgs e)
  2175. {
  2176. }
  2177. }
  2178. }