31 Commits

Author SHA1 Message Date
fierke 08db1eb681 [fix:] first semester priority 2026-06-07 20:31:30 +02:00
fierke 9be546d25f [feat:] overview of generated courses (pdf) 2026-06-07 20:19:54 +02:00
fierke 2910b1aeda Merge remote-tracking branch 'origin/main' 2026-06-07 12:51:51 +02:00
fierke 7a0e392ba8 [chore:] removed comments 2026-06-07 12:51:45 +02:00
fierke 91c6ea1269 [feat:] reordering courses if they are of the same sport and are not weighted equally 2026-06-06 14:24:25 +02:00
fierke c0da656331 [fix:] pdfsharp integration 2026-06-05 07:48:19 +02:00
fierke eea7b9f628 [fix:] sentence was too long 2026-06-04 19:15:03 +02:00
fierke a784e598de [chore:] forgot to 'git add' these... 2026-06-04 19:12:52 +02:00
fierke 66596b530b [feat:] im- and export (pdf/json) 2026-06-04 18:59:01 +02:00
fierke 5ef41b21b0 [gui:] buttons for im- and exporting 2026-06-04 18:58:32 +02:00
fierke c291ed1788 [file:] included fonts 2026-06-04 18:58:03 +02:00
fierke 78d25e6231 [fix:] Sport ids 2026-06-01 21:28:07 +02:00
fierke 73af2039ba [file:] added fonts 2026-06-01 21:27:29 +02:00
fierke c5be9d2c6e [feat:] frontend to edit courses (ai-slop) 2026-06-01 17:00:55 +02:00
fierke c46417c56d [chore:] single-semester-side limiting of courses 2026-06-01 17:00:39 +02:00
fierke 895c55a52f [chore:] cleaned up some rubbish 2026-06-01 13:14:03 +02:00
fierke e54e2e7840 [chore:] logo integration 2026-06-01 12:45:35 +02:00
fierke 0a710bea8b [chore:] window icon integration 2026-06-01 12:28:59 +02:00
fierke d4de543d71 [file:] added logo.ico 2026-06-01 12:25:33 +02:00
fierke a0eac4ae95 Merge pull request '[chore:] better crafting' (#5) from bettercrafting into main
Reviewed-on: #5
2026-05-30 11:33:11 +00:00
fierke 22e9377062 [chore:] better crafting 2026-05-29 12:52:02 +02:00
fierke 11e6aab8fd [fix:] removed triple-calling of OptimizeStudentWishCoverage() 2026-04-22 07:25:46 +02:00
fierke 13f313d4a0 Merge remote-tracking branch 'origin/main' 2026-04-21 14:14:07 +02:00
fierke 64951c579f [fix:] massive bug that prevented correct course crafting (sports's id was always 0) 2026-04-21 14:13:54 +02:00
fierke 4130c36335 [chore:] import function that won't be called 2026-04-21 14:13:14 +02:00
fierke 7f2fc99d0b [gui:] import-button in the menu 2026-04-21 14:12:54 +02:00
fierke eb640ff749 [chore:] file picker, but currently not using the file itself 2026-04-21 14:12:40 +02:00
fierke 54a564df04 [chore:] tiny little comment to prevent confusion 2026-04-21 14:11:51 +02:00
fierke 71dc63f2a2 [file:] added logo 2026-04-18 16:04:14 +02:00
fierke 68f673c8d7 Update README.md 2026-04-15 11:48:44 +00:00
fierke d5b7fd7af3 [file:] uploaded app-screenshots 2026-04-15 13:45:37 +02:00
20 changed files with 2115 additions and 407 deletions
+77
View File
@@ -0,0 +1,77 @@
using System;
using System.IO;
using PdfSharp.Drawing;
using PdfSharp.Fonts;
namespace spplus;
/// <summary>
/// Custom font resolver for PdfSharp that loads fonts from the res/fonts directory.
/// </summary>
public class CustomFontResolver : IFontResolver
{
private static string? _fontPath;
public CustomFontResolver()
{
// Try to locate the fonts directory
var basePath = AppDomain.CurrentDomain.BaseDirectory;
var possiblePaths = new[]
{
Path.Combine(basePath, "res", "fonts"),
Path.Combine(basePath, "fonts"),
Path.Combine(Directory.GetCurrentDirectory(), "res", "fonts"),
Path.Combine(Directory.GetCurrentDirectory(), "fonts"),
};
foreach (var path in possiblePaths)
{
if (Directory.Exists(path))
{
_fontPath = path;
break;
}
}
if (_fontPath == null)
{
throw new DirectoryNotFoundException(
$"Font directory not found. Searched: {string.Join(", ", possiblePaths)}");
}
}
public FontResolverInfo ResolveTypeface(string familyName, bool isBold, bool isItalic)
{
// Map family name to font file
var fileName = familyName switch
{
"Cantarell" => isBold
? isItalic ? "Cantarell-BoldItalic.ttf" : "Cantarell-Bold.ttf"
: isItalic ? "Cantarell-Italic.ttf" : "Cantarell-Regular.ttf",
_ => isBold
? isItalic ? "Cantarell-BoldItalic.ttf" : "Cantarell-Bold.ttf"
: isItalic ? "Cantarell-Italic.ttf" : "Cantarell-Regular.ttf"
};
var fontPath = Path.Combine(_fontPath, fileName);
if (!File.Exists(fontPath))
{
throw new FileNotFoundException($"Font file not found: {fontPath}");
}
// Return a FontResolverInfo with the font path
return new FontResolverInfo(fontPath);
}
public byte[]? GetFont(string faceName)
{
// faceName is the path returned by ResolveTypeface
if (File.Exists(faceName))
{
return File.ReadAllBytes(faceName);
}
return null;
}
}
+41 -8
View File
@@ -3,7 +3,7 @@
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
x:Class="spplus.MainWindow" WindowState="Maximized"
x:Class="spplus.MainWindow" WindowState="Maximized" Icon="res/logo.ico"
Title="spplus">
<Border>
<Grid RowDefinitions="30,*">
@@ -12,6 +12,7 @@
<!-- <MenuItem Click="MnuSettings_OnClick" x:Name="MnuSettings" Header="Einstellungen" /> -->
<!-- <Separator /> -->
<MenuItem x:Name="MnuExpSettings" Header="Einstellungen exportieren" Click="MnuExpSettings_OnClick" />
<MenuItem x:Name="MnuImpResult" Header="Berechnung importieren" Click="MnuImpResult_OnClick" />
<MenuItem x:Name="MnuExit" Header="Beenden" Click="MnuExit_OnClick"/>
</MenuItem>
<MenuItem Header="Hilfe">
@@ -195,8 +196,20 @@
</Grid>
<Line />
<Grid ColumnDefinitions="*,3*">
<Label Content="Maximale Sportkursanzahl pro Semester"></Label>
<NumericUpDown Grid.Column="1" x:Name="NudSportMaxPerSemester" ValueChanged="NudSportMaxPerSemester_OnValueChanged"></NumericUpDown>
<Label Content="Maximale Sportkursanzahl Semester 1"></Label>
<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>
</StackPanel>
</Grid>
@@ -209,23 +222,43 @@
<Label FontSize="20" Content="Ergebnisse" VerticalContentAlignment="Center" />
</StackPanel>
</TabItem.Header>
<Grid ColumnDefinitions="*,*" RowDefinitions="50,2*,*">
<ListBox Grid.RowSpan="2" x:Name="LbResult" Margin="0,10,10,10"></ListBox>
<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">
<Grid ColumnDefinitions="*,*" RowDefinitions="100,2*,*">
<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" RowDefinitions="*,*" ColumnDefinitions="*,*">
<Button Grid.Row="0" Grid.Column="0" Margin="0,10,5,0" x:Name="BtnExportCoursePDF" VerticalAlignment="Top" Height="35" HorizontalAlignment="Stretch" Click="BtnExportCoursePDF_OnClick" HorizontalContentAlignment="Center">
<StackPanel Orientation="Horizontal">
<LucideIcon Kind="FileText" Width="24" Height="24" />
<Label Content="Export (PDF)..." VerticalContentAlignment="Center" FontSize="12"
FontWeight="Bold" />
</StackPanel>
</Button>
<Button Grid.Column="1" Margin="5,10,0,0" x:Name="BtnExportCourseCSV" VerticalAlignment="Top" Height="35" HorizontalAlignment="Stretch" Click="BtnExportCourseCSV_OnClick" HorizontalContentAlignment="Center">
<Button Grid.Row="0" Grid.Column="1" Margin="5,10,0,0" x:Name="BtnExportCourseCSV" VerticalAlignment="Top" Height="35" HorizontalAlignment="Stretch" Click="BtnExportCourseCSV_OnClick" HorizontalContentAlignment="Center">
<StackPanel Orientation="Horizontal">
<LucideIcon Kind="FileJson" Width="24" Height="24" />
<Label Content="Export (CSV)..." VerticalContentAlignment="Center" FontSize="12"
FontWeight="Bold" />
</StackPanel>
</Button>
<Button Grid.Row="1" Grid.Column="0" Margin="0,5,5,0" x:Name="BtnExportConfiguration" VerticalAlignment="Top" Height="35" HorizontalAlignment="Stretch" Click="MnuExpSettings_OnClick" HorizontalContentAlignment="Center">
<StackPanel Orientation="Horizontal">
<LucideIcon Kind="FileJson" Width="24" Height="24" />
<Label Content="Konfiguration exportieren..." VerticalContentAlignment="Center" FontSize="12"
FontWeight="Bold" />
</StackPanel>
</Button>
<Button Grid.Row="1" Grid.Column="1" Margin="5,5,0,0" x:Name="BtnImportResults" VerticalAlignment="Top" Height="35" HorizontalAlignment="Stretch" Click="MnuImpResult_OnClick" HorizontalContentAlignment="Center">
<StackPanel Orientation="Horizontal">
<LucideIcon Kind="Import" Width="24" Height="24" />
<Label Content="Ergebnisse importieren..." VerticalContentAlignment="Center" FontSize="12"
FontWeight="Bold" />
</StackPanel>
</Button>
</Grid>
<ScrollViewer Grid.Row="1" Grid.Column="1" Margin="0,5,0,10" Background="#44CCCCCC">
<TextBlock x:Name="TbResultStatistics"></TextBlock>
+322 -11
View File
@@ -1,6 +1,8 @@
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Threading.Tasks;
using Avalonia.Controls;
using Avalonia.Input;
using Avalonia.Interactivity;
@@ -12,6 +14,26 @@ public partial class MainWindow : Window
{
public static MainWindow Instance { get; set; }
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()
{
InitializeComponent();
@@ -20,13 +42,199 @@ public partial class MainWindow : Window
RefreshCoursesList();
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 {}
}
private void MnuExpSettings_OnClick(object? sender, RoutedEventArgs e)
private void RegenerateContextMenu()
{
var res = MessageBox.Show(this, "Dieses Feature ist noch nicht implementiert", "Fehlend");
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 async void ChangeStudentCourse(Sport targetSport)
{
if (LbResult.SelectedItem is not ResultEntry selectedEntry)
return;
try
{
if (await ApplyStudentCourseChange(selectedEntry.Student, selectedEntry.Semester, targetSport))
{
CourseCrafter.ReloadResult();
RefreshResultView();
RegenerateContextMenu();
}
}
catch (Exception ex)
{
Console.WriteLine(ex.Message + "\n" + ex.StackTrace);
}
}
private async Task<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();
string? oldSportName = currentCourses
.Select(course => course.Instance.Sport.Name)
.FirstOrDefault();
var existingTargetCourses = semesterCourses
.Where(course => course.Instance.Sport.Name == targetSport.Name)
.ToList();
CourseCrafter.CourseInstance? targetCourse = existingTargetCourses
.Where(course => !course.Instance.Students.Contains(student.ID) &&
course.Instance.Students.Count < course.Instance.Sport.MaxStudents)
.OrderBy(course => course.Instance.Students.Count)
.Select(course => course.Instance)
.FirstOrDefault();
if (targetCourse == null && existingTargetCourses.Any())
{
targetCourse = existingTargetCourses
.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;
}
if (changed && !string.IsNullOrEmpty(oldSportName) &&
IsRebalanceNeededForSportSemester(oldSportName, semester))
{
var result = await MessageBox.Show(this,
$"Der Kursplan für {oldSportName} im {semester}. Semester ist nach der Änderung unausgeglichen. Soll die alte Sportart umverteilt werden?",
"Umverteilung der alten Sportart", MessageBoxButton.YesNo);
if (result == MessageBoxResult.Yes)
BalanceSportSemester(oldSportName, semester);
}
return changed;
}
private bool IsRebalanceNeededForSportSemester(string sportName, int semester)
{
var courses = CourseCrafter.GeneratedCourses
.Where(course => course.Semester == semester && course.Instance.Sport.Name == sportName)
.ToList();
if (courses.Count <= 1)
return false;
var ordered = courses
.OrderBy(course => course.Instance.Students.Count)
.ToList();
int minCount = ordered.First().Instance.Students.Count;
int maxCount = ordered.Last().Instance.Students.Count;
if (maxCount - minCount <= 1)
return false;
if (ordered.Last().Instance.Students.Count - 1 < GetEffectiveMinStudents(ordered.Last().Instance.Sport, semester))
return false;
if (ordered.First().Instance.Students.Count >= ordered.First().Instance.Sport.MaxStudents)
return false;
return true;
}
private void BalanceSportSemester(string sportName, int semester)
{
bool changed;
do
{
changed = false;
var courses = CourseCrafter.GeneratedCourses
.Where(course => course.Semester == semester && course.Instance.Sport.Name == sportName)
.OrderBy(course => course.Instance.Students.Count)
.ToList();
if (courses.Count <= 1)
break;
var target = courses.First();
var source = courses.Last();
if (source.Instance.Students.Count <= target.Instance.Students.Count + 1)
break;
if (source.Instance.Students.Count - 1 < GetEffectiveMinStudents(source.Instance.Sport, semester))
break;
if (target.Instance.Students.Count >= target.Instance.Sport.MaxStudents)
break;
var studentId = source.Instance.Students[^1];
source.Instance.Students.RemoveAt(source.Instance.Students.Count - 1);
target.Instance.Students.Add(studentId);
changed = true;
} while (changed);
}
private int GetEffectiveMinStudents(Sport sport, int semester)
{
int reduction = (semester >= 3) ? 2 : 0;
return Math.Max(1, sport.MinStudents - reduction);
}
private async void MnuExpSettings_OnClick(object? sender, RoutedEventArgs e)
{
await ExportConfigurationAsync();
}
private void MnuExit_OnClick(object? sender, RoutedEventArgs e)
@@ -159,6 +367,21 @@ public partial class MainWindow : Window
//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()
{
LbResult.Items.Clear();
@@ -168,7 +391,7 @@ public partial class MainWindow : Window
{
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)
@@ -262,6 +485,7 @@ public partial class MainWindow : Window
{
LbSportCourses.Items.Add(sp);
}
RegenerateContextMenu();
}
@@ -298,6 +522,7 @@ public partial class MainWindow : Window
try
{
((Sport)LbSportCourses.SelectedItem).Name = TbSportName.Text;
RegenerateContextMenu();
} catch {}
}
@@ -411,16 +636,53 @@ public partial class MainWindow : Window
} catch (Exception ex){}
}
private void BtnExportCoursePDF_OnClick(object? sender, RoutedEventArgs e)
private async void BtnExportCoursePDF_OnClick(object? sender, RoutedEventArgs e)
{
// Export as PDF
var topLevel = GetTopLevel(this);
var file = await topLevel!.StorageProvider.SaveFilePickerAsync(new FilePickerSaveOptions
{
Title = "PDF-Datei speichern",
SuggestedFileName = "spplus_kurse.pdf",
SuggestedFileType = new FilePickerFileType(".pdf-Datei")
{
Patterns = new[] { "*.pdf" }
}
});
if (file == null) return;
PdfExportUtility.ExportGeneratedCourses(file.Path.LocalPath);
}
private void NudSportMaxPerSemester_OnValueChanged(object? sender, NumericUpDownValueChangedEventArgs e)
private void NudSportMaxPerSemester1_OnValueChanged(object? sender, NumericUpDownValueChangedEventArgs e)
{
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 {}
}
@@ -430,16 +692,65 @@ public partial class MainWindow : Window
var file = await topLevel!.StorageProvider.SaveFilePickerAsync(new FilePickerSaveOptions
{
Title = "CSV-Datei speichern",
SuggestedFileType = new FilePickerFileType(".csv-Datei")
SuggestedFileName = "spplus_ergebnisse.json",
SuggestedFileType = new FilePickerFileType(".json-Datei")
{
Patterns = new[] { "*.csv" }
Patterns = new[] { "*.json" }
}
});
if (file == null) return;
ExportUtility.ExportToCSV(file.Path.AbsolutePath);
ExportUtility.ExportResultsToJson(file.Path.LocalPath);
}
private async void MnuImpResult_OnClick(object? sender, RoutedEventArgs e)
{
var topLevel = GetTopLevel(this);
var file = await topLevel!.StorageProvider.OpenFilePickerAsync(new FilePickerOpenOptions
{
Title = "Ergebnis-CSV laden",
AllowMultiple = false,
FileTypeFilter = new[]
{
new FilePickerFileType(".json-Datei")
{
Patterns = new[] { "*.json" }
}
}
});
if (file == null || file.Count == 0) return;
var imported = import.ImportResultFromJson(file[0].Path.LocalPath.ToString());
if (imported != null && imported.Count > 0)
{
CourseCrafter.GeneratedCourses = imported
.OrderBy(c => c.Semester)
.ThenBy(c => c.Instance.Sport.Name)
.ToList();
CourseCrafter.ReloadResult();
RefreshResultView();
}
}
private async System.Threading.Tasks.Task ExportConfigurationAsync()
{
var topLevel = GetTopLevel(this);
var file = await topLevel!.StorageProvider.SaveFilePickerAsync(new FilePickerSaveOptions
{
Title = "Konfiguration speichern",
SuggestedFileName = "spplus_konfiguration.json",
SuggestedFileType = new FilePickerFileType(".json-Datei")
{
Patterns = new[] { "*.json" }
}
});
if (file == null) return;
ExportUtility.ExportConfigurationToJson(file.Path.LocalPath);
}
}
+1 -1
View File
@@ -2,7 +2,7 @@
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d" SizeToContent="WidthAndHeight"
mc:Ignorable="d" SizeToContent="WidthAndHeight" Icon="res/logo.ico"
x:Class="spplus.MessageBox" WindowStartupLocation="CenterScreen"
Title="MessageBox">
<StackPanel>
+621
View File
@@ -0,0 +1,621 @@
using System;
using System.Collections.Generic;
using System.Linq;
using PdfSharp;
using PdfSharp.Drawing;
using PdfSharp.Pdf;
namespace spplus;
public static class PdfExportUtility
{
private const string FontFamily = "Cantarell";
private const double Margin = 36.0;
private const double FooterHeight = 28.0;
private static readonly XBrush FooterBrush = XBrushes.Gray;
public static void ExportGeneratedCourses(string filepath)
{
var document = new PdfDocument
{
Info =
{
Title = "spplus - generierte Kurse",
Author = "spplus",
Creator = "spplus"
}
};
var courses = CourseCrafter.GeneratedCourses
.OrderBy(course => course.Semester)
.ThenBy(course => course.Instance.Sport.Name)
.ToList();
if (courses.Count == 0)
{
RenderEmptyDocument(document);
document.Save(filepath);
return;
}
RenderCoursesOverview(document, courses);
var studentsById = Settings.Instance.Students
.GroupBy(student => student.ID, StringComparer.OrdinalIgnoreCase)
.ToDictionary(group => group.Key, group => group.First(), StringComparer.OrdinalIgnoreCase);
foreach (var course in courses)
{
RenderCourse(document, course, studentsById);
}
RenderMissingStudentsReport(document, studentsById, courses);
document.Save(filepath);
}
private static void RenderMissingStudentsReport(
PdfDocument document,
IReadOnlyDictionary<string, Student> studentsById,
IReadOnlyList<(int Semester, CourseCrafter.CourseInstance Instance)> courses)
{
var assignedSemesters = studentsById.Values
.ToDictionary(student => student.ID, _ => new bool[4], StringComparer.OrdinalIgnoreCase);
foreach (var course in courses)
{
if (course.Semester < 1 || course.Semester > 4)
continue;
foreach (var studentId in course.Instance.Students)
{
if (assignedSemesters.TryGetValue(studentId, out var assignment))
assignment[course.Semester - 1] = true;
}
}
var missingStudents = studentsById.Values
.Where(student => assignedSemesters.TryGetValue(student.ID, out var assignment) && assignment.Any(a => !a))
.OrderBy(student => student.Name)
.ThenBy(student => student.ID)
.ToList();
var columns = new double[] { 30.0, 90.0, 190.0, 40.0, 40.0, 40.0, 40.0 };
var headerRow = new[] { "Nr.", "ID", "Name", "Sem1", "Sem2", "Sem3", "Sem4" };
int rowIndex = 0;
int pageIndex = 0;
while (rowIndex < missingStudents.Count || pageIndex == 0)
{
var page = document.AddPage();
ConfigurePage(page);
using var gfx = XGraphics.FromPdfPage(page);
var titleFont = new XFont(FontFamily, 18, XFontStyleEx.Bold);
var subtitleFont = new XFont(FontFamily, 10, XFontStyleEx.Regular);
var tableHeaderFont = new XFont(FontFamily, 9, XFontStyleEx.Bold);
var tableBodyFont = new XFont(FontFamily, 8.5, XFontStyleEx.Regular);
double pageWidth = page.Width.Point;
double pageHeight = page.Height.Point;
double contentWidth = pageWidth - (Margin * 2);
double contentBottom = pageHeight - Margin - FooterHeight;
double y = Margin;
var headerTitle = pageIndex == 0
? "Nicht zugeordnete Schülerinnen und Schüler"
: "Fortsetzung: Nicht zugeordnete Schülerinnen und Schüler";
gfx.DrawString(headerTitle, titleFont, XBrushes.Black,
new XRect(Margin, y, contentWidth, 24), XStringFormats.TopLeft);
y += 26;
if (pageIndex == 0)
{
gfx.DrawString("Es werden nur Schülerinnen und Schüler aufgeführt, die in mindestens einem Semester keine Zuordnung erhalten haben.",
subtitleFont, XBrushes.Gray, new XRect(Margin, y, contentWidth, 20), XStringFormats.TopLeft);
y += 24;
}
if (missingStudents.Count == 0)
{
var bodyFont = new XFont(FontFamily, 11, XFontStyleEx.Regular);
gfx.DrawString("Alle Schülerinnen und Schüler sind in allen vier Semestern zugeordnet.", bodyFont, XBrushes.Black,
new XRect(Margin, y, contentWidth, 20), XStringFormats.TopLeft);
DrawFooter(gfx, page);
break;
}
y += DrawTableRow(gfx, y, columns, headerRow, tableHeaderFont, fillHeader: true);
while (rowIndex < missingStudents.Count)
{
var student = missingStudents[rowIndex];
var assignment = assignedSemesters[student.ID];
var row = BuildMissingStudentRow(rowIndex + 1, student, assignment);
double rowHeight = MeasureRowHeight(gfx, columns, row, tableBodyFont);
if (y + rowHeight > contentBottom)
break;
y += DrawTableRow(gfx, y, columns, row, tableBodyFont, fillHeader: false);
rowIndex++;
}
DrawFooter(gfx, page);
pageIndex++;
if (rowIndex >= missingStudents.Count)
break;
}
}
private static string[] BuildMissingStudentRow(int rowNumber, Student student, bool[] assignment)
{
return new[]
{
rowNumber.ToString(),
student.ID,
student.Name,
assignment[0] ? string.Empty : "X",
assignment[1] ? string.Empty : "X",
assignment[2] ? string.Empty : "X",
assignment[3] ? string.Empty : "X",
};
}
private static void RenderEmptyDocument(PdfDocument document)
{
var page = document.AddPage();
ConfigurePage(page);
using var gfx = XGraphics.FromPdfPage(page);
var titleFont = new XFont(FontFamily, 18, XFontStyleEx.Bold);
var bodyFont = new XFont(FontFamily, 11, XFontStyleEx.Regular);
gfx.DrawString("spplus", titleFont, XBrushes.Black,
new XRect(Margin, Margin, page.Width.Point - Margin * 2, 24), XStringFormats.TopLeft);
gfx.DrawString("Es wurden keine generierten Kurse gefunden.", bodyFont, XBrushes.Black,
new XRect(Margin, Margin + 34, page.Width.Point - Margin * 2, 24), XStringFormats.TopLeft);
DrawFooter(gfx, page);
}
private static void RenderCoursesOverview(
PdfDocument document,
IReadOnlyList<(int Semester, CourseCrafter.CourseInstance Instance)> courses)
{
var overviewFont = new XFont(FontFamily, 18, XFontStyleEx.Bold);
var subtitleFont = new XFont(FontFamily, 10, XFontStyleEx.Regular);
var tableHeaderFont = new XFont(FontFamily, 9, XFontStyleEx.Bold);
var tableBodyFont = new XFont(FontFamily, 8.5, XFontStyleEx.Regular);
int count = 0;
var coursesPerSemester = courses
.GroupBy(course => course.Semester)
.ToDictionary(group => group.Key, group => group.Count());
var courseIndexBySemester = new Dictionary<int, int>();
var rows = courses.Select(course =>
{
count++;
var key = course.Semester;
courseIndexBySemester.TryGetValue(key, out var index);
index++;
courseIndexBySemester[key] = index;
return new[]
{
count.ToString(),
course.Semester.ToString(),
course.Instance.Sport.Name,
$"{index}/{coursesPerSemester[key]}",
course.Instance.Students.Count.ToString()
};
}).ToList();
var columns = new[] { 50.0, 70.0, 150.0, 150.0, 100.0 };
var headerRow = new[]
{
"Nr.",
"Semester",
"Sportart",
"Kurs (Semester)",
"Anzahl SuS"
};
int rowIndex = 0;
int pageIndex = 0;
while (rowIndex < rows.Count || pageIndex == 0)
{
var page = document.AddPage();
ConfigurePage(page);
using var gfx = XGraphics.FromPdfPage(page);
double pageWidth = page.Width.Point;
double pageHeight = page.Height.Point;
double contentWidth = pageWidth - (Margin * 2);
double contentBottom = pageHeight - Margin - FooterHeight;
double y = Margin;
var headerTitle = pageIndex == 0
? "Übersicht generierter Kurse"
: "Fortsetzung: Übersicht generierter Kurse";
gfx.DrawString(headerTitle, overviewFont, XBrushes.Black,
new XRect(Margin, y, contentWidth, 24), XStringFormats.TopLeft);
y += 26;
if (pageIndex == 0)
{
gfx.DrawString("Tabellarische Übersicht aller generierten Kurse nach Semester und Sportart.",
subtitleFont, XBrushes.Gray, new XRect(Margin, y, contentWidth, 20), XStringFormats.TopLeft);
y += 24;
}
y += DrawTableRow(gfx, y, columns, headerRow, tableHeaderFont, fillHeader: true);
while (rowIndex < rows.Count)
{
var row = rows[rowIndex];
double rowHeight = MeasureRowHeight(gfx, columns, row, tableBodyFont);
if (y + rowHeight > contentBottom)
break;
y += DrawTableRow(gfx, y, columns, row, tableBodyFont, fillHeader: false);
rowIndex++;
}
DrawFooter(gfx, page);
pageIndex++;
if (rowIndex >= rows.Count)
break;
}
}
private static void RenderCourse(
PdfDocument document,
(int Semester, CourseCrafter.CourseInstance Instance) course,
IReadOnlyDictionary<string, Student> studentsById)
{
var studentIds = course.Instance.Students.ToList();
int index = 0;
int pageIndex = 0;
do
{
var page = document.AddPage();
ConfigurePage(page);
using var gfx = XGraphics.FromPdfPage(page);
var titleFont = new XFont(FontFamily, 18, XFontStyleEx.Bold);
var subtitleFont = new XFont(FontFamily, 10, XFontStyleEx.Regular);
var labelFont = new XFont(FontFamily, 9, XFontStyleEx.Bold);
var valueFont = new XFont(FontFamily, 9, XFontStyleEx.Regular);
var tableHeaderFont = new XFont(FontFamily, 9, XFontStyleEx.Bold);
var tableBodyFont = new XFont(FontFamily, 8.5, XFontStyleEx.Regular);
double pageWidth = page.Width.Point;
double pageHeight = page.Height.Point;
double contentWidth = pageWidth - (Margin * 2);
double contentBottom = pageHeight - Margin - FooterHeight;
double y = Margin;
var headerTitle = $"Sem. {course.Semester} - {course.Instance.Sport.Name}";
if (pageIndex > 0)
headerTitle += " (Fortsetzung)";
gfx.DrawString(headerTitle, titleFont, XBrushes.Black,
new XRect(Margin, y, contentWidth, 24), XStringFormats.TopLeft);
y += 26;
gfx.DrawString($"{course.Instance.Students.Count} Schülerinnen und Schüler", subtitleFont,
XBrushes.Gray, new XRect(Margin, y, contentWidth, 18), XStringFormats.TopLeft);
y += 24;
y = DrawInfoRows(gfx, y, contentWidth, labelFont, valueFont, course);
y += 10;
var columns = GetTableColumns(contentWidth);
var headerRow = new[]
{
"Nr.",
"ID",
"Name",
};
y += DrawTableRow(gfx, y, columns, headerRow, tableHeaderFont, fillHeader: true);
if (studentIds.Count == 0)
{
var emptyFont = new XFont(FontFamily, 9, XFontStyleEx.Italic);
gfx.DrawString("Keine Schülerinnen und Schüler zugeordnet.", emptyFont, XBrushes.Gray,
new XRect(Margin + 6, y + 8, contentWidth - 12, 20), XStringFormats.TopLeft);
}
else
{
while (index < studentIds.Count)
{
if (!studentsById.TryGetValue(studentIds[index], out var student))
student = new Student { ID = studentIds[index] };
var row = BuildStudentRow(index + 1, student);
double rowHeight = MeasureRowHeight(gfx, columns, row, tableBodyFont);
if (y + rowHeight > contentBottom)
break;
y += DrawTableRow(gfx, y, columns, row, tableBodyFont, fillHeader: false);
index++;
}
}
DrawFooter(gfx, page);
pageIndex++;
} while (index < studentIds.Count);
}
private static double DrawInfoRows(
XGraphics gfx,
double startY,
double contentWidth,
XFont labelFont,
XFont valueFont,
(int Semester, CourseCrafter.CourseInstance Instance) course)
{
double y = startY;
double labelWidth = 160.0;
double valueWidth = contentWidth - labelWidth;
double rowPadding = 4.0;
var rows = new List<(string Label, string Value)>
{
("Kurs-ID", course.Instance.Sport.ID.ToString()),
("Kursname", course.Instance.Sport.Name),
("Semester", course.Semester.ToString()),
("Anzahl SuS", course.Instance.Students.Count.ToString()),
("Min / Max", $"{course.Instance.Sport.MinStudents} / {course.Instance.Sport.MaxStudents}"),
("Semesterangebote", string.Join(", ", course.Instance.Sport.Semester.Select((count, idx) => $"S{idx + 1}:{count}"))),
("Alternativnamen", course.Instance.Sport.AlternativeNames.Count == 0
? "-"
: string.Join(", ", course.Instance.Sport.AlternativeNames))
};
foreach (var row in rows)
{
double rowHeight = MeasureInfoRowHeight(gfx, labelFont, valueFont, labelWidth, valueWidth, row.Label, row.Value);
gfx.DrawRectangle(XPens.LightGray, Margin, y, labelWidth, rowHeight);
gfx.DrawRectangle(XPens.LightGray, Margin + labelWidth, y, valueWidth, rowHeight);
DrawWrappedText(gfx, row.Label, labelFont, XBrushes.Black, Margin + 4, y + rowPadding, labelWidth - 8);
DrawWrappedText(gfx, row.Value, valueFont, XBrushes.Black, Margin + labelWidth + 4, y + rowPadding, valueWidth - 8);
y += rowHeight;
}
return y;
}
private static double MeasureInfoRowHeight(
XGraphics gfx,
XFont labelFont,
XFont valueFont,
double labelWidth,
double valueWidth,
string label,
string value)
{
double labelHeight = MeasureWrappedTextHeight(gfx, label, labelFont, labelWidth - 8);
double valueHeight = MeasureWrappedTextHeight(gfx, value, valueFont, valueWidth - 8);
return Math.Max(labelHeight, valueHeight) + 8.0;
}
private static string[] BuildStudentRow(int rowNumber, Student student)
{
return new[]
{
rowNumber.ToString(),
student.ID,
student.Name,
};
}
private static string GetCourseName(Student student, int index)
{
if (index < 0 || index >= student.SelectedCourseNames.Count)
return string.Empty;
return student.SelectedCourseNames[index];
}
private static double[] GetTableColumns(double contentWidth)
{
return new[]
{
30.0,
75.0,
145.0
};
}
private static double DrawTableRow(
XGraphics gfx,
double startY,
double[] columns,
IReadOnlyList<string> values,
XFont font,
bool fillHeader)
{
const double padding = 4.0;
double rowHeight = MeasureRowHeight(gfx, columns, values, font);
double x = Margin;
for (int i = 0; i < columns.Length; i++)
{
double width = columns[i];
var rect = new XRect(x, startY, width, rowHeight);
gfx.DrawRectangle(XPens.LightGray, rect);
if (fillHeader)
gfx.DrawRectangle(new XSolidBrush(XColor.FromArgb(235, 235, 235)), rect);
gfx.DrawRectangle(XPens.LightGray, rect);
var lines = WrapText(gfx, values[i], font, width - (padding * 2));
double lineHeight = font.GetHeight();
double textY = startY + padding;
foreach (var line in lines)
{
gfx.DrawString(line, font, XBrushes.Black,
new XRect(x + padding, textY, width - (padding * 2), lineHeight), XStringFormats.TopLeft);
textY += lineHeight;
}
x += width;
}
return rowHeight;
}
private static double MeasureRowHeight(
XGraphics gfx,
double[] columns,
IReadOnlyList<string> values,
XFont font)
{
const double padding = 4.0;
double maxHeight = font.GetHeight() + (padding * 2);
for (int i = 0; i < columns.Length; i++)
{
double height = MeasureWrappedTextHeight(gfx, values[i], font, columns[i] - (padding * 2)) + (padding * 2);
if (height > maxHeight)
maxHeight = height;
}
return maxHeight;
}
private static double MeasureWrappedTextHeight(XGraphics gfx, string text, XFont font, double width)
{
var lines = WrapText(gfx, text, font, width);
return lines.Count * font.GetHeight();
}
private static List<string> WrapText(XGraphics gfx, string? text, XFont font, double maxWidth)
{
var result = new List<string>();
if (string.IsNullOrWhiteSpace(text))
{
result.Add(string.Empty);
return result;
}
foreach (var paragraph in text.Replace("\r", string.Empty).Split('\n'))
{
if (string.IsNullOrWhiteSpace(paragraph))
{
result.Add(string.Empty);
continue;
}
string current = string.Empty;
foreach (var token in paragraph.Split(' ', StringSplitOptions.RemoveEmptyEntries))
{
foreach (var wordPart in SplitWordIfNeeded(gfx, token, font, maxWidth))
{
var candidate = string.IsNullOrEmpty(current) ? wordPart : $"{current} {wordPart}";
if (gfx.MeasureString(candidate, font).Width <= maxWidth)
{
current = candidate;
}
else
{
if (!string.IsNullOrEmpty(current))
result.Add(current);
current = wordPart;
}
}
}
if (!string.IsNullOrEmpty(current))
result.Add(current);
}
return result.Count == 0 ? new List<string> { string.Empty } : result;
}
private static IEnumerable<string> SplitWordIfNeeded(XGraphics gfx, string word, XFont font, double maxWidth)
{
if (gfx.MeasureString(word, font).Width <= maxWidth)
{
yield return word;
yield break;
}
string chunk = string.Empty;
foreach (char c in word)
{
string candidate = chunk + c;
if (chunk.Length > 0 && gfx.MeasureString(candidate, font).Width > maxWidth)
{
yield return chunk;
chunk = c.ToString();
}
else
{
chunk = candidate;
}
}
if (!string.IsNullOrEmpty(chunk))
yield return chunk;
}
private static void DrawWrappedText(
XGraphics gfx,
string? text,
XFont font,
XBrush brush,
double x,
double y,
double width)
{
double lineHeight = font.GetHeight();
foreach (var line in WrapText(gfx, text, font, width))
{
gfx.DrawString(line, font, brush, new XRect(x, y, width, lineHeight), XStringFormats.TopLeft);
y += lineHeight;
}
}
private static void DrawFooter(XGraphics gfx, PdfPage page)
{
var footerFont = new XFont(FontFamily, 7, XFontStyleEx.Regular);
double width = page.Width.Point - (Margin * 2);
double x = Margin;
double baseY = page.Height.Point - Margin + 2;
gfx.DrawString("generated by spplus", footerFont, FooterBrush,
new XRect(x, baseY, width, 9), XStringFormats.Center);
gfx.DrawString("(c) 2026 MyPapertown", footerFont, FooterBrush,
new XRect(x, baseY + 8, width, 9), XStringFormats.Center);
gfx.DrawString("www.mypapercloud.de/spplus", footerFont, FooterBrush,
new XRect(x, baseY + 16, width, 9), XStringFormats.Center);
}
private static void ConfigurePage(PdfPage page)
{
page.Size = PageSize.A4;
page.Orientation = PageOrientation.Portrait;
}
}
+9 -1
View File
@@ -1,5 +1,7 @@
using Avalonia;
using System;
using PdfSharp;
using PdfSharp.Fonts;
namespace spplus;
@@ -9,8 +11,14 @@ class Program
// SynchronizationContext-reliant code before AppMain is called: things aren't initialized
// yet and stuff might break.
[STAThread]
public static void Main(string[] args) => BuildAvaloniaApp()
public static void Main(string[] args)
{
// Initialize PdfSharp font resolver before any PDF operations
GlobalFontSettings.FontResolver = new CustomFontResolver();
BuildAvaloniaApp()
.StartWithClassicDesktopLifetime(args);
}
// Avalonia configuration, don't remove; also used by visual designer.
public static AppBuilder BuildAvaloniaApp()
+4
View File
@@ -2,6 +2,8 @@
Plattformunabhängiger (Windows, Linux, Mac), interaktiver Sportkursplaner für Oberstufen auf Basis einer Sportkurswahl durch SuS.
![spplus-Hauptbildschirm](img/spplus_planung.png)
## Features
* \+ Import von CSV-Dateien mit Kurswahl
* \+ Wahlansicht
@@ -21,3 +23,5 @@ Plattformunabhängiger (Windows, Linux, Mac), interaktiver Sportkursplaner für
* Suche `spplus` bzw. `spplus.exe` und führe aus
* Linux/MacOS evl.: `chmod +x spplus`
![spplus-Sportkursübersicht](img/spplus_sport.png)
+647 -143
View File
@@ -15,6 +15,66 @@ public class CourseCrafter
public static List<(int Semester, CourseInstance Instance)> GeneratedCourses
= new();
private static Sport? ResolveSportFromCourseName(string courseName)
{
if (string.IsNullOrWhiteSpace(courseName) ||
string.Equals(courseName, "null", StringComparison.OrdinalIgnoreCase))
{
return null;
}
return Settings.Instance.Sports.FirstOrDefault(sport =>
string.Equals(sport.Name, courseName, StringComparison.OrdinalIgnoreCase) ||
sport.AlternativeNames.Any(alt => string.Equals(alt, courseName, StringComparison.OrdinalIgnoreCase)));
}
public static void RebuildGeneratedCoursesFromResults(IEnumerable<Student> importedResults)
{
var importedById = importedResults
.Where(student => !string.IsNullOrWhiteSpace(student.ID))
.GroupBy(student => student.ID, StringComparer.OrdinalIgnoreCase)
.ToDictionary(group => group.Key, group => group.First(), StringComparer.OrdinalIgnoreCase);
var generatedCourses = new Dictionary<(int Semester, string SportName), CourseInstance>();
foreach (var student in Settings.Instance.Students)
{
if (!importedById.TryGetValue(student.ID, out var importedStudent))
continue;
for (int semesterIndex = 0; semesterIndex < 4; semesterIndex++)
{
if (importedStudent.Result.Length <= semesterIndex)
continue;
var resultName = importedStudent.Result[semesterIndex];
var sport = ResolveSportFromCourseName(resultName);
if (sport == null)
continue;
var key = (semesterIndex + 1, sport.Name);
if (!generatedCourses.TryGetValue(key, out var course))
{
course = new CourseInstance
{
Sport = sport,
Students = new List<string>()
};
generatedCourses[key] = course;
}
if (!course.Students.Contains(student.ID))
course.Students.Add(student.ID);
}
}
GeneratedCourses = generatedCourses
.Select(entry => (Semester: entry.Key.Semester, Instance: entry.Value))
.OrderBy(course => course.Semester)
.ThenBy(course => course.Instance.Sport.Name)
.ToList();
}
public static void Craft()
{
GeneratedCourses = new();
@@ -48,12 +108,11 @@ public class CourseCrafter
{
if (item.Item2.Count >= item.Item1.MinStudents)
{
int semester = getSemesterForSport(item.Item1);
int semester = getSemesterForSport(item.Item1, item.Item2);
if (semester <= 0) goto semeq0;
var inst = new CourseInstance();
inst.Sport = item.Item1;
inst.Students = new List<string>();
// int dist = 1;
for (int i = item.Item2.Count - 1; i >= 0; i--)
{
if (inst.Students.Count >= inst.Sport.MaxStudents)
@@ -68,7 +127,7 @@ public class CourseCrafter
item.Item2.RemoveAt(i);
}
}
if (inst.Students.Count < inst.Sport.MinStudents)
if (inst.Students.Count < getEffectiveMinStudents(inst.Sport, semester))
{
// Rückgängig machen
foreach (var s in inst.Students)
@@ -79,55 +138,14 @@ public class CourseCrafter
continue; // Kurs nicht erstellen
}
GeneratedCourses.Add((semester, inst));
//MainWindow.Instance.TbResultLog.Text += ($"{semester} -> {inst.Students.Count}\n");
//MainWindow.Instance.TbResultLog.Text += ($"{students_in_semester[0].Count} - {students_in_semester[1].Count} - {students_in_semester[2].Count} - {students_in_semester[3].Count}\n\n");
}
semeq0: ;
}
}
// Kurse auffüllen (mit restl. Leuten)
foreach (var item in initial_sportlist)
{
if (item.Item2.Count > 0)
{
foreach (var ci in GeneratedCourses)
{
FillExistingCourses();
if (item.Item1.ID == ci.Instance.Sport.ID)
{
int semester = ci.Semester;
List<string> added = new();
foreach (string stud in item.Item2)
{
if (ci.Instance.Students.Count >= ci.Instance.Sport.MaxStudents) break;
if (!students_in_semester[semester-1].Contains(stud))
{
ci.Instance.Students.Add(stud);
students_in_semester[semester-1].Add(stud);
//ci.Instance.Students.Add(stud);
added.Add(stud);
}
}
// Hinzugefügte aus Initialkurs entfernen
foreach (string s in added)
{
item.Item2.Remove(s);
}
}
//MainWindow.Instance.TbResultLog.Text += ($"{ci.Semester} -> {ci.Instance.Students.Count}\n");
//MainWindow.Instance.TbResultLog.Text += ($"{students_in_semester[0].Count} - {students_in_semester[1].Count} - {students_in_semester[2].Count} - {students_in_semester[3].Count}\n\n");
}
}
}
// Kurs umdisponieren (besser verteilen)
// Kurs umdisponieren (besser verteilen)
bool changed;
int maxIterations = 20;
@@ -138,15 +156,13 @@ public class CourseCrafter
changed = false;
iteration++;
// nach Sport gruppieren
var sports = GeneratedCourses
.GroupBy(c => c.Instance.Sport.ID);
.GroupBy(c => c.Instance.Sport.Name);
foreach (var sportGroup in sports)
{
var courses = sportGroup.ToList();
// paarweise vergleichen
for (int i = 0; i < courses.Count; i++)
{
for (int j = 0; j < courses.Count; j++)
@@ -156,28 +172,23 @@ public class CourseCrafter
var cA = courses[i];
var cB = courses[j];
// nur sinnvoll, wenn Unterschied
if (cA.Instance.Students.Count <= cB.Instance.Students.Count + 1)
continue;
// Kandidaten aus A nach B verschieben
for (int k = cA.Instance.Students.Count - 1; k >= 0; k--)
{
string stud = cA.Instance.Students[k];
// 1. Zielsemester frei?
if (!isStudentFree(cB.Semester, stud))
continue;
// 2. Zielkurs hat noch Platz?
if (cB.Instance.Students.Count >= cB.Instance.Sport.MaxStudents)
continue;
// 3. Quellkurs darf nicht unter Min fallen
if (cA.Instance.Students.Count - 1 < cA.Instance.Sport.MinStudents)
// Quellkurs darf nicht unter effektives Min fallen
if (cA.Instance.Students.Count - 1 < getEffectiveMinStudents(cA.Instance.Sport, cA.Semester))
continue;
// --- MOVE durchführen ---
cA.Instance.Students.RemoveAt(k);
students_in_semester[cA.Semester - 1].Remove(stud);
@@ -185,7 +196,7 @@ public class CourseCrafter
students_in_semester[cB.Semester - 1].Add(stud);
changed = true;
break; // nach jedem Move neu bewerten
break;
}
}
}
@@ -193,34 +204,18 @@ public class CourseCrafter
} while (changed && iteration < maxIterations);
// // --- Kurse nachträglich aufteilen, um NumCoursesPerSemester exakt zu erreichen ---
// foreach (var course in GeneratedCourses)
// {
// int sem_count_total = GeneratedCourses.Count(tuple => tuple.Semester == course.Semester);
// if (sem_count_total >= Settings.Instance.NumCoursesPerSemester) break;
//
// int sem_count = GeneratedCourses.Count(tuple => tuple.Semester == course.Semester && tuple.Instance.Sport.Name == course.Instance.Sport.Name);
//
//
// if (sem_count < course.Instance.Sport.Semester[course.Semester - 1] && course.Instance.Students.Count >= course.Instance.Sport.MinStudents *2)
// {
// // hier aufteilen
// Console.WriteLine("Könnte aufgeteilt werden.");
// }
// }
//
// --- Kurse nachträglich aufteilen, um NumCoursesPerSemester exakt zu erreichen ---
for (int semester = 1; semester <= 4; semester++)
{
int cancel = 0;
while (GeneratedCourses.Count(c => c.Semester == semester) < Settings.Instance.NumCoursesPerSemester)
while (GeneratedCourses.Count(c => c.Semester == semester) < Settings.Instance.NumCoursesPerSemester[semester-1])
{
cancel++;
if (cancel >= 5) break;
// Kandidaten suchen: splittbare Kurse, deren Sport noch Kapazität hat
if (cancel >= 20) break;
var candidate = GeneratedCourses
.Where(c => c.Semester == semester)
.Where(c => c.Instance.Students.Count >= c.Instance.Sport.MinStudents * 2)
.Where(c => c.Instance.Students.Count >= getEffectiveMinStudents(c.Instance.Sport, c.Semester) * 2)
.Where(c =>
{
int sportCount = GeneratedCourses.Count(g =>
@@ -233,10 +228,19 @@ public class CourseCrafter
.OrderByDescending(c => c.Instance.Students.Count)
.FirstOrDefault();
try
{
if (candidate.Instance == null)
break;
var students = candidate.Instance.Students;
int totalStudents = students.Count;
int movedCount = totalStudents / 2;
int remainingCount = totalStudents - movedCount;
if (movedCount < getEffectiveMinStudents(candidate.Instance.Sport, candidate.Semester) ||
remainingCount < getEffectiveMinStudents(candidate.Instance.Sport, candidate.Semester))
{
break;
}
var newCourse = new CourseInstance
{
@@ -244,43 +248,152 @@ public class CourseCrafter
Students = new List<string>()
};
List<string> moved = new();
var moved = students
.Skip(remainingCount)
.Take(movedCount)
.ToList();
for (int i = students.Count - 1; i >= 0; i--)
{
string stud = students[i];
if (newCourse.Students.Count >= candidate.Instance.Sport.MaxStudents)
break;
// Sicherstellen, dass beide Kurse >= MinStudents bleiben
if (students.Count - moved.Count <= candidate.Instance.Sport.MinStudents)
break;
newCourse.Students.Add(stud);
moved.Add(stud);
}
// Validierung
if (newCourse.Students.Count < candidate.Instance.Sport.MinStudents)
break;
// Move durchführen
foreach (var s in moved)
{
candidate.Instance.Students.Remove(s);
newCourse.Students.Add(s);
}
GeneratedCourses.Add((semester, newCourse));
}
catch
}
bool rescueChanged;
do
{
rescueChanged = FillExistingCourses();
foreach (var item in initial_sportlist)
{
if (TryCreateRescueCourse(item.Item1, item.Item2))
{
rescueChanged = true;
}
}
} while (rescueChanged);
OptimizeStudentWishCoverage();
// --- Ringtausch: Schüler mit 4 Wünschen aber < 4 Zuteilungen neu zuordnen ---
foreach (var student in Settings.Instance.Students)
{
if (student.SelectedCourseNames.Count < 4)
continue;
var currentAssignments = GetAssignmentsBySemester(student.ID);
int currentCount = currentAssignments.Count(a => a != null);
if (currentCount >= 4)
continue;
// Alle aktuellen Zuteilungen entfernen (temporär)
foreach (var assignment in currentAssignments)
{
if (assignment == null) continue;
assignment.Value.Instance.Students.Remove(student.ID);
students_in_semester[assignment.Value.Semester - 1].Remove(student.ID);
}
// Alle gewünschten Sports ermitteln
var desiredSports = student.SelectedCourseNames
.Select(ResolveSportFromSelection)
.Where(s => s != null)
.DistinctBy(s => s!.Name)
.Select(s => s!)
.ToList();
// Kandidatenkurse pro Semester: Kurse mit freiem Platz, die einem Wunschsport entsprechen
var candidatesPerSemester = new List<(int Semester, CourseInstance Instance)>[4];
for (int si = 0; si < 4; si++)
{
candidatesPerSemester[si] = GeneratedCourses
.Where(c => c.Semester == si + 1)
.Where(c => c.Instance.Students.Count < c.Instance.Sport.MaxStudents)
.Where(c => desiredSports.Any(ds => ds.Name == c.Instance.Sport.Name))
.ToList();
candidatesPerSemester[si].Insert(0, default); // default = kein Kurs
}
(int Semester, CourseInstance Instance)?[] bestAssignment = currentAssignments;
int bestScore = currentCount;
int s0 = candidatesPerSemester[0].Count;
int s1 = candidatesPerSemester[1].Count;
int s2 = candidatesPerSemester[2].Count;
int s3 = candidatesPerSemester[3].Count;
var sportSelectedCounts = student.SelectedCourseNames
.Select(ResolveSportFromSelection)
.Where(sp => sp != null)
.GroupBy(sp => sp!.Name)
.ToDictionary(g => g.Key, g => g.Count());
for (int i0 = 0; i0 < s0; i0++)
for (int i1 = 0; i1 < s1; i1++)
for (int i2 = 0; i2 < s2; i2++)
for (int i3 = 0; i3 < s3; i3++)
{
var c0 = candidatesPerSemester[0][i0];
var c1 = candidatesPerSemester[1][i1];
var c2 = candidatesPerSemester[2][i2];
var c3 = candidatesPerSemester[3][i3];
var chosen = new (int Semester, CourseInstance Instance)?[]
{
c0.Instance != null ? (c0.Semester, c0.Instance) : null,
c1.Instance != null ? (c1.Semester, c1.Instance) : null,
c2.Instance != null ? (c2.Semester, c2.Instance) : null,
c3.Instance != null ? (c3.Semester, c3.Instance) : null
};
bool valid = true;
var sportUsageCounts = new Dictionary<string, int>();
for (int si = 0; si < 4; si++)
{
if (chosen[si] == null) continue;
string sportName = chosen[si]!.Value.Instance.Sport.Name;
sportUsageCounts.TryGetValue(sportName, out int used);
int allowed = sportSelectedCounts.GetValueOrDefault(sportName, 0);
if (used >= allowed) { valid = false; break; }
sportUsageCounts[sportName] = used + 1;
}
if (!valid) continue;
int score = chosen.Count(c => c != null);
if (score > bestScore)
{
bestScore = score;
bestAssignment = chosen;
}
}
// Beste Zuteilung anwenden
foreach (var assignment in bestAssignment)
{
if (assignment == null) continue;
assignment.Value.Instance.Students.Add(student.ID);
students_in_semester[assignment.Value.Semester - 1].Add(student.ID);
}
}
BalanceCoursesBetweenSameSportAndSemester();
// ---------------------------------------------------------------------------
// Lokale Hilfsfunktionen
// ---------------------------------------------------------------------------
int getEffectiveMinStudents(Sport sport, int semester)
{
int reduction = (semester >= 3) ? 2 : 0;
return Math.Max(1, sport.MinStudents - reduction);
}
bool isStudentFree(int semester, string studentID)
{
@@ -289,21 +402,66 @@ public class CourseCrafter
if (semester != inst.Semester) continue;
foreach (string stud in inst.Instance.Students)
{
if (stud == studentID) return false; // Schüler in genanntem Semester bereits gefunden
if (stud == studentID) return false;
}
}
// Schüler nicht gefunden:
return true;
}
void BalanceCoursesBetweenSameSportAndSemester()
{
bool changed;
do
{
changed = false;
var groups = GeneratedCourses
.GroupBy(c => (c.Semester, c.Instance.Sport.Name))
.Where(g => g.Count() > 1);
foreach (var group in groups)
{
var courses = group
.OrderBy(c => c.Instance.Students.Count)
.ToList();
for (int sourceIndex = courses.Count - 1; sourceIndex > 0; sourceIndex--)
{
var sourceCourse = courses[sourceIndex];
for (int targetIndex = 0; targetIndex < sourceIndex; targetIndex++)
{
var targetCourse = courses[targetIndex];
if (sourceCourse.Instance.Students.Count <= targetCourse.Instance.Students.Count + 1)
break;
if (targetCourse.Instance.Students.Count >= targetCourse.Instance.Sport.MaxStudents)
continue;
if (sourceCourse.Instance.Students.Count - 1 < getEffectiveMinStudents(sourceCourse.Instance.Sport, sourceCourse.Semester))
continue;
var studentId = sourceCourse.Instance.Students[^1];
sourceCourse.Instance.Students.RemoveAt(sourceCourse.Instance.Students.Count - 1);
targetCourse.Instance.Students.Add(studentId);
changed = true;
break;
}
if (changed)
break;
}
if (changed)
break;
}
} while (changed);
}
bool requestExit()
{
globalCount++;
// max Kursanzahl
if (GeneratedCourses.Count >= Settings.Instance.NumCoursesPerSemester * 4) return true;
int sum = Settings.Instance.NumCoursesPerSemester[0] + Settings.Instance.NumCoursesPerSemester[1] +
Settings.Instance.NumCoursesPerSemester[2] + Settings.Instance.NumCoursesPerSemester[3];
if (GeneratedCourses.Count >= sum) return true;
int low = 0;
foreach (var item in initial_sportlist)
@@ -312,92 +470,435 @@ public class CourseCrafter
}
if (low >= initial_sportlist.Count) return true;
if (globalCount >= 20) return true;
return false;
}
int total_missing = 0;
foreach (var tuple in initial_sportlist)
{
MainWindow.Instance.TbResultTextout.Text += $"{tuple.Item1}: {tuple.Item2.Count} remaining\n";
int[] missingPerSemester = new int[4];
foreach (var studentId in tuple.Item2.Distinct())
{
for (int semesterIndex = 0; semesterIndex < 4; semesterIndex++)
{
if (!students_in_semester[semesterIndex].Contains(studentId))
{
missingPerSemester[semesterIndex]++;
total_missing++;
}
}
}
// ...existing code...
int getSemesterForSport2(Sport sp)
MainWindow.Instance.TbResultTextout.Text +=
$"{tuple.Item1}: {tuple.Item2.Count} remaining ({missingPerSemester[0]},{missingPerSemester[1]},{missingPerSemester[2]},{missingPerSemester[3]})\n";
}
MainWindow.Instance.TbResultTextout.Text += $"\n total remaining: {total_missing}";
bool FillExistingCourses()
{
int[] semcount = new int[4];
bool changed = false;
foreach (var inst in GeneratedCourses)
semcount[inst.Semester - 1]++;
int bestSem = 0;
int minCourses = int.MaxValue;
for (int i = 0; i < 4; i++)
foreach (var item in initial_sportlist)
{
if (sp.Semester[i] == 0) continue;
// prüfen, ob für diesen Sport im Semester i schon die maximale Anzahl erreicht ist
int sportCoursesInSemester = GeneratedCourses
.Count(g => g.Semester == i + 1 && g.Instance.Sport.ID == sp.ID);
if (sportCoursesInSemester >= sp.Semester[i])
if (item.Item2.Count == 0)
continue;
if (semcount[i] < Settings.Instance.NumCoursesPerSemester &&
semcount[i] < minCourses)
foreach (var ci in GeneratedCourses
.Where(ci => ci.Instance.Sport.Name == item.Item1.Name)
.OrderBy(ci => ci.Instance.Students.Count))
{
minCourses = semcount[i];
bestSem = i + 1;
int semester = ci.Semester;
List<string> added = new();
foreach (string stud in item.Item2)
{
if (ci.Instance.Students.Count >= ci.Instance.Sport.MaxStudents)
break;
if (students_in_semester[semester - 1].Contains(stud))
continue;
ci.Instance.Students.Add(stud);
students_in_semester[semester - 1].Add(stud);
added.Add(stud);
}
foreach (string s in added)
{
item.Item2.Remove(s);
changed = true;
}
}
}
return bestSem;
return changed;
}
int getSemesterForSport(Sport sp)
bool TryCreateRescueCourse(Sport sport, List<string> remainingStudents)
{
if (remainingStudents.Count == 0)
return false;
for (int semester = 1; semester <= 4; semester++)
{
int semesterIndex = semester - 1;
if (sport.Semester[semesterIndex] == 0)
continue;
int sportCoursesInSemester = GeneratedCourses.Count(c =>
c.Semester == semester && c.Instance.Sport.Name == sport.Name);
if (sportCoursesInSemester >= sport.Semester[semesterIndex])
continue;
if (GeneratedCourses.Count(c => c.Semester == semester) >= Settings.Instance.NumCoursesPerSemester[semester-1])
continue;
var directlyPlaceable = remainingStudents
.Distinct()
.Where(studentId => !students_in_semester[semesterIndex].Contains(studentId))
.ToList();
var donorMoves = new List<((int Semester, CourseInstance Instance) Course, string StudentId)>();
foreach (var course in GeneratedCourses
.Where(c => c.Instance.Sport.Name == sport.Name && c.Semester != semester)
.OrderByDescending(c => c.Instance.Students.Count))
{
int movableCount = course.Instance.Students.Count - getEffectiveMinStudents(sport, course.Semester);
if (movableCount <= 0)
continue;
foreach (var studentId in course.Instance.Students.ToList())
{
if (movableCount <= 0)
break;
if (students_in_semester[semesterIndex].Contains(studentId))
continue;
donorMoves.Add((course, studentId));
movableCount--;
}
}
if (directlyPlaceable.Count + donorMoves.Count < getEffectiveMinStudents(sport, semester))
continue;
var newCourse = new CourseInstance
{
Sport = sport,
Students = new List<string>()
};
foreach (var studentId in directlyPlaceable)
{
if (newCourse.Students.Count >= sport.MaxStudents)
break;
newCourse.Students.Add(studentId);
}
foreach (var move in donorMoves)
{
if (newCourse.Students.Count >= sport.MaxStudents)
break;
if (newCourse.Students.Contains(move.StudentId))
continue;
move.Course.Instance.Students.Remove(move.StudentId);
students_in_semester[move.Course.Semester - 1].Remove(move.StudentId);
newCourse.Students.Add(move.StudentId);
}
if (newCourse.Students.Count < getEffectiveMinStudents(sport, semester))
continue;
foreach (var studentId in newCourse.Students)
{
remainingStudents.Remove(studentId);
students_in_semester[semesterIndex].Add(studentId);
}
GeneratedCourses.Add((semester, newCourse));
return true;
}
return false;
}
void OptimizeStudentWishCoverage()
{
bool changed;
int iterations = 0;
do
{
changed = false;
iterations++;
foreach (var student in Settings.Instance.Students)
{
if (TryImproveStudentCoverage(student))
{
changed = true;
}
}
} while (changed && iterations < 20);
}
bool TryImproveStudentCoverage(Student student)
{
var assignments = GetAssignmentsBySemester(student.ID);
if (assignments.All(a => a != null))
return false;
var openSemesters = Enumerable.Range(0, 4)
.Where(i => assignments[i] == null)
.Select(i => i + 1)
.ToList();
if (openSemesters.Count == 0)
return false;
var assignedSports = assignments
.Where(a => a != null)
.Select(a => a!.Value.Instance.Sport.Name)
.ToHashSet();
var preferredSports = student.SelectedCourseNames
.Select(ResolveSportFromSelection)
.Where(sport => sport != null)
.DistinctBy(sport => sport!.Name)
.Select(sport => sport!)
.Where(sport => !assignedSports.Contains(sport.Name))
.ToList();
foreach (var desiredSport in preferredSports)
{
for (int sourceSemester = 1; sourceSemester <= 4; sourceSemester++)
{
var currentCourse = assignments[sourceSemester - 1];
if (currentCourse == null)
continue;
var currentCourseValue = currentCourse.Value;
var desiredCourse = GeneratedCourses
.FirstOrDefault(course =>
course.Semester == sourceSemester &&
course.Instance.Sport.Name == desiredSport.Name &&
course.Instance.Students.Count < course.Instance.Sport.MaxStudents);
if (desiredCourse.Instance == null)
continue;
foreach (var targetSemester in openSemesters)
{
var relocationTarget = GeneratedCourses
.FirstOrDefault(course =>
course.Semester == targetSemester &&
course.Instance.Sport.Name == currentCourseValue.Instance.Sport.Name &&
course.Instance.Students.Count < course.Instance.Sport.MaxStudents);
if (relocationTarget.Instance == null)
continue;
if (!CanRelocateStudent(student.ID, currentCourseValue, relocationTarget))
continue;
currentCourseValue.Instance.Students.Remove(student.ID);
students_in_semester[sourceSemester - 1].Remove(student.ID);
relocationTarget.Instance.Students.Add(student.ID);
students_in_semester[targetSemester - 1].Add(student.ID);
desiredCourse.Instance.Students.Add(student.ID);
students_in_semester[sourceSemester - 1].Add(student.ID);
return true;
}
}
}
return false;
}
bool CanRelocateStudent(
string studentId,
(int Semester, CourseInstance Instance) sourceCourse,
(int Semester, CourseInstance Instance) targetCourse)
{
if (sourceCourse.Semester == targetCourse.Semester)
return false;
if (sourceCourse.Instance.Students.Count - 1 < getEffectiveMinStudents(sourceCourse.Instance.Sport, sourceCourse.Semester))
return false;
if (targetCourse.Instance.Students.Count >= targetCourse.Instance.Sport.MaxStudents)
return false;
if (!isStudentFree(targetCourse.Semester, studentId))
return false;
return true;
}
bool EnsureFirstSemesterCoverage()
{
bool changed = false;
foreach (var student in Settings.Instance.Students)
{
if (student.SelectedCourseNames.Count == 0)
continue;
var assignments = GetAssignmentsBySemester(student.ID);
if (assignments[0] != null)
continue;
if (TryAssignStudentToFirstSemester(student))
changed = true;
}
return changed;
}
bool TryAssignStudentToFirstSemester(Student student)
{
var desiredSports = student.SelectedCourseNames
.Select(ResolveSportFromSelection)
.Where(sp => sp != null && sp.Semester[0] != 0)
.DistinctBy(sp => sp!.Name)
.Select(sp => sp!)
.ToList();
foreach (var sport in desiredSports)
{
// Direkt in bestehendes Kursangebot eintragen
var existingCourse = GeneratedCourses
.FirstOrDefault(c => c.Semester == 1 && c.Instance.Sport.Name == sport.Name &&
c.Instance.Students.Count < c.Instance.Sport.MaxStudents);
if (existingCourse.Instance != null)
{
existingCourse.Instance.Students.Add(student.ID);
students_in_semester[0].Add(student.ID);
return true;
}
// Versuche einen Teilnehmer aus einem bestehenden Sem.1-Kurs umzudisponieren
var firstSemesterCourses = GeneratedCourses
.Where(c => c.Semester == 1 && c.Instance.Sport.Name == sport.Name)
.ToList();
foreach (var course in firstSemesterCourses)
{
if (course.Instance.Students.Count <= getEffectiveMinStudents(sport, 1))
continue;
foreach (var otherStudent in course.Instance.Students.ToList())
{
if (otherStudent == student.ID)
continue;
var targetCourse = GeneratedCourses
.FirstOrDefault(c => c.Semester != 1 && c.Instance.Sport.Name == sport.Name &&
c.Instance.Students.Count < c.Instance.Sport.MaxStudents &&
!students_in_semester[c.Semester - 1].Contains(otherStudent));
if (targetCourse.Instance == null)
continue;
course.Instance.Students.Remove(otherStudent);
students_in_semester[0].Remove(otherStudent);
targetCourse.Instance.Students.Add(otherStudent);
students_in_semester[targetCourse.Semester - 1].Add(otherStudent);
course.Instance.Students.Add(student.ID);
students_in_semester[0].Add(student.ID);
return true;
}
}
}
return false;
}
(int Semester, CourseInstance Instance)?[] GetAssignmentsBySemester(string studentId)
{
var assignments = new (int Semester, CourseInstance Instance)?[4];
foreach (var course in GeneratedCourses)
{
if (course.Instance.Students.Contains(studentId))
{
assignments[course.Semester - 1] = course;
}
}
return assignments;
}
Sport? ResolveSportFromSelection(string selectedCourseName)
{
return ResolveSportFromCourseName(selectedCourseName);
}
int getSemesterForSport(Sport sp, List<string> interestedStudents)
{
// 1. Zähle alle Kurse pro Semester (egal welche Sportart)
int[] totalCoursesPerSemester = new int[4];
foreach (var inst in GeneratedCourses)
totalCoursesPerSemester[inst.Semester - 1]++;
int bestSem = 0;
int minCourses = int.MaxValue;
int maxFreeInterestedStudents = -1;
// 2. Kandidaten-Semester durchgehen
for (int i = 0; i < 4; i++)
{
// a) Sport darf in diesem Semester gar nicht stattfinden?
if (sp.Semester[i] == 0)
continue;
int semesterNumber = i + 1;
// b) Wie viele Kurse DIESES Sports gibt es schon in diesem Semester?
int sportCoursesInThisSemester = GeneratedCourses
.Count(g => g.Semester == semesterNumber &&
g.Instance.Sport.Name == sp.Name);
// c) Pro-Sport-Limit erreicht?
if (sportCoursesInThisSemester >= sp.Semester[i])
continue;
// d) Globales Limit pro Semester erreicht?
if (totalCoursesPerSemester[i] >= Settings.Instance.NumCoursesPerSemester)
if (totalCoursesPerSemester[i] >= Settings.Instance.NumCoursesPerSemester[i])
continue;
// e) Wähle das Semester mit bisher insgesamt den wenigsten Kursen
if (totalCoursesPerSemester[i] < minCourses)
int freeInterestedStudents = interestedStudents
.Distinct()
.Count(studentId => !students_in_semester[i].Contains(studentId));
if (freeInterestedStudents < getEffectiveMinStudents(sp, semesterNumber))
continue;
if (freeInterestedStudents > maxFreeInterestedStudents ||
(freeInterestedStudents == maxFreeInterestedStudents &&
(totalCoursesPerSemester[i] < minCourses ||
(totalCoursesPerSemester[i] == minCourses && semesterNumber < bestSem))))
{
maxFreeInterestedStudents = freeInterestedStudents;
minCourses = totalCoursesPerSemester[i];
bestSem = semesterNumber;
}
}
return bestSem; // 0, falls kein zulässiges Semester gefunden
return bestSem;
}
EnsureFirstSemesterCoverage();
ReloadResult();
}
public static void ReloadResult()
{
var errors = ValidateCourses(GeneratedCourses);
if (errors.Count == 0)
@@ -411,6 +912,11 @@ public class CourseCrafter
MainWindow.Instance.TbResultLog.Text += e + "\n";
}
foreach (var student in Settings.Instance.Students)
{
student.Result = new string[4];
}
foreach (var course in GeneratedCourses)
{
foreach (var student in Settings.Instance.Students)
@@ -421,9 +927,7 @@ public class CourseCrafter
}
}
}
}
public static string GenerateStatistics()
{
GeneratedCourses.Sort((x,y) => x.Semester.CompareTo(y.Semester) );
+48
View File
@@ -1,9 +1,19 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.IO;
using System.Text.Json;
namespace spplus;
public static class ExportUtility
{
private sealed class ConfigurationExport
{
public List<Sport> Sports { get; set; } = [];
public int[] NumCoursesPerSemester { get; set; } = [];
}
public static void ExportToCSV(string filepath)
{
char separator = ',';
@@ -16,4 +26,42 @@ public static class ExportUtility
File.WriteAllText(filepath, output);
}
private sealed class ResultExportEntry
{
public int Semester { get; set; }
public string SportName { get; set; } = string.Empty;
public List<string> Students { get; set; } = new();
}
public static void ExportResultsToJson(string filepath)
{
var list = CourseCrafter.GeneratedCourses
.Select(g => new ResultExportEntry
{
Semester = g.Semester,
SportName = g.Instance.Sport.Name,
Students = g.Instance.Students.ToList()
})
.ToList();
var json = JsonSerializer.Serialize(list, new JsonSerializerOptions { WriteIndented = true });
File.WriteAllText(filepath, json);
}
public static void ExportConfigurationToJson(string filepath)
{
var export = new ConfigurationExport
{
Sports = Settings.Instance.Sports,
NumCoursesPerSemester = Settings.Instance.NumCoursesPerSemester
};
var json = JsonSerializer.Serialize(export, new JsonSerializerOptions
{
WriteIndented = true
});
File.WriteAllText(filepath, json);
}
}
Binary file not shown.

After

Width:  |  Height:  |  Size: 82 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 194 KiB

+78
View File
@@ -1,6 +1,7 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Text.Json;
using System.Linq;
namespace spplus;
@@ -47,4 +48,81 @@ public static class import
return result;
}
public static List<Student> ImportResultFromFile(string path)
{
var dict = new Dictionary<string, string[]>(StringComparer.OrdinalIgnoreCase);
foreach (var line in File.ReadLines(path).Skip(1)) // Header überspringen
{
if (string.IsNullOrWhiteSpace(line))
continue;
var parts = line.Split(',', StringSplitOptions.None);
if (parts.Length < 5)
continue;
var studentId = parts[0].Trim();
dict[studentId] = new[]
{
parts[1].Trim(),
parts[2].Trim(),
parts[3].Trim(),
parts[4].Trim()
};
}
return dict
.Select(entry => new Student(entry.Key, string.Empty, new List<string>())
{
Result = entry.Value
})
.ToList();
}
private sealed class ResultImportEntry
{
public int Semester { get; set; }
public string SportName { get; set; } = string.Empty;
public List<string> Students { get; set; } = new();
}
public static List<(int Semester, CourseCrafter.CourseInstance Instance)> ImportResultFromJson(string path)
{
try
{
var json = File.ReadAllText(path);
var entries = JsonSerializer.Deserialize<List<ResultImportEntry>>(json);
if (entries == null) return new();
var result = new List<(int Semester, CourseCrafter.CourseInstance Instance)>();
foreach (var e in entries)
{
if (string.IsNullOrWhiteSpace(e.SportName))
continue;
var sport = Settings.Instance.Sports.FirstOrDefault(s =>
string.Equals(s.Name, e.SportName, StringComparison.OrdinalIgnoreCase) ||
s.AlternativeNames.Any(a => string.Equals(a, e.SportName, StringComparison.OrdinalIgnoreCase)));
if (sport == null)
continue;
var ci = new CourseCrafter.CourseInstance
{
Sport = sport,
Students = e.Students.Distinct().ToList()
};
result.Add((e.Semester, ci));
}
return result;
}
catch
{
return new();
}
}
}
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
View File
Binary file not shown.

After

Width:  |  Height:  |  Size: 106 KiB

BIN
View File
Binary file not shown.

After

Width:  |  Height:  |  Size: 67 KiB

+21
View File
@@ -5,6 +5,7 @@
<Nullable>enable</Nullable>
<ApplicationManifest>app.manifest</ApplicationManifest>
<AvaloniaUseCompiledBindingsByDefault>true</AvaloniaUseCompiledBindingsByDefault>
<ApplicationIcon>res/logo.ico</ApplicationIcon>
</PropertyGroup>
<ItemGroup>
@@ -18,5 +19,25 @@
<PrivateAssets Condition="'$(Configuration)' != 'Debug'">All</PrivateAssets>
</PackageReference>
<PackageReference Include="Lucide.Avalonia" Version="0.1.35"/>
<PackageReference Include="PdfSharp" Version="6.2.4"/>
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.8"/>
</ItemGroup>
<ItemGroup>
<None Remove="res\logo.ico"/>
<AvaloniaResource Include="res\logo.ico">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</AvaloniaResource>
<None Remove="res\logo.png"/>
<AvaloniaResource Include="res\logo.png">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</AvaloniaResource>
</ItemGroup>
<ItemGroup>
<None Include="res/fonts/*.ttf">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</None>
</ItemGroup>
</Project>
+17 -14
View File
@@ -85,7 +85,7 @@ public class Settings
public List<Student> Students { get; set; } = [];
public List<Sport> Sports { get; set; } = [];
public int NumCoursesPerSemester { get; set; } = 10;
public int[] NumCoursesPerSemester { get; set; } = [10,10,11,11]; // Exact Amount of courses, not a maximum
public Settings()
{
@@ -99,19 +99,22 @@ public class Settings
public static void ImportInitial()
{
Instance.Sports.Add(new Sport("Tischtennis"){ AlternativeNames = {"Sport_TT"}});
Instance.Sports.Add(new Sport("Badminton"){ AlternativeNames = {"Sport_BM"}});
Instance.Sports.Add(new Sport("Gymnastik/Tanz"){ AlternativeNames = {"Sport_Gym"}});
Instance.Sports.Add(new Sport("Schwimmen"){ AlternativeNames = {"Sport_SW"}, Semester = [1, 1, 1, 1], MaxStudents = 18});
Instance.Sports.Add(new Sport("Bouldern"){ AlternativeNames = {"Sport_BO"}, Semester = [1, 1, 1, 1]});
Instance.Sports.Add(new Sport("Basketball"){ AlternativeNames = {"Sport_BS"}});
Instance.Sports.Add(new Sport("Fitness"){ AlternativeNames = {"Sport_Fit"}});
Instance.Sports.Add(new Sport("Fußball"){ AlternativeNames = {"Sport_Fuß"}, Semester = [1, 0, 1, 0]});
Instance.Sports.Add(new Sport("Handball"){ AlternativeNames = {"Sport_HB"}});
Instance.Sports.Add(new Sport("Leichtathletik"){ AlternativeNames = {"Sport_LA"}, Semester = [1, 0, 1, 0], MaxStudents = 18});
Instance.Sports.Add(new Sport("Tennis"){ AlternativeNames = {"Sport_Te"}});
Instance.Sports.Add(new Sport("Turnen"){ AlternativeNames = {"Sport_Tur"}});
Instance.Sports.Add(new Sport("Volleyball"){ AlternativeNames = {"Sport_VB"}});
Instance.Sports.Clear();
int id = 1;
Instance.Sports.Add(new Sport("Tischtennis"){ ID = id++, AlternativeNames = {"Sport_TT"}});
Instance.Sports.Add(new Sport("Badminton"){ ID = id++, AlternativeNames = {"Sport_BM"}});
Instance.Sports.Add(new Sport("Gymnastik/Tanz"){ ID = id++, AlternativeNames = {"Sport_Gym"}});
Instance.Sports.Add(new Sport("Schwimmen"){ ID = id++, AlternativeNames = {"Sport_SW"}, Semester = [1, 1, 1, 1], MaxStudents = 18});
Instance.Sports.Add(new Sport("Bouldern"){ ID = id++, AlternativeNames = {"Sport_BO"}, Semester = [1, 1, 1, 1]});
Instance.Sports.Add(new Sport("Basketball"){ ID = id++, AlternativeNames = {"Sport_BS"}});
Instance.Sports.Add(new Sport("Fitness"){ ID = id++, AlternativeNames = {"Sport_Fit"}});
Instance.Sports.Add(new Sport("Fußball"){ ID = id++, AlternativeNames = {"Sport_Fuß"}, Semester = [1, 0, 1, 0]});
Instance.Sports.Add(new Sport("Handball"){ ID = id++, AlternativeNames = {"Sport_HB"}});
Instance.Sports.Add(new Sport("Leichtathletik"){ ID = id++, AlternativeNames = {"Sport_LA"}, Semester = [1, 0, 1, 0], MaxStudents = 18});
Instance.Sports.Add(new Sport("Tennis"){ ID = id++, AlternativeNames = {"Sport_Te"}});
Instance.Sports.Add(new Sport("Turnen"){ ID = id++, AlternativeNames = {"Sport_Tur"}});
Instance.Sports.Add(new Sport("Volleyball"){ ID = id++, AlternativeNames = {"Sport_VB"}});
}
public static string? GetSportNameFromID(int id)