[feat:] frontend to edit courses (ai-slop)

This commit is contained in:
2026-06-01 17:00:55 +02:00
parent c46417c56d
commit c5be9d2c6e
2 changed files with 186 additions and 8 deletions
+21 -3
View File
@@ -196,8 +196,20 @@
</Grid> </Grid>
<Line /> <Line />
<Grid ColumnDefinitions="*,3*"> <Grid ColumnDefinitions="*,3*">
<Label Content="Maximale Sportkursanzahl pro Semester"></Label> <Label Content="Maximale Sportkursanzahl Semester 1"></Label>
<NumericUpDown Grid.Column="1" x:Name="NudSportMaxPerSemester" ValueChanged="NudSportMaxPerSemester_OnValueChanged"></NumericUpDown> <NumericUpDown Grid.Column="1" x:Name="NudSportMaxPerSemester1" ValueChanged="NudSportMaxPerSemester1_OnValueChanged"></NumericUpDown>
</Grid>
<Grid ColumnDefinitions="*,3*">
<Label Content="Maximale Sportkursanzahl Semester 2"></Label>
<NumericUpDown Grid.Column="1" x:Name="NudSportMaxPerSemester2" ValueChanged="NudSportMaxPerSemester2_OnValueChanged"></NumericUpDown>
</Grid>
<Grid ColumnDefinitions="*,3*">
<Label Content="Maximale Sportkursanzahl Semester 3"></Label>
<NumericUpDown Grid.Column="1" x:Name="NudSportMaxPerSemester3" ValueChanged="NudSportMaxPerSemester3_OnValueChanged"></NumericUpDown>
</Grid>
<Grid ColumnDefinitions="*,3*">
<Label Content="Maximale Sportkursanzahl Semester 4"></Label>
<NumericUpDown Grid.Column="1" x:Name="NudSportMaxPerSemester4" ValueChanged="NudSportMaxPerSemester4_OnValueChanged"></NumericUpDown>
</Grid> </Grid>
</StackPanel> </StackPanel>
</Grid> </Grid>
@@ -211,7 +223,13 @@
</StackPanel> </StackPanel>
</TabItem.Header> </TabItem.Header>
<Grid ColumnDefinitions="*,*" RowDefinitions="50,2*,*"> <Grid ColumnDefinitions="*,*" RowDefinitions="50,2*,*">
<ListBox Grid.RowSpan="2" x:Name="LbResult" Margin="0,10,10,10"></ListBox> <ListBox Grid.RowSpan="2" x:Name="LbResult" Margin="0,10,10,10" PointerPressed="LbResult_OnPointerPressed">
<ListBox.ContextMenu>
<ContextMenu>
<MenuItem Header="Ändern" x:Name="MnuChange" />
</ContextMenu>
</ListBox.ContextMenu>
</ListBox>
<Grid Grid.Row="0" Grid.Column="1" Grid.ColumnDefinitions="*,*"> <Grid Grid.Row="0" Grid.Column="1" Grid.ColumnDefinitions="*,*">
<Button Grid.Column="0" Margin="0,10,0,0" x:Name="BtnExportCoursePDF" VerticalAlignment="Top" Height="35" HorizontalAlignment="Stretch" Click="BtnExportCoursePDF_OnClick" HorizontalContentAlignment="Center"> <Button Grid.Column="0" Margin="0,10,0,0" x:Name="BtnExportCoursePDF" VerticalAlignment="Top" Height="35" HorizontalAlignment="Stretch" Click="BtnExportCoursePDF_OnClick" HorizontalContentAlignment="Center">
<StackPanel Orientation="Horizontal"> <StackPanel Orientation="Horizontal">
+164 -4
View File
@@ -1,6 +1,7 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Diagnostics; using System.Diagnostics;
using System.Linq;
using Avalonia.Controls; using Avalonia.Controls;
using Avalonia.Input; using Avalonia.Input;
using Avalonia.Interactivity; using Avalonia.Interactivity;
@@ -12,6 +13,26 @@ public partial class MainWindow : Window
{ {
public static MainWindow Instance { get; set; } public static MainWindow Instance { get; set; }
public static string ApplicationVersion = "v1.2.24"; public static string ApplicationVersion = "v1.2.24";
private sealed class ResultEntry
{
public Student Student { get; }
public int Semester { get; }
public string CourseName { get; }
public ResultEntry(Student student, int semester, string? courseName)
{
Student = student;
Semester = semester;
CourseName = courseName ?? string.Empty;
}
public override string ToString()
{
return $"{Student.Name} ({Student.ID}) - {Semester}. Semester: {CourseName}";
}
}
public MainWindow() public MainWindow()
{ {
InitializeComponent(); InitializeComponent();
@@ -20,10 +41,108 @@ public partial class MainWindow : Window
RefreshCoursesList(); RefreshCoursesList();
try try
{ {
NudSportMaxPerSemester.Value = Settings.Instance.NumCoursesPerSemester; NudSportMaxPerSemester1.Value = Settings.Instance.NumCoursesPerSemester[0];
NudSportMaxPerSemester2.Value = Settings.Instance.NumCoursesPerSemester[1];
NudSportMaxPerSemester3.Value = Settings.Instance.NumCoursesPerSemester[2];
NudSportMaxPerSemester4.Value = Settings.Instance.NumCoursesPerSemester[3];
} catch {} } catch {}
} }
private void RegenerateContextMenu()
{
MnuChange.Items.Clear();
foreach (var sport in Settings.Instance.Sports)
{
var item = new MenuItem { Header = sport.Name };
item.Click += (_, _) => ChangeStudentCourse(sport);
MnuChange.Items.Add(item);
}
}
private void ChangeStudentCourse(Sport targetSport)
{
if (LbResult.SelectedItem is not ResultEntry selectedEntry)
return;
try
{
if (ApplyStudentCourseChange(selectedEntry.Student, selectedEntry.Semester, targetSport))
{
CourseCrafter.ReloadResult();
RefreshResultView();
RegenerateContextMenu();
}
}
catch (Exception ex)
{
Console.WriteLine(ex.Message + "\n" + ex.StackTrace);
}
}
private bool ApplyStudentCourseChange(Student student, int semester, Sport targetSport)
{
if (semester < 1 || semester > 4)
return false;
var semesterCourses = CourseCrafter.GeneratedCourses
.Where(course => course.Semester == semester)
.ToList();
var currentCourses = semesterCourses
.Where(course => course.Instance.Students.Contains(student.ID))
.ToList();
CourseCrafter.CourseInstance? targetCourse = currentCourses
.FirstOrDefault(course => course.Instance.Sport.Name == targetSport.Name)
.Instance;
if (targetCourse == null)
{
targetCourse = semesterCourses
.Where(course => course.Instance.Sport.Name == targetSport.Name)
.OrderBy(course => course.Instance.Students.Count)
.Select(course => course.Instance)
.FirstOrDefault();
}
bool changed = false;
foreach (var course in currentCourses)
{
if (targetCourse != null && ReferenceEquals(course.Instance, targetCourse))
continue;
if (course.Instance.Students.Remove(student.ID))
changed = true;
}
int removedEmptyCourses = CourseCrafter.GeneratedCourses.RemoveAll(course =>
course.Semester == semester &&
course.Instance.Students.Count == 0 &&
!ReferenceEquals(course.Instance, targetCourse));
if (removedEmptyCourses > 0)
changed = true;
if (targetCourse == null)
{
var newCourse = new CourseCrafter.CourseInstance
{
Sport = targetSport,
Students = new List<string> { student.ID }
};
CourseCrafter.GeneratedCourses.Add((semester, newCourse));
changed = true;
}
else if (!targetCourse.Students.Contains(student.ID))
{
targetCourse.Students.Add(student.ID);
changed = true;
}
return changed;
}
private void MnuExpSettings_OnClick(object? sender, RoutedEventArgs e) private void MnuExpSettings_OnClick(object? sender, RoutedEventArgs e)
{ {
var res = MessageBox.Show(this, "Dieses Feature ist noch nicht implementiert", "Fehlend"); var res = MessageBox.Show(this, "Dieses Feature ist noch nicht implementiert", "Fehlend");
@@ -159,6 +278,21 @@ public partial class MainWindow : Window
//TbiResults.Focus(); //TbiResults.Focus();
} }
private void LbResult_OnPointerPressed(object? sender, PointerPressedEventArgs e)
{
if (!e.GetCurrentPoint(LbResult).Properties.IsRightButtonPressed)
return;
if (e.Source is Control control && control.DataContext is ResultEntry entry)
{
LbResult.SelectedItem = entry;
}
else
{
LbResult.SelectedItem = null;
}
}
private void RefreshResultView() private void RefreshResultView()
{ {
LbResult.Items.Clear(); LbResult.Items.Clear();
@@ -168,7 +302,7 @@ public partial class MainWindow : Window
{ {
for(int i = 0; i<s.Result.Length;i++) for(int i = 0; i<s.Result.Length;i++)
{ {
LbResult.Items.Add($"{s.Name} ({s.ID}) - {i+1}. Semester: {s.Result[i]}"); LbResult.Items.Add(new ResultEntry(s, i + 1, s.Result[i]));
} }
} }
catch (Exception ex) catch (Exception ex)
@@ -262,6 +396,7 @@ public partial class MainWindow : Window
{ {
LbSportCourses.Items.Add(sp); LbSportCourses.Items.Add(sp);
} }
RegenerateContextMenu();
} }
@@ -298,6 +433,7 @@ public partial class MainWindow : Window
try try
{ {
((Sport)LbSportCourses.SelectedItem).Name = TbSportName.Text; ((Sport)LbSportCourses.SelectedItem).Name = TbSportName.Text;
RegenerateContextMenu();
} catch {} } catch {}
} }
@@ -416,11 +552,35 @@ public partial class MainWindow : Window
// Export as PDF // Export as PDF
} }
private void NudSportMaxPerSemester_OnValueChanged(object? sender, NumericUpDownValueChangedEventArgs e) private void NudSportMaxPerSemester1_OnValueChanged(object? sender, NumericUpDownValueChangedEventArgs e)
{ {
try try
{ {
Settings.Instance.NumCoursesPerSemester = Convert.ToInt32(NudSportMaxPerSemester.Value); Settings.Instance.NumCoursesPerSemester[0] = Convert.ToInt32(NudSportMaxPerSemester1.Value);
} catch {}
}
private void NudSportMaxPerSemester2_OnValueChanged(object? sender, NumericUpDownValueChangedEventArgs e)
{
try
{
Settings.Instance.NumCoursesPerSemester[1] = Convert.ToInt32(NudSportMaxPerSemester2.Value);
} catch {}
}
private void NudSportMaxPerSemester3_OnValueChanged(object? sender, NumericUpDownValueChangedEventArgs e)
{
try
{
Settings.Instance.NumCoursesPerSemester[2] = Convert.ToInt32(NudSportMaxPerSemester3.Value);
} catch {}
}
private void NudSportMaxPerSemester4_OnValueChanged(object? sender, NumericUpDownValueChangedEventArgs e)
{
try
{
Settings.Instance.NumCoursesPerSemester[3] = Convert.ToInt32(NudSportMaxPerSemester4.Value);
} catch {} } catch {}
} }