Settings so halb und a bissi aufgeräumt ...

This commit is contained in:
2024-10-20 17:58:26 +02:00
parent fbd650c174
commit 996dbadaf1
27 changed files with 707 additions and 931 deletions

View File

@@ -1,100 +1,68 @@
using CommunityToolkit.Mvvm.ComponentModel;
using CommunityToolkit.Mvvm.Input;
using Jugenddienst_Stunden.Models;
using Jugenddienst_Stunden.Types;
using System.Collections.ObjectModel;
using System.Windows.Input;
using static System.Runtime.InteropServices.JavaScript.JSType;
namespace Jugenddienst_Stunden.ViewModels;
internal class StundeViewModel : ObservableObject, IQueryAttributable
{
public int id { get; set; }
public class StundeViewModel : ObservableObject, IQueryAttributable {
public int Id { get; set; }
public string Title { get; set; } = "Eintrag bearbeiten";
public string SubTitle { get; set; } = DateTime.Today.ToString("dddd, d. MMMM yyyy");
private Settings _settings;
public Settings Settings
{
get => _settings; set
{
if (_settings != value)
{
_settings = value;
}
}
}
private DayTime _stunde;
public DayTime Stunde
{
get => _stunde;
set
{
if (_stunde != value)
{
_stunde = value;
OnPropertyChanged(nameof(Stunde));
}
}
}
public string Title { get; set; } = "Eintrag bearbeiten";
private HoursBase HoursBase = new HoursBase();
internal Settings Settings = new Settings();
public event EventHandler<string> AlertEvent;
public event EventHandler<string> InfoEvent;
public event Func<string, string, Task<bool>> ConfirmEvent;
/// <summary>
/// Gemeinden für die Auswahlliste
/// </summary>
public List<Gemeinde> OptionsGemeinde { get; private set; }
public ObservableCollection<Gemeinde> OptionsGemeinde { get; private set; }
public ObservableCollection<Projekt> OptionsProjekt { get; private set; }
public ObservableCollection<Freistellung> OptionsFreistellung { get; private set; }
public ObservableCollection<DayTime> DayTimes { get; set; }
/// <summary>
/// Projekte für die Auswahlliste
/// </summary>
public List<Projekt> OptionsProjekt { get; private set; }
//private Gemeinde _selectedGemeinde;
public Gemeinde SelectedOptionGemeinde
{
get => _stunde.GemeindeAktiv;
set
{
if (_stunde.GemeindeAktiv != value)
{
//_selectedGemeinde = value;
_stunde.GemeindeAktiv = value;
OnPropertyChanged(nameof(SelectedOptionGemeinde));
}
/// <summary>
/// Freistellungen für die Auswahlliste
/// </summary>
public List<Freistellung> OptionsFreistellung { get; private set; }
/// <summary>
/// Vorhandene Zeiten anzeigen, wenn neuer Eintrag erstellt wird
/// </summary>
public List<DayTime> DayTimes { get; set; }
private DayTime _dayTime;
/// <summary>
/// Aktueller Stundeneintrag
/// </summary>
public DayTime DayTime {
get => _dayTime;
set {
if (_dayTime != value) {
_dayTime = value;
}OnPropertyChanged(nameof(DayTime));
}
}
//private Projekt _selectedProjekt;
public Projekt SelectedOptionProjekt
{
get => _stunde.ProjektAktiv;
set
{
if (_stunde.ProjektAktiv != value)
{
//_selectedProjekt = value;
_stunde.ProjektAktiv = value;
OnPropertyChanged(nameof(SelectedOptionProjekt));
}
}
}
//private Freistellung _selectedFreistellung;
public Freistellung SelectedOptionFreistellung
{
get => _stunde.FreistellungAktiv;
set
{
if (_stunde.FreistellungAktiv != value)
{
_stunde.FreistellungAktiv = value;
OnPropertyChanged(nameof(SelectedOptionFreistellung));
}
}
}
/// <summary>
/// Dürfen Gemeinden verwendet werden?
/// </summary>
public bool GemeindeAktivSet { get; set; }
/// <summary>
/// Dürfen Projekte verwendet werden?
/// </summary>
public bool ProjektAktivSet { get; set; }
@@ -104,156 +72,131 @@ internal class StundeViewModel : ObservableObject, IQueryAttributable
//public ICommand LoadDataCommand { get; private set; }
public StundeViewModel()
{
_stunde = new DayTime();
public StundeViewModel() {
DayTime = new DayTime();
SaveCommand = new AsyncRelayCommand(Save);
//DeleteCommand = new AsyncRelayCommand(Delete);
DeleteConfirmCommand = new Command(async () => await DeleteConfirm());
LoadSettingsAsync();
}
public StundeViewModel(DayTime stunde)
{
_stunde = stunde;
public StundeViewModel(DayTime stunde) {
DayTime = new DayTime();
SaveCommand = new AsyncRelayCommand(Save);
DeleteConfirmCommand = new AsyncRelayCommand(DeleteConfirm);
LoadSettingsAsync();
}
private async Task LoadData()
{
try
{
Hours _hours = await Models.Stunde.LoadBasicData();
OptionsProjekt = new ObservableCollection<Projekt>(_hours.Projekte);
OptionsGemeinde = new ObservableCollection<Gemeinde>(_hours.Gemeinden);
OptionsFreistellung = new ObservableCollection<Freistellung>(_hours.Freistellungen);
OnPropertyChanged(nameof(OptionsGemeinde));
OnPropertyChanged(nameof(OptionsProjekt));
OnPropertyChanged(nameof(OptionsFreistellung));
_stunde.EmployeeId = _hours.EmployeeId;
}
catch (Exception e)
{
private async void LoadSettingsAsync() {
Settings = await HoursBase.LoadSettings();
OptionsGemeinde = Settings.Gemeinden;
OptionsProjekt = Settings.Projekte;
OptionsFreistellung = Settings.Freistellungen;
GemeindeAktivSet = Settings.GemeindeAktivSet;
ProjektAktivSet = Settings.ProjektAktivSet;
OnPropertyChanged(nameof(OptionsGemeinde));
OnPropertyChanged(nameof(OptionsFreistellung));
OnPropertyChanged(nameof(OptionsProjekt));
OnPropertyChanged(nameof(GemeindeAktivSet));
OnPropertyChanged(nameof(ProjektAktivSet));
}
private async Task LoadData() {
try {
Hours _hours = await Models.HoursBase.LoadBasicData();
DayTime.EmployeeId = _hours.EmployeeId;
} catch (Exception e) {
AlertEvent?.Invoke(this, e.Message);
}
}
async Task Save()
{
async Task Save() {
bool exceptionOccurred = false;
try
{
await Models.Stunde.SaveEntry(_stunde);
}
catch (Exception e)
{
try {
await Models.HoursBase.SaveEntry(DayTime);
} catch (Exception e) {
AlertEvent?.Invoke(this, e.Message);
exceptionOccurred = true;
}
if (!exceptionOccurred)
{
if (_stunde.id != null)
{
await Shell.Current.GoToAsync($"..?saved={_stunde.id}");
}
else
{
await Shell.Current.GoToAsync($"..?date={_stunde.day.ToString("yyyy-MM-dd")}");
if (!exceptionOccurred) {
if (DayTime.Id != null) {
await Shell.Current.GoToAsync($"..?saved={DayTime.Id}");
} else {
await Shell.Current.GoToAsync($"..?date={DayTime.Day.ToString("yyyy-MM-dd")}");
}
}
}
private async Task Delete()
{
await Models.Stunde.DeleteEntry(_stunde);
await Shell.Current.GoToAsync($"..?date={_stunde.day.ToString("yyyy-MM-dd")}");
private async Task Delete() {
await Models.HoursBase.DeleteEntry(DayTime);
await Shell.Current.GoToAsync($"..?date={DayTime.Day.ToString("yyyy-MM-dd")}");
}
private async Task DeleteConfirm()
{
if (ConfirmEvent != null)
{
private async Task DeleteConfirm() {
if (ConfirmEvent != null) {
bool answer = await ConfirmEvent.Invoke("Achtung", "Löschen kann nicht ungeschehen gemacht werden. Fortfahren?");
if (answer)
{
if (answer) {
//Löschen
await Models.Stunde.DeleteEntry(_stunde);
await Shell.Current.GoToAsync($"..?date={_stunde.day.ToString("yyyy-MM-dd")}");
}
else
{ //nicht Löschen
await Models.HoursBase.DeleteEntry(DayTime);
await Shell.Current.GoToAsync($"..?date={DayTime.Day.ToString("yyyy-MM-dd")}");
} else { //nicht Löschen
}
}
}
private async Task LoadSettings()
{
Settings = await Models.Stunde.LoadSettings();
}
/// <summary>
/// Anwenden der Query-Parameter
/// </summary>
async void IQueryAttributable.ApplyQueryAttributes(IDictionary<string, object> query)
{
await LoadSettings();
async void IQueryAttributable.ApplyQueryAttributes(IDictionary<string, object> query) {
var probe = query;
if (query.ContainsKey("load"))
{
if (query.ContainsKey("load")) {
//DateTime heute = DateTime.Now;
Stunde = await Models.Stunde.LoadEntry(Convert.ToInt32(query["load"]));
if (System.String.IsNullOrEmpty(Stunde.description))
{
_dayTime = await Models.HoursBase.LoadEntry(Convert.ToInt32(query["load"]));
if (System.String.IsNullOrEmpty(DayTime.Description)) {
AlertEvent?.Invoke(this, "Eintrag hat keine Daten zurückgegeben");
}
SubTitle = Stunde.day.ToString("dddd, d. MMMM yyyy");
SubTitle = DayTime.Day.ToString("dddd, d. MMMM yyyy");
OptionsProjekt = new ObservableCollection<Projekt>(Stunde.Projekte);
OptionsGemeinde = new ObservableCollection<Gemeinde>(Stunde.Gemeinden);
OptionsFreistellung = new ObservableCollection<Freistellung>(Stunde.Freistellungen);
OnPropertyChanged(nameof(OptionsGemeinde));
OnPropertyChanged(nameof(OptionsProjekt));
OnPropertyChanged(nameof(OptionsFreistellung));
//OptionsProjekt.FirstOrDefault(x => x.Id == _stunde.projekt);
_dayTime.GemeindeAktiv = OptionsGemeinde.FirstOrDefault(Gemeinde => Gemeinde.Id == DayTime.Gemeinde) ?? new Gemeinde();
SelectedOptionGemeinde = OptionsGemeinde.FirstOrDefault(item => item.Id == Stunde.gemeinde) ?? new Gemeinde();
OnPropertyChanged(nameof(SelectedOptionGemeinde));
_dayTime.ProjektAktiv = OptionsProjekt.FirstOrDefault(Projekt => Projekt.Id == DayTime.Projekt) ?? new Projekt();
SelectedOptionProjekt = OptionsProjekt.FirstOrDefault(Projekt => Projekt.Id == Stunde.projekt) ?? new Projekt();
OnPropertyChanged(nameof(SelectedOptionProjekt));
SelectedOptionFreistellung = OptionsFreistellung.FirstOrDefault(Freistellung => Freistellung.Id == Stunde.free) ?? new Freistellung();
OnPropertyChanged(nameof(SelectedOptionFreistellung));
_dayTime.FreistellungAktiv = OptionsFreistellung.FirstOrDefault(Freistellung => Freistellung.Id == DayTime.Free) ?? new Freistellung();
OnPropertyChanged(nameof(DayTime));
OnPropertyChanged(nameof(SubTitle));
}
if (query.ContainsKey("date"))
{
if (query.ContainsKey("date")) {
Title = "Neuer Eintrag";
DateTime _date = DateTime.ParseExact((string)query["date"], "yyyy-MM-dd", System.Globalization.CultureInfo.InvariantCulture);
//Bei neuem Eintrag die vorhandenen des gleichen Tages anzeigen
try
{
DayTimes = await Models.Stunde.LoadDay(_date);
}
catch (Exception)
{
try {
DayTimes = await HoursBase.LoadDay(_date);
} catch (Exception) {
//Ein Tag ohne Einträge gibt eine Fehlermeldung,
//die soll aber ignoriert werden, weil beim Neueintrag ist das ja Wurscht
}
_stunde.day = _date;
DayTime.Day = _date;
SubTitle = _date.ToString("dddd, d. MMMM yyyy");
_ = LoadData();

View File

@@ -1,5 +1,6 @@
using CommunityToolkit.Mvvm.ComponentModel;
using CommunityToolkit.Mvvm.Input;
using Jugenddienst_Stunden.Models;
using Jugenddienst_Stunden.Types;
using Newtonsoft.Json.Linq;
using System.Collections.ObjectModel;
@@ -10,15 +11,12 @@ using System.Windows.Input;
namespace Jugenddienst_Stunden.ViewModels;
internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyPropertyChanged
{
public string Name => AppInfo.Name;
public string Surname => AppInfo.VersionString;
public string MoreInfoUrl => "https://aka.ms/maui";
public string Message => "Hier werden deine geleisteten Arbeitsstunden aufgelistet";
/// <summary>
/// Stundenliste
/// </summary>
internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyPropertyChanged {
public string LoadOverview => "Lade Summen für " + DateTime.Today.ToString("MMMM");
//public static DateTime GetDay = DateTime.Today;
//public string ShowDay => "Zeit an Tag " + GetDay.ToString("ddd d. MMM") + ": ";
public ICommand NewEntryCommand { get; }
public ICommand SelectEntryCommand { get; }
@@ -30,32 +28,19 @@ internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyP
public event EventHandler<string> AlertEvent;
public event EventHandler<string> InfoEvent;
private Settings Settings { get; set; }
private HoursBase HoursBase = new HoursBase();
internal Settings Settings = new Settings();
private bool isRefreshing;
public bool IsRefreshing
{
get => isRefreshing;
set
{
if (isRefreshing != value)
{
isRefreshing = value;
OnPropertyChanged();
}
}
}
private string _title = Preferences.Default.Get("name", "") + " " + Preferences.Default.Get("surname", "");
public string Title
{
private string _title = HoursBase.name + " " + HoursBase.surname;
public string Title {
get => _title;
set => SetProperty(ref _title, value);
}
private Hours _hour;
public Hours Hours
{
public Hours Hours {
get => _hour;
}
@@ -67,26 +52,26 @@ internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyP
/// <summary>
/// Liste der Tageszeiten
/// </summary>
private ObservableCollection<DayTime> _dayTimes = new ObservableCollection<DayTime>();
public ObservableCollection<DayTime> DayTimes
{
private List<DayTime> _dayTimes = new List<DayTime>();
public List<DayTime> DayTimes {
get => _dayTimes;
set => SetProperty(ref _dayTimes, value);
set {
SetProperty(ref _dayTimes, value);
OnPropertyChanged(nameof(DayTimes));
}
}
/// <summary>
/// Mindest-Datum für den Datepicker
/// </summary>
public DateTime MinimumDate
{
public DateTime MinimumDate {
get => DateTime.Today.AddDays(-365);
}
/// <summary>
/// Höchst-Datum für den Datepicker
/// </summary>
public DateTime MaximumDate
{
public DateTime MaximumDate {
get => DateTime.Today.AddDays(60);
}
@@ -94,13 +79,10 @@ internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyP
/// Heutiges Datum, wenn das Datum geändert wird, wird auch der Tag geladen
/// </summary>
private DateTime dateToday = DateTime.Today;
public DateTime DateToday
{
public DateTime DateToday {
get => dateToday;
set
{
if (dateToday != value)
{
set {
if (dateToday != value) {
dateToday = value;
//GetDay = value;
//OnPropertyChanged();
@@ -113,56 +95,69 @@ internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyP
/// <summary>
/// Monatsübersicht: Geleistete Stunden
/// </summary>
public string? ZeitCalculated
{
get => _hour.zeit_total;
public string? ZeitCalculated {
get => Hours.Zeit_total;
}
/// <summary>
/// Monatsübersicht: Sollstunden
/// </summary>
public string? Nominal
{
get => _hour.nominal;
public string? Nominal {
get => Hours.Nominal;
}
/// <summary>
/// Monatsübersicht: Differenz zwischen Soll und geleisteten Stunden
/// </summary>
public string? Overtime
{
get => _hour.overtime;
public string? Overtime {
get => Hours.overtime;
}
/// <summary>
/// Monatsübersicht: Restüberstunden insgesamt
/// </summary>
public string OvertimeMonth
{
get => _hour.overtime_month;
public string OvertimeMonth {
get => Hours.overtime_month;
}
/// <summary>
/// Monatsübersicht: Resturlaub
/// </summary>
public string Holiday
{
get => _hour.holiday;
public string Holiday {
get => Hours.holiday;
}
/// <summary>
/// Seite neu laden
/// </summary>
private bool isRefreshing;
public bool IsRefreshing {
get => isRefreshing;
set {
if (isRefreshing != value) {
isRefreshing = value;
OnPropertyChanged();
}
}
}
/// <summary>
/// Dürfen Gemeinden verwendet werden?
/// </summary>
public bool GemeindeAktivSet { get; set; }
/// <summary>
/// Dürfen Projekte verwendet werden?
/// </summary>
public bool ProjektAktivSet { get; set; }
/// <summary>
/// CTOR
/// </summary>
public StundenViewModel()
{
_hour = new Types.Hours();
public StundenViewModel() {
_hour = new Hours();
LoadDataCommand = new AsyncRelayCommand(LoadData);
NewEntryCommand = new AsyncRelayCommand(NewEntryAsync);
@@ -171,18 +166,24 @@ internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyP
RefreshCommand = new Command(async () => await RefreshItemsAsync());
Task task = LoadDay(DateTime.Today);
LoadSettingsAsync();
}
private async void LoadSettingsAsync() {
Settings = await HoursBase.LoadSettings();
GemeindeAktivSet = Settings.GemeindeAktivSet;
ProjektAktivSet = Settings.ProjektAktivSet;
OnPropertyChanged(nameof(GemeindeAktivSet));
OnPropertyChanged(nameof(ProjektAktivSet));
}
/// <summary>
/// Öffnet eine neue Stundeneingabe
/// </summary>
/// <returns></returns>
private async Task NewEntryAsync()
{
private async Task NewEntryAsync() {
//Hier muss das Datum übergeben werden
//await Shell.Current.GoToAsync(nameof(Views.StundePage));
await Shell.Current.GoToAsync($"{nameof(Views.StundePage)}?date={dateToday:yyyy-MM-dd}");
@@ -191,99 +192,72 @@ internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyP
/// <summary>
/// Öffnet eine bestehende Stundeneingabe
/// </summary>
/// <param name="entry"></param>
/// <returns></returns>
private async Task SelectEntryAsync(DayTime entry)
{
if (entry != null && entry.id != null)
{
private async Task SelectEntryAsync(DayTime entry) {
if (entry != null && entry.Id != null) {
//var navigationParameters = new Dictionary<string, object> { { "load", entry.id } };
//await Shell.Current.GoToAsync($"{nameof(Views.StundePage)}", navigationParameters);
await Shell.Current.GoToAsync($"{nameof(Views.StundePage)}?load={entry.id}");
}
else AlertEvent?.Invoke(this, "Auswahl enthält keine Daten");
await Shell.Current.GoToAsync($"{nameof(Views.StundePage)}?load={entry.Id}");
} else AlertEvent?.Invoke(this, "Auswahl enthält keine Daten");
}
private async Task RefreshList()
{
private async Task RefreshList() {
OnPropertyChanged(nameof(DayTimes));
}
/// <summary>
/// Lädt die Monatssummen für die Übersicht
/// </summary>
/// <returns></returns>
private async Task LoadData()
{
try
{
_hour = await Models.Stunde.LoadData();
private async Task LoadData() {
try {
_hour = await HoursBase.LoadData();
RefreshProperties();
}
catch (Exception e)
{
} catch (Exception e) {
AlertEvent?.Invoke(this, e.Message);
}
}
/// <summary>
/// Lädt die Arbeitszeiten für einen Tag
/// </summary>
/// <param name="date"></param>
/// <returns></returns>
public async Task LoadDay(DateTime date)
{
public async Task LoadDay(DateTime date) {
DayTotal = new TimeOnly(0);
try
{
await LoadSettings();
DayTimes = await Models.Stunde.LoadDay(date);
LoadSettingsAsync();
try {
DayTimes = await HoursBase.LoadDay(date);
//TODO: Hier muss noch die Berechnung der Stunden erfolgen
//Es werden im Moment nur die eingetragenen Stunden gezählt
//Auf der Website bekommt der Benutzer hingegen die berechneten Stunden angezeigt (Nachstunden außerhalb des Stundenplanes zählen mehr ...)
TimeSpan span = TimeSpan.Zero;
foreach (DayTime dt in DayTimes)
{
span += dt.end - dt.begin;
foreach (DayTime dt in DayTimes) {
span += dt.End - dt.Begin;
}
DayTotal = TimeOnly.FromTimeSpan(span);
}
catch (Exception e)
{
DayTimes = new ObservableCollection<DayTime>();
} catch (Exception e) {
DayTimes = new List<DayTime>();
//TODO: hier könnte auch ein Fehler kommen, dann wäre InfoEvent falsch.
InfoEvent?.Invoke(this, e.Message);
}
finally
{
} finally {
OnPropertyChanged(nameof(DayTotal));
//OnPropertyChanged(nameof(DayTimes));
}
}
private async Task LoadSettings()
{
Settings = await Models.Stunde.LoadSettings();
}
async void IQueryAttributable.ApplyQueryAttributes(IDictionary<string, object> query)
{
if (query.ContainsKey("date"))
{
async void IQueryAttributable.ApplyQueryAttributes(IDictionary<string, object> query) {
if (query.ContainsKey("date")) {
await LoadDay(Convert.ToDateTime(query["date"]));
}
}
private async Task RefreshItemsAsync()
{
private async Task RefreshItemsAsync() {
IsRefreshing = true;
// Fügen Sie hier die Logik zum Aktualisieren der Daten hinzu
@@ -296,8 +270,7 @@ internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyP
/// <summary>
/// Refreshes all properties
/// </summary>
private void RefreshProperties()
{
private void RefreshProperties() {
OnPropertyChanged(nameof(Nominal));
OnPropertyChanged(nameof(Overtime));
OnPropertyChanged(nameof(OvertimeMonth));
@@ -309,14 +282,10 @@ internal class StundenViewModel : ObservableObject, IQueryAttributable, INotifyP
OnPropertyChanged(nameof(MaximumDate));
}
protected void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
try
{
protected void OnPropertyChanged([CallerMemberName] string propertyName = null) {
try {
base.OnPropertyChanged(new PropertyChangedEventArgs(propertyName));
}
catch (Exception ex)
{
} catch (Exception ex) {
AlertEvent?.Invoke(this, ex.Message);
//Console.WriteLine($"Fehler bei OnPropertyChanged: {ex.Message}");
}