FactoryPanel.xaml.cs 115 KB

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