Compare commits
68 Commits
image
...
865797d3f8
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
865797d3f8 | ||
|
|
cefa47a176 | ||
|
|
c4d7fd6ed5 | ||
|
|
f4f04e4a42 | ||
|
|
6807f2b6e6 | ||
|
|
0dd0c1bf4c | ||
|
|
87f26f9367 | ||
|
|
c0c18f2ddd | ||
|
|
b0866754e0 | ||
|
|
870930914e | ||
|
|
54effa67ac | ||
|
|
48ad7dda87 | ||
|
|
d09926a8b4 | ||
|
|
954d57b7a6 | ||
|
|
781da32796 | ||
|
|
b1ed916da4 | ||
|
|
94a2dbf801 | ||
|
|
dc9276e3e9 | ||
|
|
aba2007481 | ||
|
|
cf4b73735b | ||
|
|
cb3a2ae042 | ||
|
|
8ad6a143de | ||
|
|
46d5bcd00d | ||
|
|
545f59e059 | ||
|
|
cad9617451 | ||
|
|
04084b4bf7 | ||
|
|
69516b2701 | ||
|
|
def8702489 | ||
|
|
8262c4979b | ||
|
|
ad9e2ae8c1 | ||
|
|
4d4648b187 | ||
|
|
f04dba72fd | ||
|
|
03872e8bba | ||
|
|
4330b53824 | ||
|
|
7660e8ce81 | ||
|
|
9da0bf3a43 | ||
|
|
9983a58ba9 | ||
|
|
3db943d652 | ||
|
|
40f0213a73 | ||
|
|
5a4d4a7a04 | ||
|
|
c9d46be196 | ||
|
|
19c22e6ae8 | ||
|
|
d1852f28c8 | ||
|
|
e80d5b92f3 | ||
|
|
62bfdb2023 | ||
|
|
646ccb0f18 | ||
|
|
ac52bc6df9 | ||
|
|
b3212acf6d | ||
|
|
a93de45270 | ||
|
|
c7e0bfd8da | ||
|
|
5f5690e84d | ||
|
|
05b74b929e | ||
|
|
328e194611 | ||
|
|
8a65c03c2c | ||
|
|
297a7c60bd | ||
|
|
8e5a37a0c9 | ||
|
|
ad6f28023e | ||
|
|
c5c24d44c9 | ||
|
|
3687e573e0 | ||
|
|
b7a7a8e078 | ||
|
|
15780bccee | ||
|
|
82d2c48ff7 | ||
|
|
41b1f8ae1d | ||
|
|
eda6625e91 | ||
|
|
701388ee34 | ||
|
|
3e099988bc | ||
|
|
ee967cd046 | ||
|
|
76c3e6ddde |
41
.gitea/workflows/docker-image-dev.yml
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
name: Build And Push Dev Docker Image
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
docker:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
env:
|
||||||
|
DOTNET_ROLL_FORWARD: Major
|
||||||
|
REGISTRY: git.beging.de
|
||||||
|
BASE_IMAGE: git.beging.de/troogs/fs-onboarding-server
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
|
- name: Setup .NET SDK
|
||||||
|
uses: actions/setup-dotnet@v5
|
||||||
|
with:
|
||||||
|
dotnet-version: "9.0.x"
|
||||||
|
|
||||||
|
- name: Publish server project
|
||||||
|
run: dotnet publish ./FoodsharingSiegen.Server/FoodsharingSiegen.Server.csproj -c Release -o ./Publish/Server
|
||||||
|
|
||||||
|
- name: Login to Gitea registry
|
||||||
|
run: echo "${{ secrets.REGISTRY_TOKEN }}" | docker login "${{ env.REGISTRY }}" -u "${{ secrets.REGISTRY_USERNAME }}" --password-stdin
|
||||||
|
|
||||||
|
- name: Build docker image
|
||||||
|
run: |
|
||||||
|
docker build \
|
||||||
|
-f ./Docker/dockerfile.server \
|
||||||
|
-t "${{ env.BASE_IMAGE }}:dev" \
|
||||||
|
.
|
||||||
|
|
||||||
|
- name: Push docker images
|
||||||
|
run: |
|
||||||
|
docker push "${{ env.BASE_IMAGE }}:dev"
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
name: Build And Release
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- "v*"
|
|
||||||
workflow_dispatch:
|
|
||||||
inputs:
|
|
||||||
tag_name:
|
|
||||||
description: "Tag for the release (for example: v1.0.0)"
|
|
||||||
required: true
|
|
||||||
type: string
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
release:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
env:
|
|
||||||
RELEASE_TAG: ${{ github.event.inputs.tag_name || github.ref_name }}
|
|
||||||
DOTNET_ROLL_FORWARD: Major
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v6
|
|
||||||
|
|
||||||
- name: Setup .NET SDK
|
|
||||||
uses: actions/setup-dotnet@v5
|
|
||||||
with:
|
|
||||||
dotnet-version: "9.0.x"
|
|
||||||
|
|
||||||
- name: Restore dependencies
|
|
||||||
run: dotnet restore FoodsharingSiegen.sln
|
|
||||||
|
|
||||||
- name: Publish server build
|
|
||||||
run: dotnet publish ./FoodsharingSiegen.Server/FoodsharingSiegen.Server.csproj -c Release -o ./artifacts/publish
|
|
||||||
|
|
||||||
- name: Create zip package
|
|
||||||
run: |
|
|
||||||
cd artifacts
|
|
||||||
zip -r FoodsharingSiegen.Server-${{ env.RELEASE_TAG }}.zip publish
|
|
||||||
|
|
||||||
- name: Create release and upload artifact
|
|
||||||
uses: https://gitea.com/actions/gitea-release-action@v1
|
|
||||||
with:
|
|
||||||
files: artifacts/FoodsharingSiegen.Server-${{ env.RELEASE_TAG }}.zip
|
|
||||||
tag_name: ${{ env.RELEASE_TAG }}
|
|
||||||
name: Release ${{ env.RELEASE_TAG }}
|
|
||||||
target_commitish: ${{ github.sha }}
|
|
||||||
token: ${{ github.token }}
|
|
||||||
13
.gitignore
vendored
@@ -6,3 +6,16 @@ riderModule.iml
|
|||||||
|
|
||||||
Publish/
|
Publish/
|
||||||
app.db
|
app.db
|
||||||
|
FoodsharingSiegen.Server/wwwroot/buildinfo.txt
|
||||||
|
FoodsharingSiegen.Server/config/appsettings.json
|
||||||
|
|
||||||
|
# Generated CSS files from SCSS
|
||||||
|
FoodsharingSiegen.Server/Pages/AuditView.razor.css
|
||||||
|
FoodsharingSiegen.Server/Shared/DefaultLayout.razor.css
|
||||||
|
FoodsharingSiegen.Server/Shared/NavMenu.razor.css
|
||||||
|
FoodsharingSiegen.Server/wwwroot/css/site.css
|
||||||
|
FoodsharingSiegen.Server/wwwroot/css/site.min.css
|
||||||
|
FoodsharingSiegen.Server/wwwroot/css/site.css.map
|
||||||
|
FoodsharingSiegen.Server/Controls/ProspectContainer.razor.css
|
||||||
|
FoodsharingSiegen.Server/Controls/ProspectContainer.razor.css.map
|
||||||
|
FoodsharingSiegen.Server/Shared/DefaultLayout.razor.css.map
|
||||||
|
|||||||
7
Docker/.env
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
MAIL_HOST=smtp.example.com
|
||||||
|
MAIL_PORT=587
|
||||||
|
MAIL_USERNAME=your_username
|
||||||
|
MAIL_PASSWORD=your_password
|
||||||
|
MAIL_FROM_ADDRESS=noreply@example.com
|
||||||
|
MAIL_FROM_NAME="FS Einarbeitungen Musterhausen"
|
||||||
|
MAIL_USE_SSL=true
|
||||||
@@ -5,6 +5,17 @@ services:
|
|||||||
image: ghcr.io/troogs/fs-onboarding/server:latest
|
image: ghcr.io/troogs/fs-onboarding/server:latest
|
||||||
ports:
|
ports:
|
||||||
- "8100:56000"
|
- "8100:56000"
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
environment:
|
||||||
|
# Mail Configuration Examples. These override options in appsettings.json.
|
||||||
|
# All values below can also be set via the active .env file or appsettings.json.
|
||||||
|
- Settings__Mail__Host=${MAIL_HOST:-smtp.example.com}
|
||||||
|
- Settings__Mail__Port=${MAIL_PORT:-587}
|
||||||
|
- Settings__Mail__Username=${MAIL_USERNAME:-your_username}
|
||||||
|
- Settings__Mail__Password=${MAIL_PASSWORD:-your_password}
|
||||||
|
- Settings__Mail__FromAddress=${MAIL_FROM_ADDRESS:-no-reply@example.com}
|
||||||
|
- Settings__Mail__UseSsl=${MAIL_USE_SSL:-true}
|
||||||
volumes:
|
volumes:
|
||||||
- /docker/data/fs-onboarding/config:/app/config/
|
- /docker/data/fs-onboarding/config:/app/config/
|
||||||
- /docker/data/fs-onboarding/data:/app/data/
|
- /docker/data/fs-onboarding/data:/app/data/
|
||||||
@@ -59,7 +59,7 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the prospect (ab)
|
/// Gets or sets the value of the prospect (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public Prospect Prospect { get; set; }
|
public Prospect Prospect { get; set; } = null!;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the prospect id (ab)
|
/// Gets or sets the value of the prospect id (ab)
|
||||||
@@ -74,7 +74,7 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the user (ab)
|
/// Gets or sets the value of the user (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public User User { get; set; }
|
public User User { get; set; } = null!;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the user id (ab)
|
/// Gets or sets the value of the user id (ab)
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the interactions (ab)
|
/// Gets or sets the value of the interactions (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public IList<Interaction> Interactions { get; set; }
|
public IList<Interaction> Interactions { get; set; } = new List<Interaction>();
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the memo (ab)
|
/// Gets or sets the value of the memo (ab)
|
||||||
@@ -51,7 +51,7 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the name (ab)
|
/// Gets or sets the value of the name (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string Name { get; set; }
|
public string Name { get; set; } = string.Empty;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the state of the record within the system.
|
/// Gets or sets the state of the record within the system.
|
||||||
@@ -63,6 +63,16 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public bool Warning { get; set; }
|
public bool Warning { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets a token string used to securely authorize upload logic.
|
||||||
|
/// </summary>
|
||||||
|
public Guid? VerificationToken { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets uploaded identity verification images.
|
||||||
|
/// </summary>
|
||||||
|
public ICollection<ProspectImage>? Images { get; set; }
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
28
FoodsharingSiegen.Contracts/Entity/ProspectImage.cs
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
using System.ComponentModel.DataAnnotations.Schema;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Contracts.Entity
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Represents an uploaded image for a prospect's verification.
|
||||||
|
/// </summary>
|
||||||
|
public class ProspectImage
|
||||||
|
{
|
||||||
|
[Key]
|
||||||
|
public Guid Id { get; set; }
|
||||||
|
|
||||||
|
public Guid ProspectId { get; set; }
|
||||||
|
|
||||||
|
[ForeignKey("ProspectId")]
|
||||||
|
public Prospect? Prospect { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
public byte[] ImageData { get; set; } = [];
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[MaxLength(100)]
|
||||||
|
public string ContentType { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
public DateTime Created { get; set; }
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -20,7 +20,7 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the encrypted password (ab)
|
/// Gets or sets the value of the encrypted password (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string EncryptedPassword { get; set; }
|
public string EncryptedPassword { get; set; } = string.Empty;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the force logout (ab)
|
/// Gets or sets the value of the force logout (ab)
|
||||||
@@ -30,7 +30,7 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the groups (ab)
|
/// Gets or sets the value of the groups (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string Groups { get; set; }
|
public string Groups { get; set; } = string.Empty;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the groups list (ab)
|
/// Gets or sets the value of the groups list (ab)
|
||||||
@@ -57,12 +57,12 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the interactions (ab)
|
/// Gets or sets the value of the interactions (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public IList<Interaction> Interactions { get; set; }
|
public IList<Interaction> Interactions { get; set; } = new List<Interaction>();
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the mail (ab)
|
/// Gets or sets the value of the mail (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string Mail { get; set; }
|
public string Mail { get; set; } = string.Empty;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the memo (ab)
|
/// Gets or sets the value of the memo (ab)
|
||||||
@@ -72,7 +72,7 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the name (ab)
|
/// Gets or sets the value of the name (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string Name { get; set; }
|
public string Name { get; set; } = string.Empty;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the network (ab)
|
/// Gets or sets the value of the network (ab)
|
||||||
@@ -95,16 +95,21 @@ namespace FoodsharingSiegen.Contracts.Entity
|
|||||||
set => EncryptedPassword = Cryptor.Encrypt(value);
|
set => EncryptedPassword = Cryptor.Encrypt(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the reset token for password recovery (ab)
|
||||||
|
/// </summary>
|
||||||
|
public string? ResetToken { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the expiry date for the reset token (ab)
|
||||||
|
/// </summary>
|
||||||
|
public DateTime? ResetTokenExpiry { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the type (ab)
|
/// Gets or sets the value of the type (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public UserType Type { get; set; }
|
public UserType Type { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets or sets the value of the verified (ab)
|
|
||||||
/// </summary>
|
|
||||||
public bool Verified { get; set; }
|
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Public Method Clone
|
#region Public Method Clone
|
||||||
|
|||||||
@@ -62,7 +62,32 @@ namespace FoodsharingSiegen.Contracts.Enums
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// The remove interaction audit type
|
/// The remove interaction audit type
|
||||||
/// </summary>
|
/// </summary>
|
||||||
RemoveInteraction = 100
|
RemoveInteraction = 100,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The delete prospect images audit type
|
||||||
|
/// </summary>
|
||||||
|
DeleteProspectImages = 110,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The view prospect images audit type
|
||||||
|
/// </summary>
|
||||||
|
ViewProspectImages = 120,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The upload prospect image audit type
|
||||||
|
/// </summary>
|
||||||
|
UploadProspectImage = 130,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The request password reset audit type
|
||||||
|
/// </summary>
|
||||||
|
RequestPasswordReset = 140,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The change own password audit type
|
||||||
|
/// </summary>
|
||||||
|
ChangeOwnPassword = 150
|
||||||
|
|
||||||
#endregion Prospects
|
#endregion Prospects
|
||||||
}
|
}
|
||||||
|
|||||||
10
FoodsharingSiegen.Contracts/Enums/ProspectSortOption.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
namespace FoodsharingSiegen.Contracts.Enums
|
||||||
|
{
|
||||||
|
public enum ProspectSortOption
|
||||||
|
{
|
||||||
|
NameAscending,
|
||||||
|
NameDescending,
|
||||||
|
ModifiedAscending,
|
||||||
|
ModifiedDescending
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,6 +6,11 @@
|
|||||||
|
|
||||||
public bool DisableStepIn { get; set; }
|
public bool DisableStepIn { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the mail server settings.
|
||||||
|
/// </summary>
|
||||||
|
public MailSettings Mail { get; set; } = new();
|
||||||
|
|
||||||
public TermSettings Terms { get; set; } = new();
|
public TermSettings Terms { get; set; } = new();
|
||||||
|
|
||||||
public bool TestMode { get; set; }
|
public bool TestMode { get; set; }
|
||||||
|
|||||||
38
FoodsharingSiegen.Contracts/Model/MailSettings.cs
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
namespace FoodsharingSiegen.Contracts.Model
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Configuration settings for the mail service.
|
||||||
|
/// </summary>
|
||||||
|
public class MailSettings
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the SMTP server host.
|
||||||
|
/// </summary>
|
||||||
|
public string Host { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the SMTP port.
|
||||||
|
/// </summary>
|
||||||
|
public int Port { get; set; } = 587;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the username for SMTP authentication.
|
||||||
|
/// </summary>
|
||||||
|
public string Username { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the password for SMTP authentication.
|
||||||
|
/// </summary>
|
||||||
|
public string Password { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the sender address.
|
||||||
|
/// </summary>
|
||||||
|
public string FromAddress { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets whether to use SSL/TLS connection.
|
||||||
|
/// </summary>
|
||||||
|
public bool UseSsl { get; set; } = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -8,6 +8,8 @@
|
|||||||
|
|
||||||
public bool WithoutIdCheck { get; set; }
|
public bool WithoutIdCheck { get; set; }
|
||||||
|
|
||||||
|
public bool IdCheckPossible { get; set; }
|
||||||
|
|
||||||
public bool NoActivity { get; set; }
|
public bool NoActivity { get; set; }
|
||||||
|
|
||||||
public bool RecentActivity { get; set; }
|
public bool RecentActivity { get; set; }
|
||||||
|
|||||||
@@ -12,6 +12,26 @@ namespace FoodsharingSiegen.Contracts
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public const string ProspectFilter = "ProspectFilter";
|
public const string ProspectFilter = "ProspectFilter";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Represents the storage key used for prospect sorting preferences.
|
||||||
|
/// </summary>
|
||||||
|
public const string SortProspects = "SortProspects";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Represents the storage key used for sorting all prospects.
|
||||||
|
/// </summary>
|
||||||
|
public const string SortProspectsAll = "SortProspectsAll";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Represents the storage key used for sorting completed prospects.
|
||||||
|
/// </summary>
|
||||||
|
public const string SortProspectsDone = "SortProspectsDone";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Represents the storage key used for sorting prospects pending verification.
|
||||||
|
/// </summary>
|
||||||
|
public const string SortProspectsVerify = "SortProspectsVerify";
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The token key
|
/// The token key
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -2,11 +2,13 @@ using FoodsharingSiegen.Contracts;
|
|||||||
using FoodsharingSiegen.Contracts.Entity;
|
using FoodsharingSiegen.Contracts.Entity;
|
||||||
using FoodsharingSiegen.Contracts.Enums;
|
using FoodsharingSiegen.Contracts.Enums;
|
||||||
using FoodsharingSiegen.Contracts.Helper;
|
using FoodsharingSiegen.Contracts.Helper;
|
||||||
|
using FoodsharingSiegen.Contracts.Model;
|
||||||
using FoodsharingSiegen.Server.Data;
|
using FoodsharingSiegen.Server.Data;
|
||||||
using FoodsharingSiegen.Server.Service;
|
using FoodsharingSiegen.Server.Service;
|
||||||
using FoodsharingSiegen.Shared.Helper;
|
using FoodsharingSiegen.Shared.Helper;
|
||||||
using Microsoft.AspNetCore.Components.Authorization;
|
using Microsoft.AspNetCore.Components.Authorization;
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using Microsoft.Extensions.Options;
|
||||||
|
|
||||||
namespace FoodsharingSiegen.Server.Auth
|
namespace FoodsharingSiegen.Server.Auth
|
||||||
{
|
{
|
||||||
@@ -50,6 +52,16 @@ namespace FoodsharingSiegen.Server.Auth
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private User? _user;
|
private User? _user;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The mail service
|
||||||
|
/// </summary>
|
||||||
|
private readonly IMailService _mailService;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The application settings
|
||||||
|
/// </summary>
|
||||||
|
private readonly AppSettings _appSettings;
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Setup/Teardown
|
#region Setup/Teardown
|
||||||
@@ -60,14 +72,19 @@ namespace FoodsharingSiegen.Server.Auth
|
|||||||
/// <param name="context">The context</param>
|
/// <param name="context">The context</param>
|
||||||
/// <param name="localStorageService">The local storage service</param>
|
/// <param name="localStorageService">The local storage service</param>
|
||||||
/// <param name="authenticationStateProvider">The authentication state provider</param>
|
/// <param name="authenticationStateProvider">The authentication state provider</param>
|
||||||
|
/// <param name="mailService">The mail service</param>
|
||||||
public AuthService(
|
public AuthService(
|
||||||
FsContext context,
|
FsContext context,
|
||||||
LocalStorageService localStorageService,
|
LocalStorageService localStorageService,
|
||||||
AuthenticationStateProvider authenticationStateProvider)
|
AuthenticationStateProvider authenticationStateProvider,
|
||||||
|
IMailService mailService,
|
||||||
|
IOptions<AppSettings> appSettings)
|
||||||
{
|
{
|
||||||
Context = context;
|
Context = context;
|
||||||
_localStorageService = localStorageService;
|
_localStorageService = localStorageService;
|
||||||
_authenticationStateProvider = authenticationStateProvider;
|
_authenticationStateProvider = authenticationStateProvider;
|
||||||
|
_mailService = mailService;
|
||||||
|
_appSettings = appSettings.Value;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@@ -125,6 +142,12 @@ namespace FoodsharingSiegen.Server.Auth
|
|||||||
|
|
||||||
if (_user != null)
|
if (_user != null)
|
||||||
{
|
{
|
||||||
|
if (_user.Type == UserType.Unverified)
|
||||||
|
{
|
||||||
|
_user = null;
|
||||||
|
return new OperationResult(new Exception("Anmeldung nicht möglich."));
|
||||||
|
}
|
||||||
|
|
||||||
var serializedToken = AuthHelper.CreateToken(_user);
|
var serializedToken = AuthHelper.CreateToken(_user);
|
||||||
await _localStorageService.SetItem(StorageKeys.TokenKey, serializedToken);
|
await _localStorageService.SetItem(StorageKeys.TokenKey, serializedToken);
|
||||||
|
|
||||||
@@ -139,7 +162,7 @@ namespace FoodsharingSiegen.Server.Auth
|
|||||||
return new OperationResult();
|
return new OperationResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
return new OperationResult(new Exception("Benutzername oder Passwort falsch"));
|
return new OperationResult(new Exception("E-Mail-Adresse oder Passwort ist ungültig."));
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@@ -186,5 +209,98 @@ namespace FoodsharingSiegen.Server.Auth
|
|||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
#region Password Recovery
|
||||||
|
|
||||||
|
public async Task InitiateInitialPasswordSetup(string email, string baseUri)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(email)) return;
|
||||||
|
|
||||||
|
var user = await Context.Users!.FirstOrDefaultAsync(x => x.Mail.ToLower() == email.ToLower());
|
||||||
|
if (user == null) return; // Do not leak existence
|
||||||
|
|
||||||
|
var resetToken = Guid.NewGuid().ToString("N");
|
||||||
|
user.ResetToken = resetToken;
|
||||||
|
user.ResetTokenExpiry = DateTime.UtcNow.AddDays(7);
|
||||||
|
|
||||||
|
await Context.SaveChangesAsync();
|
||||||
|
|
||||||
|
var resetLink = $"{baseUri.TrimEnd('/')}/reset-password/{resetToken}";
|
||||||
|
var mailBody = $"""
|
||||||
|
Hallo {user.Name},<br>
|
||||||
|
<br>
|
||||||
|
für dich wurde ein neues Konto bei {_appSettings.Terms.Title} erstellt. <br>
|
||||||
|
<br>
|
||||||
|
Um dein Passwort festzulegen, klicke bitte auf den folgenden Link (dieser ist 7 Tage gültig):<br>
|
||||||
|
<a href='{resetLink}'>{resetLink}</a><br>
|
||||||
|
<br>
|
||||||
|
Viele Grüße<br>Dein Team {_appSettings.Terms.Title}
|
||||||
|
""";
|
||||||
|
|
||||||
|
await _mailService.SendEmailAsync(user.Mail, "Passwort festlegen", mailBody);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task InitiatePasswordReset(string email, string baseUri)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(email)) return;
|
||||||
|
|
||||||
|
var user = await Context.Users!.FirstOrDefaultAsync(x => x.Mail.ToLower() == email.ToLower());
|
||||||
|
if (user == null) return; // Do not leak existence
|
||||||
|
|
||||||
|
var resetToken = Guid.NewGuid().ToString("N");
|
||||||
|
user.ResetToken = resetToken;
|
||||||
|
user.ResetTokenExpiry = DateTime.UtcNow.AddMinutes(30);
|
||||||
|
|
||||||
|
Context.Audits?.Add(new Audit
|
||||||
|
{
|
||||||
|
Created = DateTime.Now,
|
||||||
|
Type = AuditType.RequestPasswordReset,
|
||||||
|
UserID = user.Id,
|
||||||
|
Data1 = user.Mail
|
||||||
|
});
|
||||||
|
|
||||||
|
await Context.SaveChangesAsync();
|
||||||
|
|
||||||
|
var resetLink = $"{baseUri.TrimEnd('/')}/reset-password/{resetToken}";
|
||||||
|
var mailBody = $"""
|
||||||
|
Hallo {user.Name},<br>
|
||||||
|
<br>
|
||||||
|
für dein Konto wurde eine Anfrage zum Zurücksetzen des Passworts gestellt. <br>
|
||||||
|
Wenn du diese Anfrage nicht gestellt hast, kannst du diese E-Mail ignorieren und dein Passwort bleibt unverändert.<br>
|
||||||
|
<br>
|
||||||
|
Um dein Passwort zurückzusetzen, klicke bitte auf den folgenden Link (dieser ist 30 Minuten gültig):<br>
|
||||||
|
<a href='{resetLink}'>{resetLink}</a><br>
|
||||||
|
<br>
|
||||||
|
Viele Grüße<br>Dein Team {_appSettings.Terms.Title}
|
||||||
|
""";
|
||||||
|
|
||||||
|
await _mailService.SendEmailAsync(user.Mail, "Passwort zurücksetzen", mailBody);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<OperationResult> ResetPassword(string token, string newPassword)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(token)) return new OperationResult(new Exception("Ungültiges Token."));
|
||||||
|
if (string.IsNullOrWhiteSpace(newPassword)) return new OperationResult(new Exception("Passwort darf nicht leer sein."));
|
||||||
|
|
||||||
|
var user = await Context.Users!.FirstOrDefaultAsync(x => x.ResetToken == token && x.ResetTokenExpiry > DateTime.UtcNow);
|
||||||
|
if (user == null) return new OperationResult(new Exception("Token ist ungültig oder abgelaufen."));
|
||||||
|
|
||||||
|
user.Password = newPassword;
|
||||||
|
user.ResetToken = null;
|
||||||
|
user.ResetTokenExpiry = null;
|
||||||
|
|
||||||
|
await Context.SaveChangesAsync();
|
||||||
|
|
||||||
|
return new OperationResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<bool> IsResetTokenValid(string token)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(token)) return false;
|
||||||
|
var user = await Context.Users!.FirstOrDefaultAsync(x => x.ResetToken == token && x.ResetTokenExpiry > DateTime.UtcNow);
|
||||||
|
return user != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -71,12 +71,6 @@ namespace FoodsharingSiegen.Server.BaseClasses
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Private Fields
|
|
||||||
|
|
||||||
private bool _dataInitialized;
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Override OnInitializedAsync
|
#region Override OnInitializedAsync
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -85,26 +79,8 @@ namespace FoodsharingSiegen.Server.BaseClasses
|
|||||||
protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
{
|
{
|
||||||
await AuthService.Initialize();
|
await AuthService.Initialize();
|
||||||
await base.OnInitializedAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Override SetParametersAsync
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
public override async Task SetParametersAsync(ParameterView parameters)
|
|
||||||
{
|
|
||||||
parameters.SetParameterProperties(this);
|
|
||||||
|
|
||||||
if (!_dataInitialized)
|
|
||||||
{
|
|
||||||
_dataInitialized = true;
|
|
||||||
await InitializeDataAsync();
|
await InitializeDataAsync();
|
||||||
}
|
await base.OnInitializedAsync();
|
||||||
|
|
||||||
// Da die Parameter bereits gesetzt wurden, kann die Basisklasse am Ende aufgerufen werden.
|
|
||||||
await base.SetParametersAsync(ParameterView.Empty);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|||||||
@@ -5,52 +5,63 @@
|
|||||||
@inherits FsBase
|
@inherits FsBase
|
||||||
|
|
||||||
@{
|
@{
|
||||||
|
var colorClass = "";
|
||||||
|
if(Done) colorClass = "interaction--color-done";
|
||||||
|
|
||||||
var rowClass = "";
|
var rowClass = "";
|
||||||
if (Done) rowClass += " done";
|
if (Done) rowClass += " done";
|
||||||
if (NotNeeded) rowClass += " notneeded";
|
if (NotNeeded) rowClass += " notneeded";
|
||||||
if (Alert) rowClass += " fs-alert";
|
if (Alert) rowClass += " fs-alert";
|
||||||
}
|
}
|
||||||
|
|
||||||
<tr class="@rowClass" style="border-top: 5px solid transparent;">
|
@if(!AllowInteraction || Prospect is {Complete: true })
|
||||||
<th class="text-center align-top pr-2">
|
{
|
||||||
<i class="@IconClass"></i>
|
<Button Size="Size.Small" Disabled="true">
|
||||||
</th>
|
<i class="@ButtonIconClass"></i>
|
||||||
<th class="pr-2 align-top" style="white-space: nowrap;">@Type.Translate(AppSettings):</th>
|
</Button>
|
||||||
<td class="align-top d-flex flex-column">
|
}
|
||||||
@if (Interactions.Count > 0)
|
else
|
||||||
|
{
|
||||||
|
@if(Interactions.Count == 0 || Multiple)
|
||||||
{
|
{
|
||||||
foreach (var interaction in Interactions)
|
if (Multiple) ButtonIconClass = "fa-solid fa-plus";
|
||||||
{
|
<Button Size="Size.Small" Clicked="@(async () => { if (AddClick != null) await AddClick(Type); })"><i class="@ButtonIconClass" style="color: #64ae24;"></i></Button>
|
||||||
<div style="padding-bottom: .5rem;">
|
} else {
|
||||||
<div style="white-space: nowrap;">
|
<Button Size="Size.Small" Clicked="@(async () => { await RemoveFirstAsync(Type); })"><i class="fa-solid fa-times" style="color: rgb(153, 0, 0);"></i></Button>
|
||||||
<span title="@interaction.User.Memo">@interaction.User.Name</span> (@interaction.Date.ToShortDateString())
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
<div style="white-space: nowrap; overflow: hidden; text-overflow: ellipsis">
|
||||||
|
<div class="@colorClass" style="display: inline-block;"><i class="@IconClass"></i> @Type.Translate(AppSettings)</div>
|
||||||
|
|
||||||
|
@if(Interactions.Count > 0)
|
||||||
|
{
|
||||||
|
<span title="@Interactions.First().User.Memo"> (@Interactions.First().User.Name</span> <span>@Interactions.First().Date.ToShortDateString())</span>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if(Multiple && Interactions.Count > 0)
|
||||||
|
{
|
||||||
|
|
||||||
|
@foreach (var interaction in Interactions)
|
||||||
|
{
|
||||||
|
<div class="d-flex justify-content-end">
|
||||||
@if ((Prospect is not { Complete: true } || interaction.Type == InteractionType.Complete) && AllowInteraction)
|
@if ((Prospect is not { Complete: true } || interaction.Type == InteractionType.Complete) && AllowInteraction)
|
||||||
{
|
{
|
||||||
<span> <a href=""><i class="fa-solid fa-square-xmark" @onclick="async () => { if (RemoveClick != null) await RemoveClick.Invoke(interaction.Id); }" @onclick:preventDefault></i></a></span>
|
<a style="display: inline-block;"" href=""><i class="fa-solid fa-square-xmark" @onclick="async () => { if (RemoveClick != null) await RemoveClick.Invoke(interaction.Id); }" @onclick:preventDefault></i></a>
|
||||||
|
} else {
|
||||||
|
<span>•</span>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
@FeedbackBuilder(interaction)
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
|
<span title="@interaction.User.Memo">@interaction.User.Name (@interaction.Date.ToShortDateString())</span>
|
||||||
@if (!string.IsNullOrWhiteSpace(interaction.FeedbackInfo))
|
@if (!string.IsNullOrWhiteSpace(interaction.FeedbackInfo))
|
||||||
{
|
{
|
||||||
<span>(<i>@interaction.FeedbackInfo</i>)</span>
|
<span>(<i>@interaction.FeedbackInfo</i>)</span>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
@FeedbackBuilder(interaction)
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@if (Prospect is not {Complete: true } && (Interactions.Count == 0 || Multiple) && AllowInteraction)
|
|
||||||
{
|
|
||||||
if (Multiple) ButtonIconClass = "fa-solid fa-plus";
|
|
||||||
<div class="m-auto">
|
|
||||||
<Button Size="Size.Small" Clicked="@(async () => { if (AddClick != null) await AddClick(Type); })"><i class="@ButtonIconClass" style="color: #64ae24;"></i></Button>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
@@ -94,6 +94,15 @@ namespace FoodsharingSiegen.Server.Controls
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
private async Task RemoveFirstAsync(InteractionType type)
|
||||||
|
{
|
||||||
|
if (Prospect != null && RemoveClick != null)
|
||||||
|
{
|
||||||
|
var interaction = Interactions.FirstOrDefault(x => x.Type == type);
|
||||||
|
if (interaction != null) await RemoveClick(interaction.Id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private MarkupString FeedbackBuilder(Interaction interaction)
|
private MarkupString FeedbackBuilder(Interaction interaction)
|
||||||
{
|
{
|
||||||
var infoList = new List<string>();
|
var infoList = new List<string>();
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
@using FoodsharingSiegen.Contracts.Enums
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
@using FoodsharingSiegen.Shared.Helper
|
@using FoodsharingSiegen.Shared.Helper
|
||||||
|
@using System.ComponentModel
|
||||||
@inherits FsBase
|
@inherits FsBase
|
||||||
|
|
||||||
@{
|
@{
|
||||||
var divClass = $"{CssClass} pc-main";
|
var divClass = $"{CssClass} pc-main shadow border-0";
|
||||||
if (Prospect is { Complete: true }) divClass += " complete";
|
if (Prospect is { Complete: true }) divClass += " complete";
|
||||||
if (Prospect is { Warning: true }) divClass += " warning";
|
if (Prospect is { Warning: true }) divClass += " warning";
|
||||||
if (Prospect is { RecordState: RecordState.Archived }) divClass += " deleted";
|
if (Prospect is { RecordState: RecordState.Archived }) divClass += " deleted";
|
||||||
@@ -11,28 +12,32 @@
|
|||||||
|
|
||||||
<div class="@divClass">
|
<div class="@divClass">
|
||||||
<h5 class="mb-2 d-flex">
|
<h5 class="mb-2 d-flex">
|
||||||
<div class="flex-grow-1">
|
<div class="flex-grow-1 d-flex">
|
||||||
@Prospect?.Name
|
|
||||||
<small style="font-size: .9rem;">@Prospect?.FsId</small>
|
|
||||||
</div>
|
|
||||||
<div>
|
<div>
|
||||||
@if (CurrentUser.IsInGroup(UserGroup.WelcomeTeam, UserGroup.Ambassador))
|
@if(string.IsNullOrWhiteSpace(Prospect?.Name))
|
||||||
{
|
{
|
||||||
<i class="fa-solid fa-pen-to-square link mr-2" @onclick="EditProspectAsync" @onclick:preventDefault></i>
|
<i class="fa-solid fa-exclamation-triangle text-warning"></i>
|
||||||
}
|
<doublearrows></doublearrows>
|
||||||
<a href="@(CurrentUser.NetworkLink)/profile/@Prospect?.FsId" target="_blank"><i class="fa-solid fa-eye"></i></a>
|
|
||||||
@if (CurrentUser.IsInGroup(UserGroup.WelcomeTeam, UserGroup.Ambassador))
|
<em>»Name fehlt«</em>
|
||||||
{
|
|
||||||
if (Prospect?.RecordState != RecordState.Archived)
|
|
||||||
{
|
|
||||||
<i class="fa-solid fa-box-archive link ml-2" @onclick="ArchiveProspectAsync" title="Archivieren" @onclick:preventDefault></i>
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
<i class="fa-solid fa-recycle link ml-2" @onclick="RestoreProspectAsync" title="Wiederherstellen" @onclick:preventDefault></i>
|
@Prospect?.Name
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div style="flex-grow: 1;"></div>
|
||||||
|
|
||||||
|
@if (Prospect?.FsId != null && Prospect.FsId != 0)
|
||||||
|
{
|
||||||
|
<a href="@(CurrentUser.NetworkLink)/profile/@Prospect?.FsId" target="_blank">
|
||||||
|
<small style="font-size: .9rem;">
|
||||||
|
<i class="fa-solid fa-eye"></i> @Prospect?.FsId
|
||||||
|
</small>
|
||||||
|
</a>
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
|
||||||
</div>
|
</div>
|
||||||
</h5>
|
</h5>
|
||||||
|
|
||||||
@@ -46,8 +51,7 @@
|
|||||||
</Alert>
|
</Alert>
|
||||||
}
|
}
|
||||||
|
|
||||||
<table class="flex-column" style="width: 100%;">
|
<div class="interaction-grid mb-3">
|
||||||
|
|
||||||
<InteractionRow
|
<InteractionRow
|
||||||
Prospect="Prospect"
|
Prospect="Prospect"
|
||||||
Type="InteractionType.Welcome"
|
Type="InteractionType.Welcome"
|
||||||
@@ -83,12 +87,6 @@
|
|||||||
IconClass="fa-solid fa-basket-shopping">
|
IconClass="fa-solid fa-basket-shopping">
|
||||||
</InteractionRow>
|
</InteractionRow>
|
||||||
|
|
||||||
<tr>
|
|
||||||
<td colspan="3">
|
|
||||||
<hr style="margin: 10px 0;">
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
|
|
||||||
<InteractionRow
|
<InteractionRow
|
||||||
Prospect="Prospect"
|
Prospect="Prospect"
|
||||||
Type="InteractionType.ReleasedForVerification"
|
Type="InteractionType.ReleasedForVerification"
|
||||||
@@ -133,27 +131,91 @@
|
|||||||
ButtonIconClass="fa-solid fa-check"
|
ButtonIconClass="fa-solid fa-check"
|
||||||
IconClass="fa-solid fa-user-check">
|
IconClass="fa-solid fa-user-check">
|
||||||
</InteractionRow>
|
</InteractionRow>
|
||||||
|
|
||||||
<tr>
|
|
||||||
<td colspan="3">
|
|
||||||
<hr style="margin: 10px 0;">
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
|
|
||||||
<InteractionRow
|
|
||||||
Prospect="Prospect"
|
|
||||||
AllowInteraction="@CurrentUser.IsInGroup(UserGroup.Ambassador)"
|
|
||||||
Type="InteractionType.Complete"
|
|
||||||
AddClick="AddInteraction"
|
|
||||||
RemoveClick="@RemoveInteraction"
|
|
||||||
ButtonIconClass="fa-solid fa-check"
|
|
||||||
IconClass="fa-solid fa-flag-checkered">
|
|
||||||
</InteractionRow>
|
|
||||||
}
|
}
|
||||||
|
</div>
|
||||||
</table>
|
|
||||||
|
|
||||||
<div class="flex-grow-1"></div>
|
<div class="flex-grow-1"></div>
|
||||||
|
|
||||||
<small class="text-center" style="margin-top: .5rem;">Zuletzt geändert: @Prospect?.Modified?.ToLocalTime()</small>
|
@if(Prospect?.Images?.Count > 0)
|
||||||
|
{
|
||||||
|
<div class="text-center mt-3">
|
||||||
|
<Badge Color="Color.Info" Style="margin-bottom: 0.5rem; cursor: pointer;" @onclick="OpenVerificationDialogAsync"><i class="fa-solid fa-address-card"></i>-Prüfung möglich</Badge></div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<div class="text-center d-flex justify-content-center gap-2 mt-1">
|
||||||
|
|
||||||
|
<Button
|
||||||
|
Color="Color.Secondary"
|
||||||
|
Height="Height.Px(35)"
|
||||||
|
title="Bearbeiten"
|
||||||
|
Clicked="EditProspectAsync"
|
||||||
|
Visibility="@(CurrentUser.IsInGroup(UserGroup.WelcomeTeam, UserGroup.Ambassador) ? Visibility.Default : Visibility.Invisible)"
|
||||||
|
><i class="fa-solid fa-pen-to-square"></i>
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
|
||||||
|
@if(StateFilter > ProspectStateFilter.OnBoarding && CurrentUser.IsInGroup(UserGroup.Ambassador))
|
||||||
|
{
|
||||||
|
@if(Prospect?.Complete != true)
|
||||||
|
{
|
||||||
|
<Button
|
||||||
|
Color="Color.Primary"
|
||||||
|
Height="Height.Px(35)"
|
||||||
|
title="Fertigstellen"
|
||||||
|
Clicked="@(() => AddInteraction(InteractionType.Complete))"
|
||||||
|
Visibility="@(CurrentUser.IsInGroup(UserGroup.Ambassador) ? Visibility.Default : Visibility.Invisible)"
|
||||||
|
><i class="fa-solid fa-flag-checkered"></i>
|
||||||
|
</Button>
|
||||||
|
} else {
|
||||||
|
<Button
|
||||||
|
Color="Color.Primary"
|
||||||
|
Height="Height.Px(35)"
|
||||||
|
title="Fertigstellen rückgängig"
|
||||||
|
Clicked="@(() => AddInteraction(InteractionType.Complete))"
|
||||||
|
Visibility="@(CurrentUser.IsInGroup(UserGroup.Ambassador) ? Visibility.Default : Visibility.Invisible)"
|
||||||
|
|
||||||
|
>
|
||||||
|
<span class="fa-stack" style="vertical-align: top;">
|
||||||
|
<i class="fa-solid fa-slash fa-xl"></i>
|
||||||
|
<i class="fa-solid fa-flag fa-stack-1x"></i>
|
||||||
|
</span>
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
<Button
|
||||||
|
Color="Color.Info"
|
||||||
|
Height="Height.Px(35)"
|
||||||
|
title="Identitätsprüfung"
|
||||||
|
Clicked="OpenVerificationDialogAsync"
|
||||||
|
Visibility="@(CurrentUser.IsInGroup(UserGroup.WelcomeTeam, UserGroup.Ambassador) ? Visibility.Default : Visibility.Invisible)"
|
||||||
|
><i class="fa-solid fa-address-card"></i>
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
@if (Prospect?.RecordState != RecordState.Archived)
|
||||||
|
{
|
||||||
|
<Button
|
||||||
|
Color="Color.Danger"
|
||||||
|
Height="Height.Px(35)"
|
||||||
|
title="Archivieren"
|
||||||
|
Clicked="ArchiveProspectAsync"
|
||||||
|
Visibility="@(CurrentUser.IsInGroup(UserGroup.WelcomeTeam, UserGroup.Ambassador) ? Visibility.Default : Visibility.Invisible)"
|
||||||
|
><i class="fa-solid fa-box-archive"></i>
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<Button
|
||||||
|
Color="Color.Success"
|
||||||
|
Height="Height.Px(35)"
|
||||||
|
title="Wiederherstellen"
|
||||||
|
Clicked="RestoreProspectAsync"
|
||||||
|
Visibility="@(CurrentUser.IsInGroup(UserGroup.WelcomeTeam, UserGroup.Ambassador) ? Visibility.Default : Visibility.Invisible)"
|
||||||
|
><i class="fa-solid fa-recycle"></i>
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<small class="text-center" style="margin-top: .5rem;">Geändert: @Prospect?.Modified?.ToLocalTime()</small>
|
||||||
</div>
|
</div>
|
||||||
@@ -43,7 +43,20 @@ namespace FoodsharingSiegen.Server.Controls
|
|||||||
{
|
{
|
||||||
var headerText = $"{type.Translate(AppSettings)} für {Prospect.Name} eintragen";
|
var headerText = $"{type.Translate(AppSettings)} für {Prospect.Name} eintragen";
|
||||||
|
|
||||||
await InteractionDialog.ShowAsync(ModalService, new(type, Prospect.Id, headerText, OnDataChanged));
|
Func<Task> onSuccess = async () =>
|
||||||
|
{
|
||||||
|
if (type == InteractionType.IdCheck && Prospect.Images != null && Prospect.Images.Count > 0)
|
||||||
|
{
|
||||||
|
await ConfirmDialog.ShowAsync(ModalService, "Personalausweisbilder löschen?", $"Möchtest du die Personalausweisbilder von {Prospect.Name} löschen? Diese werden für die weitere Bearbeitung nicht mehr benötigt und enthalten persönliche Daten.", async () =>
|
||||||
|
{
|
||||||
|
var result = await ProspectService.DeleteVerificationImagesAsync(Prospect.Id);
|
||||||
|
await OnDataChanged();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
await OnDataChanged();
|
||||||
|
};
|
||||||
|
|
||||||
|
await InteractionDialog.ShowAsync(ModalService, new(type, Prospect.Id, headerText, onSuccess));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -87,6 +100,16 @@ namespace FoodsharingSiegen.Server.Controls
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
#region Private Method OpenVerificationDialogAsync
|
||||||
|
|
||||||
|
private async Task OpenVerificationDialogAsync()
|
||||||
|
{
|
||||||
|
if (Prospect == null) return;
|
||||||
|
await VerificationSettingsDialog.ShowAsync(ModalService, Prospect, OnDataChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
#region Private Method RemoveInteraction
|
#region Private Method RemoveInteraction
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -106,6 +129,27 @@ namespace FoodsharingSiegen.Server.Controls
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task RemoveInteraction(InteractionType type)
|
||||||
|
{
|
||||||
|
var typeName = type.Translate(AppSettings);
|
||||||
|
await ConfirmDialog.ShowAsync(ModalService, "Bestätigen", $"Alle {typeName}-Interaktionen wirklich entfernen?", async () =>
|
||||||
|
{
|
||||||
|
var interactions = Prospect?.Interactions.Where(x => x.Type == type);
|
||||||
|
|
||||||
|
var dataChanged = false;
|
||||||
|
|
||||||
|
foreach (var interaction in interactions ?? [])
|
||||||
|
{
|
||||||
|
var removeR = await ProspectService.RemoveInteraction(interaction.Id);
|
||||||
|
if (!removeR.Success) continue;
|
||||||
|
|
||||||
|
dataChanged = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(dataChanged && OnDataChanged != null) await OnDataChanged();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Private Method RestoreProspectAsync
|
#region Private Method RestoreProspectAsync
|
||||||
|
|||||||
@@ -1,50 +0,0 @@
|
|||||||
::deep a,
|
|
||||||
::deep a.invert:hover{
|
|
||||||
|
|
||||||
color: #64ae24;
|
|
||||||
}
|
|
||||||
|
|
||||||
::deep a.invert,
|
|
||||||
::deep a:hover {
|
|
||||||
color: #533a20;
|
|
||||||
}
|
|
||||||
|
|
||||||
.green {
|
|
||||||
color: #64ae24;
|
|
||||||
}
|
|
||||||
|
|
||||||
.pc-main {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
flex-basis: 0;
|
|
||||||
flex-grow: 1;
|
|
||||||
max-width: 480px;
|
|
||||||
border: 1px solid #533a20;
|
|
||||||
border-radius: 3px;
|
|
||||||
margin: 5px;
|
|
||||||
padding: .5rem .5rem 0 .5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.pc-main.warning {
|
|
||||||
-webkit-box-shadow: 0 0 9px 4px rgba(214,100,23,0.87);
|
|
||||||
-moz-box-shadow: 0 0 9px 4px rgba(214,100,23,0.87);
|
|
||||||
box-shadow: 0 0 9px 4px rgba(214,100,23,0.87);
|
|
||||||
}
|
|
||||||
|
|
||||||
.pc-main.deleted {
|
|
||||||
-webkit-box-shadow: 0 0 9px 4px rgb(214 23 23 / 87%);
|
|
||||||
-moz-box-shadow: 0 0 9px 4px rgb(214 23 23 / 87%);
|
|
||||||
box-shadow: 0 0 9px 4px rgb(214 23 23 / 87%);
|
|
||||||
}
|
|
||||||
|
|
||||||
.complete {
|
|
||||||
background: #76ff003b;
|
|
||||||
}
|
|
||||||
|
|
||||||
i.link {
|
|
||||||
cursor: pointer; color: #64ae24;
|
|
||||||
}
|
|
||||||
|
|
||||||
i.link:hover {
|
|
||||||
color: #000;
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,69 @@
|
|||||||
|
::deep a {
|
||||||
|
color: #64ae24;
|
||||||
|
display: block;
|
||||||
|
|
||||||
|
&.invert,
|
||||||
|
&:hover {
|
||||||
|
color: #533a20;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.invert:hover {
|
||||||
|
color: #64ae24;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.green {
|
||||||
|
color: #64ae24;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pc-main {
|
||||||
|
background-color: #FFF;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
max-width: none;
|
||||||
|
border-radius: 12px;
|
||||||
|
margin: 0;
|
||||||
|
padding: 1rem 1rem 0 1rem;
|
||||||
|
|
||||||
|
@media (max-width: 576px) {
|
||||||
|
padding: .5rem .5rem 0 .5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.warning {
|
||||||
|
box-shadow: rgb(255 145 0 / 36%) 0px 8px 10px 1px, rgb(255 145 0 / 44%) 0px 3px 14px 2px, rgb(255 145 0 / 49%) 0px 5px 5px -3px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.deleted {
|
||||||
|
box-shadow: rgba(255, 0, 0, 0.36) 0px 8px 10px 1px, rgba(255, 0, 0, 0.44) 0px 3px 14px 2px, rgba(255, 0, 0, 0.49) 0px 5px 5px -3px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.complete {
|
||||||
|
background: #76ff003b;
|
||||||
|
}
|
||||||
|
|
||||||
|
i.link {
|
||||||
|
cursor: pointer;
|
||||||
|
color: #64ae24;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.interaction-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: auto 1fr;
|
||||||
|
align-items: center;
|
||||||
|
gap: .5rem;
|
||||||
|
|
||||||
|
::deep .interaction {
|
||||||
|
&--color {
|
||||||
|
&-done{
|
||||||
|
color: #64ae24;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,102 +0,0 @@
|
|||||||
@using FoodsharingSiegen.Contracts.Enums
|
|
||||||
@using FoodsharingSiegen.Contracts.Model
|
|
||||||
@inherits FsBase
|
|
||||||
@code {
|
|
||||||
|
|
||||||
[Parameter] public ProspectFilter Filter { get; set; } = new();
|
|
||||||
|
|
||||||
[Parameter] public EventCallback<ProspectFilter> FilterChanged { get; set; }
|
|
||||||
|
|
||||||
[Parameter] public ProspectStateFilter StateFilter { get; set; }
|
|
||||||
|
|
||||||
|
|
||||||
private async Task WithoutStepInBriefingChangedAsync(bool arg)
|
|
||||||
{
|
|
||||||
Filter.WithoutStepInBriefing = arg;
|
|
||||||
await FilterChanged.InvokeAsync(Filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task TextChanged(string arg)
|
|
||||||
{
|
|
||||||
Filter.Text = arg;
|
|
||||||
await FilterChanged.InvokeAsync(Filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task WithoutIdCheckChangedAsync(bool arg)
|
|
||||||
{
|
|
||||||
Filter.WithoutIdCheck = arg;
|
|
||||||
await FilterChanged.InvokeAsync(Filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task NoActivityChangedAsync(bool arg)
|
|
||||||
{
|
|
||||||
Filter.NoActivity = arg;
|
|
||||||
await FilterChanged.InvokeAsync(Filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task DeletedOnlyChangedAsync(bool arg)
|
|
||||||
{
|
|
||||||
Filter.DeletedOnly = arg;
|
|
||||||
await FilterChanged.InvokeAsync(Filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task RecentActivityChangedAsync(bool arg)
|
|
||||||
{
|
|
||||||
Filter.RecentActivity = arg;
|
|
||||||
await FilterChanged.InvokeAsync(Filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
<div class="card">
|
|
||||||
<div class="card-header" style="padding: .5rem;">
|
|
||||||
<i class="fa-solid fa-filter"></i> Suchfilter
|
|
||||||
</div>
|
|
||||||
<div class="card-body" style="padding: .5rem;">
|
|
||||||
|
|
||||||
@if (!AppSettings.DisableStepIn)
|
|
||||||
{
|
|
||||||
@* WITHOUT STEP IN BRIEFING *@
|
|
||||||
@if (new[] { ProspectStateFilter.All, ProspectStateFilter.OnBoarding, ProspectStateFilter.Completed }.Contains(StateFilter))
|
|
||||||
{
|
|
||||||
<div style="margin-left: 1rem;">
|
|
||||||
<Switch TValue="bool" Checked="Filter.WithoutStepInBriefing" CheckedChanged="WithoutStepInBriefingChangedAsync" Color="Color.Primary">Ohne @AppSettings.Terms.StepInName</Switch>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@* WITHOUT ID CHECK *@
|
|
||||||
@if (new[] { ProspectStateFilter.All, ProspectStateFilter.Verification, ProspectStateFilter.Completed }.Contains(StateFilter))
|
|
||||||
{
|
|
||||||
<div style="margin-left: 1rem;">
|
|
||||||
<Switch TValue="bool" Checked="Filter.WithoutIdCheck" CheckedChanged="WithoutIdCheckChangedAsync" Color="Color.Primary">Perso noch nicht geprüft</Switch>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
|
|
||||||
@* RECENT ACTIVITY *@
|
|
||||||
@if (new[] { ProspectStateFilter.All, ProspectStateFilter.OnBoarding, ProspectStateFilter.Verification }.Contains(StateFilter))
|
|
||||||
{
|
|
||||||
<div style="margin-left: 1rem;">
|
|
||||||
<Switch TValue="bool" Checked="Filter.RecentActivity" CheckedChanged="RecentActivityChangedAsync" Color="Color.Primary">Kürzlich geändert (< 6 Monate)</Switch>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
|
|
||||||
@* NO ACTIVITY *@
|
|
||||||
@if (new[] { ProspectStateFilter.All, ProspectStateFilter.OnBoarding, ProspectStateFilter.Verification }.Contains(StateFilter))
|
|
||||||
{
|
|
||||||
<div style="margin-left: 1rem;">
|
|
||||||
<Switch TValue="bool" Checked="Filter.NoActivity" CheckedChanged="NoActivityChangedAsync" Color="Color.Primary">Lange keine Aktivität (> 6 Monate)</Switch>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
|
|
||||||
@* DELETED ONLY *@
|
|
||||||
@if (new[] { ProspectStateFilter.All }.Contains(StateFilter))
|
|
||||||
{
|
|
||||||
<div style="margin-left: 1rem;">
|
|
||||||
<Switch TValue="bool" Checked="Filter.DeletedOnly" CheckedChanged="DeletedOnlyChangedAsync" Color="Color.Primary">Gelöschte</Switch>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
|
|
||||||
<TextEdit Text="@Filter.Text" TextChanged="TextChanged" Placeholder="Suchen..." Debounce="true" DebounceInterval="200"/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@@ -7,12 +7,14 @@
|
|||||||
|
|
||||||
[Parameter] public Func<Task>? OnDataChanged { get; set; }
|
[Parameter] public Func<Task>? OnDataChanged { get; set; }
|
||||||
|
|
||||||
|
[Parameter] public string GridClass { get; set; } = string.Empty;
|
||||||
|
|
||||||
}
|
}
|
||||||
<h5>@(Prospects?.Count ?? 0) Einträge</h5>
|
<h6>@(Prospects?.Count ?? 0) Ergebnisse</h6>
|
||||||
|
|
||||||
@if (Prospects?.Any() == true)
|
@if (Prospects?.Any() == true)
|
||||||
{
|
{
|
||||||
<div class="row m-0">
|
<div class="prospect-grid @GridClass">
|
||||||
<Repeater Items="@Prospects">
|
<Repeater Items="@Prospects">
|
||||||
<ProspectContainer
|
<ProspectContainer
|
||||||
Prospect="context"
|
Prospect="context"
|
||||||
|
|||||||
8
FoodsharingSiegen.Server/Controls/ProspectGrid.razor.css
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
.prospect-grid {
|
||||||
|
display: grid;
|
||||||
|
gap: 1rem;
|
||||||
|
width: 100%;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(320px, 1fr));
|
||||||
|
justify-content: center;
|
||||||
|
align-items: stretch;
|
||||||
|
}
|
||||||
56
FoodsharingSiegen.Server/Controls/ProspectSortControl.razor
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
|
@inherits FoodsharingSiegen.Server.BaseClasses.FsBase
|
||||||
|
|
||||||
|
<Button Color="Color.Primary"
|
||||||
|
Width="Width.Px(50)"
|
||||||
|
Height="Height.Px(50)"
|
||||||
|
title="Sortieren"
|
||||||
|
style="min-width: auto;"
|
||||||
|
Clicked="@OpenSortDialogAsync">
|
||||||
|
<i class="fa-solid fa-sort"></i>
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<Button Color="Color.Info"
|
||||||
|
Width="Width.Px(50)"
|
||||||
|
Height="Height.Px(50)"
|
||||||
|
title="Filtern"
|
||||||
|
style="min-width: auto;"
|
||||||
|
Clicked="@OpenFilterDialogAsync">
|
||||||
|
<i class="fa-solid fa-filter"></i>
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<div style="flex-grow: 1;" class="mt-3">
|
||||||
|
<TextEdit Text="@Filter.Text" TextChanged="TextChangedAsync" Placeholder="Suchen..." Debounce="true" DebounceInterval="200" Style="border: 1px solid #64ae24; background: #fff;" Class="pl-2" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="badge-row mt-1 mb-3">
|
||||||
|
@if (HasCustomSort)
|
||||||
|
{
|
||||||
|
<Badge class="mr-1" Color="Color.Primary" Closable="true" CloseClicked="@EventCallback.Factory.Create(this, ResetSortAsync)">@CurrentSortText</Badge>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (Filter.WithoutStepInBriefing)
|
||||||
|
{
|
||||||
|
<Badge class="mr-1 mb-1" Color="Color.Info" Closable="true" CloseClicked="@EventCallback.Factory.Create(this, () => DisableFilterAsync(nameof(Filter.WithoutStepInBriefing)))">Ohne @AppSettings.Terms.StepInName</Badge>
|
||||||
|
}
|
||||||
|
@if (Filter.WithoutIdCheck)
|
||||||
|
{
|
||||||
|
<Badge class="mr-1 mb-1" Color="Color.Info" Closable="true" CloseClicked="@EventCallback.Factory.Create(this, () => DisableFilterAsync(nameof(Filter.WithoutIdCheck)))">Perso noch nicht geprüft</Badge>
|
||||||
|
}
|
||||||
|
@if (Filter.RecentActivity)
|
||||||
|
{
|
||||||
|
<Badge class="mr-1 mb-1" Color="Color.Info" Closable="true" CloseClicked="@EventCallback.Factory.Create(this, () => DisableFilterAsync(nameof(Filter.RecentActivity)))">Kürzlich geändert</Badge>
|
||||||
|
}
|
||||||
|
@if (Filter.NoActivity)
|
||||||
|
{
|
||||||
|
<Badge class="mr-1 mb-1" Color="Color.Info" Closable="true" CloseClicked="@EventCallback.Factory.Create(this, () => DisableFilterAsync(nameof(Filter.NoActivity)))">Lange keine Aktivität</Badge>
|
||||||
|
}
|
||||||
|
@if (Filter.DeletedOnly)
|
||||||
|
{
|
||||||
|
<Badge class="mr-1 mb-1" Color="Color.Info" Closable="true" CloseClicked="@EventCallback.Factory.Create(this, () => DisableFilterAsync(nameof(Filter.DeletedOnly)))">Gelöschte</Badge>
|
||||||
|
}
|
||||||
|
@if(Filter.IdCheckPossible)
|
||||||
|
{
|
||||||
|
<Badge class="mr-1 mb-1" Color="Color.Info" Closable="true" CloseClicked="@EventCallback.Factory.Create(this, () => DisableFilterAsync(nameof(Filter.IdCheckPossible)))">Perso-Prüfung möglich</Badge>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
109
FoodsharingSiegen.Server/Controls/ProspectSortControl.razor.cs
Normal file
@@ -0,0 +1,109 @@
|
|||||||
|
using Blazorise;
|
||||||
|
using FoodsharingSiegen.Contracts.Entity;
|
||||||
|
using FoodsharingSiegen.Contracts.Enums;
|
||||||
|
using FoodsharingSiegen.Contracts.Model;
|
||||||
|
using FoodsharingSiegen.Server.Data.Service;
|
||||||
|
using FoodsharingSiegen.Server.Dialogs;
|
||||||
|
using FoodsharingSiegen.Server.Service;
|
||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Controls;
|
||||||
|
|
||||||
|
public partial class ProspectSortControl
|
||||||
|
{
|
||||||
|
[Parameter]
|
||||||
|
public ProspectSortOption CurrentSort { get; set; } = ProspectSortOption.NameAscending;
|
||||||
|
|
||||||
|
[Parameter]
|
||||||
|
public EventCallback<ProspectSortOption> CurrentSortChanged { get; set; }
|
||||||
|
|
||||||
|
[Parameter]
|
||||||
|
public EventCallback OnSortChanged { get; set; }
|
||||||
|
|
||||||
|
[Parameter]
|
||||||
|
public string? StorageKey { get; set; }
|
||||||
|
|
||||||
|
[Parameter] public ProspectFilter Filter { get; set; } = new();
|
||||||
|
[Parameter] public EventCallback<ProspectFilter> FilterChanged { get; set; }
|
||||||
|
[Parameter] public ProspectStateFilter StateFilter { get; set; } = ProspectStateFilter.All;
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
if (!string.IsNullOrEmpty(StorageKey))
|
||||||
|
{
|
||||||
|
var savedSort = await LocalStorageService.GetItem<ProspectSortOption?>(StorageKey);
|
||||||
|
if (savedSort.HasValue)
|
||||||
|
{
|
||||||
|
CurrentSort = savedSort.Value;
|
||||||
|
await CurrentSortChanged.InvokeAsync(CurrentSort);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
await base.OnInitializedAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task OpenSortDialogAsync()
|
||||||
|
{
|
||||||
|
await ProspectSortDialog.ShowAsync(ModalService, CurrentSort, async option =>
|
||||||
|
{
|
||||||
|
await UpdateSortAsync(option);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task OpenFilterDialogAsync()
|
||||||
|
{
|
||||||
|
await ProspectFilterDialog.ShowAsync(ModalService, Filter, StateFilter, async (f) =>
|
||||||
|
{
|
||||||
|
Filter = f;
|
||||||
|
await FilterChanged.InvokeAsync(Filter);
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task TextChangedAsync(string text)
|
||||||
|
{
|
||||||
|
Filter.Text = text;
|
||||||
|
await FilterChanged.InvokeAsync(Filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task DisableFilterAsync(string filterPropName)
|
||||||
|
{
|
||||||
|
switch (filterPropName)
|
||||||
|
{
|
||||||
|
case nameof(Filter.WithoutStepInBriefing): Filter.WithoutStepInBriefing = false; break;
|
||||||
|
case nameof(Filter.WithoutIdCheck): Filter.WithoutIdCheck = false; break;
|
||||||
|
case nameof(Filter.RecentActivity): Filter.RecentActivity = false; break;
|
||||||
|
case nameof(Filter.NoActivity): Filter.NoActivity = false; break;
|
||||||
|
case nameof(Filter.DeletedOnly): Filter.DeletedOnly = false; break;
|
||||||
|
case nameof(Filter.IdCheckPossible): Filter.IdCheckPossible = false; break;
|
||||||
|
}
|
||||||
|
await FilterChanged.InvokeAsync(Filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool HasCustomSort => CurrentSort != ProspectSortOption.NameAscending;
|
||||||
|
|
||||||
|
private string CurrentSortText => CurrentSort switch
|
||||||
|
{
|
||||||
|
ProspectSortOption.NameDescending => "Sortierung: Name (absteigend)",
|
||||||
|
ProspectSortOption.ModifiedAscending => "Sortierung: Zuletzt geändert (aufsteigend)",
|
||||||
|
ProspectSortOption.ModifiedDescending => "Sortierung: Zuletzt geaendert (absteigend)",
|
||||||
|
_ => string.Empty
|
||||||
|
};
|
||||||
|
|
||||||
|
private async Task ResetSortAsync()
|
||||||
|
{
|
||||||
|
await UpdateSortAsync(ProspectSortOption.NameAscending);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task UpdateSortAsync(ProspectSortOption option)
|
||||||
|
{
|
||||||
|
CurrentSort = option;
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(StorageKey))
|
||||||
|
{
|
||||||
|
await LocalStorageService.SetItem(StorageKey, CurrentSort);
|
||||||
|
}
|
||||||
|
|
||||||
|
await CurrentSortChanged.InvokeAsync(CurrentSort);
|
||||||
|
await OnSortChanged.InvokeAsync();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -37,6 +37,16 @@ namespace FoodsharingSiegen.Server.Data
|
|||||||
return $"hat dem Neuling {audit.Data1} folgendes hinzugefügt: {audit.Data2}";
|
return $"hat dem Neuling {audit.Data1} folgendes hinzugefügt: {audit.Data2}";
|
||||||
case AuditType.RemoveInteraction:
|
case AuditType.RemoveInteraction:
|
||||||
return $"hat eine Interaktion bei {audit.Data1} gelöscht.";
|
return $"hat eine Interaktion bei {audit.Data1} gelöscht.";
|
||||||
|
case AuditType.DeleteProspectImages:
|
||||||
|
return $"hat die Bilder von {audit.Data1} gelöscht.";
|
||||||
|
case AuditType.ViewProspectImages:
|
||||||
|
return $"hat die Bilder von {audit.Data1} angesehen.";
|
||||||
|
case AuditType.UploadProspectImage:
|
||||||
|
return $"hat ein Bild für {audit.Data1} hochgeladen.";
|
||||||
|
case AuditType.RequestPasswordReset:
|
||||||
|
return $"hat ein Passwort-Reset für {audit.Data1} angefordert.";
|
||||||
|
case AuditType.ChangeOwnPassword:
|
||||||
|
return $"hat das eigene Passwort geändert.";
|
||||||
case AuditType.None:
|
case AuditType.None:
|
||||||
default:
|
default:
|
||||||
return $"{audit.Data1}, {audit.Data2}";
|
return $"{audit.Data1}, {audit.Data2}";
|
||||||
|
|||||||
@@ -27,6 +27,11 @@ namespace FoodsharingSiegen.Server.Data
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public DbSet<Prospect>? Prospects { get; set; }
|
public DbSet<Prospect>? Prospects { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the uploaded verification images mapping.
|
||||||
|
/// </summary>
|
||||||
|
public DbSet<ProspectImage>? ProspectImages { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the users (ab)
|
/// Gets or sets the value of the users (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -61,29 +61,56 @@ namespace FoodsharingSiegen.Server.Data.Service
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Public Method Load
|
#region Public Method GetCount
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Loads the count (a. beging, 23.05.2022)
|
/// Gets the total count (ab)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="count">The count</param>
|
|
||||||
/// <param name="type">The type</param>
|
/// <param name="type">The type</param>
|
||||||
/// <returns>A task containing an operation result of list audit</returns>
|
/// <returns>A task containing an operation result of count</returns>
|
||||||
public async Task<OperationResult<List<Audit>>> Load(int count, AuditType? type = null)
|
public async Task<OperationResult<int>> GetCount(AuditType? type = null)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await Task.CompletedTask;
|
var query = Context.Audits?.AsQueryable();
|
||||||
|
|
||||||
var query = Context.Audits?.Include(x => x.User).OrderByDescending(x => x.Created).AsQueryable();
|
|
||||||
|
|
||||||
if (count > 0)
|
|
||||||
query = query?.Take(count);
|
|
||||||
|
|
||||||
if (type != null)
|
if (type != null)
|
||||||
query = query?.Where(x => x.Type == type);
|
query = query?.Where(x => x.Type == type);
|
||||||
|
|
||||||
var mat = query?.ToList();
|
if (query == null) return new(0);
|
||||||
|
|
||||||
|
var count = await query.CountAsync();
|
||||||
|
return new(count);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
return new(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region Public Method LoadPage
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Loads the page of audits (ab)
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="skip">The skip count</param>
|
||||||
|
/// <param name="take">The take count</param>
|
||||||
|
/// <param name="type">The type</param>
|
||||||
|
/// <returns>A task containing an operation result of list audit</returns>
|
||||||
|
public async Task<OperationResult<List<Audit>>> LoadPage(int skip, int take, AuditType? type = null)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var query = Context.Audits?.Include(x => x.User).OrderByDescending(x => x.Created).AsQueryable();
|
||||||
|
|
||||||
|
if (type != null)
|
||||||
|
query = query?.Where(x => x.Type == type);
|
||||||
|
|
||||||
|
query = query?.Skip(skip).Take(take);
|
||||||
|
|
||||||
|
var mat = await query!.ToListAsync();
|
||||||
|
|
||||||
if (mat != null) return new(mat);
|
if (mat != null) return new(mat);
|
||||||
|
|
||||||
|
|||||||
@@ -111,7 +111,10 @@ namespace FoodsharingSiegen.Server.Data.Service
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var prospectsQuery = Context.Prospects!.AsNoTracking().Include(x => x.Interactions.OrderBy(i => i.Date)).ThenInclude(x => x.User).OrderBy(x => x.Name).AsQueryable();
|
var prospectsQuery = Context.Prospects!.AsNoTracking()
|
||||||
|
.Include(x => x.Images)
|
||||||
|
.Include(x => x.Interactions.OrderBy(i => i.Date)).ThenInclude(x => x.User)
|
||||||
|
.OrderBy(x => x.Name).AsQueryable();
|
||||||
|
|
||||||
if(parameter.MustHaveInteractions != null && parameter.MustHaveInteractions.Any())
|
if(parameter.MustHaveInteractions != null && parameter.MustHaveInteractions.Any())
|
||||||
prospectsQuery = prospectsQuery.Where(x => x.Interactions.Any(i => parameter.MustHaveInteractions.Contains(i.Type)));
|
prospectsQuery = prospectsQuery.Where(x => x.Interactions.Any(i => parameter.MustHaveInteractions.Contains(i.Type)));
|
||||||
@@ -206,5 +209,152 @@ namespace FoodsharingSiegen.Server.Data.Service
|
|||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
#region Image Upload Features
|
||||||
|
|
||||||
|
public async Task<OperationResult<Guid>> GenerateVerificationTokenAsync(Guid prospectId)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var prospect = await Context.Prospects!
|
||||||
|
.Include(x => x.Interactions)
|
||||||
|
.FirstOrDefaultAsync(x => x.Id == prospectId);
|
||||||
|
|
||||||
|
if (prospect == null) return new(new Exception("Prospect not found"));
|
||||||
|
|
||||||
|
if (prospect.Interactions.Any(x => x.Type == InteractionType.Verify))
|
||||||
|
return new(new Exception("Die Identitätsprüfung wurde bereits abgeschlossen. Es kann kein neuer Token generiert werden."));
|
||||||
|
|
||||||
|
prospect.VerificationToken = Guid.NewGuid();
|
||||||
|
prospect.Modified = DateTime.UtcNow;
|
||||||
|
|
||||||
|
await Context.SaveChangesAsync();
|
||||||
|
return new(prospect.VerificationToken.Value);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
return new(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<OperationResult<Prospect>> GetProspectByVerificationTokenAsync(Guid token)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var prospect = await Context.Prospects!
|
||||||
|
.Include(x => x.Interactions)
|
||||||
|
.Include(x => x.Images)
|
||||||
|
.AsNoTracking()
|
||||||
|
.FirstOrDefaultAsync(x => x.VerificationToken == token);
|
||||||
|
|
||||||
|
if (prospect == null) return new(new Exception("Ungültiger oder abgelaufener Token."));
|
||||||
|
|
||||||
|
if (prospect.Interactions.Any(x => x.Type == InteractionType.Verify))
|
||||||
|
return new(new Exception("Die Identitätsprüfung wurde bereits abgeschlossen. Ein Hochladen weiterer Bilder ist nicht mehr möglich."));
|
||||||
|
|
||||||
|
return new(prospect);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
return new(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<OperationResult> AddVerificationImageAsync(Guid prospectId, byte[] imageData, string contentType)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var prospect = await Context.Prospects!
|
||||||
|
.Include(x => x.Interactions)
|
||||||
|
.AsNoTracking()
|
||||||
|
.FirstOrDefaultAsync(x => x.Id == prospectId);
|
||||||
|
|
||||||
|
if (prospect == null) return new(new Exception("Foodsaver nicht gefunden."));
|
||||||
|
|
||||||
|
if (prospect.Interactions.Any(x => x.Type == InteractionType.Verify))
|
||||||
|
return new(new Exception("Die Identitätsprüfung wurde bereits abgeschlossen. Ein Hochladen weiterer Bilder ist nicht mehr möglich."));
|
||||||
|
|
||||||
|
// Verify max 5 images
|
||||||
|
var imgCount = await Context.ProspectImages!.CountAsync(x => x.ProspectId == prospectId);
|
||||||
|
if (imgCount >= 5) return new(new Exception("Maximum 5 images allowed"));
|
||||||
|
|
||||||
|
var image = new ProspectImage
|
||||||
|
{
|
||||||
|
Id = Guid.NewGuid(),
|
||||||
|
ProspectId = prospectId,
|
||||||
|
ImageData = imageData,
|
||||||
|
ContentType = contentType,
|
||||||
|
Created = DateTime.UtcNow
|
||||||
|
};
|
||||||
|
|
||||||
|
await Context.ProspectImages!.AddAsync(image);
|
||||||
|
await Context.SaveChangesAsync();
|
||||||
|
|
||||||
|
await AuditService.Insert(AuditType.UploadProspectImage, prospect.Name);
|
||||||
|
|
||||||
|
return new();
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
return new(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<OperationResult<List<ProspectImage>>> GetVerificationImagesAsync(Guid prospectId)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var images = await Context.ProspectImages!
|
||||||
|
.AsNoTracking()
|
||||||
|
.Where(x => x.ProspectId == prospectId)
|
||||||
|
.OrderBy(x => x.Created)
|
||||||
|
.ToListAsync();
|
||||||
|
|
||||||
|
var prospectName = await Context.Prospects!
|
||||||
|
.Where(x => x.Id == prospectId)
|
||||||
|
.Select(x => x.Name)
|
||||||
|
.FirstOrDefaultAsync();
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(prospectName))
|
||||||
|
{
|
||||||
|
await AuditService.Insert(AuditType.ViewProspectImages, prospectName);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new(images);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
return new(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<OperationResult> DeleteVerificationImagesAsync(Guid prospectId)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var images = await Context.ProspectImages!.Where(x => x.ProspectId == prospectId).ToListAsync();
|
||||||
|
if (images.Any())
|
||||||
|
{
|
||||||
|
Context.ProspectImages!.RemoveRange(images);
|
||||||
|
|
||||||
|
var prospect = await Context.Prospects!.FirstOrDefaultAsync(x => x.Id == prospectId);
|
||||||
|
if (prospect != null)
|
||||||
|
{
|
||||||
|
prospect.VerificationToken = null; // Clear token when images are deleted
|
||||||
|
await AuditService.Insert(AuditType.DeleteProspectImages, prospect.Name);
|
||||||
|
}
|
||||||
|
|
||||||
|
await Context.SaveChangesAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
return new();
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
return new(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -105,6 +105,13 @@ namespace FoodsharingSiegen.Server.Data.Service
|
|||||||
var user = await Context.Users!.Include(x => x.Interactions).FirstOrDefaultAsync(x => x.Id == userId);
|
var user = await Context.Users!.Include(x => x.Interactions).FirstOrDefaultAsync(x => x.Id == userId);
|
||||||
if (user == null) return new(new Exception("User not found"));
|
if (user == null) return new(new Exception("User not found"));
|
||||||
|
|
||||||
|
if (user.Type == UserType.Admin)
|
||||||
|
{
|
||||||
|
var adminCount = await Context.Users!.CountAsync(x => x.Type == UserType.Admin && x.Id != userId);
|
||||||
|
if (adminCount == 0)
|
||||||
|
return new(new Exception("Der letzte Administrator kann nicht gelöscht werden."));
|
||||||
|
}
|
||||||
|
|
||||||
// Interaktionen vom aktuellen Nutzer übernehmen
|
// Interaktionen vom aktuellen Nutzer übernehmen
|
||||||
if(CurrentUser?.Id != null)
|
if(CurrentUser?.Id != null)
|
||||||
foreach (var userInteraction in user.Interactions)
|
foreach (var userInteraction in user.Interactions)
|
||||||
@@ -151,8 +158,14 @@ namespace FoodsharingSiegen.Server.Data.Service
|
|||||||
|
|
||||||
if (saveR < 1) return new(new Exception("Fehler beim Speichern"));
|
if (saveR < 1) return new(new Exception("Fehler beim Speichern"));
|
||||||
|
|
||||||
var auditData = CurrentUser?.Id == user.Id ? "sich selbst" : user.Mail;
|
if (CurrentUser?.Id == user.Id)
|
||||||
await AuditService.Insert(AuditType.SetUserPassword, auditData);
|
{
|
||||||
|
await AuditService.Insert(AuditType.ChangeOwnPassword);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await AuditService.Insert(AuditType.SetUserPassword, user.Mail);
|
||||||
|
}
|
||||||
|
|
||||||
return new();
|
return new();
|
||||||
}
|
}
|
||||||
@@ -178,8 +191,14 @@ namespace FoodsharingSiegen.Server.Data.Service
|
|||||||
var entityUser = await Context.Users!.FirstOrDefaultAsync(x => x.Id == user.Id);
|
var entityUser = await Context.Users!.FirstOrDefaultAsync(x => x.Id == user.Id);
|
||||||
if (entityUser == null) return new(new Exception("User not found"));
|
if (entityUser == null) return new(new Exception("User not found"));
|
||||||
|
|
||||||
|
if (entityUser.Type == UserType.Admin && user.Type != UserType.Admin)
|
||||||
|
{
|
||||||
|
var adminCount = await Context.Users!.CountAsync(x => x.Type == UserType.Admin && x.Id != user.Id);
|
||||||
|
if (adminCount == 0)
|
||||||
|
return new(new Exception("Der Typ des letzten Administrators kann nicht geändert werden."));
|
||||||
|
}
|
||||||
|
|
||||||
if (entityUser.Mail != user.Mail ||
|
if (entityUser.Mail != user.Mail ||
|
||||||
entityUser.Verified != user.Verified ||
|
|
||||||
entityUser.Type != user.Type ||
|
entityUser.Type != user.Type ||
|
||||||
entityUser.Groups != user.Groups)
|
entityUser.Groups != user.Groups)
|
||||||
entityUser.ForceLogout = true;
|
entityUser.ForceLogout = true;
|
||||||
@@ -188,7 +207,6 @@ namespace FoodsharingSiegen.Server.Data.Service
|
|||||||
entityUser.Mail = user.Mail;
|
entityUser.Mail = user.Mail;
|
||||||
entityUser.Name = user.Name;
|
entityUser.Name = user.Name;
|
||||||
entityUser.Type = user.Type;
|
entityUser.Type = user.Type;
|
||||||
entityUser.Verified = user.Verified;
|
|
||||||
entityUser.Groups = user.Groups;
|
entityUser.Groups = user.Groups;
|
||||||
entityUser.Network = user.Network;
|
entityUser.Network = user.Network;
|
||||||
|
|
||||||
|
|||||||
@@ -127,9 +127,10 @@ namespace FoodsharingSiegen.Server.Dialogs
|
|||||||
Interaction.UserID = CurrentUser.Id;
|
Interaction.UserID = CurrentUser.Id;
|
||||||
|
|
||||||
var addR = await ProspectService.AddInteraction(Interaction);
|
var addR = await ProspectService.AddInteraction(Interaction);
|
||||||
if (addR.Success && OnSuccess != null) await OnSuccess.Invoke();
|
|
||||||
|
|
||||||
await ModalService.Hide();
|
await ModalService.Hide();
|
||||||
|
|
||||||
|
if (addR.Success && OnSuccess != null) await OnSuccess.Invoke();
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|||||||
58
FoodsharingSiegen.Server/Dialogs/ProspectFilterDialog.razor
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
@inherits FsBase
|
||||||
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
|
|
||||||
|
<div class="d-grid gap-2">
|
||||||
|
@if (!AppSettings.DisableStepIn && new[] { ProspectStateFilter.All, ProspectStateFilter.OnBoarding, ProspectStateFilter.Completed }.Contains(StateFilter))
|
||||||
|
{
|
||||||
|
<div style="margin-left: 1rem;">
|
||||||
|
<Switch TValue="bool" Checked="ModalFilter.WithoutStepInBriefing" CheckedChanged="(v) => { ModalFilter.WithoutStepInBriefing = v; StateHasChanged(); }" Color="Color.Primary">
|
||||||
|
Ohne @AppSettings.Terms.StepInName
|
||||||
|
</Switch>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (new[] { ProspectStateFilter.All, ProspectStateFilter.Verification, ProspectStateFilter.Completed }.Contains(StateFilter))
|
||||||
|
{
|
||||||
|
<div style="margin-left: 1rem;">
|
||||||
|
<Switch TValue="bool" Checked="ModalFilter.WithoutIdCheck" CheckedChanged="(v) => { ModalFilter.WithoutIdCheck = v; StateHasChanged(); }" Color="Color.Primary">
|
||||||
|
Perso noch nicht geprüft
|
||||||
|
</Switch>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if(new[] { ProspectStateFilter.OnBoarding, ProspectStateFilter.Verification }.Contains(StateFilter))
|
||||||
|
{
|
||||||
|
<div style="margin-left: 1rem;">
|
||||||
|
<Switch TValue="bool" Checked="ModalFilter.IdCheckPossible" CheckedChanged="(v) => { ModalFilter.IdCheckPossible = v; StateHasChanged(); }" Color="Color.Primary">
|
||||||
|
Perso-Prüfung möglich
|
||||||
|
</Switch>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (new[] { ProspectStateFilter.All, ProspectStateFilter.OnBoarding, ProspectStateFilter.Verification }.Contains(StateFilter))
|
||||||
|
{
|
||||||
|
<div style="margin-left: 1rem;">
|
||||||
|
<Switch TValue="bool" Checked="ModalFilter.RecentActivity" CheckedChanged="(v) => { ModalFilter.RecentActivity = v; StateHasChanged(); }" Color="Color.Primary">
|
||||||
|
Kürzlich geändert (< 6 Monate)
|
||||||
|
</Switch>
|
||||||
|
</div>
|
||||||
|
<div style="margin-left: 1rem;">
|
||||||
|
<Switch TValue="bool" Checked="ModalFilter.NoActivity" CheckedChanged="(v) => { ModalFilter.NoActivity = v; StateHasChanged(); }" Color="Color.Primary">
|
||||||
|
Lange keine Aktivität (> 6 Monate)
|
||||||
|
</Switch>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (StateFilter == ProspectStateFilter.All)
|
||||||
|
{
|
||||||
|
<div style="margin-left: 1rem;">
|
||||||
|
<Switch TValue="bool" Checked="ModalFilter.DeletedOnly" CheckedChanged="(v) => { ModalFilter.DeletedOnly = v; StateHasChanged(); }" Color="Color.Primary">
|
||||||
|
Gelöschte
|
||||||
|
</Switch>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<div class="d-flex justify-content-end mt-3">
|
||||||
|
<Button Color="Color.Primary" Clicked="ApplyAsync" Block="true">Anwenden</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
using Blazorise;
|
||||||
|
using FoodsharingSiegen.Contracts.Enums;
|
||||||
|
using FoodsharingSiegen.Contracts.Model;
|
||||||
|
using FoodsharingSiegen.Server.BaseClasses;
|
||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Dialogs
|
||||||
|
{
|
||||||
|
public partial class ProspectFilterDialog : FsBase
|
||||||
|
{
|
||||||
|
[Parameter] public ProspectFilter CurrentFilter { get; set; } = new();
|
||||||
|
[Parameter] public ProspectStateFilter StateFilter { get; set; } = ProspectStateFilter.All;
|
||||||
|
[Parameter] public Func<ProspectFilter, Task>? OnFilterApplied { get; set; }
|
||||||
|
|
||||||
|
public ProspectFilter ModalFilter { get; set; } = new();
|
||||||
|
|
||||||
|
protected override void OnInitialized()
|
||||||
|
{
|
||||||
|
// Clone the filter so changes are not immediately mapped to the parent until "Anwenden" is pressed
|
||||||
|
ModalFilter = new ProspectFilter
|
||||||
|
{
|
||||||
|
Text = CurrentFilter.Text,
|
||||||
|
WithoutStepInBriefing = CurrentFilter.WithoutStepInBriefing,
|
||||||
|
WithoutIdCheck = CurrentFilter.WithoutIdCheck,
|
||||||
|
NoActivity = CurrentFilter.NoActivity,
|
||||||
|
RecentActivity = CurrentFilter.RecentActivity,
|
||||||
|
DeletedOnly = CurrentFilter.DeletedOnly,
|
||||||
|
IdCheckPossible = CurrentFilter.IdCheckPossible
|
||||||
|
};
|
||||||
|
base.OnInitialized();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task ShowAsync(IModalService modalService, ProspectFilter currentFilter, ProspectStateFilter stateFilter, Func<ProspectFilter, Task> onFilterApplied)
|
||||||
|
{
|
||||||
|
await modalService.Show<ProspectFilterDialog>("Filtern", p =>
|
||||||
|
{
|
||||||
|
p.Add(nameof(CurrentFilter), currentFilter);
|
||||||
|
p.Add(nameof(StateFilter), stateFilter);
|
||||||
|
p.Add(nameof(OnFilterApplied), onFilterApplied);
|
||||||
|
}, new ModalInstanceOptions
|
||||||
|
{
|
||||||
|
Size = ModalSize.Small,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task ApplyAsync()
|
||||||
|
{
|
||||||
|
if (OnFilterApplied != null) await OnFilterApplied(ModalFilter);
|
||||||
|
await ModalService.Hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
@inherits FsBase
|
||||||
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
|
|
||||||
|
<div class="d-grid gap-2">
|
||||||
|
<Button Color="@GetSortButtonColor(ProspectSortOption.NameAscending)" Clicked="() => SelectAsync(ProspectSortOption.NameAscending)">Name (aufsteigend)</Button>
|
||||||
|
<Button Color="@GetSortButtonColor(ProspectSortOption.NameDescending)" Clicked="() => SelectAsync(ProspectSortOption.NameDescending)">Name (absteigend)</Button>
|
||||||
|
<Button Color="@GetSortButtonColor(ProspectSortOption.ModifiedAscending)" Clicked="() => SelectAsync(ProspectSortOption.ModifiedAscending)">Geändert (aufsteigend)</Button>
|
||||||
|
<Button Color="@GetSortButtonColor(ProspectSortOption.ModifiedDescending)" Clicked="() => SelectAsync(ProspectSortOption.ModifiedDescending)">Geändert (absteigend)</Button>
|
||||||
|
</div>
|
||||||
39
FoodsharingSiegen.Server/Dialogs/ProspectSortDialog.razor.cs
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
using Blazorise;
|
||||||
|
using FoodsharingSiegen.Contracts.Enums;
|
||||||
|
using FoodsharingSiegen.Server.BaseClasses;
|
||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Dialogs
|
||||||
|
{
|
||||||
|
public partial class ProspectSortDialog : FsBase
|
||||||
|
{
|
||||||
|
[Parameter]
|
||||||
|
public ProspectSortOption CurrentSort { get; set; } = ProspectSortOption.NameAscending;
|
||||||
|
|
||||||
|
[Parameter]
|
||||||
|
public Func<ProspectSortOption, Task>? OnSortSelected { get; set; }
|
||||||
|
|
||||||
|
public static async Task ShowAsync(IModalService modalService, ProspectSortOption currentSort, Func<ProspectSortOption, Task> onSortSelected)
|
||||||
|
{
|
||||||
|
await modalService.Show<ProspectSortDialog>("Sortieren", p =>
|
||||||
|
{
|
||||||
|
p.Add(nameof(CurrentSort), currentSort);
|
||||||
|
p.Add(nameof(OnSortSelected), onSortSelected);
|
||||||
|
}, new ModalInstanceOptions
|
||||||
|
{
|
||||||
|
Size = ModalSize.Small,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private Color GetSortButtonColor(ProspectSortOption option)
|
||||||
|
{
|
||||||
|
return CurrentSort == option ? Color.Success : Color.Secondary;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task SelectAsync(ProspectSortOption option)
|
||||||
|
{
|
||||||
|
if (OnSortSelected != null) await OnSortSelected(option);
|
||||||
|
await ModalService.Hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
@using Blazorise
|
||||||
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
|
@inherits FsBase
|
||||||
|
|
||||||
|
<div class="mt-1 mb-3">
|
||||||
|
<div class="d-grid gap-3">
|
||||||
|
|
||||||
|
@if (ShowLinkPanel)
|
||||||
|
{
|
||||||
|
<div class="border p-3 rounded">
|
||||||
|
<p class="mb-2 text-muted">Kopiere diesen Link und teile ihn mit <strong>@Prospect?.Name</strong>:</p>
|
||||||
|
<div>
|
||||||
|
<input type="text" class="form-control" value="@LinkUrl" readonly />
|
||||||
|
@if(CopySuccess)
|
||||||
|
{
|
||||||
|
<div class="alert alert-success py-2 mt-2 mb-0">Link wurde in die Zwischenablage kopiert!</div>
|
||||||
|
}
|
||||||
|
<Button Color="Color.Secondary" Clicked="CopyLink" Style="width: 100%;" Class="mt-2">
|
||||||
|
<i class="fa-solid fa-copy mr-2"></i>Link kopieren
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<Button Color="Color.Info" Clicked="@GenerateLinkAsync">
|
||||||
|
<i class="fa-solid fa-link me-2"></i> Upload-Link erstellen / anzeigen
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
@if(CurrentUser.IsAdmin() || CurrentUser.IsInGroup(UserGroup.Ambassador))
|
||||||
|
{
|
||||||
|
<Button Color="Color.Success" Clicked="@ViewImagesAsync" Disabled="@(ImageCount == 0)">
|
||||||
|
<i class="fa-solid fa-images me-2"></i> Hochgeladene Bilder ansehen (@ImageCount)
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
<Button Color="Color.Danger" Clicked="@DeleteImagesAsync" Disabled="@(ImageCount == 0)">
|
||||||
|
<i class="fa-solid fa-trash-can me-2"></i> Alle Bilder löschen
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
@@ -0,0 +1,110 @@
|
|||||||
|
using Blazorise;
|
||||||
|
using FoodsharingSiegen.Contracts.Entity;
|
||||||
|
using FoodsharingSiegen.Server.BaseClasses;
|
||||||
|
using FoodsharingSiegen.Server.Data.Service;
|
||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
using Microsoft.JSInterop;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Dialogs
|
||||||
|
{
|
||||||
|
public partial class VerificationSettingsDialog : FsBase
|
||||||
|
{
|
||||||
|
[Inject]
|
||||||
|
public ProspectService ProspectService { get; set; } = null!;
|
||||||
|
|
||||||
|
[Inject]
|
||||||
|
public IJSRuntime JS { get; set; } = null!;
|
||||||
|
|
||||||
|
[Parameter]
|
||||||
|
public Prospect? Prospect { get; set; }
|
||||||
|
|
||||||
|
[Parameter]
|
||||||
|
public Func<Task>? OnDataChanged { get; set; }
|
||||||
|
|
||||||
|
private int ImageCount { get; set; } = 0;
|
||||||
|
private bool ShowLinkPanel { get; set; } = false;
|
||||||
|
|
||||||
|
private bool CopySuccess { get; set; } = false;
|
||||||
|
private string LinkUrl { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
if (Prospect != null)
|
||||||
|
{
|
||||||
|
var result = await ProspectService.GetVerificationImagesAsync(Prospect.Id);
|
||||||
|
if (result.Success && result.Data != null)
|
||||||
|
{
|
||||||
|
ImageCount = result.Data.Count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task ShowAsync(IModalService modalService, Prospect? prospect, Func<Task>? onDataChanged)
|
||||||
|
{
|
||||||
|
var title = "Identitätsprüfung";
|
||||||
|
var action = new Action<ModalProviderParameterBuilder<VerificationSettingsDialog>>(b =>
|
||||||
|
{
|
||||||
|
b.Add(nameof(Prospect), prospect);
|
||||||
|
b.Add(nameof(OnDataChanged), onDataChanged);
|
||||||
|
});
|
||||||
|
|
||||||
|
await modalService.Show(title, action, new ModalInstanceOptions { Size = ModalSize.Default });
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task GenerateLinkAsync()
|
||||||
|
{
|
||||||
|
if (Prospect == null) return;
|
||||||
|
|
||||||
|
Guid token = Prospect.VerificationToken ?? Guid.Empty;
|
||||||
|
|
||||||
|
if (token == Guid.Empty)
|
||||||
|
{
|
||||||
|
var result = await ProspectService.GenerateVerificationTokenAsync(Prospect.Id);
|
||||||
|
if (result.Success)
|
||||||
|
{
|
||||||
|
token = result.Data;
|
||||||
|
if (OnDataChanged != null) await OnDataChanged();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await Notification.Error(result.Exception?.Message ?? "Ein Fehler ist aufgetreten.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LinkUrl = NavigationManager.BaseUri + "verify/" + token.ToString();
|
||||||
|
ShowLinkPanel = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task CopyLink()
|
||||||
|
{
|
||||||
|
await JS.InvokeVoidAsync("navigator.clipboard.writeText", LinkUrl);
|
||||||
|
CopySuccess = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task ViewImagesAsync()
|
||||||
|
{
|
||||||
|
await ModalService.Hide();
|
||||||
|
if (Prospect != null)
|
||||||
|
{
|
||||||
|
await ViewImagesDialog.ShowAsync(ModalService, Prospect);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task DeleteImagesAsync()
|
||||||
|
{
|
||||||
|
if (Prospect == null) return;
|
||||||
|
|
||||||
|
await ConfirmDialog.ShowAsync(ModalService, "Bilder Löschen", $"Sollen alle Identitätsprüfungsbilder von {Prospect.Name} unwiderruflich gelöscht werden?", async () =>
|
||||||
|
{
|
||||||
|
var result = await ProspectService.DeleteVerificationImagesAsync(Prospect.Id);
|
||||||
|
if (result.Success)
|
||||||
|
{
|
||||||
|
ImageCount = 0;
|
||||||
|
|
||||||
|
if(OnDataChanged != null) await OnDataChanged();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
48
FoodsharingSiegen.Server/Dialogs/ViewImagesDialog.razor
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
@using Blazorise
|
||||||
|
@inherits FsBase
|
||||||
|
|
||||||
|
@if (SelectedImageIndex.HasValue)
|
||||||
|
{
|
||||||
|
<div class="text-center position-relative">
|
||||||
|
<Button Color="Color.Secondary" Class="position-absolute start-0 top-0 m-2 z-3 text-white bg-dark border-0 rounded-circle w-40px h-40px fs-4 lh-1" Clicked="() => SelectedImageIndex = null">
|
||||||
|
<i class="fa-solid fa-close"></i>
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<a href="@(_images[SelectedImageIndex.Value])" target="_blank" download>
|
||||||
|
<img src="@(_images[SelectedImageIndex.Value])" class="img-fluid rounded" style="max-height: 80vh;" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<div class="d-flex justify-content-between position-absolute top-50 start-0 w-100">
|
||||||
|
<Button Color="Color.Dark" Class="rounded-circle" Disabled="@(SelectedImageIndex.Value == 0)" Clicked="() => SelectedImageIndex--"><i class="fa-solid fa-chevron-left"></i></Button>
|
||||||
|
<Button Color="Color.Dark" Class="rounded-circle" Disabled="@(SelectedImageIndex.Value == _images.Count - 1)" Clicked="() => SelectedImageIndex++"><i class="fa-solid fa-chevron-right"></i></Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
@if (_isLoading)
|
||||||
|
{
|
||||||
|
<div class="text-center my-3"><div class="spinner-border text-success"></div></div>
|
||||||
|
}
|
||||||
|
else if (_images.Any())
|
||||||
|
{
|
||||||
|
<div class="row row-cols-1 row-cols-md-3 g-2">
|
||||||
|
@for (int i = 0; i < _images.Count; i++)
|
||||||
|
{
|
||||||
|
var index = i;
|
||||||
|
<div class="col">
|
||||||
|
<div class="card h-100 shadow-sm" style="cursor:pointer;" @onclick="() => SelectedImageIndex = index">
|
||||||
|
<img src="@(_images[index])" class="card-img-top mh-100 object-fit-cover" style="height: 200px" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="alert alert-info">Keine Bilder vorhanden.</div>
|
||||||
|
}
|
||||||
|
<div class="mt-3 text-end">
|
||||||
|
<Button Color="Color.Secondary" Clicked="@(() => ModalService.Hide())">Schließen</Button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
50
FoodsharingSiegen.Server/Dialogs/ViewImagesDialog.razor.cs
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
using Blazorise;
|
||||||
|
using FoodsharingSiegen.Contracts.Entity;
|
||||||
|
using FoodsharingSiegen.Server.BaseClasses;
|
||||||
|
using FoodsharingSiegen.Server.Data.Service;
|
||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Dialogs
|
||||||
|
{
|
||||||
|
public partial class ViewImagesDialog : FsBase
|
||||||
|
{
|
||||||
|
[Inject]
|
||||||
|
public ProspectService ProspectService { get; set; } = null!;
|
||||||
|
|
||||||
|
[Parameter]
|
||||||
|
public Prospect? Prospect { get; set; }
|
||||||
|
|
||||||
|
private bool _isLoading = true;
|
||||||
|
private List<string> _images = new();
|
||||||
|
private int? SelectedImageIndex { get; set; }
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
if (Prospect != null)
|
||||||
|
{
|
||||||
|
var result = await ProspectService.GetVerificationImagesAsync(Prospect.Id);
|
||||||
|
if (result.Success && result.Data != null)
|
||||||
|
{
|
||||||
|
foreach (var image in result.Data)
|
||||||
|
{
|
||||||
|
var base64 = Convert.ToBase64String(image.ImageData);
|
||||||
|
var imgSrc = $"data:{image.ContentType};base64,{base64}";
|
||||||
|
_images.Add(imgSrc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_isLoading = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task ShowAsync(IModalService modalService, Prospect prospect)
|
||||||
|
{
|
||||||
|
var title = $"Bilder für {prospect.Name}";
|
||||||
|
var action = new Action<ModalProviderParameterBuilder<ViewImagesDialog>>(b =>
|
||||||
|
{
|
||||||
|
b.Add(nameof(Prospect), prospect);
|
||||||
|
});
|
||||||
|
|
||||||
|
await modalService.Show(title, action, new ModalInstanceOptions { Size = ModalSize.ExtraLarge });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -22,13 +22,14 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<LibSassOutputStyle>expanded</LibSassOutputStyle>
|
<DartSassOutputStyle>expanded</DartSassOutputStyle>
|
||||||
<LibSassOutputLevel>verbose</LibSassOutputLevel>
|
<DartSassOutputLevel>verbose</DartSassOutputLevel>
|
||||||
<LibSassMessageLevel>High</LibSassMessageLevel>
|
<DartSassMessageLevel>High</DartSassMessageLevel>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="LibSassBuilder" Version="2.0.1" />
|
<PackageReference Include="AspNetCore.SassCompiler" Version="1.81.1" />
|
||||||
|
<PackageReference Include="MailKit" Version="4.16.0" />
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="9.0.3" />
|
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="9.0.3" />
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="9.0.3">
|
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="9.0.3">
|
||||||
<PrivateAssets>all</PrivateAssets>
|
<PrivateAssets>all</PrivateAssets>
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ namespace FoodsharingSiegen.Server.Migrations
|
|||||||
{
|
{
|
||||||
[DbContext(typeof(FsContext))]
|
[DbContext(typeof(FsContext))]
|
||||||
[Migration("20220521155432_init")]
|
[Migration("20220521155432_init")]
|
||||||
partial class init
|
partial class Init
|
||||||
{
|
{
|
||||||
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ using Microsoft.EntityFrameworkCore.Migrations;
|
|||||||
|
|
||||||
namespace FoodsharingSiegen.Server.Migrations
|
namespace FoodsharingSiegen.Server.Migrations
|
||||||
{
|
{
|
||||||
public partial class init : Migration
|
public partial class Init : Migration
|
||||||
{
|
{
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
{
|
{
|
||||||
|
|||||||
223
FoodsharingSiegen.Server/Migrations/20260418111848_PasswordRecovery.Designer.cs
generated
Normal file
@@ -0,0 +1,223 @@
|
|||||||
|
// <auto-generated />
|
||||||
|
using System;
|
||||||
|
using FoodsharingSiegen.Server.Data;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using Microsoft.EntityFrameworkCore.Infrastructure;
|
||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Migrations
|
||||||
|
{
|
||||||
|
[DbContext(typeof(FsContext))]
|
||||||
|
[Migration("20260418111848_PasswordRecovery")]
|
||||||
|
partial class PasswordRecovery
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
||||||
|
{
|
||||||
|
#pragma warning disable 612, 618
|
||||||
|
modelBuilder.HasAnnotation("ProductVersion", "9.0.3");
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Audit", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Data1")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Data2")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid?>("UserID")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.HasIndex("UserID");
|
||||||
|
|
||||||
|
b.ToTable("Audits");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Interaction", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("Alert")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Date")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Feedback")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("FeedbackInfo")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Info1")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Info2")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("NotNeeded")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid>("ProspectID")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid>("UserID")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.HasIndex("ProspectID");
|
||||||
|
|
||||||
|
b.HasIndex("UserID");
|
||||||
|
|
||||||
|
b.ToTable("Interactions");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Prospect", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("FsId")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("Memo")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime?>("Modified")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Name")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("RecordState")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<bool>("Warning")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.ToTable("Prospects");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.User", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("EncryptedPassword")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("ForceLogout")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("Groups")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Mail")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Memo")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Name")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Network")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("ResetToken")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime?>("ResetTokenExpiry")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<bool>("Verified")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.ToTable("Users");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Audit", b =>
|
||||||
|
{
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.User", "User")
|
||||||
|
.WithMany()
|
||||||
|
.HasForeignKey("UserID");
|
||||||
|
|
||||||
|
b.Navigation("User");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Interaction", b =>
|
||||||
|
{
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.Prospect", "Prospect")
|
||||||
|
.WithMany("Interactions")
|
||||||
|
.HasForeignKey("ProspectID")
|
||||||
|
.OnDelete(DeleteBehavior.Cascade)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.User", "User")
|
||||||
|
.WithMany("Interactions")
|
||||||
|
.HasForeignKey("UserID")
|
||||||
|
.OnDelete(DeleteBehavior.Cascade)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
b.Navigation("Prospect");
|
||||||
|
|
||||||
|
b.Navigation("User");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Prospect", b =>
|
||||||
|
{
|
||||||
|
b.Navigation("Interactions");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.User", b =>
|
||||||
|
{
|
||||||
|
b.Navigation("Interactions");
|
||||||
|
});
|
||||||
|
#pragma warning restore 612, 618
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
using System;
|
||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Migrations
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
public partial class PasswordRecovery : Migration
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<string>(
|
||||||
|
name: "ResetToken",
|
||||||
|
table: "Users",
|
||||||
|
type: "TEXT",
|
||||||
|
nullable: true);
|
||||||
|
|
||||||
|
migrationBuilder.AddColumn<DateTime>(
|
||||||
|
name: "ResetTokenExpiry",
|
||||||
|
table: "Users",
|
||||||
|
type: "TEXT",
|
||||||
|
nullable: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "ResetToken",
|
||||||
|
table: "Users");
|
||||||
|
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "ResetTokenExpiry",
|
||||||
|
table: "Users");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
267
FoodsharingSiegen.Server/Migrations/20260420122950_AddImageVerification.Designer.cs
generated
Normal file
@@ -0,0 +1,267 @@
|
|||||||
|
// <auto-generated />
|
||||||
|
using System;
|
||||||
|
using FoodsharingSiegen.Server.Data;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using Microsoft.EntityFrameworkCore.Infrastructure;
|
||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Migrations
|
||||||
|
{
|
||||||
|
[DbContext(typeof(FsContext))]
|
||||||
|
[Migration("20260420122950_AddImageVerification")]
|
||||||
|
partial class AddImageVerification
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
||||||
|
{
|
||||||
|
#pragma warning disable 612, 618
|
||||||
|
modelBuilder.HasAnnotation("ProductVersion", "9.0.3");
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Audit", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Data1")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Data2")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid?>("UserID")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.HasIndex("UserID");
|
||||||
|
|
||||||
|
b.ToTable("Audits");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Interaction", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("Alert")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Date")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Feedback")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("FeedbackInfo")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Info1")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Info2")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("NotNeeded")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid>("ProspectID")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid>("UserID")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.HasIndex("ProspectID");
|
||||||
|
|
||||||
|
b.HasIndex("UserID");
|
||||||
|
|
||||||
|
b.ToTable("Interactions");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Prospect", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("FsId")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("Memo")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime?>("Modified")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Name")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("RecordState")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid?>("VerificationToken")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("Warning")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.ToTable("Prospects");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.ProspectImage", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("ContentType")
|
||||||
|
.IsRequired()
|
||||||
|
.HasMaxLength(100)
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<byte[]>("ImageData")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("BLOB");
|
||||||
|
|
||||||
|
b.Property<Guid>("ProspectId")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.HasIndex("ProspectId");
|
||||||
|
|
||||||
|
b.ToTable("ProspectImages");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.User", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("EncryptedPassword")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("ForceLogout")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("Groups")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Mail")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Memo")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Name")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Network")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("ResetToken")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime?>("ResetTokenExpiry")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<bool>("Verified")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.ToTable("Users");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Audit", b =>
|
||||||
|
{
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.User", "User")
|
||||||
|
.WithMany()
|
||||||
|
.HasForeignKey("UserID");
|
||||||
|
|
||||||
|
b.Navigation("User");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Interaction", b =>
|
||||||
|
{
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.Prospect", "Prospect")
|
||||||
|
.WithMany("Interactions")
|
||||||
|
.HasForeignKey("ProspectID")
|
||||||
|
.OnDelete(DeleteBehavior.Cascade)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.User", "User")
|
||||||
|
.WithMany("Interactions")
|
||||||
|
.HasForeignKey("UserID")
|
||||||
|
.OnDelete(DeleteBehavior.Cascade)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
b.Navigation("Prospect");
|
||||||
|
|
||||||
|
b.Navigation("User");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.ProspectImage", b =>
|
||||||
|
{
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.Prospect", "Prospect")
|
||||||
|
.WithMany("Images")
|
||||||
|
.HasForeignKey("ProspectId")
|
||||||
|
.OnDelete(DeleteBehavior.Cascade)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
b.Navigation("Prospect");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Prospect", b =>
|
||||||
|
{
|
||||||
|
b.Navigation("Images");
|
||||||
|
|
||||||
|
b.Navigation("Interactions");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.User", b =>
|
||||||
|
{
|
||||||
|
b.Navigation("Interactions");
|
||||||
|
});
|
||||||
|
#pragma warning restore 612, 618
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,58 @@
|
|||||||
|
using System;
|
||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Migrations
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
public partial class AddImageVerification : Migration
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<Guid>(
|
||||||
|
name: "VerificationToken",
|
||||||
|
table: "Prospects",
|
||||||
|
type: "TEXT",
|
||||||
|
nullable: true);
|
||||||
|
|
||||||
|
migrationBuilder.CreateTable(
|
||||||
|
name: "ProspectImages",
|
||||||
|
columns: table => new
|
||||||
|
{
|
||||||
|
Id = table.Column<Guid>(type: "TEXT", nullable: false),
|
||||||
|
ProspectId = table.Column<Guid>(type: "TEXT", nullable: false),
|
||||||
|
ImageData = table.Column<byte[]>(type: "BLOB", nullable: false),
|
||||||
|
ContentType = table.Column<string>(type: "TEXT", maxLength: 100, nullable: false),
|
||||||
|
Created = table.Column<DateTime>(type: "TEXT", nullable: false)
|
||||||
|
},
|
||||||
|
constraints: table =>
|
||||||
|
{
|
||||||
|
table.PrimaryKey("PK_ProspectImages", x => x.Id);
|
||||||
|
table.ForeignKey(
|
||||||
|
name: "FK_ProspectImages_Prospects_ProspectId",
|
||||||
|
column: x => x.ProspectId,
|
||||||
|
principalTable: "Prospects",
|
||||||
|
principalColumn: "Id",
|
||||||
|
onDelete: ReferentialAction.Cascade);
|
||||||
|
});
|
||||||
|
|
||||||
|
migrationBuilder.CreateIndex(
|
||||||
|
name: "IX_ProspectImages_ProspectId",
|
||||||
|
table: "ProspectImages",
|
||||||
|
column: "ProspectId");
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropTable(
|
||||||
|
name: "ProspectImages");
|
||||||
|
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "VerificationToken",
|
||||||
|
table: "Prospects");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
264
FoodsharingSiegen.Server/Migrations/20260426081244_RemoveUserVerified.Designer.cs
generated
Normal file
@@ -0,0 +1,264 @@
|
|||||||
|
// <auto-generated />
|
||||||
|
using System;
|
||||||
|
using FoodsharingSiegen.Server.Data;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using Microsoft.EntityFrameworkCore.Infrastructure;
|
||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Migrations
|
||||||
|
{
|
||||||
|
[DbContext(typeof(FsContext))]
|
||||||
|
[Migration("20260426081244_RemoveUserVerified")]
|
||||||
|
partial class RemoveUserVerified
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
||||||
|
{
|
||||||
|
#pragma warning disable 612, 618
|
||||||
|
modelBuilder.HasAnnotation("ProductVersion", "9.0.3");
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Audit", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Data1")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Data2")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid?>("UserID")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.HasIndex("UserID");
|
||||||
|
|
||||||
|
b.ToTable("Audits");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Interaction", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("Alert")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Date")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Feedback")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("FeedbackInfo")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Info1")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Info2")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("NotNeeded")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid>("ProspectID")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid>("UserID")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.HasIndex("ProspectID");
|
||||||
|
|
||||||
|
b.HasIndex("UserID");
|
||||||
|
|
||||||
|
b.ToTable("Interactions");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Prospect", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("FsId")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("Memo")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime?>("Modified")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Name")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("RecordState")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid?>("VerificationToken")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("Warning")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.ToTable("Prospects");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.ProspectImage", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("ContentType")
|
||||||
|
.IsRequired()
|
||||||
|
.HasMaxLength(100)
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<byte[]>("ImageData")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("BLOB");
|
||||||
|
|
||||||
|
b.Property<Guid>("ProspectId")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.HasIndex("ProspectId");
|
||||||
|
|
||||||
|
b.ToTable("ProspectImages");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.User", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("EncryptedPassword")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<bool>("ForceLogout")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("Groups")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Mail")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Memo")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("Name")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Network")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<string>("ResetToken")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime?>("ResetTokenExpiry")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.ToTable("Users");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Audit", b =>
|
||||||
|
{
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.User", "User")
|
||||||
|
.WithMany()
|
||||||
|
.HasForeignKey("UserID");
|
||||||
|
|
||||||
|
b.Navigation("User");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Interaction", b =>
|
||||||
|
{
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.Prospect", "Prospect")
|
||||||
|
.WithMany("Interactions")
|
||||||
|
.HasForeignKey("ProspectID")
|
||||||
|
.OnDelete(DeleteBehavior.Cascade)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.User", "User")
|
||||||
|
.WithMany("Interactions")
|
||||||
|
.HasForeignKey("UserID")
|
||||||
|
.OnDelete(DeleteBehavior.Cascade)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
b.Navigation("Prospect");
|
||||||
|
|
||||||
|
b.Navigation("User");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.ProspectImage", b =>
|
||||||
|
{
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.Prospect", "Prospect")
|
||||||
|
.WithMany("Images")
|
||||||
|
.HasForeignKey("ProspectId")
|
||||||
|
.OnDelete(DeleteBehavior.Cascade)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
b.Navigation("Prospect");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Prospect", b =>
|
||||||
|
{
|
||||||
|
b.Navigation("Images");
|
||||||
|
|
||||||
|
b.Navigation("Interactions");
|
||||||
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.User", b =>
|
||||||
|
{
|
||||||
|
b.Navigation("Interactions");
|
||||||
|
});
|
||||||
|
#pragma warning restore 612, 618
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Migrations
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
public partial class RemoveUserVerified : Migration
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "Verified",
|
||||||
|
table: "Users");
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<bool>(
|
||||||
|
name: "Verified",
|
||||||
|
table: "Users",
|
||||||
|
type: "INTEGER",
|
||||||
|
nullable: false,
|
||||||
|
defaultValue: false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -118,6 +118,9 @@ namespace FoodsharingSiegen.Server.Migrations
|
|||||||
b.Property<int>("RecordState")
|
b.Property<int>("RecordState")
|
||||||
.HasColumnType("INTEGER");
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<Guid?>("VerificationToken")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
b.Property<bool>("Warning")
|
b.Property<bool>("Warning")
|
||||||
.HasColumnType("INTEGER");
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
@@ -126,6 +129,34 @@ namespace FoodsharingSiegen.Server.Migrations
|
|||||||
b.ToTable("Prospects");
|
b.ToTable("Prospects");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.ProspectImage", b =>
|
||||||
|
{
|
||||||
|
b.Property<Guid>("Id")
|
||||||
|
.ValueGeneratedOnAdd()
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("ContentType")
|
||||||
|
.IsRequired()
|
||||||
|
.HasMaxLength(100)
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<DateTime>("Created")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<byte[]>("ImageData")
|
||||||
|
.IsRequired()
|
||||||
|
.HasColumnType("BLOB");
|
||||||
|
|
||||||
|
b.Property<Guid>("ProspectId")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.HasKey("Id");
|
||||||
|
|
||||||
|
b.HasIndex("ProspectId");
|
||||||
|
|
||||||
|
b.ToTable("ProspectImages");
|
||||||
|
});
|
||||||
|
|
||||||
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.User", b =>
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.User", b =>
|
||||||
{
|
{
|
||||||
b.Property<Guid>("Id")
|
b.Property<Guid>("Id")
|
||||||
@@ -160,10 +191,13 @@ namespace FoodsharingSiegen.Server.Migrations
|
|||||||
b.Property<int>("Network")
|
b.Property<int>("Network")
|
||||||
.HasColumnType("INTEGER");
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
b.Property<int>("Type")
|
b.Property<string>("ResetToken")
|
||||||
.HasColumnType("INTEGER");
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
b.Property<bool>("Verified")
|
b.Property<DateTime?>("ResetTokenExpiry")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<int>("Type")
|
||||||
.HasColumnType("INTEGER");
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
b.HasKey("Id");
|
b.HasKey("Id");
|
||||||
@@ -199,8 +233,21 @@ namespace FoodsharingSiegen.Server.Migrations
|
|||||||
b.Navigation("User");
|
b.Navigation("User");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.ProspectImage", b =>
|
||||||
|
{
|
||||||
|
b.HasOne("FoodsharingSiegen.Contracts.Entity.Prospect", "Prospect")
|
||||||
|
.WithMany("Images")
|
||||||
|
.HasForeignKey("ProspectId")
|
||||||
|
.OnDelete(DeleteBehavior.Cascade)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
b.Navigation("Prospect");
|
||||||
|
});
|
||||||
|
|
||||||
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Prospect", b =>
|
modelBuilder.Entity("FoodsharingSiegen.Contracts.Entity.Prospect", b =>
|
||||||
{
|
{
|
||||||
|
b.Navigation("Images");
|
||||||
|
|
||||||
b.Navigation("Interactions");
|
b.Navigation("Interactions");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -13,6 +13,8 @@
|
|||||||
|
|
||||||
<DataGrid TItem="Audit"
|
<DataGrid TItem="Audit"
|
||||||
Data="@Audits"
|
Data="@Audits"
|
||||||
|
ReadData="@OnReadData"
|
||||||
|
TotalItems="@TotalAudits"
|
||||||
VirtualizeOptions="@(new() { DataGridHeight = "100%", DataGridMaxHeight = "100%"})"
|
VirtualizeOptions="@(new() { DataGridHeight = "100%", DataGridMaxHeight = "100%"})"
|
||||||
Virtualize="true"
|
Virtualize="true"
|
||||||
Responsive>
|
Responsive>
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
|
using Blazorise.DataGrid;
|
||||||
using FoodsharingSiegen.Contracts.Entity;
|
using FoodsharingSiegen.Contracts.Entity;
|
||||||
|
using FoodsharingSiegen.Contracts.Helper;
|
||||||
using FoodsharingSiegen.Server.Data.Service;
|
using FoodsharingSiegen.Server.Data.Service;
|
||||||
using Microsoft.AspNetCore.Components;
|
using Microsoft.AspNetCore.Components;
|
||||||
|
|
||||||
@@ -26,16 +28,47 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private List<Audit>? Audits { get; set; }
|
private List<Audit>? Audits { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the value of the total audits (ab)
|
||||||
|
/// </summary>
|
||||||
|
private int TotalAudits { get; set; }
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Override InitializeDataAsync
|
#region Override InitializeDataAsync
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
protected override async Task InitializeDataAsync()
|
protected override Task InitializeDataAsync()
|
||||||
{
|
{
|
||||||
var loadR = await AuditService?.Load(100)!;
|
if (!CurrentUser.IsAdmin()) NavigationManager.NavigateTo("/");
|
||||||
if (loadR.Success)
|
return Task.CompletedTask;
|
||||||
Audits = loadR.Data;
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region Private Method OnReadData
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Called when data is read (ab)
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="e">The params</param>
|
||||||
|
private async Task OnReadData(DataGridReadDataEventArgs<Audit> e)
|
||||||
|
{
|
||||||
|
if (!CurrentUser.IsAdmin()) return;
|
||||||
|
|
||||||
|
var countLoad = await AuditService?.GetCount()!;
|
||||||
|
if (countLoad.Success)
|
||||||
|
TotalAudits = countLoad.Data;
|
||||||
|
|
||||||
|
// Default fallback if VirtualizeCount is not set, though Blazor shouldn't do this usually
|
||||||
|
var limit = e.VirtualizeCount > 0 ? e.VirtualizeCount : 50;
|
||||||
|
var offset = e.VirtualizeOffset;
|
||||||
|
|
||||||
|
var itemsLoad = await AuditService?.LoadPage(offset, limit)!;
|
||||||
|
if (itemsLoad.Success)
|
||||||
|
Audits = itemsLoad.Data;
|
||||||
|
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|||||||
@@ -1,3 +0,0 @@
|
|||||||
.p-audit > ::deep form {
|
|
||||||
height: calc(100% - 68px);
|
|
||||||
}
|
|
||||||
63
FoodsharingSiegen.Server/Pages/ForgotPassword.razor
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
@page "/forgot-password"
|
||||||
|
@using FoodsharingSiegen.Shared.Helper
|
||||||
|
@layout LoginLayout
|
||||||
|
|
||||||
|
@inherits FoodsharingSiegen.Server.BaseClasses.FsBase
|
||||||
|
|
||||||
|
<PageTitle>@AppSettings.Terms.Title - Passwort vergessen</PageTitle>
|
||||||
|
|
||||||
|
<div class="d-flex justify-content-center align-items-center-sm" style="min-height: 100vh; align-items: start;">
|
||||||
|
<div class="card shadow border-0" style="width: 100%; max-width: 420px; border-radius: 12px; margin: 1rem;">
|
||||||
|
|
||||||
|
<div class="card-body p-4 p-md-5">
|
||||||
|
<div class="text-center mb-4">
|
||||||
|
<i class="fa-solid fa-leaf mb-3" style="font-size: 3rem; color: #64ae24;"></i>
|
||||||
|
<h4 class="font-weight-bold" style="color: #533a20;"><small style="font-size: .6em;" class="d-block">Einarbeitungen</small> @AppSettings.Terms.Title</h4>
|
||||||
|
<p class="text-muted">Passwort zurücksetzen</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if (IsSubmitted)
|
||||||
|
{
|
||||||
|
<div class="alert alert-success text-center">
|
||||||
|
Wenn ein Benutzerkonto mit dieser E-Mail-Adresse existiert, wurde eine E-Mail mit weiteren Anweisungen versendet.
|
||||||
|
<br><br>
|
||||||
|
<small><b>Hinweis:</b> Bitte überprüfe auch deinen Spam-Ordner, falls du künftige E-Mails nicht im regulären Posteingang findest.</small>
|
||||||
|
</div>
|
||||||
|
<div class="text-center mt-4">
|
||||||
|
<a href="/login" class="btn btn-outline-primary"><i class="fas fa-arrow-left mr-2"></i> Zurück zum Login</a>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
@if (!string.IsNullOrEmpty(ErrorMessage))
|
||||||
|
{
|
||||||
|
<div class="alert alert-danger text-center">
|
||||||
|
@ErrorMessage
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<Validation Validator="ValidationHelper.ValidateMail" @bind-Status="@IsValidMail">
|
||||||
|
<Field>
|
||||||
|
<FieldLabel>E-Mail Adresse</FieldLabel>
|
||||||
|
<TextEdit @bind-Text="MailAddress" Role="TextRole.Email" Placeholder="E-Mail" KeyUp="TextEdit_KeyUp" Size="Size.Large"></TextEdit>
|
||||||
|
</Field>
|
||||||
|
</Validation>
|
||||||
|
|
||||||
|
<Button Class="mt-4 w-100" Color="Color.Primary" Size="Size.Large" Clicked="SubmitRequest" Disabled="@(IsValidMail != ValidationStatus.Success || IsLoading)">
|
||||||
|
@if (IsLoading)
|
||||||
|
{
|
||||||
|
<i class="fas fa-spinner fa-spin mr-2"></i> <span>Wird gesendet...</span>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<i class="fas fa-paper-plane mr-2"></i> <span>Passwort zurücksetzen</span>
|
||||||
|
}
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<div class="text-center mt-3">
|
||||||
|
<a href="/login" style="font-size: 0.85rem; color: #64ae24; text-decoration: none;"><i class="fas fa-arrow-left mr-1"></i> Zurück zum Login</a>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
53
FoodsharingSiegen.Server/Pages/ForgotPassword.razor.cs
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
using System;
|
||||||
|
using Blazorise;
|
||||||
|
using FoodsharingSiegen.Server.BaseClasses;
|
||||||
|
using FoodsharingSiegen.Server.Auth;
|
||||||
|
using FoodsharingSiegen.Shared.Helper;
|
||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
using Microsoft.AspNetCore.Components.Web;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Pages
|
||||||
|
{
|
||||||
|
public partial class ForgotPassword : FsBase
|
||||||
|
{
|
||||||
|
public string MailAddress { get; set; } = string.Empty;
|
||||||
|
public ValidationStatus IsValidMail { get; set; } = ValidationStatus.None;
|
||||||
|
|
||||||
|
public bool IsSubmitted { get; set; }
|
||||||
|
public bool IsLoading { get; set; }
|
||||||
|
public string? ErrorMessage { get; set; }
|
||||||
|
|
||||||
|
public async Task SubmitRequest()
|
||||||
|
{
|
||||||
|
if (IsValidMail != ValidationStatus.Success) return;
|
||||||
|
|
||||||
|
IsLoading = true;
|
||||||
|
ErrorMessage = null;
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await AuthService.InitiatePasswordReset(MailAddress, NavigationManager.BaseUri);
|
||||||
|
IsSubmitted = true;
|
||||||
|
}
|
||||||
|
catch (Exception)
|
||||||
|
{
|
||||||
|
ErrorMessage = "Es gab ein Problem bei der Verarbeitung der Anfrage. Bitte versuche es später erneut oder wende dich an einen Administrator.";
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
IsLoading = false;
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task TextEdit_KeyUp(KeyboardEventArgs e)
|
||||||
|
{
|
||||||
|
if (e.Key == "Enter" && IsValidMail == ValidationStatus.Success)
|
||||||
|
{
|
||||||
|
await SubmitRequest();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,24 +4,45 @@
|
|||||||
|
|
||||||
@inherits FoodsharingSiegen.Server.BaseClasses.FsBase
|
@inherits FoodsharingSiegen.Server.BaseClasses.FsBase
|
||||||
|
|
||||||
<PageTitle>@AppSettings.Terms.Title</PageTitle>
|
<PageTitle>@AppSettings.Terms.Title - Login</PageTitle>
|
||||||
|
|
||||||
|
<div class="d-flex justify-content-center align-items-center-sm" style="min-height: 100vh; align-items: start;">
|
||||||
|
<div class="card shadow border-0" style="width: 100%; max-width: 420px; border-radius: 12px; margin: 1rem;">
|
||||||
|
|
||||||
|
<div class="card-body p-4 p-md-5">
|
||||||
|
<div class="text-center mb-4">
|
||||||
|
<i class="fa-solid fa-leaf mb-3" style="font-size: 3rem; color: #64ae24;"></i>
|
||||||
|
<h4 class="font-weight-bold" style="color: #533a20;"><small style="font-size: .6em;" class="d-block">Einarbeitungen</small> @AppSettings.Terms.Title</h4>
|
||||||
|
<p class="text-muted">Bitte melde dich an, um fortzufahren.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="d-flex justify-content-center align-items-center" style="height: 100vh;">
|
|
||||||
<div class="card" style="width: 100%; max-width: 380px;">
|
|
||||||
<div class="card-header">@AppSettings.Terms.Title</div>
|
|
||||||
<div class="card-body">
|
|
||||||
<Validation Validator="ValidationHelper.ValidateMail" @bind-Status="@IsValidMail">
|
<Validation Validator="ValidationHelper.ValidateMail" @bind-Status="@IsValidMail">
|
||||||
<TextEdit @bind-Text="MailAddress" Role="TextRole.Email" Placeholder="E-Mail" Class="mt-0" KeyUp="TextEdit_KeyUp"></TextEdit>
|
<Field>
|
||||||
|
<FieldLabel>E-Mail Adresse</FieldLabel>
|
||||||
|
<TextEdit @bind-Text="MailAddress" Role="TextRole.Email" Placeholder="E-Mail" KeyUp="TextEdit_KeyUp" Size="Size.Large"></TextEdit>
|
||||||
|
</Field>
|
||||||
</Validation>
|
</Validation>
|
||||||
|
|
||||||
<Validation Validator="ValidationHelper.ValidatePassword" @bind-Status="@IsValidPassword">
|
<Validation Validator="ValidationHelper.ValidatePassword" @bind-Status="@IsValidPassword">
|
||||||
<TextEdit @bind-Text="Password" Role="TextRole.Password" Placeholder="Passwort" Class="my-3" KeyUp="TextEdit_KeyUp"></TextEdit>
|
<Field Class="mt-3">
|
||||||
|
<FieldLabel>Passwort</FieldLabel>
|
||||||
|
<TextEdit @bind-Text="Password" Role="TextRole.Password" Placeholder="Passwort" KeyUp="TextEdit_KeyUp" Size="Size.Large"></TextEdit>
|
||||||
|
</Field>
|
||||||
</Validation>
|
</Validation>
|
||||||
|
|
||||||
<div class="d-flex justify-content-center">
|
@if (!string.IsNullOrEmpty(LoginErrorMessage))
|
||||||
<Button Clicked="PerformLogin" Disabled="@(IsValidMail != ValidationStatus.Success || IsValidPassword != ValidationStatus.Success)">Einloggen</Button>
|
{
|
||||||
|
<div class="text-danger mt-3 text-center">
|
||||||
|
<i class="fas fa-exclamation-triangle mr-1"></i> @LoginErrorMessage
|
||||||
</div>
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<Button Class="mt-4 w-100" Color="Color.Primary" Size="Size.Large" Clicked="PerformLogin" Disabled="@(IsValidMail != ValidationStatus.Success || IsValidPassword != ValidationStatus.Success)">
|
||||||
|
<i class="fas fa-sign-in-alt mr-2"></i> Einloggen
|
||||||
|
</Button>
|
||||||
|
<div class="d-flex justify-content-center align-items-center mt-2">
|
||||||
|
<a href="/forgot-password" tabindex="-1" style="font-size: 0.85rem; color: #64ae24; text-decoration: none;">Passwort vergessen?</a>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -33,6 +33,11 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private string? Password { get; set; }
|
private string? Password { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the login error message
|
||||||
|
/// </summary>
|
||||||
|
private string? LoginErrorMessage { get; set; }
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Private Method PerformLogin
|
#region Private Method PerformLogin
|
||||||
@@ -42,12 +47,14 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private async Task PerformLogin()
|
private async Task PerformLogin()
|
||||||
{
|
{
|
||||||
|
LoginErrorMessage = null;
|
||||||
//Todo Eingaben Validieren [04.04.22 - Andre Beging]
|
//Todo Eingaben Validieren [04.04.22 - Andre Beging]
|
||||||
|
|
||||||
if (string.IsNullOrWhiteSpace(MailAddress) || string.IsNullOrWhiteSpace(Password))
|
if (string.IsNullOrWhiteSpace(MailAddress) || string.IsNullOrWhiteSpace(Password))
|
||||||
{
|
{
|
||||||
MailAddress = string.Empty;
|
MailAddress = string.Empty;
|
||||||
Password = string.Empty;
|
Password = string.Empty;
|
||||||
|
LoginErrorMessage = "E-Mail-Adresse oder Passwort ist ungültig.";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -55,7 +62,7 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
if (loginR.Success)
|
if (loginR.Success)
|
||||||
NavigationManager.NavigateTo("/", true);
|
NavigationManager.NavigateTo("/", true);
|
||||||
else
|
else
|
||||||
await Notification.Error(loginR.ErrorMessage)!;
|
LoginErrorMessage = loginR.ErrorMessage;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|||||||
@@ -29,14 +29,14 @@
|
|||||||
<TextEdit @bind-Text="User.Mail" ReadOnly="true"></TextEdit>
|
<TextEdit @bind-Text="User.Mail" ReadOnly="true"></TextEdit>
|
||||||
</FieldBody>
|
</FieldBody>
|
||||||
</Field>
|
</Field>
|
||||||
@* <Validation Validator="ValidationRule.None"> *@
|
<Validation Validator="ValidationRule.None">
|
||||||
@* <Field ColumnSize="ColumnSize.Is12"> *@
|
<Field ColumnSize="ColumnSize.Is12">
|
||||||
@* <FieldLabel>Info über dich</FieldLabel> *@
|
<FieldLabel>Info über dich</FieldLabel>
|
||||||
@* <FieldBody> *@
|
<FieldBody>
|
||||||
@* <MemoEdit Rows="3" Placeholder="z.B. Bieb bei Rewe Musterhausen" @bind-Text="User.Memo"/> *@
|
<MemoEdit Rows="3" Placeholder="z.B. Bieb bei Rewe Musterhausen" @bind-Text="User.Memo"/>
|
||||||
@* </FieldBody> *@
|
</FieldBody>
|
||||||
@* </Field> *@
|
</Field>
|
||||||
@* </Validation> *@
|
</Validation>
|
||||||
</Validations>
|
</Validations>
|
||||||
</Fields>
|
</Fields>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -2,12 +2,13 @@
|
|||||||
@page "/prospect"
|
@page "/prospect"
|
||||||
@page "/prospects"
|
@page "/prospects"
|
||||||
|
|
||||||
|
@using FoodsharingSiegen.Contracts
|
||||||
@using FoodsharingSiegen.Contracts.Enums
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
@using FoodsharingSiegen.Shared.Helper
|
@using FoodsharingSiegen.Shared.Helper
|
||||||
@inherits FsBase
|
@inherits FsBase
|
||||||
|
|
||||||
<PageTitle>Neue Foodsaver - @AppSettings.Terms.Title</PageTitle>
|
<PageTitle>Aktuelle - @AppSettings.Terms.Title</PageTitle>
|
||||||
<h2>Neue Foodsaver</h2>
|
<h2>Aktuelle Einarbeitungen</h2>
|
||||||
|
|
||||||
@if (AppSettings.TestMode)
|
@if (AppSettings.TestMode)
|
||||||
{
|
{
|
||||||
@@ -16,20 +17,25 @@
|
|||||||
|
|
||||||
<Button
|
<Button
|
||||||
Color="Color.Primary"
|
Color="Color.Primary"
|
||||||
|
Height="Height.Px(50)"
|
||||||
|
title="Hinzufügen"
|
||||||
|
style="min-width: auto;"
|
||||||
Clicked="@CreateProspectAsync"
|
Clicked="@CreateProspectAsync"
|
||||||
Visibility="@(CurrentUser.IsInGroup(UserGroup.WelcomeTeam, UserGroup.Ambassador) ? Visibility.Default : Visibility.Invisible)"
|
Visibility="@(CurrentUser.IsInGroup(UserGroup.WelcomeTeam, UserGroup.Ambassador) ? Visibility.Default : Visibility.Invisible)"
|
||||||
>Hinzufügen
|
><i class="fa-solid fa-plus"></i> Neu
|
||||||
</Button>
|
</Button>
|
||||||
|
|
|
||||||
|
|
||||||
|
<ProspectSortControl @bind-CurrentSort="CurrentSort" StorageKey="@StorageKeys.SortProspects" OnSortChanged="StateHasChanged" Filter="Filter" FilterChanged="FilterChangedAsync" StateFilter="ProspectStateFilter.OnBoarding" />
|
||||||
|
|
||||||
@{
|
@{
|
||||||
var filterList = ProspectList.ApplyFilter(Filter);
|
var filterList = ProspectList.ApplyFilter(Filter);
|
||||||
|
var sortList = filterList.ApplySort(CurrentSort);
|
||||||
}
|
}
|
||||||
|
|
||||||
<hr/>
|
|
||||||
<ProspectFilterControl Filter="Filter" FilterChanged="FilterChangedAsync" StateFilter="ProspectStateFilter.OnBoarding"></ProspectFilterControl>
|
|
||||||
<hr />
|
|
||||||
<ProspectGrid
|
<ProspectGrid
|
||||||
Prospects="filterList"
|
Prospects="sortList"
|
||||||
OnDataChanged="@LoadProspects"
|
OnDataChanged="@LoadProspects"
|
||||||
StateFilter="ProspectStateFilter.OnBoarding">
|
StateFilter="ProspectStateFilter.OnBoarding">
|
||||||
</ProspectGrid>
|
</ProspectGrid>
|
||||||
@@ -36,6 +36,8 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private List<Prospect>? ProspectList { get; set; }
|
private List<Prospect>? ProspectList { get; set; }
|
||||||
|
|
||||||
|
private ProspectSortOption CurrentSort { get; set; } = ProspectSortOption.NameAscending;
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Override InitializeDataAsync
|
#region Override InitializeDataAsync
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
@page "/all"
|
@page "/all"
|
||||||
@page "/archive"
|
@page "/archive"
|
||||||
|
|
||||||
|
@using FoodsharingSiegen.Contracts
|
||||||
@using FoodsharingSiegen.Contracts.Enums
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
@using FoodsharingSiegen.Shared.Helper
|
@using FoodsharingSiegen.Shared.Helper
|
||||||
@inherits FsBase
|
@inherits FsBase
|
||||||
@@ -13,15 +14,15 @@
|
|||||||
<div class="alert alert-danger"><strong>TESTMODUS!</strong> Änderungen werden wieder gelöscht.</div>
|
<div class="alert alert-danger"><strong>TESTMODUS!</strong> Änderungen werden wieder gelöscht.</div>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
<ProspectSortControl @bind-CurrentSort="CurrentSort" StorageKey="@StorageKeys.SortProspectsAll" OnSortChanged="StateHasChanged" Filter="Filter" FilterChanged="FilterChangedAsync" StateFilter="ProspectStateFilter.All" />
|
||||||
|
|
||||||
@{
|
@{
|
||||||
var filterList = ProspectList.ApplyFilter(Filter);
|
var filterList = ProspectList.ApplyFilter(Filter);
|
||||||
|
var sortList = filterList.ApplySort(CurrentSort);
|
||||||
}
|
}
|
||||||
|
|
||||||
<hr/>
|
|
||||||
<ProspectFilterControl Filter="Filter" FilterChanged="FilterChangedAsync" StateFilter="ProspectStateFilter.All"></ProspectFilterControl>
|
|
||||||
<hr />
|
|
||||||
<ProspectGrid
|
<ProspectGrid
|
||||||
Prospects="filterList"
|
Prospects="sortList"
|
||||||
OnDataChanged="@LoadProspects"
|
OnDataChanged="@LoadProspects"
|
||||||
StateFilter="ProspectStateFilter.All">
|
StateFilter="ProspectStateFilter.All">
|
||||||
</ProspectGrid>
|
</ProspectGrid>
|
||||||
@@ -39,6 +39,8 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private List<Prospect>? ProspectList { get; set; }
|
private List<Prospect>? ProspectList { get; set; }
|
||||||
|
|
||||||
|
private ProspectSortOption CurrentSort { get; set; } = ProspectSortOption.NameAscending;
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Override InitializeDataAsync
|
#region Override InitializeDataAsync
|
||||||
|
|||||||
@@ -1,26 +1,27 @@
|
|||||||
@page "/done"
|
@page "/done"
|
||||||
|
|
||||||
|
@using FoodsharingSiegen.Contracts
|
||||||
@using FoodsharingSiegen.Contracts.Enums
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
@using FoodsharingSiegen.Shared.Helper
|
@using FoodsharingSiegen.Shared.Helper
|
||||||
@inherits FsBase
|
@inherits FsBase
|
||||||
|
|
||||||
<PageTitle>Abgeschlossen - @AppSettings.Terms.Title</PageTitle>
|
<PageTitle>Fertige - @AppSettings.Terms.Title</PageTitle>
|
||||||
<h2>Abgeschlossen</h2>
|
<h2>Abgeschlossene Einarbeitungen</h2>
|
||||||
|
|
||||||
@if (AppSettings.TestMode)
|
@if (AppSettings.TestMode)
|
||||||
{
|
{
|
||||||
<div class="alert alert-danger"><strong>TESTMODUS!</strong> Änderungen werden wieder gelöscht.</div>
|
<div class="alert alert-danger"><strong>TESTMODUS!</strong> Änderungen werden wieder gelöscht.</div>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
<ProspectSortControl @bind-CurrentSort="CurrentSort" StorageKey="@StorageKeys.SortProspectsDone" OnSortChanged="StateHasChanged" Filter="Filter" FilterChanged="FilterChangedAsync" StateFilter="ProspectStateFilter.Completed" />
|
||||||
|
|
||||||
@{
|
@{
|
||||||
var filterList = ProspectList.ApplyFilter(Filter);
|
var filterList = ProspectList.ApplyFilter(Filter);
|
||||||
|
var sortList = filterList.ApplySort(CurrentSort);
|
||||||
}
|
}
|
||||||
|
|
||||||
<hr />
|
|
||||||
<ProspectFilterControl Filter="Filter" FilterChanged="FilterChangedAsync" StateFilter="ProspectStateFilter.Completed"></ProspectFilterControl>
|
|
||||||
<hr />
|
|
||||||
<ProspectGrid
|
<ProspectGrid
|
||||||
Prospects="filterList"
|
Prospects="sortList"
|
||||||
OnDataChanged="@LoadProspects"
|
OnDataChanged="@LoadProspects"
|
||||||
StateFilter="ProspectStateFilter.Completed">
|
StateFilter="ProspectStateFilter.Completed">
|
||||||
</ProspectGrid>
|
</ProspectGrid>
|
||||||
@@ -31,6 +31,8 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private List<Prospect>? ProspectList { get; set; }
|
private List<Prospect>? ProspectList { get; set; }
|
||||||
|
|
||||||
|
private ProspectSortOption CurrentSort { get; set; } = ProspectSortOption.NameAscending;
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Override InitializeDataAsync
|
#region Override InitializeDataAsync
|
||||||
|
|||||||
@@ -1,26 +1,28 @@
|
|||||||
@page "/verify"
|
@page "/verify"
|
||||||
|
|
||||||
|
@using FoodsharingSiegen.Contracts
|
||||||
@using FoodsharingSiegen.Contracts.Enums
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
@using FoodsharingSiegen.Shared.Helper
|
@using FoodsharingSiegen.Shared.Helper
|
||||||
@inherits FsBase
|
@inherits FsBase
|
||||||
|
|
||||||
<PageTitle>Freischalten - @AppSettings.Terms.Title</PageTitle>
|
<PageTitle>Freischalten - @AppSettings.Terms.Title</PageTitle>
|
||||||
<h2>Freischalten</h2>
|
<h2>Zum Freischalten freigegeben</h2>
|
||||||
|
|
||||||
@if (AppSettings.TestMode)
|
@if (AppSettings.TestMode)
|
||||||
{
|
{
|
||||||
<div class="alert alert-danger"><strong>TESTMODUS!</strong> Änderungen werden wieder gelöscht.</div>
|
<div class="alert alert-danger"><strong>TESTMODUS!</strong> Änderungen werden wieder gelöscht.</div>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
<ProspectSortControl @bind-CurrentSort="CurrentSort" StorageKey="@StorageKeys.SortProspectsVerify" OnSortChanged="StateHasChanged" Filter="Filter" FilterChanged="FilterChangedAsync" StateFilter="ProspectStateFilter.Verification" />
|
||||||
|
|
||||||
@{
|
@{
|
||||||
var filterList = ProspectList.ApplyFilter(Filter);
|
var filterList = ProspectList.ApplyFilter(Filter);
|
||||||
|
var sortList = filterList.ApplySort(CurrentSort);
|
||||||
}
|
}
|
||||||
|
|
||||||
<hr/>
|
|
||||||
<ProspectFilterControl Filter="Filter" FilterChanged="FilterChangedAsync" StateFilter="ProspectStateFilter.Verification"></ProspectFilterControl>
|
|
||||||
<hr />
|
|
||||||
<ProspectGrid
|
<ProspectGrid
|
||||||
Prospects="filterList"
|
Prospects="sortList"
|
||||||
OnDataChanged="@LoadProspects"
|
OnDataChanged="@LoadProspects"
|
||||||
StateFilter="ProspectStateFilter.Verification">
|
StateFilter="ProspectStateFilter.Verification">
|
||||||
</ProspectGrid>
|
</ProspectGrid>
|
||||||
@@ -37,6 +37,8 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private List<Prospect>? ProspectList { get; set; }
|
private List<Prospect>? ProspectList { get; set; }
|
||||||
|
|
||||||
|
private ProspectSortOption CurrentSort { get; set; } = ProspectSortOption.NameAscending;
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Override InitializeDataAsync
|
#region Override InitializeDataAsync
|
||||||
|
|||||||
86
FoodsharingSiegen.Server/Pages/ResetPassword.razor
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
@page "/reset-password/{Token}"
|
||||||
|
@using FoodsharingSiegen.Shared.Helper
|
||||||
|
@layout LoginLayout
|
||||||
|
|
||||||
|
@inherits FoodsharingSiegen.Server.BaseClasses.FsBase
|
||||||
|
|
||||||
|
<PageTitle>@AppSettings.Terms.Title - Neues Passwort setzen</PageTitle>
|
||||||
|
|
||||||
|
<div class="d-flex justify-content-center align-items-center-sm" style="min-height: 100vh; align-items: start;">
|
||||||
|
<div class="card shadow border-0" style="width: 100%; max-width: 420px; border-radius: 12px; margin: 1rem;">
|
||||||
|
|
||||||
|
<div class="card-body p-4 p-md-5">
|
||||||
|
<div class="text-center mb-4">
|
||||||
|
<i class="fa-solid fa-leaf mb-3" style="font-size: 3rem; color: #64ae24;"></i>
|
||||||
|
<h4 class="font-weight-bold" style="color: #533a20;"><small style="font-size: .6em;" class="d-block">Einarbeitungen</small> @AppSettings.Terms.Title</h4>
|
||||||
|
<p class="text-muted">Neues Passwort festlegen</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if (IsInitializing)
|
||||||
|
{
|
||||||
|
<div class="text-center mt-4">
|
||||||
|
<i class="fas fa-spinner fa-spin fa-2x mb-3" style="color: #64ae24;"></i>
|
||||||
|
<p>Token wird überprüft...</p>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else if (!IsTokenValid)
|
||||||
|
{
|
||||||
|
<div class="alert alert-danger text-center">
|
||||||
|
<i class="fas fa-exclamation-triangle fa-2x mb-2 d-block"></i>
|
||||||
|
Der Link zum Zurücksetzen des Passworts ist ungültig oder abgelaufen. Bitte fordere einen neuen an.
|
||||||
|
</div>
|
||||||
|
<div class="text-center mt-4">
|
||||||
|
<a href="/forgot-password" class="btn btn-outline-primary w-100 mb-2"><i class="fas fa-redo mr-2"></i> Neuen Link anfordern</a>
|
||||||
|
<a href="/login" class="btn btn-link w-100" style="color: #64ae24;">Zurück zum Login</a>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else if (IsSuccess)
|
||||||
|
{
|
||||||
|
<div class="alert alert-success text-center">
|
||||||
|
Passwort erfolgreich aktualisiert. Du kannst dich jetzt anmelden.
|
||||||
|
</div>
|
||||||
|
<div class="text-center mt-4">
|
||||||
|
<a href="/login" class="btn btn-outline-primary"><i class="fas fa-sign-in-alt mr-2"></i> Zum Login</a>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="alert alert-info" style="font-size: 0.85rem;">
|
||||||
|
<i class="fas fa-info-circle mr-1"></i> Dein Passwort muss mindestens 8 Zeichen lang sein und eine Zahl enthalten.
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Validation Validator="ValidationHelper.ValidatePassword" @bind-Status="@IsValidPassword">
|
||||||
|
<Field>
|
||||||
|
<FieldLabel>Neues Passwort</FieldLabel>
|
||||||
|
<TextEdit @bind-Text="NewPassword" Role="TextRole.Password" Placeholder="Passwort (min. 8 Zeichen, min. 1 Zahl)" KeyUp="TextEdit_KeyUp" Size="Size.Large"></TextEdit>
|
||||||
|
</Field>
|
||||||
|
</Validation>
|
||||||
|
|
||||||
|
<Validation Validator="ValidatePasswordConfirmation" @bind-Status="@IsValidPasswordConfirmation">
|
||||||
|
<Field Class="mt-3">
|
||||||
|
<FieldLabel>Passwort bestätigen</FieldLabel>
|
||||||
|
<TextEdit @bind-Text="ConfirmPassword" Role="TextRole.Password" Placeholder="Passwort bestätigen" KeyUp="TextEdit_KeyUp" Size="Size.Large"></TextEdit>
|
||||||
|
</Field>
|
||||||
|
</Validation>
|
||||||
|
|
||||||
|
@if (!string.IsNullOrEmpty(ErrorMessage))
|
||||||
|
{
|
||||||
|
<div class="text-danger mt-3 text-center">
|
||||||
|
<i class="fas fa-exclamation-triangle mr-1"></i> @ErrorMessage
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<Button Class="mt-4 w-100" Color="Color.Primary" Size="Size.Large" Clicked="SubmitReset" Disabled="@(IsValidPassword != ValidationStatus.Success || IsValidPasswordConfirmation != ValidationStatus.Success || IsLoading)">
|
||||||
|
@if (IsLoading)
|
||||||
|
{
|
||||||
|
<i class="fas fa-spinner fa-spin mr-2"></i> <span>Speichern...</span>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<i class="fas fa-save mr-2"></i> <span>Passwort speichern</span>
|
||||||
|
}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
77
FoodsharingSiegen.Server/Pages/ResetPassword.razor.cs
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
using Blazorise;
|
||||||
|
using FoodsharingSiegen.Server.BaseClasses;
|
||||||
|
using FoodsharingSiegen.Server.Auth;
|
||||||
|
using FoodsharingSiegen.Shared.Helper;
|
||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
using Microsoft.AspNetCore.Components.Web;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Pages
|
||||||
|
{
|
||||||
|
public partial class ResetPassword : FsBase
|
||||||
|
{
|
||||||
|
[Parameter]
|
||||||
|
public string Token { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
public string NewPassword { get; set; } = string.Empty;
|
||||||
|
public string ConfirmPassword { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
public ValidationStatus IsValidPassword { get; set; } = ValidationStatus.None;
|
||||||
|
public ValidationStatus IsValidPasswordConfirmation { get; set; } = ValidationStatus.None;
|
||||||
|
|
||||||
|
public string ErrorMessage { get; set; } = string.Empty;
|
||||||
|
public bool IsSuccess { get; set; }
|
||||||
|
public bool IsLoading { get; set; }
|
||||||
|
public bool IsInitializing { get; set; } = true;
|
||||||
|
public bool IsTokenValid { get; set; }
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
IsTokenValid = await AuthService.IsResetTokenValid(Token);
|
||||||
|
IsInitializing = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ValidatePasswordConfirmation(ValidatorEventArgs args)
|
||||||
|
{
|
||||||
|
var confirmPassword = System.Convert.ToString(args.Value);
|
||||||
|
if (string.IsNullOrWhiteSpace(confirmPassword))
|
||||||
|
{
|
||||||
|
args.Status = ValidationStatus.None;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
args.Status = confirmPassword == NewPassword ? ValidationStatus.Success : ValidationStatus.Error;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task SubmitReset()
|
||||||
|
{
|
||||||
|
if (IsValidPassword != ValidationStatus.Success || IsValidPasswordConfirmation != ValidationStatus.Success) return;
|
||||||
|
|
||||||
|
IsLoading = true;
|
||||||
|
ErrorMessage = string.Empty;
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
|
||||||
|
var result = await AuthService.ResetPassword(Token, NewPassword);
|
||||||
|
|
||||||
|
if (result.Success)
|
||||||
|
{
|
||||||
|
IsSuccess = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ErrorMessage = result.Exception?.Message ?? "Ein Fehler ist aufgetreten.";
|
||||||
|
}
|
||||||
|
|
||||||
|
IsLoading = false;
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task TextEdit_KeyUp(KeyboardEventArgs e)
|
||||||
|
{
|
||||||
|
if (e.Key == "Enter" && IsValidPassword == ValidationStatus.Success && IsValidPasswordConfirmation == ValidationStatus.Success)
|
||||||
|
{
|
||||||
|
await SubmitReset();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
30
FoodsharingSiegen.Server/Pages/Settings.razor
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
@page "/settings"
|
||||||
|
|
||||||
|
@inherits FsBase
|
||||||
|
|
||||||
|
<PageTitle>Einstellungen - @AppSettings.Terms.Title</PageTitle>
|
||||||
|
|
||||||
|
<div style="width: 100%; max-width: 500px;">
|
||||||
|
<h2>Einstellungen</h2>
|
||||||
|
|
||||||
|
<div class="card mt-3">
|
||||||
|
<div class="card-header" style="padding: .5rem 0 0 1rem;"><h5>Mail Service</h5></div>
|
||||||
|
<div class="card-body">
|
||||||
|
<p>
|
||||||
|
<strong>SMTP Server:</strong> @AppSettings.Mail.Host<br />
|
||||||
|
<strong>Port:</strong> @AppSettings.Mail.Port<br />
|
||||||
|
<strong>SSL:</strong> @(AppSettings.Mail.UseSsl ? "Ja" : "Nein")<br />
|
||||||
|
</p>
|
||||||
|
<hr />
|
||||||
|
<Fields Class="my-3">
|
||||||
|
<Field ColumnSize="ColumnSize.Is12">
|
||||||
|
<FieldLabel>Testmail-Empfänger</FieldLabel>
|
||||||
|
<FieldBody>
|
||||||
|
<TextEdit @bind-Text="TestMailReceiver" Placeholder="E-Mail Adresse"></TextEdit>
|
||||||
|
</FieldBody>
|
||||||
|
</Field>
|
||||||
|
</Fields>
|
||||||
|
<Button Color="Color.Primary" Clicked="SendTestMail">Testmail senden</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
46
FoodsharingSiegen.Server/Pages/Settings.razor.cs
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
using FoodsharingSiegen.Contracts.Helper;
|
||||||
|
using FoodsharingSiegen.Server.Service;
|
||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
using System;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Pages
|
||||||
|
{
|
||||||
|
public partial class Settings
|
||||||
|
{
|
||||||
|
[Inject]
|
||||||
|
public IMailService MailService { get; set; } = null!;
|
||||||
|
|
||||||
|
private string TestMailReceiver { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
protected override async Task InitializeDataAsync()
|
||||||
|
{
|
||||||
|
if (!CurrentUser.IsAdmin())
|
||||||
|
{
|
||||||
|
NavigationManager.NavigateTo("/");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
TestMailReceiver = CurrentUser.Mail ?? string.Empty;
|
||||||
|
await Task.CompletedTask;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task SendTestMail()
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(TestMailReceiver))
|
||||||
|
{
|
||||||
|
await Notification.Error("Bitte eine Empfänger-Adresse angeben.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await MailService.SendEmailAsync(TestMailReceiver, "Testmail", "Dies ist eine Testmail.");
|
||||||
|
await Notification.Success("Testmail versendet.");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
await Notification.Error($"Fehler beim Senden: {ex.Message}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
123
FoodsharingSiegen.Server/Pages/UploadVerification.razor
Normal file
@@ -0,0 +1,123 @@
|
|||||||
|
@page "/verify/{Token:guid}"
|
||||||
|
@using FoodsharingSiegen.Contracts.Entity
|
||||||
|
@using System.IO
|
||||||
|
@layout LoginLayout
|
||||||
|
|
||||||
|
<PageTitle>Identitätsprüfung - @AppSettings.Value.Terms.Title</PageTitle>
|
||||||
|
|
||||||
|
<div class="row min-vh-100 align-items-center-sm justify-content-center p-0 p-md-5 m-0">
|
||||||
|
<div class="col-12 col-md-10 col-lg-8 col-xl-5 login-form p-2">
|
||||||
|
<div class="card shadow-sm border-0">
|
||||||
|
<div class="card-body p-3 p-md-5">
|
||||||
|
<div class="text-center mb-4">
|
||||||
|
|
||||||
|
<h5 class="mb-0 text-success d-block d-sm-none"><i
|
||||||
|
class="fa-solid fa-address-card me-2"></i><br>Identitätsprüfung</h5>
|
||||||
|
<h4 class="mb-0 text-success d-none d-sm-block"><i
|
||||||
|
class="fa-solid fa-address-card me-2"></i>Identitätsprüfung</h4>
|
||||||
|
|
||||||
|
<p class="text-muted mt-2">@(AppSettings.Value.Terms.Title)</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if (_isLoading)
|
||||||
|
{
|
||||||
|
<div class="text-center my-5">
|
||||||
|
<div class="spinner-border text-success" role="status">
|
||||||
|
<span class="visually-hidden">Laden...</span>
|
||||||
|
</div>
|
||||||
|
<p class="mt-3 text-muted">Lade Daten...</p>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else if (_prospect == null)
|
||||||
|
{
|
||||||
|
<div class="alert alert-danger" role="alert">
|
||||||
|
<i class="fa-solid fa-triangle-exclamation me-2"></i> @(_message ?? "Der Link ist ungültig oder abgelaufen. Bitte fordere einen neuen Link an.")
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="alert alert-info mb-4">
|
||||||
|
<strong>Hinweis:</strong> Dies ist die Upload-Seite für Foodsaver <b>@_prospect.FsId</b>.
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mb-3 text-center">
|
||||||
|
|
||||||
|
@if (_uploadedCount >= 5)
|
||||||
|
{
|
||||||
|
<div class="alert alert-warning py-2 mb-0">Du hast die maximale Anzahl von 5 Bildern erreicht.</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<InputFile id="fileInput" OnChange="OnInputFileChange" class="d-none" accept="image/*" />
|
||||||
|
<label for="fileInput" class="btn btn-outline-success w-100" style="height: 5rem;">
|
||||||
|
<i class="fa-solid fa-images me-2"></i>Bild auswählen
|
||||||
|
</label>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if (!string.IsNullOrEmpty(_message))
|
||||||
|
{
|
||||||
|
<div class="alert alert-@(_isSuccess ? "success" : "danger") alert-dismissible fade show" role="alert">
|
||||||
|
@_message
|
||||||
|
<button type="button" class="btn-close" @onclick="() => _message = null"></button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (_isSuccess)
|
||||||
|
{
|
||||||
|
<div class="alert alert-success text-center">
|
||||||
|
<i class="fa-solid fa-check-circle fa-2x mb-2"></i><br />
|
||||||
|
Vielen Dank für den Upload. Wenn du alle benötigten Bilder hochgeladen hast, kannst du die Seite
|
||||||
|
schließen. Du musst nichts weiter tun.
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (_uploadedCount < 5)
|
||||||
|
{
|
||||||
|
<div class="text-center mb-3">
|
||||||
|
<span class="badge bg-secondary text-wrap">Es können noch bis zu @(5 - _uploadedCount) Bilder
|
||||||
|
hochgeladen werden</span>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<div class="mb-1 text-muted">
|
||||||
|
Um dich auf der Foodsharing-Plattform als Foodsaver freischalten zu können, muss ein*e
|
||||||
|
Botschafter*in Name, Adresse und Geburtsdatum im Profil des Foodsavers auf Korrektheit durch
|
||||||
|
Vergleich mit einem Ausweisdokument prüfen. Das ist wichtig, damit die <a
|
||||||
|
href="https://wiki.foodsharing.network/wiki/Rechtsvereinbarung"
|
||||||
|
target="_blank">Rechtsvereinbarung</a> Bestand hat und wir die Zusagen erfüllen, die wir den
|
||||||
|
Spenderbetrieben geben.<br>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mb-3 text-muted">
|
||||||
|
<a class="mt-3"
|
||||||
|
href="https://wiki.foodsharing.network/wiki/Foodsaver#3.4_Verifizierung_(der_Daten),_Foodsaver-Ausweis_und_Freischaltung"
|
||||||
|
target="_blank">Mehr dazu im Wiki</a>.
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mb-4">
|
||||||
|
<h5 class="fw-bold">Anleitung:</h5>
|
||||||
|
<ul class="text-muted">
|
||||||
|
<li>Lade hier die Vorder- und Rückseite deines Personalausweises oder Reisepasses hoch</li>
|
||||||
|
<li>Dein Name und Adresse müssen gut und lesbar erkennbar sein</li>
|
||||||
|
<li>Wir nutzen diese Bilder ausschließlich zur Identitätsprüfung</li>
|
||||||
|
<li>Ausschließlich die Botschafter haben Zugriff auf diese Bilder</li>
|
||||||
|
<li>Die Bilder werden nach der Überprüfung sofort und unwiderruflich von uns gelöscht</li>
|
||||||
|
<li>Bereits hochgeladene Bilder werden hier hier aus Datenschutzgründen nicht angezeigt</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if (_isUploading)
|
||||||
|
{
|
||||||
|
<div class="text-center my-3">
|
||||||
|
<div class="spinner-border text-primary spinner-border-sm" role="status">
|
||||||
|
<span class="visually-hidden">Laden...</span>
|
||||||
|
</div>
|
||||||
|
<span class="ms-2">Bilder werden hochgeladen und verarbeitet...</span>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
142
FoodsharingSiegen.Server/Pages/UploadVerification.razor.cs
Normal file
@@ -0,0 +1,142 @@
|
|||||||
|
using FoodsharingSiegen.Contracts.Entity;
|
||||||
|
using FoodsharingSiegen.Contracts.Model;
|
||||||
|
using FoodsharingSiegen.Server.Data.Service;
|
||||||
|
using FoodsharingSiegen.Shared.Helper;
|
||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
using Microsoft.AspNetCore.Components.Forms;
|
||||||
|
using Microsoft.Extensions.Options;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Pages
|
||||||
|
{
|
||||||
|
public partial class UploadVerification : ComponentBase
|
||||||
|
{
|
||||||
|
[Inject]
|
||||||
|
public ProspectService ProspectService { get; set; } = null!;
|
||||||
|
|
||||||
|
[Inject]
|
||||||
|
public IOptions<AppSettings> AppSettings { get; set; } = null!;
|
||||||
|
|
||||||
|
[Parameter]
|
||||||
|
public Guid Token { get; set; }
|
||||||
|
|
||||||
|
private bool _isLoading = true;
|
||||||
|
private Prospect? _prospect;
|
||||||
|
|
||||||
|
private int _uploadedCount = 0;
|
||||||
|
private bool _isUploading = false;
|
||||||
|
|
||||||
|
private string? _message;
|
||||||
|
private bool _isSuccess;
|
||||||
|
|
||||||
|
private const int MaxAllowedFiles = 5;
|
||||||
|
private const long MaxFileSize = 10 * 1024 * 1024; // 10 MB
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
await LoadData();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task LoadData()
|
||||||
|
{
|
||||||
|
_isLoading = true;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var result = await ProspectService.GetProspectByVerificationTokenAsync(Token);
|
||||||
|
|
||||||
|
if (result.Success && result.Data != null)
|
||||||
|
{
|
||||||
|
_prospect = result.Data;
|
||||||
|
_uploadedCount = _prospect.Images?.Count ?? 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_prospect = null;
|
||||||
|
_message = result.Exception?.Message ?? "Ein Fehler ist aufgetreten.";
|
||||||
|
_isSuccess = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_message = $"Fehler: {ex.Message}";
|
||||||
|
_isSuccess = false;
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
_isLoading = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task OnInputFileChange(InputFileChangeEventArgs e)
|
||||||
|
{
|
||||||
|
if (_prospect == null) return;
|
||||||
|
|
||||||
|
_message = null;
|
||||||
|
|
||||||
|
var files = e.GetMultipleFiles(MaxAllowedFiles);
|
||||||
|
|
||||||
|
if (_uploadedCount + files.Count > MaxAllowedFiles)
|
||||||
|
{
|
||||||
|
_message = $"Es sind maximal {MaxAllowedFiles} Bilder erlaubt.";
|
||||||
|
_isSuccess = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_isUploading = true;
|
||||||
|
|
||||||
|
int successCount = 0;
|
||||||
|
|
||||||
|
foreach (var file in files)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (file.Size > MaxFileSize)
|
||||||
|
{
|
||||||
|
_message = $"Bild '{file.Name}' überschreitet die erlaubte Größe von 10 MB.";
|
||||||
|
_isSuccess = false;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resize the image to a max dimension of 1000 pixels (longest edge) to save DB space
|
||||||
|
var resizedImageFile = await file.RequestImageFileAsync(file.ContentType, 1000, 1000);
|
||||||
|
|
||||||
|
using var stream = resizedImageFile.OpenReadStream(MaxFileSize);
|
||||||
|
using var memoryStream = new MemoryStream();
|
||||||
|
|
||||||
|
await stream.CopyToAsync(memoryStream);
|
||||||
|
|
||||||
|
byte[] imageData = memoryStream.ToArray();
|
||||||
|
|
||||||
|
var saveResult = await ProspectService.AddVerificationImageAsync(_prospect.Id, imageData, resizedImageFile.ContentType);
|
||||||
|
|
||||||
|
if (saveResult.Success)
|
||||||
|
{
|
||||||
|
successCount++;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_message = $"Fehler beim Speichern von {file.Name}: {saveResult.Exception?.Message}";
|
||||||
|
_isSuccess = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_message = $"Fehler bei {file.Name}: {ex.Message}";
|
||||||
|
_isSuccess = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_isUploading = false;
|
||||||
|
|
||||||
|
if (successCount > 0)
|
||||||
|
{
|
||||||
|
_message = $"{successCount} Bild(er) erfolgreich hinzugefügt.";
|
||||||
|
_isSuccess = true;
|
||||||
|
_uploadedCount += successCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
StateHasChanged();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,111 +6,137 @@
|
|||||||
|
|
||||||
@inherits FoodsharingSiegen.Server.BaseClasses.FsBase
|
@inherits FoodsharingSiegen.Server.BaseClasses.FsBase
|
||||||
|
|
||||||
@code {
|
|
||||||
|
|
||||||
private RenderFragment PopupTitleTemplate(PopupTitleContext<User> value)
|
|
||||||
{
|
|
||||||
var header = "Benutzer erstellen";
|
|
||||||
if (value.EditState == DataGridEditState.Edit) header = "Benutzer bearbeiten";
|
|
||||||
|
|
||||||
return builder =>
|
|
||||||
{
|
|
||||||
builder.OpenElement(0, "span");
|
|
||||||
builder.AddContent(1, header);
|
|
||||||
builder.CloseElement();
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
<PageTitle>Benutzerverwaltung - @AppSettings.Terms.Title</PageTitle>
|
<PageTitle>Benutzerverwaltung - @AppSettings.Terms.Title</PageTitle>
|
||||||
|
|
||||||
<h2>Benutzerverwaltung <span style="font-size: .5em; line-height: 0;">Admin</span></h2>
|
<div class="d-flex justify-content-between align-items-center mb-3">
|
||||||
|
<h2>Benutzerverwaltung <span style="font-size: .5em; line-height: 0;">Admin</span></h2>
|
||||||
<div class="my-2">
|
<Button Color="Color.Success" Clicked="CreateNewUser">
|
||||||
<Button Color="Color.Primary" Disabled="@(SelectedUser == null)" Clicked="async () => await PasswordModal?.Show(SelectedUser!)!"><i class="fa-solid fa-key"></i> setzen</Button>
|
<Icon Name="IconName.Add" /> Benutzer erstellen
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<DataGrid TItem="User"
|
<div class="user-grid" style="max-width: 1000px;">
|
||||||
@ref="UserDataGrid"
|
@if (SortedUsers != null)
|
||||||
Data="@UserList"
|
{
|
||||||
CommandMode="DataGridCommandMode.Commands"
|
@foreach (var user in SortedUsers)
|
||||||
EditMode="DataGridEditMode.Popup"
|
{
|
||||||
PopupTitleTemplate="PopupTitleTemplate"
|
<Card Class="user-card">
|
||||||
RowInserted="RowInserted"
|
<CardBody>
|
||||||
RowUpdated="RowUpdated"
|
<CardTitle Size="4">@user.Name</CardTitle>
|
||||||
PageSize="50"
|
<CardSubtitle Class="mb-2 text-muted">@user.Mail</CardSubtitle>
|
||||||
@bind-SelectedRow="SelectedUser"
|
<CardText>
|
||||||
RowDoubleClicked="arg => UserDataGrid?.Edit(arg.Item)!"
|
Typ:
|
||||||
Editable
|
@if (user.Type == UserType.Unverified)
|
||||||
Responsive="true">
|
{
|
||||||
<DataGridColumns>
|
<span style="color: red" class="fw-bold">@user.Type</span>
|
||||||
<DataGridCommandColumn TItem="User" Width="100px" CellClass="@(_ => "px-0 d-flex align-items-center justify-content-center")">
|
}
|
||||||
<NewCommandTemplate>
|
else if (user.Type == UserType.Admin)
|
||||||
<Button Size="Size.ExtraSmall" Color="Color.Success" Clicked="@context.Clicked" Class="mr-1" Style="min-width: auto;">
|
{
|
||||||
<i class="oi oi-plus"></i>
|
<span style="color: blue" class="fw-bold">@user.Type</span>
|
||||||
</Button>
|
}
|
||||||
</NewCommandTemplate>
|
else
|
||||||
<EditCommandTemplate>
|
{
|
||||||
<Button Size="Size.ExtraSmall" Color="Color.Secondary" Clicked="@context.Clicked" Class="mr-1" Style="min-width: auto;">
|
@user.Type
|
||||||
<i class="oi oi-pencil"></i>
|
}
|
||||||
</Button>
|
</CardText>
|
||||||
</EditCommandTemplate>
|
|
||||||
<DeleteCommandTemplate>
|
<div class="mb-3">
|
||||||
<Button Size="Size.ExtraSmall" Color="Color.Danger" Clicked="() => RemoveUserAsync(context.Item)" Class="mr-1" Style="min-width: auto;">
|
@if (user.GroupsList != null && user.GroupsList.Any())
|
||||||
<i class="oi oi-trash"></i>
|
{
|
||||||
</Button>
|
@foreach(var group in user.GroupsList)
|
||||||
</DeleteCommandTemplate>
|
{
|
||||||
<ClearFilterCommandTemplate>
|
<Badge Color="Color.Primary" Class="me-1" Style="font-size: 0.8em;">@group.ToString()</Badge>
|
||||||
<Button Size="Size.ExtraSmall" Color="Color.Danger" Clicked="@context.Clicked" Style="min-width: auto;">
|
}
|
||||||
<i class="o"></i> <i class="fas fa-trash"></i>
|
}
|
||||||
</Button>
|
else
|
||||||
</ClearFilterCommandTemplate>
|
{
|
||||||
</DataGridCommandColumn>
|
<span class="text-muted" style="font-style: italic; font-size: 0.8em;">Keine Gruppen</span>
|
||||||
<DataGridCheckColumn TItem="User" Field="@nameof(User.Verified)" Caption="Verifiziert" Editable="true" Width="100px">
|
}
|
||||||
<DisplayTemplate>
|
</div>
|
||||||
<Check TValue="bool" Checked="context.Verified" Disabled="true" ReadOnly="true"/>
|
</CardBody>
|
||||||
</DisplayTemplate>
|
<CardFooter Class="d-flex justify-content-between">
|
||||||
</DataGridCheckColumn>
|
<Button Color="Color.Primary" Size="Size.Small" Clicked="() => EditUser(user)"><Icon Name="IconName.Edit" /></Button>
|
||||||
<DataGridColumn TItem="User" Field="@nameof(User.Type)" Caption="Typ" Editable="true" Width="200px">
|
<Button Color="Color.Info" Size="Size.Small" Clicked="() => SetPassword(user)"><i class="fa-solid fa-key"></i></Button>
|
||||||
<EditTemplate>
|
<Button Color="Color.Secondary" Size="Size.Small" Clicked="() => SendPasswordSetupMail(user)"><Icon Name="IconName.Mail" /></Button>
|
||||||
<Select TValue="UserType" SelectedValue="@((UserType)context.CellValue)" SelectedValueChanged="@(v => context.CellValue = v)">
|
@if (!(user.Type == UserType.Admin && SortedUsers.Count(x => x.Type == UserType.Admin) <= 1))
|
||||||
|
{
|
||||||
|
<Button Color="Color.Danger" Size="Size.Small" Clicked="() => RemoveUserAsync(user)"><Icon Name="IconName.Delete" /></Button>
|
||||||
|
}
|
||||||
|
</CardFooter>
|
||||||
|
</Card>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.user-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(1, 1fr);
|
||||||
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
@@media (min-width: 700px) {
|
||||||
|
.user-grid {
|
||||||
|
grid-template-columns: repeat(2, 1fr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@media (min-width: 1250px) {
|
||||||
|
.user-grid {
|
||||||
|
grid-template-columns: repeat(3, 1fr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<Modal @ref="editUserModal">
|
||||||
|
<ModalContent Centered>
|
||||||
|
<ModalHeader>
|
||||||
|
<ModalTitle>@(IsEditing ? "Benutzer bearbeiten" : "Benutzer erstellen")</ModalTitle>
|
||||||
|
<CloseButton />
|
||||||
|
</ModalHeader>
|
||||||
|
<ModalBody>
|
||||||
|
@if (EditModel != null)
|
||||||
|
{
|
||||||
|
<Field>
|
||||||
|
<FieldLabel>Name</FieldLabel>
|
||||||
|
<TextEdit @bind-Text="EditModel.Name" />
|
||||||
|
</Field>
|
||||||
|
<Field>
|
||||||
|
<FieldLabel>E-Mail</FieldLabel>
|
||||||
|
<TextEdit @bind-Text="EditModel.Mail" />
|
||||||
|
</Field>
|
||||||
|
<Field>
|
||||||
|
<FieldLabel>Typ</FieldLabel>
|
||||||
|
<Select TValue="UserType" SelectedValue="EditModel.Type" SelectedValueChanged="@(v => EditModel.Type = v)" Disabled="@IsLastAdmin">
|
||||||
@foreach (var enumValue in Enum.GetValues<UserType>())
|
@foreach (var enumValue in Enum.GetValues<UserType>())
|
||||||
{
|
{
|
||||||
<SelectItem TValue="UserType" Value="enumValue">@enumValue</SelectItem>
|
<SelectItem TValue="UserType" Value="enumValue">@enumValue</SelectItem>
|
||||||
}
|
}
|
||||||
</Select>
|
</Select>
|
||||||
</EditTemplate>
|
@if (IsLastAdmin)
|
||||||
</DataGridColumn>
|
{
|
||||||
<DataGridColumn TItem="User" Field="@nameof(User.Name)" Caption="Name" Editable="true" Width="250px"></DataGridColumn>
|
<small class="text-danger mt-1 d-block">Das ist der letzte Administrator-Account. Der Typ kann nicht geändert werden.</small>
|
||||||
<DataGridColumn TItem="User" Field="@nameof(User.Mail)" Caption="E-Mail" Editable="true"></DataGridColumn>
|
}
|
||||||
<DataGridColumn TItem="User" Field="@nameof(User.GroupsList)" Caption="Gruppen" Editable="true">
|
</Field>
|
||||||
<EditTemplate>
|
<Field>
|
||||||
|
<FieldLabel>Gruppen</FieldLabel>
|
||||||
<Autocomplete TItem="UserGroup"
|
<Autocomplete TItem="UserGroup"
|
||||||
TValue="UserGroup"
|
TValue="UserGroup"
|
||||||
Size="Size.ExtraSmall"
|
|
||||||
Data="@UserGroups"
|
Data="@UserGroups"
|
||||||
TextField="@(( item ) => item.ToString())"
|
TextField="@(( item ) => item.ToString())"
|
||||||
ValueField="@(( item ) => item)"
|
ValueField="@(( item ) => item)"
|
||||||
Multiple="true"
|
SelectionMode="AutocompleteSelectionMode.Multiple"
|
||||||
SelectedValues="@((List<UserGroup>) context.CellValue)"
|
SelectedValues="@EditModel.GroupsList"
|
||||||
SelectedValuesChanged="@(v => context.CellValue = v)"
|
SelectedValuesChanged="@(v => { EditModel.GroupsList = v.ToList(); })"
|
||||||
@bind-SelectedTexts="SelectedCompanyTexts">
|
@bind-SelectedTexts="SelectedGroupTexts">
|
||||||
</Autocomplete>
|
</Autocomplete>
|
||||||
<small>Verfügbar: @string.Join(", ", Enum.GetValues<UserGroup>())</small>
|
<small>Verfügbar: @string.Join(", ", Enum.GetValues<UserGroup>())</small>
|
||||||
</EditTemplate>
|
</Field>
|
||||||
<DisplayTemplate>
|
|
||||||
@if (string.IsNullOrWhiteSpace(context.Groups))
|
|
||||||
{
|
|
||||||
<span style="font-style: italic;">Keine Gruppen</span>
|
|
||||||
}
|
}
|
||||||
else
|
</ModalBody>
|
||||||
{
|
<ModalFooter>
|
||||||
<span>@string.Join(", ", context.GroupsList)</span>
|
<Button Color="Color.Secondary" Clicked="() => editUserModal?.Hide()">Abbrechen</Button>
|
||||||
}
|
<Button Color="Color.Primary" Clicked="SaveUser">Speichern</Button>
|
||||||
</DisplayTemplate>
|
</ModalFooter>
|
||||||
</DataGridColumn>
|
</ModalContent>
|
||||||
</DataGridColumns>
|
</Modal>
|
||||||
</DataGrid>
|
|
||||||
|
|
||||||
<SetPasswordModal @ref="PasswordModal" OnPasswortSet="OnPasswordSet"></SetPasswordModal>
|
<SetPasswordModal @ref="PasswordModal" OnPasswortSet="OnPasswordSet"></SetPasswordModal>
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
using Blazorise.DataGrid;
|
using Blazorise;
|
||||||
using FoodsharingSiegen.Contracts.Entity;
|
using FoodsharingSiegen.Contracts.Entity;
|
||||||
using FoodsharingSiegen.Contracts.Enums;
|
using FoodsharingSiegen.Contracts.Enums;
|
||||||
using FoodsharingSiegen.Contracts.Helper;
|
using FoodsharingSiegen.Contracts.Helper;
|
||||||
@@ -22,6 +22,12 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
[Inject]
|
[Inject]
|
||||||
public UserService UserService { get; set; } = null!;
|
public UserService UserService { get; set; } = null!;
|
||||||
|
|
||||||
|
[Inject]
|
||||||
|
public new FoodsharingSiegen.Server.Auth.AuthService AuthService { get; set; } = null!;
|
||||||
|
|
||||||
|
[Inject]
|
||||||
|
public new NavigationManager NavigationManager { get; set; } = null!;
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Private Properties
|
#region Private Properties
|
||||||
@@ -32,19 +38,29 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
private SetPasswordModal? PasswordModal { get; set; }
|
private SetPasswordModal? PasswordModal { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the selected company texts (ab)
|
/// Gets or sets the edit user modal
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private List<string> SelectedCompanyTexts { get; set; } = new();
|
private Modal? editUserModal { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the selected user (ab)
|
/// Gets or sets the selected group texts
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private User? SelectedUser { get; set; }
|
private List<string> SelectedGroupTexts { get; set; } = new();
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the value of the user data grid (ab)
|
/// Gets or sets the edit model
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private DataGrid<User>? UserDataGrid { get; set; }
|
private User? EditModel { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets a value indicating whether we are editing an existing user
|
||||||
|
/// </summary>
|
||||||
|
private bool IsEditing { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets a value indicating whether the current editing user is the last admin
|
||||||
|
/// </summary>
|
||||||
|
private bool IsLastAdmin => IsEditing && EditModel?.Type == UserType.Admin && UserList?.Count(x => x.Type == UserType.Admin) <= 1;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the value of the user groups (ab)
|
/// Gets the value of the user groups (ab)
|
||||||
@@ -56,6 +72,11 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private List<User>? UserList { get; set; }
|
private List<User>? UserList { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the sorted users list
|
||||||
|
/// </summary>
|
||||||
|
private IEnumerable<User> SortedUsers => UserList?.OrderByDescending(x => x.Type).ThenBy(x => x.Name) ?? Enumerable.Empty<User>();
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Override InitializeDataAsync
|
#region Override InitializeDataAsync
|
||||||
@@ -84,6 +105,66 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
#region Actions
|
||||||
|
|
||||||
|
private void CreateNewUser()
|
||||||
|
{
|
||||||
|
EditModel = new User();
|
||||||
|
IsEditing = false;
|
||||||
|
editUserModal?.Show();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void EditUser(User user)
|
||||||
|
{
|
||||||
|
EditModel = user.Clone();
|
||||||
|
IsEditing = true;
|
||||||
|
editUserModal?.Show();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void SetPassword(User user)
|
||||||
|
{
|
||||||
|
PasswordModal?.Show(user);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task SendPasswordSetupMail(User user)
|
||||||
|
{
|
||||||
|
await ConfirmDialog.ShowAsync(ModalService, "Bestätigen", $"Soll eine E-Mail zum Festlegen des Passworts an {user.Mail} gesendet werden?", async () =>
|
||||||
|
{
|
||||||
|
await AuthService.InitiateInitialPasswordSetup(user.Mail, NavigationManager.BaseUri);
|
||||||
|
if (Notification != null)
|
||||||
|
{
|
||||||
|
await Notification.Success("E-Mail gesendet. Bitte weise den Benutzer darauf hin, auch den Spam-Ordner zu prüfen.");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task SaveUser()
|
||||||
|
{
|
||||||
|
if (EditModel == null) return;
|
||||||
|
|
||||||
|
if (IsEditing)
|
||||||
|
{
|
||||||
|
var updateR = await UserService.Update(EditModel);
|
||||||
|
if (!updateR.Success)
|
||||||
|
await Notification.Error($"Fehler beim Speichern: {updateR.ErrorMessage}")!;
|
||||||
|
else
|
||||||
|
await Notification.Success("Benutzer aktualisiert");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var addUserR = await UserService.AddUserAsync(EditModel);
|
||||||
|
if (!addUserR.Success)
|
||||||
|
await Notification.Error($"Fehler beim Anlegen: {addUserR.ErrorMessage}")!;
|
||||||
|
else
|
||||||
|
await Notification.Success("Benutzer erstellt");
|
||||||
|
}
|
||||||
|
|
||||||
|
await editUserModal?.Hide()!;
|
||||||
|
await LoadUsers();
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
#region Private Method OnPasswordSet
|
#region Private Method OnPasswordSet
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -108,12 +189,6 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
/// <returns>A task that represents the asynchronous remove operation.</returns>
|
/// <returns>A task that represents the asynchronous remove operation.</returns>
|
||||||
private async Task RemoveUserAsync(User user)
|
private async Task RemoveUserAsync(User user)
|
||||||
{
|
{
|
||||||
if (user.IsAdmin())
|
|
||||||
{
|
|
||||||
await Notification.Error("Admins können nicht gelöscht werden!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await ConfirmDialog.ShowAsync(ModalService, "Bestätigen", $"User {user.Mail} löschen?", async () =>
|
await ConfirmDialog.ShowAsync(ModalService, "Bestätigen", $"User {user.Mail} löschen?", async () =>
|
||||||
{
|
{
|
||||||
var removeR = await UserService.RemoveAsync(user.Id);
|
var removeR = await UserService.RemoveAsync(user.Id);
|
||||||
@@ -125,39 +200,5 @@ namespace FoodsharingSiegen.Server.Pages
|
|||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Private Method RowInserted
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Rows the inserted using the specified arg (a. beging, 01.04.2022)
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="arg">The arg</param>
|
|
||||||
private async Task RowInserted(SavedRowItem<User, Dictionary<string, object>> arg)
|
|
||||||
{
|
|
||||||
var addUserR = await UserService.AddUserAsync(arg.Item);
|
|
||||||
if (!addUserR.Success)
|
|
||||||
await Notification.Error($"Fehler beim Anlegen: {addUserR.ErrorMessage}")!;
|
|
||||||
else
|
|
||||||
await LoadUsers();
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Private Method RowUpdated
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Rows the updated using the specified arg (a. beging, 01.04.2022)
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="arg">The arg</param>
|
|
||||||
private async Task RowUpdated(SavedRowItem<User, Dictionary<string, object>> arg)
|
|
||||||
{
|
|
||||||
if (arg.Item?.Id == null || arg.Item.Id.Equals(Guid.Empty) || arg.Values?.Any() != true) return;
|
|
||||||
|
|
||||||
var updateR = await UserService.Update(arg.Item);
|
|
||||||
if (!updateR.Success)
|
|
||||||
await Notification.Error($"Fehler beim Speichern: {updateR.ErrorMessage}")!;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -8,18 +8,24 @@
|
|||||||
<meta charset="utf-8"/>
|
<meta charset="utf-8"/>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
|
||||||
<base href="~/"/>
|
<base href="~/"/>
|
||||||
<link rel="stylesheet" href="css/bootstrap/bootstrap.min.css"/>
|
<link rel="icon" type="image/png" href="/favicon-96x96.png" sizes="96x96" />
|
||||||
<link href="css/site.css" rel="stylesheet"/>
|
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
|
||||||
<link href="FoodsharingSiegen.Server.styles.css" rel="stylesheet"/>
|
<link rel="shortcut icon" href="/favicon.ico" />
|
||||||
|
<link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png" />
|
||||||
|
<meta name="apple-mobile-web-app-title" content="Foodsharing Einarbeitungen" />
|
||||||
|
<link rel="manifest" href="/site.webmanifest" />
|
||||||
|
<link rel="stylesheet" href="~/css/bootstrap/bootstrap.min.css" asp-append-version="true" />
|
||||||
|
<link href="~/css/site.css" rel="stylesheet" asp-append-version="true" />
|
||||||
|
<link href="~/FoodsharingSiegen.Server.styles.css" rel="stylesheet" asp-append-version="true" />
|
||||||
|
|
||||||
<!-- Material CSS -->
|
<!-- Material CSS -->
|
||||||
<link href="css/material.min.css" rel="stylesheet">
|
<link href="~/css/material.min.css" rel="stylesheet" asp-append-version="true" />
|
||||||
|
|
||||||
<!-- Add Material font (Roboto) and Material icon as needed -->
|
<!-- Add Material font (Roboto) and Material icon as needed -->
|
||||||
<link href="https://fonts.googleapis.com/css?family=Roboto:300,300i,400,400i,500,500i,700,700i|Roboto+Mono:300,400,700|Roboto+Slab:300,400,700" rel="stylesheet">
|
<link href="https://fonts.googleapis.com/css?family=Roboto:300,300i,400,400i,500,500i,700,700i|Roboto+Mono:300,400,700|Roboto+Slab:300,400,700" rel="stylesheet">
|
||||||
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
|
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
|
||||||
|
|
||||||
<link href="css/all.min.css" rel="stylesheet" />
|
<link href="~/css/all.min.css" rel="stylesheet" asp-append-version="true" />
|
||||||
<link href="_content/Blazorise/blazorise.css?v=1.7.5.0" rel="stylesheet" />
|
<link href="_content/Blazorise/blazorise.css?v=1.7.5.0" rel="stylesheet" />
|
||||||
<link href="_content/Blazorise.Material/blazorise.material.css?v=1.7.5.0" rel="stylesheet" />
|
<link href="_content/Blazorise.Material/blazorise.material.css?v=1.7.5.0" rel="stylesheet" />
|
||||||
<link href="_content/Blazorise.Icons.Material/blazorise.icons.material.css?v=1.7.5.0" rel="stylesheet" />
|
<link href="_content/Blazorise.Icons.Material/blazorise.icons.material.css?v=1.7.5.0" rel="stylesheet" />
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
|
using AspNetCore.SassCompiler;
|
||||||
using Blazorise;
|
using Blazorise;
|
||||||
using Blazorise.Icons.Material;
|
using Blazorise.Icons.Material;
|
||||||
using Blazorise.Material;
|
using Blazorise.Material;
|
||||||
@@ -19,6 +20,9 @@ builder.WebHost.UseUrls("http://+:8700");
|
|||||||
builder.Services.AddRazorPages();
|
builder.Services.AddRazorPages();
|
||||||
builder.Services.AddServerSideBlazor();
|
builder.Services.AddServerSideBlazor();
|
||||||
builder.AddDatabaseContext();
|
builder.AddDatabaseContext();
|
||||||
|
#if DEBUG
|
||||||
|
builder.Services.AddSassCompiler();
|
||||||
|
#endif
|
||||||
|
|
||||||
// DI
|
// DI
|
||||||
builder.Services.AddScoped<LocalStorageService>();
|
builder.Services.AddScoped<LocalStorageService>();
|
||||||
@@ -29,6 +33,7 @@ builder.Services.AddScoped<AuditService>();
|
|||||||
builder.Services.AddScoped<AuthService>();
|
builder.Services.AddScoped<AuthService>();
|
||||||
builder.Services.AddScoped<UserService>();
|
builder.Services.AddScoped<UserService>();
|
||||||
builder.Services.AddScoped<ProspectService>();
|
builder.Services.AddScoped<ProspectService>();
|
||||||
|
builder.Services.AddScoped<IMailService, MailService>();
|
||||||
|
|
||||||
builder.Services
|
builder.Services
|
||||||
.AddBlazorise()
|
.AddBlazorise()
|
||||||
|
|||||||
@@ -12,6 +12,7 @@
|
|||||||
"commandName": "Project",
|
"commandName": "Project",
|
||||||
"dotnetRunMessages": true,
|
"dotnetRunMessages": true,
|
||||||
"launchBrowser": true,
|
"launchBrowser": true,
|
||||||
|
"launchUrl": "http://localhost:56000/",
|
||||||
"applicationUrl": "http://localhost:56000",
|
"applicationUrl": "http://localhost:56000",
|
||||||
"environmentVariables": {
|
"environmentVariables": {
|
||||||
"ASPNETCORE_ENVIRONMENT": "Development"
|
"ASPNETCORE_ENVIRONMENT": "Development"
|
||||||
|
|||||||
19
FoodsharingSiegen.Server/Service/IMailService.cs
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Service
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Service interface for sending emails.
|
||||||
|
/// </summary>
|
||||||
|
public interface IMailService
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sends an email asynchronously.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="toEmail">The recipient's email address.</param>
|
||||||
|
/// <param name="subject">The subject of the email.</param>
|
||||||
|
/// <param name="htmlBody">The HTML content of the email body.</param>
|
||||||
|
/// <returns>A task that represents the asynchronous email sending operation.</returns>
|
||||||
|
Task SendEmailAsync(string toEmail, string subject, string htmlBody);
|
||||||
|
}
|
||||||
|
}
|
||||||
57
FoodsharingSiegen.Server/Service/MailService.cs
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
using System.Threading.Tasks;
|
||||||
|
using FoodsharingSiegen.Contracts.Model;
|
||||||
|
using MailKit.Net.Smtp;
|
||||||
|
using MailKit.Security;
|
||||||
|
using Microsoft.Extensions.Options;
|
||||||
|
using MimeKit;
|
||||||
|
using MimeKit.Text;
|
||||||
|
|
||||||
|
namespace FoodsharingSiegen.Server.Service
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Default implementation of <see cref="IMailService"/> which sends emails via SMTP using MailKit.
|
||||||
|
/// </summary>
|
||||||
|
public class MailService : IMailService
|
||||||
|
{
|
||||||
|
private readonly MailSettings _mailSettings;
|
||||||
|
private readonly TermSettings _termSettings;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Initializes a new instance of the <see cref="MailService"/> class.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="appSettings">The configured application settings injected by DI, containing the <see cref="MailSettings"/>.</param>
|
||||||
|
public MailService(IOptions<AppSettings> appSettings)
|
||||||
|
{
|
||||||
|
_mailSettings = appSettings.Value.Mail;
|
||||||
|
_termSettings = appSettings.Value.Terms;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc/>
|
||||||
|
public async Task SendEmailAsync(string toEmail, string subject, string htmlBody)
|
||||||
|
{
|
||||||
|
var email = new MimeMessage();
|
||||||
|
email.From.Add(new MailboxAddress(_termSettings.Title, _mailSettings.FromAddress));
|
||||||
|
email.To.Add(MailboxAddress.Parse(toEmail));
|
||||||
|
email.Subject = subject;
|
||||||
|
|
||||||
|
var textPart = new TextPart(TextFormat.Html)
|
||||||
|
{
|
||||||
|
Text = htmlBody
|
||||||
|
};
|
||||||
|
email.Body = textPart;
|
||||||
|
|
||||||
|
using var smtp = new SmtpClient();
|
||||||
|
var secureOptions = _mailSettings.UseSsl ? SecureSocketOptions.StartTls : SecureSocketOptions.Auto;
|
||||||
|
|
||||||
|
await smtp.ConnectAsync(_mailSettings.Host, _mailSettings.Port, secureOptions);
|
||||||
|
|
||||||
|
if (!string.IsNullOrWhiteSpace(_mailSettings.Username))
|
||||||
|
{
|
||||||
|
await smtp.AuthenticateAsync(_mailSettings.Username, _mailSettings.Password);
|
||||||
|
}
|
||||||
|
|
||||||
|
await smtp.SendAsync(email);
|
||||||
|
await smtp.DisconnectAsync(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -24,5 +24,5 @@
|
|||||||
</NotAuthorized>
|
</NotAuthorized>
|
||||||
</AuthorizeView>
|
</AuthorizeView>
|
||||||
|
|
||||||
<NotificationAlert/>
|
<NotificationProvider/>
|
||||||
<MessageAlert/>
|
<MessageProvider/>
|
||||||
@@ -1,180 +0,0 @@
|
|||||||
.page {
|
|
||||||
display: flex;
|
|
||||||
height: 100vh;
|
|
||||||
}
|
|
||||||
|
|
||||||
.page aside {
|
|
||||||
background-color: #f1e7c9;
|
|
||||||
width: 0;
|
|
||||||
box-shadow: 5px 5px 5px #acacac;
|
|
||||||
transition: width 250ms;
|
|
||||||
overflow: hidden auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 992px) {
|
|
||||||
.page aside {
|
|
||||||
width: 250px;
|
|
||||||
min-width: 250px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.page main {
|
|
||||||
flex: 1;
|
|
||||||
width: 100%;
|
|
||||||
overflow: hidden auto;
|
|
||||||
padding: 60px 10px 10px 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 992px) {
|
|
||||||
.page main {
|
|
||||||
padding: 20px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#menu-toggler {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
#menu-toggler-label {
|
|
||||||
position: fixed;
|
|
||||||
top: 15px;
|
|
||||||
left: 15px;
|
|
||||||
cursor: pointer;
|
|
||||||
height: 30px;
|
|
||||||
width: calc(100% - 40px);
|
|
||||||
z-index: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 992px) {
|
|
||||||
#menu-toggler-label {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#menu-toggler-label span {
|
|
||||||
height: 6px;
|
|
||||||
background-color: #64ae24;
|
|
||||||
position: absolute;
|
|
||||||
border-radius: 3px;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
width: 30px;
|
|
||||||
transition: all 0.25s ease-out;
|
|
||||||
}
|
|
||||||
|
|
||||||
#menu-toggler-label span:nth-child(2) {
|
|
||||||
top: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#menu-toggler-label span:nth-child(3) {
|
|
||||||
top: 24px;
|
|
||||||
}
|
|
||||||
|
|
||||||
input:checked ~ aside {
|
|
||||||
width: 250px;
|
|
||||||
}
|
|
||||||
|
|
||||||
input:checked ~ #menu-toggler-label span:nth-child(1) {
|
|
||||||
transform: translateY(12px) rotate(45deg);
|
|
||||||
}
|
|
||||||
|
|
||||||
input:checked ~ #menu-toggler-label span:nth-child(2) {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
input:checked ~ #menu-toggler-label span:nth-child(3) {
|
|
||||||
transform: translateY(-12px) rotate(-45deg);
|
|
||||||
}
|
|
||||||
|
|
||||||
::deep h2 {
|
|
||||||
font-size: 1.8rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 576px) {
|
|
||||||
::deep h2 {
|
|
||||||
font-size: 3.2rem;
|
|
||||||
line-height: 3.2rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 768px) {
|
|
||||||
::deep h2 {
|
|
||||||
font-size: 3.8rem;
|
|
||||||
line-height: 3.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 992px) {
|
|
||||||
::deep h2 {
|
|
||||||
font-size: 3.8rem;
|
|
||||||
line-height: 3.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 1200px) {
|
|
||||||
::deep h2 {
|
|
||||||
font-size: 3.8rem;
|
|
||||||
line-height: 3.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 1400px) {
|
|
||||||
::deep h2 {
|
|
||||||
font-size: 3.8rem;
|
|
||||||
line-height: 3.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 1950px) {
|
|
||||||
::deep h2 {
|
|
||||||
font-size: 3.2rem;
|
|
||||||
line-height: 3.2rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
::deep h3 {
|
|
||||||
font-size: 1.4rem;
|
|
||||||
font-weight: 300;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 576px) {
|
|
||||||
::deep h3 {
|
|
||||||
font-size: 2.4rem;
|
|
||||||
line-height: 2.4rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 768px) {
|
|
||||||
::deep h3 {
|
|
||||||
font-size: 2.8rem;
|
|
||||||
line-height: 2.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 992px) {
|
|
||||||
::deep h3 {
|
|
||||||
font-size: 2.8rem;
|
|
||||||
line-height: 2.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 1200px) {
|
|
||||||
::deep h3 {
|
|
||||||
font-size: 2.8rem;
|
|
||||||
line-height: 2.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 1400px) {
|
|
||||||
::deep h3 {
|
|
||||||
font-size: 2.8rem;
|
|
||||||
line-height: 2.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 1950px) {
|
|
||||||
::deep h3 {
|
|
||||||
font-size: 2.4rem;
|
|
||||||
line-height: 2.4rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -7,7 +7,7 @@ $sidebarBreakpoint: $breakpointM;
|
|||||||
height: 100vh;
|
height: 100vh;
|
||||||
|
|
||||||
aside {
|
aside {
|
||||||
background-color: #f1e7c9;
|
background-color: #f0f0f0;
|
||||||
width: 0;
|
width: 0;
|
||||||
box-shadow: 5px 5px 5px #acacac;
|
box-shadow: 5px 5px 5px #acacac;
|
||||||
transition: width 250ms;
|
transition: width 250ms;
|
||||||
|
|||||||
@@ -8,11 +8,6 @@ main {
|
|||||||
flex: 1;
|
flex: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar {
|
|
||||||
/*background-image: linear-gradient(180deg, rgb(5, 39, 103) 0%, #3a0647 70%);*/
|
|
||||||
background-color: #f1e7c9 !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.top-row {
|
.top-row {
|
||||||
background-color: #f7f7f7;
|
background-color: #f7f7f7;
|
||||||
border-bottom: 1px solid #d6d5d5;
|
border-bottom: 1px solid #d6d5d5;
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
@using FoodsharingSiegen.Contracts.Enums
|
@using FoodsharingSiegen.Contracts.Enums
|
||||||
<nav class="d-flex flex-column h-100">
|
<nav class="d-flex flex-column h-100">
|
||||||
<div class="nav-logo"></div>
|
<i class="fa-solid fa-leaf mb-3 mt-3 text-center" style="font-size: 4rem; color: #64ae24;"></i>
|
||||||
<div class="d-flex px-3 justify-content-center text-center font-weight-bold">
|
<div class="px-3 justify-content-center text-center font-weight-bold">
|
||||||
Einarbeitungen<br/>
|
<span class="d-block">Einarbeitungen</span>
|
||||||
@(AppSettings.Terms.TitleShort ?? AppSettings.Terms.Title)
|
<h5 class="d-block">@(AppSettings.Terms.Title ?? AppSettings.Terms.TitleShort)</h5>
|
||||||
</div>
|
</div>
|
||||||
<div class="d-flex px-3 mt-3 justify-content-center text-center font-weight-bold">
|
<div class="d-flex px-3 justify-content-center text-center text-muted">
|
||||||
Hallo @CurrentUser.Name!
|
Hallo @CurrentUser.Name!
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -49,14 +49,21 @@
|
|||||||
|
|
||||||
@if (CurrentUser.IsAdmin())
|
@if (CurrentUser.IsAdmin())
|
||||||
{
|
{
|
||||||
<div class="nav-item px-3 pb-0">
|
<div class="nav-item px-3">
|
||||||
<div @onclick="NavLinkClickedAsync">
|
<div @onclick="NavLinkClickedAsync">
|
||||||
<NavLink class="nav-link" href="users" Match="NavLinkMatch.All">
|
<NavLink class="nav-link" href="users" Match="NavLinkMatch.All">
|
||||||
<span class="fas fa-users mr-2" aria-hidden="true" style="font-size: 1.4em;"></span> Benutzer
|
<span class="fas fa-users mr-2" aria-hidden="true" style="font-size: 1.4em;"></span> Benutzer
|
||||||
</NavLink>
|
</NavLink>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
|
||||||
|
<div class="nav-item px-3 pb-0">
|
||||||
|
<div @onclick="NavLinkClickedAsync">
|
||||||
|
<NavLink class="nav-link" href="settings" Match="NavLinkMatch.All">
|
||||||
|
<span class="fas fa-cog mr-2" aria-hidden="true" style="font-size: 1.4em;"></span> Einstellungen
|
||||||
|
</NavLink>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="nav-item px-3">
|
<div class="nav-item px-3">
|
||||||
<div @onclick="NavLinkClickedAsync">
|
<div @onclick="NavLinkClickedAsync">
|
||||||
@@ -65,6 +72,7 @@
|
|||||||
</NavLink>
|
</NavLink>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
<div class="flex-grow-1"></div>
|
<div class="flex-grow-1"></div>
|
||||||
|
|
||||||
@@ -84,5 +92,5 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="pb-1 text-center small">@($"v{Version ?? "0"}")</div>
|
<div class="pb-1 text-center small text-muted">@($"Version {Version ?? "0"}")</div>
|
||||||
</nav>
|
</nav>
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
nav {
|
|
||||||
padding-top: 50px;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 992px) {
|
|
||||||
nav {
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
nav .nav-logo {
|
|
||||||
background-image: url(img/logo_text.png);
|
|
||||||
height: 55px;
|
|
||||||
width: 200px;
|
|
||||||
margin: auto;
|
|
||||||
background-size: contain;
|
|
||||||
background-repeat: no-repeat;
|
|
||||||
background-position: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (min-width: 992px) {
|
|
||||||
nav .nav-logo {
|
|
||||||
height: 220px;
|
|
||||||
background-image: url(img/logo.png);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.oi {
|
|
||||||
width: 2rem;
|
|
||||||
font-size: 1.1rem;
|
|
||||||
vertical-align: text-top;
|
|
||||||
top: -2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item {
|
|
||||||
font-size: 0.9rem;
|
|
||||||
padding-bottom: 0.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item:first-of-type {
|
|
||||||
padding-top: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item:last-of-type {
|
|
||||||
padding-bottom: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item ::deep a {
|
|
||||||
color: #533a20;
|
|
||||||
border-radius: 4px;
|
|
||||||
height: 3rem;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
line-height: 3rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item ::deep a.active {
|
|
||||||
background-color: rgba(255, 255, 255, 0.25);
|
|
||||||
color: #64ae24;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item ::deep a:hover {
|
|
||||||
background-color: rgba(255, 255, 255, 0.1);
|
|
||||||
color: #64ae24;
|
|
||||||
}
|
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
{
|
{
|
||||||
"Kestrel": {
|
"Kestrel": {
|
||||||
"Endpoints": {
|
"Endpoints": {
|
||||||
"Http": {
|
"Http": {
|
||||||
@@ -9,10 +9,18 @@
|
|||||||
"DetailedErrors": true,
|
"DetailedErrors": true,
|
||||||
"Settings": {
|
"Settings": {
|
||||||
"TestMode": true,
|
"TestMode": true,
|
||||||
|
"Mail": {
|
||||||
|
"Host": "mail.example.com",
|
||||||
|
"Port": 587,
|
||||||
|
"Username": "your_username",
|
||||||
|
"Password": "your_password",
|
||||||
|
"FromAddress": "no-reply@example.com",
|
||||||
|
"UseSsl": true
|
||||||
|
},
|
||||||
"Terms": {
|
"Terms": {
|
||||||
"Title": "Foodsharing Musterhausen",
|
"Title": "Foodsharing Musterhausen",
|
||||||
"TitleShort": "Musterhausen",
|
"TitleShort": "Musterhausen",
|
||||||
"StepInName": "Krabbelgruppe2"
|
"StepInName": "Neulingstreffen"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
16
FoodsharingSiegen.Server/sasscompiler.json
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
{
|
||||||
|
"GenerateScopedCss": true,
|
||||||
|
"ScopedCssFolders": [
|
||||||
|
"Views",
|
||||||
|
"Pages",
|
||||||
|
"Shared",
|
||||||
|
"Components",
|
||||||
|
"Controls"
|
||||||
|
],
|
||||||
|
"Compilations" : [
|
||||||
|
{
|
||||||
|
"Source": "wwwroot/css/site.scss",
|
||||||
|
"Target": "wwwroot/css/site.css"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
BIN
FoodsharingSiegen.Server/wwwroot/apple-touch-icon.png
Normal file
|
After Width: | Height: | Size: 5.9 KiB |
@@ -1 +0,0 @@
|
|||||||
20260410
|
|
||||||
80
FoodsharingSiegen.Server/wwwroot/css/_utilities.scss
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
// Breakpoints matching typical Bootstrap breakpoints
|
||||||
|
$breakpoints: (
|
||||||
|
sm: 576px,
|
||||||
|
md: 768px,
|
||||||
|
lg: 992px,
|
||||||
|
xl: 1200px
|
||||||
|
);
|
||||||
|
|
||||||
|
// Spacer values (1 to 5) mapped to rem
|
||||||
|
$spacers: (
|
||||||
|
0: 0,
|
||||||
|
1: 0.25rem,
|
||||||
|
2: 0.5rem,
|
||||||
|
3: 1rem,
|
||||||
|
4: 1.5rem,
|
||||||
|
5: 3rem
|
||||||
|
);
|
||||||
|
|
||||||
|
@mixin generate-utilities($breakpoint: "") {
|
||||||
|
@each $size-key, $size-val in $spacers {
|
||||||
|
|
||||||
|
// Padding and Margin (all sides)
|
||||||
|
.p#{$breakpoint}-#{$size-key} { padding: $size-val !important; }
|
||||||
|
.m#{$breakpoint}-#{$size-key} { margin: $size-val !important; }
|
||||||
|
|
||||||
|
// Top
|
||||||
|
.pt#{$breakpoint}-#{$size-key} { padding-top: $size-val !important; }
|
||||||
|
.mt#{$breakpoint}-#{$size-key} { margin-top: $size-val !important; }
|
||||||
|
|
||||||
|
// Right / End
|
||||||
|
.pr#{$breakpoint}-#{$size-key} { padding-right: $size-val !important; }
|
||||||
|
.mr#{$breakpoint}-#{$size-key} { margin-right: $size-val !important; }
|
||||||
|
|
||||||
|
// Bottom
|
||||||
|
.pb#{$breakpoint}-#{$size-key} { padding-bottom: $size-val !important; }
|
||||||
|
.mb#{$breakpoint}-#{$size-key} { margin-bottom: $size-val !important; }
|
||||||
|
|
||||||
|
// Left / Start
|
||||||
|
.pl#{$breakpoint}-#{$size-key} { padding-left: $size-val !important; }
|
||||||
|
.ml#{$breakpoint}-#{$size-key} { margin-left: $size-val !important; }
|
||||||
|
|
||||||
|
// X-axis (left and right)
|
||||||
|
.px#{$breakpoint}-#{$size-key} {
|
||||||
|
padding-left: $size-val !important;
|
||||||
|
padding-right: $size-val !important;
|
||||||
|
}
|
||||||
|
.mx#{$breakpoint}-#{$size-key} {
|
||||||
|
margin-left: $size-val !important;
|
||||||
|
margin-right: $size-val !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Y-axis (top and bottom)
|
||||||
|
.py#{$breakpoint}-#{$size-key} {
|
||||||
|
padding-top: $size-val !important;
|
||||||
|
padding-bottom: $size-val !important;
|
||||||
|
}
|
||||||
|
.my#{$breakpoint}-#{$size-key} {
|
||||||
|
margin-top: $size-val !important;
|
||||||
|
margin-bottom: $size-val !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.align-items-center#{$breakpoint} {
|
||||||
|
align-items: center !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.align-items-start#{$breakpoint} {
|
||||||
|
align-items: flex-start !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 1. Generate default utilities (without breakpoint modifier)
|
||||||
|
@include generate-utilities("");
|
||||||
|
|
||||||
|
// 2. Generate utilities for each breakpoint using an @each loop
|
||||||
|
@each $bp-name, $bp-min-width in $breakpoints {
|
||||||
|
@media (min-width: $bp-min-width) {
|
||||||
|
@include generate-utilities("-#{$bp-name}");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,7 +1,10 @@
|
|||||||
|
@import 'utilities';
|
||||||
|
|
||||||
@import url('open-iconic/font/css/open-iconic-bootstrap.min.css');
|
@import url('open-iconic/font/css/open-iconic-bootstrap.min.css');
|
||||||
|
|
||||||
html, body {
|
html, body {
|
||||||
font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif;
|
font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif;
|
||||||
|
background-color: #f5ffed !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
h1:focus {
|
h1:focus {
|
||||||
@@ -66,3 +69,7 @@ a, .btn-link {
|
|||||||
.blazor-error-boundary::after {
|
.blazor-error-boundary::after {
|
||||||
content: "An error has occurred."
|
content: "An error has occurred."
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.badge .badge-close {
|
||||||
|
padding: 0 .3rem;
|
||||||
|
}
|
||||||
BIN
FoodsharingSiegen.Server/wwwroot/favicon-96x96.png
Normal file
|
After Width: | Height: | Size: 3.5 KiB |
|
Before Width: | Height: | Size: 5.3 KiB After Width: | Height: | Size: 15 KiB |
1
FoodsharingSiegen.Server/wwwroot/favicon.svg
Normal file
|
After Width: | Height: | Size: 62 KiB |
21
FoodsharingSiegen.Server/wwwroot/site.webmanifest
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
{
|
||||||
|
"name": "Foodsharing Einarbeitungen",
|
||||||
|
"short_name": "FS Onboarding",
|
||||||
|
"icons": [
|
||||||
|
{
|
||||||
|
"src": "/web-app-manifest-192x192.png",
|
||||||
|
"sizes": "192x192",
|
||||||
|
"type": "image/png",
|
||||||
|
"purpose": "maskable"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "/web-app-manifest-512x512.png",
|
||||||
|
"sizes": "512x512",
|
||||||
|
"type": "image/png",
|
||||||
|
"purpose": "maskable"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"theme_color": "#e8ffe2",
|
||||||
|
"background_color": "#f2ffe9",
|
||||||
|
"display": "standalone"
|
||||||
|
}
|
||||||
BIN
FoodsharingSiegen.Server/wwwroot/web-app-manifest-192x192.png
Normal file
|
After Width: | Height: | Size: 7.5 KiB |
BIN
FoodsharingSiegen.Server/wwwroot/web-app-manifest-512x512.png
Normal file
|
After Width: | Height: | Size: 26 KiB |
@@ -45,6 +45,10 @@ namespace FoodsharingSiegen.Shared.Helper
|
|||||||
if (filter.DeletedOnly)
|
if (filter.DeletedOnly)
|
||||||
filterListQ = filterListQ.Where(x => x.RecordState == RecordState.Archived);
|
filterListQ = filterListQ.Where(x => x.RecordState == RecordState.Archived);
|
||||||
|
|
||||||
|
// Show only prospects with possible IdCheck
|
||||||
|
if (filter.IdCheckPossible)
|
||||||
|
filterListQ = filterListQ.Where(x => x.Images != null && x.Images.Count > 0);
|
||||||
|
|
||||||
// No Activity Filter
|
// No Activity Filter
|
||||||
if (filter.NoActivity)
|
if (filter.NoActivity)
|
||||||
filterListQ = filterListQ.Where(x => DateTime.Now - x.Modified > TimeSpan.FromDays(180));
|
filterListQ = filterListQ.Where(x => DateTime.Now - x.Modified > TimeSpan.FromDays(180));
|
||||||
@@ -56,6 +60,20 @@ namespace FoodsharingSiegen.Shared.Helper
|
|||||||
return filterListQ.ToList();
|
return filterListQ.ToList();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static List<Prospect> ApplySort(this List<Prospect>? prospectList, ProspectSortOption sortOption)
|
||||||
|
{
|
||||||
|
if (prospectList == null) return [];
|
||||||
|
|
||||||
|
return sortOption switch
|
||||||
|
{
|
||||||
|
ProspectSortOption.NameAscending => prospectList.OrderBy(x => x.Name, StringComparer.OrdinalIgnoreCase).ToList(),
|
||||||
|
ProspectSortOption.NameDescending => prospectList.OrderByDescending(x => x.Name, StringComparer.OrdinalIgnoreCase).ToList(),
|
||||||
|
ProspectSortOption.ModifiedAscending => prospectList.OrderBy(x => x.Modified ?? x.Created).ToList(),
|
||||||
|
ProspectSortOption.ModifiedDescending => prospectList.OrderByDescending(x => x.Modified ?? x.Created).ToList(),
|
||||||
|
_ => prospectList
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||