LicenseRenewalForm.xaml.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Windows;
  5. using InABox.Clients;
  6. using InABox.Core;
  7. using InABox.DynamicGrid;
  8. using InABox.Wpf;
  9. using InABox.WPF;
  10. using Stripe;
  11. using System.Diagnostics.CodeAnalysis;
  12. using System.Windows.Controls;
  13. using System.Drawing;
  14. using Image = System.Windows.Controls.Image;
  15. using System.Diagnostics;
  16. using InABox.Configuration;
  17. namespace PRSServer.Forms.DatabaseLicense
  18. {
  19. public class CreditCardGrid : DynamicGrid<CreditCardDetails>
  20. {
  21. private static readonly EncryptedLocalConfiguration<CreditCardSettings> DetailsConfiguration = new("g6+BoQpyti5bHsTZOY5Nbqq3Q3c90n0m3qZaQ3eAwkk=");
  22. protected override void Reload(Filters<CreditCardDetails> criteria, Columns<CreditCardDetails> columns, ref SortOrder<CreditCardDetails>? sort, Action<CoreTable, Exception> action)
  23. {
  24. }
  25. private static CreditCardDetails LoadSaved()
  26. {
  27. var settings = DetailsConfiguration.Load();
  28. var details = new CreditCardDetails
  29. {
  30. CardNumber = settings.CardNumber,
  31. Month = settings.Month,
  32. Year = settings.Year,
  33. Cvv = settings.Cvv,
  34. // If the card number is not blank, it means we loaded from a saved file, and therefore the default should be to still save details
  35. SaveDetails = !string.IsNullOrWhiteSpace(settings.CardNumber)
  36. };
  37. return details;
  38. }
  39. private static void SaveDetails(CreditCardDetails details)
  40. {
  41. var settings = new CreditCardSettings
  42. {
  43. CardNumber = details.CardNumber,
  44. Month = details.Month,
  45. Year = details.Year,
  46. Cvv = details.Cvv
  47. };
  48. DetailsConfiguration.Save(settings);
  49. }
  50. private static void ClearDetails()
  51. {
  52. DetailsConfiguration.Delete();
  53. }
  54. public static bool DoPayment(double amount, [NotNullWhen(true)] out string? transactionID)
  55. {
  56. var details = LoadSaved();
  57. details.Amount = amount;
  58. var grid = new CreditCardGrid();
  59. if (grid.EditItems(new[] { details }))
  60. {
  61. if (details.SaveDetails)
  62. {
  63. SaveDetails(details);
  64. }
  65. else
  66. {
  67. ClearDetails();
  68. }
  69. transactionID = grid.TransactionID;
  70. return true;
  71. }
  72. transactionID = null;
  73. return false;
  74. }
  75. protected override CreditCardDetails LoadItem(CoreRow row)
  76. {
  77. return new CreditCardDetails();
  78. }
  79. private string TransactionID = "";
  80. protected override void DoValidate(CreditCardDetails[] items, List<string> errors)
  81. {
  82. base.DoValidate(items, errors);
  83. if (items.Any(x => string.IsNullOrWhiteSpace(x.CardNumber)))
  84. errors.Add("Please supply a card number.");
  85. if (items.Any(x => string.IsNullOrWhiteSpace(x.Month)))
  86. errors.Add("Please supply an expiration month.");
  87. if (items.Any(x => string.IsNullOrWhiteSpace(x.Year)))
  88. errors.Add("Please supply an expiration year.");
  89. if (items.Any(x => string.IsNullOrWhiteSpace(x.Cvv)))
  90. errors.Add("Please supply a CVV.");
  91. }
  92. public override void SaveItem(CreditCardDetails item)
  93. {
  94. TransactionID = "";
  95. var error = "";
  96. Progress.ShowModal("Processing Payment", (progress) =>
  97. {
  98. try
  99. {
  100. StripeConfiguration.ApiKey = "sk_test_51MRSuPIyPMVqmkXN8TeGxDAGBeFx0pLzn3fsHBR8X1oMBKQVwGPEbuv6DNIu0qSmuflpmFfQ4N8c3vzdknKa7G0o00wTOXwCeW";
  101. var cardoptions = new TokenCreateOptions()
  102. {
  103. Card = new TokenCardOptions()
  104. {
  105. Number = item.CardNumber.Replace(" ", ""),
  106. ExpMonth = item.Month,
  107. ExpYear = item.Year,
  108. Cvc = item.Cvv
  109. }
  110. };
  111. var service = new TokenService();
  112. var token = service.Create(cardoptions);
  113. var chargeoptions = new ChargeCreateOptions()
  114. {
  115. Amount = Convert.ToInt64(item.Amount * 100),
  116. Currency = "AUD",
  117. Description = "PRS Renewal",
  118. Source = token.Id
  119. };
  120. var chargeservice = new ChargeService();
  121. var charge = chargeservice.Create(chargeoptions);
  122. if (charge.Paid)
  123. TransactionID = charge.Id;
  124. else
  125. error = string.Format("{0}: {1}", charge.FailureCode, charge.FailureMessage);
  126. }
  127. catch (Exception ex)
  128. {
  129. error = $"{ex.Message}";
  130. }
  131. });
  132. if (!String.IsNullOrWhiteSpace(error))
  133. throw new Exception(error);
  134. }
  135. protected override void DeleteItems(params CoreRow[] rows)
  136. {
  137. }
  138. protected override void Init()
  139. {
  140. }
  141. protected override void DoReconfigure(FluentList<DynamicGridOption> options)
  142. {
  143. }
  144. }
  145. /// <summary>
  146. /// Interaction logic for LicenseRenewal.xaml
  147. /// </summary>
  148. public partial class LicenseRenewalForm : ThemableWindow
  149. {
  150. private LicenseData? _currentLicense;
  151. private double _gross;
  152. private double _discount;
  153. private DateTime _renewalAvailable;
  154. public LicenseData? CurrentLicense
  155. {
  156. get => _currentLicense;
  157. set
  158. {
  159. if(value == null)
  160. {
  161. return;
  162. }
  163. PayWithStripe.IsEnabled = false;
  164. PayTooltip.Visibility = Visibility.Visible;
  165. PayTooltip.Content = new Label { Content = "Loading..." };
  166. _currentLicense = value;
  167. LastRenewal.Value = value.LastRenewal;
  168. CurrentExpiry.Value = value.Expiry;
  169. Modules.Renewed = value.LastRenewal;
  170. RenewalAvailableFrom = value.RenewalAvailable;
  171. Modules.Refresh(false, true);
  172. }
  173. }
  174. public int RenewalPeriod
  175. {
  176. get
  177. {
  178. if (int.TryParse(RenewalPeriodEditor.Value?.ToString(), out var period))
  179. {
  180. return period;
  181. }
  182. return 0;
  183. }
  184. set
  185. {
  186. RenewalPeriodEditor.Value = value;
  187. CalculateDiscounts();
  188. }
  189. }
  190. public DateTime RenewalDate
  191. {
  192. get
  193. {
  194. if(CurrentLicense == null)
  195. {
  196. return DateTime.MinValue;
  197. }
  198. if (CurrentLicense.Expiry > DateTime.Now)
  199. {
  200. return CurrentLicense.Expiry.Date;
  201. }
  202. return DateTime.Today;
  203. }
  204. }
  205. public DateTime RenewalAvailableFrom
  206. {
  207. get => _renewalAvailable;
  208. set
  209. {
  210. _renewalAvailable = value;
  211. if (!CanRenew)
  212. {
  213. RenewalAvailableLabel.Content = $"Renewal available from {value:dd MMM yyyy}";
  214. }
  215. }
  216. }
  217. public bool CanRenew => RenewalAvailableFrom.Date <= DateTime.Today || (CurrentLicense != null && CurrentLicense.Expiry <= DateTime.Now);
  218. public DateTime NewExpiration
  219. {
  220. get => RenewalDate.AddMonths(RenewalPeriod);
  221. }
  222. public double Gross { get => _gross; }
  223. public double Discount { get => _discount; }
  224. public double Net { get => _gross - _discount; }
  225. public LicenseRenewalForm()
  226. {
  227. InitializeComponent();
  228. LastRenewal.EditorDefinition = new DateEditor();
  229. LastRenewal.Configure();
  230. LastRenewal.IsEnabled = false;
  231. CurrentExpiry.EditorDefinition = new DateEditor();
  232. CurrentExpiry.Configure();
  233. CurrentExpiry.IsEnabled = false;
  234. RenewalPeriodEditor.EditorDefinition = new ComboLookupEditor(typeof(RenewalPeriodLookups));
  235. RenewalPeriodEditor.Configure();
  236. var lookups = new RenewalPeriodLookups(null).AsTable("RenewalPeriod");
  237. RenewalPeriodEditor.LoadLookups(lookups);
  238. NewExpiry.EditorDefinition = new DateEditor();
  239. NewExpiry.Configure();
  240. NewExpiry.IsEnabled = false;
  241. GrossLicenseFee.EditorDefinition = new CurrencyEditor();
  242. GrossLicenseFee.Configure();
  243. GrossLicenseFee.IsEnabled = false;
  244. DiscountEditor.EditorDefinition = new DoubleEditor();
  245. DiscountEditor.Configure();
  246. DiscountEditor.IsEnabled = false;
  247. NettLicenseFee.EditorDefinition = new CurrencyEditor();
  248. NettLicenseFee.Configure();
  249. NettLicenseFee.IsEnabled = false;
  250. Help.Content = new Image { Source = Properties.Resources.help.AsBitmapImage(Color.White) };
  251. Help.Click += Help_Click;
  252. Modules.AfterRefresh += ModulesRefreshed;
  253. //double months = Math.Max(0.0F,CalcMonths(CurrentExpiry.Value, NewExpiry.Value));
  254. //double total = 0.0F;
  255. //foreach (var row in Modules.Data.Rows)
  256. // total += row.Get<LicenseRenewal, double>(x => x.ExGST) * months;
  257. //GrossLicenseFee.Value = total;
  258. ////GrossLicenseFee.Width = 150;
  259. //Discount.Value = LicenseUtils.GetUserDiscount(Modules.Licenses);
  260. ////Discount.Width = 150;
  261. //NettLicenseFee.Value = total * ((100.0F - Discount.Value) / 100.0F);
  262. ////NettLicenseFee.Width = 150;
  263. }
  264. private void Help_Click(object sender, RoutedEventArgs e)
  265. {
  266. Process.Start(new ProcessStartInfo("https://prsdigital.com.au/wiki/index.php/License_Renewal") { UseShellExecute = true });
  267. }
  268. private void Window_Loaded(object sender, RoutedEventArgs e)
  269. {
  270. if (!LicenseTrackingItemGrid.LicenseClient.Ping("ping"))
  271. {
  272. MessageBox.Show("The PRS server is not available right now. Please try again later.");
  273. Close();
  274. return;
  275. }
  276. Modules.Refresh(true, false);
  277. var license = LoadCurrentLicense();
  278. if(license != null)
  279. {
  280. CurrentLicense = license;
  281. }
  282. LastRenewal.Loaded = true;
  283. CurrentExpiry.Loaded = true;
  284. NewExpiry.Loaded = true;
  285. RenewalPeriodEditor.Loaded = true;
  286. }
  287. private void ModulesRefreshed(object sender, AfterRefreshEventArgs args)
  288. {
  289. if (CurrentLicense == null) return;
  290. var lookups = new RenewalPeriodLookups(null).AsTable("RenewalPeriod");
  291. RenewalPeriodEditor.LoadLookups(lookups);
  292. RenewalPeriod = LicenseUtils.TimeDiscountLevels().OrderBy(x => x).First();
  293. PayWithStripe.IsEnabled = CanRenew;
  294. if (!PayWithStripe.IsEnabled)
  295. {
  296. PayTooltip.Visibility = Visibility.Visible;
  297. PayTooltip.Content = new Label { Content = $"Renewal available from {RenewalAvailableFrom:dd MMM yyyy}" };
  298. }
  299. else
  300. {
  301. PayTooltip.Visibility = Visibility.Collapsed;
  302. }
  303. OK.IsEnabled = true;
  304. }
  305. private static LicenseData? LoadCurrentLicense()
  306. {
  307. var license = new Client<License>().Query(new Filter<License>().All(), new Columns<License>(x => x.Data))
  308. .Rows.FirstOrDefault()?.ToObject<License>();
  309. if(license == null)
  310. {
  311. MessageBox.Show("No current license found. Please see the documentation on how to proceed.");
  312. return null;
  313. }
  314. if(!LicenseUtils.TryDecryptLicense(license.Data, out var data, out var error))
  315. {
  316. MessageBox.Show("Current license is corrupt. Please see the documentation on how to proceed.");
  317. return null;
  318. }
  319. return data;
  320. }
  321. private void RenewalPeriod_OnEditorValueChanged(IDynamicEditorControl sender, Dictionary<string, object> values)
  322. {
  323. CalculateDiscounts();
  324. }
  325. private void CalculateDiscounts()
  326. {
  327. var periodInMonths = RenewalPeriod;
  328. NewExpiry.Value = NewExpiration;
  329. double total = 0.0F;
  330. foreach (var row in Modules.Data.Rows)
  331. total += row.Get<LicenseTrackingItem, double>(x => x.ExGST) * periodInMonths;
  332. _gross = total;
  333. _discount = total * (LicenseUtils.GetUserDiscount(Modules.Licenses) / 100.0F) +
  334. total * (LicenseUtils.GetTimeDiscount(periodInMonths) / 100.0F);
  335. GrossLicenseFee.Value = Gross;
  336. DiscountEditor.Value = Discount;
  337. NettLicenseFee.Value = Net;
  338. PayWithStripe.IsEnabled = CanRenew && Net > 0;
  339. }
  340. private class RenewalPeriodLookups : LookupGenerator<object>
  341. {
  342. public RenewalPeriodLookups(object[] items) : base(items)
  343. {
  344. }
  345. protected override void DoGenerateLookups()
  346. {
  347. foreach (var period in LicenseUtils.TimeDiscountLevels())
  348. AddValue(period, string.Format("{0} month{1}", period, period > 1 ? "s" : ""));
  349. }
  350. }
  351. private License GetNewLicense(LicenseRenewal licenseRenewal)
  352. {
  353. var license = LicenseTrackingItemGrid.LicenseClient.PostRequest<License>(licenseRenewal, "RenewLicense");
  354. if(license == null)
  355. {
  356. throw new Exception("The server did not return a valid license.");
  357. }
  358. return license;
  359. }
  360. private void SaveLicense(License license)
  361. {
  362. using var client = new Client<License>();
  363. var oldLicenses = client
  364. .Query(new Filter<License>().All(), new Columns<License>(x => x.ID))
  365. .Rows.Select(x => x.ToObject<License>()).ToList();
  366. client.Delete(oldLicenses, "");
  367. client.Save(license, "");
  368. }
  369. private void PayNowClick(object sender, RoutedEventArgs e)
  370. {
  371. if (!LicenseTrackingItemGrid.LicenseClient.Ping("ping"))
  372. {
  373. MessageBox.Show("The PRS server is not available right now. Please try again later.");
  374. return;
  375. }
  376. var companyInformation = new Client<CompanyInformation>().Load().FirstOrDefault();
  377. var isCompanyEmpty = string.IsNullOrWhiteSpace(companyInformation?.CompanyName);
  378. var isABNEmpty = string.IsNullOrWhiteSpace(companyInformation?.ABN);
  379. if (companyInformation == null || (isCompanyEmpty && isABNEmpty))
  380. {
  381. MessageBox.Show("The company information for this database has not been set. Please set company information before proceeding. The company name and ABN are required. This can be done through PRS Desktop, via System -> Company Information");
  382. return;
  383. }
  384. else if (isCompanyEmpty)
  385. {
  386. MessageBox.Show("Please set your company's name before proceeding. This can be done through PRS Desktop, via System -> Company Information");
  387. return;
  388. }
  389. else if (isABNEmpty)
  390. {
  391. MessageBox.Show("Please set your company's ABN before proceeding. This can be done through PRS Desktop, via System -> Company Information");
  392. return;
  393. }
  394. if (!CreditCardGrid.DoPayment(Net, out var transactionID))
  395. return;
  396. /*var grid = new CreditCardGrid();
  397. var ccDetails = grid.LoadSaved();
  398. ccDetails.Amount = Net;
  399. if (grid.EditItems(new CreditCardDetails[] { ccDetails }))
  400. {
  401. var transactionID = grid.TransactionID;*/
  402. var licenseRenewal = new LicenseRenewal
  403. {
  404. Company = companyInformation,
  405. DateRenewed = RenewalDate,
  406. OldLicense = CurrentLicense,
  407. NewExpiry = NewExpiration,
  408. LicenseTracking = Modules.Data.Rows.Select(x => x.ToObject<LicenseTrackingItem>()).ToArray(),
  409. Gross = Gross,
  410. Discount = Discount,
  411. Net = Net,
  412. TransactionID = transactionID
  413. };
  414. string? error = null;
  415. License? license = null;
  416. Progress.ShowModal("Payment success, renewing license", progress =>
  417. {
  418. try
  419. {
  420. license = GetNewLicense(licenseRenewal);
  421. SaveLicense(license);
  422. }
  423. catch(Exception e)
  424. {
  425. error = $"There was an error retrieving your new license:\n\n{e.Message}\n\nPlease contact the PRS team, and keep your transaction ID for future reference. Your transaction ID is '{transactionID}'";
  426. }
  427. });
  428. if(error != null)
  429. {
  430. MessageBox.Show(error);
  431. }
  432. else
  433. {
  434. MessageBox.Show("License Renewed!");
  435. if(!LicenseUtils.TryDecryptLicense(license.Data, out var licenseData, out error))
  436. {
  437. MessageBox.Show(error);
  438. }
  439. CurrentLicense = licenseData ?? new LicenseData();
  440. }
  441. //}
  442. }
  443. private void OK_Click(object sender, RoutedEventArgs e)
  444. {
  445. DialogResult = true;
  446. Close();
  447. }
  448. private void Cancel_Click(object sender, RoutedEventArgs e)
  449. {
  450. DialogResult = false;
  451. Close();
  452. }
  453. }
  454. }