FactoryPanel.xaml.cs 115 KB

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