From 0942f0b308a1b9278b4c8e6113917f73ce138d84 Mon Sep 17 00:00:00 2001 From: Florian Edlmayer Date: Thu, 15 Jan 2026 15:22:05 +0100 Subject: [PATCH] Initial commit --- Client/AssemblyInfo.cs | 4 + Client/Interop.cs | 15 ++ .../Edit.razor | 112 +++++++++++++ .../Index.razor | 79 ++++++++++ .../ModuleInfo.cs | 19 +++ .../Settings.razor | 47 ++++++ .../Edit.resx | 141 +++++++++++++++++ .../Index.resx | 147 ++++++++++++++++++ .../Settings.resx | 126 +++++++++++++++ ...tenverein.Module.PremiumArea.Client.csproj | 38 +++++ Client/Services/PremiumAreaService.cs | 55 +++++++ Client/Startup/ClientStartup.cs | 18 +++ Client/_Imports.razor | 24 +++ ...enverein.Module.PremiumArea.Package.csproj | 29 ++++ ...bsolventenverein.Module.PremiumArea.nuspec | 38 +++++ Package/debug.cmd | 11 ++ Package/debug.sh | 12 ++ Package/icon.png | Bin 0 -> 7800 bytes Package/release.cmd | 8 + Package/release.sh | 7 + SZUAbsolventenverein.Module.PremiumArea.slnx | 7 + Server/AssemblyInfo.cs | 4 + Server/Controllers/PremiumAreaController.cs | 114 ++++++++++++++ Server/Manager/PremiumAreaManager.cs | 87 +++++++++++ .../Migrations/01000000_InitializeModule.cs | 30 ++++ .../PremiumAreaEntityBuilder.cs | 36 +++++ Server/Repository/PremiumAreaContext.cs | 26 ++++ Server/Repository/PremiumAreaRepository.cs | 75 +++++++++ ...tenverein.Module.PremiumArea.Server.csproj | 38 +++++ Server/Services/PremiumAreaService.cs | 101 ++++++++++++ Server/Startup/ServerStartup.cs | 28 ++++ Server/wwwroot/Module.css | 1 + Server/wwwroot/Module.js | 5 + Shared/Models/PremiumArea.cs | 15 ++ ...tenverein.Module.PremiumArea.Shared.csproj | 22 +++ exteneral.module.template.json | 6 + 36 files changed, 1525 insertions(+) create mode 100644 Client/AssemblyInfo.cs create mode 100644 Client/Interop.cs create mode 100644 Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Edit.razor create mode 100644 Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Index.razor create mode 100644 Client/Modules/SZUAbsolventenverein.Module.PremiumArea/ModuleInfo.cs create mode 100644 Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Settings.razor create mode 100644 Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Edit.resx create mode 100644 Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Index.resx create mode 100644 Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Settings.resx create mode 100644 Client/SZUAbsolventenverein.Module.PremiumArea.Client.csproj create mode 100644 Client/Services/PremiumAreaService.cs create mode 100644 Client/Startup/ClientStartup.cs create mode 100644 Client/_Imports.razor create mode 100644 Package/SZUAbsolventenverein.Module.PremiumArea.Package.csproj create mode 100644 Package/SZUAbsolventenverein.Module.PremiumArea.nuspec create mode 100644 Package/debug.cmd create mode 100644 Package/debug.sh create mode 100644 Package/icon.png create mode 100644 Package/release.cmd create mode 100644 Package/release.sh create mode 100644 SZUAbsolventenverein.Module.PremiumArea.slnx create mode 100644 Server/AssemblyInfo.cs create mode 100644 Server/Controllers/PremiumAreaController.cs create mode 100644 Server/Manager/PremiumAreaManager.cs create mode 100644 Server/Migrations/01000000_InitializeModule.cs create mode 100644 Server/Migrations/EntityBuilders/PremiumAreaEntityBuilder.cs create mode 100644 Server/Repository/PremiumAreaContext.cs create mode 100644 Server/Repository/PremiumAreaRepository.cs create mode 100644 Server/SZUAbsolventenverein.Module.PremiumArea.Server.csproj create mode 100644 Server/Services/PremiumAreaService.cs create mode 100644 Server/Startup/ServerStartup.cs create mode 100644 Server/wwwroot/Module.css create mode 100644 Server/wwwroot/Module.js create mode 100644 Shared/Models/PremiumArea.cs create mode 100644 Shared/SZUAbsolventenverein.Module.PremiumArea.Shared.csproj create mode 100644 exteneral.module.template.json diff --git a/Client/AssemblyInfo.cs b/Client/AssemblyInfo.cs new file mode 100644 index 0000000..0501391 --- /dev/null +++ b/Client/AssemblyInfo.cs @@ -0,0 +1,4 @@ +using System.Resources; +using Microsoft.Extensions.Localization; + +[assembly: RootNamespace("SZUAbsolventenverein.Module.PremiumArea.Client")] diff --git a/Client/Interop.cs b/Client/Interop.cs new file mode 100644 index 0000000..19b23f0 --- /dev/null +++ b/Client/Interop.cs @@ -0,0 +1,15 @@ +using Microsoft.JSInterop; +using System.Threading.Tasks; + +namespace SZUAbsolventenverein.Module.PremiumArea +{ + public class Interop + { + private readonly IJSRuntime _jsRuntime; + + public Interop(IJSRuntime jsRuntime) + { + _jsRuntime = jsRuntime; + } + } +} diff --git a/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Edit.razor b/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Edit.razor new file mode 100644 index 0000000..f00b3e9 --- /dev/null +++ b/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Edit.razor @@ -0,0 +1,112 @@ +@using Oqtane.Modules.Controls +@using SZUAbsolventenverein.Module.PremiumArea.Services +@using SZUAbsolventenverein.Module.PremiumArea.Models + +@namespace SZUAbsolventenverein.Module.PremiumArea +@inherits ModuleBase +@inject IPremiumAreaService PremiumAreaService +@inject NavigationManager NavigationManager +@inject IStringLocalizer Localizer + +
+
+
+ +
+ +
+
+
+ + @Localizer["Cancel"] +

+ @if (PageState.Action == "Edit") + { + + } +
+ +@code { + public override SecurityAccessLevel SecurityAccessLevel => SecurityAccessLevel.Edit; + + public override string Actions => "Add,Edit"; + + public override string Title => "Manage PremiumArea"; + + public override List Resources => new List() + { + new Stylesheet("_content/SZUAbsolventenverein.Module.PremiumArea/Module.css") + }; + + private ElementReference form; + private bool validated = false; + + private int _id; + private string _name; + private string _createdby; + private DateTime _createdon; + private string _modifiedby; + private DateTime _modifiedon; + + protected override async Task OnInitializedAsync() + { + try + { + if (PageState.Action == "Edit") + { + _id = Int32.Parse(PageState.QueryString["id"]); + PremiumArea PremiumArea = await PremiumAreaService.GetPremiumAreaAsync(_id, ModuleState.ModuleId); + if (PremiumArea != null) + { + _name = PremiumArea.Name; + _createdby = PremiumArea.CreatedBy; + _createdon = PremiumArea.CreatedOn; + _modifiedby = PremiumArea.ModifiedBy; + _modifiedon = PremiumArea.ModifiedOn; + } + } + } + catch (Exception ex) + { + await logger.LogError(ex, "Error Loading PremiumArea {PremiumAreaId} {Error}", _id, ex.Message); + AddModuleMessage(Localizer["Message.LoadError"], MessageType.Error); + } + } + + private async Task Save() + { + try + { + validated = true; + var interop = new Oqtane.UI.Interop(JSRuntime); + if (await interop.FormValid(form)) + { + if (PageState.Action == "Add") + { + PremiumArea PremiumArea = new PremiumArea(); + PremiumArea.ModuleId = ModuleState.ModuleId; + PremiumArea.Name = _name; + PremiumArea = await PremiumAreaService.AddPremiumAreaAsync(PremiumArea); + await logger.LogInformation("PremiumArea Added {PremiumArea}", PremiumArea); + } + else + { + PremiumArea PremiumArea = await PremiumAreaService.GetPremiumAreaAsync(_id, ModuleState.ModuleId); + PremiumArea.Name = _name; + await PremiumAreaService.UpdatePremiumAreaAsync(PremiumArea); + await logger.LogInformation("PremiumArea Updated {PremiumArea}", PremiumArea); + } + NavigationManager.NavigateTo(NavigateUrl()); + } + else + { + AddModuleMessage(Localizer["Message.SaveValidation"], MessageType.Warning); + } + } + catch (Exception ex) + { + await logger.LogError(ex, "Error Saving PremiumArea {Error}", ex.Message); + AddModuleMessage(Localizer["Message.SaveError"], MessageType.Error); + } + } +} diff --git a/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Index.razor b/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Index.razor new file mode 100644 index 0000000..2052382 --- /dev/null +++ b/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Index.razor @@ -0,0 +1,79 @@ +@using SZUAbsolventenverein.Module.PremiumArea.Services +@using SZUAbsolventenverein.Module.PremiumArea.Models + +@namespace SZUAbsolventenverein.Module.PremiumArea +@inherits ModuleBase +@inject IPremiumAreaService PremiumAreaService +@inject NavigationManager NavigationManager +@inject IStringLocalizer Localizer + +@if (_PremiumAreas == null) +{ +

Loading...

+} +else +{ + +
+
+ @if (@_PremiumAreas.Count != 0) + { + +
+   +   + @Localizer["Name"] +
+ + + + @context.Name + +
+ } + else + { +

@Localizer["Message.DisplayNone"]

+ } +} + +@code { + public override string RenderMode => RenderModes.Static; + + public override List Resources => new List() + { + new Stylesheet("_content/SZUAbsolventenverein.Module.PremiumArea/Module.css"), + new Script("_content/SZUAbsolventenverein.Module.PremiumArea/Module.js") + }; + + List _PremiumAreas; + + protected override async Task OnInitializedAsync() + { + try + { + _PremiumAreas = await PremiumAreaService.GetPremiumAreasAsync(ModuleState.ModuleId); + } + catch (Exception ex) + { + await logger.LogError(ex, "Error Loading PremiumArea {Error}", ex.Message); + AddModuleMessage(Localizer["Message.LoadError"], MessageType.Error); + } + } + + private async Task Delete(PremiumArea PremiumArea) + { + try + { + await PremiumAreaService.DeletePremiumAreaAsync(PremiumArea.PremiumAreaId, ModuleState.ModuleId); + await logger.LogInformation("PremiumArea Deleted {PremiumArea}", PremiumArea); + _PremiumAreas = await PremiumAreaService.GetPremiumAreasAsync(ModuleState.ModuleId); + StateHasChanged(); + } + catch (Exception ex) + { + await logger.LogError(ex, "Error Deleting PremiumArea {PremiumArea} {Error}", PremiumArea, ex.Message); + AddModuleMessage(Localizer["Message.DeleteError"], MessageType.Error); + } + } +} \ No newline at end of file diff --git a/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/ModuleInfo.cs b/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/ModuleInfo.cs new file mode 100644 index 0000000..f393a02 --- /dev/null +++ b/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/ModuleInfo.cs @@ -0,0 +1,19 @@ +using Oqtane.Models; +using Oqtane.Modules; + +namespace SZUAbsolventenverein.Module.PremiumArea +{ + public class ModuleInfo : IModule + { + public ModuleDefinition ModuleDefinition => new ModuleDefinition + { + Name = "PremiumArea", + Description = "This module adds a premium member system to Octane. Users receive premium status after completing a payment. Premium members get access to exclusive features and content.", + Version = "1.0.0", + ServerManagerType = "SZUAbsolventenverein.Module.PremiumArea.Manager.PremiumAreaManager, SZUAbsolventenverein.Module.PremiumArea.Server.Oqtane", + ReleaseVersions = "1.0.0", + Dependencies = "SZUAbsolventenverein.Module.PremiumArea.Shared.Oqtane", + PackageName = "SZUAbsolventenverein.Module.PremiumArea" + }; + } +} diff --git a/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Settings.razor b/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Settings.razor new file mode 100644 index 0000000..fc77636 --- /dev/null +++ b/Client/Modules/SZUAbsolventenverein.Module.PremiumArea/Settings.razor @@ -0,0 +1,47 @@ +@namespace SZUAbsolventenverein.Module.PremiumArea +@inherits ModuleBase +@inject ISettingService SettingService +@inject IStringLocalizer Localizer + +
+
+ +
+ +
+
+
+ +@code { + private string resourceType = "SZUAbsolventenverein.Module.PremiumArea.Settings, SZUAbsolventenverein.Module.PremiumArea.Client.Oqtane"; // for localization + public override string Title => "PremiumArea Settings"; + + string _value; + + protected override async Task OnInitializedAsync() + { + try + { + Dictionary settings = await SettingService.GetModuleSettingsAsync(ModuleState.ModuleId); + _value = SettingService.GetSetting(settings, "SettingName", ""); + } + catch (Exception ex) + { + AddModuleMessage(ex.Message, MessageType.Error); + } + } + + public async Task UpdateSettings() + { + try + { + var settings = await SettingService.GetModuleSettingsAsync(ModuleState.ModuleId); + settings = SettingService.SetSetting(settings, "SettingName", _value); + await SettingService.UpdateModuleSettingsAsync(settings, ModuleState.ModuleId); + } + catch (Exception ex) + { + AddModuleMessage(ex.Message, MessageType.Error); + } + } +} diff --git a/Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Edit.resx b/Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Edit.resx new file mode 100644 index 0000000..f004544 --- /dev/null +++ b/Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Edit.resx @@ -0,0 +1,141 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 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 + + + Name: + + + Enter the name + + + Save + + + Cancel + + + Error Loading PremiumArea + + + Please Provide All Required Information + + + Error Saving PremiumArea + + \ No newline at end of file diff --git a/Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Index.resx b/Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Index.resx new file mode 100644 index 0000000..6463291 --- /dev/null +++ b/Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Index.resx @@ -0,0 +1,147 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 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 + + + Name + + + Add PremiumArea + + + Edit + + + Delete + + + Delete PremiumArea + + + Are You Sure You Wish To Delete This PremiumArea? + + + No PremiumAreas To Display + + + Error Loading PremiumArea + + + Error Deleting PremiumArea + + \ No newline at end of file diff --git a/Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Settings.resx b/Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Settings.resx new file mode 100644 index 0000000..83dc88f --- /dev/null +++ b/Client/Resources/SZUAbsolventenverein.Module.PremiumArea/Settings.resx @@ -0,0 +1,126 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 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 + + + Name: + + + Enter a value + + \ No newline at end of file diff --git a/Client/SZUAbsolventenverein.Module.PremiumArea.Client.csproj b/Client/SZUAbsolventenverein.Module.PremiumArea.Client.csproj new file mode 100644 index 0000000..e7eb8db --- /dev/null +++ b/Client/SZUAbsolventenverein.Module.PremiumArea.Client.csproj @@ -0,0 +1,38 @@ + + + + net10.0 + 1.0.0 + SZUAbsolventenverein + SZUAbsolventenverein + This module adds a premium member system to Octane. Users receive premium status after completing a payment. Premium members get access to exclusive features and content. + SZUAbsolventenverein.Module.PremiumArea + SZUAbsolventenverein + SZUAbsolventenverein.Module.PremiumArea.Client.Oqtane + true + + + + + + + + + + + + + + + + ..\..\oqtane.framework\Oqtane.Server\bin\Debug\net10.0\Oqtane.Client.dll + ..\..\oqtane.framework\Oqtane.Server\bin\Debug\net10.0\Oqtane.Shared.dll + + + + + false + false + + + diff --git a/Client/Services/PremiumAreaService.cs b/Client/Services/PremiumAreaService.cs new file mode 100644 index 0000000..35bfc7f --- /dev/null +++ b/Client/Services/PremiumAreaService.cs @@ -0,0 +1,55 @@ +using System.Collections.Generic; +using System.Linq; +using System.Net.Http; +using System.Threading.Tasks; +using Oqtane.Services; +using Oqtane.Shared; + +namespace SZUAbsolventenverein.Module.PremiumArea.Services +{ + public interface IPremiumAreaService + { + Task> GetPremiumAreasAsync(int ModuleId); + + Task GetPremiumAreaAsync(int PremiumAreaId, int ModuleId); + + Task AddPremiumAreaAsync(Models.PremiumArea PremiumArea); + + Task UpdatePremiumAreaAsync(Models.PremiumArea PremiumArea); + + Task DeletePremiumAreaAsync(int PremiumAreaId, int ModuleId); + } + + public class PremiumAreaService : ServiceBase, IPremiumAreaService + { + public PremiumAreaService(HttpClient http, SiteState siteState) : base(http, siteState) { } + + private string Apiurl => CreateApiUrl("PremiumArea"); + + public async Task> GetPremiumAreasAsync(int ModuleId) + { + List PremiumAreas = await GetJsonAsync>(CreateAuthorizationPolicyUrl($"{Apiurl}?moduleid={ModuleId}", EntityNames.Module, ModuleId), Enumerable.Empty().ToList()); + return PremiumAreas.OrderBy(item => item.Name).ToList(); + } + + public async Task GetPremiumAreaAsync(int PremiumAreaId, int ModuleId) + { + return await GetJsonAsync(CreateAuthorizationPolicyUrl($"{Apiurl}/{PremiumAreaId}/{ModuleId}", EntityNames.Module, ModuleId)); + } + + public async Task AddPremiumAreaAsync(Models.PremiumArea PremiumArea) + { + return await PostJsonAsync(CreateAuthorizationPolicyUrl($"{Apiurl}", EntityNames.Module, PremiumArea.ModuleId), PremiumArea); + } + + public async Task UpdatePremiumAreaAsync(Models.PremiumArea PremiumArea) + { + return await PutJsonAsync(CreateAuthorizationPolicyUrl($"{Apiurl}/{PremiumArea.PremiumAreaId}", EntityNames.Module, PremiumArea.ModuleId), PremiumArea); + } + + public async Task DeletePremiumAreaAsync(int PremiumAreaId, int ModuleId) + { + await DeleteAsync(CreateAuthorizationPolicyUrl($"{Apiurl}/{PremiumAreaId}/{ModuleId}", EntityNames.Module, ModuleId)); + } + } +} diff --git a/Client/Startup/ClientStartup.cs b/Client/Startup/ClientStartup.cs new file mode 100644 index 0000000..5c5d9d3 --- /dev/null +++ b/Client/Startup/ClientStartup.cs @@ -0,0 +1,18 @@ +using Microsoft.Extensions.DependencyInjection; +using System.Linq; +using Oqtane.Services; +using SZUAbsolventenverein.Module.PremiumArea.Services; + +namespace SZUAbsolventenverein.Module.PremiumArea.Startup +{ + public class ClientStartup : IClientStartup + { + public void ConfigureServices(IServiceCollection services) + { + if (!services.Any(s => s.ServiceType == typeof(IPremiumAreaService))) + { + services.AddScoped(); + } + } + } +} diff --git a/Client/_Imports.razor b/Client/_Imports.razor new file mode 100644 index 0000000..2f6fb1b --- /dev/null +++ b/Client/_Imports.razor @@ -0,0 +1,24 @@ +@using System +@using System.Linq +@using System.Collections.Generic +@using System.Net.Http +@using System.Net.Http.Json + +@using Microsoft.AspNetCore.Components.Authorization +@using Microsoft.AspNetCore.Components.Routing +@using Microsoft.AspNetCore.Components.Web +@using Microsoft.Extensions.Localization +@using Microsoft.JSInterop + +@using Oqtane.Models +@using Oqtane.Modules +@using Oqtane.Modules.Controls +@using Oqtane.Providers +@using Oqtane.Security +@using Oqtane.Services +@using Oqtane.Shared +@using Oqtane.Themes +@using Oqtane.Themes.Controls +@using Oqtane.UI +@using Oqtane.Enums +@using Oqtane.Interfaces \ No newline at end of file diff --git a/Package/SZUAbsolventenverein.Module.PremiumArea.Package.csproj b/Package/SZUAbsolventenverein.Module.PremiumArea.Package.csproj new file mode 100644 index 0000000..60561ee --- /dev/null +++ b/Package/SZUAbsolventenverein.Module.PremiumArea.Package.csproj @@ -0,0 +1,29 @@ + + + + net10.0 + false + false + + + + + True + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/Package/SZUAbsolventenverein.Module.PremiumArea.nuspec b/Package/SZUAbsolventenverein.Module.PremiumArea.nuspec new file mode 100644 index 0000000..a8db0f8 --- /dev/null +++ b/Package/SZUAbsolventenverein.Module.PremiumArea.nuspec @@ -0,0 +1,38 @@ + + + + $projectname$ + 1.0.0 + SZUAbsolventenverein + SZUAbsolventenverein + PremiumArea + This module adds a premium member system to Octane. Users receive premium status after completing a payment. Premium members get access to exclusive features and content. + SZUAbsolventenverein + false + MIT + https://github.com/oqtane/oqtane.framework + icon.png + oqtane module + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/Package/debug.cmd b/Package/debug.cmd new file mode 100644 index 0000000..cebe487 --- /dev/null +++ b/Package/debug.cmd @@ -0,0 +1,11 @@ +@echo off +set TargetFramework=%1 +set ProjectName=%2 + +XCOPY "..\Client\bin\Debug\%TargetFramework%\%ProjectName%.Client.Oqtane.dll" "..\..\oqtane.framework\Oqtane.Server\bin\Debug\%TargetFramework%\" /Y +XCOPY "..\Client\bin\Debug\%TargetFramework%\%ProjectName%.Client.Oqtane.pdb" "..\..\oqtane.framework\Oqtane.Server\bin\Debug\%TargetFramework%\" /Y +XCOPY "..\Server\bin\Debug\%TargetFramework%\%ProjectName%.Server.Oqtane.dll" "..\..\oqtane.framework\Oqtane.Server\bin\Debug\%TargetFramework%\" /Y +XCOPY "..\Server\bin\Debug\%TargetFramework%\%ProjectName%.Server.Oqtane.pdb" "..\..\oqtane.framework\Oqtane.Server\bin\Debug\%TargetFramework%\" /Y +XCOPY "..\Shared\bin\Debug\%TargetFramework%\%ProjectName%.Shared.Oqtane.dll" "..\..\oqtane.framework\Oqtane.Server\bin\Debug\%TargetFramework%\" /Y +XCOPY "..\Shared\bin\Debug\%TargetFramework%\%ProjectName%.Shared.Oqtane.pdb" "..\..\oqtane.framework\Oqtane.Server\bin\Debug\%TargetFramework%\" /Y +XCOPY "..\Server\wwwroot\*" "..\..\oqtane.framework\Oqtane.Server\wwwroot\_content\%ProjectName%\" /Y /S /I \ No newline at end of file diff --git a/Package/debug.sh b/Package/debug.sh new file mode 100644 index 0000000..4a8b343 --- /dev/null +++ b/Package/debug.sh @@ -0,0 +1,12 @@ +#!/bin/bash + +TargetFramework=$1 +ProjectName=$2 + +cp -f "../Client/bin/Debug/$TargetFramework/$ProjectName$.Client.Oqtane.dll" "../../oqtane.framework/Oqtane.Server/bin/Debug/$TargetFramework/" +cp -f "../Client/bin/Debug/$TargetFramework/$ProjectName$.Client.Oqtane.pdb" "../../oqtane.framework/Oqtane.Server/bin/Debug/$TargetFramework/" +cp -f "../Server/bin/Debug/$TargetFramework/$ProjectName$.Server.Oqtane.dll" "../../oqtane.framework/Oqtane.Server/bin/Debug/$TargetFramework/" +cp -f "../Server/bin/Debug/$TargetFramework/$ProjectName$.Server.Oqtane.pdb" "../../oqtane.framework/Oqtane.Server/bin/Debug/$TargetFramework/" +cp -f "../Shared/bin/Debug/$TargetFramework/$ProjectName$.Shared.Oqtane.dll" "../../oqtane.framework/Oqtane.Server/bin/Debug/$TargetFramework/" +cp -f "../Shared/bin/Debug/$TargetFramework/$ProjectName$.Shared.Oqtane.pdb" "../../oqtane.framework/Oqtane.Server/bin/Debug/$TargetFramework/" +cp -rf "../Server/wwwroot/"* "../../oqtane.framework/Oqtane.Server/wwwroot/_content/%ProjectName%/" diff --git a/Package/icon.png b/Package/icon.png new file mode 100644 index 0000000000000000000000000000000000000000..7422cf2f313e352d93b08d4e87fec74e52117713 GIT binary patch literal 7800 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 M>>IU0xA6S`1HzXf%>V!Z literal 0 HcmV?d00001 diff --git a/Package/release.cmd b/Package/release.cmd new file mode 100644 index 0000000..9eb2046 --- /dev/null +++ b/Package/release.cmd @@ -0,0 +1,8 @@ +@echo off +set TargetFramework=%1 +set ProjectName=%2 + +del "*.nupkg" +"..\..\oqtane.framework\oqtane.package\FixProps.exe" +"..\..\oqtane.framework\oqtane.package\nuget.exe" pack %ProjectName%.nuspec -Properties targetframework=%TargetFramework%;projectname=%ProjectName% +XCOPY "*.nupkg" "..\..\oqtane.framework\Oqtane.Server\Packages\" /Y \ No newline at end of file diff --git a/Package/release.sh b/Package/release.sh new file mode 100644 index 0000000..443bf64 --- /dev/null +++ b/Package/release.sh @@ -0,0 +1,7 @@ +TargetFramework=$1 +ProjectName=$2 + +find . -name "*.nupkg" -delete +"..\..\oqtane.framework\oqtane.package\FixProps.exe" +"..\..\oqtane.framework\oqtane.package\nuget.exe" pack %ProjectName%.nuspec -Properties targetframework=%TargetFramework%;projectname=%ProjectName% +cp -f "*.nupkg" "..\..\oqtane.framework\Oqtane.Server\Packages\" \ No newline at end of file diff --git a/SZUAbsolventenverein.Module.PremiumArea.slnx b/SZUAbsolventenverein.Module.PremiumArea.slnx new file mode 100644 index 0000000..07bfea7 --- /dev/null +++ b/SZUAbsolventenverein.Module.PremiumArea.slnx @@ -0,0 +1,7 @@ + + + + + + + diff --git a/Server/AssemblyInfo.cs b/Server/AssemblyInfo.cs new file mode 100644 index 0000000..85e5e7b --- /dev/null +++ b/Server/AssemblyInfo.cs @@ -0,0 +1,4 @@ +using System.Resources; +using Microsoft.Extensions.Localization; + +[assembly: RootNamespace("SZUAbsolventenverein.Module.PremiumArea.Server")] diff --git a/Server/Controllers/PremiumAreaController.cs b/Server/Controllers/PremiumAreaController.cs new file mode 100644 index 0000000..7204f8d --- /dev/null +++ b/Server/Controllers/PremiumAreaController.cs @@ -0,0 +1,114 @@ +using Microsoft.AspNetCore.Mvc; +using Microsoft.AspNetCore.Authorization; +using System.Collections.Generic; +using Microsoft.AspNetCore.Http; +using Oqtane.Shared; +using Oqtane.Enums; +using Oqtane.Infrastructure; +using SZUAbsolventenverein.Module.PremiumArea.Services; +using Oqtane.Controllers; +using System.Net; +using System.Threading.Tasks; + +namespace SZUAbsolventenverein.Module.PremiumArea.Controllers +{ + [Route(ControllerRoutes.ApiRoute)] + public class PremiumAreaController : ModuleControllerBase + { + private readonly IPremiumAreaService _PremiumAreaService; + + public PremiumAreaController(IPremiumAreaService PremiumAreaService, ILogManager logger, IHttpContextAccessor accessor) : base(logger, accessor) + { + _PremiumAreaService = PremiumAreaService; + } + + // GET: api/?moduleid=x + [HttpGet] + [Authorize(Policy = PolicyNames.ViewModule)] + public async Task> Get(string moduleid) + { + int ModuleId; + if (int.TryParse(moduleid, out ModuleId) && IsAuthorizedEntityId(EntityNames.Module, ModuleId)) + { + return await _PremiumAreaService.GetPremiumAreasAsync(ModuleId); + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Get Attempt {ModuleId}", moduleid); + HttpContext.Response.StatusCode = (int)HttpStatusCode.Forbidden; + return null; + } + } + + // GET api//5 + [HttpGet("{id}/{moduleid}")] + [Authorize(Policy = PolicyNames.ViewModule)] + public async Task Get(int id, int moduleid) + { + Models.PremiumArea PremiumArea = await _PremiumAreaService.GetPremiumAreaAsync(id, moduleid); + if (PremiumArea != null && IsAuthorizedEntityId(EntityNames.Module, PremiumArea.ModuleId)) + { + return PremiumArea; + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Get Attempt {PremiumAreaId} {ModuleId}", id, moduleid); + HttpContext.Response.StatusCode = (int)HttpStatusCode.Forbidden; + return null; + } + } + + // POST api/ + [HttpPost] + [Authorize(Policy = PolicyNames.EditModule)] + public async Task Post([FromBody] Models.PremiumArea PremiumArea) + { + if (ModelState.IsValid && IsAuthorizedEntityId(EntityNames.Module, PremiumArea.ModuleId)) + { + PremiumArea = await _PremiumAreaService.AddPremiumAreaAsync(PremiumArea); + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Post Attempt {PremiumArea}", PremiumArea); + HttpContext.Response.StatusCode = (int)HttpStatusCode.Forbidden; + PremiumArea = null; + } + return PremiumArea; + } + + // PUT api//5 + [HttpPut("{id}")] + [Authorize(Policy = PolicyNames.EditModule)] + public async Task Put(int id, [FromBody] Models.PremiumArea PremiumArea) + { + if (ModelState.IsValid && PremiumArea.PremiumAreaId == id && IsAuthorizedEntityId(EntityNames.Module, PremiumArea.ModuleId)) + { + PremiumArea = await _PremiumAreaService.UpdatePremiumAreaAsync(PremiumArea); + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Put Attempt {PremiumArea}", PremiumArea); + HttpContext.Response.StatusCode = (int)HttpStatusCode.Forbidden; + PremiumArea = null; + } + return PremiumArea; + } + + // DELETE api//5 + [HttpDelete("{id}/{moduleid}")] + [Authorize(Policy = PolicyNames.EditModule)] + public async Task Delete(int id, int moduleid) + { + Models.PremiumArea PremiumArea = await _PremiumAreaService.GetPremiumAreaAsync(id, moduleid); + if (PremiumArea != null && IsAuthorizedEntityId(EntityNames.Module, PremiumArea.ModuleId)) + { + await _PremiumAreaService.DeletePremiumAreaAsync(id, PremiumArea.ModuleId); + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Delete Attempt {PremiumAreaId} {ModuleId}", id, moduleid); + HttpContext.Response.StatusCode = (int)HttpStatusCode.Forbidden; + } + } + } +} diff --git a/Server/Manager/PremiumAreaManager.cs b/Server/Manager/PremiumAreaManager.cs new file mode 100644 index 0000000..091b57a --- /dev/null +++ b/Server/Manager/PremiumAreaManager.cs @@ -0,0 +1,87 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text.Json; +using Oqtane.Modules; +using Oqtane.Models; +using Oqtane.Infrastructure; +using Oqtane.Interfaces; +using Oqtane.Enums; +using Oqtane.Repository; +using SZUAbsolventenverein.Module.PremiumArea.Repository; +using System.Threading.Tasks; + +namespace SZUAbsolventenverein.Module.PremiumArea.Manager +{ + public class PremiumAreaManager : MigratableModuleBase, IInstallable, IPortable, ISearchable + { + private readonly IPremiumAreaRepository _PremiumAreaRepository; + private readonly IDBContextDependencies _DBContextDependencies; + + public PremiumAreaManager(IPremiumAreaRepository PremiumAreaRepository, IDBContextDependencies DBContextDependencies) + { + _PremiumAreaRepository = PremiumAreaRepository; + _DBContextDependencies = DBContextDependencies; + } + + public bool Install(Tenant tenant, string version) + { + return Migrate(new PremiumAreaContext(_DBContextDependencies), tenant, MigrationType.Up); + } + + public bool Uninstall(Tenant tenant) + { + return Migrate(new PremiumAreaContext(_DBContextDependencies), tenant, MigrationType.Down); + } + + public string ExportModule(Oqtane.Models.Module module) + { + string content = ""; + List PremiumAreas = _PremiumAreaRepository.GetPremiumAreas(module.ModuleId).ToList(); + if (PremiumAreas != null) + { + content = JsonSerializer.Serialize(PremiumAreas); + } + return content; + } + + public void ImportModule(Oqtane.Models.Module module, string content, string version) + { + List PremiumAreas = null; + if (!string.IsNullOrEmpty(content)) + { + PremiumAreas = JsonSerializer.Deserialize>(content); + } + if (PremiumAreas != null) + { + foreach(var PremiumArea in PremiumAreas) + { + _PremiumAreaRepository.AddPremiumArea(new Models.PremiumArea { ModuleId = module.ModuleId, Name = PremiumArea.Name }); + } + } + } + + public Task> GetSearchContentsAsync(PageModule pageModule, DateTime lastIndexedOn) + { + var searchContentList = new List(); + + foreach (var PremiumArea in _PremiumAreaRepository.GetPremiumAreas(pageModule.ModuleId)) + { + if (PremiumArea.ModifiedOn >= lastIndexedOn) + { + searchContentList.Add(new SearchContent + { + EntityName = "SZUAbsolventenvereinPremiumArea", + EntityId = PremiumArea.PremiumAreaId.ToString(), + Title = PremiumArea.Name, + Body = PremiumArea.Name, + ContentModifiedBy = PremiumArea.ModifiedBy, + ContentModifiedOn = PremiumArea.ModifiedOn + }); + } + } + + return Task.FromResult(searchContentList); + } + } +} diff --git a/Server/Migrations/01000000_InitializeModule.cs b/Server/Migrations/01000000_InitializeModule.cs new file mode 100644 index 0000000..2843086 --- /dev/null +++ b/Server/Migrations/01000000_InitializeModule.cs @@ -0,0 +1,30 @@ +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Oqtane.Databases.Interfaces; +using Oqtane.Migrations; +using SZUAbsolventenverein.Module.PremiumArea.Migrations.EntityBuilders; +using SZUAbsolventenverein.Module.PremiumArea.Repository; + +namespace SZUAbsolventenverein.Module.PremiumArea.Migrations +{ + [DbContext(typeof(PremiumAreaContext))] + [Migration("SZUAbsolventenverein.Module.PremiumArea.01.00.00.00")] + public class InitializeModule : MultiDatabaseMigration + { + public InitializeModule(IDatabase database) : base(database) + { + } + + protected override void Up(MigrationBuilder migrationBuilder) + { + var entityBuilder = new PremiumAreaEntityBuilder(migrationBuilder, ActiveDatabase); + entityBuilder.Create(); + } + + protected override void Down(MigrationBuilder migrationBuilder) + { + var entityBuilder = new PremiumAreaEntityBuilder(migrationBuilder, ActiveDatabase); + entityBuilder.Drop(); + } + } +} diff --git a/Server/Migrations/EntityBuilders/PremiumAreaEntityBuilder.cs b/Server/Migrations/EntityBuilders/PremiumAreaEntityBuilder.cs new file mode 100644 index 0000000..fced43e --- /dev/null +++ b/Server/Migrations/EntityBuilders/PremiumAreaEntityBuilder.cs @@ -0,0 +1,36 @@ +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Migrations.Operations; +using Microsoft.EntityFrameworkCore.Migrations.Operations.Builders; +using Oqtane.Databases.Interfaces; +using Oqtane.Migrations; +using Oqtane.Migrations.EntityBuilders; + +namespace SZUAbsolventenverein.Module.PremiumArea.Migrations.EntityBuilders +{ + public class PremiumAreaEntityBuilder : AuditableBaseEntityBuilder + { + private const string _entityTableName = "SZUAbsolventenvereinPremiumArea"; + private readonly PrimaryKey _primaryKey = new("PK_SZUAbsolventenvereinPremiumArea", x => x.PremiumAreaId); + private readonly ForeignKey _moduleForeignKey = new("FK_SZUAbsolventenvereinPremiumArea_Module", x => x.ModuleId, "Module", "ModuleId", ReferentialAction.Cascade); + + public PremiumAreaEntityBuilder(MigrationBuilder migrationBuilder, IDatabase database) : base(migrationBuilder, database) + { + EntityTableName = _entityTableName; + PrimaryKey = _primaryKey; + ForeignKeys.Add(_moduleForeignKey); + } + + protected override PremiumAreaEntityBuilder BuildTable(ColumnsBuilder table) + { + PremiumAreaId = AddAutoIncrementColumn(table,"PremiumAreaId"); + ModuleId = AddIntegerColumn(table,"ModuleId"); + Name = AddMaxStringColumn(table,"Name"); + AddAuditableColumns(table); + return this; + } + + public OperationBuilder PremiumAreaId { get; set; } + public OperationBuilder ModuleId { get; set; } + public OperationBuilder Name { get; set; } + } +} diff --git a/Server/Repository/PremiumAreaContext.cs b/Server/Repository/PremiumAreaContext.cs new file mode 100644 index 0000000..8ac03c8 --- /dev/null +++ b/Server/Repository/PremiumAreaContext.cs @@ -0,0 +1,26 @@ +using Microsoft.EntityFrameworkCore; +using Microsoft.AspNetCore.Http; +using Oqtane.Modules; +using Oqtane.Repository; +using Oqtane.Infrastructure; +using Oqtane.Repository.Databases.Interfaces; + +namespace SZUAbsolventenverein.Module.PremiumArea.Repository +{ + public class PremiumAreaContext : DBContextBase, ITransientService, IMultiDatabase + { + public virtual DbSet PremiumArea { get; set; } + + public PremiumAreaContext(IDBContextDependencies DBContextDependencies) : base(DBContextDependencies) + { + // ContextBase handles multi-tenant database connections + } + + protected override void OnModelCreating(ModelBuilder builder) + { + base.OnModelCreating(builder); + + builder.Entity().ToTable(ActiveDatabase.RewriteName("SZUAbsolventenvereinPremiumArea")); + } + } +} diff --git a/Server/Repository/PremiumAreaRepository.cs b/Server/Repository/PremiumAreaRepository.cs new file mode 100644 index 0000000..0c4ddbb --- /dev/null +++ b/Server/Repository/PremiumAreaRepository.cs @@ -0,0 +1,75 @@ +using Microsoft.EntityFrameworkCore; +using System.Linq; +using System.Collections.Generic; +using Oqtane.Modules; + +namespace SZUAbsolventenverein.Module.PremiumArea.Repository +{ + public interface IPremiumAreaRepository + { + IEnumerable GetPremiumAreas(int ModuleId); + Models.PremiumArea GetPremiumArea(int PremiumAreaId); + Models.PremiumArea GetPremiumArea(int PremiumAreaId, bool tracking); + Models.PremiumArea AddPremiumArea(Models.PremiumArea PremiumArea); + Models.PremiumArea UpdatePremiumArea(Models.PremiumArea PremiumArea); + void DeletePremiumArea(int PremiumAreaId); + } + + public class PremiumAreaRepository : IPremiumAreaRepository, ITransientService + { + private readonly IDbContextFactory _factory; + + public PremiumAreaRepository(IDbContextFactory factory) + { + _factory = factory; + } + + public IEnumerable GetPremiumAreas(int ModuleId) + { + using var db = _factory.CreateDbContext(); + return db.PremiumArea.Where(item => item.ModuleId == ModuleId).ToList(); + } + + public Models.PremiumArea GetPremiumArea(int PremiumAreaId) + { + return GetPremiumArea(PremiumAreaId, true); + } + + public Models.PremiumArea GetPremiumArea(int PremiumAreaId, bool tracking) + { + using var db = _factory.CreateDbContext(); + if (tracking) + { + return db.PremiumArea.Find(PremiumAreaId); + } + else + { + return db.PremiumArea.AsNoTracking().FirstOrDefault(item => item.PremiumAreaId == PremiumAreaId); + } + } + + public Models.PremiumArea AddPremiumArea(Models.PremiumArea PremiumArea) + { + using var db = _factory.CreateDbContext(); + db.PremiumArea.Add(PremiumArea); + db.SaveChanges(); + return PremiumArea; + } + + public Models.PremiumArea UpdatePremiumArea(Models.PremiumArea PremiumArea) + { + using var db = _factory.CreateDbContext(); + db.Entry(PremiumArea).State = EntityState.Modified; + db.SaveChanges(); + return PremiumArea; + } + + public void DeletePremiumArea(int PremiumAreaId) + { + using var db = _factory.CreateDbContext(); + Models.PremiumArea PremiumArea = db.PremiumArea.Find(PremiumAreaId); + db.PremiumArea.Remove(PremiumArea); + db.SaveChanges(); + } + } +} diff --git a/Server/SZUAbsolventenverein.Module.PremiumArea.Server.csproj b/Server/SZUAbsolventenverein.Module.PremiumArea.Server.csproj new file mode 100644 index 0000000..c96d25d --- /dev/null +++ b/Server/SZUAbsolventenverein.Module.PremiumArea.Server.csproj @@ -0,0 +1,38 @@ + + + + net10.0 + true + 1.0.0 + SZUAbsolventenverein.Module.PremiumArea + SZUAbsolventenverein + SZUAbsolventenverein + This module adds a premium member system to Octane. Users receive premium status after completing a payment. Premium members get access to exclusive features and content. + SZUAbsolventenverein + SZUAbsolventenverein.Module.PremiumArea.Server.Oqtane + true + false + + + + + + + + + + + + + + + + + + + + + ..\..\oqtane.framework\Oqtane.Server\bin\Debug\net10.0\Oqtane.Server.dll + ..\..\oqtane.framework\Oqtane.Server\bin\Debug\net10.0\Oqtane.Shared.dll + + diff --git a/Server/Services/PremiumAreaService.cs b/Server/Services/PremiumAreaService.cs new file mode 100644 index 0000000..18c616b --- /dev/null +++ b/Server/Services/PremiumAreaService.cs @@ -0,0 +1,101 @@ +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; +using Microsoft.AspNetCore.Http; +using Oqtane.Enums; +using Oqtane.Infrastructure; +using Oqtane.Models; +using Oqtane.Security; +using Oqtane.Shared; +using SZUAbsolventenverein.Module.PremiumArea.Repository; + +namespace SZUAbsolventenverein.Module.PremiumArea.Services +{ + public class ServerPremiumAreaService : IPremiumAreaService + { + private readonly IPremiumAreaRepository _PremiumAreaRepository; + private readonly IUserPermissions _userPermissions; + private readonly ILogManager _logger; + private readonly IHttpContextAccessor _accessor; + private readonly Alias _alias; + + public ServerPremiumAreaService(IPremiumAreaRepository PremiumAreaRepository, IUserPermissions userPermissions, ITenantManager tenantManager, ILogManager logger, IHttpContextAccessor accessor) + { + _PremiumAreaRepository = PremiumAreaRepository; + _userPermissions = userPermissions; + _logger = logger; + _accessor = accessor; + _alias = tenantManager.GetAlias(); + } + + public Task> GetPremiumAreasAsync(int ModuleId) + { + if (_userPermissions.IsAuthorized(_accessor.HttpContext.User, _alias.SiteId, EntityNames.Module, ModuleId, PermissionNames.View)) + { + return Task.FromResult(_PremiumAreaRepository.GetPremiumAreas(ModuleId).ToList()); + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Get Attempt {ModuleId}", ModuleId); + return null; + } + } + + public Task GetPremiumAreaAsync(int PremiumAreaId, int ModuleId) + { + if (_userPermissions.IsAuthorized(_accessor.HttpContext.User, _alias.SiteId, EntityNames.Module, ModuleId, PermissionNames.View)) + { + return Task.FromResult(_PremiumAreaRepository.GetPremiumArea(PremiumAreaId)); + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Get Attempt {PremiumAreaId} {ModuleId}", PremiumAreaId, ModuleId); + return null; + } + } + + public Task AddPremiumAreaAsync(Models.PremiumArea PremiumArea) + { + if (_userPermissions.IsAuthorized(_accessor.HttpContext.User, _alias.SiteId, EntityNames.Module, PremiumArea.ModuleId, PermissionNames.Edit)) + { + PremiumArea = _PremiumAreaRepository.AddPremiumArea(PremiumArea); + _logger.Log(LogLevel.Information, this, LogFunction.Create, "PremiumArea Added {PremiumArea}", PremiumArea); + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Add Attempt {PremiumArea}", PremiumArea); + PremiumArea = null; + } + return Task.FromResult(PremiumArea); + } + + public Task UpdatePremiumAreaAsync(Models.PremiumArea PremiumArea) + { + if (_userPermissions.IsAuthorized(_accessor.HttpContext.User, _alias.SiteId, EntityNames.Module, PremiumArea.ModuleId, PermissionNames.Edit)) + { + PremiumArea = _PremiumAreaRepository.UpdatePremiumArea(PremiumArea); + _logger.Log(LogLevel.Information, this, LogFunction.Update, "PremiumArea Updated {PremiumArea}", PremiumArea); + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Update Attempt {PremiumArea}", PremiumArea); + PremiumArea = null; + } + return Task.FromResult(PremiumArea); + } + + public Task DeletePremiumAreaAsync(int PremiumAreaId, int ModuleId) + { + if (_userPermissions.IsAuthorized(_accessor.HttpContext.User, _alias.SiteId, EntityNames.Module, ModuleId, PermissionNames.Edit)) + { + _PremiumAreaRepository.DeletePremiumArea(PremiumAreaId); + _logger.Log(LogLevel.Information, this, LogFunction.Delete, "PremiumArea Deleted {PremiumAreaId}", PremiumAreaId); + } + else + { + _logger.Log(LogLevel.Error, this, LogFunction.Security, "Unauthorized PremiumArea Delete Attempt {PremiumAreaId} {ModuleId}", PremiumAreaId, ModuleId); + } + return Task.CompletedTask; + } + } +} diff --git a/Server/Startup/ServerStartup.cs b/Server/Startup/ServerStartup.cs new file mode 100644 index 0000000..68f7da9 --- /dev/null +++ b/Server/Startup/ServerStartup.cs @@ -0,0 +1,28 @@ +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Hosting; +using Microsoft.Extensions.DependencyInjection; +using Oqtane.Infrastructure; +using SZUAbsolventenverein.Module.PremiumArea.Repository; +using SZUAbsolventenverein.Module.PremiumArea.Services; + +namespace SZUAbsolventenverein.Module.PremiumArea.Startup +{ + public class ServerStartup : IServerStartup + { + public void Configure(IApplicationBuilder app, IWebHostEnvironment env) + { + // not implemented + } + + public void ConfigureMvc(IMvcBuilder mvcBuilder) + { + // not implemented + } + + public void ConfigureServices(IServiceCollection services) + { + services.AddTransient(); + services.AddDbContextFactory(opt => { }, ServiceLifetime.Transient); + } + } +} diff --git a/Server/wwwroot/Module.css b/Server/wwwroot/Module.css new file mode 100644 index 0000000..0856a26 --- /dev/null +++ b/Server/wwwroot/Module.css @@ -0,0 +1 @@ +/* Module Custom Styles */ \ No newline at end of file diff --git a/Server/wwwroot/Module.js b/Server/wwwroot/Module.js new file mode 100644 index 0000000..d5eb845 --- /dev/null +++ b/Server/wwwroot/Module.js @@ -0,0 +1,5 @@ +/* Module Script */ +var SZUAbsolventenverein = SZUAbsolventenverein || {}; + +SZUAbsolventenverein.PremiumArea = { +}; \ No newline at end of file diff --git a/Shared/Models/PremiumArea.cs b/Shared/Models/PremiumArea.cs new file mode 100644 index 0000000..fcc6661 --- /dev/null +++ b/Shared/Models/PremiumArea.cs @@ -0,0 +1,15 @@ +using System.ComponentModel.DataAnnotations; +using System.ComponentModel.DataAnnotations.Schema; +using Oqtane.Models; + +namespace SZUAbsolventenverein.Module.PremiumArea.Models +{ + [Table("SZUAbsolventenvereinPremiumArea")] + public class PremiumArea : ModelBase + { + [Key] + public int PremiumAreaId { get; set; } + public int ModuleId { get; set; } + public string Name { get; set; } + } +} diff --git a/Shared/SZUAbsolventenverein.Module.PremiumArea.Shared.csproj b/Shared/SZUAbsolventenverein.Module.PremiumArea.Shared.csproj new file mode 100644 index 0000000..9b03ec3 --- /dev/null +++ b/Shared/SZUAbsolventenverein.Module.PremiumArea.Shared.csproj @@ -0,0 +1,22 @@ + + + + net10.0 + 1.0.0 + SZUAbsolventenverein.Module.PremiumArea + SZUAbsolventenverein + SZUAbsolventenverein + This module adds a premium member system to Octane. Users receive premium status after completing a payment. Premium members get access to exclusive features and content. + SZUAbsolventenverein + SZUAbsolventenverein.Module.PremiumArea.Shared.Oqtane + + + + + + + + ..\..\oqtane.framework\Oqtane.Server\bin\Debug\net10.0\Oqtane.Shared.dll + + + diff --git a/exteneral.module.template.json b/exteneral.module.template.json new file mode 100644 index 0000000..f38a7f6 --- /dev/null +++ b/exteneral.module.template.json @@ -0,0 +1,6 @@ +{ + "Title": "Default Module Template", + "Type": "External", + "Version": "10.0.0", + "Namespace": "SZUAbsolventenverein.Module.PremiumArea" +}