diff --git a/samples/todoapp/TodoApp.Uno/.editorconfig b/samples/todoapp/TodoApp.Uno/.editorconfig new file mode 100644 index 00000000..b947be64 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/.editorconfig @@ -0,0 +1,167 @@ +; This file is for unifying the coding style for different editors and IDEs. +; More information at http://editorconfig.org + +# This file is the top-most EditorConfig file +root = true + +########################################## +# Common Settings +########################################## + +[*] +indent_style = space +end_of_line = crlf +trim_trailing_whitespace = true +insert_final_newline = true +charset = utf-8 + +########################################## +# File Extension Settings +########################################## + +[*.{yml,yaml}] +indent_size = 2 + +[.vsconfig] +indent_size = 2 +end_of_line = lf + +[*.sln] +indent_style = tab +indent_size = 2 + +[*.{csproj,proj,projitems,shproj}] +indent_size = 2 + +[*.{json,slnf}] +indent_size = 2 +end_of_line = lf + +[*.{props,targets}] +indent_size = 2 + +[*.xaml] +indent_size = 2 +charset = utf-8-bom + +[*.xml] +indent_size = 2 +end_of_line = lf + +[*.plist] +indent_size = 2 +indent_style = tab +end_of_line = lf + +[*.manifest] +indent_size = 2 + +[*.appxmanifest] +indent_size = 2 + +[*.{json,css,webmanifest}] +indent_size = 2 +end_of_line = lf + +[web.config] +indent_size = 2 +end_of_line = lf + +[*.sh] +indent_size = 2 +end_of_line = lf + +[*.cs] +# EOL should be normalized by Git. See https://github.com/dotnet/format/issues/1099 +end_of_line = unset + +# See https://github.com/dotnet/roslyn/issues/20356#issuecomment-310143926 +trim_trailing_whitespace = false + +tab_width = 4 +indent_size = 4 + +# Sort using and Import directives with System.* appearing first +dotnet_sort_system_directives_first = true + +# Avoid "this." and "Me." if not necessary +dotnet_style_qualification_for_field = false:suggestion +dotnet_style_qualification_for_property = false:suggestion +dotnet_style_qualification_for_method = false:suggestion +dotnet_style_qualification_for_event = false:suggestion + +#### Naming styles #### + +# Naming rules + +dotnet_naming_rule.interface_should_be_begins_with_i.severity = suggestion +dotnet_naming_rule.interface_should_be_begins_with_i.symbols = interface +dotnet_naming_rule.interface_should_be_begins_with_i.style = begins_with_i + +dotnet_naming_rule.types_should_be_pascal_case.severity = suggestion +dotnet_naming_rule.types_should_be_pascal_case.symbols = types +dotnet_naming_rule.types_should_be_pascal_case.style = pascal_case + +dotnet_naming_rule.non_field_members_should_be_pascal_case.severity = suggestion +dotnet_naming_rule.non_field_members_should_be_pascal_case.symbols = non_field_members +dotnet_naming_rule.non_field_members_should_be_pascal_case.style = pascal_case + +# Symbol specifications + +dotnet_naming_symbols.interface.applicable_kinds = interface +dotnet_naming_symbols.interface.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected +dotnet_naming_symbols.interface.required_modifiers = + +dotnet_naming_symbols.types.applicable_kinds = class, struct, interface, enum +dotnet_naming_symbols.types.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected +dotnet_naming_symbols.types.required_modifiers = + +dotnet_naming_symbols.non_field_members.applicable_kinds = property, event, method +dotnet_naming_symbols.non_field_members.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected +dotnet_naming_symbols.non_field_members.required_modifiers = + +# Naming styles + +dotnet_naming_style.begins_with_i.required_prefix = I +dotnet_naming_style.begins_with_i.required_suffix = +dotnet_naming_style.begins_with_i.word_separator = +dotnet_naming_style.begins_with_i.capitalization = pascal_case + +dotnet_naming_style.pascal_case.required_prefix = +dotnet_naming_style.pascal_case.required_suffix = +dotnet_naming_style.pascal_case.word_separator = +dotnet_naming_style.pascal_case.capitalization = pascal_case + +dotnet_naming_style.pascal_case.required_prefix = +dotnet_naming_style.pascal_case.required_suffix = +dotnet_naming_style.pascal_case.word_separator = +dotnet_naming_style.pascal_case.capitalization = pascal_case +dotnet_style_operator_placement_when_wrapping = beginning_of_line +dotnet_style_coalesce_expression = true:suggestion +dotnet_style_null_propagation = true:suggestion +dotnet_style_prefer_is_null_check_over_reference_equality_method = true:suggestion +dotnet_style_prefer_auto_properties = true:silent +dotnet_style_object_initializer = true:suggestion +dotnet_style_collection_initializer = true:suggestion +dotnet_style_prefer_simplified_boolean_expressions = true:suggestion +dotnet_style_prefer_conditional_expression_over_assignment = true:silent +dotnet_style_prefer_conditional_expression_over_return = true:silent +dotnet_style_explicit_tuple_names = true:suggestion +dotnet_style_prefer_inferred_tuple_names = true:suggestion + +csharp_indent_labels = one_less_than_current +csharp_using_directive_placement = outside_namespace:silent +csharp_prefer_simple_using_statement = true:suggestion +csharp_prefer_braces = true:silent +csharp_style_namespace_declarations = file_scoped:warning +csharp_style_prefer_method_group_conversion = true:silent +csharp_style_prefer_top_level_statements = true:silent +csharp_style_prefer_primary_constructors = true:suggestion +csharp_style_expression_bodied_methods = false:silent +csharp_style_expression_bodied_constructors = false:silent +csharp_style_expression_bodied_operators = false:silent +csharp_style_expression_bodied_properties = true:silent +csharp_style_expression_bodied_indexers = true:silent +csharp_style_expression_bodied_accessors = true:silent +csharp_style_expression_bodied_lambdas = true:silent +csharp_style_expression_bodied_local_functions = false:silent diff --git a/samples/todoapp/TodoApp.Uno/.gitignore b/samples/todoapp/TodoApp.Uno/.gitignore new file mode 100644 index 00000000..ef88c205 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/.gitignore @@ -0,0 +1,403 @@ +## Ignore Visual Studio temporary files, build results, and +## files generated by popular Visual Studio add-ons. +## +## Get latest from https://github.com/github/gitignore/blob/main/VisualStudio.gitignore + +# User-specific files +*.rsuser +*.suo +*.user +*.userosscache +*.sln.docstates + +# User-specific files (MonoDevelop/Xamarin Studio) +*.userprefs + +# Mono auto generated files +mono_crash.* + +# Build results +[Dd]ebug/ +[Dd]ebugPublic/ +[Rr]elease/ +[Rr]eleases/ +x64/ +x86/ +[Ww][Ii][Nn]32/ +[Aa][Rr][Mm]/ +[Aa][Rr][Mm]64/ +bld/ +[Bb]in/ +[Oo]bj/ +[Ll]og/ +[Ll]ogs/ + +# Visual Studio 2015/2017 cache/options directory +.vs/ +# Uncomment if you have tasks that create the project's static files in wwwroot +#wwwroot/ + +# Visual Studio 2017 auto generated files +Generated\ Files/ + +# MSTest test Results +[Tt]est[Rr]esult*/ +[Bb]uild[Ll]og.* + +# NUnit +*.VisualState.xml +TestResult.xml +nunit-*.xml + +# Build Results of an ATL Project +[Dd]ebugPS/ +[Rr]eleasePS/ +dlldata.c + +# Benchmark Results +BenchmarkDotNet.Artifacts/ + +# .NET Core +project.lock.json +project.fragment.lock.json +artifacts/ + +# ASP.NET Scaffolding +ScaffoldingReadMe.txt + +# StyleCop +StyleCopReport.xml + +# Files built by Visual Studio +*_i.c +*_p.c +*_h.h +*.ilk +*.meta +*.obj +*.iobj +*.pch +*.pdb +*.ipdb +*.pgc +*.pgd +*.rsp +*.sbr +*.tlb +*.tli +*.tlh +*.tmp +*.tmp_proj +*_wpftmp.csproj +*.log +*.tlog +*.vspscc +*.vssscc +.builds +*.pidb +*.svclog +*.scc + +# Chutzpah Test files +_Chutzpah* + +# Visual C++ cache files +ipch/ +*.aps +*.ncb +*.opendb +*.opensdf +*.sdf +*.cachefile +*.VC.db +*.VC.VC.opendb + +# Visual Studio profiler +*.psess +*.vsp +*.vspx +*.sap + +# Visual Studio Trace Files +*.e2e + +# TFS 2012 Local Workspace +$tf/ + +# Guidance Automation Toolkit +*.gpState + +# ReSharper is a .NET coding add-in +_ReSharper*/ +*.[Rr]e[Ss]harper +*.DotSettings.user + +# TeamCity is a build add-in +_TeamCity* + +# DotCover is a Code Coverage Tool +*.dotCover + +# AxoCover is a Code Coverage Tool +.axoCover/* +!.axoCover/settings.json + +# Coverlet is a free, cross platform Code Coverage Tool +coverage*.json +coverage*.xml +coverage*.info + +# Visual Studio code coverage results +*.coverage +*.coveragexml + +# NCrunch +_NCrunch_* +.*crunch*.local.xml +nCrunchTemp_* + +# MightyMoose +*.mm.* +AutoTest.Net/ + +# Web workbench (sass) +.sass-cache/ + +# Installshield output folder +[Ee]xpress/ + +# DocProject is a documentation generator add-in +DocProject/buildhelp/ +DocProject/Help/*.HxT +DocProject/Help/*.HxC +DocProject/Help/*.hhc +DocProject/Help/*.hhk +DocProject/Help/*.hhp +DocProject/Help/Html2 +DocProject/Help/html + +# Click-Once directory +publish/ + +# Publish Web Output +*.[Pp]ublish.xml +*.azurePubxml +# Note: Comment the next line if you want to checkin your web deploy settings, +# but database connection strings (with potential passwords) will be unencrypted +*.pubxml +*.publishproj + +# Microsoft Azure Web App publish settings. Comment the next line if you want to +# checkin your Azure Web App publish settings, but sensitive information contained +# in these scripts will be unencrypted +PublishScripts/ + +# NuGet Packages +*.nupkg +# NuGet Symbol Packages +*.snupkg +# The packages folder can be ignored because of Package Restore +**/[Pp]ackages/* +# except build/, which is used as an MSBuild target. +!**/[Pp]ackages/build/ +# Uncomment if necessary however generally it will be regenerated when needed +#!**/[Pp]ackages/repositories.config +# NuGet v3's project.json files produces more ignorable files +*.nuget.props +*.nuget.targets + +# Microsoft Azure Build Output +csx/ +*.build.csdef + +# Microsoft Azure Emulator +ecf/ +rcf/ + +# Windows Store app package directories and files +AppPackages/ +BundleArtifacts/ +Package.StoreAssociation.xml +_pkginfo.txt +*.appx +*.appxbundle +*.appxupload + +# Visual Studio cache files +# files ending in .cache can be ignored +*.[Cc]ache +# but keep track of directories ending in .cache +!?*.[Cc]ache/ + +# Others +ClientBin/ +~$* +*~ +*.dbmdl +*.dbproj.schemaview +*.jfm +*.pfx +*.publishsettings +orleans.codegen.cs + +# Including strong name files can present a security risk +# (https://github.com/github/gitignore/pull/2483#issue-259490424) +#*.snk + +# Since there are multiple workflows, uncomment next line to ignore bower_components +# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622) +#bower_components/ + +# RIA/Silverlight projects +Generated_Code/ + +# Backup & report files from converting an old project file +# to a newer Visual Studio version. Backup files are not needed, +# because we have git ;-) +_UpgradeReport_Files/ +Backup*/ +UpgradeLog*.XML +UpgradeLog*.htm +ServiceFabricBackup/ +*.rptproj.bak + +# SQL Server files +*.mdf +*.ldf +*.ndf + +# Business Intelligence projects +*.rdl.data +*.bim.layout +*.bim_*.settings +*.rptproj.rsuser +*- [Bb]ackup.rdl +*- [Bb]ackup ([0-9]).rdl +*- [Bb]ackup ([0-9][0-9]).rdl + +# Microsoft Fakes +FakesAssemblies/ + +# GhostDoc plugin setting file +*.GhostDoc.xml + +# Node.js Tools for Visual Studio +.ntvs_analysis.dat +node_modules/ + +# Visual Studio 6 build log +*.plg + +# Visual Studio 6 workspace options file +*.opt + +# Visual Studio 6 auto-generated workspace file (contains which files were open etc.) +*.vbw + +# Visual Studio 6 auto-generated project file (contains which files were open etc.) +*.vbp + +# Visual Studio 6 workspace and project file (working project files containing files to include in project) +*.dsw +*.dsp + +# Visual Studio 6 technical files +*.ncb +*.aps + +# Visual Studio LightSwitch build output +**/*.HTMLClient/GeneratedArtifacts +**/*.DesktopClient/GeneratedArtifacts +**/*.DesktopClient/ModelManifest.xml +**/*.Server/GeneratedArtifacts +**/*.Server/ModelManifest.xml +_Pvt_Extensions + +# Paket dependency manager +.paket/paket.exe +paket-files/ + +# FAKE - F# Make +.fake/ + +# CodeRush personal settings +.cr/personal + +# Python Tools for Visual Studio (PTVS) +__pycache__/ +*.pyc + +# Cake - Uncomment if you are using it +# tools/** +# !tools/packages.config + +# Tabs Studio +*.tss + +# Telerik's JustMock configuration file +*.jmconfig + +# BizTalk build output +*.btp.cs +*.btm.cs +*.odx.cs +*.xsd.cs + +# OpenCover UI analysis results +OpenCover/ + +# Azure Stream Analytics local run output +ASALocalRun/ + +# MSBuild Binary and Structured Log +*.binlog + +# NVidia Nsight GPU debugger configuration file +*.nvuser + +# MFractors (Xamarin productivity tool) working folder +.mfractor/ + +# Local History for Visual Studio +.localhistory/ + +# Visual Studio History (VSHistory) files +.vshistory/ + +# BeatPulse healthcheck temp database +healthchecksdb + +# Backup folder for Package Reference Convert tool in Visual Studio 2017 +MigrationBackup/ + +# Ionide (cross platform F# VS Code tools) working folder +.ionide/ + +# Fody - auto-generated XML schema +FodyWeavers.xsd + +# VS Code files for those working on multiple tools +.vscode/* +!.vscode/settings.json +!.vscode/tasks.json +!.vscode/launch.json +!.vscode/extensions.json +*.code-workspace + +# Local History for Visual Studio Code +.history/ + +# Windows Installer files from build outputs +*.cab +*.msi +*.msix +*.msm +*.msp + +# JetBrains Rider +*.sln.iml + +# Single Target Config +solution-config.props +# Publish Profiles +!**/Properties/PublishProfiles/*.pubxml \ No newline at end of file diff --git a/samples/todoapp/TodoApp.Uno/.run/Readme.md b/samples/todoapp/TodoApp.Uno/.run/Readme.md new file mode 100644 index 00000000..6e72638a --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/.run/Readme.md @@ -0,0 +1,3 @@ +# About the `.run` folder + +This folder is present to add support for the [Rider IDE](https://aka.platform.uno/rider-getstarted). You can remove this folder safely if you're not using Rider. diff --git a/samples/todoapp/TodoApp.Uno/.run/TodoApp.Uno.run.xml b/samples/todoapp/TodoApp.Uno/.run/TodoApp.Uno.run.xml new file mode 100644 index 00000000..6f98af60 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/.run/TodoApp.Uno.run.xml @@ -0,0 +1,62 @@ + + + + + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/.vscode/extensions.json b/samples/todoapp/TodoApp.Uno/.vscode/extensions.json new file mode 100644 index 00000000..a63ad400 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/.vscode/extensions.json @@ -0,0 +1,5 @@ +{ + "recommendations": [ + "unoplatform.vscode" + ], +} diff --git a/samples/todoapp/TodoApp.Uno/.vscode/launch.json b/samples/todoapp/TodoApp.Uno/.vscode/launch.json new file mode 100644 index 00000000..339fe686 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/.vscode/launch.json @@ -0,0 +1,56 @@ +{ + // Use IntelliSense to find out which attributes exist for C# debugging + // Use hover for the description of the existing attributes + // For further information visit https://github.com/OmniSharp/omnisharp-vscode/blob/master/debugger-launchjson.md + "version": "0.2.0", + "configurations": [ + { + "name": "Uno Platform Mobile Debug", + "type": "Uno", + "request": "launch", + // any Uno* task will do, this is simply to satisfy vscode requirement when a launch.json is present + "preLaunchTask": "Uno: android | Debug | android-x64" + }, + { + // Use IntelliSense to find out which attributes exist for C# debugging + // Use hover for the description of the existing attributes + // For further information visit https://github.com/OmniSharp/omnisharp-vscode/blob/master/debugger-launchjson.md + "name": "Uno Platform WebAssembly Debug (Chrome)", + "type": "chrome", + "request": "launch", + "url": "http://localhost:5000", + "webRoot": "${workspaceFolder}/TodoApp.Uno", + "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", + "timeout": 30000, + "preLaunchTask": "build-wasm", + "server": { + "runtimeExecutable": "dotnet", + "program": "run", + "args": ["--no-build","-f","net8.0-browserwasm","--launch-profile", "TodoApp.Uno (WebAssembly)"], + "outputCapture": "std", + "timeout": 30000, + "cwd": "${workspaceFolder}/TodoApp.Uno" + } + }, + { + // Use IntelliSense to find out which attributes exist for C# debugging + // Use hover for the description of the existing attributes + // For further information visit https://github.com/OmniSharp/omnisharp-vscode/blob/master/debugger-launchjson.md + "name": "Uno Platform Desktop Debug", + "type": "coreclr", + "request": "launch", + "preLaunchTask": "build-desktop", + // If you have changed target frameworks, make sure to update the program path. + "program": "${workspaceFolder}/TodoApp.Uno/bin/Debug/net8.0-desktop/TodoApp.Uno.dll", + "args": [], + "launchSettingsProfile": "TodoApp.Uno (Desktop)", + "env": { + "DOTNET_MODIFIABLE_ASSEMBLIES": "debug" + }, + "cwd": "${workspaceFolder}/TodoApp.Uno", + // For more information about the 'console' field, see https://aka.ms/VSCode-CS-LaunchJson-Console + "console": "internalConsole", + "stopAtEntry": false + }, + ] +} diff --git a/samples/todoapp/TodoApp.Uno/.vscode/settings.json b/samples/todoapp/TodoApp.Uno/.vscode/settings.json new file mode 100644 index 00000000..3405922d --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/.vscode/settings.json @@ -0,0 +1,10 @@ +{ + "explorer.fileNesting.enabled": true, + "explorer.fileNesting.expand": false, + "explorer.fileNesting.patterns": { + "*.xaml": "$(capture).xaml.cs" + }, + "files.associations": { + "global.json": "jsonc" + } +} diff --git a/samples/todoapp/TodoApp.Uno/.vscode/tasks.json b/samples/todoapp/TodoApp.Uno/.vscode/tasks.json new file mode 100644 index 00000000..6d08f7ee --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/.vscode/tasks.json @@ -0,0 +1,57 @@ +{ + "version": "2.0.0", + "tasks": [ + { + "label": "build-wasm", + "command": "dotnet", + "type": "process", + "args": [ + "build", + "${workspaceFolder}/TodoApp.Uno/TodoApp.Uno.csproj", + "/property:GenerateFullPaths=true", + "/property:TargetFramework=net8.0-browserwasm", + "/consoleloggerparameters:NoSummary" + ], + "problemMatcher": "$msCompile" + }, + { + "label": "publish-wasm", + "command": "dotnet", + "type": "process", + "args": [ + "publish", + "${workspaceFolder}/TodoApp.Uno/TodoApp.Uno.csproj", + "/property:GenerateFullPaths=true", + "/property:TargetFramework=net8.0-browserwasm", + "/consoleloggerparameters:NoSummary" + ], + "problemMatcher": "$msCompile" + }, + { + "label": "build-desktop", + "command": "dotnet", + "type": "process", + "args": [ + "build", + "${workspaceFolder}/TodoApp.Uno/TodoApp.Uno.csproj", + "/property:GenerateFullPaths=true", + "/property:TargetFramework=net8.0-desktop", + "/consoleloggerparameters:NoSummary" + ], + "problemMatcher": "$msCompile" + }, + { + "label": "publish-desktop", + "command": "dotnet", + "type": "process", + "args": [ + "publish", + "${workspaceFolder}/TodoApp.Uno/TodoApp.Uno.csproj", + "/property:GenerateFullPaths=true", + "/property:TargetFramework=net8.0-desktop", + "/consoleloggerparameters:NoSummary" + ], + "problemMatcher": "$msCompile" + } + ] +} diff --git a/samples/todoapp/TodoApp.Uno/.vsconfig b/samples/todoapp/TodoApp.Uno/.vsconfig new file mode 100644 index 00000000..0439670d --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/.vsconfig @@ -0,0 +1,34 @@ +{ + "version": "1.0", + "components": [ + "Microsoft.VisualStudio.Component.CoreEditor", + "Microsoft.VisualStudio.Workload.CoreEditor", + "Microsoft.NetCore.Component.SDK", + "Microsoft.NetCore.Component.DevelopmentTools", + "Microsoft.Net.ComponentGroup.DevelopmentPrerequisites", + "Microsoft.VisualStudio.Component.TextTemplating", + "Microsoft.VisualStudio.ComponentGroup.WebToolsExtensions", + "Microsoft.NetCore.Component.Web", + "Microsoft.VisualStudio.Component.IISExpress", + "Component.Microsoft.Web.LibraryManager", + "Microsoft.VisualStudio.ComponentGroup.Web", + "Microsoft.VisualStudio.Component.Web", + "Microsoft.VisualStudio.ComponentGroup.Web.Client", + "Microsoft.VisualStudio.Workload.NetWeb", + "Microsoft.VisualStudio.ComponentGroup.WebToolsExtensions.TemplateEngine", + "Microsoft.VisualStudio.ComponentGroup.MSIX.Packaging", + "Microsoft.VisualStudio.Component.ManagedDesktop.Prerequisites", + "Microsoft.VisualStudio.Component.Debugger.JustInTime", + "Microsoft.VisualStudio.Workload.ManagedDesktop", + "Component.Xamarin.RemotedSimulator", + "Microsoft.VisualStudio.Component.MonoDebugger", + "Microsoft.VisualStudio.ComponentGroup.Maui.All", + "Component.Android.SDK34", + "Component.OpenJDK", + "Microsoft.VisualStudio.Workload.NetCrossPlat", + "Microsoft.VisualStudio.Workload.NetCoreTools" + ], + "extensions": [ + "https://marketplace.visualstudio.com/items?itemName=unoplatform.uno-platform-addin-2022" + ] +} diff --git a/samples/todoapp/TodoApp.Uno/Directory.Build.props b/samples/todoapp/TodoApp.Uno/Directory.Build.props new file mode 100644 index 00000000..3e51b4ea --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/Directory.Build.props @@ -0,0 +1,15 @@ + + + enable + enable + true + true + + $(NoWarn);NU1507;NETSDK1201;PRI257 + + diff --git a/samples/todoapp/TodoApp.Uno/Directory.Build.targets b/samples/todoapp/TodoApp.Uno/Directory.Build.targets new file mode 100644 index 00000000..f75adf7e --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/Directory.Build.targets @@ -0,0 +1,2 @@ + + diff --git a/samples/todoapp/TodoApp.Uno/Directory.Packages.props b/samples/todoapp/TodoApp.Uno/Directory.Packages.props new file mode 100644 index 00000000..f0ec5d86 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/Directory.Packages.props @@ -0,0 +1,19 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno.sln b/samples/todoapp/TodoApp.Uno/TodoApp.Uno.sln new file mode 100644 index 00000000..cc4fa640 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno.sln @@ -0,0 +1,47 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 17 +VisualStudioVersion = 17.10.35201.131 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "TodoApp.Uno", "TodoApp.Uno\TodoApp.Uno.csproj", "{5FD4876C-302E-42DF-B28B-55035009E545}" +EndProject +Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{F20ED0E3-BD89-4202-A1C8-E809AA315AFB}" + ProjectSection(SolutionItems) = preProject + .gitignore = .gitignore + Directory.Build.props = Directory.Build.props + Directory.Build.targets = Directory.Build.targets + Directory.Packages.props = Directory.Packages.props + global.json = global.json + EndProjectSection +EndProject +Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "library", "library", "{0AAB65A5-8A90-47AF-AAED-AF27E82D50A4}" +EndProject +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CommunityToolkit.Datasync.Client", "..\..\..\src\CommunityToolkit.Datasync.Client\CommunityToolkit.Datasync.Client.csproj", "{2D6F113F-D3A1-43A5-BF57-E72E5D5F5963}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Release|Any CPU = Release|Any CPU + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {5FD4876C-302E-42DF-B28B-55035009E545}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {5FD4876C-302E-42DF-B28B-55035009E545}.Debug|Any CPU.Build.0 = Debug|Any CPU + {5FD4876C-302E-42DF-B28B-55035009E545}.Debug|Any CPU.Deploy.0 = Debug|Any CPU + {5FD4876C-302E-42DF-B28B-55035009E545}.Release|Any CPU.ActiveCfg = Release|Any CPU + {5FD4876C-302E-42DF-B28B-55035009E545}.Release|Any CPU.Build.0 = Release|Any CPU + {5FD4876C-302E-42DF-B28B-55035009E545}.Release|Any CPU.Deploy.0 = Release|Any CPU + {2D6F113F-D3A1-43A5-BF57-E72E5D5F5963}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {2D6F113F-D3A1-43A5-BF57-E72E5D5F5963}.Debug|Any CPU.Build.0 = Debug|Any CPU + {2D6F113F-D3A1-43A5-BF57-E72E5D5F5963}.Release|Any CPU.ActiveCfg = Release|Any CPU + {2D6F113F-D3A1-43A5-BF57-E72E5D5F5963}.Release|Any CPU.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(NestedProjects) = preSolution + {2D6F113F-D3A1-43A5-BF57-E72E5D5F5963} = {0AAB65A5-8A90-47AF-AAED-AF27E82D50A4} + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {EA315B6A-4F6C-4482-A88B-61900C1B6934} + EndGlobalSection +EndGlobal diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/App.xaml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/App.xaml new file mode 100644 index 00000000..fe2094c7 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/App.xaml @@ -0,0 +1,28 @@ + + + + + + + + + + + + + + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/App.xaml.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/App.xaml.cs new file mode 100644 index 00000000..9a55a5af --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/App.xaml.cs @@ -0,0 +1,134 @@ +using System; +using Microsoft.Data.Sqlite; +using Microsoft.EntityFrameworkCore; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.UI.Xaml; +using Microsoft.UI.Xaml.Controls; +using TodoApp.Uno.Database; +using TodoApp.Uno.ViewModels; +using TodoApp.Uno.Views; +using Uno.Resizetizer; + +namespace TodoApp.Uno; +public partial class App : Application +{ + /// + /// 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(); + + + } + + protected Window? MainWindow { get; private set; } + public IHost? Host { get; private set; } + + private SqliteConnection dbConnection; + + protected async override void OnLaunched(LaunchActivatedEventArgs args) + { + AppContext.SetSwitch("System.Reflection.NullabilityInfoContext.IsSupported", true); + + this.dbConnection = new SqliteConnection("Data Source=:memory:"); + this.dbConnection.Open(); + + var builder = this.CreateBuilder(args) + // Add navigation support for toolkit controls such as TabBar and NavigationView + .UseToolkitNavigation() + .Configure(host => host +#if DEBUG + // Switch to Development environment when running in DEBUG + .UseEnvironment(Environments.Development) +#endif + .UseLogging(configure: (context, logBuilder) => + { + // Configure log levels for different categories of logging + logBuilder + .SetMinimumLevel( + context.HostingEnvironment.IsDevelopment() ? + LogLevel.Information : + LogLevel.Warning) + + // Default filters for core Uno Platform namespaces + .CoreLogLevel(LogLevel.Warning); + + // Uno Platform namespace filter groups + // Uncomment individual methods to see more detailed logging + //// Generic Xaml events + //logBuilder.XamlLogLevel(LogLevel.Debug); + //// Layout specific messages + //logBuilder.XamlLayoutLogLevel(LogLevel.Debug); + //// Storage messages + //logBuilder.StorageLogLevel(LogLevel.Debug); + //// Binding related messages + //logBuilder.XamlBindingLogLevel(LogLevel.Debug); + //// Binder memory references tracking + //logBuilder.BinderMemoryReferenceLogLevel(LogLevel.Debug); + //// DevServer and HotReload related + //logBuilder.HotReloadCoreLogLevel(LogLevel.Information); + //// Debug JS interop + //logBuilder.WebAssemblyLogLevel(LogLevel.Debug); + + }, enableUnoLogging: true) + .UseConfiguration(configure: configBuilder => + configBuilder + .EmbeddedSource() + .Section() + ) + + .ConfigureServices((context, services) => + { + // TODO: Register your services + //services.AddSingleton(); + services.AddDbContext(options => options.UseSqlite(this.dbConnection)); + services.AddScoped(); + services.AddTransient(); + + }) + .UseNavigation(RegisterRoutes) + ); + MainWindow = builder.Window; + +#if DEBUG + MainWindow.EnableHotReload(); +#endif + MainWindow.SetWindowIcon(); + + // We have to build here so that the services are initialized and we can call InitializeDatabase + // before navigating. This enables us to load the data when the page is loaded + Host = builder.Build(); + + InitializeDatabase(); + + // We still make this navigation call so we use the Uno Navigation extension, which automatically wires + // up view models with their page using the ViewMap and RouteMap below. + Host = await builder.NavigateAsync(); + + } + + private void InitializeDatabase() + { + IDbInitializer? initializer = Host?.Services?.GetRequiredService(); + initializer?.Initialize(); + } + + private static void RegisterRoutes(IViewRegistry views, IRouteRegistry routes) + { + views.Register( + new ViewMap(ViewModel: typeof(ShellViewModel)), + new ViewMap() + ); + + routes.Register( + new RouteMap("", View: views.FindByViewModel(), + Nested: + [ + new ("TodoList", View: views.FindByViewModel(), IsDefault: true) + ] + ) + ); + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/AddItem.png b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/AddItem.png new file mode 100644 index 00000000..33664793 Binary files /dev/null and b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/AddItem.png differ diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Icons/icon.svg b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Icons/icon.svg new file mode 100644 index 00000000..a15af53a --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Icons/icon.svg @@ -0,0 +1,42 @@ + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Icons/icon_foreground.svg b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Icons/icon_foreground.svg new file mode 100644 index 00000000..8ffc41ae --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Icons/icon_foreground.svg @@ -0,0 +1,137 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Images/back.svg b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Images/back.svg new file mode 100644 index 00000000..bcd7851b --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Images/back.svg @@ -0,0 +1,3 @@ + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/LockScreenLogo.scale-200.png b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/LockScreenLogo.scale-200.png new file mode 100644 index 00000000..7440f0d4 Binary files /dev/null and b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/LockScreenLogo.scale-200.png differ diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/RefreshItems.png b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/RefreshItems.png new file mode 100644 index 00000000..51748edd Binary files /dev/null and b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/RefreshItems.png differ diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/SharedAssets.md b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/SharedAssets.md new file mode 100644 index 00000000..1b84a74a --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/SharedAssets.md @@ -0,0 +1,32 @@ +# Shared Assets + +See documentation about assets here: https://github.com/unoplatform/uno/blob/master/doc/articles/features/working-with-assets.md + +## Here is a cheat sheet + +1. Add the image file to the `Assets` directory of a shared project. +2. Set the build action to `Content`. +3. (Recommended) Provide an asset for various scales/dpi + +### Examples + +```text +\Assets\Images\logo.scale-100.png +\Assets\Images\logo.scale-200.png +\Assets\Images\logo.scale-400.png + +\Assets\Images\scale-100\logo.png +\Assets\Images\scale-200\logo.png +\Assets\Images\scale-400\logo.png +``` + +### Table of scales + +| Scale | WinUI | iOS/MacCatalyst | Android | +|-------|:-----------:|:---------------:|:-------:| +| `100` | scale-100 | @1x | mdpi | +| `125` | scale-125 | N/A | N/A | +| `150` | scale-150 | N/A | hdpi | +| `200` | scale-200 | @2x | xhdpi | +| `300` | scale-300 | @3x | xxhdpi | +| `400` | scale-400 | N/A | xxxhdpi | diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Splash/splash_screen.svg b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Splash/splash_screen.svg new file mode 100644 index 00000000..8ffc41ae --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Splash/splash_screen.svg @@ -0,0 +1,137 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/SplashScreen.scale-200.png b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/SplashScreen.scale-200.png new file mode 100644 index 00000000..32f486a8 Binary files /dev/null and b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/SplashScreen.scale-200.png differ diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Square150x150Logo.scale-200.png b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Square150x150Logo.scale-200.png new file mode 100644 index 00000000..53ee3777 Binary files /dev/null and b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Square150x150Logo.scale-200.png differ diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Square44x44Logo.scale-200.png b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Square44x44Logo.scale-200.png new file mode 100644 index 00000000..f713bba6 Binary files /dev/null and b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Square44x44Logo.scale-200.png differ diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Square44x44Logo.targetsize-24_altform-unplated.png b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Square44x44Logo.targetsize-24_altform-unplated.png new file mode 100644 index 00000000..dc9f5bea Binary files /dev/null and b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Square44x44Logo.targetsize-24_altform-unplated.png differ diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/StoreLogo.png b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/StoreLogo.png new file mode 100644 index 00000000..a4586f26 Binary files /dev/null and b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/StoreLogo.png differ diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Wide310x150Logo.scale-200.png b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Wide310x150Logo.scale-200.png new file mode 100644 index 00000000..8b4a5d0d Binary files /dev/null and b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Assets/Wide310x150Logo.scale-200.png differ diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/AppDbContext.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/AppDbContext.cs new file mode 100644 index 00000000..b29abca4 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/AppDbContext.cs @@ -0,0 +1,65 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. +// See the LICENSE file in the project root for more information. + +using System; +using System.Linq; +using System.Threading; +using System.Threading.Tasks; +using CommunityToolkit.Datasync.Client.Http; +using CommunityToolkit.Datasync.Client.Offline; +using Microsoft.EntityFrameworkCore; +using TodoApp.Uno.Services; + +namespace TodoApp.Uno.Database; + +public class AppDbContext(DbContextOptions options) : DbContext(options) +//public class AppDbContext(DbContextOptions options) : OfflineDbContext(options) +{ + public DbSet TodoItems => Set(); + + //protected override void OnDatasyncInitialization(DatasyncOfflineOptionsBuilder optionsBuilder) + //{ + // HttpClientOptions clientOptions = new() + // { + // Endpoint = new Uri("https://yoursite.azurewebsites.net/"), + // HttpPipeline = [new LoggingHandler()] + // }; + // _ = optionsBuilder.UseHttpClientOptions(clientOptions); + //} + + public async Task SynchronizeAsync(CancellationToken cancellationToken = default) + { + //PushResult pushResult = await this.PushAsync(cancellationToken); + //if (!pushResult.IsSuccessful) + //{ + // throw new ApplicationException($"Push failed: {pushResult.FailedRequests.FirstOrDefault().Value.ReasonPhrase}"); + //} + + //PullResult pullResult = await this.PullAsync(cancellationToken); + //if (!pullResult.IsSuccessful) + //{ + // throw new ApplicationException($"Pull failed: {pullResult.FailedRequests.FirstOrDefault().Value.ReasonPhrase}"); + //} + } +} + +/// +/// Use this class to initialize the database. In this sample, we just create +/// the database using . However, you +/// may want to use migrations. +/// +/// The context for the database. +public class DbContextInitializer(AppDbContext context) : IDbInitializer +{ + /// + public void Initialize() + { + _ = context.Database.EnsureCreated(); + Task.Run(async () => await context.SynchronizeAsync()); + } + + /// + public Task InitializeAsync(CancellationToken cancellationToken = default) + => context.Database.EnsureCreatedAsync(cancellationToken); +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/IDbInitializer.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/IDbInitializer.cs new file mode 100644 index 00000000..2f845444 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/IDbInitializer.cs @@ -0,0 +1,26 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. +// See the LICENSE file in the project root for more information. + +using System.Threading; +using System.Threading.Tasks; + +namespace TodoApp.Uno.Database; + +/// +/// An interface to initialize a database. +/// +public interface IDbInitializer +{ + /// + /// Synchronously initialize the database. + /// + void Initialize(); + + /// + /// Asynchronously initialize the database. + /// + /// A to observe. + /// A task that resolves when complete. + Task InitializeAsync(CancellationToken cancellationToken = default); +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/OfflineClientEntity.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/OfflineClientEntity.cs new file mode 100644 index 00000000..cf244bf9 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/OfflineClientEntity.cs @@ -0,0 +1,20 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. +// See the LICENSE file in the project root for more information. + +using System; +using System.ComponentModel.DataAnnotations; + +namespace TodoApp.Uno.Database; + +/// +/// An abstract class for working with offline entities. +/// +public abstract class OfflineClientEntity +{ + [System.ComponentModel.DataAnnotations.Key] + public string Id { get; set; } + public DateTimeOffset? UpdatedAt { get; set; } + public string? Version { get; set; } + public bool Deleted { get; set; } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/TodoItem.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/TodoItem.cs new file mode 100644 index 00000000..bc0ecba1 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Database/TodoItem.cs @@ -0,0 +1,16 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. +// See the LICENSE file in the project root for more information. + +using System.Text.Json; + +namespace TodoApp.Uno.Database; + +public class TodoItem : OfflineClientEntity +{ + public string Title { get; set; } = string.Empty; + public bool IsComplete { get; set; } = false; + + public override string ToString() + => JsonSerializer.Serialize(this); +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/GlobalUsings.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/GlobalUsings.cs new file mode 100644 index 00000000..b3669d5f --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/GlobalUsings.cs @@ -0,0 +1,11 @@ +global using System.Collections.Immutable; +global using CommunityToolkit.Mvvm.ComponentModel; +global using CommunityToolkit.Mvvm.Input; +global using Microsoft.Extensions.DependencyInjection; +global using Microsoft.Extensions.Hosting; +global using Microsoft.Extensions.Localization; +global using Microsoft.Extensions.Logging; +global using Microsoft.Extensions.Options; +global using TodoApp.Uno.Models; +global using TodoApp.Uno.Presentation; +global using ApplicationExecutionState = Windows.ApplicationModel.Activation.ApplicationExecutionState; diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Models/AppConfig.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Models/AppConfig.cs new file mode 100644 index 00000000..74aab031 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Models/AppConfig.cs @@ -0,0 +1,6 @@ +namespace TodoApp.Uno.Models; + +public record AppConfig +{ + public string? Environment { get; init; } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Package.appxmanifest b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Package.appxmanifest new file mode 100644 index 00000000..9ef38146 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Package.appxmanifest @@ -0,0 +1,31 @@ + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/AndroidManifest.xml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/AndroidManifest.xml new file mode 100644 index 00000000..95ae0753 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/AndroidManifest.xml @@ -0,0 +1,4 @@ + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Assets/AboutAssets.txt b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Assets/AboutAssets.txt new file mode 100644 index 00000000..89ab409d --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Assets/AboutAssets.txt @@ -0,0 +1,22 @@ +To add cross-platform image assets for your Uno Platform app, use the Assets folder +in the shared project instead. Assets in this folder are Android-only assets. + +Any raw assets you want to be deployed with your application can be placed in +this directory (and child directories) and given a Build Action of "AndroidAsset". + +These files will be deployed with your package and will be accessible using Android's +AssetManager, like this: + +public class ReadAsset : Activity +{ + protected override void OnCreate (Bundle bundle) + { + base.OnCreate (bundle); + + InputStream input = Assets.Open ("my_asset.txt"); + } +} + +Additionally, some Android functions will automatically load asset files: + +Typeface tf = Typeface.CreateFromAsset (Context.Assets, "fonts/samplefont.ttf"); diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Main.Android.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Main.Android.cs new file mode 100644 index 00000000..8fc5119a --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Main.Android.cs @@ -0,0 +1,43 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using Android.App; +using Android.Content; +using Android.OS; +using Android.Runtime; +using Android.Views; +using Android.Widget; +using Com.Nostra13.Universalimageloader.Core; +using Microsoft.UI.Xaml.Media; + +namespace TodoApp.Uno.Droid; +[global::Android.App.ApplicationAttribute( + Label = "@string/ApplicationName", + Icon = "@mipmap/icon", + LargeHeap = true, + HardwareAccelerated = true, + Theme = "@style/AppTheme" +)] +public class Application : Microsoft.UI.Xaml.NativeApplication +{ + + public Application(IntPtr javaReference, JniHandleOwnership transfer) + : base(() => new App(), javaReference, transfer) + { + ConfigureUniversalImageLoader(); + } + + private static void ConfigureUniversalImageLoader() + { + // Create global configuration and initialize ImageLoader with this config + ImageLoaderConfiguration config = new ImageLoaderConfiguration + .Builder(Context) + .Build(); + + ImageLoader.Instance.Init(config); + + ImageSource.DefaultImageLoader = ImageLoader.Instance.LoadImageAsync; + } +} + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/MainActivity.Android.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/MainActivity.Android.cs new file mode 100644 index 00000000..dbea655b --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/MainActivity.Android.cs @@ -0,0 +1,15 @@ +using Android.App; +using Android.Content.PM; +using Android.OS; +using Android.Views; +using Android.Widget; + +namespace TodoApp.Uno.Droid; +[Activity( + MainLauncher = true, + ConfigurationChanges = global::Uno.UI.ActivityHelper.AllConfigChanges, + WindowSoftInputMode = SoftInput.AdjustNothing | SoftInput.StateHidden +)] +public class MainActivity : Microsoft.UI.Xaml.ApplicationActivity +{ +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Resources/AboutResources.txt b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Resources/AboutResources.txt new file mode 100644 index 00000000..17e3b133 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Resources/AboutResources.txt @@ -0,0 +1,47 @@ +To add cross-platform image assets for your Uno Platform app, use the Assets folder +in the shared project instead. Resources in this folder are Android-only. + +Images, layout descriptions, binary blobs and string dictionaries can be included +in your application as resource files. Various Android APIs are designed to +operate on the resource IDs instead of dealing with images, strings or binary blobs +directly. + +For example, a sample Android app that contains a user interface layout (main.axml), +an internationalization string table (strings.xml) and some icons (drawable-XXX/icon.png) +would keep its resources in the "Resources" directory of the application: + +Resources/ + drawable/ + icon.png + + layout/ + main.axml + + values/ + strings.xml + +In order to get the build system to recognize Android resources, set the build action to +"AndroidResource". The native Android APIs do not operate directly with filenames, but +instead operate on resource IDs. When you compile an Android application that uses resources, +the build system will package the resources for distribution and generate a class called "R" +(this is an Android convention) that contains the tokens for each one of the resources +included. For example, for the above Resources layout, this is what the R class would expose: + +public class R { + public class drawable { + public const int icon = 0x123; + } + + public class layout { + public const int main = 0x456; + } + + public class strings { + public const int first_string = 0xabc; + public const int second_string = 0xbcd; + } +} + +You would then use R.drawable.icon to reference the drawable/icon.png file, or R.layout.main +to reference the layout/main.axml file, or R.strings.first_string to reference the first +string in the dictionary file values/strings.xml. diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Resources/values/Strings.xml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Resources/values/Strings.xml new file mode 100644 index 00000000..188b899a --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Resources/values/Strings.xml @@ -0,0 +1,5 @@ + + + Hello World, Click Me! + TodoApp.Uno + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Resources/values/Styles.xml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Resources/values/Styles.xml new file mode 100644 index 00000000..c02bd06f --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/Resources/values/Styles.xml @@ -0,0 +1,22 @@ + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/environment.conf b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/environment.conf new file mode 100644 index 00000000..fa6c2e32 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Android/environment.conf @@ -0,0 +1,2 @@ +# See this for more details: http://developer.xamarin.com/guides/android/advanced_topics/garbage_collection/ +MONO_GC_PARAMS=bridge-implementation=tarjan,nursery-size=32m,soft-heap-limit=256m \ No newline at end of file diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Desktop/Program.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Desktop/Program.cs new file mode 100644 index 00000000..648eb542 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/Desktop/Program.cs @@ -0,0 +1,19 @@ +using Uno.UI.Runtime.Skia; + +namespace TodoApp.Uno; +public class Program +{ + [STAThread] + public static void Main(string[] args) + { + var host = SkiaHostBuilder.Create() + .App(() => new App()) + .UseX11() + .UseLinuxFrameBuffer() + .UseMacOS() + .UseWindows() + .Build(); + + host.Run(); + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Entitlements.plist b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Entitlements.plist new file mode 100644 index 00000000..24c31036 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Entitlements.plist @@ -0,0 +1,6 @@ + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Info.plist b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Info.plist new file mode 100644 index 00000000..1bb02ddc --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Info.plist @@ -0,0 +1,26 @@ + + + + + UIDeviceFamily + + 2 + + LSApplicationCategoryType + public.app-category.utilities + UISupportedInterfaceOrientations + + UIInterfaceOrientationPortrait + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + XSAppIconAssets + Assets.xcassets/icon.appiconset + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Main.maccatalyst.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Main.maccatalyst.cs new file mode 100644 index 00000000..a2e34523 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Main.maccatalyst.cs @@ -0,0 +1,13 @@ +using UIKit; + +namespace TodoApp.Uno.MacCatalyst; +public class EntryPoint +{ + // This is the main entry point of the application. + public static void Main(string[] args) + { + // if you want to use a different Application Delegate class from "AppDelegate" + // you can specify it here. + UIApplication.Main(args, null, typeof(App)); + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Media.xcassets/LaunchImages.launchimage/Contents.json b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Media.xcassets/LaunchImages.launchimage/Contents.json new file mode 100644 index 00000000..69555e44 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/MacCatalyst/Media.xcassets/LaunchImages.launchimage/Contents.json @@ -0,0 +1,58 @@ +{ + "images": [ + { + "orientation": "portrait", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "2x", + "size": "640x960", + "idiom": "iphone" + }, + { + "orientation": "portrait", + "extent": "full-screen", + "minimum-system-version": "7.0", + "subtype": "retina4", + "scale": "2x", + "size": "640x1136", + "idiom": "iphone" + }, + { + "orientation": "portrait", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "1x", + "size": "768x1024", + "idiom": "ipad" + }, + { + "orientation": "landscape", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "1x", + "size": "1024x768", + "idiom": "ipad" + }, + { + "orientation": "portrait", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "2x", + "size": "1536x2048", + "idiom": "ipad" + }, + { + "orientation": "landscape", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "2x", + "size": "2048x1536", + "idiom": "ipad" + } + ], + "properties": {}, + "info": { + "version": 1, + "author": "" + } +} \ No newline at end of file diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/LinkerConfig.xml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/LinkerConfig.xml new file mode 100644 index 00000000..6947a56c --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/LinkerConfig.xml @@ -0,0 +1,10 @@ + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/Program.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/Program.cs new file mode 100644 index 00000000..10e4a9fc --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/Program.cs @@ -0,0 +1,13 @@ +namespace TodoApp.Uno; + +public class Program +{ + private static App? _app; + + public static int Main(string[] args) + { + Microsoft.UI.Xaml.Application.Start(_ => _app = new App()); + + return 0; + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/WasmCSS/Fonts.css b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/WasmCSS/Fonts.css new file mode 100644 index 00000000..4fdd6055 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/WasmCSS/Fonts.css @@ -0,0 +1,28 @@ +/** + When adding fonts here, make sure to add them using a base64 data uri, otherwise + fonts loading are delayed, and text may get displayed incorrectly. +*/ + +/* https://github.com/unoplatform/uno/issues/3954 */ +@font-face { + font-family: 'Segoe UI'; + src: local('Segoe UI'), local('-apple-system'), local('BlinkMacSystemFont'), local('Inter'), local('Cantarell'), local('Ubuntu'), local('Roboto'), local('Open Sans'), local('Noto Sans'), local('Helvetica Neue'), local('sans-serif'); +} + +@font-face { + font-family: 'Roboto'; + src: url(./Uno.Fonts.Roboto/Fonts/Roboto-Light.ttf) format('truetype'); + font-weight: 300; +} + +@font-face { + font-family: 'Roboto'; + src: url(./Uno.Fonts.Roboto/Fonts/Roboto-Regular.ttf) format('truetype'); + font-weight: 400; +} + +@font-face { + font-family: 'Roboto'; + src: url(./Uno.Fonts.Roboto/Fonts/Roboto-Medium.ttf) format('truetype'); + font-weight: 500; +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/WasmScripts/AppManifest.js b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/WasmScripts/AppManifest.js new file mode 100644 index 00000000..9fccab4c --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/WasmScripts/AppManifest.js @@ -0,0 +1,3 @@ +var UnoAppManifest = { + displayName: "TodoApp.Uno" +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/manifest.webmanifest b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/manifest.webmanifest new file mode 100644 index 00000000..d1f86976 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/manifest.webmanifest @@ -0,0 +1,10 @@ +{ + "background_color": "#ffffff", + "description": "TodoApp.Uno", + "display": "standalone", + "name": "TodoApp.Uno", + "short_name": "TodoApp.Uno", + "start_url": "/index.html", + "theme_color": "#ffffff", + "scope": "/" +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/wwwroot/staticwebapp.config.json b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/wwwroot/staticwebapp.config.json new file mode 100644 index 00000000..79c1b17c --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/wwwroot/staticwebapp.config.json @@ -0,0 +1,30 @@ +{ + "navigationFallback": { + "rewrite": "/index.html", + "exclude": [ + "*.{css,js}", + "*.{png}", + "*.{c,h,wasm,clr,pdb,dat,txt}" + ] + }, + "routes": [ + { + "route": "/package_*", + "headers": { + "cache-control": "public, immutable, max-age=31536000" + } + }, + { + "route": "/*.ttf", + "headers": { + "cache-control": "public, immutable, max-age=31536000" + } + }, + { + "route": "/*", + "headers": { + "cache-control": "must-revalidate, max-age=3600" + } + } + ] +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/wwwroot/web.config b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/wwwroot/web.config new file mode 100644 index 00000000..8f5a860f --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/WebAssembly/wwwroot/web.config @@ -0,0 +1,78 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Entitlements.plist b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Entitlements.plist new file mode 100644 index 00000000..24c31036 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Entitlements.plist @@ -0,0 +1,6 @@ + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Info.plist b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Info.plist new file mode 100644 index 00000000..ea3dcb4b --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Info.plist @@ -0,0 +1,43 @@ + + + + + LSRequiresIPhoneOS + + UIDeviceFamily + + 1 + 2 + + UIRequiredDeviceCapabilities + + armv7 + arm64 + + UISupportedInterfaceOrientations + + UIInterfaceOrientationPortrait + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UISupportedInterfaceOrientations~ipad + + UIInterfaceOrientationPortrait + UIInterfaceOrientationPortraitUpsideDown + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UIViewControllerBasedStatusBarAppearance + + XSAppIconAssets + Assets.xcassets/icon.appiconset + UIApplicationSupportsIndirectInputEvents + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Main.iOS.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Main.iOS.cs new file mode 100644 index 00000000..f3fa2955 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Main.iOS.cs @@ -0,0 +1,13 @@ +using UIKit; + +namespace TodoApp.Uno.iOS; +public class EntryPoint +{ + // This is the main entry point of the application. + public static void Main(string[] args) + { + // if you want to use a different Application Delegate class from "AppDelegate" + // you can specify it here. + UIApplication.Main(args, null, typeof(App)); + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Media.xcassets/LaunchImages.launchimage/Contents.json b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Media.xcassets/LaunchImages.launchimage/Contents.json new file mode 100644 index 00000000..69555e44 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/Media.xcassets/LaunchImages.launchimage/Contents.json @@ -0,0 +1,58 @@ +{ + "images": [ + { + "orientation": "portrait", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "2x", + "size": "640x960", + "idiom": "iphone" + }, + { + "orientation": "portrait", + "extent": "full-screen", + "minimum-system-version": "7.0", + "subtype": "retina4", + "scale": "2x", + "size": "640x1136", + "idiom": "iphone" + }, + { + "orientation": "portrait", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "1x", + "size": "768x1024", + "idiom": "ipad" + }, + { + "orientation": "landscape", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "1x", + "size": "1024x768", + "idiom": "ipad" + }, + { + "orientation": "portrait", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "2x", + "size": "1536x2048", + "idiom": "ipad" + }, + { + "orientation": "landscape", + "extent": "full-screen", + "minimum-system-version": "7.0", + "scale": "2x", + "size": "2048x1536", + "idiom": "ipad" + } + ], + "properties": {}, + "info": { + "version": 1, + "author": "" + } +} \ No newline at end of file diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/PrivacyInfo.xcprivacy b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/PrivacyInfo.xcprivacy new file mode 100644 index 00000000..902abb05 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Platforms/iOS/PrivacyInfo.xcprivacy @@ -0,0 +1,41 @@ + + + + + + + + NSPrivacyAccessedAPIType + NSPrivacyAccessedAPICategoryFileTimestamp + NSPrivacyAccessedAPITypeReasons + + C617.1 + + + + NSPrivacyAccessedAPIType + NSPrivacyAccessedAPICategorySystemBootTime + NSPrivacyAccessedAPITypeReasons + + 35F9.1 + + + + NSPrivacyAccessedAPIType + NSPrivacyAccessedAPICategoryDiskSpace + NSPrivacyAccessedAPITypeReasons + + E174.1 + + + + + + NSPrivacyAccessedAPIType + NSPrivacyAccessedAPICategoryUserDefaults + NSPrivacyAccessedAPITypeReasons + + CA92.1 + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Presentation/Shell.xaml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Presentation/Shell.xaml new file mode 100644 index 00000000..72b978f4 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Presentation/Shell.xaml @@ -0,0 +1,36 @@ + + + + + + + + + + + + + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Presentation/Shell.xaml.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Presentation/Shell.xaml.cs new file mode 100644 index 00000000..2226fd10 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Presentation/Shell.xaml.cs @@ -0,0 +1,10 @@ +namespace TodoApp.Uno.Presentation; + +public sealed partial class Shell : UserControl, IContentControlProvider +{ + public Shell() + { + this.InitializeComponent(); + } + public ContentControl ContentControl => Splash; +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Presentation/ShellViewModel.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Presentation/ShellViewModel.cs new file mode 100644 index 00000000..8bc20f1e --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Presentation/ShellViewModel.cs @@ -0,0 +1,13 @@ +namespace TodoApp.Uno.Presentation; + +public class ShellViewModel +{ + private readonly INavigator _navigator; + + public ShellViewModel( + INavigator navigator) + { + _navigator = navigator; + // Add code here to initialize or attach event handlers to singleton services + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/PublishProfiles/win-arm64.pubxml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/PublishProfiles/win-arm64.pubxml new file mode 100644 index 00000000..5d5632ab --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/PublishProfiles/win-arm64.pubxml @@ -0,0 +1,18 @@ + + + + + FileSystem + arm64 + win-arm64 + bin\$(Configuration)\$(TargetFramework)\$(RuntimeIdentifier)\publish\ + true + False + False + True + False + True + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/PublishProfiles/win-x64.pubxml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/PublishProfiles/win-x64.pubxml new file mode 100644 index 00000000..12b83447 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/PublishProfiles/win-x64.pubxml @@ -0,0 +1,18 @@ + + + + + FileSystem + x64 + win-x64 + bin\$(Configuration)\$(TargetFramework)\$(RuntimeIdentifier)\publish\ + true + False + False + True + False + True + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/PublishProfiles/win-x86.pubxml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/PublishProfiles/win-x86.pubxml new file mode 100644 index 00000000..65b8f1db --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/PublishProfiles/win-x86.pubxml @@ -0,0 +1,23 @@ + + + + + FileSystem + x86 + win-x86 + bin\$(Configuration)\$(TargetFramework)\$(RuntimeIdentifier)\publish\ + true + False + False + True + False + True + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/launchSettings.json b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/launchSettings.json new file mode 100644 index 00000000..415af151 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Properties/launchSettings.json @@ -0,0 +1,50 @@ +{ + "iisSettings": { + "windowsAuthentication": false, + "anonymousAuthentication": true, + "iisExpress": { + "applicationUrl": "http://localhost:8080", + "sslPort": 0 + } + }, + "profiles": { + // This profile is first in order for dotnet run to pick it up by default + "TodoApp.Uno (WebAssembly)": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "http://localhost:5000", + "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "TodoApp.Uno (WebAssembly IIS Express)": { + "commandName": "IISExpress", + "launchBrowser": true, + "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + // Note: In order to select this profile, you'll need to comment the `Packaged` profile below until this is fixed: https://aka.platform.uno/wasdk-maui-debug-profile-issue + "TodoApp.Uno (WinAppSDK Unpackaged)": { + "commandName": "Project", + "compatibleTargetFramework": "windows" + }, + "TodoApp.Uno (WinAppSDK Packaged)": { + "commandName": "MsixPackage", + "compatibleTargetFramework": "windows" + }, + "TodoApp.Uno (Desktop)": { + "commandName": "Project", + "compatibleTargetFramework": "desktop" + }, + "TodoApp.Uno (Desktop WSL2)": { + "commandName": "WSL2", + "commandLineArgs": "{ProjectDir}/bin/Debug/net8.0-desktop/TodoApp.Uno.dll", + "distributionName": "", + "compatibleTargetFramework": "desktop" + } + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ReadMe.md b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ReadMe.md new file mode 100644 index 00000000..93482da2 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ReadMe.md @@ -0,0 +1,7 @@ +# Getting Started + +Welcome to the Uno Platform! + +To discover how to get started with your new app: https://aka.platform.uno/get-started + +For more information on how to use the Uno.Sdk or upgrade Uno Platform packages in your solution: https://aka.platform.uno/using-uno-sdk \ No newline at end of file diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Services/LoggingHandler.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Services/LoggingHandler.cs new file mode 100644 index 00000000..67eac603 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Services/LoggingHandler.cs @@ -0,0 +1,49 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. +// See the LICENSE file in the project root for more information. + +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Linq; +using System.Net.Http; +using System.Text; +using System.Threading; +using System.Threading.Tasks; + +namespace TodoApp.Uno.Services; + +/// +/// A delegating handler that logs the request/response to stdout. +/// +public class LoggingHandler : DelegatingHandler +{ + public LoggingHandler() : base() + { + } + + public LoggingHandler(HttpMessageHandler innerHandler) : base(innerHandler) + { + } + + protected override async Task SendAsync(HttpRequestMessage request, CancellationToken cancellationToken) + { + Debug.WriteLine($"[HTTP] >>> {request.Method} {request.RequestUri}"); + await WriteContentAsync(request.Content, cancellationToken); + + HttpResponseMessage response = await base.SendAsync(request, cancellationToken); + + Debug.WriteLine($"[HTTP] <<< {response.StatusCode} {response.ReasonPhrase}"); + await WriteContentAsync(response.Content, cancellationToken); + + return response; + } + + private static async Task WriteContentAsync(HttpContent content, CancellationToken cancellationToken = default) + { + if (content != null) + { + Debug.WriteLine($"[HTTP] >>> {await content.ReadAsStringAsync(cancellationToken)}"); + } + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/en/Resources.resw b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/en/Resources.resw new file mode 100644 index 00000000..be03cf9b --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/en/Resources.resw @@ -0,0 +1,123 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + text/microsoft-resx + + + 2.0 + + + System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + TodoApp.Uno-en + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/es/Resources.resw b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/es/Resources.resw new file mode 100644 index 00000000..61da9ea1 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/es/Resources.resw @@ -0,0 +1,123 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + text/microsoft-resx + + + 2.0 + + + System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + TodoApp.Uno-es + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/fr/Resources.resw b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/fr/Resources.resw new file mode 100644 index 00000000..7d3ea6a7 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/fr/Resources.resw @@ -0,0 +1,123 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + text/microsoft-resx + + + 2.0 + + + System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + TodoApp.Uno-fr + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/pt-BR/Resources.resw b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/pt-BR/Resources.resw new file mode 100644 index 00000000..8aabbc06 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Strings/pt-BR/Resources.resw @@ -0,0 +1,123 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + text/microsoft-resx + + + 2.0 + + + System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + TodoApp.Uno-pt-BR + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Styles/ColorPaletteOverride.json b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Styles/ColorPaletteOverride.json new file mode 100644 index 00000000..cd154921 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Styles/ColorPaletteOverride.json @@ -0,0 +1,76 @@ +{ + "description": "Default Uno Material colors", + "seed": "#5946d2", + "coreColors": { + "primary": "#5946d2", + "secondary": "#6b4ea2" + }, + "extendedColors": [], + "schemes": { + "light": { + "primary": "#5946d2", + "surfaceTint": "#5946d2", + "onPrimary": "#ffffff", + "primaryContainer": "#e5deff", + "onPrimaryContainer": "#170065", + "secondary": "#6b4ea2", + "onSecondary": "#ffffff", + "secondaryContainer": "#ebddff", + "onSecondaryContainer": "#220555", + "tertiary": "#0061a4", + "onTertiary": "#ffffff", + "tertiaryContainer": "#cfe4ff", + "onTertiaryContainer": "#001d36", + "error": "#b3261e", + "onError": "#ffffff", + "errorContainer": "#f9dedc", + "onErrorContainer": "#410e0b", + "background": "#fcfbff", + "onBackground": "#1c1b1f", + "surface": "#ffffff", + "onSurface": "#1c1b1f", + "surfaceVariant": "#f2eff5", + "onSurfaceVariant": "#8b8494", + "outline": "#79747e", + "outlineVariant": "#c9c5d0", + "shadow": "#000000", + "scrim": "#000000", + "inverseSurface": "#e6e1e5", + "inverseOnSurface": "#1c1b1f", + "inversePrimary": "#2a009f" + }, + "dark": { + "primary": "#c7bfff", + "surfaceTint": "#c7bfff", + "onPrimary": "#2a009f", + "primaryContainer": "#4129ba", + "onPrimaryContainer": "#e4dfff", + "secondary": "#cdc2dc", + "onSecondary": "#332d41", + "secondaryContainer": "#433c52", + "onSecondaryContainer": "#eddfff", + "tertiary": "#9fcaff", + "onTertiary": "#003258", + "tertiaryContainer": "#00497d", + "onTertiaryContainer": "#d1e4ff", + "error": "#ffb4ab", + "onError": "#690005", + "errorContainer": "#93000a", + "onErrorContainer": "#ffdad6", + "background": "#1c1b1f", + "onBackground": "#e5e1e6", + "surface": "#302d37", + "onSurface": "#e6e1e5", + "surfaceVariant": "#47464f", + "onSurfaceVariant": "#c9c5d0", + "outline": "#928f99", + "outlineVariant": "#57545d", + "shadow": "#000000", + "scrim": "#000000", + "inverseSurface": "#e6e1e5", + "inverseOnSurface": "#1c1b1f", + "inversePrimary": "#2a009f" + } + }, + "palettes": {} +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Styles/ColorPaletteOverride.xaml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Styles/ColorPaletteOverride.xaml new file mode 100644 index 00000000..b47f3309 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Styles/ColorPaletteOverride.xaml @@ -0,0 +1,65 @@ + + + + + #5946D2 + #FFFFFF + #E5DEFF + #170065 + #6B4EA2 + #FFFFFF + #EBDDFF + #220555 + #0061A4 + #FFFFFF + #CFE4FF + #001D36 + #B3261E + #F9DEDC + #FFFFFF + #410E0B + #FCFBFF + #1C1B1F + #FFFFFF + #1C1B1F + #F2EFF5 + #8B8494 + #79747E + #F4EFF4 + #313033 + #C8BFFF + #5946D2 + #C9C5D0 + + + #C7BFFF + #2A009F + #4129BA + #E4DFFF + #CDC2DC + #332D41 + #433C52 + #EDDFFF + #9FCAFF + #003258 + #00497D + #D1E4FF + #FFB4AB + #93000A + #690005 + #FFDAD6 + #1C1B1F + #E5E1E6 + #302D37 + #E6E1E5 + #47464F + #C9C5D0 + #928F99 + #1C1B1F + #E6E1E5 + #2A009F + #C7BFFF + #57545D + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/TodoApp.Uno.csproj b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/TodoApp.Uno.csproj new file mode 100644 index 00000000..986128df --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/TodoApp.Uno.csproj @@ -0,0 +1,90 @@ + + + net8.0-android;net8.0-ios;net8.0-maccatalyst;net8.0-windows10.0.26100;net8.0-browserwasm;net8.0-desktop + + Exe + true + + + TodoApp.Uno + + com.companyname.TodoApp.Uno + + 1.0 + 1 + + Your.Name + + TodoApp.Uno powered by Uno Platform. + + + 10.0.26100.57 + + + + + Dsp; + Hosting; + Toolkit; + Logging; + Lottie; + Mvvm; + Configuration; + + + + Navigation; + + + True + + + + manual + VS: WildCard Development + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + MSBuild:Compile + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ViewModels/NotificationEventArgs.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ViewModels/NotificationEventArgs.cs new file mode 100644 index 00000000..439cca78 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ViewModels/NotificationEventArgs.cs @@ -0,0 +1,9 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. +// See the LICENSE file in the project root for more information. + +namespace TodoApp.Uno.ViewModels; + +internal record NotificationEventArgs(string Title, string Message, bool IsError) +{ +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ViewModels/TodoItemViewModel.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ViewModels/TodoItemViewModel.cs new file mode 100644 index 00000000..bfd471af --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ViewModels/TodoItemViewModel.cs @@ -0,0 +1,47 @@ +using CommunityToolkit.Mvvm.ComponentModel; +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using TodoApp.Uno.Database; + +namespace TodoApp.Uno.ViewModels +{ + public partial class TodoItemViewModel(TodoItem todoItem) : ObservableObject + { + public readonly TodoItem _todoItem = todoItem; + + public string Title + { + get => _todoItem.Title; + set => SetProperty(_todoItem.Title, value, _todoItem, (item, value) => item.Title = value); + } + + public bool IsComplete + { + get => _todoItem.IsComplete; + set => SetProperty(_todoItem.IsComplete, value, _todoItem, (item, value) => item.IsComplete = value); + } + + public string Version + { + get => _todoItem.Version; + set => SetProperty(_todoItem.Version, value, _todoItem, (item, value) => item.Version = value); + } + + public string Id => _todoItem.Id; + + public DateTimeOffset? UpdatedAt + { + get => _todoItem.UpdatedAt; + set => SetProperty(_todoItem.UpdatedAt, value, _todoItem, (item, value) => item.UpdatedAt = value); + } + + public bool Deleted + { + get => _todoItem.Deleted; + set => SetProperty(_todoItem.Deleted, value, _todoItem, (item, value) => item.Deleted = value); + } + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ViewModels/TodoListViewModel.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ViewModels/TodoListViewModel.cs new file mode 100644 index 00000000..c39f2d9d --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/ViewModels/TodoListViewModel.cs @@ -0,0 +1,126 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. +// See the LICENSE file in the project root for more information. + +using System; +using System.Collections.Generic; +using System.Threading; +using System.Threading.Tasks; +using CommunityToolkit.Datasync.Client; +using CommunityToolkit.Mvvm.ComponentModel; +using CommunityToolkit.Mvvm.Input; +using Microsoft.EntityFrameworkCore; +using TodoApp.Uno.Database; + +namespace TodoApp.Uno.ViewModels; + +/// +/// The view model for the TodoListWindow. +/// +public partial class TodoListViewModel(AppDbContext service) : ObservableRecipient +{ + internal event EventHandler NotificationHandler; + + [ObservableProperty] + private bool isRefreshing; + + [ObservableProperty] + private ConcurrentObservableCollection items = []; + + [ObservableProperty] + private string title = string.Empty; + + [RelayCommand] + public async Task AddItemAsync(CancellationToken cancellationToken = default) + { + try + { + // Create a new item + TodoItem addition = new() + { + Id = Guid.NewGuid().ToString("N"), + Title = Title, + }; + + // Add te item to the database + _ = service.TodoItems.Add(addition); + _ = await service.SaveChangesAsync(cancellationToken); + + // Add the item to the end of the list. + Items.Add(new TodoItemViewModel(addition)); + + // Update the title field ready for ext insertion. + Title = string.Empty; + } + catch (Exception ex) + { + NotificationHandler?.Invoke(this, new NotificationEventArgs(ex.GetType().Name, ex.Message, true)); + } + finally + { + NotificationHandler?.Invoke(this, new NotificationEventArgs("Item Added", "", false)); + } + } + + [RelayCommand] + public async Task EditItemAsync(string itemId, CancellationToken cancellationToken = default) + { + try + { + // Retrieve the item (by ID) from the service. + TodoItem item = await service.TodoItems.FindAsync([itemId], cancellationToken) + ?? throw new ApplicationException($"Item with ID '{itemId}' not found."); + + // Update the item in the database + item.IsComplete = !item.IsComplete; + _ = service.TodoItems.Update(item); + _ = await service.SaveChangesAsync(cancellationToken); + + // Update the item in the list + _ = Items.ReplaceIf(x => x.Id == itemId, new TodoItemViewModel(item)); + } + catch (Exception ex) + { + NotificationHandler?.Invoke(this, new NotificationEventArgs(ex.GetType().Name, ex.Message, true)); + } + finally + { + NotificationHandler?.Invoke(this, new NotificationEventArgs("Item Updated", "", false)); + } + } + + [RelayCommand] + public async Task LoadPageAsync(CancellationToken cancellationToken = default) + { + await RefreshItemsAsync(cancellationToken); + } + + [RelayCommand] + public async Task RefreshItemsAsync(CancellationToken cancellationToken = default) + { + try + { + IsRefreshing = true; + + // Synchronize data with the remote service (if any). + await service.SynchronizeAsync(cancellationToken); + + // Pull all items from the database. + IEnumerable itemsFromDatabase = await service.TodoItems.ToListAsync(cancellationToken); + + IEnumerable items = itemsFromDatabase.OrderBy(item => item.Id).Select(x => new TodoItemViewModel(x)); + + // Replace all the items in the collection. + Items.ReplaceAll(items); + } + catch (Exception ex) + { + NotificationHandler?.Invoke(this, new NotificationEventArgs(ex.GetType().Name, ex.Message, true)); + } + finally + { + IsRefreshing = false; + NotificationHandler?.Invoke(this, new NotificationEventArgs("Items Refreshed", "", false)); + } + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Views/TodoListPage.xaml b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Views/TodoListPage.xaml new file mode 100644 index 00000000..90ce4871 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Views/TodoListPage.xaml @@ -0,0 +1,112 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Views/TodoListPage.xaml.cs b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Views/TodoListPage.xaml.cs new file mode 100644 index 00000000..5b1529ff --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/Views/TodoListPage.xaml.cs @@ -0,0 +1,67 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Runtime.InteropServices.WindowsRuntime; +using CommunityToolkit.WinUI.Behaviors; +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 TodoApp.Uno.ViewModels; +using Windows.Foundation; +using Windows.Foundation.Collections; +using Windows.UI.ViewManagement; + +namespace TodoApp.Uno.Views; + +public sealed partial class TodoListPage : Page +{ + public TodoListViewModel ViewModel => (TodoListViewModel)DataContext!; + + public TodoListPage() + { + this.InitializeComponent(); + + DataContextChanged += DataContextChangedHandler; + } + + + + // on loaded override + private async void DataContextChangedHandler(object sender, DataContextChangedEventArgs e) + { + // Ensure e.NewValue is not null and is a TodoListViewModel (for some reason on iOS it is the ShellViewModel on one of the calls) + if (e.NewValue != null && e.NewValue is TodoListViewModel && ViewModel.LoadPageCommand.CanExecute(null)) + { + ViewModel.NotificationHandler += PublishNotification; + await ViewModel.LoadPageCommand.ExecuteAsync(null); + } + } + + internal void PublishNotification(object sender, NotificationEventArgs args) + { + Notification notification = new() + { + Title = args.Title, + Message = args.Message, + Severity = args.IsError ? InfoBarSeverity.Error : InfoBarSeverity.Informational, + Duration = args.IsError ? null : TimeSpan.FromSeconds(2) + }; + _ = this.NotificationQueue.Show(notification); + } + + private void TextBox_KeyDown(object sender, KeyRoutedEventArgs e) + { + // if key is enter, run AddItem command on ViewModel + if (e.Key == Windows.System.VirtualKey.Enter) + { + e.Handled = true; + ViewModel.AddItemCommand.Execute(null); + ViewModel.Title = string.Empty; + } + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/app.manifest b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/app.manifest new file mode 100644 index 00000000..bde52249 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/app.manifest @@ -0,0 +1,25 @@ + + + + + + + + + + + + + + + true/PM + PerMonitorV2, PerMonitor + + + diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/appsettings.development.json b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/appsettings.development.json new file mode 100644 index 00000000..2cf8df52 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/appsettings.development.json @@ -0,0 +1,9 @@ +{ + "AppConfig": { + "Environment": "Development" + }, + "ApiClient": { + "Url": "https://localhost:5002", + "UseNativeHandler": true + } +} diff --git a/samples/todoapp/TodoApp.Uno/TodoApp.Uno/appsettings.json b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/appsettings.json new file mode 100644 index 00000000..aa890ef1 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/TodoApp.Uno/appsettings.json @@ -0,0 +1,16 @@ +{ + "AppConfig": { + "Environment": "Production" + }, + "ApiClient": { + "UseNativeHandler": true + }, + "LocalizationConfiguration": { + "Cultures": [ + "es", + "fr", + "pt-BR", + "en" + ] + } +} diff --git a/samples/todoapp/TodoApp.Uno/global.json b/samples/todoapp/TodoApp.Uno/global.json new file mode 100644 index 00000000..694b05a7 --- /dev/null +++ b/samples/todoapp/TodoApp.Uno/global.json @@ -0,0 +1,9 @@ +{ + // To update the version of Uno please update the version of the Uno.Sdk here. See https://aka.platform.uno/upgrade-uno-packages for more information. + "msbuild-sdks": { + "Uno.Sdk": "5.4.5" + }, + "sdk":{ + "allowPrerelease": false + } +}