LicenseRenewalForm.xaml.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513
  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. }
  139. /// <summary>
  140. /// Interaction logic for LicenseRenewal.xaml
  141. /// </summary>
  142. public partial class LicenseRenewalForm : ThemableWindow
  143. {
  144. private LicenseData? _currentLicense;
  145. private double _gross;
  146. private double _discount;
  147. private DateTime _renewalAvailable;
  148. public LicenseData? CurrentLicense
  149. {
  150. get => _currentLicense;
  151. set
  152. {
  153. if(value == null)
  154. {
  155. return;
  156. }
  157. PayWithStripe.IsEnabled = false;
  158. PayTooltip.Visibility = Visibility.Visible;
  159. PayTooltip.Content = new Label { Content = "Loading..." };
  160. _currentLicense = value;
  161. LastRenewal.Value = value.LastRenewal;
  162. CurrentExpiry.Value = value.Expiry;
  163. Modules.Renewed = value.LastRenewal;
  164. RenewalAvailableFrom = value.RenewalAvailable;
  165. Modules.Refresh(false, true);
  166. }
  167. }
  168. public int RenewalPeriod
  169. {
  170. get
  171. {
  172. if (int.TryParse(RenewalPeriodEditor.Value?.ToString(), out var period))
  173. {
  174. return period;
  175. }
  176. return 0;
  177. }
  178. set
  179. {
  180. RenewalPeriodEditor.Value = value;
  181. CalculateDiscounts();
  182. }
  183. }
  184. public DateTime RenewalDate
  185. {
  186. get
  187. {
  188. if(CurrentLicense == null)
  189. {
  190. return DateTime.MinValue;
  191. }
  192. if (CurrentLicense.Expiry > DateTime.Now)
  193. {
  194. return CurrentLicense.Expiry.Date;
  195. }
  196. return DateTime.Today;
  197. }
  198. }
  199. public DateTime RenewalAvailableFrom
  200. {
  201. get => _renewalAvailable;
  202. set
  203. {
  204. _renewalAvailable = value;
  205. if (!CanRenew)
  206. {
  207. RenewalAvailableLabel.Content = $"Renewal available from {value:dd MMM yyyy}";
  208. }
  209. }
  210. }
  211. public bool CanRenew => RenewalAvailableFrom.Date <= DateTime.Today || (CurrentLicense != null && CurrentLicense.Expiry <= DateTime.Now);
  212. public DateTime NewExpiration
  213. {
  214. get => RenewalDate.AddMonths(RenewalPeriod);
  215. }
  216. public double Gross { get => _gross; }
  217. public double Discount { get => _discount; }
  218. public double Net { get => _gross - _discount; }
  219. public LicenseRenewalForm()
  220. {
  221. InitializeComponent();
  222. LastRenewal.EditorDefinition = new DateEditor();
  223. LastRenewal.Configure();
  224. LastRenewal.IsEnabled = false;
  225. CurrentExpiry.EditorDefinition = new DateEditor();
  226. CurrentExpiry.Configure();
  227. CurrentExpiry.IsEnabled = false;
  228. RenewalPeriodEditor.EditorDefinition = new ComboLookupEditor(typeof(RenewalPeriodLookups));
  229. RenewalPeriodEditor.Configure();
  230. var lookups = new RenewalPeriodLookups(null).AsTable("RenewalPeriod");
  231. RenewalPeriodEditor.LoadLookups(lookups);
  232. NewExpiry.EditorDefinition = new DateEditor();
  233. NewExpiry.Configure();
  234. NewExpiry.IsEnabled = false;
  235. GrossLicenseFee.EditorDefinition = new CurrencyEditor();
  236. GrossLicenseFee.Configure();
  237. GrossLicenseFee.IsEnabled = false;
  238. DiscountEditor.EditorDefinition = new DoubleEditor();
  239. DiscountEditor.Configure();
  240. DiscountEditor.IsEnabled = false;
  241. NettLicenseFee.EditorDefinition = new CurrencyEditor();
  242. NettLicenseFee.Configure();
  243. NettLicenseFee.IsEnabled = false;
  244. Help.Content = new Image { Source = Properties.Resources.help.AsBitmapImage(Color.White) };
  245. Help.Click += Help_Click;
  246. Modules.AfterRefresh += ModulesRefreshed;
  247. //double months = Math.Max(0.0F,CalcMonths(CurrentExpiry.Value, NewExpiry.Value));
  248. //double total = 0.0F;
  249. //foreach (var row in Modules.Data.Rows)
  250. // total += row.Get<LicenseRenewal, double>(x => x.ExGST) * months;
  251. //GrossLicenseFee.Value = total;
  252. ////GrossLicenseFee.Width = 150;
  253. //Discount.Value = LicenseUtils.GetUserDiscount(Modules.Licenses);
  254. ////Discount.Width = 150;
  255. //NettLicenseFee.Value = total * ((100.0F - Discount.Value) / 100.0F);
  256. ////NettLicenseFee.Width = 150;
  257. }
  258. private void Help_Click(object sender, RoutedEventArgs e)
  259. {
  260. Process.Start(new ProcessStartInfo("https://prsdigital.com.au/wiki/index.php/License_Renewal") { UseShellExecute = true });
  261. }
  262. private void Window_Loaded(object sender, RoutedEventArgs e)
  263. {
  264. if (!LicenseTrackingItemGrid.LicenseClient.Ping("ping"))
  265. {
  266. MessageBox.Show("The PRS server is not available right now. Please try again later.");
  267. Close();
  268. return;
  269. }
  270. Modules.Refresh(true, false);
  271. var license = LoadCurrentLicense();
  272. if(license != null)
  273. {
  274. CurrentLicense = license;
  275. }
  276. LastRenewal.Loaded = true;
  277. CurrentExpiry.Loaded = true;
  278. NewExpiry.Loaded = true;
  279. RenewalPeriodEditor.Loaded = true;
  280. }
  281. private void ModulesRefreshed(object sender, AfterRefreshEventArgs args)
  282. {
  283. if (CurrentLicense == null) return;
  284. var lookups = new RenewalPeriodLookups(null).AsTable("RenewalPeriod");
  285. RenewalPeriodEditor.LoadLookups(lookups);
  286. RenewalPeriod = LicenseUtils.TimeDiscountLevels().OrderBy(x => x).First();
  287. PayWithStripe.IsEnabled = CanRenew;
  288. if (!PayWithStripe.IsEnabled)
  289. {
  290. PayTooltip.Visibility = Visibility.Visible;
  291. PayTooltip.Content = new Label { Content = $"Renewal available from {RenewalAvailableFrom:dd MMM yyyy}" };
  292. }
  293. else
  294. {
  295. PayTooltip.Visibility = Visibility.Collapsed;
  296. }
  297. OK.IsEnabled = true;
  298. }
  299. private static LicenseData? LoadCurrentLicense()
  300. {
  301. var license = new Client<License>().Query(new Filter<License>().All(), new Columns<License>(x => x.Data))
  302. .Rows.FirstOrDefault()?.ToObject<License>();
  303. if(license == null)
  304. {
  305. MessageBox.Show("No current license found. Please see the documentation on how to proceed.");
  306. return null;
  307. }
  308. if(!LicenseUtils.TryDecryptLicense(license.Data, out var data, out var error))
  309. {
  310. MessageBox.Show("Current license is corrupt. Please see the documentation on how to proceed.");
  311. return null;
  312. }
  313. return data;
  314. }
  315. private void RenewalPeriod_OnEditorValueChanged(IDynamicEditorControl sender, Dictionary<string, object> values)
  316. {
  317. CalculateDiscounts();
  318. }
  319. private void CalculateDiscounts()
  320. {
  321. var periodInMonths = RenewalPeriod;
  322. NewExpiry.Value = NewExpiration;
  323. double total = 0.0F;
  324. foreach (var row in Modules.Data.Rows)
  325. total += row.Get<LicenseTrackingItem, double>(x => x.ExGST) * periodInMonths;
  326. _gross = total;
  327. _discount = total * (LicenseUtils.GetUserDiscount(Modules.Licenses) / 100.0F) +
  328. total * (LicenseUtils.GetTimeDiscount(periodInMonths) / 100.0F);
  329. GrossLicenseFee.Value = Gross;
  330. DiscountEditor.Value = Discount;
  331. NettLicenseFee.Value = Net;
  332. PayWithStripe.IsEnabled = CanRenew && Net > 0;
  333. }
  334. private class RenewalPeriodLookups : LookupGenerator<object>
  335. {
  336. public RenewalPeriodLookups(object[] items) : base(items)
  337. {
  338. }
  339. protected override void DoGenerateLookups()
  340. {
  341. foreach (var period in LicenseUtils.TimeDiscountLevels())
  342. AddValue(period, string.Format("{0} month{1}", period, period > 1 ? "s" : ""));
  343. }
  344. }
  345. private License GetNewLicense(LicenseRenewal licenseRenewal)
  346. {
  347. var license = LicenseTrackingItemGrid.LicenseClient.PostRequest<License>(licenseRenewal, "RenewLicense");
  348. if(license == null)
  349. {
  350. throw new Exception("The server did not return a valid license.");
  351. }
  352. return license;
  353. }
  354. private void SaveLicense(License license)
  355. {
  356. using var client = new Client<License>();
  357. var oldLicenses = client
  358. .Query(new Filter<License>().All(), new Columns<License>(x => x.ID))
  359. .Rows.Select(x => x.ToObject<License>()).ToList();
  360. client.Delete(oldLicenses, "");
  361. client.Save(license, "");
  362. }
  363. private void PayNowClick(object sender, RoutedEventArgs e)
  364. {
  365. if (!LicenseTrackingItemGrid.LicenseClient.Ping("ping"))
  366. {
  367. MessageBox.Show("The PRS server is not available right now. Please try again later.");
  368. return;
  369. }
  370. var companyInformation = new Client<CompanyInformation>().Load().FirstOrDefault();
  371. var isCompanyEmpty = string.IsNullOrWhiteSpace(companyInformation?.CompanyName);
  372. var isABNEmpty = string.IsNullOrWhiteSpace(companyInformation?.ABN);
  373. if (companyInformation == null || (isCompanyEmpty && isABNEmpty))
  374. {
  375. 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");
  376. return;
  377. }
  378. else if (isCompanyEmpty)
  379. {
  380. MessageBox.Show("Please set your company's name before proceeding. This can be done through PRS Desktop, via System -> Company Information");
  381. return;
  382. }
  383. else if (isABNEmpty)
  384. {
  385. MessageBox.Show("Please set your company's ABN before proceeding. This can be done through PRS Desktop, via System -> Company Information");
  386. return;
  387. }
  388. if (!CreditCardGrid.DoPayment(Net, out var transactionID))
  389. return;
  390. /*var grid = new CreditCardGrid();
  391. var ccDetails = grid.LoadSaved();
  392. ccDetails.Amount = Net;
  393. if (grid.EditItems(new CreditCardDetails[] { ccDetails }))
  394. {
  395. var transactionID = grid.TransactionID;*/
  396. var licenseRenewal = new LicenseRenewal
  397. {
  398. Company = companyInformation,
  399. DateRenewed = RenewalDate,
  400. OldLicense = CurrentLicense,
  401. NewExpiry = NewExpiration,
  402. LicenseTracking = Modules.Data.Rows.Select(x => x.ToObject<LicenseTrackingItem>()).ToArray(),
  403. Gross = Gross,
  404. Discount = Discount,
  405. Net = Net,
  406. TransactionID = transactionID
  407. };
  408. string? error = null;
  409. License? license = null;
  410. Progress.ShowModal("Payment success, renewing license", progress =>
  411. {
  412. try
  413. {
  414. license = GetNewLicense(licenseRenewal);
  415. SaveLicense(license);
  416. }
  417. catch(Exception e)
  418. {
  419. 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}'";
  420. }
  421. });
  422. if(error != null)
  423. {
  424. MessageBox.Show(error);
  425. }
  426. else
  427. {
  428. MessageBox.Show("License Renewed!");
  429. if(!LicenseUtils.TryDecryptLicense(license.Data, out var licenseData, out error))
  430. {
  431. MessageBox.Show(error);
  432. }
  433. CurrentLicense = licenseData ?? new LicenseData();
  434. }
  435. //}
  436. }
  437. private void OK_Click(object sender, RoutedEventArgs e)
  438. {
  439. DialogResult = true;
  440. Close();
  441. }
  442. private void Cancel_Click(object sender, RoutedEventArgs e)
  443. {
  444. DialogResult = false;
  445. Close();
  446. }
  447. }
  448. }