FactoryPanel.xaml.cs 115 KB

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