From 7ec132eee34a82fb9f6d30cd0910e34b2a63322e Mon Sep 17 00:00:00 2001 From: Alvin Ashcraft Date: Sat, 23 Sep 2023 15:10:27 -0400 Subject: [PATCH] Chapter 14 samples --- Chapter14/Ch14-MyMediaCollection.sln | 43 +++ Chapter14/MyMediaCollection/App.xaml | 18 ++ Chapter14/MyMediaCollection/App.xaml.cs | 131 ++++++++ .../Assets/LockScreenLogo.scale-200.png | Bin 0 -> 432 bytes .../Assets/SplashScreen.scale-200.png | Bin 0 -> 5372 bytes .../Assets/Square150x150Logo.scale-200.png | Bin 0 -> 1755 bytes .../Assets/Square44x44Logo.scale-200.png | Bin 0 -> 637 bytes ...x44Logo.targetsize-24_altform-unplated.png | Bin 0 -> 283 bytes .../MyMediaCollection/Assets/StoreLogo.png | Bin 0 -> 456 bytes .../Assets/Wide310x150Logo.scale-200.png | Bin 0 -> 2097 bytes Chapter14/MyMediaCollection/Enums/ItemType.cs | 9 + .../MyMediaCollection/Enums/LocationType.cs | 8 + .../Helpers/NotificationManager.cs | 88 +++++ .../Helpers/NotificationShared.cs | 43 +++ .../Helpers/ToastWithAvatar.cs | 42 +++ .../Helpers/ToastWithText.cs | 47 +++ .../Interfaces/IDataService.cs | 22 ++ .../Interfaces/INavigationService.cs | 12 + .../Interfaces/IValidatable.cs | 7 + Chapter14/MyMediaCollection/MainWindow.xaml | 10 + .../MyMediaCollection/MainWindow.xaml.cs | 47 +++ .../MyMediaCollection/Model/MediaItem.cs | 17 + Chapter14/MyMediaCollection/Model/Medium.cs | 13 + .../MyMediaCollection.csproj | 79 +++++ .../MyMediaCollection/Package.appxmanifest | 63 ++++ .../Properties/launchSettings.json | 10 + .../MyMediaCollection/Services/DataService.cs | 175 ++++++++++ .../Services/NavigationService.cs | 84 +++++ .../Services/SqliteDataService.cs | 300 ++++++++++++++++++ .../ViewModels/ItemDetailsViewModel.cs | 142 +++++++++ .../ViewModels/MainViewModel.cs | 121 +++++++ .../Views/ItemDetailsPage.xaml | 99 ++++++ .../Views/ItemDetailsPage.xaml.cs | 60 ++++ .../MyMediaCollection/Views/MainPage.xaml | 89 ++++++ .../MyMediaCollection/Views/MainPage.xaml.cs | 93 ++++++ Chapter14/MyMediaCollection/app.manifest | 25 ++ Chapter14/ProjectTracker/App.xaml | 16 + Chapter14/ProjectTracker/App.xaml.cs | 50 +++ .../Assets/LockScreenLogo.scale-200.png | Bin 0 -> 432 bytes .../Assets/SplashScreen.scale-200.png | Bin 0 -> 5372 bytes .../Assets/Square150x150Logo.scale-200.png | Bin 0 -> 1755 bytes .../Assets/Square44x44Logo.scale-200.png | Bin 0 -> 637 bytes ...x44Logo.targetsize-24_altform-unplated.png | Bin 0 -> 283 bytes Chapter14/ProjectTracker/Assets/StoreLogo.png | Bin 0 -> 456 bytes .../Assets/Wide310x150Logo.scale-200.png | Bin 0 -> 2097 bytes Chapter14/ProjectTracker/MainWindow.xaml | 14 + Chapter14/ProjectTracker/MainWindow.xaml.cs | 36 +++ Chapter14/ProjectTracker/Package.appxmanifest | 51 +++ .../ProjectTracker/ProjectTracker.csproj | 58 ++++ Chapter14/ProjectTracker/ProjectTracker.sln | 43 +++ .../Properties/launchSettings.json | 10 + Chapter14/ProjectTracker/app.manifest | 20 ++ 52 files changed, 2195 insertions(+) create mode 100644 Chapter14/Ch14-MyMediaCollection.sln create mode 100644 Chapter14/MyMediaCollection/App.xaml create mode 100644 Chapter14/MyMediaCollection/App.xaml.cs create mode 100644 Chapter14/MyMediaCollection/Assets/LockScreenLogo.scale-200.png create mode 100644 Chapter14/MyMediaCollection/Assets/SplashScreen.scale-200.png create mode 100644 Chapter14/MyMediaCollection/Assets/Square150x150Logo.scale-200.png create mode 100644 Chapter14/MyMediaCollection/Assets/Square44x44Logo.scale-200.png create mode 100644 Chapter14/MyMediaCollection/Assets/Square44x44Logo.targetsize-24_altform-unplated.png create mode 100644 Chapter14/MyMediaCollection/Assets/StoreLogo.png create mode 100644 Chapter14/MyMediaCollection/Assets/Wide310x150Logo.scale-200.png create mode 100644 Chapter14/MyMediaCollection/Enums/ItemType.cs create mode 100644 Chapter14/MyMediaCollection/Enums/LocationType.cs create mode 100644 Chapter14/MyMediaCollection/Helpers/NotificationManager.cs create mode 100644 Chapter14/MyMediaCollection/Helpers/NotificationShared.cs create mode 100644 Chapter14/MyMediaCollection/Helpers/ToastWithAvatar.cs create mode 100644 Chapter14/MyMediaCollection/Helpers/ToastWithText.cs create mode 100644 Chapter14/MyMediaCollection/Interfaces/IDataService.cs create mode 100644 Chapter14/MyMediaCollection/Interfaces/INavigationService.cs create mode 100644 Chapter14/MyMediaCollection/Interfaces/IValidatable.cs create mode 100644 Chapter14/MyMediaCollection/MainWindow.xaml create mode 100644 Chapter14/MyMediaCollection/MainWindow.xaml.cs create mode 100644 Chapter14/MyMediaCollection/Model/MediaItem.cs create mode 100644 Chapter14/MyMediaCollection/Model/Medium.cs create mode 100644 Chapter14/MyMediaCollection/MyMediaCollection.csproj create mode 100644 Chapter14/MyMediaCollection/Package.appxmanifest create mode 100644 Chapter14/MyMediaCollection/Properties/launchSettings.json create mode 100644 Chapter14/MyMediaCollection/Services/DataService.cs create mode 100644 Chapter14/MyMediaCollection/Services/NavigationService.cs create mode 100644 Chapter14/MyMediaCollection/Services/SqliteDataService.cs create mode 100644 Chapter14/MyMediaCollection/ViewModels/ItemDetailsViewModel.cs create mode 100644 Chapter14/MyMediaCollection/ViewModels/MainViewModel.cs create mode 100644 Chapter14/MyMediaCollection/Views/ItemDetailsPage.xaml create mode 100644 Chapter14/MyMediaCollection/Views/ItemDetailsPage.xaml.cs create mode 100644 Chapter14/MyMediaCollection/Views/MainPage.xaml create mode 100644 Chapter14/MyMediaCollection/Views/MainPage.xaml.cs create mode 100644 Chapter14/MyMediaCollection/app.manifest create mode 100644 Chapter14/ProjectTracker/App.xaml create mode 100644 Chapter14/ProjectTracker/App.xaml.cs create mode 100644 Chapter14/ProjectTracker/Assets/LockScreenLogo.scale-200.png create mode 100644 Chapter14/ProjectTracker/Assets/SplashScreen.scale-200.png create mode 100644 Chapter14/ProjectTracker/Assets/Square150x150Logo.scale-200.png create mode 100644 Chapter14/ProjectTracker/Assets/Square44x44Logo.scale-200.png create mode 100644 Chapter14/ProjectTracker/Assets/Square44x44Logo.targetsize-24_altform-unplated.png create mode 100644 Chapter14/ProjectTracker/Assets/StoreLogo.png create mode 100644 Chapter14/ProjectTracker/Assets/Wide310x150Logo.scale-200.png create mode 100644 Chapter14/ProjectTracker/MainWindow.xaml create mode 100644 Chapter14/ProjectTracker/MainWindow.xaml.cs create mode 100644 Chapter14/ProjectTracker/Package.appxmanifest create mode 100644 Chapter14/ProjectTracker/ProjectTracker.csproj create mode 100644 Chapter14/ProjectTracker/ProjectTracker.sln create mode 100644 Chapter14/ProjectTracker/Properties/launchSettings.json create mode 100644 Chapter14/ProjectTracker/app.manifest diff --git a/Chapter14/Ch14-MyMediaCollection.sln b/Chapter14/Ch14-MyMediaCollection.sln new file mode 100644 index 0000000..d74113c --- /dev/null +++ b/Chapter14/Ch14-MyMediaCollection.sln @@ -0,0 +1,43 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 17 +VisualStudioVersion = 17.5.33530.505 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "MyMediaCollection", "MyMediaCollection\MyMediaCollection.csproj", "{972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|ARM64 = Debug|ARM64 + Debug|x64 = Debug|x64 + Debug|x86 = Debug|x86 + Release|ARM64 = Release|ARM64 + Release|x64 = Release|x64 + Release|x86 = Release|x86 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Debug|ARM64.ActiveCfg = Debug|ARM64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Debug|ARM64.Build.0 = Debug|ARM64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Debug|ARM64.Deploy.0 = Debug|ARM64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Debug|x64.ActiveCfg = Debug|x64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Debug|x64.Build.0 = Debug|x64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Debug|x64.Deploy.0 = Debug|x64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Debug|x86.ActiveCfg = Debug|x86 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Debug|x86.Build.0 = Debug|x86 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Debug|x86.Deploy.0 = Debug|x86 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Release|ARM64.ActiveCfg = Release|ARM64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Release|ARM64.Build.0 = Release|ARM64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Release|ARM64.Deploy.0 = Release|ARM64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Release|x64.ActiveCfg = Release|x64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Release|x64.Build.0 = Release|x64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Release|x64.Deploy.0 = Release|x64 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Release|x86.ActiveCfg = Release|x86 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Release|x86.Build.0 = Release|x86 + {972D5C0D-86E6-4A2F-A6FD-8D4FE3380707}.Release|x86.Deploy.0 = Release|x86 + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {C9277197-C949-4948-80CD-0A685EE6DCBB} + EndGlobalSection +EndGlobal diff --git a/Chapter14/MyMediaCollection/App.xaml b/Chapter14/MyMediaCollection/App.xaml new file mode 100644 index 0000000..eeb9e6e --- /dev/null +++ b/Chapter14/MyMediaCollection/App.xaml @@ -0,0 +1,18 @@ + + + + + + + + + + + + + + diff --git a/Chapter14/MyMediaCollection/App.xaml.cs b/Chapter14/MyMediaCollection/App.xaml.cs new file mode 100644 index 0000000..9a2ce58 --- /dev/null +++ b/Chapter14/MyMediaCollection/App.xaml.cs @@ -0,0 +1,131 @@ +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using Microsoft.UI.Xaml; +using Microsoft.UI.Xaml.Controls; +using Microsoft.UI.Xaml.Navigation; +using Microsoft.Windows.AppLifecycle; +using Microsoft.Windows.AppNotifications; +using MyMediaCollection.Helpers; +using MyMediaCollection.Interfaces; +using MyMediaCollection.Services; +using MyMediaCollection.ViewModels; +using MyMediaCollection.Views; +using System; +using System.Runtime.InteropServices; +using System.Threading.Tasks; +using WinRT.Interop; + +namespace MyMediaCollection +{ + /// + /// Provides application-specific behavior to supplement the default Application class. + /// + public partial class App : Application + { + [DllImport("user32.dll", SetLastError = true)] + static extern void SwitchToThisWindow(IntPtr hWnd, bool turnOn); + + private NotificationManager notificationManager; + + public static IHost HostContainer { get; private set; } + + /// + /// Initializes the singleton application object. This is the first line of authored code + /// executed, and as such is the logical equivalent of main() or WinMain(). + /// + public App() + { + this.InitializeComponent(); + notificationManager = new NotificationManager(); + notificationManager.Init(); + AppDomain.CurrentDomain.ProcessExit += CurrentDomain_ProcessExit; + } + + private void CurrentDomain_ProcessExit(object sender, EventArgs e) + { + notificationManager.Unregister(); + } + + public static void ToForeground() + { + if (m_window != null) + { + IntPtr handle = WindowNative.GetWindowHandle(m_window); + if (handle != IntPtr.Zero) + { + SwitchToThisWindow(handle, true); + } + } + } + + public static string GetFullPathToExe() + { + var path = AppDomain.CurrentDomain.BaseDirectory; + var pos = path.LastIndexOf("\\"); + return path.Substring(0, pos); + } + + public static string GetFullPathToAsset(string assetName) + { + return $"{GetFullPathToExe()}\\Assets\\{assetName}"; + } + + /// + /// Invoked when the application is launched. + /// + /// Details about the launch request and process. + protected override async void OnLaunched(LaunchActivatedEventArgs args) + { + m_window = new MainWindow(); + var rootFrame = new Frame(); + await RegisterComponentsAsync(rootFrame); + rootFrame.NavigationFailed += RootFrame_NavigationFailed; + rootFrame.Navigate(typeof(MainPage), args); + m_window.Content = rootFrame; + + var currentInstance = AppInstance.GetCurrent(); + if (currentInstance.IsCurrent) + { + AppActivationArguments activationArgs = currentInstance.GetActivatedEventArgs(); + if (activationArgs != null) + { + ExtendedActivationKind extendedKind = activationArgs.Kind; + if (extendedKind == ExtendedActivationKind.AppNotification) + { + var notificationActivatedEventArgs = (AppNotificationActivatedEventArgs)activationArgs.Data; + notificationManager.ProcessLaunchActivationArgs(notificationActivatedEventArgs); + } + } + } + + m_window.Activate(); + } + + private void RootFrame_NavigationFailed(object sender, NavigationFailedEventArgs e) + { + throw new Exception($"Error loading page {e.SourcePageType.FullName}"); + } + + private static Window m_window; + + internal Window Window => m_window; + + private async Task RegisterComponentsAsync(Frame rootFrame) + { + var navigationService = new NavigationService(rootFrame); + navigationService.Configure(nameof(MainPage), typeof(MainPage)); + navigationService.Configure(nameof(ItemDetailsPage), typeof(ItemDetailsPage)); + var dataService = new SqliteDataService(); + await dataService.InitializeDataAsync(); + + HostContainer = Host.CreateDefaultBuilder() + .ConfigureServices(services => + { + services.AddSingleton(navigationService); + services.AddSingleton(dataService); + services.AddTransient(); + services.AddTransient(); + }).Build(); + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Assets/LockScreenLogo.scale-200.png b/Chapter14/MyMediaCollection/Assets/LockScreenLogo.scale-200.png new file mode 100644 index 0000000000000000000000000000000000000000..7440f0d4bf7c7e26e4e36328738c68e624ee851e GIT binary patch literal 432 zcmeAS@N?(olHy`uVBq!ia0vp^1|ZDA1|-9oezr3(FqV6|IEGZ*x-#9g>~Mkr+x6^F zy~CDX2QIMs&Gcs3RnRBoxBA!*(Mfw0KTCYuYk0WlEIV>qBmPl! zq4ukrvfADX@#p8fbLY(H47N+k`FZ(FZh?cDro7>{8mkBO3>^oaIx`3!Jl)Qq)HI!+ z(S=1{o~eT)&W^=Ea8C`-17(Jv5(nHFJ{dOjGdxLVkY_y6&S1whfuFI4MM0kF0f&cO zPDVpV%nz;Id$>+0Ga5e9625-JcI)oq=#Pa3p^>8BB}21BUw@eN!-6@w%X+^`+Vn?! zryu|3T>kVWNBYyBc=7Y6H#s1Ah!OI_nezW zXTqOdkv2Az6KKBV=$yHdF^R3Fqw(TZEoNSZX>reXJ#bwX42%f|Pgg&ebxsLQ010xn AssI20 literal 0 HcmV?d00001 diff --git a/Chapter14/MyMediaCollection/Assets/SplashScreen.scale-200.png b/Chapter14/MyMediaCollection/Assets/SplashScreen.scale-200.png new file mode 100644 index 0000000000000000000000000000000000000000..32f486a86792a5e34cd9a8261b394c49b48f86be GIT binary patch literal 5372 zcmd5=Z){Ul6u)iv53sCbIJKLzl(EF%0tzcEY@|pLrfgF~2Dk$KFtU+$kbYqDN5W%7 z>?DBo!@y06eh{Oux>brrNT^{MO(tkiC@nH(2}}G_1|uvcMD(0{?|W^Gxo!tG~hW2Rn&7%b`-Kd_^`BCrb>XVtRKONoEw6%NswzMxk+kbocuk&}kJ#hSP z>8uR{r%LJ?I#)aaWW;uEixz+DzyTpp)MTEo&R%nEA92~g{^eXQwKV1m{xl5K<@k3FacT+Z zrwfy=VocIptI>t%@p5a;Rt=WXVnU;2SUdr7Yk>gw_2z_ICK^23$|Cg7{3Eg5j@N*F zetT?>30(*S_7ld-Yt&u7T{(hEjjM#vPlXibjrq?;pBBx3*>_2~VFGdsH5L zQKme_LAebV}aOX#+rQafZtp+4jK}V!>pn1?+eUH$0%6}z(Kul9!^2z zXi+d@jnx)RW7!j9uFEdv5N&1sCW#Z6Ej5Y7c;o28Q7i%U0(2v5J>o9P zl$#C8&9r)nL;?J65^GIeSOHYr3B7}}R~}@2Tx_xo5*YdU#g1bO}95cq69J!efdlE+xj1qG#ZUqh~1Sn#dBsZfDvcupM zXOFoyJ0$s+RHQKpzr#T>c&EUbq)lGvZDxuI!9unMI=#;ob2&gT)WqOjt6^X`_N21r`&eh6h0xpT!n6Z9rvE&+bFU$vTJO2? z#^tBNOx*2N)~(+TH8d>ep6``8V=3JEfdUUahVZ-xN+k#V&32x|%qnX(XBii5<@`%^ zV#Ky4f1!6RJqJXBU3M4~tmj2;;r`8_j&w?h5g35uMH(QI$Xpesb zG|*XRT?kh6M(jj0Y&vF^M*9g-iDMW%G%9%Pa}6ERQ9b0%6z1v}Ja=|L@G#5ZI>JS9 z*(K12nMvS?oyG8s9|q~{w`ajtI`KSHSiJ;)%X@M&eCE(VqI#F(XL?L@A$TUT?6av5 zkPWIR391XjSC%d6L}7F71Qpw(;c_~)mSZo-&Fm^FHlPX|Fu}1B3E+9j0}o1a(4HFS zUItE22CC%XZi!b4%~vWn>rpV9&CUEvt!?Q{Pr*L~51&(0Sz{VJJFrJtWw2PwXd|J{ zgH%3vAY$flodH=4&ruCHX;(3t;o}n?!0~3EE|5qRz$!VIkphxa4@_jyfiE9m;0 zjcYJ2;26N&MTB8X4joZ&?SUe|VS$^I%dt{!c2O;%3SdqW@K_14r8eyC1s&VcU5+2~ z_O1Cc*w|aIA=VC6AT_EFoL}W#Rl;7CZe)e}RS*e;8CVyM6i8a(yO@|S709VYY(y2g zc+QxB>Bw^B^2Db~*o)=i$m-aUNQFkYy5(eJW$cez>C{POds*p3cy#tHnvActP;dBP zdEf)C;lq}&#PE?XCD<~ngrzYUg|nS`#MS`Rd7cT>xlR19P#~4Qg5!J}@glCUq)z_2 zjvyv%aSq0 z)njao1dV0XNw&c@qmj1e*jgQ$l@_urW5G4RSY#rT1z`#%3;{EB`aJK|TH^lb_3nAT z-_Q4X-(K&IS8UyqsnjYdippfmN-HT!X2MT;Dpcy~-#$k6V z|MR4vU#O&p7TC46pTflb3 zoUJ;ZRf#&8&EwXy5s%!&(q6cN62swD#FH%O-RJsjWPZN3^^@FCIQ&MxXIFo7!I#VI zkpIstuWqUV5uhgs07?k$*!`uiZ=5b#$lI|0c+XJvj(}zSE3MN#EyOK zql(#yA}~Ibl*r(s1}Z^5mmn*-n93g?-ccM+^PN?6HH~h0hjy6@XY*^i<-V)+OZ;p7 z7j`p_sT55xnYsedNIIel^QIIg7i@`2Qi}x5$!tk29$2OQI zs^kQXAKE}5ZJu$)2@Dxn?}}O@f@6@^!%9Tj+o>=jd!^ZuvBE4jb4g}Z5WMBtcmy^~ zoFGVS5|0FA!(1Q%fL?Bj*L+9ZL{mjSO8lzqrQ0UCZ)X zPwk$1HNFgaK%NxGpuXz}#ywXvf2JQ?BQ5uOZM2up4S#ieaxS$!o9o6Z=czNQb} zwAh|xLZ>+WyN%o?^uCAQw&&4o?S$DJ`WP(Hr*grL*qNXlqU0osCQ(Up5F(^$Z5;n&oJIO4uF`k&QL*j{f zU=;#MZ5{@b%qMbjTB3dh-5#mqY>%{0jgS+WdHyG literal 0 HcmV?d00001 diff --git a/Chapter14/MyMediaCollection/Assets/Square44x44Logo.scale-200.png b/Chapter14/MyMediaCollection/Assets/Square44x44Logo.scale-200.png new file mode 100644 index 0000000000000000000000000000000000000000..f713bba67f551ef91020b75716a4dc8ebd744b1c GIT binary patch literal 637 zcmeAS@N?(olHy`uVBq!ia0vp^5g^RL1|$oo8kjIJFu8cTIEGZ*dUI*J;2{SImxtDO zm%3!R$UazoY}x{$j0P5ABYXWr(l=jxJ6ps1W{tV=^>{Dl><3nv3A}sm=EZ)#l3`NR zpZda3^rNox*D1%NC98Z~L*6zipLw~Gxn&(Y-;KmJ+aR6eLabU-L#y8HW%7P-E_-VlLqIabbHPHKT*)fT@9iWJ7iWgOT9%0}Lrj>lztPxWq6sPw3pi z#-<=#$jjrP_DD*i!RLsn0mIA=>4~N)IMYWIf=j%-zuKCdMG%tHYot70D1| zvWa0wMhauW#S>1CnI_;>!1Q3zMA17@DOVq{MQ+{U7^a&yA+%dMCG;WNPV0i;w$tu; zX^b}UKziPM)(<;)ruW;-`)bBN+rQNM*Zs_>?n$|FVFo-e*PZb*@U7VAd+tHb4e?=Blc~}S6K)wL}r*Gf`BM#QB z+y>N$mCswb4d{^{S9v_!eQj4fTRMOwOCi?lSk9%<=vAz}jM-*PQtH@Odn1LZcd^j#o> hW$4xn+CT+ep9lJ{OAO?njobhL002ovPDHLkV1nYebbkN< literal 0 HcmV?d00001 diff --git a/Chapter14/MyMediaCollection/Assets/StoreLogo.png b/Chapter14/MyMediaCollection/Assets/StoreLogo.png new file mode 100644 index 0000000000000000000000000000000000000000..a4586f26bdf7841cad10f39cdffe2aca3af252c1 GIT binary patch literal 456 zcmeAS@N?(olHy`uVBq!ia0vp^Mj*_=1|;R|J2o;fF!p=8IEGZ*dUM0H=rDtTTVkd2 z(%lbKn@VS_lUaADVB&;Z6F#LM+mPsa?e>FnHo;HND^!P`-lX%BH~FOg%y&x+t*x!? zg$#_1A1kgsSvO(fw`bOmo;lrJX8byO1j^gf7qohR%mmt z@L)WX;>gqgK|tWJvQ5j;4;=gt4HXVKSMYRv5RhY5vS~TqfK_NAP*r{h!!g^BZ;w4r z7CGdsai)y;fJQc`7{Zc2b==h%o`Op$|bg6a&nL{*m7-=0>k4M4-PXlU;G-?%*(*g>iFt^ U$m#7DfHB12>FVdQ&MBb@0G`#n8vpc0sq%A~kJcD9FY~qQRMt?ZR3YyDZt}Od;|mgpc{2dv9AHF){kXU%k({ z=Y8JidEayHTkG@twPZ|U3_^%3ct-OgLSiFAqDN!|tbCX@c@?4P`2x*TMK!+Q4b?k0 ziW7!!KF6dPWcF<%I|iznM~`QJ_V7sHGV_D`dhgpA9Vd@&X}ErK+j~_rdv;Bp?OA@a zFXOk7eWOJe5NcK;6h$FaM&7JxNc#-@QTwzW6x#d_zmQNkz5) zPI;kh;3d;5UCJU+9a(cOxX(|edWoOiAEdGU#kPJ&xnc2||3vDbuhBCkj-pb0as$Zl z5;}4n=**n6(1g`JEtSy;SG6X;#-F~Oz3lESG2b5`j@wAwY4Yp<=4Xeb>iH=6aicF?DxD&q{`!&}ct zBI)aycwuobQAf&678Uf+Mmh-@9RUhyH~>?w0dixO0#jZjEc9R^=5NZw=|a(kcB?9^ zfnTiEFXp-q#B;Tn>(O%$A*ud^Rg&eVH6Y_5Y%!E39RR&s?XpG`gKwU!6FE1 z7X)DC7)*(5g}lh`4`{i~DZcWupZI`K)_4P)VE{@gc7@Xsd^86zl~_mOYH?I4!aGeX z^E(_=L6?PgveDQ+r%P@UISEXrkn`LHJZ##+!-anV>6h)IkKp;E@p8+3&(5%kS2)ld*J*rJccZM0iyaAx7+F~GW1UWFK&3X$PE1^}NH zgAG9ck5K!{07OwU@j@Do>TbH=CDEo#4m0cEyAuXy_<&jlzJVcKweSJ5 z&=q~iIn18$w8yb=rmEmHxVEUA^?RwnB?6Qlp1os8@*dWTGL2bhzZ!s*xqScR?EPL` zo(JwNdKUUYy7GtvZ3asXm)cgFvCx9EmAi;|w=a0iGiv%%VYKh`P0Wma4y`Xyx|T~( zAmfGbgbEEC7)j8b@WA@+5W3a61HJXC1dX@6_T|Czk0I0zBk%tnW~()VWITGI!`$c< gARL?UBrYYkwoDw4eo*CrzXGTrZ@;GF>596)00d&n@&Et; literal 0 HcmV?d00001 diff --git a/Chapter14/MyMediaCollection/Enums/ItemType.cs b/Chapter14/MyMediaCollection/Enums/ItemType.cs new file mode 100644 index 0000000..2e50873 --- /dev/null +++ b/Chapter14/MyMediaCollection/Enums/ItemType.cs @@ -0,0 +1,9 @@ +namespace MyMediaCollection.Enums +{ + public enum ItemType + { + Music, + Video, + Book + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Enums/LocationType.cs b/Chapter14/MyMediaCollection/Enums/LocationType.cs new file mode 100644 index 0000000..a8d19aa --- /dev/null +++ b/Chapter14/MyMediaCollection/Enums/LocationType.cs @@ -0,0 +1,8 @@ +namespace MyMediaCollection.Enums +{ + public enum LocationType + { + InCollection, + Loaned + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Helpers/NotificationManager.cs b/Chapter14/MyMediaCollection/Helpers/NotificationManager.cs new file mode 100644 index 0000000..a9fa398 --- /dev/null +++ b/Chapter14/MyMediaCollection/Helpers/NotificationManager.cs @@ -0,0 +1,88 @@ +using Microsoft.Windows.AppNotifications; +using System; +using System.Collections.Generic; + +namespace MyMediaCollection.Helpers +{ + internal class NotificationManager + { + private bool isRegistered; + private Dictionary> notificationHandlers; + + public NotificationManager() + { + isRegistered = false; + + // When adding new a scenario, be sure to add its notification handler here. + notificationHandlers = new Dictionary> + { + { ToastWithAvatar.ScenarioId, ToastWithAvatar.NotificationReceived }, + { ToastWithText.ScenarioId, ToastWithText.NotificationReceived } + }; + } + + ~NotificationManager() + { + Unregister(); + } + + public void Unregister() + { + if (isRegistered) + { + AppNotificationManager.Default.Unregister(); + isRegistered = false; + } + } + + public void Init() + { + AppNotificationManager notificationManager = AppNotificationManager.Default; + + // Add handler before calling Register. + notificationManager.NotificationInvoked += OnNotificationInvoked; + notificationManager.Register(); + + isRegistered = true; + } + + public void ProcessLaunchActivationArgs(AppNotificationActivatedEventArgs notificationActivatedEventArgs) + { + DispatchNotification(notificationActivatedEventArgs); + NotificationShared.AppLaunchedFromNotification(); + } + + private bool DispatchNotification(AppNotificationActivatedEventArgs notificationActivatedEventArgs) + { + var scenarioId = notificationActivatedEventArgs.Arguments[NotificationShared.scenarioTag]; + if (scenarioId.Length != 0) + { + try + { + notificationHandlers[int.Parse(scenarioId)](notificationActivatedEventArgs); + return true; + } + catch + { + // No matching NotificationHandler for scenarioId. + return false; + } + } + else + { + // No scenarioId provided + return false; + } + } + + public void OnNotificationInvoked(object sender, AppNotificationActivatedEventArgs notificationActivatedEventArgs) + { + NotificationShared.NotificationReceived(); + + if (!DispatchNotification(notificationActivatedEventArgs)) + { + NotificationShared.UnrecognizedToastOriginator(); + } + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Helpers/NotificationShared.cs b/Chapter14/MyMediaCollection/Helpers/NotificationShared.cs new file mode 100644 index 0000000..8fad23a --- /dev/null +++ b/Chapter14/MyMediaCollection/Helpers/NotificationShared.cs @@ -0,0 +1,43 @@ +using Microsoft.UI.Xaml.Controls; +using MyMediaCollection.Views; + +namespace MyMediaCollection.Helpers +{ + public class NotificationShared + { + public const string scenarioTag = "scenarioId"; + + public struct Notification + { + public string Originator; + public string Action; + public bool HasInput; + public string Input; + }; + + public static void CouldNotSendToast() + { + MainPage.Current.NotifyUser("Could not send toast", InfoBarSeverity.Error); + } + + public static void ToastSentSuccessfully() + { + MainPage.Current.NotifyUser("Toast sent successfully!", InfoBarSeverity.Success); + } + + public static void AppLaunchedFromNotification() + { + MainPage.Current.NotifyUser("App launched from notifications", InfoBarSeverity.Informational); + } + + public static void NotificationReceived() + { + MainPage.Current.NotifyUser("Notification received", InfoBarSeverity.Informational); + } + + public static void UnrecognizedToastOriginator() + { + MainPage.Current.NotifyUser("Unrecognized Toast Originator or Unknown Error", InfoBarSeverity.Error); + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Helpers/ToastWithAvatar.cs b/Chapter14/MyMediaCollection/Helpers/ToastWithAvatar.cs new file mode 100644 index 0000000..049946b --- /dev/null +++ b/Chapter14/MyMediaCollection/Helpers/ToastWithAvatar.cs @@ -0,0 +1,42 @@ +using Microsoft.Windows.AppNotifications.Builder; +using Microsoft.Windows.AppNotifications; +using MyMediaCollection.Views; + +namespace MyMediaCollection.Helpers +{ + public class ToastWithAvatar + { + public const int ScenarioId = 1; + public const string ScenarioName = "Local Toast with Image"; + + public static bool SendToast() + { + var appNotification = new AppNotificationBuilder() + .AddArgument("action", "ToastClick") + .AddArgument(NotificationShared.scenarioTag, ScenarioId.ToString()) + .SetAppLogoOverride(new System.Uri($"file://{App.GetFullPathToAsset("Square150x150Logo.scale-200.png")}"), AppNotificationImageCrop.Circle) + .AddText(ScenarioName) + .AddText("This is a notification message.") + .AddButton(new AppNotificationButton("Open App") + .AddArgument("action", "OpenApp") + .AddArgument(NotificationShared.scenarioTag, ScenarioId.ToString())) + .BuildNotification(); + + AppNotificationManager.Default.Show(appNotification); + + // If notification is sent, it will have an Id. Success. + return appNotification.Id != 0; + } + + public static void NotificationReceived(AppNotificationActivatedEventArgs notificationActivatedEventArgs) + { + var notification = new NotificationShared.Notification + { + Originator = ScenarioName, + Action = notificationActivatedEventArgs.Arguments["action"] + }; + MainPage.Current.NotificationReceived(notification); + App.ToForeground(); + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Helpers/ToastWithText.cs b/Chapter14/MyMediaCollection/Helpers/ToastWithText.cs new file mode 100644 index 0000000..e383298 --- /dev/null +++ b/Chapter14/MyMediaCollection/Helpers/ToastWithText.cs @@ -0,0 +1,47 @@ +using Microsoft.Windows.AppNotifications.Builder; +using Microsoft.Windows.AppNotifications; +using MyMediaCollection.Views; + +namespace MyMediaCollection.Helpers +{ + public class ToastWithText + { + public const int ScenarioId = 2; + public const string ScenarioName = "Local Toast with Image and Text Entry"; + const string textboxReplyId = "textboxReply"; + + public static bool SendToast() + { + var appNotification = new AppNotificationBuilder() + .AddArgument("action", "ToastClick") + .AddArgument(NotificationShared.scenarioTag, ScenarioId.ToString()) + .SetAppLogoOverride(new System.Uri($"file://{App.GetFullPathToAsset("Square150x150Logo.scale-200.png")}"), AppNotificationImageCrop.Circle) + .AddText(ScenarioName) + .AddText("This is a notification message.") + .AddTextBox(textboxReplyId, "Enter a reply", "Reply box") + .AddButton(new AppNotificationButton("Reply") + .AddArgument("action", "Reply") + .AddArgument(NotificationShared.scenarioTag, ScenarioId.ToString()) + .SetInputId(textboxReplyId)) + .BuildNotification(); + + AppNotificationManager.Default.Show(appNotification); + + // If notification is sent, it will have an Id. Success. + return appNotification.Id != 0; + } + + public static void NotificationReceived(AppNotificationActivatedEventArgs notificationActivatedEventArgs) + { + var notification = new NotificationShared.Notification + { + Originator = ScenarioName, + Action = notificationActivatedEventArgs.Arguments["action"], + HasInput = true, + Input = notificationActivatedEventArgs.UserInput[textboxReplyId] + }; + MainPage.Current.NotificationReceived(notification); + App.ToForeground(); + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Interfaces/IDataService.cs b/Chapter14/MyMediaCollection/Interfaces/IDataService.cs new file mode 100644 index 0000000..7d6721b --- /dev/null +++ b/Chapter14/MyMediaCollection/Interfaces/IDataService.cs @@ -0,0 +1,22 @@ +using MyMediaCollection.Enums; +using MyMediaCollection.Model; +using System.Collections.Generic; +using System.Threading.Tasks; + +namespace MyMediaCollection.Interfaces +{ + public interface IDataService + { + Task> GetItemsAsync(); + Task GetItemAsync(int id); + Task AddItemAsync(MediaItem item); + Task UpdateItemAsync(MediaItem item); + Task DeleteItemAsync(MediaItem item); + IList GetItemTypes(); + Medium GetMedium(string name); + IList GetMediums(); + IList GetMediums(ItemType itemType); + IList GetLocationTypes(); + Task InitializeDataAsync(); + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Interfaces/INavigationService.cs b/Chapter14/MyMediaCollection/Interfaces/INavigationService.cs new file mode 100644 index 0000000..c8ccee6 --- /dev/null +++ b/Chapter14/MyMediaCollection/Interfaces/INavigationService.cs @@ -0,0 +1,12 @@ +using System; + +namespace MyMediaCollection.Interfaces +{ + public interface INavigationService + { + string CurrentPage { get; } + void NavigateTo(string page); + void NavigateTo(string page, object parameter); + void GoBack(); + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Interfaces/IValidatable.cs b/Chapter14/MyMediaCollection/Interfaces/IValidatable.cs new file mode 100644 index 0000000..95cebf4 --- /dev/null +++ b/Chapter14/MyMediaCollection/Interfaces/IValidatable.cs @@ -0,0 +1,7 @@ +namespace MyMediaCollection.Interfaces +{ + public interface IValidatable + { + void Validate(string memberName, object value); + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/MainWindow.xaml b/Chapter14/MyMediaCollection/MainWindow.xaml new file mode 100644 index 0000000..7ebe85e --- /dev/null +++ b/Chapter14/MyMediaCollection/MainWindow.xaml @@ -0,0 +1,10 @@ + + + \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/MainWindow.xaml.cs b/Chapter14/MyMediaCollection/MainWindow.xaml.cs new file mode 100644 index 0000000..6da1b61 --- /dev/null +++ b/Chapter14/MyMediaCollection/MainWindow.xaml.cs @@ -0,0 +1,47 @@ +using Microsoft.UI; +using Microsoft.UI.Composition.SystemBackdrops; +using Microsoft.UI.Windowing; +using Microsoft.UI.Xaml; +using System; +using WinRT.Interop; +using Microsoft.UI.Xaml.Media; + +namespace MyMediaCollection +{ + /// + /// An empty window that can be used on its own or navigated to within a Frame. + /// + public sealed partial class MainWindow : Window + { + private AppWindow _appWindow; + private const string AppTitle = "My Media Collection"; + + public MainWindow() + { + this.InitializeComponent(); + SystemBackdrop = new MicaBackdrop + { + Kind = MicaKind.BaseAlt + }; + _appWindow = GetCurrentAppWindow(); + _appWindow.Title = AppTitle; + } + + private AppWindow GetCurrentAppWindow() + { + IntPtr handle = WindowNative.GetWindowHandle(this); + WindowId windowId = Win32Interop.GetWindowIdFromWindow(handle); + return AppWindow.GetFromWindowId(windowId); + } + + internal void SetPageTitle(string title) + { + if (_appWindow == null) + { + _appWindow = GetCurrentAppWindow(); + } + + _appWindow.Title = $"{AppTitle} - {title}"; + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Model/MediaItem.cs b/Chapter14/MyMediaCollection/Model/MediaItem.cs new file mode 100644 index 0000000..329af7f --- /dev/null +++ b/Chapter14/MyMediaCollection/Model/MediaItem.cs @@ -0,0 +1,17 @@ +using Dapper.Contrib.Extensions; +using MyMediaCollection.Enums; + +namespace MyMediaCollection.Model +{ + public class MediaItem + { + [Key] + public int Id { get; set; } + public string Name { get; set; } + public ItemType MediaType { get; set; } + [Computed] + public Medium MediumInfo { get; set; } + public LocationType Location { get; set; } + public int MediumId => MediumInfo.Id; + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Model/Medium.cs b/Chapter14/MyMediaCollection/Model/Medium.cs new file mode 100644 index 0000000..3c23205 --- /dev/null +++ b/Chapter14/MyMediaCollection/Model/Medium.cs @@ -0,0 +1,13 @@ +using Dapper.Contrib.Extensions; +using MyMediaCollection.Enums; + +namespace MyMediaCollection.Model +{ + public class Medium + { + [Key] + public int Id { get; set; } + public string Name { get; set; } + public ItemType MediaType { get; set; } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/MyMediaCollection.csproj b/Chapter14/MyMediaCollection/MyMediaCollection.csproj new file mode 100644 index 0000000..60a8ea6 --- /dev/null +++ b/Chapter14/MyMediaCollection/MyMediaCollection.csproj @@ -0,0 +1,79 @@ + + + WinExe + net6.0-windows10.0.19041.0 + 10.0.17763.0 + MyMediaCollection + app.manifest + x86;x64;ARM64 + win10-x86;win10-x64;win10-arm64 + win10-$(Platform).pubxml + true + true + 10.0.18362.0 + False + True + CF7197F6406374B9861069CD5BE203E41888B4BA + SHA256 + True + False + True + Never + C:\Installers\MyMediaCollection + 0 + True + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + MSBuild:Compile + + + + + MSBuild:Compile + + + + + + true + + diff --git a/Chapter14/MyMediaCollection/Package.appxmanifest b/Chapter14/MyMediaCollection/Package.appxmanifest new file mode 100644 index 0000000..9b2994a --- /dev/null +++ b/Chapter14/MyMediaCollection/Package.appxmanifest @@ -0,0 +1,63 @@ + + + + + + + + My Media Collection + Alvinitech + Assets\StoreLogo.png + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Chapter14/MyMediaCollection/Properties/launchSettings.json b/Chapter14/MyMediaCollection/Properties/launchSettings.json new file mode 100644 index 0000000..b7ae329 --- /dev/null +++ b/Chapter14/MyMediaCollection/Properties/launchSettings.json @@ -0,0 +1,10 @@ +{ + "profiles": { + "MyMediaCollection (Package)": { + "commandName": "MsixPackage" + }, + "MyMediaCollection (Unpackaged)": { + "commandName": "Project" + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Services/DataService.cs b/Chapter14/MyMediaCollection/Services/DataService.cs new file mode 100644 index 0000000..a81fcab --- /dev/null +++ b/Chapter14/MyMediaCollection/Services/DataService.cs @@ -0,0 +1,175 @@ +using MyMediaCollection.Enums; +using MyMediaCollection.Interfaces; +using MyMediaCollection.Model; +using System; +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; + +namespace MyMediaCollection.Services +{ + public class DataService : IDataService + { + private IList _items; + private IList _itemTypes; + private IList _mediums; + private IList _locationTypes; + + public async Task InitializeDataAsync() + { + PopulateItemTypes(); + PopulateMediums(); + PopulateLocationTypes(); + PopulateItems(); + await Task.Delay(1); + } + + private void PopulateItems() + { + var cd = new MediaItem + { + Id = 1, + Name = "Classical Favorites", + MediaType = ItemType.Music, + MediumInfo = _mediums.FirstOrDefault(m => m.Name == "CD"), + Location = LocationType.InCollection + }; + + var book = new MediaItem + { + Id = 2, + Name = "Classic Fairy Tales", + MediaType = ItemType.Book, + MediumInfo = _mediums.FirstOrDefault(m => m.Name == "Hardcover"), + Location = LocationType.InCollection + }; + + var bluRay = new MediaItem + { + Id = 3, + Name = "The Mummy", + MediaType = ItemType.Video, + MediumInfo = _mediums.FirstOrDefault(m => m.Name == "Blu Ray"), + Location = LocationType.InCollection + }; + + _items = new List + { + cd, + book, + bluRay + }; + } + + private void PopulateMediums() + { + var cd = new Medium { Id = 1, MediaType = ItemType.Music, Name = "CD" }; + var vinyl = new Medium { Id = 2, MediaType = ItemType.Music, Name = "Vinyl" }; + var hardcover = new Medium { Id = 3, MediaType = ItemType.Book, Name = "Hardcover" }; + var paperback = new Medium { Id = 4, MediaType = ItemType.Book, Name = "Paperback" }; + var dvd = new Medium { Id = 5, MediaType = ItemType.Video, Name = "DVD" }; + var bluRay = new Medium { Id = 6, MediaType = ItemType.Video, Name = "Blu Ray" }; + + _mediums = new List + { + cd, + vinyl, + hardcover, + paperback, + dvd, + bluRay + }; + } + + private void PopulateItemTypes() + { + _itemTypes = new List + { + ItemType.Book, + ItemType.Music, + ItemType.Video + }; + } + + private void PopulateLocationTypes() + { + _locationTypes = new List + { + LocationType.InCollection, + LocationType.Loaned + }; + } + + public async Task AddItemAsync(MediaItem item) + { + item.Id = _items.Max(i => i.Id) + 1; + _items.Add(item); + await Task.Delay(1); + + return item.Id; + } + + public async Task GetItemAsync(int id) + { + await Task.Delay(1); + return _items.FirstOrDefault(i => i.Id == id); + } + + public async Task> GetItemsAsync() + { + await Task.Delay(1); + return _items; + } + + public IList GetItemTypes() + { + return _itemTypes; + } + + public IList GetMediums() + { + return _mediums; + } + + public IList GetMediums(ItemType itemType) + { + return _mediums + .Where(m => m.MediaType == itemType) + .ToList(); + } + + public IList GetLocationTypes() + { + return _locationTypes; + } + + public async Task UpdateItemAsync(MediaItem item) + { + var idx = -1; + var matchedItem = + (from x in _items + let ind = idx++ + where x.Id == item.Id + select ind).FirstOrDefault(); + + if (idx == -1) + { + throw new Exception("Unable to update item. Item not found in collection."); + } + + _items[idx] = item; + await Task.Delay(1); + } + + public async Task DeleteItemAsync(MediaItem item) + { + await Task.Delay(1); + throw new NotImplementedException(); + } + + public Medium GetMedium(string name) + { + return _mediums.FirstOrDefault(m => m.Name == name); + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Services/NavigationService.cs b/Chapter14/MyMediaCollection/Services/NavigationService.cs new file mode 100644 index 0000000..63f8c95 --- /dev/null +++ b/Chapter14/MyMediaCollection/Services/NavigationService.cs @@ -0,0 +1,84 @@ +using Microsoft.UI.Xaml.Controls; +using MyMediaCollection.Interfaces; +using System; +using System.Collections.Concurrent; +using System.Collections.Generic; +using System.Linq; + +namespace MyMediaCollection.Services +{ + public class NavigationService : INavigationService + { + public NavigationService(Frame rootFrame) + { + AppFrame = rootFrame; + } + + private readonly IDictionary _pages = new ConcurrentDictionary(); + + public const string RootPage = "(Root)"; + + public const string UnknownPage = "(Unknown)"; + + private static Frame AppFrame; + + public void Configure(string page, Type type) + { + if (_pages.Values.Any(v => v == type)) + { + throw new ArgumentException($"The {type.Name} view has already been registered under another name."); + } + + _pages[page] = type; + } + + /// + /// Gets the name of the currently displayed page. + /// + public string CurrentPage + { + get + { + var frame = AppFrame; + + if (frame.BackStackDepth == 0) + return RootPage; + + if (frame.Content == null) + return UnknownPage; + + var type = frame.Content.GetType(); + + if (_pages.Values.All(v => v != type)) + return UnknownPage; + + var item = _pages.Single(i => i.Value == type); + + return item.Key; + } + } + + public void NavigateTo(string page) + { + NavigateTo(page, null); + } + + public void NavigateTo(string page, object parameter) + { + if (!_pages.ContainsKey(page)) + { + throw new ArgumentException($"Unable to find a page registered with the name {page}."); + } + + AppFrame.Navigate(_pages[page], parameter); + } + + public void GoBack() + { + if (AppFrame?.CanGoBack == true) + { + AppFrame.GoBack(); + } + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Services/SqliteDataService.cs b/Chapter14/MyMediaCollection/Services/SqliteDataService.cs new file mode 100644 index 0000000..5a88079 --- /dev/null +++ b/Chapter14/MyMediaCollection/Services/SqliteDataService.cs @@ -0,0 +1,300 @@ +using Dapper; +using Dapper.Contrib.Extensions; +using Microsoft.Data.Sqlite; +using MyMediaCollection.Enums; +using MyMediaCollection.Interfaces; +using MyMediaCollection.Model; +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Threading.Tasks; +using Windows.Storage; + +namespace MyMediaCollection.Services +{ + public class SqliteDataService : IDataService + { + private IList _itemTypes; + private IList _mediums; + private IList _locationTypes; + private const string DbName = "mediaCollectionData.db"; + + private async Task GetOpenConnectionAsync() + { + await ApplicationData.Current.LocalFolder.CreateFileAsync(DbName, CreationCollisionOption.OpenIfExists).AsTask(); + + string dbPath = Path.Combine(ApplicationData.Current.LocalFolder.Path, DbName); + var cn = new SqliteConnection($"Filename={dbPath}"); + cn.Open(); + + return cn; + } + + private async Task CreateMediumTableAsync(SqliteConnection db) + { + string tableCommand = @"CREATE TABLE IF NOT + EXISTS Mediums (Id INTEGER PRIMARY KEY AUTOINCREMENT, + Name NVARCHAR(30) NOT NULL, + MediumType INTEGER NOT NULL)"; + + using var createTable = new SqliteCommand(tableCommand, db); + + await createTable.ExecuteNonQueryAsync(); + } + + private async Task CreateMediaItemTableAsync(SqliteConnection db) + { + string tableCommand = @"CREATE TABLE IF NOT + EXISTS MediaItems (Id INTEGER PRIMARY KEY AUTOINCREMENT, + Name NVARCHAR(1000) NOT NULL, + ItemType INTEGER NOT NULL, + MediumId INTEGER NOT NULL, + LocationType INTEGER, + CONSTRAINT fk_mediums + FOREIGN KEY(MediumId) + REFERENCES Mediums(Id))"; + + using var createTable = new SqliteCommand(tableCommand, db); + + await createTable.ExecuteNonQueryAsync(); + } + + private async Task InsertMediumAsync(SqliteConnection db, Medium medium) + { + //// IMPLEMENTATION WITHOUT DAPPER + //using var insertCommand = new SqliteCommand + //{ + // Connection = db, + // CommandText = "INSERT INTO Mediums VALUES (NULL, @Name, @MediumType);" + //}; + //insertCommand.Parameters.AddWithValue("@Name", medium.Name); + //insertCommand.Parameters.AddWithValue("@MediumType", (int)medium.MediaType); + //await insertCommand.ExecuteNonQueryAsync(); + + var newIds = await db.QueryAsync( + $@"INSERT INTO Mediums + ({nameof(medium.Name)}, MediumType) + VALUES + (@{nameof(medium.Name)}, @{nameof(medium.MediaType)}); + SELECT last_insert_rowid()", medium); + + medium.Id = (int)newIds.First(); + } + + private async Task> GetAllMediumsAsync(SqliteConnection db) + { + //// IMPLEMENTATION WITHOUT DAPPER + //IList mediums = new List(); + //using var selectCommand = new SqliteCommand("SELECT Id, Name, MediumType FROM Mediums", db); + //using SqliteDataReader query = await selectCommand.ExecuteReaderAsync(); + + //while (query.Read()) + //{ + // var medium = new Medium + // { + // Id = query.GetInt32(0), + // Name = query.GetString(1), + // MediaType = (ItemType)query.GetInt32(2) + // }; + // mediums.Add(medium); + //} + + //return mediums; + + var mediums = + await db.QueryAsync(@"SELECT Id, + Name, + MediumType AS MediaType + FROM Mediums"); + + return mediums.ToList(); + } + + private async Task> GetAllMediaItemsAsync(SqliteConnection db) + { + var itemsResult = await db.QueryAsync + ( + @"SELECT + [MediaItems].[Id], + [MediaItems].[Name], + [MediaItems].[ItemType] AS MediaType, + [MediaItems].[LocationType] AS Location, + [Mediums].[Id], + [Mediums].[Name], + [Mediums].[MediumType] AS MediaType + FROM + [MediaItems] + JOIN + [Mediums] + ON + [Mediums].[Id] = [MediaItems].[MediumId]", + (item, medium) => + { + item.MediumInfo = medium; + return item; + } + ); + + return itemsResult.ToList(); + } + + private async Task InsertMediaItemAsync(SqliteConnection db, MediaItem item) + { + var newIds = await db.QueryAsync( + @"INSERT INTO MediaItems + (Name, ItemType, MediumId, LocationType) + VALUES + (@Name, @MediaType, @MediumId, @Location); + SELECT last_insert_rowid()", item); + + return (int)newIds.First(); + } + + private async Task UpdateMediaItemAsync(SqliteConnection db, MediaItem item) + { + await db.QueryAsync( + @"UPDATE MediaItems + SET Name = @Name, + ItemType = @MediaType, + MediumId = @MediumId, + LocationType = @Location + WHERE Id = @Id;", item); + } + + private async Task DeleteMediaItemAsync(SqliteConnection db, int id) + { + await db.DeleteAsync(new MediaItem { Id = id }); + } + + private async Task PopulateMediumsAsync(SqliteConnection db) + { + _mediums = await GetAllMediumsAsync(db); + + if (_mediums.Count == 0) + { + var cd = new Medium { Id = 1, MediaType = ItemType.Music, Name = "CD" }; + var vinyl = new Medium { Id = 2, MediaType = ItemType.Music, Name = "Vinyl" }; + var hardcover = new Medium { Id = 3, MediaType = ItemType.Book, Name = "Hardcover" }; + var paperback = new Medium { Id = 4, MediaType = ItemType.Book, Name = "Paperback" }; + var dvd = new Medium { Id = 5, MediaType = ItemType.Video, Name = "DVD" }; + var bluRay = new Medium { Id = 6, MediaType = ItemType.Video, Name = "Blu Ray" }; + + var mediums = new List + { + cd, + vinyl, + hardcover, + paperback, + dvd, + bluRay + }; + + foreach (var medium in mediums) + { + await InsertMediumAsync(db, medium); + } + + _mediums = await GetAllMediumsAsync(db); + } + } + + private void PopulateItemTypes() + { + _itemTypes = new List + { + ItemType.Book, + ItemType.Music, + ItemType.Video + }; + } + + private void PopulateLocationTypes() + { + _locationTypes = new List + { + LocationType.InCollection, + LocationType.Loaned + }; + } + + public async Task InitializeDataAsync() + { + using (var db = await GetOpenConnectionAsync()) + { + await CreateMediumTableAsync(db); + await CreateMediaItemTableAsync(db); + + PopulateItemTypes(); + await PopulateMediumsAsync(db); + PopulateLocationTypes(); + } + } + + public async Task AddItemAsync(MediaItem item) + { + using var db = await GetOpenConnectionAsync(); + return await InsertMediaItemAsync(db, item); + } + + public async Task GetItemAsync(int id) + { + IList mediaItems; + using var db = await GetOpenConnectionAsync(); + mediaItems = await GetAllMediaItemsAsync(db); + + //Filter the list to get the item for our Id + return mediaItems.FirstOrDefault(i => i.Id == id); + } + + public async Task> GetItemsAsync() + { + using var db = await GetOpenConnectionAsync(); + return await GetAllMediaItemsAsync(db); + } + + public IList GetItemTypes() + { + return _itemTypes; + } + + public IList GetLocationTypes() + { + return _locationTypes; + } + + public Medium GetMedium(string name) + { + return _mediums.FirstOrDefault(m => m.Name == name); + } + + public Medium GetMedium(int id) + { + return _mediums.FirstOrDefault(m => m.Id == id); + } + + public IList GetMediums() + { + return _mediums; + } + + public IList GetMediums(ItemType itemType) + { + return _mediums + .Where(m => m.MediaType == itemType) + .ToList(); + } + + public async Task UpdateItemAsync(MediaItem item) + { + using var db = await GetOpenConnectionAsync(); + await UpdateMediaItemAsync(db, item); + } + + public async Task DeleteItemAsync(MediaItem item) + { + using var db = await GetOpenConnectionAsync(); + await DeleteMediaItemAsync(db, item.Id); + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/ViewModels/ItemDetailsViewModel.cs b/Chapter14/MyMediaCollection/ViewModels/ItemDetailsViewModel.cs new file mode 100644 index 0000000..1054800 --- /dev/null +++ b/Chapter14/MyMediaCollection/ViewModels/ItemDetailsViewModel.cs @@ -0,0 +1,142 @@ +using CommunityToolkit.Mvvm.ComponentModel; +using CommunityToolkit.Mvvm.Input; +using MyMediaCollection.Enums; +using MyMediaCollection.Interfaces; +using MyMediaCollection.Model; +using System; +using System.Collections.ObjectModel; +using System.Linq; +using System.Threading.Tasks; + +namespace MyMediaCollection.ViewModels +{ + public partial class ItemDetailsViewModel : ObservableObject + { + [ObservableProperty] + private ObservableCollection locationTypes = new(); + [ObservableProperty] + private ObservableCollection mediums = new(); + [ObservableProperty] + private ObservableCollection itemTypes = new(); + private int _itemId; + [ObservableProperty] + private string itemName; + [ObservableProperty] + private string selectedMedium; + [ObservableProperty] + private string selectedItemType; + [ObservableProperty] + private string selectedLocation; + [ObservableProperty] + private bool isDirty; + private int _selectedItemId = -1; + protected INavigationService _navigationService; + protected IDataService _dataService; + + public ItemDetailsViewModel(INavigationService navigationService, IDataService dataService) + { + _navigationService = navigationService; + _dataService = dataService; + + PopulateLists(); + } + + public void InitializeItemDetailData(int itemId) + { + _selectedItemId = itemId; + IsDirty = false; + } + + private void PopulateLists() + { + ItemTypes.Clear(); + foreach (string iType in Enum.GetNames(typeof(ItemType))) + ItemTypes.Add(iType); + + LocationTypes.Clear(); + foreach (string lType in Enum.GetNames(typeof(LocationType))) + LocationTypes.Add(lType); + + Mediums = new ObservableCollection(); + } + + private async Task SaveAsync() + { + MediaItem item; + + if (_itemId > 0) + { + item = await _dataService.GetItemAsync(_itemId); + + item.Name = ItemName; + item.Location = (LocationType)Enum.Parse(typeof(LocationType), SelectedLocation); + item.MediaType = (ItemType)Enum.Parse(typeof(ItemType), SelectedItemType); + item.MediumInfo = _dataService.GetMedium(SelectedMedium); + + await _dataService.UpdateItemAsync(item); + } + else + { + item = new MediaItem + { + Name = ItemName, + Location = (LocationType)Enum.Parse(typeof(LocationType), SelectedLocation), + MediaType = (ItemType)Enum.Parse(typeof(ItemType), SelectedItemType), + MediumInfo = _dataService.GetMedium(SelectedMedium) + }; + + await _dataService.AddItemAsync(item); + } + } + + public async Task SaveItemAndContinueAsync() + { + await SaveAsync(); + _itemId = 0; + ItemName = string.Empty; + SelectedMedium = null; + SelectedLocation = null; + SelectedItemType = null; + IsDirty = false; + } + + public async Task SaveItemAndReturnAsync() + { + await SaveAsync(); + _navigationService.GoBack(); + } + + partial void OnItemNameChanged(string value) + { + IsDirty = true; + } + + partial void OnSelectedMediumChanged(string value) + { + IsDirty = true; + } + + partial void OnSelectedItemTypeChanged(string value) + { + IsDirty = true; + Mediums.Clear(); + + if (!string.IsNullOrWhiteSpace(value)) + { + foreach (string med in _dataService.GetMediums((ItemType)Enum.Parse(typeof(ItemType), SelectedItemType)).Select(m => m.Name)) + Mediums.Add(med); + } + } + + partial void OnSelectedLocationChanged(string value) + { + IsDirty = true; + } + + [RelayCommand] + private void Cancel() + { + _navigationService.GoBack(); + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/ViewModels/MainViewModel.cs b/Chapter14/MyMediaCollection/ViewModels/MainViewModel.cs new file mode 100644 index 0000000..d718f18 --- /dev/null +++ b/Chapter14/MyMediaCollection/ViewModels/MainViewModel.cs @@ -0,0 +1,121 @@ +using CommunityToolkit.Mvvm.ComponentModel; +using CommunityToolkit.Mvvm.Input; +using Microsoft.UI.Xaml.Input; +using MyMediaCollection.Helpers; +using MyMediaCollection.Interfaces; +using MyMediaCollection.Model; +using System.Collections.ObjectModel; +using System.Threading.Tasks; + +namespace MyMediaCollection.ViewModels +{ + public partial class MainViewModel : ObservableObject + { + [ObservableProperty] + private string selectedMedium; + [ObservableProperty] + private ObservableCollection items = new ObservableCollection(); + private ObservableCollection allItems; + [ObservableProperty] + private ObservableCollection mediums; + [ObservableProperty] + [NotifyCanExecuteChangedFor(nameof(DeleteCommand))] + private MediaItem selectedMediaItem; + private INavigationService _navigationService; + private IDataService _dataService; + private const string AllMediums = "All"; + + public MainViewModel(INavigationService navigationService, IDataService dataService) + { + _navigationService = navigationService; + _dataService = dataService; + + PopulateDataAsync(); + } + + public async Task PopulateDataAsync() + { + Items.Clear(); + + foreach (var item in await _dataService.GetItemsAsync()) + { + Items.Add(item); + } + + allItems = new ObservableCollection(Items); + + Mediums = new ObservableCollection + { + AllMediums + }; + + foreach (var itemType in _dataService.GetItemTypes()) + { + Mediums.Add(itemType.ToString()); + } + + SelectedMedium = Mediums[0]; + } + + partial void OnSelectedMediumChanged(string value) + { + Items.Clear(); + + foreach (var item in allItems) + { + if (string.IsNullOrWhiteSpace(value) + || value == "All" + || value == item.MediaType.ToString()) + { + Items.Add(item); + } + } + } + + [RelayCommand] + private void AddEdit() + { + var selectedItemId = -1; + + if (SelectedMediaItem != null) + { + selectedItemId = SelectedMediaItem.Id; + } + + _navigationService.NavigateTo("ItemDetailsPage", selectedItemId); + } + + public void ListViewDoubleTapped(object sender, DoubleTappedRoutedEventArgs e) + { + AddEdit(); + } + + [RelayCommand(CanExecute = nameof(CanDeleteItem))] + private async Task DeleteAsync() + { + await _dataService.DeleteItemAsync(SelectedMediaItem); + allItems.Remove(SelectedMediaItem); + Items.Remove(SelectedMediaItem); + } + + private bool CanDeleteItem() => SelectedMediaItem != null; + + [RelayCommand] + private void SendToast() + { + if (ToastWithAvatar.SendToast()) + NotificationShared.ToastSentSuccessfully(); + else + NotificationShared.CouldNotSendToast(); + } + + [RelayCommand] + private void SendToastWithText() + { + if (ToastWithText.SendToast()) + NotificationShared.ToastSentSuccessfully(); + else + NotificationShared.CouldNotSendToast(); + } + } +} \ No newline at end of file diff --git a/Chapter14/MyMediaCollection/Views/ItemDetailsPage.xaml b/Chapter14/MyMediaCollection/Views/ItemDetailsPage.xaml new file mode 100644 index 0000000..1431228 --- /dev/null +++ b/Chapter14/MyMediaCollection/Views/ItemDetailsPage.xaml @@ -0,0 +1,99 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Chapter14/ProjectTracker/MainWindow.xaml.cs b/Chapter14/ProjectTracker/MainWindow.xaml.cs new file mode 100644 index 0000000..4afe923 --- /dev/null +++ b/Chapter14/ProjectTracker/MainWindow.xaml.cs @@ -0,0 +1,36 @@ +using Microsoft.UI.Xaml; +using Microsoft.UI.Xaml.Controls; +using Microsoft.UI.Xaml.Controls.Primitives; +using Microsoft.UI.Xaml.Data; +using Microsoft.UI.Xaml.Input; +using Microsoft.UI.Xaml.Media; +using Microsoft.UI.Xaml.Navigation; +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Runtime.InteropServices.WindowsRuntime; +using Windows.Foundation; +using Windows.Foundation.Collections; + +// To learn more about WinUI, the WinUI project structure, +// and more about our project templates, see: http://aka.ms/winui-project-info. + +namespace ProjectTracker +{ + /// + /// An empty window that can be used on its own or navigated to within a Frame. + /// + public sealed partial class MainWindow : Window + { + public MainWindow() + { + this.InitializeComponent(); + } + + private void myButton_Click(object sender, RoutedEventArgs e) + { + myButton.Content = "Clicked"; + } + } +} diff --git a/Chapter14/ProjectTracker/Package.appxmanifest b/Chapter14/ProjectTracker/Package.appxmanifest new file mode 100644 index 0000000..bbeb09c --- /dev/null +++ b/Chapter14/ProjectTracker/Package.appxmanifest @@ -0,0 +1,51 @@ + + + + + + + + + + ProjectTracker + alash + Assets\StoreLogo.png + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Chapter14/ProjectTracker/ProjectTracker.csproj b/Chapter14/ProjectTracker/ProjectTracker.csproj new file mode 100644 index 0000000..f461359 --- /dev/null +++ b/Chapter14/ProjectTracker/ProjectTracker.csproj @@ -0,0 +1,58 @@ + + + WinExe + net6.0-windows10.0.19041.0 + 10.0.17763.0 + ProjectTracker + app.manifest + x86;x64;ARM64 + win10-x86;win10-x64;win10-arm64 + win10-$(Platform).pubxml + true + true + True + True + EA11267DD5E2484524B1D22D01DEBB813D349174 + SHA256 + True + False + True + Never + C:\Installers\ProjectTracker + 0 + + + + + + + + + + + + + + + + + + + + + + + + + + true + + diff --git a/Chapter14/ProjectTracker/ProjectTracker.sln b/Chapter14/ProjectTracker/ProjectTracker.sln new file mode 100644 index 0000000..156790c --- /dev/null +++ b/Chapter14/ProjectTracker/ProjectTracker.sln @@ -0,0 +1,43 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 17 +VisualStudioVersion = 17.7.34031.279 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ProjectTracker", "ProjectTracker.csproj", "{93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|ARM64 = Debug|ARM64 + Debug|x64 = Debug|x64 + Debug|x86 = Debug|x86 + Release|ARM64 = Release|ARM64 + Release|x64 = Release|x64 + Release|x86 = Release|x86 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Debug|ARM64.ActiveCfg = Debug|ARM64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Debug|ARM64.Build.0 = Debug|ARM64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Debug|ARM64.Deploy.0 = Debug|ARM64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Debug|x64.ActiveCfg = Debug|x64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Debug|x64.Build.0 = Debug|x64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Debug|x64.Deploy.0 = Debug|x64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Debug|x86.ActiveCfg = Debug|x86 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Debug|x86.Build.0 = Debug|x86 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Debug|x86.Deploy.0 = Debug|x86 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Release|ARM64.ActiveCfg = Release|ARM64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Release|ARM64.Build.0 = Release|ARM64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Release|ARM64.Deploy.0 = Release|ARM64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Release|x64.ActiveCfg = Release|x64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Release|x64.Build.0 = Release|x64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Release|x64.Deploy.0 = Release|x64 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Release|x86.ActiveCfg = Release|x86 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Release|x86.Build.0 = Release|x86 + {93BA0760-DFF3-43B8-89BB-B2CCC0F4E5E3}.Release|x86.Deploy.0 = Release|x86 + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {281EFFB3-0143-45CA-AEFF-5597FED3569B} + EndGlobalSection +EndGlobal diff --git a/Chapter14/ProjectTracker/Properties/launchSettings.json b/Chapter14/ProjectTracker/Properties/launchSettings.json new file mode 100644 index 0000000..ad1147d --- /dev/null +++ b/Chapter14/ProjectTracker/Properties/launchSettings.json @@ -0,0 +1,10 @@ +{ + "profiles": { + "ProjectTracker (Package)": { + "commandName": "MsixPackage" + }, + "ProjectTracker (Unpackaged)": { + "commandName": "Project" + } + } +} \ No newline at end of file diff --git a/Chapter14/ProjectTracker/app.manifest b/Chapter14/ProjectTracker/app.manifest new file mode 100644 index 0000000..e8e47a0 --- /dev/null +++ b/Chapter14/ProjectTracker/app.manifest @@ -0,0 +1,20 @@ + + + + + + + + + + + + + + PerMonitorV2 + + + \ No newline at end of file