Compare commits
	
		
			41 Commits
		
	
	
		
			823cb859ea
			...
			dev/pdf
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| a634b7a69e | |||
| e1f0f85401 | |||
| b6644f3584 | |||
| 7eda8eb538 | |||
| 0d7696cbc6 | |||
| 5001f24d9b | |||
| ea8e78fd9f | |||
| 6da58d6753 | |||
| 89eb5d255d | |||
| 1b8fb747e8 | |||
| 74cded42d8 | |||
| 22350142fc | |||
| 659fb80049 | |||
| cbc4028f8d | |||
| e4d423385a | |||
| c9c2d801b0 | |||
| 94c7c8a36e | |||
| d69ec600cd | |||
| 95d5c4ab9d | |||
| bf841ad5c6 | |||
| a1aae9dc56 | |||
| 750fb1ff58 | |||
| f4e9915e7f | |||
| 18046bbe18 | |||
| 75929e3b7d | |||
| 627f5b7e5b | |||
| 9e5dc760d5 | |||
| 0ffb910e37 | |||
| 566776910a | |||
| 4d00143a74 | |||
| c093127a8c | |||
| 3dd4b134c8 | |||
| 7e27c944f3 | |||
| 5fbe53faf6 | |||
| 15a2a9c075 | |||
| 90193e9346 | |||
| e8f1113293 | |||
| db6fc10c28 | |||
| 55b0332600 | |||
| 0e1e0b2de0 | |||
| 7ceef2c344 | 
| @@ -23,7 +23,9 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v4 | ||||||
|  |         with: | ||||||
|  |           # Disabling shallow clone is recommended for improving relevancy of reporting | ||||||
|  |           fetch-depth: 0 | ||||||
|       - name: Setup go |       - name: Setup go | ||||||
|         uses: actions/setup-go@v5 |         uses: actions/setup-go@v5 | ||||||
|         with: |         with: | ||||||
| @@ -45,4 +47,25 @@ jobs: | |||||||
|           restore-keys: |- |           restore-keys: |- | ||||||
|             arbeitszeitmessung- |             arbeitszeitmessung- | ||||||
|       - name: Run Go Tests |       - name: Run Go Tests | ||||||
|         run: cd Backend && go test ./... |         run: cd Backend && mkdir .test && go test ./... -coverprofile=.test/coverage.out -json > .test/report.json | ||||||
|  |       - name: Verify coverage report exists | ||||||
|  |         run: | | ||||||
|  |           if [ -f "Backend/.test/coverage.out" ]; then | ||||||
|  |             echo "Coverage report found" | ||||||
|  |           else | ||||||
|  |             echo "Coverage report not found" | ||||||
|  |           fi | ||||||
|  |       - uses: SonarSource/sonarqube-scan-action@v6 | ||||||
|  |         env: | ||||||
|  |           SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} | ||||||
|  |           SONAR_HOST_URL: ${{ secrets.SONAR_HOST_URL }} | ||||||
|  |         with: | ||||||
|  |           projectBaseDir: Backend | ||||||
|  |           args: > | ||||||
|  |             -Dsonar.projectVersion=${{ gitea.sha_short }} | ||||||
|  |       - uses: SonarSource/sonarqube-quality-gate-action@v1 | ||||||
|  |         timeout-minutes: 5 | ||||||
|  |         env: | ||||||
|  |           SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} | ||||||
|  |         with: | ||||||
|  |           scanMetadataReportFile: Backend/.scannerwork/report-task.txt | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ RUN go mod download && go mod verify | |||||||
| COPY . . | COPY . . | ||||||
| RUN go build -o server . | RUN go build -o server . | ||||||
|  |  | ||||||
| FROM alpine | FROM alpine:3.22 | ||||||
| RUN apk add --no-cache tzdata | RUN apk add --no-cache tzdata | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| COPY --from=build /app/server /app/server | COPY --from=build /app/server /app/server | ||||||
|   | |||||||
							
								
								
									
										161
									
								
								Backend/endpoints/pdf-create.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										161
									
								
								Backend/endpoints/pdf-create.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,161 @@ | |||||||
|  | package endpoints | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"arbeitszeitmessung/helper" | ||||||
|  | 	"arbeitszeitmessung/helper/paramParser" | ||||||
|  | 	"arbeitszeitmessung/models" | ||||||
|  | 	"bytes" | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
|  | 	"log/slog" | ||||||
|  | 	"net/http" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/Dadido3/go-typst" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func convertDaysToTypst(days []models.IWorkDay, u models.User) ([]typstDay, error) { | ||||||
|  | 	var typstDays []typstDay | ||||||
|  | 	for _, day := range days { | ||||||
|  | 		var thisTypstDay typstDay | ||||||
|  | 		work, pause, overtime := day.GetAllWorkTimesVirtual(u) | ||||||
|  | 		thisTypstDay.Date = day.Date().Format("02.01.2006") | ||||||
|  | 		thisTypstDay.Worktime = helper.FormatDurationFill(work, true) | ||||||
|  | 		thisTypstDay.Pausetime = helper.FormatDurationFill(pause, true) | ||||||
|  | 		thisTypstDay.Overtime = helper.FormatDurationFill(overtime, true) | ||||||
|  | 		thisTypstDay.IsFriday = day.Date().Weekday() == time.Friday | ||||||
|  |  | ||||||
|  | 		thisTypstDay.DayParts = convertDayToTypstDayParts(day, u) | ||||||
|  | 		typstDays = append(typstDays, thisTypstDay) | ||||||
|  | 	} | ||||||
|  | 	return typstDays, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func convertDayToTypstDayParts(day models.IWorkDay, user models.User) []typstDayPart { | ||||||
|  | 	var typstDayParts []typstDayPart | ||||||
|  | 	if day.IsWorkDay() { | ||||||
|  | 		workDay, _ := day.(*models.WorkDay) | ||||||
|  | 		for i := 0; i < len(workDay.Bookings); i += 2 { | ||||||
|  | 			var typstDayPart typstDayPart | ||||||
|  | 			typstDayPart.BookingFrom = workDay.Bookings[i].Timestamp.Format("15:04") | ||||||
|  | 			typstDayPart.BookingTo = workDay.Bookings[i+1].Timestamp.Format("15:04") | ||||||
|  | 			typstDayPart.WorkType = workDay.Bookings[i].BookingType.Name | ||||||
|  | 			typstDayPart.IsWorkDay = true | ||||||
|  | 			typstDayParts = append(typstDayParts, typstDayPart) | ||||||
|  | 		} | ||||||
|  | 		if day.IsKurzArbeit() { | ||||||
|  | 			tsFrom, tsTo := workDay.GenerateKurzArbeitBookings(user) | ||||||
|  | 			typstDayParts = append(typstDayParts, typstDayPart{ | ||||||
|  | 				BookingFrom: tsFrom.Format("15:04"), | ||||||
|  | 				BookingTo:   tsTo.Format("15:04"), | ||||||
|  | 				WorkType:    "Kurzarbeit", | ||||||
|  | 				IsWorkDay:   true, | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		absentDay, _ := day.(*models.Absence) | ||||||
|  | 		typstDayParts = append(typstDayParts, typstDayPart{IsWorkDay: false, WorkType: absentDay.AbwesenheitTyp.Name}) | ||||||
|  | 	} | ||||||
|  | 	return typstDayParts | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func renderPDF(days []typstDay, metadata typstMetadata) (bytes.Buffer, error) { | ||||||
|  | 	var markup bytes.Buffer | ||||||
|  | 	var output bytes.Buffer | ||||||
|  |  | ||||||
|  | 	if err := typst.InjectValues(&markup, map[string]any{"meta": metadata, "days": days}); err != nil { | ||||||
|  | 		return output, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Import the template and invoke the template function with the custom data. | ||||||
|  | 	// Show is used to replace the current document with whatever content the template function in `template.typ` returns. | ||||||
|  | 	markup.WriteString(` | ||||||
|  | 		#import "template.typ": abrechnung | ||||||
|  | 		#show: doc => abrechnung(meta, days) | ||||||
|  | 		`) | ||||||
|  |  | ||||||
|  | 	// Compile the prepared markup with Typst and write the result it into `output.pdf`. | ||||||
|  | 	// f, err := os.Create("output.pdf") | ||||||
|  | 	// if err != nil { | ||||||
|  | 	// 	log.Panicf("Failed to create output file: %v.", err) | ||||||
|  | 	// } | ||||||
|  | 	// defer f.Close() | ||||||
|  | 	// | ||||||
|  |  | ||||||
|  | 	typstCLI := typst.CLI{} | ||||||
|  | 	if err := typstCLI.Compile(&markup, &output, nil); err != nil { | ||||||
|  | 		return output, err | ||||||
|  | 	} | ||||||
|  | 	return output, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func PDFHandler(w http.ResponseWriter, r *http.Request) { | ||||||
|  | 	helper.RequiresLogin(Session, w, r) | ||||||
|  | 	pp := paramParser.New(r.URL.Query()) | ||||||
|  | 	startDate := pp.ParseTimestamp("start_date", time.DateOnly, time.Now()) | ||||||
|  | 	if startDate.Day() > 1 { | ||||||
|  | 		startDate = startDate.AddDate(0, 0, -(startDate.Day() - 1)) | ||||||
|  | 	} | ||||||
|  | 	endDate := startDate.AddDate(0, 1, -1) | ||||||
|  |  | ||||||
|  | 	user, err := models.GetUserFromSession(Session, r.Context()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Println("Error getting user!") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	//TODO: only accepted weeks | ||||||
|  |  | ||||||
|  | 	weeks := models.GetDays(user, startDate, endDate, false) | ||||||
|  | 	var aggregatedOvertime, aggregatedWorkTime time.Duration | ||||||
|  | 	for _, day := range weeks { | ||||||
|  | 		aggregatedOvertime += day.TimeOvertimeReal(user) | ||||||
|  | 		aggregatedWorkTime += day.TimeWorkVirtual(user) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	typstDays, err := convertDaysToTypst(weeks, user) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Panicf("Failed to convert days!") | ||||||
|  | 	} | ||||||
|  | 	metadata := typstMetadata{ | ||||||
|  | 		EmployeeName:     fmt.Sprintf("%s %s", user.Vorname, user.Name), | ||||||
|  | 		TimeRange:        fmt.Sprintf("%s - %s", startDate.Format("02.01.2006"), endDate.Format("02.01.2006")), | ||||||
|  | 		Overtime:         helper.FormatDurationFill(aggregatedOvertime, true), | ||||||
|  | 		WorkTime:         helper.FormatDurationFill(aggregatedWorkTime, true), | ||||||
|  | 		OvertimeTotal:    "", | ||||||
|  | 		CurrentTimestamp: time.Now().Format("02.01.2006 - 15:04 Uhr"), | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	output, err := renderPDF(typstDays, metadata) | ||||||
|  | 	if err != nil { | ||||||
|  | 		slog.Warn("Could not create pdf report", slog.Any("Error", err)) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	w.Header().Set("Content-type", "application/pdf") | ||||||
|  | 	output.WriteTo(w) | ||||||
|  | 	w.WriteHeader(200) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type typstMetadata struct { | ||||||
|  | 	TimeRange        string `json:"time-range"` | ||||||
|  | 	EmployeeName     string `json:"employee-name"` | ||||||
|  | 	WorkTime         string `json:"worktime"` | ||||||
|  | 	Overtime         string `json:"overtime"` | ||||||
|  | 	OvertimeTotal    string `json:"overtime-total"` | ||||||
|  | 	CurrentTimestamp string `json:"current-timestamp"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type typstDayPart struct { | ||||||
|  | 	BookingFrom string `json:"booking-from"` | ||||||
|  | 	BookingTo   string `json:"booking-to"` | ||||||
|  | 	WorkType    string `json:"worktype"` | ||||||
|  | 	IsWorkDay   bool   `json:"is-workday"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type typstDay struct { | ||||||
|  | 	Date      string         `json:"date"` | ||||||
|  | 	DayParts  []typstDayPart `json:"day-parts"` | ||||||
|  | 	Worktime  string         `json:"worktime"` | ||||||
|  | 	Pausetime string         `json:"pausetime"` | ||||||
|  | 	Overtime  string         `json:"overtime"` | ||||||
|  | 	IsFriday  bool           `json:"is-weekend"` | ||||||
|  | } | ||||||
| @@ -4,33 +4,22 @@ import ( | |||||||
| 	"arbeitszeitmessung/helper" | 	"arbeitszeitmessung/helper" | ||||||
| 	"arbeitszeitmessung/models" | 	"arbeitszeitmessung/models" | ||||||
| 	"arbeitszeitmessung/templates" | 	"arbeitszeitmessung/templates" | ||||||
| 	"log" | 	"log/slog" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func PDFHandler(w http.ResponseWriter, r *http.Request) { | func PDFFormHandler(w http.ResponseWriter, r *http.Request) { | ||||||
| 	helper.RequiresLogin(Session, w, r) | 	helper.RequiresLogin(Session, w, r) | ||||||
| 	startDate, err := parseTimestamp(r, "start", time.Now().Format("2006-01-02")) |  | ||||||
| 	if err != nil { |  | ||||||
| 		log.Println("Error parsing 'start_date' time", err) |  | ||||||
| 		http.Error(w, "Timestamp 'start_date' cannot be parsed!", http.StatusBadRequest) |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	if startDate.Day() > 1 { |  | ||||||
| 		startDate = startDate.AddDate(0, 0, -(startDate.Day() - 1)) |  | ||||||
| 	} |  | ||||||
| 	endDate := startDate.AddDate(0, 1, -1) |  | ||||||
|  |  | ||||||
| 	user, err := models.GetUserFromSession(Session, r.Context()) | 	user, err := models.GetUserFromSession(Session, r.Context()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Error getting user!") | 		slog.Warn("Error getting user!", slog.Any("Error", err)) | ||||||
|  | 		// TODO add error handling | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	//TODO: only accepted weeks | 	teamMembers, err := user.GetTeamMembers() | ||||||
|  | 	if err != nil { | ||||||
| 	weeks := models.GetWorkDays(user, startDate, endDate) | 		slog.Warn("Error getting team members!", slog.Any("Error", err)) | ||||||
|  | 	} | ||||||
| 	// log.Printf("Using Dates: %s - %s\n", startDate.String(), endDate.String()) | 	templates.PDFForm(teamMembers).Render(r.Context(), w) | ||||||
| 	templates.PDFReportEmploye(user, weeks, startDate, endDate).Render(r.Context(), w) |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -65,7 +65,7 @@ func showWeeks(w http.ResponseWriter, r *http.Request) { | |||||||
| 	submissionDate := r.URL.Query().Get("submission_date") | 	submissionDate := r.URL.Query().Get("submission_date") | ||||||
| 	lastSub := user.GetLastWorkWeekSubmission() | 	lastSub := user.GetLastWorkWeekSubmission() | ||||||
| 	if submissionDate != "" { | 	if submissionDate != "" { | ||||||
| 		submissionDate, err := time.Parse("2006-01-02", submissionDate) | 		submissionDate, err := time.Parse(time.DateOnly, submissionDate) | ||||||
| 		if err == nil { | 		if err == nil { | ||||||
| 			lastSub = helper.GetMonday(submissionDate) | 			lastSub = helper.GetMonday(submissionDate) | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -28,10 +28,9 @@ func teamPresence(w http.ResponseWriter, r *http.Request) { | |||||||
| 		log.Println("Error getting user!", err) | 		log.Println("Error getting user!", err) | ||||||
| 	} | 	} | ||||||
| 	team, err := user.GetTeamMembers() | 	team, err := user.GetTeamMembers() | ||||||
| 	teamPresence := make(map[bool][]models.User) | 	teamPresence := make(map[models.User]bool) | ||||||
| 	for _, user := range team { | 	for _, user := range team { | ||||||
| 		present := user.CheckAnwesenheit() | 		teamPresence[user] = user.CheckAnwesenheit() | ||||||
| 		teamPresence[present] = append(teamPresence[present], user) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import ( | |||||||
| 	"arbeitszeitmessung/models" | 	"arbeitszeitmessung/models" | ||||||
| 	"arbeitszeitmessung/templates" | 	"arbeitszeitmessung/templates" | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"database/sql" | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"log" | 	"log" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| @@ -30,12 +31,28 @@ func TimeHandler(w http.ResponseWriter, r *http.Request) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func AbsencHandler(w http.ResponseWriter, r *http.Request) { | ||||||
|  | 	helper.RequiresLogin(Session, w, r) | ||||||
|  | 	helper.SetCors(w) | ||||||
|  | 	switch r.Method { | ||||||
|  | 	case http.MethodPost: | ||||||
|  | 		err := updateAbsence(r) | ||||||
|  | 		if err != nil { | ||||||
|  | 			http.Error(w, "Internal error", http.StatusInternalServerError) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		http.Redirect(w, r, "/time", 301) | ||||||
|  | 	default: | ||||||
|  | 		http.Error(w, "Method not allowed!", http.StatusMethodNotAllowed) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| func parseTimestamp(r *http.Request, getKey string, fallback string) (time.Time, error) { | func parseTimestamp(r *http.Request, getKey string, fallback string) (time.Time, error) { | ||||||
| 	getTimestamp := r.URL.Query().Get(getKey) | 	getTimestamp := r.URL.Query().Get(getKey) | ||||||
| 	if getTimestamp == "" { | 	if getTimestamp == "" { | ||||||
| 		getTimestamp = fallback | 		getTimestamp = fallback | ||||||
| 	} | 	} | ||||||
| 	Timestamp, err := time.Parse("2006-01-02", getTimestamp) | 	Timestamp, err := time.Parse(time.DateOnly, getTimestamp) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return time.Now(), err | 		return time.Now(), err | ||||||
| 	} | 	} | ||||||
| @@ -52,13 +69,13 @@ func getBookings(w http.ResponseWriter, r *http.Request) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// TODO add config for timeoffset | 	// TODO add config for timeoffset | ||||||
| 	tsFrom, err := parseTimestamp(r, "time_from", time.Now().AddDate(0, -1, 0).Format("2006-01-02")) | 	tsFrom, err := parseTimestamp(r, "time_from", time.Now().AddDate(0, -1, 0).Format(time.DateOnly)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Error parsing 'from' time", err) | 		log.Println("Error parsing 'from' time", err) | ||||||
| 		http.Error(w, "Timestamp 'from' cannot be parsed!", http.StatusBadRequest) | 		http.Error(w, "Timestamp 'from' cannot be parsed!", http.StatusBadRequest) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	tsTo, err := parseTimestamp(r, "time_to", time.Now().Format("2006-01-02")) | 	tsTo, err := parseTimestamp(r, "time_to", time.Now().Format(time.DateOnly)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Error parsing 'to' time", err) | 		log.Println("Error parsing 'to' time", err) | ||||||
| 		http.Error(w, "Timestamp 'to' cannot be parsed!", http.StatusBadRequest) | 		http.Error(w, "Timestamp 'to' cannot be parsed!", http.StatusBadRequest) | ||||||
| @@ -66,18 +83,18 @@ func getBookings(w http.ResponseWriter, r *http.Request) { | |||||||
| 	} | 	} | ||||||
| 	tsTo = tsTo.AddDate(0, 0, 1) // so that today is inside | 	tsTo = tsTo.AddDate(0, 0, 1) // so that today is inside | ||||||
|  |  | ||||||
| 	workDays := models.GetWorkDays(user, tsFrom, tsTo) | 	days := models.GetDays(user, tsFrom, tsTo, true) | ||||||
| 	sort.Slice(workDays, func(i, j int) bool { | 	sort.Slice(days, func(i, j int) bool { | ||||||
| 		return workDays[i].Day.After(workDays[j].Day) | 		return days[i].Date().After(days[j].Date()) | ||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| 	lastSub := user.GetLastWorkWeekSubmission() | 	lastSub := user.GetLastWorkWeekSubmission() | ||||||
| 	var aggregatedOvertime time.Duration | 	var aggregatedOvertime time.Duration | ||||||
| 	for _, day := range workDays { | 	for _, day := range days { | ||||||
| 		if day.Day.Before(lastSub) { | 		if day.Date().Before(lastSub) { | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
| 		aggregatedOvertime += day.CalcOvertime(user) | 		aggregatedOvertime += day.TimeOvertimeReal(user) | ||||||
| 	} | 	} | ||||||
| 	if reportedOvertime, err := user.GetReportedOvertime(); err == nil { | 	if reportedOvertime, err := user.GetReportedOvertime(); err == nil { | ||||||
| 		user.Overtime = (reportedOvertime + aggregatedOvertime).Round(time.Minute) | 		user.Overtime = (reportedOvertime + aggregatedOvertime).Round(time.Minute) | ||||||
| @@ -88,12 +105,13 @@ func getBookings(w http.ResponseWriter, r *http.Request) { | |||||||
| 	if r.Header.Get("Accept") == "application/json" { | 	if r.Header.Get("Accept") == "application/json" { | ||||||
| 		w.Header().Set("Content-Type", "application/json") | 		w.Header().Set("Content-Type", "application/json") | ||||||
| 		w.WriteHeader(http.StatusOK) | 		w.WriteHeader(http.StatusOK) | ||||||
| 		json.NewEncoder(w).Encode(workDays) | 		json.NewEncoder(w).Encode(days) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ctx := context.WithValue(r.Context(), "user", user) | 	ctx := context.WithValue(r.Context(), "user", user) | ||||||
| 	templates.TimePage(workDays, lastSub).Render(ctx, w) | 	ctx = context.WithValue(ctx, "days", days) | ||||||
|  | 	templates.TimePage([]models.WorkDay{}, lastSub).Render(ctx, w) | ||||||
| } | } | ||||||
|  |  | ||||||
| func updateBooking(w http.ResponseWriter, r *http.Request) { | func updateBooking(w http.ResponseWriter, r *http.Request) { | ||||||
| @@ -109,6 +127,7 @@ func updateBooking(w http.ResponseWriter, r *http.Request) { | |||||||
| 		log.Println("No user found!", err) | 		log.Println("No user found!", err) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	switch r.FormValue("action") { | 	switch r.FormValue("action") { | ||||||
| 	case "add": | 	case "add": | ||||||
| 		timestamp, err := time.ParseInLocation("2006-01-02|15:04", r.FormValue("date")+"|"+r.FormValue("timestamp"), loc) | 		timestamp, err := time.ParseInLocation("2006-01-02|15:04", r.FormValue("date")+"|"+r.FormValue("timestamp"), loc) | ||||||
| @@ -131,14 +150,14 @@ func updateBooking(w http.ResponseWriter, r *http.Request) { | |||||||
| 			log.Printf("Error inserting booking %v -> %v\n", newBooking, err) | 			log.Printf("Error inserting booking %v -> %v\n", newBooking, err) | ||||||
| 		} | 		} | ||||||
| 	case "change": | 	case "change": | ||||||
| 		absenceType, err := strconv.Atoi(r.FormValue("absence")) | 		// absenceType, err := strconv.Atoi(r.FormValue("absence")) | ||||||
| 		if err != nil { | 		// if err != nil { | ||||||
| 			log.Println("Error parsing absence type.", err) | 		// 	log.Println("Error parsing absence type.", err) | ||||||
| 			absenceType = 0 | 		// 	absenceType = 0 | ||||||
| 		} | 		// } | ||||||
| 		if absenceType != 0 { | 		// if absenceType != 0 { | ||||||
| 			createAbsence(absenceType, user, loc, r) | 		// 	createAbsence(absenceType, user, loc, r) | ||||||
| 		} | 		// } | ||||||
| 		for index, possibleBooking := range r.PostForm { | 		for index, possibleBooking := range r.PostForm { | ||||||
| 			if len(index) > 7 && index[:7] == "booking" { | 			if len(index) > 7 && index[:7] == "booking" { | ||||||
| 				booking_id, err := strconv.Atoi(index[8:]) | 				booking_id, err := strconv.Atoi(index[8:]) | ||||||
| @@ -156,29 +175,89 @@ func updateBooking(w http.ResponseWriter, r *http.Request) { | |||||||
| 					log.Println("Error parsing time!", err) | 					log.Println("Error parsing time!", err) | ||||||
| 					continue | 					continue | ||||||
| 				} | 				} | ||||||
| 				// log.Println("Parsing time", parsedTime) |  | ||||||
| 				booking.UpdateTime(parsedTime) | 				booking.UpdateTime(parsedTime) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 	default: | ||||||
|  | 		log.Println("No action from /time found") | ||||||
| 	} | 	} | ||||||
| 	getBookings(w, r) | 	getBookings(w, r) | ||||||
| } | } | ||||||
|  |  | ||||||
| func createAbsence(absenceType int, user models.User, loc *time.Location, r *http.Request) { | func updateAbsence(r *http.Request) error { | ||||||
| 	absenceDate, err := time.ParseInLocation("2006-01-02", r.FormValue("date"), loc) | 	r.ParseForm() | ||||||
|  | 	var loc *time.Location | ||||||
|  | 	loc, err := time.LoadLocation(helper.GetEnv("TZ", "Europe/Berlin")) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Cannot get date from input! Skipping absence creation", err) | 		log.Println("Error loading location", err) | ||||||
| 		return | 		loc = time.Local | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	absence, err := models.NewAbsence(user.CardUID, absenceType, absenceDate) | 	dateFrom, err := time.ParseInLocation(time.DateOnly, r.FormValue("date_from"), loc) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Error creating absence!", err) | 		log.Println("Error parsing date_from input for absence", err) | ||||||
| 		return | 		return err | ||||||
| 	} | 	} | ||||||
| 	err = absence.Insert() |  | ||||||
|  | 	dateTo, err := time.ParseInLocation(time.DateOnly, r.FormValue("date_to"), loc) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Error inserting absence!", err) | 		log.Println("Error parsing date_to input for absence", err) | ||||||
| 		return | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	absenceTypeId, err := strconv.Atoi(r.FormValue("aw_type")) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Println("Error parsing aw_type", err) | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	absenceId, err := strconv.Atoi(r.FormValue("aw_id")) | ||||||
|  | 	if err != nil && r.FormValue("aw_id") == "" { | ||||||
|  | 		absenceId = 0 | ||||||
|  | 	} else if err != nil { | ||||||
|  | 		log.Println("Error parsing aw_id", err) | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	absenceType, err := models.GetAbsenceTypeById(int8(absenceTypeId)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Println("No matching absence type found!") | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	newAbsence := models.Absence{DateFrom: dateFrom, DateTo: dateTo, AbwesenheitTyp: absenceType} | ||||||
|  |  | ||||||
|  | 	absence, err := models.GetAbsenceById(absenceId) | ||||||
|  | 	if err == sql.ErrNoRows { | ||||||
|  | 		err = nil | ||||||
|  | 		log.Println("Absence not found creating new!") | ||||||
|  |  | ||||||
|  | 		user, err := models.GetUserFromSession(Session, r.Context()) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Println("No user found!", err) | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		newAbsence.CardUID = user.CardUID | ||||||
|  | 		newAbsence.Insert() | ||||||
|  | 	} | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Println("Cannot get Absence for id: ", absenceId, err) | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	if r.FormValue("action") == "delete" { | ||||||
|  | 		log.Println("Deleting Absence!", "Not implemented") | ||||||
|  | 		// TODO | ||||||
|  | 		//absence.Delete() | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if absence.Update(newAbsence) { | ||||||
|  | 		err = absence.Save() | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Println("Error saving updated absence!", err) | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ package endpoints | |||||||
| import ( | import ( | ||||||
| 	"arbeitszeitmessung/models" | 	"arbeitszeitmessung/models" | ||||||
| 	"arbeitszeitmessung/templates" | 	"arbeitszeitmessung/templates" | ||||||
|  | 	"context" | ||||||
| 	"log" | 	"log" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
| @@ -38,5 +39,9 @@ func changePassword(w http.ResponseWriter, r *http.Request) { | |||||||
| } | } | ||||||
|  |  | ||||||
| func showUserPage(w http.ResponseWriter, r *http.Request, status int) { | func showUserPage(w http.ResponseWriter, r *http.Request, status int) { | ||||||
| 	templates.UserPage(status).Render(r.Context(), w) | 	var ctx context.Context | ||||||
|  | 	if user, err := models.GetUserFromSession(Session, r.Context()); err == nil { | ||||||
|  | 		ctx = context.WithValue(r.Context(), "user", user) | ||||||
|  | 	} | ||||||
|  | 	templates.SettingsPage(status).Render(ctx, w) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -29,6 +29,7 @@ func LoginHandler(w http.ResponseWriter, r *http.Request) { | |||||||
|  |  | ||||||
| func UserSettingsHandler(w http.ResponseWriter, r *http.Request) { | func UserSettingsHandler(w http.ResponseWriter, r *http.Request) { | ||||||
| 	helper.RequiresLogin(Session, w, r) | 	helper.RequiresLogin(Session, w, r) | ||||||
|  |  | ||||||
| 	switch r.Method { | 	switch r.Method { | ||||||
| 	case http.MethodGet: | 	case http.MethodGet: | ||||||
| 		showUserPage(w, r, 0) | 		showUserPage(w, r, 0) | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ require github.com/a-h/templ v0.3.943 | |||||||
| require github.com/alexedwards/scs/v2 v2.8.0 | require github.com/alexedwards/scs/v2 v2.8.0 | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
|  | 	github.com/Dadido3/go-typst v0.3.0 | ||||||
| 	github.com/golang-migrate/migrate/v4 v4.18.3 | 	github.com/golang-migrate/migrate/v4 v4.18.3 | ||||||
| 	github.com/joho/godotenv v1.5.1 | 	github.com/joho/godotenv v1.5.1 | ||||||
| ) | ) | ||||||
| @@ -16,6 +17,14 @@ require ( | |||||||
| require ( | require ( | ||||||
| 	github.com/hashicorp/errwrap v1.1.0 // indirect | 	github.com/hashicorp/errwrap v1.1.0 // indirect | ||||||
| 	github.com/hashicorp/go-multierror v1.1.1 // indirect | 	github.com/hashicorp/go-multierror v1.1.1 // indirect | ||||||
|  | 	github.com/smasher164/xid v0.1.2 // indirect | ||||||
| 	go.uber.org/atomic v1.7.0 // indirect | 	go.uber.org/atomic v1.7.0 // indirect | ||||||
| 	golang.org/x/sys v0.36.0 // indirect | 	golang.org/x/mod v0.29.0 // indirect | ||||||
|  | 	golang.org/x/sync v0.17.0 // indirect | ||||||
|  | 	golang.org/x/sys v0.37.0 // indirect | ||||||
|  | 	golang.org/x/telemetry v0.0.0-20251008203120-078029d740a8 // indirect | ||||||
|  | 	golang.org/x/text v0.23.0 // indirect | ||||||
|  | 	golang.org/x/tools v0.38.0 // indirect | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | tool golang.org/x/tools/cmd/deadcode | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 h1:L/gRVlceqvL25UVaW/CKtUDjefjrs0SPonmDGUVOYP0= | github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 h1:L/gRVlceqvL25UVaW/CKtUDjefjrs0SPonmDGUVOYP0= | ||||||
| github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E= | github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E= | ||||||
|  | github.com/Dadido3/go-typst v0.3.0 h1:Itix2FtQgBiOuHUNqgGUAK11Oo2WMlZGGGpCiQNK1IA= | ||||||
|  | github.com/Dadido3/go-typst v0.3.0/go.mod h1:QYis9sT70u65kn1SkFfyPRmHsPxgoxWbAixwfPReOZA= | ||||||
| github.com/Microsoft/go-winio v0.6.2 h1:F2VQgta7ecxGYO8k3ZZz3RS8fVIXVxONVUPlNERoyfY= | github.com/Microsoft/go-winio v0.6.2 h1:F2VQgta7ecxGYO8k3ZZz3RS8fVIXVxONVUPlNERoyfY= | ||||||
| github.com/Microsoft/go-winio v0.6.2/go.mod h1:yd8OoFMLzJbo9gZq8j5qaps8bJ9aShtEA8Ipt1oGCvU= | github.com/Microsoft/go-winio v0.6.2/go.mod h1:yd8OoFMLzJbo9gZq8j5qaps8bJ9aShtEA8Ipt1oGCvU= | ||||||
| github.com/a-h/templ v0.3.943 h1:o+mT/4yqhZ33F3ootBiHwaY4HM5EVaOJfIshvd5UNTY= | github.com/a-h/templ v0.3.943 h1:o+mT/4yqhZ33F3ootBiHwaY4HM5EVaOJfIshvd5UNTY= | ||||||
| @@ -54,6 +56,8 @@ github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= | |||||||
| github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||||
|  | github.com/smasher164/xid v0.1.2 h1:erplXSdBRIIw+MrwjJ/m8sLN2XY16UGzpTA0E2Ru6HA= | ||||||
|  | github.com/smasher164/xid v0.1.2/go.mod h1:tgivm8CQl19fH1c5y+8F4mA+qY6n2i6qDRBlY/6nm+I= | ||||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||||
| github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA= | github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA= | ||||||
| @@ -68,7 +72,19 @@ go.opentelemetry.io/otel/trace v1.29.0 h1:J/8ZNK4XgR7a21DZUAsbF8pZ5Jcw1VhACmnYt3 | |||||||
| go.opentelemetry.io/otel/trace v1.29.0/go.mod h1:eHl3w0sp3paPkYstJOmAimxhiFXPg+MMTlEh3nsQgWQ= | go.opentelemetry.io/otel/trace v1.29.0/go.mod h1:eHl3w0sp3paPkYstJOmAimxhiFXPg+MMTlEh3nsQgWQ= | ||||||
| go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw= | go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw= | ||||||
| go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= | go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= | ||||||
| golang.org/x/sys v0.36.0 h1:KVRy2GtZBrk1cBYA7MKu5bEZFxQk4NIDV6RLVcC8o0k= | golang.org/x/mod v0.29.0 h1:HV8lRxZC4l2cr3Zq1LvtOsi/ThTgWnUk/y64QSs8GwA= | ||||||
| golang.org/x/sys v0.36.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks= | golang.org/x/mod v0.29.0/go.mod h1:NyhrlYXJ2H4eJiRy/WDBO6HMqZQ6q9nk4JzS3NuCK+w= | ||||||
|  | golang.org/x/sync v0.17.0 h1:l60nONMj9l5drqw6jlhIELNv9I0A4OFgRsG9k2oT9Ug= | ||||||
|  | golang.org/x/sync v0.17.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI= | ||||||
|  | golang.org/x/sys v0.37.0 h1:fdNQudmxPjkdUTPnLn5mdQv7Zwvbvpaxqs831goi9kQ= | ||||||
|  | golang.org/x/sys v0.37.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks= | ||||||
|  | golang.org/x/telemetry v0.0.0-20251008203120-078029d740a8 h1:LvzTn0GQhWuvKH/kVRS3R3bVAsdQWI7hvfLHGgh9+lU= | ||||||
|  | golang.org/x/telemetry v0.0.0-20251008203120-078029d740a8/go.mod h1:Pi4ztBfryZoJEkyFTI5/Ocsu2jXyDr6iSdgJiYE/uwE= | ||||||
|  | golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
|  | golang.org/x/text v0.23.0 h1:D71I7dUrlY+VX0gQShAThNGHFxZ13dGLBHQLVl1mJlY= | ||||||
|  | golang.org/x/text v0.23.0/go.mod h1:/BLNzu4aZCJ1+kcD0DNRotWKage4q2rGVAg4o22unh4= | ||||||
|  | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||||
|  | golang.org/x/tools v0.38.0 h1:Hx2Xv8hISq8Lm16jvBZ2VQf+RLmbd7wVUsALibYI/IQ= | ||||||
|  | golang.org/x/tools v0.38.0/go.mod h1:yEsQ/d/YK8cjh0L6rZlY8tgtlKiBNTL14pGDJPJpYQs= | ||||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||||
| gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ type FileLog struct { | |||||||
| var Logs map[string]FileLog = make(map[string]FileLog) | var Logs map[string]FileLog = make(map[string]FileLog) | ||||||
|  |  | ||||||
| func NewAudit() (i *log.Logger, close func() error) { | func NewAudit() (i *log.Logger, close func() error) { | ||||||
| 	LOG_FILE := "logs/" + time.Now().Format("2006-01-02") + ".log" | 	LOG_FILE := "logs/" + time.Now().Format(time.DateOnly) + ".log" | ||||||
| 	logFile, err := os.OpenFile(LOG_FILE, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644) | 	logFile, err := os.OpenFile(LOG_FILE, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Panic(err) | 		log.Panic(err) | ||||||
|   | |||||||
							
								
								
									
										31
									
								
								Backend/helper/paramParser/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								Backend/helper/paramParser/main.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | |||||||
|  | package paramParser | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"log/slog" | ||||||
|  | 	"net/url" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type ParamsParser struct { | ||||||
|  | 	urlParams url.Values | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func New(_urlParams url.Values) ParamsParser { | ||||||
|  | 	return ParamsParser{ | ||||||
|  | 		urlParams: _urlParams, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (p *ParamsParser) ParseTimestamp(key string, format string, fallback time.Time) time.Time { | ||||||
|  | 	paramTimestamp := p.urlParams.Get(key) | ||||||
|  | 	if paramTimestamp == "" { | ||||||
|  | 		return fallback | ||||||
|  | 	} | ||||||
|  | 	if timestamp, err := time.Parse(format, paramTimestamp); err == nil { | ||||||
|  | 		return timestamp | ||||||
|  | 	} else { | ||||||
|  | 		slog.Warn("Error parsing HTTP Params to timestamp", slog.Any("key", key), slog.Any("error", err)) | ||||||
|  | 		return fallback | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -6,8 +6,8 @@ import ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestGetMonday(t *testing.T) { | func TestGetMonday(t *testing.T) { | ||||||
| 	isMonday, err := time.Parse("2006-01-02", "2025-07-14") | 	isMonday, err := time.Parse(time.DateOnly, "2025-07-14") | ||||||
| 	notMonday, err := time.Parse("2006-01-02", "2025-07-16") | 	notMonday, err := time.Parse(time.DateOnly, "2025-07-16") | ||||||
| 	if err != nil || isMonday.Equal(notMonday) { | 	if err != nil || isMonday.Equal(notMonday) { | ||||||
| 		t.Errorf("U stupid? %e", err) | 		t.Errorf("U stupid? %e", err) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -3,7 +3,23 @@ package helper | |||||||
| import "time" | import "time" | ||||||
|  |  | ||||||
| type TimeFormValue struct { | type TimeFormValue struct { | ||||||
| 	TsFrom time.Time | 	TsFrom  time.Time | ||||||
| 	TsTo time.Time | 	TsTo    time.Time | ||||||
| 	CardUID string | 	CardUID string | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func BoolToInt(b bool) int { | ||||||
|  | 	var i int = 0 | ||||||
|  | 	if b { | ||||||
|  | 		i = 1 | ||||||
|  | 	} | ||||||
|  | 	return i | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func BoolToInt8(b bool) int8 { | ||||||
|  | 	var i int8 = 0 | ||||||
|  | 	if b { | ||||||
|  | 		i = 1 | ||||||
|  | 	} | ||||||
|  | 	return i | ||||||
|  | } | ||||||
|   | |||||||
| @@ -5,8 +5,7 @@ import ( | |||||||
| 	"arbeitszeitmessung/helper" | 	"arbeitszeitmessung/helper" | ||||||
| 	"arbeitszeitmessung/models" | 	"arbeitszeitmessung/models" | ||||||
| 	"context" | 	"context" | ||||||
| 	"fmt" | 	"log/slog" | ||||||
| 	"log" |  | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"os" | 	"os" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -17,23 +16,25 @@ import ( | |||||||
|  |  | ||||||
| func main() { | func main() { | ||||||
| 	var err error | 	var err error | ||||||
|  | 	var logLevel slog.LevelVar | ||||||
|  | 	logLevel.Set(slog.LevelWarn) | ||||||
|  |  | ||||||
|  | 	logger := slog.New(slog.NewJSONHandler(os.Stdout, &slog.HandlerOptions{Level: &logLevel})) | ||||||
|  | 	slog.SetDefault(logger) | ||||||
|  |  | ||||||
| 	err = godotenv.Load(".env") | 	err = godotenv.Load(".env") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("No .env file found in directory!") | 		slog.Info("No .env file found in directory!") | ||||||
| 	} | 	} | ||||||
| 	if helper.GetEnv("GO_ENV", "production") == "debug" { | 	if helper.GetEnv("GO_ENV", "production") == "debug" { | ||||||
| 		log.Println("Debug mode enabled") | 		logLevel.Set(slog.LevelDebug) | ||||||
| 		log.Println("Environment Variables") |  | ||||||
| 		envs := os.Environ() | 		envs := os.Environ() | ||||||
| 		for _, e := range envs { | 		slog.Debug("Debug mode enabled", "Environment Variables", envs) | ||||||
| 			fmt.Println(e) |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	models.DB, err = OpenDatabase() | 	models.DB, err = OpenDatabase() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Fatal(err) | 		slog.Error("Error while opening the database", "Error", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	fs := http.FileServer(http.Dir("./static")) | 	fs := http.FileServer(http.Dir("./static")) | ||||||
| @@ -43,28 +44,31 @@ func main() { | |||||||
|  |  | ||||||
| 	// handles the different http routes | 	// handles the different http routes | ||||||
| 	server.HandleFunc("/time/new", endpoints.TimeCreateHandler) | 	server.HandleFunc("/time/new", endpoints.TimeCreateHandler) | ||||||
|  | 	server.Handle("/absence", ParamsMiddleware(endpoints.AbsencHandler)) | ||||||
| 	server.Handle("/time", ParamsMiddleware(endpoints.TimeHandler)) | 	server.Handle("/time", ParamsMiddleware(endpoints.TimeHandler)) | ||||||
| 	server.HandleFunc("/logout", endpoints.LogoutHandler) | 	server.HandleFunc("/logout", endpoints.LogoutHandler) | ||||||
| 	server.HandleFunc("/user/{action}", endpoints.UserHandler) | 	server.HandleFunc("/user/{action}", endpoints.UserHandler) | ||||||
| 	// server.HandleFunc("/user/login", endpoints.LoginHandler) | 	// server.HandleFunc("/user/login", endpoints.LoginHandler) | ||||||
| 	// server.HandleFunc("/user/settings", endpoints.UserSettingsHandler) | 	// server.HandleFunc("/user/settings", endpoints.UserSettingsHandler) | ||||||
| 	server.HandleFunc("/team", endpoints.TeamHandler) | 	server.HandleFunc("/team", endpoints.TeamHandler) | ||||||
| 	server.HandleFunc("/team/presence", endpoints.TeamPresenceHandler) | 	server.HandleFunc("/presence", endpoints.TeamPresenceHandler) | ||||||
| 	server.HandleFunc("/pdf", endpoints.PDFHandler) | 	server.Handle("/pdf", ParamsMiddleware(endpoints.PDFFormHandler)) | ||||||
|  | 	server.HandleFunc("/pdf/generate", endpoints.PDFHandler) | ||||||
| 	server.Handle("/", http.RedirectHandler("/time", http.StatusPermanentRedirect)) | 	server.Handle("/", http.RedirectHandler("/time", http.StatusPermanentRedirect)) | ||||||
| 	server.Handle("/static/", http.StripPrefix("/static/", fs)) | 	server.Handle("/static/", http.StripPrefix("/static/", fs)) | ||||||
|  |  | ||||||
| 	serverSessionMiddleware := endpoints.Session.LoadAndSave(server) | 	serverSessionMiddleware := endpoints.Session.LoadAndSave(server) | ||||||
|  |  | ||||||
| 	// starting the http server | 	// starting the http server | ||||||
| 	fmt.Printf("Server is running at http://localhost:%s\n", helper.GetEnv("EXPOSED_PORT", "8080")) | 	slog.Info("Server is running at http://localhost:8080") | ||||||
| 	log.Fatal(http.ListenAndServe(":8080", serverSessionMiddleware)) | 	slog.Error("Error starting Server", "Error", http.ListenAndServe(":8080", serverSessionMiddleware)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func ParamsMiddleware(next http.HandlerFunc) http.Handler { | func ParamsMiddleware(next http.HandlerFunc) http.Handler { | ||||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||||
| 		queryParams := r.URL.Query() | 		queryParams := r.URL.Query() | ||||||
| 		ctx := context.WithValue(r.Context(), "urlParams", queryParams) | 		ctx := context.WithValue(r.Context(), "urlParams", queryParams) | ||||||
|  | 		slog.Debug("ParamsMiddleware added urlParams", slog.Any("urlParams", queryParams)) | ||||||
| 		next.ServeHTTP(w, r.WithContext(ctx)) | 		next.ServeHTTP(w, r.WithContext(ctx)) | ||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,22 +1,25 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"encoding/json" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"log" | 	"log" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type AbsenceType struct { | type AbsenceType struct { | ||||||
| 	Id       int8 | 	Id       int8   `json:"abwesenheit_id"` | ||||||
| 	Name     string | 	Name     string `json:"abwesenheit_name"` | ||||||
| 	WorkTime int8 | 	WorkTime int8   `json:"arbeitszeit_equivalent"` | ||||||
| } | } | ||||||
|  |  | ||||||
| type Absence struct { | type Absence struct { | ||||||
|  | 	Day            time.Time | ||||||
| 	CounterId      int | 	CounterId      int | ||||||
| 	CardUID        string | 	CardUID        string | ||||||
| 	AbwesenheitTyp AbsenceType | 	AbwesenheitTyp AbsenceType | ||||||
| 	Datum          time.Time | 	DateFrom       time.Time | ||||||
|  | 	DateTo         time.Time | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewAbsence(card_uid string, abwesenheit_typ int, datum time.Time) (Absence, error) { | func NewAbsence(card_uid string, abwesenheit_typ int, datum time.Time) (Absence, error) { | ||||||
| @@ -24,7 +27,7 @@ func NewAbsence(card_uid string, abwesenheit_typ int, datum time.Time) (Absence, | |||||||
| 		return Absence{ | 		return Absence{ | ||||||
| 			CardUID:        card_uid, | 			CardUID:        card_uid, | ||||||
| 			AbwesenheitTyp: AbsenceType{0, "Custom absence", 100}, | 			AbwesenheitTyp: AbsenceType{0, "Custom absence", 100}, | ||||||
| 			Datum:          datum, | 			DateFrom:       datum, | ||||||
| 		}, nil | 		}, nil | ||||||
| 	} | 	} | ||||||
| 	_absenceType, ok := GetAbsenceTypesCached()[int8(abwesenheit_typ)] | 	_absenceType, ok := GetAbsenceTypesCached()[int8(abwesenheit_typ)] | ||||||
| @@ -34,18 +37,75 @@ func NewAbsence(card_uid string, abwesenheit_typ int, datum time.Time) (Absence, | |||||||
| 	return Absence{ | 	return Absence{ | ||||||
| 		CardUID:        card_uid, | 		CardUID:        card_uid, | ||||||
| 		AbwesenheitTyp: _absenceType, | 		AbwesenheitTyp: _absenceType, | ||||||
| 		Datum:          datum, | 		DateFrom:       datum, | ||||||
| 	}, nil | 	}, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (a *Absence) Date() time.Time { | ||||||
|  | 	return a.Day.Truncate(24 * time.Hour) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) IsMultiDay() bool { | ||||||
|  | 	return !a.DateFrom.Equal(a.DateTo) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) TimeWorkVirtual(u User) time.Duration { | ||||||
|  | 	return a.TimeWorkReal(u) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) TimeWorkReal(u User) time.Duration { | ||||||
|  | 	if a.AbwesenheitTyp.WorkTime > 1 { | ||||||
|  | 		return time.Duration(u.ArbeitszeitPerTag * float32(time.Hour)).Round(time.Minute) | ||||||
|  | 	} | ||||||
|  | 	return 0 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) TimePauseReal(u User) (work, pause time.Duration) { | ||||||
|  | 	return 0, 0 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) TimeOvertimeReal(u User) time.Duration { | ||||||
|  | 	if a.AbwesenheitTyp.WorkTime > 1 { | ||||||
|  | 		return 0 | ||||||
|  | 	} | ||||||
|  | 	return -u.ArbeitszeitProTag() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) ToString() string { | ||||||
|  | 	return "Abwesenheit" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) IsWorkDay() bool { | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) IsKurzArbeit() bool { | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) GetDayProgress(u User) int8 { | ||||||
|  | 	return 100 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) RequiresAction() bool { | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (a *Absence) GetAllWorkTimesVirtual(u User) (work, pause, overtime time.Duration) { | ||||||
|  | 	if a.AbwesenheitTyp.WorkTime > 1 { | ||||||
|  | 		return u.ArbeitszeitProTag(), 0, 0 | ||||||
|  | 	} | ||||||
|  | 	return 0, 0, 0 | ||||||
|  | } | ||||||
|  |  | ||||||
| func (a *Absence) Insert() error { | func (a *Absence) Insert() error { | ||||||
| 	qStr, err := DB.Prepare(`INSERT INTO abwesenheit (card_uid, abwesenheit_typ, datum) VALUES ($1, $2, $3) RETURNING counter_id;`) | 	qStr, err := DB.Prepare(`INSERT INTO abwesenheit (card_uid, abwesenheit_typ, datum_from, datum_to) VALUES ($1, $2, $3, $4) RETURNING counter_id;`) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Error preparing sql Statement", err) | 		log.Println("Error preparing sql Statement", err) | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	defer qStr.Close() | 	defer qStr.Close() | ||||||
| 	err = qStr.QueryRow(a.CardUID, a.AbwesenheitTyp.Id, a.Datum).Scan(&a.CounterId) | 	err = qStr.QueryRow(a.CardUID, a.AbwesenheitTyp.Id, a.DateFrom, a.DateTo).Scan(&a.CounterId) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Error executing insert statement", err) | 		log.Println("Error executing insert statement", err) | ||||||
| 		return err | 		return err | ||||||
| @@ -53,14 +113,31 @@ func (a *Absence) Insert() error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (a *Absence) Save() error { | ||||||
|  | 	qStr, err := DB.Prepare(` | ||||||
|  | 		UPDATE abwesenheit SET card_uid = $2, abwesenheit_typ = $3, datum_from = $4, datum_to = $5 WHERE counter_id = $1; | ||||||
|  | 	`) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Println("Error preparing sql Statement (Absence Save)", err) | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	defer qStr.Close() | ||||||
|  | 	_, err = qStr.Query(a.CounterId, a.CardUID, a.AbwesenheitTyp.Id, a.DateFrom, a.DateTo) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Println("Error executing update statement", err) | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
| func GetAbsenceById(counterId int) (Absence, error) { | func GetAbsenceById(counterId int) (Absence, error) { | ||||||
| 	var absence Absence = Absence{CounterId: counterId} | 	var absence Absence = Absence{CounterId: counterId} | ||||||
| 	qStr, err := DB.Prepare("SELECT card_uid, abwesenheit_typ, datum FROM abwesenheit WHERE counter_id = $1;") | 	qStr, err := DB.Prepare("SELECT card_uid, abwesenheit_typ, datum_from, datum_to FROM abwesenheit WHERE counter_id = $1;") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return absence, err | 		return absence, err | ||||||
| 	} | 	} | ||||||
| 	defer qStr.Close() | 	defer qStr.Close() | ||||||
| 	err = qStr.QueryRow(counterId).Scan(&absence.CardUID, &absence.AbwesenheitTyp.Id, &absence.Datum) | 	err = qStr.QueryRow(counterId).Scan(&absence.CardUID, &absence.AbwesenheitTyp.Id, &absence.DateFrom, &absence.DateTo) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return absence, err | 		return absence, err | ||||||
| 	} | 	} | ||||||
| @@ -69,7 +146,32 @@ func GetAbsenceById(counterId int) (Absence, error) { | |||||||
|  |  | ||||||
| func GetAbsencesByCardUID(card_uid string, tsFrom time.Time, tsTo time.Time) ([]Absence, error) { | func GetAbsencesByCardUID(card_uid string, tsFrom time.Time, tsTo time.Time) ([]Absence, error) { | ||||||
| 	var absences []Absence | 	var absences []Absence | ||||||
| 	qStr, err := DB.Prepare("SELECT counter_id, abwesenheit_typ, datum FROM abwesenheit WHERE card_uid = $1 AND datum BETWEEN $2::DATE AND $3::DATE ORDER BY datum;") | 	// qStr, err := DB.Prepare(`SELECT counter_id, abwesenheit_typ, datum_from, datum_to FROM abwesenheit WHERE card_uid = $1 AND datum_from <= $2 AND datum_to >= $3 ORDER BY datum_from;`) | ||||||
|  | 	qStr, err := DB.Prepare(` | ||||||
|  | 	SELECT | ||||||
|  | 	    ab.counter_id, | ||||||
|  | 	    gs::DATE AS work_date, | ||||||
|  | 	    ab.card_uid, | ||||||
|  | 	    ab.datum_from, | ||||||
|  | 	    ab.datum_to, | ||||||
|  | 	    jsonb_build_object( | ||||||
|  | 	        'abwesenheit_id', sat.abwesenheit_id, | ||||||
|  | 	        'abwesenheit_name', sat.abwesenheit_name, | ||||||
|  | 	        'arbeitszeit_equivalent', sat.arbeitszeit_equivalent | ||||||
|  | 	    ) AS abwesenheit_info | ||||||
|  | 	FROM generate_series( | ||||||
|  | 	        $2, | ||||||
|  | 	        $3, | ||||||
|  | 	        INTERVAL '1 day' | ||||||
|  | 	    ) gs | ||||||
|  | 	JOIN abwesenheit ab | ||||||
|  | 	  ON ab.card_uid = $1 | ||||||
|  | 	 AND ab.datum_from::DATE <= gs::DATE | ||||||
|  | 	 AND ab.datum_to::DATE >= gs::DATE | ||||||
|  | 	LEFT JOIN s_abwesenheit_typen sat | ||||||
|  | 	  ON ab.abwesenheit_typ = sat.abwesenheit_id | ||||||
|  | 	ORDER BY gs::DATE, ab.counter_id; | ||||||
|  | 	`) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return absences, err | 		return absences, err | ||||||
| 	} | 	} | ||||||
| @@ -81,15 +183,16 @@ func GetAbsencesByCardUID(card_uid string, tsFrom time.Time, tsTo time.Time) ([] | |||||||
| 	defer rows.Close() | 	defer rows.Close() | ||||||
| 	for rows.Next() { | 	for rows.Next() { | ||||||
| 		var absence Absence | 		var absence Absence | ||||||
| 		if err := rows.Scan(&absence.CounterId, &absence.AbwesenheitTyp.Id, &absence.Datum); err != nil { | 		var abwesenheitsTyp []byte | ||||||
|  | 		if err := rows.Scan(&absence.CounterId, &absence.Day, &absence.CardUID, &absence.DateFrom, &absence.DateTo, &abwesenheitsTyp); err != nil { | ||||||
| 			return absences, err | 			return absences, err | ||||||
| 		} | 		} | ||||||
| 		absence.AbwesenheitTyp, err = GetAbsenceTypeById(absence.AbwesenheitTyp.Id) | 		err = json.Unmarshal(abwesenheitsTyp, &absence.AbwesenheitTyp) | ||||||
| 		if err == nil { | 		if err != nil { | ||||||
| 			absences = append(absences, absence) | 			log.Println("Error parsing abwesenheitsTyp to JSON!", err) | ||||||
| 		} else { | 			return absences, nil | ||||||
| 			log.Println("Cannot populate absence type!", err) |  | ||||||
| 		} | 		} | ||||||
|  | 		absences = append(absences, absence) | ||||||
| 	} | 	} | ||||||
| 	if err = rows.Err(); err != nil { | 	if err = rows.Err(); err != nil { | ||||||
| 		return absences, err | 		return absences, err | ||||||
| @@ -97,6 +200,27 @@ func GetAbsencesByCardUID(card_uid string, tsFrom time.Time, tsTo time.Time) ([] | |||||||
| 	return absences, nil | 	return absences, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (a *Absence) Update(na Absence) bool { | ||||||
|  | 	change := false | ||||||
|  | 	if a.CardUID != na.CardUID && na.CardUID != "" { | ||||||
|  | 		a.CardUID = na.CardUID | ||||||
|  | 		change = true | ||||||
|  | 	} | ||||||
|  | 	if a.AbwesenheitTyp != na.AbwesenheitTyp && na.AbwesenheitTyp.Id != 0 { | ||||||
|  | 		a.AbwesenheitTyp = na.AbwesenheitTyp | ||||||
|  | 		change = true | ||||||
|  | 	} | ||||||
|  | 	if !a.DateFrom.Equal(na.DateFrom) && !na.DateFrom.IsZero() { | ||||||
|  | 		a.DateFrom = na.DateFrom | ||||||
|  | 		change = true | ||||||
|  | 	} | ||||||
|  | 	if !a.DateTo.Equal(na.DateTo) && !na.DateTo.IsZero() { | ||||||
|  | 		a.DateTo = na.DateTo | ||||||
|  | 		change = true | ||||||
|  | 	} | ||||||
|  | 	return change | ||||||
|  | } | ||||||
|  |  | ||||||
| func GetAbsenceTypes() (map[int8]AbsenceType, error) { | func GetAbsenceTypes() (map[int8]AbsenceType, error) { | ||||||
| 	var types = make(map[int8]AbsenceType) | 	var types = make(map[int8]AbsenceType) | ||||||
| 	qStr, err := DB.Prepare("SELECT abwesenheit_id, abwesenheit_name, arbeitszeit_equivalent FROM s_abwesenheit_typen;") | 	qStr, err := DB.Prepare("SELECT abwesenheit_id, abwesenheit_name, arbeitszeit_equivalent FROM s_abwesenheit_typen;") | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ import ( | |||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"log" | 	"log" | ||||||
|  | 	"log/slog" | ||||||
| 	"net/url" | 	"net/url" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -87,6 +88,27 @@ func (b *Booking) Verify() bool { | |||||||
| 	return true | 	return true | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (b *Booking) IsSubmittedAndChecked() bool { | ||||||
|  | 	qStr, err := DB.Prepare(`SELECT bestaetigt from wochen_report WHERE $1 = ANY(anwesenheiten);`) | ||||||
|  | 	if err != nil { | ||||||
|  | 		slog.Warn("Error when preparing SQL Statement", "error", err) | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	defer qStr.Close() | ||||||
|  | 	var isSubmittedAndChecked bool = false | ||||||
|  |  | ||||||
|  | 	err = qStr.QueryRow(b.CounterId).Scan(&isSubmittedAndChecked) | ||||||
|  | 	if err == sql.ErrNoRows { | ||||||
|  | 		// No rows found ==> not even submitted | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err != nil { | ||||||
|  | 		slog.Warn("Unexpected error when executing SQL Statement", "error", err) | ||||||
|  | 	} | ||||||
|  | 	return isSubmittedAndChecked | ||||||
|  | } | ||||||
|  |  | ||||||
| func (b *Booking) Insert() error { | func (b *Booking) Insert() error { | ||||||
| 	if !checkLastBooking(*b) { | 	if !checkLastBooking(*b) { | ||||||
| 		return SameBookingError{} | 		return SameBookingError{} | ||||||
|   | |||||||
| @@ -85,6 +85,15 @@ func (u *User) GetAll() ([]User, error) { | |||||||
| 	return users, nil | 	return users, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Returns the worktime per day rounded to minutes | ||||||
|  | func (u *User) ArbeitszeitProTag() time.Duration { | ||||||
|  | 	return time.Duration(u.ArbeitszeitPerTag * float32(time.Hour)).Round(time.Minute) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (u *User) ArbeitszeitProWoche() time.Duration { | ||||||
|  | 	return time.Duration(u.ArbeitszeitPerWoche * float32(time.Hour)).Round(time.Minute) | ||||||
|  | } | ||||||
|  |  | ||||||
| // Returns true if there is a booking 1 for today -> meaning the user is at work | // Returns true if there is a booking 1 for today -> meaning the user is at work | ||||||
| // Returns false if there is no booking today or the user is already booked out of the system | // Returns false if there is no booking today or the user is already booked out of the system | ||||||
| func (u *User) CheckAnwesenheit() bool { | func (u *User) CheckAnwesenheit() bool { | ||||||
| @@ -164,7 +173,7 @@ func (u *User) ChangePass(password, newPassword string) (bool, error) { | |||||||
|  |  | ||||||
| func (u *User) GetTeamMembers() ([]User, error) { | func (u *User) GetTeamMembers() ([]User, error) { | ||||||
| 	var teamMembers []User | 	var teamMembers []User | ||||||
| 	qStr, err := DB.Prepare(`SELECT personal_nummer FROM s_personal_daten WHERE vorgesetzter_pers_nr = $1`) | 	qStr, err := DB.Prepare(`SELECT personal_nummer FROM s_personal_daten WHERE vorgesetzter_pers_nr = $1 ORDER BY "nachname";`) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return teamMembers, err | 		return teamMembers, err | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -2,21 +2,182 @@ package models | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"arbeitszeitmessung/helper" | 	"arbeitszeitmessung/helper" | ||||||
| 	"database/sql" |  | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
|  | 	"fmt" | ||||||
| 	"log" | 	"log" | ||||||
| 	"strconv" | 	"log/slog" | ||||||
|  | 	"sort" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | type IWorkDay interface { | ||||||
|  | 	Date() time.Time | ||||||
|  | 	TimeWorkVirtual(User) time.Duration | ||||||
|  | 	TimeWorkReal(User) time.Duration | ||||||
|  | 	TimePauseReal(User) (work, pause time.Duration) | ||||||
|  | 	TimeOvertimeReal(User) time.Duration | ||||||
|  | 	GetAllWorkTimesVirtual(User) (work, pause, overtime time.Duration) | ||||||
|  | 	ToString() string | ||||||
|  | 	IsWorkDay() bool | ||||||
|  | 	IsKurzArbeit() bool | ||||||
|  | 	GetDayProgress(User) int8 | ||||||
|  | 	RequiresAction() bool | ||||||
|  | } | ||||||
|  |  | ||||||
| type WorkDay struct { | type WorkDay struct { | ||||||
| 	Day       time.Time `json:"day"` | 	Day               time.Time `json:"day"` | ||||||
| 	Bookings  []Booking `json:"bookings"` | 	Bookings          []Booking `json:"bookings"` | ||||||
| 	workTime  time.Duration | 	workTime          time.Duration | ||||||
| 	pauseTime time.Duration | 	pauseTime         time.Duration | ||||||
| 	TimeFrom  time.Time | 	realWorkTime      time.Duration | ||||||
| 	TimeTo    time.Time | 	realPauseTime     time.Duration | ||||||
| 	Absence   Absence | 	TimeFrom          time.Time | ||||||
|  | 	TimeTo            time.Time | ||||||
|  | 	kurzArbeit        bool | ||||||
|  | 	kurzArbeitAbsence Absence | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func GetDays(user User, tsFrom, tsTo time.Time, orderedForward bool) []IWorkDay { | ||||||
|  | 	var allDays map[string]IWorkDay = make(map[string]IWorkDay) | ||||||
|  |  | ||||||
|  | 	for _, day := range GetWorkDays(user, tsFrom, tsTo) { | ||||||
|  | 		allDays[day.Date().Format(time.DateOnly)] = &day | ||||||
|  | 	} | ||||||
|  | 	absences, err := GetAbsencesByCardUID(user.CardUID, tsFrom, tsTo) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Println("Error gettings absences for all Days!", err) | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	for _, day := range absences { | ||||||
|  | 		if helper.IsWeekend(day.Date()) { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		if day.AbwesenheitTyp.WorkTime == 1 { | ||||||
|  | 			if workDay, ok := allDays[day.Date().Format(time.DateOnly)].(*WorkDay); ok && len(workDay.Bookings) > 0 { | ||||||
|  | 				workDay.kurzArbeit = true | ||||||
|  | 				workDay.kurzArbeitAbsence = day | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			allDays[day.Date().Format(time.DateOnly)] = &day | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	sortedDays := sortDays(allDays, orderedForward) | ||||||
|  | 	return sortedDays | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func sortDays(days map[string]IWorkDay, forward bool) []IWorkDay { | ||||||
|  | 	var sortedDays []IWorkDay | ||||||
|  | 	for _, day := range days { | ||||||
|  | 		sortedDays = append(sortedDays, day) | ||||||
|  | 	} | ||||||
|  | 	if forward { | ||||||
|  | 		sort.Slice(sortedDays, func(i, j int) bool { | ||||||
|  | 			return sortedDays[i].Date().After(sortedDays[j].Date()) | ||||||
|  | 		}) | ||||||
|  | 	} else { | ||||||
|  | 		sort.Slice(sortedDays, func(i, j int) bool { | ||||||
|  | 			return sortedDays[i].Date().Before(sortedDays[j].Date()) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | 	return sortedDays | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) Date() time.Time { | ||||||
|  | 	return d.Day | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) GenerateKurzArbeitBookings(u User) (time.Time, time.Time) { | ||||||
|  | 	var timeFrom, timeTo time.Time | ||||||
|  | 	if d.workTime >= u.ArbeitszeitProTag() { | ||||||
|  | 		return timeFrom, timeTo | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	timeFrom = d.Bookings[len(d.Bookings)-1].Timestamp.Add(time.Minute) | ||||||
|  | 	timeTo = timeFrom.Add(u.ArbeitszeitProTag() - d.workTime) | ||||||
|  | 	slog.Debug("Added duration as Kurzarbeit", "date", d.Date().String(), "duration", timeTo.Sub(timeFrom).String()) | ||||||
|  |  | ||||||
|  | 	return timeFrom, timeTo | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) TimeWorkVirtual(u User) time.Duration { | ||||||
|  | 	if d.IsKurzArbeit() { | ||||||
|  | 		return u.ArbeitszeitProTag() | ||||||
|  | 	} | ||||||
|  | 	return d.workTime | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) GetKurzArbeit() *Absence { | ||||||
|  | 	return &d.kurzArbeitAbsence | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) TimeWorkReal(u User) time.Duration { | ||||||
|  | 	d.realWorkTime, d.realPauseTime = 0, 0 | ||||||
|  | 	var lastBooking Booking | ||||||
|  | 	for _, booking := range d.Bookings { | ||||||
|  | 		if booking.CheckInOut%2 == 1 { | ||||||
|  | 			if !lastBooking.Timestamp.IsZero() { | ||||||
|  | 				d.realPauseTime += booking.Timestamp.Sub(lastBooking.Timestamp) | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			d.realWorkTime += booking.Timestamp.Sub(lastBooking.Timestamp) | ||||||
|  | 		} | ||||||
|  | 		lastBooking = booking | ||||||
|  | 	} | ||||||
|  | 	if helper.IsSameDate(d.Date(), time.Now()) && len(d.Bookings)%2 == 1 { | ||||||
|  | 		d.realWorkTime += time.Since(lastBooking.Timestamp.Local()) | ||||||
|  | 	} | ||||||
|  | 	slog.Debug("Calculated RealWorkTime for user", "user", u, slog.String("worktime", d.realWorkTime.String())) | ||||||
|  | 	return d.realWorkTime | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) TimeOvertimeReal(u User) time.Duration { | ||||||
|  | 	workTime := d.TimeWorkVirtual(u) | ||||||
|  | 	if workTime == 0 { | ||||||
|  | 		workTime, _ = d.TimePauseReal(u) | ||||||
|  | 	} | ||||||
|  | 	if helper.IsWeekend(d.Day) && len(d.Bookings) == 0 { | ||||||
|  | 		return 0 | ||||||
|  | 	} | ||||||
|  | 	var overtime time.Duration | ||||||
|  | 	overtime = workTime - u.ArbeitszeitProTag() | ||||||
|  | 	return overtime | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) TimePauseReal(u User) (work, pause time.Duration) { | ||||||
|  | 	if d.realWorkTime == 0 { | ||||||
|  | 		d.TimeWorkReal(u) | ||||||
|  | 	} | ||||||
|  | 	d.workTime, d.pauseTime = d.realWorkTime, d.realPauseTime | ||||||
|  | 	if d.realWorkTime <= 6*time.Hour || d.realPauseTime > 45*time.Minute { | ||||||
|  | 		return d.realWorkTime, d.realPauseTime | ||||||
|  | 	} | ||||||
|  | 	if d.realWorkTime <= (9*time.Hour) && d.realPauseTime < 30*time.Minute { | ||||||
|  | 		diff := 30*time.Minute - d.pauseTime | ||||||
|  | 		d.workTime -= diff | ||||||
|  | 		d.pauseTime += diff | ||||||
|  | 	} else if d.realPauseTime < 45*time.Minute { | ||||||
|  | 		diff := 45*time.Minute - d.pauseTime | ||||||
|  | 		d.workTime -= diff | ||||||
|  | 		d.pauseTime += diff | ||||||
|  | 	} | ||||||
|  | 	return d.workTime, d.pauseTime | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) ToString() string { | ||||||
|  | 	return fmt.Sprintf("WorkDay: %s with %d bookings and worktime: %s", d.Date().Format(time.DateOnly), len(d.Bookings), helper.FormatDuration(d.workTime)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) IsWorkDay() bool { | ||||||
|  | 	return true | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) SetKurzArbeit(kurzArbeit bool) { | ||||||
|  | 	d.kurzArbeit = kurzArbeit | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *WorkDay) IsKurzArbeit() bool { | ||||||
|  | 	return d.kurzArbeit | ||||||
| } | } | ||||||
|  |  | ||||||
| func GetWorkDays(user User, tsFrom, tsTo time.Time) []WorkDay { | func GetWorkDays(user User, tsFrom, tsTo time.Time) []WorkDay { | ||||||
| @@ -24,72 +185,60 @@ func GetWorkDays(user User, tsFrom, tsTo time.Time) []WorkDay { | |||||||
| 	var workSec, pauseSec float64 | 	var workSec, pauseSec float64 | ||||||
|  |  | ||||||
| 	qStr, err := DB.Prepare(` | 	qStr, err := DB.Prepare(` | ||||||
| 		WITH all_days AS ( | 	WITH all_days AS ( | ||||||
|     SELECT generate_series($2::DATE, $3::DATE - INTERVAL '1 day', INTERVAL '1 day')::DATE AS work_date |     	SELECT generate_series($2::DATE, $3::DATE - INTERVAL '1 day', INTERVAL '1 day')::DATE AS work_date), | ||||||
| ), |     ordered_bookings AS ( | ||||||
| ordered_bookings AS ( | 	    SELECT | ||||||
|     SELECT | 	        a.timestamp::DATE AS work_date, | ||||||
|         a.timestamp::DATE AS work_date, | 	        a.timestamp, | ||||||
|         a.timestamp, | 	        a.check_in_out, | ||||||
|         a.check_in_out, | 	        a.counter_id, | ||||||
|         a.counter_id, | 	        a.anwesenheit_typ, | ||||||
|         a.anwesenheit_typ, | 	        sat.anwesenheit_name AS anwesenheit_typ_name, | ||||||
|         sat.anwesenheit_name AS anwesenheit_typ_name, | 	        LAG(a.timestamp) OVER (PARTITION BY a.card_uid, a.timestamp::DATE ORDER BY a.timestamp) AS prev_timestamp, | ||||||
|         LAG(a.timestamp) OVER (PARTITION BY a.card_uid, a.timestamp::DATE ORDER BY a.timestamp) AS prev_timestamp, | 	        LAG(a.check_in_out) OVER (PARTITION BY a.card_uid, a.timestamp::DATE ORDER BY a.timestamp) AS prev_check | ||||||
|         LAG(a.check_in_out) OVER (PARTITION BY a.card_uid, a.timestamp::DATE ORDER BY a.timestamp) AS prev_check | 	    FROM anwesenheit a | ||||||
|     FROM anwesenheit a | 	    LEFT JOIN s_anwesenheit_typen sat ON a.anwesenheit_typ = sat.anwesenheit_id | ||||||
|     LEFT JOIN s_anwesenheit_typen sat ON a.anwesenheit_typ = sat.anwesenheit_id | 	    WHERE a.card_uid = $1 | ||||||
|     WHERE a.card_uid = $1 | 	      AND a.timestamp::DATE >= $2 | ||||||
|       AND a.timestamp::DATE >= $2 | 	      AND a.timestamp::DATE <= $3 | ||||||
|       AND a.timestamp::DATE <= $3 | 	) | ||||||
| ), | 	SELECT | ||||||
| abwesenheiten AS ( | 	    d.work_date, | ||||||
|     SELECT | 	    COALESCE(MIN(b.timestamp), NOW()) AS time_from, | ||||||
|         datum::DATE AS work_date, | 	    COALESCE(MAX(b.timestamp), NOW()) AS time_to, | ||||||
|         abwesenheit_typ | 	    COALESCE( | ||||||
|     FROM abwesenheit | 	        EXTRACT(EPOCH FROM SUM( | ||||||
|     WHERE card_uid = $1 | 	            CASE | ||||||
|       AND datum::DATE >= $2 | 	                WHEN b.prev_check IN (1, 3) AND b.check_in_out IN (2, 4, 254) | ||||||
|       AND datum::DATE <= $3 | 	                THEN b.timestamp - b.prev_timestamp | ||||||
| ) | 	                ELSE INTERVAL '0' | ||||||
| SELECT | 	            END | ||||||
|     d.work_date, | 	        )), 0 | ||||||
|     COALESCE(MIN(b.timestamp), NOW()) AS time_from, | 	    ) AS total_work_seconds, | ||||||
|     COALESCE(MAX(b.timestamp), NOW()) AS time_to, | 	    COALESCE( | ||||||
|     COALESCE( | 	        EXTRACT(EPOCH FROM SUM( | ||||||
|         EXTRACT(EPOCH FROM SUM( | 	            CASE | ||||||
|             CASE | 	                WHEN b.prev_check IN (2, 4, 254) AND b.check_in_out IN (1, 3) | ||||||
|                 WHEN b.prev_check IN (1, 3) AND b.check_in_out IN (2, 4, 254) | 	                THEN b.timestamp - b.prev_timestamp | ||||||
|                 THEN b.timestamp - b.prev_timestamp | 	                ELSE INTERVAL '0' | ||||||
|                 ELSE INTERVAL '0' | 	            END | ||||||
|             END | 	        )), 0 | ||||||
|         )), 0 | 	    ) AS total_pause_seconds, | ||||||
|     ) AS total_work_seconds, | 	    COALESCE(jsonb_agg(jsonb_build_object( | ||||||
|     COALESCE( | 	        'check_in_out', b.check_in_out, | ||||||
|         EXTRACT(EPOCH FROM SUM( | 	        'timestamp', b.timestamp, | ||||||
|             CASE | 	        'counter_id', b.counter_id, | ||||||
|                 WHEN b.prev_check IN (2, 4, 254) AND b.check_in_out IN (1, 3) | 	        'anwesenheit_typ', b.anwesenheit_typ, | ||||||
|                 THEN b.timestamp - b.prev_timestamp | 	        'anwesenheit_typ', jsonb_build_object( | ||||||
|                 ELSE INTERVAL '0' | 	            'anwesenheit_id', b.anwesenheit_typ, | ||||||
|             END | 	            'anwesenheit_name', b.anwesenheit_typ_name | ||||||
|         )), 0 | 	        ) | ||||||
|     ) AS total_pause_seconds, | 	    ) ORDER BY b.timestamp), '[]'::jsonb) AS bookings | ||||||
|     COALESCE(jsonb_agg(jsonb_build_object( | 	FROM all_days d | ||||||
|         'check_in_out', b.check_in_out, | 	LEFT JOIN ordered_bookings b ON d.work_date = b.work_date | ||||||
|         'timestamp', b.timestamp, | 	GROUP BY d.work_date | ||||||
|         'counter_id', b.counter_id, | 	ORDER BY d.work_date ASC;`) | ||||||
|         'anwesenheit_typ', b.anwesenheit_typ, |  | ||||||
|         'anwesenheit_typ', jsonb_build_object( |  | ||||||
|             'anwesenheit_id', b.anwesenheit_typ, |  | ||||||
|             'anwesenheit_name', b.anwesenheit_typ_name |  | ||||||
|         ) |  | ||||||
|     ) ORDER BY b.timestamp), '[]'::jsonb) AS bookings, |  | ||||||
|     a.abwesenheit_typ |  | ||||||
| FROM all_days d |  | ||||||
| LEFT JOIN ordered_bookings b ON d.work_date = b.work_date |  | ||||||
| LEFT JOIN abwesenheiten a ON d.work_date = a.work_date |  | ||||||
| GROUP BY d.work_date, a.abwesenheit_typ |  | ||||||
| ORDER BY d.work_date ASC;`) |  | ||||||
|  |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Error preparing SQL statement", err) | 		log.Println("Error preparing SQL statement", err) | ||||||
| @@ -103,12 +252,11 @@ ORDER BY d.work_date ASC;`) | |||||||
| 		return workDays | 		return workDays | ||||||
| 	} | 	} | ||||||
| 	defer rows.Close() | 	defer rows.Close() | ||||||
| 	emptyDays, _ := strconv.ParseBool(helper.GetEnv("EMPTY_DAYS", "false")) | 	// emptyDays, _ := strconv.ParseBool(helper.GetEnv("EMPTY_DAYS", "false")) | ||||||
| 	for rows.Next() { | 	for rows.Next() { | ||||||
| 		var workDay WorkDay | 		var workDay WorkDay | ||||||
| 		var bookings []byte | 		var bookings []byte | ||||||
| 		var absenceType sql.NullInt16 | 		if err := rows.Scan(&workDay.Day, &workDay.TimeFrom, &workDay.TimeTo, &workSec, &pauseSec, &bookings); err != nil { | ||||||
| 		if err := rows.Scan(&workDay.Day, &workDay.TimeFrom, &workDay.TimeTo, &workSec, &pauseSec, &bookings, &absenceType); err != nil { |  | ||||||
| 			log.Println("Error scanning row!", err) | 			log.Println("Error scanning row!", err) | ||||||
| 			return workDays | 			return workDays | ||||||
| 		} | 		} | ||||||
| @@ -123,89 +271,28 @@ ORDER BY d.work_date ASC;`) | |||||||
| 		if len(workDay.Bookings) == 1 && workDay.Bookings[0].CounterId == 0 { | 		if len(workDay.Bookings) == 1 && workDay.Bookings[0].CounterId == 0 { | ||||||
| 			workDay.Bookings = []Booking{} | 			workDay.Bookings = []Booking{} | ||||||
| 		} | 		} | ||||||
|  | 		workDay.TimePauseReal(user) | ||||||
| 		if absenceType.Valid { | 		if len(workDay.Bookings) > 1 || !helper.IsWeekend(workDay.Date()) { | ||||||
| 			workDay.Absence, err = NewAbsence(user.CardUID, int(absenceType.Int16), workDay.Day) |  | ||||||
| 			workDay.CalcRealWorkTime(user) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if workDay.Day.Equal(time.Now().Truncate(24 * time.Hour)) { |  | ||||||
| 			workDay.CalcRealWorkTime(user) |  | ||||||
| 			workDay.CalcWorkPauseDiff(user) |  | ||||||
| 		} else { |  | ||||||
| 			workDay.CalcWorkPauseDiff(user) |  | ||||||
| 		} |  | ||||||
| 		if emptyDays && workDay.Day.Weekday() >= 1 && workDay.Day.Weekday() <= 5 { |  | ||||||
| 			workDays = append(workDays, workDay) |  | ||||||
| 		} else if len(workDay.Bookings) > 0 || (workDay.Absence != Absence{}) { |  | ||||||
| 			workDays = append(workDays, workDay) | 			workDays = append(workDays, workDay) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	if err = rows.Err(); err != nil { | 	if err = rows.Err(); err != nil { | ||||||
|  | 		log.Println("Error in workday rows!", err) | ||||||
| 		return workDays | 		return workDays | ||||||
| 	} | 	} | ||||||
| 	return workDays | 	return workDays | ||||||
| } | } | ||||||
|  |  | ||||||
| func (d *WorkDay) CalcWorkPauseDiff(user User) (work, pause time.Duration) { | func (d *WorkDay) GetAllWorkTimesReal(user User) (work, pause, overtime time.Duration) { | ||||||
| 	if d.workTime == 0 { | 	if d.pauseTime == 0 || d.workTime == 0 { | ||||||
| 		d.CalcRealWorkTime(user) | 		d.TimePauseReal(user) | ||||||
| 	} | 	} | ||||||
| 	if d.Absence.AbwesenheitTyp.WorkTime > 0 { | 	return d.workTime.Round(time.Minute), d.pauseTime.Round(time.Minute), d.TimeOvertimeReal(user) | ||||||
| 		return d.workTime, d.pauseTime |  | ||||||
| 	} |  | ||||||
| 	if d.workTime <= 6*time.Hour || d.pauseTime > 45*time.Minute { |  | ||||||
| 		return d.workTime, d.pauseTime |  | ||||||
| 	} |  | ||||||
| 	if d.workTime <= (9*time.Hour) && d.pauseTime < 30*time.Minute { |  | ||||||
| 		diff := 30*time.Minute - d.pauseTime |  | ||||||
| 		d.workTime -= diff |  | ||||||
| 		d.pauseTime += diff |  | ||||||
| 	} else if d.pauseTime < 45*time.Minute { |  | ||||||
| 		diff := 45*time.Minute - d.pauseTime |  | ||||||
| 		d.workTime -= diff |  | ||||||
| 		d.pauseTime += diff |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return d.workTime, d.pauseTime |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (d *WorkDay) CalcRealWorkTime(user User) time.Duration { | func (d *WorkDay) GetAllWorkTimesVirtual(user User) (work, pause, overtime time.Duration) { | ||||||
| 	if (len(d.Bookings) < 1 && d.Absence == Absence{}) { | 	_, pause, overtime = d.GetAllWorkTimesReal(user) | ||||||
| 		return 0 | 	return d.TimeWorkVirtual(user), pause, overtime | ||||||
| 	} |  | ||||||
| 	var realWorkTime, realPauseTime time.Duration |  | ||||||
| 	var lastBooking Booking |  | ||||||
| 	for _, booking := range d.Bookings { |  | ||||||
| 		if booking.CheckInOut%2 == 1 { |  | ||||||
| 			if !lastBooking.Timestamp.IsZero() { |  | ||||||
| 				realPauseTime += booking.Timestamp.Sub(lastBooking.Timestamp) |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			realWorkTime += booking.Timestamp.Sub(lastBooking.Timestamp) |  | ||||||
| 		} |  | ||||||
| 		lastBooking = booking |  | ||||||
| 	} |  | ||||||
| 	if helper.IsSameDate(d.Day, time.Now()) && len(d.Bookings)%2 == 1 { |  | ||||||
| 		realWorkTime += time.Since(lastBooking.Timestamp.Local()) |  | ||||||
| 	} |  | ||||||
| 	if d.Absence.AbwesenheitTyp.WorkTime > 0 { |  | ||||||
| 		realWorkTime = time.Duration(user.ArbeitszeitPerTag * float32(time.Hour)).Round(time.Minute) |  | ||||||
| 	} |  | ||||||
| 	d.workTime = realWorkTime |  | ||||||
| 	d.pauseTime = realPauseTime |  | ||||||
|  |  | ||||||
| 	return realWorkTime |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (d *WorkDay) GetWorkTimeString() (work string, pause string) { |  | ||||||
| 	workString := helper.FormatDuration(d.workTime) |  | ||||||
| 	pauseString := helper.FormatDuration(d.pauseTime) |  | ||||||
| 	return workString, pauseString |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (d *WorkDay) GetAllWorkTimes(user User) (work, pause, overtime time.Duration) { |  | ||||||
| 	return d.workTime.Round(time.Minute), d.pauseTime.Round(time.Minute), d.CalcOvertime(user) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| // returns bool wheter the workday was ended with an automatic logout | // returns bool wheter the workday was ended with an automatic logout | ||||||
| @@ -216,21 +303,23 @@ func (d *WorkDay) RequiresAction() bool { | |||||||
| 	return d.Bookings[len(d.Bookings)-1].CheckInOut == 254 | 	return d.Bookings[len(d.Bookings)-1].CheckInOut == 254 | ||||||
| } | } | ||||||
|  |  | ||||||
| // returns a integer percentage of how much day has been worked of | func (d *WorkDay) GetDayProgress(u User) int8 { | ||||||
| func (d *WorkDay) GetWorkDayProgress(user User) uint8 { | 	if d.RequiresAction() { | ||||||
| 	defaultWorkTime := time.Duration(user.ArbeitszeitPerTag * float32(time.Hour)).Round(time.Minute) | 		return -1 | ||||||
| 	progress := (d.workTime.Seconds() / defaultWorkTime.Seconds()) * 100 | 	} | ||||||
| 	return uint8(progress) | 	workTime := d.TimeWorkVirtual(u) | ||||||
|  | 	progress := (workTime.Seconds() / u.ArbeitszeitProTag().Seconds()) * 100 | ||||||
|  | 	return int8(progress) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (d *WorkDay) CalcOvertime(user User) time.Duration { | // func (d *WorkDay) CalcOvertime(user User) time.Duration { | ||||||
| 	if d.workTime == 0 { | // 	if d.workTime == 0 { | ||||||
| 		d.CalcWorkPauseDiff(user) | // 		d.TimePauseReal(user) | ||||||
| 	} | // 	} | ||||||
| 	if helper.IsWeekend(d.Day) && len(d.Bookings) == 0 { | // 	if helper.IsWeekend(d.Day) && len(d.Bookings) == 0 { | ||||||
| 		return 0 | // 		return 0 | ||||||
| 	} | // 	} | ||||||
| 	var overtime time.Duration | // 	var overtime time.Duration | ||||||
| 	overtime = d.workTime - time.Duration(user.ArbeitszeitPerTag*float32(time.Hour)).Round(time.Minute) | // 	overtime = d.workTime - user.ArbeitszeitProTag() | ||||||
| 	return overtime | // 	return overtime | ||||||
| } | // } | ||||||
|   | |||||||
| @@ -16,15 +16,14 @@ func CatchError[T any](val T, err error) T { | |||||||
| } | } | ||||||
|  |  | ||||||
| var testWorkDay = models.WorkDay{ | var testWorkDay = models.WorkDay{ | ||||||
| 	Day:      CatchError(time.Parse("2006-01-02", "2025-01-01")), | 	Day:      CatchError(time.Parse(time.DateOnly, "2025-01-01")), | ||||||
| 	Bookings: testBookings8hrs, | 	Bookings: testBookings8hrs, | ||||||
| 	TimeFrom: CatchError(time.Parse("2006-01-02 15:04", "2025-01-01 08:00")), | 	TimeFrom: CatchError(time.Parse("2006-01-02 15:04", "2025-01-01 08:00")), | ||||||
| 	TimeTo:   CatchError(time.Parse("2006-01-02 15:04", "2025-01-01 16:30")), | 	TimeTo:   CatchError(time.Parse("2006-01-02 15:04", "2025-01-01 16:30")), | ||||||
| 	Absence:  models.Absence{}, |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestCalcRealWorkTime(t *testing.T) { | func TestCalcRealWorkTime(t *testing.T) { | ||||||
| 	workTime := testWorkDay.CalcRealWorkTime(testUser) | 	workTime := testWorkDay.TimeWorkReal(testUser) | ||||||
| 	if workTime != time.Hour*8 { | 	if workTime != time.Hour*8 { | ||||||
| 		t.Errorf("Calc Worktime Default not working, time should be 8h, but was %s", helper.FormatDuration(workTime)) | 		t.Errorf("Calc Worktime Default not working, time should be 8h, but was %s", helper.FormatDuration(workTime)) | ||||||
| 	} | 	} | ||||||
| @@ -64,10 +63,10 @@ func TestCalcWorkPauseDiff(t *testing.T) { | |||||||
| 	for _, test := range testCases { | 	for _, test := range testCases { | ||||||
| 		t.Run(test.Name, func(t *testing.T) { | 		t.Run(test.Name, func(t *testing.T) { | ||||||
| 			testWorkDay.Bookings = test.bookings | 			testWorkDay.Bookings = test.bookings | ||||||
| 			testWorkDay.CalcRealWorkTime(testUser) | 			testWorkDay.TimeWorkReal(testUser) | ||||||
| 			testWorkDay.CalcWorkPauseDiff(testUser) | 			testWorkDay.TimePauseReal(testUser) | ||||||
| 			testWorkDay.CalcOvertime(testUser) | 			testWorkDay.TimeOvertimeReal(testUser) | ||||||
| 			workTime, pauseTime, overTime := testWorkDay.GetAllWorkTimes(testUser) | 			workTime, pauseTime, overTime := testWorkDay.GetAllWorkTimesReal(testUser) | ||||||
| 			if workTime != test.expectedWorkTime { | 			if workTime != test.expectedWorkTime { | ||||||
| 				t.Errorf("Calculated wrong workTime: should be %s, but was %s", helper.FormatDuration(test.expectedWorkTime), helper.FormatDuration(workTime)) | 				t.Errorf("Calculated wrong workTime: should be %s, but was %s", helper.FormatDuration(test.expectedWorkTime), helper.FormatDuration(workTime)) | ||||||
| 			} | 			} | ||||||
|   | |||||||
| @@ -1,26 +1,28 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"arbeitszeitmessung/helper" |  | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"log" | 	"log" | ||||||
|  | 	"log/slog" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/lib/pq" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Workweeks are | // Workweeks are | ||||||
|  |  | ||||||
| type WorkWeek struct { | type WorkWeek struct { | ||||||
| 	Id           int | 	Id              int | ||||||
| 	WorkDays     []WorkDay | 	WorkDays        []WorkDay | ||||||
| 	Absences     []Absence | 	Absences        []Absence | ||||||
| 	User         User | 	Days            []IWorkDay | ||||||
| 	WeekStart    time.Time | 	User            User | ||||||
| 	Worktime     time.Duration | 	WeekStart       time.Time | ||||||
| 	Overtime     time.Duration | 	Worktime        time.Duration | ||||||
| 	Status       WeekStatus | 	WorkTimeVirtual time.Duration | ||||||
| 	overtimeDiff time.Duration | 	Overtime        time.Duration | ||||||
| 	worktimeDiff time.Duration | 	Status          WeekStatus | ||||||
| } | } | ||||||
|  |  | ||||||
| type WeekStatus int8 | type WeekStatus int8 | ||||||
| @@ -39,20 +41,23 @@ func NewWorkWeek(user User, tsMonday time.Time, populate bool) WorkWeek { | |||||||
| 		Status:    WeekStatusNone, | 		Status:    WeekStatusNone, | ||||||
| 	} | 	} | ||||||
| 	if populate { | 	if populate { | ||||||
| 		week.PopulateWithBookings(0, 0) | 		week.PopulateWithDays(0, 0) | ||||||
| 	} | 	} | ||||||
| 	return week | 	return week | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *WorkWeek) PopulateWithBookings(worktime time.Duration, overtime time.Duration) { | func (w *WorkWeek) PopulateWithDays(worktime time.Duration, overtime time.Duration) { | ||||||
| 	w.WorkDays = GetWorkDays(w.User, w.WeekStart, w.WeekStart.Add(7*24*time.Hour)) | 	slog.Debug("Got Days with overtime and worktime", slog.String("worktime", worktime.String()), slog.String("overtime", overtime.String())) | ||||||
| 	if absences, err := GetAbsencesByCardUID(w.User.CardUID, w.WeekStart, w.WeekStart.Add(7*24*time.Hour)); err == nil { | 	w.Days = GetDays(w.User, w.WeekStart, w.WeekStart.Add(6*24*time.Hour), false) | ||||||
| 		w.Absences = absences |  | ||||||
| 	} else { | 	for _, day := range w.Days { | ||||||
| 		log.Printf("Error populating absences in workWeek (%s): %v", w.WeekStart, err) | 		work, _ := day.TimePauseReal(w.User) | ||||||
|  | 		w.Worktime += work | ||||||
|  | 		w.WorkTimeVirtual += day.TimeWorkVirtual(w.User) | ||||||
| 	} | 	} | ||||||
| 	w.Worktime = w.aggregateWorkTime() | 	slog.Debug("Got worktime for user", "user", w.User, "worktime", w.Worktime.String(), "virtualWorkTime", w.WorkTimeVirtual.String()) | ||||||
| 	w.Overtime = w.Worktime - time.Duration(w.User.ArbeitszeitPerWoche*float32(time.Hour)) |  | ||||||
|  | 	w.Overtime = w.WorkTimeVirtual - w.User.ArbeitszeitProWoche() | ||||||
|  |  | ||||||
| 	w.Worktime = w.Worktime.Round(time.Minute) | 	w.Worktime = w.Worktime.Round(time.Minute) | ||||||
| 	w.Overtime = w.Overtime.Round(time.Minute) | 	w.Overtime = w.Overtime.Round(time.Minute) | ||||||
| @@ -63,8 +68,6 @@ func (w *WorkWeek) PopulateWithBookings(worktime time.Duration, overtime time.Du | |||||||
|  |  | ||||||
| 	if overtime != w.Overtime || worktime != w.Worktime { | 	if overtime != w.Overtime || worktime != w.Worktime { | ||||||
| 		w.Status = WeekStatusDifferences | 		w.Status = WeekStatusDifferences | ||||||
| 		w.overtimeDiff = overtime |  | ||||||
| 		w.worktimeDiff = worktime |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -99,10 +102,6 @@ func (w *WorkWeek) CheckStatus() WeekStatus { | |||||||
| 	return w.Status | 	return w.Status | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *WorkWeek) GetWorkHourString() string { |  | ||||||
| 	return helper.FormatDuration(w.Worktime) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (w *WorkWeek) aggregateWorkTime() time.Duration { | func (w *WorkWeek) aggregateWorkTime() time.Duration { | ||||||
| 	var workTime time.Duration | 	var workTime time.Duration | ||||||
| 	for _, day := range w.WorkDays { | 	for _, day := range w.WorkDays { | ||||||
| @@ -133,12 +132,13 @@ func (w *WorkWeek) GetSendWeeks(user User) []WorkWeek { | |||||||
| 	defer rows.Close() | 	defer rows.Close() | ||||||
| 	for rows.Next() { | 	for rows.Next() { | ||||||
| 		var week WorkWeek = WorkWeek{User: user} | 		var week WorkWeek = WorkWeek{User: user} | ||||||
| 		if err := rows.Scan(&week.Id, &week.WeekStart, &week.Worktime, &week.Overtime); err != nil { | 		var workTime, overTime time.Duration | ||||||
|  | 		if err := rows.Scan(&week.Id, &week.WeekStart, &workTime, &overTime); err != nil { | ||||||
| 			log.Println("Error scanning row!", err) | 			log.Println("Error scanning row!", err) | ||||||
| 			return weeks | 			return weeks | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		week.PopulateWithBookings(week.Worktime, week.Overtime) | 		week.PopulateWithDays(workTime, overTime) | ||||||
| 		weeks = append(weeks, week) | 		weeks = append(weeks, week) | ||||||
| 	} | 	} | ||||||
| 	if err = rows.Err(); err != nil { | 	if err = rows.Err(); err != nil { | ||||||
| @@ -150,30 +150,70 @@ func (w *WorkWeek) GetSendWeeks(user User) []WorkWeek { | |||||||
|  |  | ||||||
| var ErrRunningWeek = errors.New("Week is in running week") | var ErrRunningWeek = errors.New("Week is in running week") | ||||||
|  |  | ||||||
|  | func (w *WorkWeek) GetBookingIds() (anwesenheitsIds, abwesenheitsIds []int64, err error) { | ||||||
|  | 	qStr, err := DB.Prepare(` | ||||||
|  | 		SELECT | ||||||
|  |   (SELECT array_agg(counter_id ORDER BY counter_id) | ||||||
|  |    FROM anwesenheit | ||||||
|  |    WHERE card_uid = $1 | ||||||
|  |      AND timestamp::DATE >= $2 | ||||||
|  |      AND timestamp::DATE < $3) AS anwesenheit, | ||||||
|  |  | ||||||
|  |   (SELECT array_agg(counter_id ORDER BY counter_id) | ||||||
|  |    FROM abwesenheit | ||||||
|  |    WHERE card_uid = $1 | ||||||
|  |      AND datum_from < $3 | ||||||
|  |      AND datum_to >= $2) AS abwesenheit; | ||||||
|  |      `) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, nil, err | ||||||
|  | 	} | ||||||
|  | 	defer qStr.Close() | ||||||
|  |  | ||||||
|  | 	slog.Debug("Inserting parameters into qStr:", "user card_uid", w.User.CardUID, "week_start", w.WeekStart, "week_end", w.WeekStart.AddDate(0, 0, 5)) | ||||||
|  |  | ||||||
|  | 	err = qStr.QueryRow(w.User.CardUID, w.WeekStart, w.WeekStart.AddDate(0, 0, 5)).Scan(pq.Array(&anwesenheitsIds), pq.Array(&abwesenheitsIds)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return anwesenheitsIds, abwesenheitsIds, err | ||||||
|  | 	} | ||||||
|  | 	return anwesenheitsIds, abwesenheitsIds, nil | ||||||
|  | } | ||||||
|  |  | ||||||
| // creates a new entry in the woche_report table with the given workweek | // creates a new entry in the woche_report table with the given workweek | ||||||
| func (w *WorkWeek) SendWeek() error { | func (w *WorkWeek) SendWeek() error { | ||||||
| 	var qStr *sql.Stmt | 	var qStr *sql.Stmt | ||||||
| 	var err error | 	var err error | ||||||
|  |  | ||||||
|  | 	slog.Info("Sending workWeek to team head", "week", w.WeekStart.String()) | ||||||
|  |  | ||||||
|  | 	anwBookings, awBookings, err := w.GetBookingIds() | ||||||
|  | 	if err != nil { | ||||||
|  | 		slog.Warn("Error querying bookings from work week", slog.Any("error", err)) | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	slog.Debug("Recieved Booking Ids", "anwesenheiten", anwBookings) | ||||||
|  |  | ||||||
| 	if time.Since(w.WeekStart) < 5*24*time.Hour { | 	if time.Since(w.WeekStart) < 5*24*time.Hour { | ||||||
| 		log.Println("Cannot send week, because it's the running week!") | 		slog.Warn("Cannot send week, because it's the running week!") | ||||||
| 		return ErrRunningWeek | 		return ErrRunningWeek | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if w.CheckStatus() != WeekStatusNone { | 	if w.CheckStatus() != WeekStatusNone { | ||||||
| 		qStr, err = DB.Prepare(`UPDATE "wochen_report" SET bestaetigt = FALSE, arbeitszeit = make_interval(secs => $3::numeric / 1000000000), ueberstunden = make_interval(secs => $4::numeric / 1000000000) WHERE personal_nummer = $1 AND woche_start = $2;`) | 		qStr, err = DB.Prepare(`UPDATE "wochen_report" SET bestaetigt = FALSE, arbeitszeit = make_interval(secs => $3::numeric / 1000000000), ueberstunden = make_interval(secs => $4::numeric / 1000000000), anwesenheiten=$5, abwesenheiten=$6 WHERE personal_nummer = $1 AND woche_start = $2;`) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Println("Error preparing SQL statement", err) | 			slog.Warn("Error preparing SQL statement", "error", err) | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		qStr, err = DB.Prepare(`INSERT INTO wochen_report (personal_nummer, woche_start, arbeitszeit, ueberstunden) VALUES ($1, $2, make_interval(secs => $3::numeric / 1000000000), make_interval(secs => $4::numeric / 1000000000));`) | 		qStr, err = DB.Prepare(`INSERT INTO wochen_report (personal_nummer, woche_start, arbeitszeit, ueberstunden, anwesenheiten, abwesenheiten) VALUES ($1, $2, make_interval(secs => $3::numeric / 1000000000), make_interval(secs => $4::numeric / 1000000000), $5, $6);`) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Println("Error preparing SQL statement", err) | 			slog.Warn("Error preparing SQL statement", "error", err) | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	_, err = qStr.Exec(w.User.PersonalNummer, w.WeekStart, int64(w.Worktime), int64(w.Overtime)) |  | ||||||
|  | 	_, err = qStr.Exec(w.User.PersonalNummer, w.WeekStart, int64(w.Worktime), int64(w.Overtime), pq.Array(anwBookings), pq.Array(awBookings)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Println("Error executing query!", err) | 		log.Println("Error executing query!", err) | ||||||
| 		return err | 		return err | ||||||
| @@ -196,8 +236,8 @@ func (w *WorkWeek) Accept() error { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (w *WorkWeek) RequiresAction() bool { | func (w *WorkWeek) RequiresAction() bool { | ||||||
| 	var requiresAction bool = true | 	var requiresAction bool = false | ||||||
| 	for _, day := range w.WorkDays { | 	for _, day := range w.Days { | ||||||
| 		requiresAction = requiresAction || day.RequiresAction() | 		requiresAction = requiresAction || day.RequiresAction() | ||||||
| 	} | 	} | ||||||
| 	return requiresAction | 	return requiresAction | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ import ( | |||||||
|  |  | ||||||
| func SetupWorkWeekFixture(t *testing.T) models.WorkWeek { | func SetupWorkWeekFixture(t *testing.T) models.WorkWeek { | ||||||
| 	t.Helper() | 	t.Helper() | ||||||
| 	monday, err := time.Parse("2006-01-02", "2025-01-10") | 	monday, err := time.Parse(time.DateOnly, "2025-01-10") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Fatal(err) | 		t.Fatal(err) | ||||||
| 	} | 	} | ||||||
| @@ -16,7 +16,7 @@ func SetupWorkWeekFixture(t *testing.T) models.WorkWeek { | |||||||
| } | } | ||||||
|  |  | ||||||
| func TestNewWorkWeekNoPopulate(t *testing.T) { | func TestNewWorkWeekNoPopulate(t *testing.T) { | ||||||
| 	monday, err := time.Parse("2006-01-02", "2025-01-10") | 	monday, err := time.Parse(time.DateOnly, "2025-01-10") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Fatal(err) | 		t.Fatal(err) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| sonar.projectKey=Arbeitszeitmessung | sonar.projectKey=arbeitszeitmessung | ||||||
| sonar.sources=. | sonar.sources=. | ||||||
| sonar.exclusions=**/*_test.go | sonar.exclusions=**/*_test.go, **/*_templ.go | ||||||
|  |  | ||||||
| sonar.tests=. | sonar.tests=. | ||||||
| sonar.test.inclusions=**/*_test.go | sonar.test.inclusions=**/*_test.go | ||||||
|   | |||||||
| @@ -34,13 +34,14 @@ | |||||||
|   body { |   body { | ||||||
|     -webkit-print-color-adjust: exact !important; |     -webkit-print-color-adjust: exact !important; | ||||||
|     print-color-adjust: exact !important; |     print-color-adjust: exact !important; | ||||||
|  |     background-color: white; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @layer components { | @layer components { | ||||||
|   .grid-main { |   .grid-main { | ||||||
|     display: grid; |     display: grid; | ||||||
|     grid-template-columns: 2fr auto 1fr; |     grid-template-columns: 4fr 3fr 3fr 1fr; | ||||||
|     align-items: stretch; |     align-items: stretch; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -88,6 +89,11 @@ | |||||||
|     transition-duration: var(--tw-duration, var(--default-transition-duration)); |     transition-duration: var(--tw-duration, var(--default-transition-duration)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   input.btn, | ||||||
|  |   select.btn { | ||||||
|  |     transition-duration: 300ms; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   .btn:hover { |   .btn:hover { | ||||||
|     color: var(--color-white); |     color: var(--color-white); | ||||||
|     background-color: var(--color-neutral-700); |     background-color: var(--color-neutral-700); | ||||||
| @@ -98,9 +104,46 @@ | |||||||
|     pointer-events: none; |     pointer-events: none; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .btn:active, |   input.btn, | ||||||
|   .btn:focus { |   select.btn { | ||||||
|     background-color: var(--color-neutral-700); |     text-align: left; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   input.btn:hover, | ||||||
|  |   select.btn:hover { | ||||||
|  |     border-color: var(--color-neutral-300); | ||||||
|  |     background-color: var(--color-neutral-100); | ||||||
|  |     color: var(--color-neutral-800); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .edit-box { | ||||||
|  |     border-radius: var(--radius-md); | ||||||
|  |     overflow: hidden; | ||||||
|  |     border-color: var(--color-neutral-500); | ||||||
|  |     transition-property: background-color, border-color; | ||||||
|  |     transition-timing-function: var(--default-transition-timing-function) * 2; | ||||||
|  |     transition-duration: var(--default-transition-duration); | ||||||
|  |     outline: none; | ||||||
|  |  | ||||||
|  |     &:is(:where(.group):is(.edit) *) { | ||||||
|  |       border-width: 1px; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .edit-box:hover { | ||||||
|  |     &:is(:where(.group):is(.edit) *) { | ||||||
|  |       background-color: var(--color-white); | ||||||
|  |       border-color: var(--color-neutral-300); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .edit-box input:focus { | ||||||
|  |     outline: none; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   div.edit { | ||||||
|  |     border-width: 1px; | ||||||
|  |     background-color: var(--color-neutral-300); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @media (width >=48rem) { |   @media (width >=48rem) { | ||||||
|   | |||||||
| @@ -9,8 +9,13 @@ | |||||||
|       "Courier New", monospace; |       "Courier New", monospace; | ||||||
|     --color-red-500: oklch(63.7% 0.237 25.331); |     --color-red-500: oklch(63.7% 0.237 25.331); | ||||||
|     --color-red-600: oklch(57.7% 0.245 27.325); |     --color-red-600: oklch(57.7% 0.245 27.325); | ||||||
|  |     --color-red-700: oklch(50.5% 0.213 27.518); | ||||||
|     --color-orange-500: oklch(70.5% 0.213 47.604); |     --color-orange-500: oklch(70.5% 0.213 47.604); | ||||||
|     --color-purple-600: oklch(55.8% 0.288 302.321); |     --color-purple-600: oklch(55.8% 0.288 302.321); | ||||||
|  |     --color-slate-300: oklch(86.9% 0.022 252.894); | ||||||
|  |     --color-slate-600: oklch(44.6% 0.043 257.281); | ||||||
|  |     --color-slate-700: oklch(37.2% 0.044 257.287); | ||||||
|  |     --color-slate-800: oklch(27.9% 0.041 260.031); | ||||||
|     --color-neutral-100: oklch(97% 0 0); |     --color-neutral-100: oklch(97% 0 0); | ||||||
|     --color-neutral-200: oklch(92.2% 0 0); |     --color-neutral-200: oklch(92.2% 0 0); | ||||||
|     --color-neutral-300: oklch(87% 0 0); |     --color-neutral-300: oklch(87% 0 0); | ||||||
| @@ -19,7 +24,6 @@ | |||||||
|     --color-neutral-600: oklch(43.9% 0 0); |     --color-neutral-600: oklch(43.9% 0 0); | ||||||
|     --color-neutral-700: oklch(37.1% 0 0); |     --color-neutral-700: oklch(37.1% 0 0); | ||||||
|     --color-neutral-800: oklch(26.9% 0 0); |     --color-neutral-800: oklch(26.9% 0 0); | ||||||
|     --color-neutral-900: oklch(20.5% 0 0); |  | ||||||
|     --color-black: #000; |     --color-black: #000; | ||||||
|     --color-white: #fff; |     --color-white: #fff; | ||||||
|     --spacing: 0.25rem; |     --spacing: 0.25rem; | ||||||
| @@ -190,9 +194,42 @@ | |||||||
|   .\@container { |   .\@container { | ||||||
|     container-type: inline-size; |     container-type: inline-size; | ||||||
|   } |   } | ||||||
|  |   .absolute { | ||||||
|  |     position: absolute; | ||||||
|  |   } | ||||||
|   .relative { |   .relative { | ||||||
|     position: relative; |     position: relative; | ||||||
|   } |   } | ||||||
|  |   .top-1 { | ||||||
|  |     top: calc(var(--spacing) * 1); | ||||||
|  |   } | ||||||
|  |   .top-1\/2 { | ||||||
|  |     top: calc(1/2 * 100%); | ||||||
|  |   } | ||||||
|  |   .top-2 { | ||||||
|  |     top: calc(var(--spacing) * 2); | ||||||
|  |   } | ||||||
|  |   .top-2\.5 { | ||||||
|  |     top: calc(var(--spacing) * 2.5); | ||||||
|  |   } | ||||||
|  |   .top-\[0\.125rem\] { | ||||||
|  |     top: 0.125rem; | ||||||
|  |   } | ||||||
|  |   .right-1 { | ||||||
|  |     right: calc(var(--spacing) * 1); | ||||||
|  |   } | ||||||
|  |   .right-2 { | ||||||
|  |     right: calc(var(--spacing) * 2); | ||||||
|  |   } | ||||||
|  |   .right-2\.5 { | ||||||
|  |     right: calc(var(--spacing) * 2.5); | ||||||
|  |   } | ||||||
|  |   .left-1 { | ||||||
|  |     left: calc(var(--spacing) * 1); | ||||||
|  |   } | ||||||
|  |   .left-1\/2 { | ||||||
|  |     left: calc(1/2 * 100%); | ||||||
|  |   } | ||||||
|   .col-span-2 { |   .col-span-2 { | ||||||
|     grid-column: span 2 / span 2; |     grid-column: span 2 / span 2; | ||||||
|   } |   } | ||||||
| @@ -211,10 +248,19 @@ | |||||||
|   .mt-1 { |   .mt-1 { | ||||||
|     margin-top: calc(var(--spacing) * 1); |     margin-top: calc(var(--spacing) * 1); | ||||||
|   } |   } | ||||||
|  |   .mb-1 { | ||||||
|  |     margin-bottom: calc(var(--spacing) * 1); | ||||||
|  |   } | ||||||
|   .mb-2 { |   .mb-2 { | ||||||
|     margin-bottom: calc(var(--spacing) * 2); |     margin-bottom: calc(var(--spacing) * 2); | ||||||
|   } |   } | ||||||
|   .icon-\[material-symbols-light--add-circle-outline\] { |   .ml-1 { | ||||||
|  |     margin-left: calc(var(--spacing) * 1); | ||||||
|  |   } | ||||||
|  |   .ml-2 { | ||||||
|  |     margin-left: calc(var(--spacing) * 2); | ||||||
|  |   } | ||||||
|  |   .icon-\[material-symbols-light--cancel-outline\] { | ||||||
|     display: inline-block; |     display: inline-block; | ||||||
|     width: 1.25em; |     width: 1.25em; | ||||||
|     height: 1.25em; |     height: 1.25em; | ||||||
| @@ -225,7 +271,7 @@ | |||||||
|     mask-repeat: no-repeat; |     mask-repeat: no-repeat; | ||||||
|     -webkit-mask-size: 100% 100%; |     -webkit-mask-size: 100% 100%; | ||||||
|     mask-size: 100% 100%; |     mask-size: 100% 100%; | ||||||
|     --svg: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24' height='24'%3E%3Cpath fill='black' d='M11.5 16.5h1v-4h4v-1h-4v-4h-1v4h-4v1h4zm.503 4.5q-1.867 0-3.51-.708q-1.643-.709-2.859-1.924t-1.925-2.856T3 12.003t.709-3.51Q4.417 6.85 5.63 5.634t2.857-1.925T11.997 3t3.51.709q1.643.708 2.859 1.922t1.925 2.857t.709 3.509t-.708 3.51t-1.924 2.859t-2.856 1.925t-3.509.709M12 20q3.35 0 5.675-2.325T20 12t-2.325-5.675T12 4T6.325 6.325T4 12t2.325 5.675T12 20m0-8'/%3E%3C/svg%3E"); |     --svg: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24' height='24'%3E%3Cpath fill='black' d='m8.4 16.308l3.6-3.6l3.6 3.6l.708-.708l-3.6-3.6l3.6-3.6l-.708-.708l-3.6 3.6l-3.6-3.6l-.708.708l3.6 3.6l-3.6 3.6zM12.003 21q-1.866 0-3.51-.708q-1.643-.709-2.859-1.924t-1.925-2.856T3 12.003t.709-3.51Q4.417 6.85 5.63 5.634t2.857-1.925T11.997 3t3.51.709q1.643.708 2.859 1.922t1.925 2.857t.709 3.509t-.708 3.51t-1.924 2.859t-2.856 1.925t-3.509.709M12 20q3.35 0 5.675-2.325T20 12t-2.325-5.675T12 4T6.325 6.325T4 12t2.325 5.675T12 20m0-8'/%3E%3C/svg%3E"); | ||||||
|   } |   } | ||||||
|   .icon-\[material-symbols-light--check-circle-outline\] { |   .icon-\[material-symbols-light--check-circle-outline\] { | ||||||
|     display: inline-block; |     display: inline-block; | ||||||
| @@ -253,6 +299,19 @@ | |||||||
|     mask-size: 100% 100%; |     mask-size: 100% 100%; | ||||||
|     --svg: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24' height='24'%3E%3Cpath fill='black' d='M12.003 21q-1.866 0-3.51-.708q-1.643-.709-2.859-1.924t-1.925-2.856T3 12.003t.709-3.51Q4.417 6.85 5.63 5.634t2.857-1.925T11.997 3t3.51.709q1.643.708 2.859 1.922t1.925 2.857t.709 3.509t-.708 3.51t-1.924 2.859t-2.856 1.925t-3.509.709M12 20q3.35 0 5.675-2.325T20 12t-2.325-5.675T12 4T6.325 6.325T4 12t2.325 5.675T12 20m0-8'/%3E%3C/svg%3E"); |     --svg: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24' height='24'%3E%3Cpath fill='black' d='M12.003 21q-1.866 0-3.51-.708q-1.643-.709-2.859-1.924t-1.925-2.856T3 12.003t.709-3.51Q4.417 6.85 5.63 5.634t2.857-1.925T11.997 3t3.51.709q1.643.708 2.859 1.922t1.925 2.857t.709 3.509t-.708 3.51t-1.924 2.859t-2.856 1.925t-3.509.709M12 20q3.35 0 5.675-2.325T20 12t-2.325-5.675T12 4T6.325 6.325T4 12t2.325 5.675T12 20m0-8'/%3E%3C/svg%3E"); | ||||||
|   } |   } | ||||||
|  |   .icon-\[material-symbols-light--delete-outline\] { | ||||||
|  |     display: inline-block; | ||||||
|  |     width: 1.25em; | ||||||
|  |     height: 1.25em; | ||||||
|  |     background-color: currentColor; | ||||||
|  |     -webkit-mask-image: var(--svg); | ||||||
|  |     mask-image: var(--svg); | ||||||
|  |     -webkit-mask-repeat: no-repeat; | ||||||
|  |     mask-repeat: no-repeat; | ||||||
|  |     -webkit-mask-size: 100% 100%; | ||||||
|  |     mask-size: 100% 100%; | ||||||
|  |     --svg: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24' height='24'%3E%3Cpath fill='black' d='M7.616 20q-.672 0-1.144-.472T6 18.385V6H5V5h4v-.77h6V5h4v1h-1v12.385q0 .69-.462 1.153T16.384 20zM17 6H7v12.385q0 .269.173.442t.443.173h8.769q.23 0 .423-.192t.192-.424zM9.808 17h1V8h-1zm3.384 0h1V8h-1zM7 6v13z'/%3E%3C/svg%3E"); | ||||||
|  |   } | ||||||
|   .icon-\[material-symbols-light--more-time\] { |   .icon-\[material-symbols-light--more-time\] { | ||||||
|     display: inline-block; |     display: inline-block; | ||||||
|     width: 1.25em; |     width: 1.25em; | ||||||
| @@ -279,7 +338,7 @@ | |||||||
|     mask-size: 100% 100%; |     mask-size: 100% 100%; | ||||||
|     --svg: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24' height='24'%3E%3Cpath fill='black' d='M9.808 14.616h1V9.385h-1zm3.384 0h1V9.385h-1zM12.003 21q-1.866 0-3.51-.705q-1.643-.706-2.859-1.915t-1.925-2.843T3 12.039q0-.905.167-1.778t.497-1.713l.78.78q-.219.65-.331 1.32T4 12q0 3.35 2.325 5.675T12 20t5.675-2.325T20 12t-2.325-5.675T12 4q-.675 0-1.332.112t-1.3.332l-.776-.775q.789-.315 1.606-.492T11.885 3q1.887 0 3.546.701t2.894 1.926t1.955 2.866t.72 3.505t-.708 3.509t-1.924 2.859t-2.856 1.925t-3.509.709M5.923 6.808q-.356 0-.62-.265q-.264-.264-.264-.62t.264-.62t.62-.264t.62.264t.265.62t-.265.62t-.62.265M12 12'/%3E%3C/svg%3E"); |     --svg: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24' height='24'%3E%3Cpath fill='black' d='M9.808 14.616h1V9.385h-1zm3.384 0h1V9.385h-1zM12.003 21q-1.866 0-3.51-.705q-1.643-.706-2.859-1.915t-1.925-2.843T3 12.039q0-.905.167-1.778t.497-1.713l.78.78q-.219.65-.331 1.32T4 12q0 3.35 2.325 5.675T12 20t5.675-2.325T20 12t-2.325-5.675T12 4q-.675 0-1.332.112t-1.3.332l-.776-.775q.789-.315 1.606-.492T11.885 3q1.887 0 3.546.701t2.894 1.926t1.955 2.866t.72 3.505t-.708 3.509t-1.924 2.859t-2.856 1.925t-3.509.709M5.923 6.808q-.356 0-.62-.265q-.264-.264-.264-.62t.264-.62t.62-.264t.62.264t.265.62t-.265.62t-.62.265M12 12'/%3E%3C/svg%3E"); | ||||||
|   } |   } | ||||||
|   .icon-\[material-symbols-light--nest-clock-farsight-analog-outline\] { |   .icon-\[material-symbols-light--schedule-outline\] { | ||||||
|     display: inline-block; |     display: inline-block; | ||||||
|     width: 1.25em; |     width: 1.25em; | ||||||
|     height: 1.25em; |     height: 1.25em; | ||||||
| @@ -290,7 +349,10 @@ | |||||||
|     mask-repeat: no-repeat; |     mask-repeat: no-repeat; | ||||||
|     -webkit-mask-size: 100% 100%; |     -webkit-mask-size: 100% 100%; | ||||||
|     mask-size: 100% 100%; |     mask-size: 100% 100%; | ||||||
|     --svg: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24' height='24'%3E%3Cpath fill='black' d='M14.935 16.223L11.5 12.789V7.923h1v4.464l3.123 3.123zM11.5 6V4h1v2zm6.5 6.5v-1h2v1zM11.5 20v-2h1v2zM4 12.5v-1h2v1zm8.003 8.5q-1.867 0-3.51-.708q-1.643-.709-2.859-1.924t-1.925-2.856T3 12.003t.709-3.51Q4.417 6.85 5.63 5.634t2.857-1.925T11.997 3t3.51.709q1.643.708 2.859 1.922t1.925 2.857t.709 3.509t-.708 3.51t-1.924 2.859t-2.856 1.925t-3.509.709M12 20q3.35 0 5.675-2.325T20 12t-2.325-5.675T12 4T6.325 6.325T4 12t2.325 5.675T12 20m0-8'/%3E%3C/svg%3E"); |     --svg: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24' height='24'%3E%3Cpath fill='black' d='m15.646 16.354l.708-.708l-3.854-3.854V7h-1v5.208zM12.003 21q-1.866 0-3.51-.708q-1.643-.709-2.859-1.924t-1.925-2.856T3 12.003t.709-3.51Q4.417 6.85 5.63 5.634t2.857-1.925T11.997 3t3.51.709q1.643.708 2.859 1.922t1.925 2.857t.709 3.509t-.708 3.51t-1.924 2.859t-2.856 1.925t-3.509.709M12 20q3.325 0 5.663-2.337T20 12t-2.337-5.663T12 4T6.337 6.338T4 12t2.338 5.663T12 20'/%3E%3C/svg%3E"); | ||||||
|  |   } | ||||||
|  |   .block { | ||||||
|  |     display: block; | ||||||
|   } |   } | ||||||
|   .flex { |   .flex { | ||||||
|     display: flex; |     display: flex; | ||||||
| @@ -304,6 +366,9 @@ | |||||||
|   .inline { |   .inline { | ||||||
|     display: inline; |     display: inline; | ||||||
|   } |   } | ||||||
|  |   .inline-flex { | ||||||
|  |     display: inline-flex; | ||||||
|  |   } | ||||||
|   .table { |   .table { | ||||||
|     display: table; |     display: table; | ||||||
|   } |   } | ||||||
| @@ -315,12 +380,25 @@ | |||||||
|     width: calc(var(--spacing) * 4); |     width: calc(var(--spacing) * 4); | ||||||
|     height: calc(var(--spacing) * 4); |     height: calc(var(--spacing) * 4); | ||||||
|   } |   } | ||||||
|  |   .size-5 { | ||||||
|  |     width: calc(var(--spacing) * 5); | ||||||
|  |     height: calc(var(--spacing) * 5); | ||||||
|  |   } | ||||||
|   .h-2 { |   .h-2 { | ||||||
|     height: calc(var(--spacing) * 2); |     height: calc(var(--spacing) * 2); | ||||||
|   } |   } | ||||||
|  |   .h-3 { | ||||||
|  |     height: calc(var(--spacing) * 3); | ||||||
|  |   } | ||||||
|  |   .h-3\.5 { | ||||||
|  |     height: calc(var(--spacing) * 3.5); | ||||||
|  |   } | ||||||
|   .h-4 { |   .h-4 { | ||||||
|     height: calc(var(--spacing) * 4); |     height: calc(var(--spacing) * 4); | ||||||
|   } |   } | ||||||
|  |   .h-5 { | ||||||
|  |     height: calc(var(--spacing) * 5); | ||||||
|  |   } | ||||||
|   .h-8 { |   .h-8 { | ||||||
|     height: calc(var(--spacing) * 8); |     height: calc(var(--spacing) * 8); | ||||||
|   } |   } | ||||||
| @@ -330,15 +408,24 @@ | |||||||
|   .h-full { |   .h-full { | ||||||
|     height: 100%; |     height: 100%; | ||||||
|   } |   } | ||||||
|   .w-1\/7 { |  | ||||||
|     width: calc(1/7 * 100%); |  | ||||||
|   } |  | ||||||
|   .w-2 { |   .w-2 { | ||||||
|     width: calc(var(--spacing) * 2); |     width: calc(var(--spacing) * 2); | ||||||
|   } |   } | ||||||
|  |   .w-3 { | ||||||
|  |     width: calc(var(--spacing) * 3); | ||||||
|  |   } | ||||||
|  |   .w-3\.5 { | ||||||
|  |     width: calc(var(--spacing) * 3.5); | ||||||
|  |   } | ||||||
|   .w-4 { |   .w-4 { | ||||||
|     width: calc(var(--spacing) * 4); |     width: calc(var(--spacing) * 4); | ||||||
|   } |   } | ||||||
|  |   .w-5 { | ||||||
|  |     width: calc(var(--spacing) * 5); | ||||||
|  |   } | ||||||
|  |   .w-9 { | ||||||
|  |     width: calc(var(--spacing) * 9); | ||||||
|  |   } | ||||||
|   .w-9\/10 { |   .w-9\/10 { | ||||||
|     width: calc(9/10 * 100%); |     width: calc(9/10 * 100%); | ||||||
|   } |   } | ||||||
| @@ -351,24 +438,58 @@ | |||||||
|   .w-full { |   .w-full { | ||||||
|     width: 100%; |     width: 100%; | ||||||
|   } |   } | ||||||
|  |   .flex-shrink { | ||||||
|  |     flex-shrink: 1; | ||||||
|  |   } | ||||||
|   .flex-shrink-0 { |   .flex-shrink-0 { | ||||||
|     flex-shrink: 0; |     flex-shrink: 0; | ||||||
|   } |   } | ||||||
|   .flex-grow { |   .flex-grow { | ||||||
|     flex-grow: 1; |     flex-grow: 1; | ||||||
|   } |   } | ||||||
|   .grow { |   .grow-0 { | ||||||
|     flex-grow: 1; |     flex-grow: 0; | ||||||
|   } |   } | ||||||
|   .grow-1 { |   .grow-1 { | ||||||
|     flex-grow: 1; |     flex-grow: 1; | ||||||
|   } |   } | ||||||
|  |   .basis-\[content\] { | ||||||
|  |     flex-basis: content; | ||||||
|  |   } | ||||||
|   .border-collapse { |   .border-collapse { | ||||||
|     border-collapse: collapse; |     border-collapse: collapse; | ||||||
|   } |   } | ||||||
|  |   .-translate-x-1 { | ||||||
|  |     --tw-translate-x: calc(var(--spacing) * -1); | ||||||
|  |     translate: var(--tw-translate-x) var(--tw-translate-y); | ||||||
|  |   } | ||||||
|  |   .-translate-x-1\/2 { | ||||||
|  |     --tw-translate-x: calc(calc(1/2 * 100%) * -1); | ||||||
|  |     translate: var(--tw-translate-x) var(--tw-translate-y); | ||||||
|  |   } | ||||||
|  |   .-translate-y-1 { | ||||||
|  |     --tw-translate-y: calc(var(--spacing) * -1); | ||||||
|  |     translate: var(--tw-translate-x) var(--tw-translate-y); | ||||||
|  |   } | ||||||
|  |   .-translate-y-1\/2 { | ||||||
|  |     --tw-translate-y: calc(calc(1/2 * 100%) * -1); | ||||||
|  |     translate: var(--tw-translate-x) var(--tw-translate-y); | ||||||
|  |   } | ||||||
|  |   .transform { | ||||||
|  |     transform: var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,); | ||||||
|  |   } | ||||||
|   .cursor-pointer { |   .cursor-pointer { | ||||||
|     cursor: pointer; |     cursor: pointer; | ||||||
|   } |   } | ||||||
|  |   .resize { | ||||||
|  |     resize: both; | ||||||
|  |   } | ||||||
|  |   .scroll-m-2 { | ||||||
|  |     scroll-margin: calc(var(--spacing) * 2); | ||||||
|  |   } | ||||||
|  |   .appearance-none { | ||||||
|  |     appearance: none; | ||||||
|  |   } | ||||||
|   .break-after-page { |   .break-after-page { | ||||||
|     break-after: page; |     break-after: page; | ||||||
|   } |   } | ||||||
| @@ -405,6 +526,9 @@ | |||||||
|   .items-center { |   .items-center { | ||||||
|     align-items: center; |     align-items: center; | ||||||
|   } |   } | ||||||
|  |   .items-end { | ||||||
|  |     align-items: flex-end; | ||||||
|  |   } | ||||||
|   .justify-around { |   .justify-around { | ||||||
|     justify-content: space-around; |     justify-content: space-around; | ||||||
|   } |   } | ||||||
| @@ -448,23 +572,25 @@ | |||||||
|   .overflow-hidden { |   .overflow-hidden { | ||||||
|     overflow: hidden; |     overflow: hidden; | ||||||
|   } |   } | ||||||
|  |   .rounded { | ||||||
|  |     border-radius: 0.25rem; | ||||||
|  |   } | ||||||
|   .rounded-full { |   .rounded-full { | ||||||
|     border-radius: calc(infinity * 1px); |     border-radius: calc(infinity * 1px); | ||||||
|   } |   } | ||||||
|   .rounded-md { |   .rounded-md { | ||||||
|     border-radius: var(--radius-md); |     border-radius: var(--radius-md); | ||||||
|   } |   } | ||||||
|  |   .rounded-none { | ||||||
|  |     border-radius: 0; | ||||||
|  |   } | ||||||
|   .border { |   .border { | ||||||
|     border-style: var(--tw-border-style); |     border-style: var(--tw-border-style); | ||||||
|     border-width: 1px; |     border-width: 1px; | ||||||
|   } |   } | ||||||
|   .border-1 { |   .border-0 { | ||||||
|     border-style: var(--tw-border-style); |     border-style: var(--tw-border-style); | ||||||
|     border-width: 1px; |     border-width: 0px; | ||||||
|   } |  | ||||||
|   .border-t-1 { |  | ||||||
|     border-top-style: var(--tw-border-style); |  | ||||||
|     border-top-width: 1px; |  | ||||||
|   } |   } | ||||||
|   .border-r-0 { |   .border-r-0 { | ||||||
|     border-right-style: var(--tw-border-style); |     border-right-style: var(--tw-border-style); | ||||||
| @@ -478,17 +604,27 @@ | |||||||
|     border-bottom-style: var(--tw-border-style); |     border-bottom-style: var(--tw-border-style); | ||||||
|     border-bottom-width: 0px; |     border-bottom-width: 0px; | ||||||
|   } |   } | ||||||
|   .border-neutral-200 { |   .border-dashed { | ||||||
|     border-color: var(--color-neutral-200); |     --tw-border-style: dashed; | ||||||
|  |     border-style: dashed; | ||||||
|   } |   } | ||||||
|   .border-neutral-300 { |   .border-neutral-300 { | ||||||
|     border-color: var(--color-neutral-300); |     border-color: var(--color-neutral-300); | ||||||
|   } |   } | ||||||
|  |   .border-neutral-500 { | ||||||
|  |     border-color: var(--color-neutral-500); | ||||||
|  |   } | ||||||
|   .border-neutral-600 { |   .border-neutral-600 { | ||||||
|     border-color: var(--color-neutral-600); |     border-color: var(--color-neutral-600); | ||||||
|   } |   } | ||||||
|   .border-neutral-900 { |   .border-slate-300 { | ||||||
|     border-color: var(--color-neutral-900); |     border-color: var(--color-slate-300); | ||||||
|  |   } | ||||||
|  |   .border-slate-700 { | ||||||
|  |     border-color: var(--color-slate-700); | ||||||
|  |   } | ||||||
|  |   .border-slate-800 { | ||||||
|  |     border-color: var(--color-slate-800); | ||||||
|   } |   } | ||||||
|   .bg-accent { |   .bg-accent { | ||||||
|     background-color: var(--color-accent); |     background-color: var(--color-accent); | ||||||
| @@ -511,6 +647,9 @@ | |||||||
|   .bg-red-600 { |   .bg-red-600 { | ||||||
|     background-color: var(--color-red-600); |     background-color: var(--color-red-600); | ||||||
|   } |   } | ||||||
|  |   .mask-repeat { | ||||||
|  |     mask-repeat: repeat; | ||||||
|  |   } | ||||||
|   .p-1 { |   .p-1 { | ||||||
|     padding: calc(var(--spacing) * 1); |     padding: calc(var(--spacing) * 1); | ||||||
|   } |   } | ||||||
| @@ -554,6 +693,9 @@ | |||||||
|   .text-accent { |   .text-accent { | ||||||
|     color: var(--color-accent); |     color: var(--color-accent); | ||||||
|   } |   } | ||||||
|  |   .text-black { | ||||||
|  |     color: var(--color-black); | ||||||
|  |   } | ||||||
|   .text-neutral-300 { |   .text-neutral-300 { | ||||||
|     color: var(--color-neutral-300); |     color: var(--color-neutral-300); | ||||||
|   } |   } | ||||||
| @@ -572,9 +714,28 @@ | |||||||
|   .text-red-600 { |   .text-red-600 { | ||||||
|     color: var(--color-red-600); |     color: var(--color-red-600); | ||||||
|   } |   } | ||||||
|  |   .text-slate-600 { | ||||||
|  |     color: var(--color-slate-600); | ||||||
|  |   } | ||||||
|  |   .text-slate-700 { | ||||||
|  |     color: var(--color-slate-700); | ||||||
|  |   } | ||||||
|  |   .text-white { | ||||||
|  |     color: var(--color-white); | ||||||
|  |   } | ||||||
|   .uppercase { |   .uppercase { | ||||||
|     text-transform: uppercase; |     text-transform: uppercase; | ||||||
|   } |   } | ||||||
|  |   .underline { | ||||||
|  |     text-decoration-line: underline; | ||||||
|  |   } | ||||||
|  |   .opacity-0 { | ||||||
|  |     opacity: 0%; | ||||||
|  |   } | ||||||
|  |   .outline { | ||||||
|  |     outline-style: var(--tw-outline-style); | ||||||
|  |     outline-width: 1px; | ||||||
|  |   } | ||||||
|   .filter { |   .filter { | ||||||
|     filter: var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,); |     filter: var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,); | ||||||
|   } |   } | ||||||
| @@ -583,6 +744,11 @@ | |||||||
|     transition-timing-function: var(--tw-ease, var(--default-transition-timing-function)); |     transition-timing-function: var(--tw-ease, var(--default-transition-timing-function)); | ||||||
|     transition-duration: var(--tw-duration, var(--default-transition-duration)); |     transition-duration: var(--tw-duration, var(--default-transition-duration)); | ||||||
|   } |   } | ||||||
|  |   .transition-all { | ||||||
|  |     transition-property: all; | ||||||
|  |     transition-timing-function: var(--tw-ease, var(--default-transition-timing-function)); | ||||||
|  |     transition-duration: var(--tw-duration, var(--default-transition-duration)); | ||||||
|  |   } | ||||||
|   .transition-colors { |   .transition-colors { | ||||||
|     transition-property: color, background-color, border-color, outline-color, text-decoration-color, fill, stroke, --tw-gradient-from, --tw-gradient-via, --tw-gradient-to; |     transition-property: color, background-color, border-color, outline-color, text-decoration-color, fill, stroke, --tw-gradient-from, --tw-gradient-via, --tw-gradient-to; | ||||||
|     transition-timing-function: var(--tw-ease, var(--default-transition-timing-function)); |     transition-timing-function: var(--tw-ease, var(--default-transition-timing-function)); | ||||||
| @@ -592,15 +758,9 @@ | |||||||
|     --tw-duration: 300ms; |     --tw-duration: 300ms; | ||||||
|     transition-duration: 300ms; |     transition-duration: 300ms; | ||||||
|   } |   } | ||||||
|   .\*\:\*\:\*\:border-1 { |   .select-none { | ||||||
|     :is(& > *) { |     -webkit-user-select: none; | ||||||
|       :is(& > *) { |     user-select: none; | ||||||
|         :is(& > *) { |  | ||||||
|           border-style: var(--tw-border-style); |  | ||||||
|           border-width: 1px; |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|   .\*\:text-center { |   .\*\:text-center { | ||||||
|     :is(& > *) { |     :is(& > *) { | ||||||
| @@ -628,9 +788,9 @@ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .group-\[\.edit\]\:block { |   .group-\[\.edit\]\:ml-2 { | ||||||
|     &:is(:where(.group):is(.edit) *) { |     &:is(:where(.group):is(.edit) *) { | ||||||
|       display: block; |       margin-left: calc(var(--spacing) * 2); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .group-\[\.edit\]\:flex { |   .group-\[\.edit\]\:flex { | ||||||
| @@ -648,16 +808,34 @@ | |||||||
|       display: inline; |       display: inline; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |   .group-\[\.edit\]\/button\:block { | ||||||
|  |     &:is(:where(.group\/button):is(.edit) *) { | ||||||
|  |       display: block; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   .group-\[\.edit\]\/button\:hidden { | ||||||
|  |     &:is(:where(.group\/button):is(.edit) *) { | ||||||
|  |       display: none; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   .peer-checked\:opacity-100 { | ||||||
|  |     &:is(:where(.peer):checked ~ *) { | ||||||
|  |       opacity: 100%; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|   .placeholder\:text-neutral-400 { |   .placeholder\:text-neutral-400 { | ||||||
|     &::placeholder { |     &::placeholder { | ||||||
|       color: var(--color-neutral-400); |       color: var(--color-neutral-400); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .hover\:border-neutral-300 { |   .checked\:border-slate-800 { | ||||||
|     &:hover { |     &:checked { | ||||||
|       @media (hover: hover) { |       border-color: var(--color-slate-800); | ||||||
|         border-color: var(--color-neutral-300); |     } | ||||||
|       } |   } | ||||||
|  |   .checked\:bg-slate-800 { | ||||||
|  |     &:checked { | ||||||
|  |       background-color: var(--color-slate-800); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .hover\:border-neutral-500 { |   .hover\:border-neutral-500 { | ||||||
| @@ -681,17 +859,10 @@ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .hover\:bg-red-600 { |   .hover\:bg-red-700 { | ||||||
|     &:hover { |     &:hover { | ||||||
|       @media (hover: hover) { |       @media (hover: hover) { | ||||||
|         background-color: var(--color-red-600); |         background-color: var(--color-red-700); | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   .hover\:text-accent { |  | ||||||
|     &:hover { |  | ||||||
|       @media (hover: hover) { |  | ||||||
|         color: var(--color-accent); |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -702,11 +873,6 @@ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .focus\:border-neutral-400 { |  | ||||||
|     &:focus { |  | ||||||
|       border-color: var(--color-neutral-400); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   .focus\:bg-neutral-700 { |   .focus\:bg-neutral-700 { | ||||||
|     &:focus { |     &:focus { | ||||||
|       background-color: var(--color-neutral-700); |       background-color: var(--color-neutral-700); | ||||||
| @@ -733,11 +899,6 @@ | |||||||
|       opacity: 50%; |       opacity: 50%; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .max-md\:flex { |  | ||||||
|     @media (width < 48rem) { |  | ||||||
|       display: flex; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   .max-md\:grid { |   .max-md\:grid { | ||||||
|     @media (width < 48rem) { |     @media (width < 48rem) { | ||||||
|       display: grid; |       display: grid; | ||||||
| @@ -748,11 +909,6 @@ | |||||||
|       display: none; |       display: none; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .max-md\:flex-col { |  | ||||||
|     @media (width < 48rem) { |  | ||||||
|       flex-direction: column; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   .max-md\:divide-y-1 { |   .max-md\:divide-y-1 { | ||||||
|     @media (width < 48rem) { |     @media (width < 48rem) { | ||||||
|       :where(& > :not(:last-child)) { |       :where(& > :not(:last-child)) { | ||||||
| @@ -764,12 +920,6 @@ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .max-md\:border-b-1 { |  | ||||||
|     @media (width < 48rem) { |  | ||||||
|       border-bottom-style: var(--tw-border-style); |  | ||||||
|       border-bottom-width: 1px; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   .max-md\:bg-neutral-300 { |   .max-md\:bg-neutral-300 { | ||||||
|     @media (width < 48rem) { |     @media (width < 48rem) { | ||||||
|       background-color: var(--color-neutral-300); |       background-color: var(--color-neutral-300); | ||||||
| @@ -785,11 +935,6 @@ | |||||||
|       grid-column: span 3 / span 3; |       grid-column: span 3 / span 3; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .md\:col-span-4 { |  | ||||||
|     @media (width >= 48rem) { |  | ||||||
|       grid-column: span 4 / span 4; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   .md\:mx-\[10\%\] { |   .md\:mx-\[10\%\] { | ||||||
|     @media (width >= 48rem) { |     @media (width >= 48rem) { | ||||||
|       margin-inline: 10%; |       margin-inline: 10%; | ||||||
| @@ -815,6 +960,11 @@ | |||||||
|       width: calc(1/2 * 100%); |       width: calc(1/2 * 100%); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |   .md\:flex-row { | ||||||
|  |     @media (width >= 48rem) { | ||||||
|  |       flex-direction: row; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|   .md\:px-4 { |   .md\:px-4 { | ||||||
|     @media (width >= 48rem) { |     @media (width >= 48rem) { | ||||||
|       padding-inline: calc(var(--spacing) * 4); |       padding-inline: calc(var(--spacing) * 4); | ||||||
| @@ -825,8 +975,8 @@ | |||||||
|       color: transparent; |       color: transparent; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .group-\[\.edit\]\:md\:block { |   .group-\[\.edit\]\/button\:md\:block { | ||||||
|     &:is(:where(.group):is(.edit) *) { |     &:is(:where(.group\/button):is(.edit) *) { | ||||||
|       @media (width >= 48rem) { |       @media (width >= 48rem) { | ||||||
|         display: block; |         display: block; | ||||||
|       } |       } | ||||||
| @@ -878,12 +1028,13 @@ | |||||||
|   body { |   body { | ||||||
|     -webkit-print-color-adjust: exact !important; |     -webkit-print-color-adjust: exact !important; | ||||||
|     print-color-adjust: exact !important; |     print-color-adjust: exact !important; | ||||||
|  |     background-color: white; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @layer components { | @layer components { | ||||||
|   .grid-main { |   .grid-main { | ||||||
|     display: grid; |     display: grid; | ||||||
|     grid-template-columns: 2fr auto 1fr; |     grid-template-columns: 4fr 3fr 3fr 1fr; | ||||||
|     align-items: stretch; |     align-items: stretch; | ||||||
|   } |   } | ||||||
|   .grid-sub { |   .grid-sub { | ||||||
| @@ -919,6 +1070,9 @@ | |||||||
|     transition-timing-function: var( --tw-ease, var(--default-transition-timing-function) ); |     transition-timing-function: var( --tw-ease, var(--default-transition-timing-function) ); | ||||||
|     transition-duration: var(--tw-duration, var(--default-transition-duration)); |     transition-duration: var(--tw-duration, var(--default-transition-duration)); | ||||||
|   } |   } | ||||||
|  |   input.btn, select.btn { | ||||||
|  |     transition-duration: 300ms; | ||||||
|  |   } | ||||||
|   .btn:hover { |   .btn:hover { | ||||||
|     color: var(--color-white); |     color: var(--color-white); | ||||||
|     background-color: var(--color-neutral-700); |     background-color: var(--color-neutral-700); | ||||||
| @@ -927,8 +1081,38 @@ | |||||||
|     opacity: 50%; |     opacity: 50%; | ||||||
|     pointer-events: none; |     pointer-events: none; | ||||||
|   } |   } | ||||||
|   .btn:active, .btn:focus { |   input.btn, select.btn { | ||||||
|     background-color: var(--color-neutral-700); |     text-align: left; | ||||||
|  |   } | ||||||
|  |   input.btn:hover, select.btn:hover { | ||||||
|  |     border-color: var(--color-neutral-300); | ||||||
|  |     background-color: var(--color-neutral-100); | ||||||
|  |     color: var(--color-neutral-800); | ||||||
|  |   } | ||||||
|  |   .edit-box { | ||||||
|  |     border-radius: var(--radius-md); | ||||||
|  |     overflow: hidden; | ||||||
|  |     border-color: var(--color-neutral-500); | ||||||
|  |     transition-property: background-color, border-color; | ||||||
|  |     transition-timing-function: var(--default-transition-timing-function) * 2; | ||||||
|  |     transition-duration: var(--default-transition-duration); | ||||||
|  |     outline: none; | ||||||
|  |     &:is(:where(.group):is(.edit) *) { | ||||||
|  |       border-width: 1px; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   .edit-box:hover { | ||||||
|  |     &:is(:where(.group):is(.edit) *) { | ||||||
|  |       background-color: var(--color-white); | ||||||
|  |       border-color: var(--color-neutral-300); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   .edit-box input:focus { | ||||||
|  |     outline: none; | ||||||
|  |   } | ||||||
|  |   div.edit { | ||||||
|  |     border-width: 1px; | ||||||
|  |     background-color: var(--color-neutral-300); | ||||||
|   } |   } | ||||||
|   @media (width >=48rem) { |   @media (width >=48rem) { | ||||||
|     .grid-main { |     .grid-main { | ||||||
| @@ -943,6 +1127,41 @@ | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | @property --tw-translate-x { | ||||||
|  |   syntax: "*"; | ||||||
|  |   inherits: false; | ||||||
|  |   initial-value: 0; | ||||||
|  | } | ||||||
|  | @property --tw-translate-y { | ||||||
|  |   syntax: "*"; | ||||||
|  |   inherits: false; | ||||||
|  |   initial-value: 0; | ||||||
|  | } | ||||||
|  | @property --tw-translate-z { | ||||||
|  |   syntax: "*"; | ||||||
|  |   inherits: false; | ||||||
|  |   initial-value: 0; | ||||||
|  | } | ||||||
|  | @property --tw-rotate-x { | ||||||
|  |   syntax: "*"; | ||||||
|  |   inherits: false; | ||||||
|  | } | ||||||
|  | @property --tw-rotate-y { | ||||||
|  |   syntax: "*"; | ||||||
|  |   inherits: false; | ||||||
|  | } | ||||||
|  | @property --tw-rotate-z { | ||||||
|  |   syntax: "*"; | ||||||
|  |   inherits: false; | ||||||
|  | } | ||||||
|  | @property --tw-skew-x { | ||||||
|  |   syntax: "*"; | ||||||
|  |   inherits: false; | ||||||
|  | } | ||||||
|  | @property --tw-skew-y { | ||||||
|  |   syntax: "*"; | ||||||
|  |   inherits: false; | ||||||
|  | } | ||||||
| @property --tw-divide-x-reverse { | @property --tw-divide-x-reverse { | ||||||
|   syntax: "*"; |   syntax: "*"; | ||||||
|   inherits: false; |   inherits: false; | ||||||
| @@ -962,6 +1181,11 @@ | |||||||
|   syntax: "*"; |   syntax: "*"; | ||||||
|   inherits: false; |   inherits: false; | ||||||
| } | } | ||||||
|  | @property --tw-outline-style { | ||||||
|  |   syntax: "*"; | ||||||
|  |   inherits: false; | ||||||
|  |   initial-value: solid; | ||||||
|  | } | ||||||
| @property --tw-blur { | @property --tw-blur { | ||||||
|   syntax: "*"; |   syntax: "*"; | ||||||
|   inherits: false; |   inherits: false; | ||||||
| @@ -1022,10 +1246,19 @@ | |||||||
| @layer properties { | @layer properties { | ||||||
|   @supports ((-webkit-hyphens: none) and (not (margin-trim: inline))) or ((-moz-orient: inline) and (not (color:rgb(from red r g b)))) { |   @supports ((-webkit-hyphens: none) and (not (margin-trim: inline))) or ((-moz-orient: inline) and (not (color:rgb(from red r g b)))) { | ||||||
|     *, ::before, ::after, ::backdrop { |     *, ::before, ::after, ::backdrop { | ||||||
|  |       --tw-translate-x: 0; | ||||||
|  |       --tw-translate-y: 0; | ||||||
|  |       --tw-translate-z: 0; | ||||||
|  |       --tw-rotate-x: initial; | ||||||
|  |       --tw-rotate-y: initial; | ||||||
|  |       --tw-rotate-z: initial; | ||||||
|  |       --tw-skew-x: initial; | ||||||
|  |       --tw-skew-y: initial; | ||||||
|       --tw-divide-x-reverse: 0; |       --tw-divide-x-reverse: 0; | ||||||
|       --tw-border-style: solid; |       --tw-border-style: solid; | ||||||
|       --tw-divide-y-reverse: 0; |       --tw-divide-y-reverse: 0; | ||||||
|       --tw-font-weight: initial; |       --tw-font-weight: initial; | ||||||
|  |       --tw-outline-style: solid; | ||||||
|       --tw-blur: initial; |       --tw-blur: initial; | ||||||
|       --tw-brightness: initial; |       --tw-brightness: initial; | ||||||
|       --tw-contrast: initial; |       --tw-contrast: initial; | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								Backend/static/logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								Backend/static/logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 17 KiB | 
| @@ -1,34 +1,87 @@ | |||||||
| function editDay(element, event, formId) { | function clearEditState() { | ||||||
|   var form = element.closest(".grid-sub").querySelector(".all-booking-component > form"); |   for (let e of document.querySelectorAll(".edit")) { | ||||||
|   form.classList.toggle("edit"); |     e.classList.remove("edit"); | ||||||
|   element.classList.toggle("edit"); |   } | ||||||
|   if (element.classList.contains("edit")) { |   toggleAbsenceEdit(false); | ||||||
|     event.preventDefault(); | } | ||||||
|     form.querySelectorAll("input, select").forEach((input) => { |  | ||||||
|       input.disabled = false; | function clearButtonState() { | ||||||
|     }); |   for (let b of document.querySelectorAll(".change-button-component")) { | ||||||
|   } else { |     b.type = "button"; | ||||||
|     form.submit(); |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| function editAbwesenheit(element, event) { | function editWorkday(element, event, id, isWorkDay) { | ||||||
|   var newBookingComponent = element.closest(".grid-sub").querySelector(".new-booking-component"); |   event.preventDefault(); | ||||||
|   if (element.value == 0) { |   let form = document.getElementById(id); | ||||||
|     newBookingComponent.style.display = ""; |   if (form == null) { | ||||||
|  |     form = element.closest(".grid-sub").querySelector(".all-booking-component > form"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   clearEditState(); | ||||||
|  |   element.closest(".grid-sub").classList.add("edit"); | ||||||
|  |   toggleAbsenceEdit(!isWorkDay); | ||||||
|  |  | ||||||
|  |   if (isWorkDay) { | ||||||
|  |     element.classList.add("edit"); | ||||||
|  |     if (element.type == "button") { | ||||||
|  |       for (let input of form.querySelectorAll("input, select")) { | ||||||
|  |         input.disabled = false; | ||||||
|  |       } | ||||||
|  |       clearButtonState(); | ||||||
|  |       element.type = "submit"; | ||||||
|  |     } else { | ||||||
|  |       form.submit(); | ||||||
|  |     } | ||||||
|   } else { |   } else { | ||||||
|     newBookingComponent.style.display = "none"; |     const absenceForm = document.getElementById("absence_form"); | ||||||
|  |  | ||||||
|  |     if (id == 0) { | ||||||
|  |       absenceForm.querySelector("[name=date_from]").value = form.id.replace("time-", ""); | ||||||
|  |       absenceForm.querySelector("[name=date_to]").value = form.id.replace("time-", ""); | ||||||
|  |     } else { | ||||||
|  |       syncFields(form, absenceForm, ["date_from", "date_to", "aw_type", "aw_id"]); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function toggleAbsenceEdit(state) { | ||||||
|  |   const form = document.getElementById("absence_form"); | ||||||
|  |   if (state) { | ||||||
|  |     form.classList.remove("hidden"); | ||||||
|  |     form.scrollIntoView({ | ||||||
|  |       behavior: "smooth", | ||||||
|  |       block: "start", | ||||||
|  |       inline: "nearest", | ||||||
|  |     }); | ||||||
|  |   } else { | ||||||
|  |     form.classList.add("hidden"); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function syncFields(from, to, fieldsToSync) { | ||||||
|  |   for (let field of fieldsToSync) { | ||||||
|  |     const src = from.querySelector(`[name=${field}]`); | ||||||
|  |     const target = to.querySelector(`[name=${field}]`); | ||||||
|  |     if (!src || !target) return; | ||||||
|  |     target.value = src.value; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| function navigateWeek(element, event, direction) { | function navigateWeek(element, event, direction) { | ||||||
|   var dateInput = element.closest("form").querySelector("input[type=date]"); |   const dateInput = element.closest("form").querySelector("input[type=date]"); | ||||||
|   var date = dateInput.valueAsDate; |   const date = dateInput.valueAsDate; | ||||||
|   date.setDate(date.getDate() + 7 * direction); |   date.setDate(date.getDate() + 7 * direction); | ||||||
|   date.setHours(10); |   date.setHours(10); | ||||||
|   dateInput.valueAsDate = date; |   dateInput.valueAsDate = date; | ||||||
| } | } | ||||||
|  |  | ||||||
| function logoutUser() { | function logoutUser() { | ||||||
|   fetch("/user/logout", {}).then(() => window.location.reload()); |   fetch("/user/logout", {}).then(() => globalThis.location.reload()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function checkAll(pattern, state) { | ||||||
|  |   for (let input of document.querySelectorAll(`input[id^=${pattern}]`)) { | ||||||
|  |     input.checked = state; | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										92
									
								
								Backend/template.typ
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								Backend/template.typ
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,92 @@ | |||||||
|  | #let table-header(..headers) = { | ||||||
|  |   table.header( | ||||||
|  |     ..headers.pos().map(h => strong(h)) | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #let abrechnung(meta, days) = { | ||||||
|  |   set page(paper: "a4", margin: (x:1.5cm, y:2.25cm), | ||||||
|  |     footer:[#grid( | ||||||
|  |       columns: (3fr,  .65fr), | ||||||
|  |       align: left + horizon, | ||||||
|  |       inset: .5em, | ||||||
|  |       [#meta.EmployeeName -- #meta.TimeRange], grid.cell(rowspan: 2)[#image("static/logo.png")], | ||||||
|  |       [Arbeitszeitrechnung maschinell erstellt am #meta.CurrentTimestamp], | ||||||
|  |     ) | ||||||
|  |       ]) | ||||||
|  |   set text(font: "Noto Sans", size:10pt, fill: luma(10%)) | ||||||
|  |   set table( | ||||||
|  |     stroke: 0.5pt + luma(10%), | ||||||
|  |     inset: .5em, | ||||||
|  |     align: center + horizon, | ||||||
|  |   ) | ||||||
|  |   show text: it => { | ||||||
|  |     if it.text == "0min"{ | ||||||
|  |       text(oklch(70.8%, 0, 0deg))[#it] | ||||||
|  |     }else if it.text.starts-with("-"){ | ||||||
|  |       text(red)[#it] | ||||||
|  |     }else{ | ||||||
|  |       it | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   [= Abrechnung Arbeitszeit -- #meta.EmployeeName] | ||||||
|  |  | ||||||
|  |   [Zeitraum: #meta.TimeRange] | ||||||
|  |  | ||||||
|  |   table( | ||||||
|  |   columns: (1fr, 1fr, 1fr, 1fr, 1fr, 1fr, 1.25fr), | ||||||
|  |   fill: (x, y) => | ||||||
|  |     if y == 0 { oklch(87%, 0, 0deg) }, | ||||||
|  |   table-header( | ||||||
|  |     [Datum], [Kommen], [Gehen], [Arbeitsart], [Stunden], [Pause], [Überstunden] | ||||||
|  |   ), | ||||||
|  |   .. for day in days { | ||||||
|  |     ( | ||||||
|  |     [#day.Date], | ||||||
|  |     if day.DayParts.len() == 0{ | ||||||
|  |         table.cell(colspan: 3)[Keine Buchungen] | ||||||
|  |     }else if not day.DayParts.first().IsWorkDay{ | ||||||
|  |       table.cell(colspan: 3)[#day.DayParts.first().WorkType] | ||||||
|  |     } | ||||||
|  |     else { | ||||||
|  |  | ||||||
|  |     table.cell(colspan: 3, inset: 0em)[ | ||||||
|  |  | ||||||
|  |       #table( | ||||||
|  |         columns: (1fr, 1fr, 1fr), | ||||||
|  |         .. for Zeit in day.DayParts { | ||||||
|  |       ( | ||||||
|  |       [#Zeit.BookingFrom], | ||||||
|  |       [#Zeit.BookingTo], | ||||||
|  |       [#Zeit.WorkType], | ||||||
|  |       ) | ||||||
|  |     }, | ||||||
|  |       ) | ||||||
|  |     ] | ||||||
|  |     }, | ||||||
|  |     [#day.Worktime], | ||||||
|  |     [#day.Pausetime], | ||||||
|  |     [#day.Overtime], | ||||||
|  |     ) | ||||||
|  |     if day.IsFriday { | ||||||
|  |       ( table.cell(colspan: 7, fill: oklch(87%, 0, 0deg))[Wochenende], )   // note the trailing comma | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   ) | ||||||
|  |  | ||||||
|  |   table( | ||||||
|  |     columns: (3fr, 1fr), | ||||||
|  |     align: right, | ||||||
|  |     inset: (x: .25em, y:.75em), | ||||||
|  |     stroke: none, | ||||||
|  |     table.hline(start: 0, end: 2, stroke: stroke(dash:"dashed", thickness:.5pt)), | ||||||
|  |   [Arbeitszeit :], table.cell(align: left)[#meta.WorkTime], | ||||||
|  |   [Überstunden :], table.cell(align: left)[#meta.Overtime], | ||||||
|  |   [Überstunden :],table.cell(align: left)[#meta.OvertimeTotal], | ||||||
|  |   table.hline(start: 0, end: 2), | ||||||
|  |  | ||||||
|  | ) | ||||||
|  | } | ||||||
| @@ -4,8 +4,9 @@ templ headerComponent() { | |||||||
| 	<div class="flex flex-row justify-between md:mx-[10%] py-2 items-center"> | 	<div class="flex flex-row justify-between md:mx-[10%] py-2 items-center"> | ||||||
| 		<a href="/time">Zeitverwaltung</a> | 		<a href="/time">Zeitverwaltung</a> | ||||||
| 		<a href="/team">Abrechnung</a> | 		<a href="/team">Abrechnung</a> | ||||||
|  | 		<a href="/pdf">PDF</a> | ||||||
| 		if true { | 		if true { | ||||||
| 			<a href="/team/presence">Anwesenheit</a> | 			<a href="/presence">Anwesenheit</a> | ||||||
| 		} | 		} | ||||||
| 		<a href="/user/settings">Einstellungen</a> | 		<a href="/user/settings">Einstellungen</a> | ||||||
| 		@LogoutButton() | 		@LogoutButton() | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| // Code generated by templ - DO NOT EDIT. | // Code generated by templ - DO NOT EDIT. | ||||||
|  |  | ||||||
| // templ: version: v0.3.943 | // templ: version: v0.3.924 | ||||||
| package templates | package templates | ||||||
|  |  | ||||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||||
| @@ -29,12 +29,12 @@ func headerComponent() templ.Component { | |||||||
| 			templ_7745c5c3_Var1 = templ.NopComponent | 			templ_7745c5c3_Var1 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<div class=\"flex flex-row justify-between md:mx-[10%] py-2 items-center\"><a href=\"/time\">Zeitverwaltung</a> <a href=\"/team\">Abrechnung</a> ") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<div class=\"flex flex-row justify-between md:mx-[10%] py-2 items-center\"><a href=\"/time\">Zeitverwaltung</a> <a href=\"/team\">Abrechnung</a> <a href=\"/pdf\">PDF</a> ") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		if true { | 		if true { | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "<a href=\"/team/presence\">Anwesenheit</a> ") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "<a href=\"/presence\">Anwesenheit</a> ") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
|   | |||||||
| @@ -1,12 +1,6 @@ | |||||||
| package templates | package templates | ||||||
|  |  | ||||||
| import ( | import "arbeitszeitmessung/models" | ||||||
| 	"arbeitszeitmessung/helper" |  | ||||||
| 	"arbeitszeitmessung/models" |  | ||||||
| 	"fmt" |  | ||||||
| 	"strconv" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| templ Base() { | templ Base() { | ||||||
| 	<!DOCTYPE html> | 	<!DOCTYPE html> | ||||||
| @@ -18,21 +12,6 @@ templ Base() { | |||||||
| 	</head> | 	</head> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ TimePage(workDays []models.WorkDay, lastSub time.Time) { |  | ||||||
| 	@Base() |  | ||||||
| 	@headerComponent() |  | ||||||
| 	<div class="grid-main divide-y-1"> |  | ||||||
| 		@inputForm() |  | ||||||
| 		for _, day := range workDays { |  | ||||||
| 			@dayComponent(day, day.Day.Before(lastSub)) |  | ||||||
| 			if (day.Day.Weekday() == time.Monday) { |  | ||||||
| 				<div class="grid-sub responsive bg-neutral-300 h-2"></div> |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	</div> |  | ||||||
| 	@LegendComponent() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| templ LoginPage(success bool, errorMsg string) { | templ LoginPage(success bool, errorMsg string) { | ||||||
| 	@Base() | 	@Base() | ||||||
| 	<div class="w-full h-[100vh] flex flex-col justify-center items-center"> | 	<div class="w-full h-[100vh] flex flex-col justify-center items-center"> | ||||||
| @@ -49,7 +28,10 @@ templ LoginPage(success bool, errorMsg string) { | |||||||
| 	</div> | 	</div> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ UserPage(status int) { | templ SettingsPage(status int) { | ||||||
|  | 	{{ | ||||||
|  | 		user := ctx.Value("user").(models.User) | ||||||
|  | 	}} | ||||||
| 	@Base() | 	@Base() | ||||||
| 	@headerComponent() | 	@headerComponent() | ||||||
| 	<div class="grid-main divide-y-1"> | 	<div class="grid-main divide-y-1"> | ||||||
| @@ -72,6 +54,14 @@ templ UserPage(status int) { | |||||||
| 				<button name="action" value="change-pass" type="submit" class="btn">Ändern</button> | 				<button name="action" value="change-pass" type="submit" class="btn">Ändern</button> | ||||||
| 			</div> | 			</div> | ||||||
| 		</form> | 		</form> | ||||||
|  | 		<div class="grid-sub responsive lg:divide-x-1"> | ||||||
|  | 			<h1 class="grid-cell font-bold uppercase text-xl text-center">Nutzerdaten</h1> | ||||||
|  | 			<div class="grid-cell col-span-3"> | ||||||
|  | 				<p>Nutzername: <span class="text-neutral-500">{ user.Vorname } { user.Name }</span></p> | ||||||
|  | 				<p>Personalnummer: <span class="text-neutral-500">{ user.PersonalNummer }</span></p> | ||||||
|  | 			</div> | ||||||
|  | 			<div></div> | ||||||
|  | 		</div> | ||||||
| 		<div class="grid-sub responsive lg:divide-x-1"> | 		<div class="grid-sub responsive lg:divide-x-1"> | ||||||
| 			<h1 class="grid-cell font-bold uppercase text-xl text-center">Nutzer abmelden</h1> | 			<h1 class="grid-cell font-bold uppercase text-xl text-center">Nutzer abmelden</h1> | ||||||
| 			<div class="grid-cell col-span-3"> | 			<div class="grid-cell col-span-3"> | ||||||
| @@ -95,99 +85,24 @@ templ statusCheckMark(status models.WeekStatus, target models.WeekStatus) { | |||||||
| templ TeamPage(weeks []models.WorkWeek, userWeek models.WorkWeek) { | templ TeamPage(weeks []models.WorkWeek, userWeek models.WorkWeek) { | ||||||
| 	@Base() | 	@Base() | ||||||
| 	@headerComponent() | 	@headerComponent() | ||||||
| 	{{ |  | ||||||
| 		progress := (float32(userWeek.Worktime.Hours()) / userWeek.User.ArbeitszeitPerWoche) * 100 |  | ||||||
| 	}} |  | ||||||
| 	<div class="grid-main divide-y-1"> | 	<div class="grid-main divide-y-1"> | ||||||
| 		<div class="grid-sub lg:divide-x-1 max-md:divide-y-1 responsive @container"> | 		<div class="grid-sub lg:divide-x-1 max-md:divide-y-1 responsive @container"> | ||||||
| 			<div class="grid-cell col-span-full bg-neutral-300 lg:border-0"> | 			<div class="grid-cell col-span-full bg-neutral-300 lg:border-0"> | ||||||
| 				<h2 class="text-2xl uppercase font-bold">Eigene Abrechnung</h2> | 				<h2 class="text-xl uppercase font-bold">Eigene Abrechnung</h2> | ||||||
| 			</div> |  | ||||||
| 			<div class="grid-cell flex flex-col max-md:border-b-1 max-md:bg-neutral-300 gap-2 "> |  | ||||||
| 				<div class="lg:hidden"> |  | ||||||
| 					@weekPicker(userWeek.WeekStart) |  | ||||||
| 				</div> |  | ||||||
| 				<h2 class="uppercase font-bold">{ fmt.Sprintf("%s %s", userWeek.User.Vorname, userWeek.User.Name) }</h2> |  | ||||||
| 				<div class="grid grid-cols-5 gap-2 lg:grid-cols-1"> |  | ||||||
| 					<div class="col-span-2"> |  | ||||||
| 						<span class="flex flex-row gap-2 items-center"> |  | ||||||
| 							@statusCheckMark(userWeek.CheckStatus(), models.WeekStatusSent) |  | ||||||
| 							Gesendet |  | ||||||
| 						</span> |  | ||||||
| 						<span class="flex flex-row gap-2 items-center"> |  | ||||||
| 							@statusCheckMark(userWeek.CheckStatus(), models.WeekStatusAccepted) |  | ||||||
| 							Akzeptiert |  | ||||||
| 						</span> |  | ||||||
| 					</div> |  | ||||||
| 					<div class="flex flex-row gap-2 col-span-3"> |  | ||||||
| 						@timeGaugeComponent(uint8(progress), false, false) |  | ||||||
| 						<div> |  | ||||||
| 							<p>Arbeitszeit: { fmt.Sprintf("%s", helper.FormatDuration(userWeek.Worktime)) }</p> |  | ||||||
| 							<p>Überstunden: { fmt.Sprintf("%s", helper.FormatDuration(userWeek.Overtime)) }</p> |  | ||||||
| 						</div> |  | ||||||
| 					</div> |  | ||||||
| 				</div> |  | ||||||
| 			</div> |  | ||||||
| 			<div class="grid-cell col-span-3 flex flex-col @7xl:grid @7xl:grid-cols-5 gap-2 py-4 content-baseline"> |  | ||||||
| 				for _, day := range userWeek.WorkDays { |  | ||||||
| 					@weekDayComponent(userWeek.User, day) |  | ||||||
| 				} |  | ||||||
| 			</div> |  | ||||||
| 			<div class="grid-cell flex flex-col gap-2 justify-between"> |  | ||||||
| 				<div class="max-md:hidden"> |  | ||||||
| 					@weekPicker(userWeek.WeekStart) |  | ||||||
| 				</div> |  | ||||||
| 				<form method="post" class="flex flex-col gap-2"> |  | ||||||
| 					<input type="hidden" name="method" value="send"/> |  | ||||||
| 					<input type="hidden" name="user" value={ strconv.Itoa(userWeek.User.PersonalNummer) }/> |  | ||||||
| 					<input type="hidden" name="week" value={ userWeek.WeekStart.Format(time.DateOnly) }/> |  | ||||||
| 					switch userWeek.CheckStatus() { |  | ||||||
| 						case models.WeekStatusNone: |  | ||||||
| 							<p class="text-sm">an Vorgesetzten senden</p> |  | ||||||
| 						case models.WeekStatusSent: |  | ||||||
| 							<p class="text-sm">an Vorgesetzten gesendet</p> |  | ||||||
| 						case models.WeekStatusAccepted: |  | ||||||
| 							<p class="text-sm">vom Vorgesetzten bestätigt</p> |  | ||||||
| 					} |  | ||||||
| 					<button disabled?={ userWeek.Status < models.WeekStatusSent } type="submit" class="btn">Korrigieren</button> |  | ||||||
| 					<button disabled?={ time.Since(userWeek.WeekStart) < 24*7*time.Hour || userWeek.Status >= models.WeekStatusSent || userWeek.RequiresAction() } type="submit" class="btn">Senden</button> |  | ||||||
| 				</form> |  | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
|  | 		@workWeekComponent(userWeek, false) | ||||||
| 		if len(weeks) > 0 { | 		if len(weeks) > 0 { | ||||||
| 			<div class="grid-cell col-span-full bg-neutral-300"> | 			<div class="grid-cell col-span-full bg-neutral-300"> | ||||||
| 				<h2 class="text-2xl uppercase font-bold">Abrechnung Mitarbeiter</h2> | 				<h2 class="text-xl uppercase font-bold">Abrechnung Mitarbeiter</h2> | ||||||
| 			</div> | 			</div> | ||||||
| 		} | 		} | ||||||
| 		for _, week := range weeks { | 		for _, week := range weeks { | ||||||
| 			@employeComponent(week) | 			@workWeekComponent(week, true) | ||||||
| 		} | 		} | ||||||
| 	</div> | 	</div> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ TeamPresencePage(teamPresence map[bool][]models.User) { |  | ||||||
| 	@Base() |  | ||||||
| 	@headerComponent() |  | ||||||
| 	<div class="grid-main divide-y-1"> |  | ||||||
| 		<div class="grid-sub divide-x-1"> |  | ||||||
| 			<h2 class="grid-cell font-bold uppercase">Anwesend</h2> |  | ||||||
| 			<div class="flex flex-col col-span-2 md:col-span-4"> |  | ||||||
| 				for _, user := range teamPresence[true] { |  | ||||||
| 					@userPresenceComponent(user, true) |  | ||||||
| 				} |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 		<div class="grid-sub divide-x-1"> |  | ||||||
| 			<h2 class="grid-cell font-bold uppercase">Nicht Anwesend</h2> |  | ||||||
| 			<div class="flex flex-col col-span-2 md:col-span-4"> |  | ||||||
| 				for _, user := range teamPresence[false] { |  | ||||||
| 					@userPresenceComponent(user, false) |  | ||||||
| 				} |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 	</div> |  | ||||||
| } |  | ||||||
|  |  | ||||||
| templ LogoutButton() { | templ LogoutButton() { | ||||||
| 	<button onclick="logoutUser()" type="button" class="cursor-pointer">Abmelden</button> | 	<button onclick="logoutUser()" type="button" class="cursor-pointer">Abmelden</button> | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| // Code generated by templ - DO NOT EDIT. | // Code generated by templ - DO NOT EDIT. | ||||||
|  |  | ||||||
| // templ: version: v0.3.943 | // templ: version: v0.3.924 | ||||||
| package templates | package templates | ||||||
|  |  | ||||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||||
| @@ -8,13 +8,7 @@ package templates | |||||||
| import "github.com/a-h/templ" | import "github.com/a-h/templ" | ||||||
| import templruntime "github.com/a-h/templ/runtime" | import templruntime "github.com/a-h/templ/runtime" | ||||||
|  |  | ||||||
| import ( | import "arbeitszeitmessung/models" | ||||||
| 	"arbeitszeitmessung/helper" |  | ||||||
| 	"arbeitszeitmessung/models" |  | ||||||
| 	"fmt" |  | ||||||
| 	"strconv" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func Base() templ.Component { | func Base() templ.Component { | ||||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
| @@ -45,7 +39,7 @@ func Base() templ.Component { | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func TimePage(workDays []models.WorkDay, lastSub time.Time) templ.Component { | func LoginPage(success bool, errorMsg string) templ.Component { | ||||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
| @@ -70,95 +64,30 @@ func TimePage(workDays []models.WorkDay, lastSub time.Time) templ.Component { | |||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = headerComponent().Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "<div class=\"w-full h-[100vh] flex flex-col justify-center items-center\"><form method=\"POST\" class=\"w-9/10 md:w-1/2 flex flex-col gap-4 p-2 mb-2\"><h1 class=\"font-bold uppercase text-xl text-center mb-2\">Benutzer Anmelden</h1><input name=\"personal_nummer\" placeholder=\"Personalnummer\" type=\"text\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> <input name=\"password\" placeholder=\"Passwort\" type=\"password\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> ") | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "<div class=\"grid-main divide-y-1\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = inputForm().Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		for _, day := range workDays { |  | ||||||
| 			templ_7745c5c3_Err = dayComponent(day, day.Day.Before(lastSub)).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, " ") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 			if day.Day.Weekday() == time.Monday { |  | ||||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "<div class=\"grid-sub responsive bg-neutral-300 h-2\"></div>") |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "</div>") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = LegendComponent().Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		return nil |  | ||||||
| 	}) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func LoginPage(success bool, errorMsg string) templ.Component { |  | ||||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { |  | ||||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context |  | ||||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { |  | ||||||
| 			return templ_7745c5c3_CtxErr |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) |  | ||||||
| 		if !templ_7745c5c3_IsBuffer { |  | ||||||
| 			defer func() { |  | ||||||
| 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) |  | ||||||
| 				if templ_7745c5c3_Err == nil { |  | ||||||
| 					templ_7745c5c3_Err = templ_7745c5c3_BufErr |  | ||||||
| 				} |  | ||||||
| 			}() |  | ||||||
| 		} |  | ||||||
| 		ctx = templ.InitializeContext(ctx) |  | ||||||
| 		templ_7745c5c3_Var3 := templ.GetChildren(ctx) |  | ||||||
| 		if templ_7745c5c3_Var3 == nil { |  | ||||||
| 			templ_7745c5c3_Var3 = templ.NopComponent |  | ||||||
| 		} |  | ||||||
| 		ctx = templ.ClearChildren(ctx) |  | ||||||
| 		templ_7745c5c3_Err = Base().Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "<div class=\"w-full h-[100vh] flex flex-col justify-center items-center\"><form method=\"POST\" class=\"w-9/10 md:w-1/2 flex flex-col gap-4 p-2 mb-2\"><h1 class=\"font-bold uppercase text-xl text-center mb-2\">Benutzer Anmelden</h1><input name=\"personal_nummer\" placeholder=\"Personalnummer\" type=\"text\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> <input name=\"password\" placeholder=\"Passwort\" type=\"password\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> ") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		if !success { | 		if !success { | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "<p class=\"text-red-600 text-sm\">Login fehlgeschlagen, bitte erneut versuchen!</p><p class=\"text-red-600 text-sm\">") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "<p class=\"text-red-600 text-sm\">Login fehlgeschlagen, bitte erneut versuchen!</p><p class=\"text-red-600 text-sm\">") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 			var templ_7745c5c3_Var4 string | 			var templ_7745c5c3_Var3 string | ||||||
| 			templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(errorMsg) | 			templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(errorMsg) | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 45, Col: 46} | 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 24, Col: 46} | ||||||
| 			} | 			} | ||||||
| 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4)) | 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3)) | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, "</p>") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "</p>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "<button type=\"submit\" class=\"cursor-pointer rounded-md text-neutral-800 p-2 md:px-4 border text-center text-sm hover:text-white transition-colors  border-neutral-300 focus:bg-neutral-700 active:bg-neutral-700 hover:bg-neutral-700 disabled:pointer-events-none disabled:opacity-50\">Login</button></form></div>") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "<button type=\"submit\" class=\"cursor-pointer rounded-md text-neutral-800 p-2 md:px-4 border text-center text-sm hover:text-white transition-colors  border-neutral-300 focus:bg-neutral-700 active:bg-neutral-700 hover:bg-neutral-700 disabled:pointer-events-none disabled:opacity-50\">Login</button></form></div>") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| @@ -166,7 +95,7 @@ func LoginPage(success bool, errorMsg string) templ.Component { | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func UserPage(status int) templ.Component { | func SettingsPage(status int) templ.Component { | ||||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
| @@ -182,11 +111,13 @@ func UserPage(status int) templ.Component { | |||||||
| 			}() | 			}() | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.InitializeContext(ctx) | 		ctx = templ.InitializeContext(ctx) | ||||||
| 		templ_7745c5c3_Var5 := templ.GetChildren(ctx) | 		templ_7745c5c3_Var4 := templ.GetChildren(ctx) | ||||||
| 		if templ_7745c5c3_Var5 == nil { | 		if templ_7745c5c3_Var4 == nil { | ||||||
| 			templ_7745c5c3_Var5 = templ.NopComponent | 			templ_7745c5c3_Var4 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
|  |  | ||||||
|  | 		user := ctx.Value("user").(models.User) | ||||||
| 		templ_7745c5c3_Err = Base().Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = Base().Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| @@ -195,28 +126,67 @@ func UserPage(status int) templ.Component { | |||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "<div class=\"grid-main divide-y-1\"><form method=\"POST\" class=\"grid-sub responsive lg:divide-x-1\"><h1 class=\"grid-cell font-bold uppercase text-xl text-center\">Passwort ändern</h1><div class=\"grid-cell col-span-3 flex flex-col gap-2\"><input name=\"password\" placeholder=\"Aktuelles Passwort\" type=\"password\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> <input name=\"new_password\" placeholder=\"Neues Passwort\" type=\"password\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> <input name=\"new_password_repeat\" placeholder=\"Neues Passwort wiederholen\" type=\"password\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> ") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "<div class=\"grid-main divide-y-1\"><form method=\"POST\" class=\"grid-sub responsive lg:divide-x-1\"><h1 class=\"grid-cell font-bold uppercase text-xl text-center\">Passwort ändern</h1><div class=\"grid-cell col-span-3 flex flex-col gap-2\"><input name=\"password\" placeholder=\"Aktuelles Passwort\" type=\"password\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> <input name=\"new_password\" placeholder=\"Neues Passwort\" type=\"password\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> <input name=\"new_password_repeat\" placeholder=\"Neues Passwort wiederholen\" type=\"password\" class=\"w-full placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-300  rounded-md px-3 py-2 transition duration-300 ease focus:outline-none hover:border-neutral-500\"> ") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		switch { | 		switch { | ||||||
| 		case status == 401: | 		case status == 401: | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, "<p class=\"text-red-600 text-sm\">Aktuelles Passwort nicht korrekt!</p>") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "<p class=\"text-red-600 text-sm\">Aktuelles Passwort nicht korrekt!</p>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		case status >= 400: | 		case status >= 400: | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 12, "<p class=\"text-red-600 text-sm\">Passwortwechsel fehlgeschlagen, bitte erneut versuchen!</p>") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, "<p class=\"text-red-600 text-sm\">Passwortwechsel fehlgeschlagen, bitte erneut versuchen!</p>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		case status == 202: | 		case status == 202: | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, "<p class=\"text-accent text-sm\">Passwortänderung erfolgreich</p>") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "<p class=\"text-accent text-sm\">Passwortänderung erfolgreich</p>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 14, "</div><div class=\"grid-cell\"><button name=\"action\" value=\"change-pass\" type=\"submit\" class=\"btn\">Ändern</button></div></form><div class=\"grid-sub responsive lg:divide-x-1\"><h1 class=\"grid-cell font-bold uppercase text-xl text-center\">Nutzer abmelden</h1><div class=\"grid-cell col-span-3\"><p>Nutzer von Weboberfläche abmelden.</p></div><div class=\"grid-cell\"><button onclick=\"logoutUser\" type=\"button\" class=\"btn\">Abmelden</button></div></div></div>") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "</div><div class=\"grid-cell\"><button name=\"action\" value=\"change-pass\" type=\"submit\" class=\"btn\">Ändern</button></div></form><div class=\"grid-sub responsive lg:divide-x-1\"><h1 class=\"grid-cell font-bold uppercase text-xl text-center\">Nutzerdaten</h1><div class=\"grid-cell col-span-3\"><p>Nutzername: <span class=\"text-neutral-500\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var5 string | ||||||
|  | 		templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(user.Vorname) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 60, Col: 64} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var5)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, " ") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var6 string | ||||||
|  | 		templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(user.Name) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 60, Col: 78} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 12, "</span></p><p>Personalnummer: <span class=\"text-neutral-500\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var7 string | ||||||
|  | 		templ_7745c5c3_Var7, templ_7745c5c3_Err = templ.JoinStringErrs(user.PersonalNummer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 61, Col: 75} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var7)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, "</span></p></div><div></div></div><div class=\"grid-sub responsive lg:divide-x-1\"><h1 class=\"grid-cell font-bold uppercase text-xl text-center\">Nutzer abmelden</h1><div class=\"grid-cell col-span-3\"><p>Nutzer von Weboberfläche abmelden.</p></div><div class=\"grid-cell\"><button onclick=\"logoutUser\" type=\"button\" class=\"btn\">Abmelden</button></div></div></div>") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| @@ -240,18 +210,18 @@ func statusCheckMark(status models.WeekStatus, target models.WeekStatus) templ.C | |||||||
| 			}() | 			}() | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.InitializeContext(ctx) | 		ctx = templ.InitializeContext(ctx) | ||||||
| 		templ_7745c5c3_Var6 := templ.GetChildren(ctx) | 		templ_7745c5c3_Var8 := templ.GetChildren(ctx) | ||||||
| 		if templ_7745c5c3_Var6 == nil { | 		if templ_7745c5c3_Var8 == nil { | ||||||
| 			templ_7745c5c3_Var6 = templ.NopComponent | 			templ_7745c5c3_Var8 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		if status >= target { | 		if status >= target { | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, "<div class=\"icon-[material-symbols-light--check-circle-outline]\"></div>") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 14, "<div class=\"icon-[material-symbols-light--check-circle-outline]\"></div>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "<div class=\"icon-[material-symbols-light--circle-outline]\"></div>") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, "<div class=\"icon-[material-symbols-light--circle-outline]\"></div>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| @@ -276,9 +246,9 @@ func TeamPage(weeks []models.WorkWeek, userWeek models.WorkWeek) templ.Component | |||||||
| 			}() | 			}() | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.InitializeContext(ctx) | 		ctx = templ.InitializeContext(ctx) | ||||||
| 		templ_7745c5c3_Var7 := templ.GetChildren(ctx) | 		templ_7745c5c3_Var9 := templ.GetChildren(ctx) | ||||||
| 		if templ_7745c5c3_Var7 == nil { | 		if templ_7745c5c3_Var9 == nil { | ||||||
| 			templ_7745c5c3_Var7 = templ.NopComponent | 			templ_7745c5c3_Var9 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		templ_7745c5c3_Err = Base().Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = Base().Render(ctx, templ_7745c5c3_Buffer) | ||||||
| @@ -289,238 +259,27 @@ func TeamPage(weeks []models.WorkWeek, userWeek models.WorkWeek) templ.Component | |||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "<div class=\"grid-main divide-y-1\"><div class=\"grid-sub lg:divide-x-1 max-md:divide-y-1 responsive @container\"><div class=\"grid-cell col-span-full bg-neutral-300 lg:border-0\"><h2 class=\"text-xl uppercase font-bold\">Eigene Abrechnung</h2></div></div>") | ||||||
| 		progress := (float32(userWeek.Worktime.Hours()) / userWeek.User.ArbeitszeitPerWoche) * 100 |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "<div class=\"grid-main divide-y-1\"><div class=\"grid-sub lg:divide-x-1 max-md:divide-y-1 responsive @container\"><div class=\"grid-cell col-span-full bg-neutral-300 lg:border-0\"><h2 class=\"text-2xl uppercase font-bold\">Eigene Abrechnung</h2></div><div class=\"grid-cell flex flex-col max-md:border-b-1 max-md:bg-neutral-300 gap-2 \"><div class=\"lg:hidden\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = weekPicker(userWeek.WeekStart).Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = workWeekComponent(userWeek, false).Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 18, "</div><h2 class=\"uppercase font-bold\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		var templ_7745c5c3_Var8 string |  | ||||||
| 		templ_7745c5c3_Var8, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%s %s", userWeek.User.Vorname, userWeek.User.Name)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 110, Col: 101} |  | ||||||
| 		} |  | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var8)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "</h2><div class=\"grid grid-cols-5 gap-2 lg:grid-cols-1\"><div class=\"col-span-2\"><span class=\"flex flex-row gap-2 items-center\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = statusCheckMark(userWeek.CheckStatus(), models.WeekStatusSent).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "Gesendet</span> <span class=\"flex flex-row gap-2 items-center\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = statusCheckMark(userWeek.CheckStatus(), models.WeekStatusAccepted).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, "Akzeptiert</span></div><div class=\"flex flex-row gap-2 col-span-3\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = timeGaugeComponent(uint8(progress), false, false).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 22, "<div><p>Arbeitszeit: ") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		var templ_7745c5c3_Var9 string |  | ||||||
| 		templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%s", helper.FormatDuration(userWeek.Worktime))) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 125, Col: 84} |  | ||||||
| 		} |  | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var9)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, "</p><p>Überstunden: ") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		var templ_7745c5c3_Var10 string |  | ||||||
| 		templ_7745c5c3_Var10, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%s", helper.FormatDuration(userWeek.Overtime))) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 126, Col: 85} |  | ||||||
| 		} |  | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var10)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, "</p></div></div></div></div><div class=\"grid-cell col-span-3 flex flex-col @7xl:grid @7xl:grid-cols-5 gap-2 py-4 content-baseline\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		for _, day := range userWeek.WorkDays { |  | ||||||
| 			templ_7745c5c3_Err = weekDayComponent(userWeek.User, day).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, "</div><div class=\"grid-cell flex flex-col gap-2 justify-between\"><div class=\"max-md:hidden\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = weekPicker(userWeek.WeekStart).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, "</div><form method=\"post\" class=\"flex flex-col gap-2\"><input type=\"hidden\" name=\"method\" value=\"send\"> <input type=\"hidden\" name=\"user\" value=\"") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		var templ_7745c5c3_Var11 string |  | ||||||
| 		templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(strconv.Itoa(userWeek.User.PersonalNummer)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 142, Col: 88} |  | ||||||
| 		} |  | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 27, "\"> <input type=\"hidden\" name=\"week\" value=\"") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		var templ_7745c5c3_Var12 string |  | ||||||
| 		templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(userWeek.WeekStart.Format(time.DateOnly)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 143, Col: 86} |  | ||||||
| 		} |  | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, "\"> ") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		switch userWeek.CheckStatus() { |  | ||||||
| 		case models.WeekStatusNone: |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, "<p class=\"text-sm\">an Vorgesetzten senden</p>") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		case models.WeekStatusSent: |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 30, "<p class=\"text-sm\">an Vorgesetzten gesendet</p>") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		case models.WeekStatusAccepted: |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 31, "<p class=\"text-sm\">vom Vorgesetzten bestätigt</p>") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 32, "<button") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		if userWeek.Status < models.WeekStatusSent { |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 33, " disabled") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 34, " type=\"submit\" class=\"btn\">Korrigieren</button> <button") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		if time.Since(userWeek.WeekStart) < 24*7*time.Hour || userWeek.Status >= models.WeekStatusSent || userWeek.RequiresAction() { |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 35, " disabled") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 36, " type=\"submit\" class=\"btn\">Senden</button></form></div></div>") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		if len(weeks) > 0 { | 		if len(weeks) > 0 { | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 37, "<div class=\"grid-cell col-span-full bg-neutral-300\"><h2 class=\"text-2xl uppercase font-bold\">Abrechnung Mitarbeiter</h2></div>") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "<div class=\"grid-cell col-span-full bg-neutral-300\"><h2 class=\"text-xl uppercase font-bold\">Abrechnung Mitarbeiter</h2></div>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		for _, week := range weeks { | 		for _, week := range weeks { | ||||||
| 			templ_7745c5c3_Err = employeComponent(week).Render(ctx, templ_7745c5c3_Buffer) | 			templ_7745c5c3_Err = workWeekComponent(week, true).Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 38, "</div>") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 18, "</div>") | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		return nil |  | ||||||
| 	}) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func TeamPresencePage(teamPresence map[bool][]models.User) templ.Component { |  | ||||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { |  | ||||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context |  | ||||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { |  | ||||||
| 			return templ_7745c5c3_CtxErr |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) |  | ||||||
| 		if !templ_7745c5c3_IsBuffer { |  | ||||||
| 			defer func() { |  | ||||||
| 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) |  | ||||||
| 				if templ_7745c5c3_Err == nil { |  | ||||||
| 					templ_7745c5c3_Err = templ_7745c5c3_BufErr |  | ||||||
| 				} |  | ||||||
| 			}() |  | ||||||
| 		} |  | ||||||
| 		ctx = templ.InitializeContext(ctx) |  | ||||||
| 		templ_7745c5c3_Var13 := templ.GetChildren(ctx) |  | ||||||
| 		if templ_7745c5c3_Var13 == nil { |  | ||||||
| 			templ_7745c5c3_Var13 = templ.NopComponent |  | ||||||
| 		} |  | ||||||
| 		ctx = templ.ClearChildren(ctx) |  | ||||||
| 		templ_7745c5c3_Err = Base().Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = headerComponent().Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 39, "<div class=\"grid-main divide-y-1\"><div class=\"grid-sub divide-x-1\"><h2 class=\"grid-cell font-bold uppercase\">Anwesend</h2><div class=\"flex flex-col col-span-2 md:col-span-4\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		for _, user := range teamPresence[true] { |  | ||||||
| 			templ_7745c5c3_Err = userPresenceComponent(user, true).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 40, "</div></div><div class=\"grid-sub divide-x-1\"><h2 class=\"grid-cell font-bold uppercase\">Nicht Anwesend</h2><div class=\"flex flex-col col-span-2 md:col-span-4\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		for _, user := range teamPresence[false] { |  | ||||||
| 			templ_7745c5c3_Err = userPresenceComponent(user, false).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 41, "</div></div></div>") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| @@ -544,12 +303,12 @@ func LogoutButton() templ.Component { | |||||||
| 			}() | 			}() | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.InitializeContext(ctx) | 		ctx = templ.InitializeContext(ctx) | ||||||
| 		templ_7745c5c3_Var14 := templ.GetChildren(ctx) | 		templ_7745c5c3_Var10 := templ.GetChildren(ctx) | ||||||
| 		if templ_7745c5c3_Var14 == nil { | 		if templ_7745c5c3_Var10 == nil { | ||||||
| 			templ_7745c5c3_Var14 = templ.NopComponent | 			templ_7745c5c3_Var10 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 42, "<button onclick=\"logoutUser()\" type=\"button\" class=\"cursor-pointer\">Abmelden</button>") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "<button onclick=\"logoutUser()\" type=\"button\" class=\"cursor-pointer\">Abmelden</button>") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -3,10 +3,62 @@ package templates | |||||||
| import ( | import ( | ||||||
| 	"arbeitszeitmessung/helper" | 	"arbeitszeitmessung/helper" | ||||||
| 	"arbeitszeitmessung/models" | 	"arbeitszeitmessung/models" | ||||||
|  | 	"fmt" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| templ PDFReportEmploye(e models.User, workDays []models.WorkDay, tsStart time.Time, tsEnd time.Time) { | templ PDFForm(teamMembers []models.User) { | ||||||
|  | 	@Base() | ||||||
|  | 	@headerComponent() | ||||||
|  | 	<div class="grid-main divide-y-1"> | ||||||
|  | 		<div class="grid-cell col-span-full bg-neutral-300"> | ||||||
|  | 			<h1 class="text-xl uppercase font-bold">PDF Abrechnung erstellen</h1> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="grid-sub divide-x-1 responsive"> | ||||||
|  | 			<div class="grid-cell">Zeitraum wählen</div> | ||||||
|  | 			<div class="grid-cell col-span-3"> | ||||||
|  | 				<label class="block mb-1 text-sm text-neutral-700">Abrechnungsmonat</label> | ||||||
|  | 				<input name="start_date" type="date" value="" class="btn bg-neutral-100"/> | ||||||
|  | 			</div> | ||||||
|  | 			<div></div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="grid-sub divide-x-1 responsive"> | ||||||
|  | 			<div class="grid-cell">Mitarbeiter wählen</div> | ||||||
|  | 			<div class="grid-cell col-span-3 flex flex-col gap-2"> | ||||||
|  | 				<div class="flex flex-row gap-2"> | ||||||
|  | 					<button class="btn" type="button" onclick={ templ.JSFuncCall("checkAll", "pdf-", templ.JSExpression("true")) }>Alle</button> | ||||||
|  | 					<button class="btn" type="button" onclick={ templ.JSFuncCall("checkAll", "pdf-", templ.JSExpression("false")) }>Keine</button> | ||||||
|  | 				</div> | ||||||
|  | 				for _, member := range teamMembers { | ||||||
|  | 					@CheckboxComponent(fmt.Sprintf("pdf-%d", member.PersonalNummer), fmt.Sprintf("%s %s", member.Vorname, member.Name)) | ||||||
|  | 				} | ||||||
|  | 			</div> | ||||||
|  | 			<div></div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="grid-sub divide-x-1 responsive"> | ||||||
|  | 			<div class="grid-cell">Direktvorschau oder Download</div> | ||||||
|  | 			<div class="grid-cell col-span-3 flex gap-2 flex-col md:flex-row"> | ||||||
|  | 				<button class="btn" type="button" name="action" value="download">Download</button> | ||||||
|  | 				<button class="btn" type="button" name="action" value="preview" onclick="">Vorschau</button> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | } | ||||||
|  |  | ||||||
|  | templ CheckboxComponent(id, label string) { | ||||||
|  | 	<div class="inline-flex items-center"> | ||||||
|  | 		<label class="flex items-center cursor-pointer relative" for={ id }> | ||||||
|  | 			<input type="checkbox" class="peer h-5 w-5 cursor-pointer transition-all appearance-none rounded border border-slate-800 checked:bg-slate-800 checked:border-slate-800" id={ id }/> | ||||||
|  | 			<span class="absolute text-white opacity-0 peer-checked:opacity-100 top-1/2 left-1/2 transform -translate-x-1/2 -translate-y-1/2"> | ||||||
|  | 				<svg xmlns="http://www.w3.org/2000/svg" class="h-3.5 w-3.5" viewBox="0 0 20 20" fill="currentColor" stroke="currentColor" stroke-width="1"> | ||||||
|  | 					<path fill-rule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clip-rule="evenodd"></path> | ||||||
|  | 				</svg> | ||||||
|  | 			</span> | ||||||
|  | 		</label> <label class="cursor-pointer ml-2 text-slate-600 select-none" for={ id }>{ label }</label> | ||||||
|  | 	</div> | ||||||
|  | } | ||||||
|  |  | ||||||
|  | templ PDFReportEmploye(e models.User, overtime, worktime time.Duration, workDays []models.IWorkDay, tsStart time.Time, tsEnd time.Time) { | ||||||
| 	{{ | 	{{ | ||||||
| 		_, kw := tsStart.ISOWeek() | 		_, kw := tsStart.ISOWeek() | ||||||
| 		noBorder := "" | 		noBorder := "" | ||||||
| @@ -14,10 +66,10 @@ templ PDFReportEmploye(e models.User, workDays []models.WorkDay, tsStart time.Ti | |||||||
| 	@Base() | 	@Base() | ||||||
| 	<content class="p-8 relative flex flex-col gap-4 break-after-page"> | 	<content class="p-8 relative flex flex-col gap-4 break-after-page"> | ||||||
| 		<div> | 		<div> | ||||||
| 			<h1 class="text-2xl font-bold">Kim Mustermensch</h1> | 			<h1 class="text-2xl font-bold">{ e.Vorname } { e.Name }</h1> | ||||||
| 			<p>Zeitraum: <span>{ tsStart.Format("02.01.2006") }</span> - <span>{ tsEnd.Format("02.01.2006") }</span></p> | 			<p>Zeitraum: <span>{ tsStart.Format("02.01.2006") }</span> - <span>{ tsEnd.Format("02.01.2006") }</span></p> | ||||||
| 			<p>Arbeitszeit: <span></span></p> | 			<p>Arbeitszeit: <span>{ helper.FormatDuration(worktime) }</span></p> | ||||||
| 			<p>Überstunden: <span></span></p> | 			<p>Überstunden: <span>{ helper.FormatDuration(overtime) }</span></p> | ||||||
| 		</div> | 		</div> | ||||||
| 		<div class="grid grid-rows-6 grid-cols-[3fr_2fr_2fr_2fr_3fr_3fr_3fr] *:not-print:p-2 *:text-center auto-rows-min divide-neutral-300 divide-x-1 divide-y-1"> | 		<div class="grid grid-rows-6 grid-cols-[3fr_2fr_2fr_2fr_3fr_3fr_3fr] *:not-print:p-2 *:text-center auto-rows-min divide-neutral-300 divide-x-1 divide-y-1"> | ||||||
| 			<p class="bg-neutral-300 border-neutral-600">{ kw }</p> | 			<p class="bg-neutral-300 border-neutral-600">{ kw }</p> | ||||||
| @@ -33,23 +85,37 @@ templ PDFReportEmploye(e models.User, workDays []models.WorkDay, tsStart time.Ti | |||||||
| 						noBorder = "border-b-0" | 						noBorder = "border-b-0" | ||||||
| 					} | 					} | ||||||
| 				}} | 				}} | ||||||
| 				<p class={ noBorder }>{ day.Day.Format("02.01.2006") }</p> | 				<p class={ noBorder }>{ day.Date().Format("02.01.2006") }</p> | ||||||
|  |  | ||||||
| 				<div class={ "grid grid-cols-subgrid col-span-3 " + noBorder }> | 				<div class={ "grid grid-cols-subgrid col-span-3 " + noBorder }> | ||||||
| 					for bookingI := 0; bookingI < len(day.Bookings); bookingI+= 2 { | 					if day.IsWorkDay() { | ||||||
| 						<p>{ day.Bookings[bookingI].Timestamp.Format("15:04") }</p> | 						{{ | ||||||
| 						<p>{ day.Bookings[bookingI+1].Timestamp.Format("15:04") }</p> | 						workDay, _ := day.(*models.WorkDay) | ||||||
| 						<p>{ day.Bookings[bookingI].BookingType.Name } </p> | 						}} | ||||||
| 					} | 						for bookingI := 0; bookingI < len(workDay.Bookings); bookingI+= 2 { | ||||||
| 					if (day.Absence != models.Absence{}) { | 							<p>{ workDay.Bookings[bookingI].Timestamp.Format("15:04") }</p> | ||||||
| 						<p class="col-span-full">{ day.Absence.AbwesenheitTyp.Name }</p> | 							<p>{ workDay.Bookings[bookingI+1].Timestamp.Format("15:04") }</p> | ||||||
|  | 							<p>{ workDay.Bookings[bookingI].BookingType.Name } </p> | ||||||
|  | 						} | ||||||
|  | 						if workDay.IsKurzArbeit() { | ||||||
|  | 							{{ | ||||||
|  | 								timeFrom, timeTo := workDay.GenerateKurzArbeitBookings(e) | ||||||
|  | 							}} | ||||||
|  | 							<p>{ timeFrom.Format("15:04") }</p> | ||||||
|  | 							<p>{ timeTo.Format("15:04") }</p> | ||||||
|  | 							<p>Kurzarbeit</p> | ||||||
|  | 						} | ||||||
|  | 					} else { | ||||||
|  | 						{{ | ||||||
|  | 						absentDay, _ := day.(*models.Absence) | ||||||
|  | 						}} | ||||||
|  | 						<p class="col-span-full">{ absentDay.AbwesenheitTyp.Name }</p> | ||||||
| 					} | 					} | ||||||
| 				</div> | 				</div> | ||||||
| 				{{ work, pause, overtime := day.GetAllWorkTimes(e) }} | 				{{ work, pause, overtime := day.GetAllWorkTimesVirtual(e) }} | ||||||
| 				@ColorDuration(work, noBorder) | 				@ColorDuration(work, noBorder) | ||||||
| 				@ColorDuration(pause, noBorder) | 				@ColorDuration(pause, noBorder) | ||||||
| 				@ColorDuration(overtime, noBorder + " border-r-0") | 				@ColorDuration(overtime, noBorder+" border-r-0") | ||||||
| 				if day.Day.Weekday() == time.Friday { | 				if day.Date().Weekday() == time.Friday { | ||||||
| 					<p class="col-span-full bg-neutral-300">Wochenende</p> | 					<p class="col-span-full bg-neutral-300">Wochenende</p> | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| @@ -57,144 +123,12 @@ templ PDFReportEmploye(e models.User, workDays []models.WorkDay, tsStart time.Ti | |||||||
| 	</content> | 	</content> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ PDFReportEmployeTable(e models.User, workDays []models.WorkDay, tsStart time.Time, tsEnd time.Time) { | templ ColorDuration(d time.Duration, classes string) { | ||||||
| 	{{ | 	{{ | ||||||
| 		_, kw := tsStart.ISOWeek() |  | ||||||
| 		noBorder := "" |  | ||||||
| 	}} |  | ||||||
| 	@Base() |  | ||||||
| 	<content class="p-8 relative flex flex-col gap-4 break-after-page"> |  | ||||||
| 		<div> |  | ||||||
| 			<h1 class="text-2xl font-bold">Kim Mustermensch</h1> |  | ||||||
| 			<p>Zeitraum: <span>{ tsStart.Format("02.01.2006") }</span> - <span>{ tsEnd.Format("02.01.2006") }</span></p> |  | ||||||
| 			<p>Arbeitszeit: <span></span></p> |  | ||||||
| 			<p>Überstunden: <span></span></p> |  | ||||||
| 		</div> |  | ||||||
| 		<table class="*:*:*:border-1 *:text-center border-1 border-collapse"> |  | ||||||
| 			<tr> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">{ kw }</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Kommen</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Gehen</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Arbeitsart</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Stunden</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Pause</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7 border-r-0">Überstunden</th> |  | ||||||
| 			</tr> |  | ||||||
| 			for index, day := range workDays { |  | ||||||
| 				{{ |  | ||||||
| 					if index == len(workDays)-1 { |  | ||||||
| 						noBorder = "border-b-0" |  | ||||||
| 					} |  | ||||||
| 				}} |  | ||||||
| 				<tr> |  | ||||||
| 					<td class={ noBorder }>{ day.Day.Format("02.01.2006") }</td> |  | ||||||
| 					<td colspan="3"> |  | ||||||
| 				<table  class={ "w-full border-collapse" + noBorder }> |  | ||||||
| 					for bookingI := 0; bookingI < len(day.Bookings); bookingI+= 2 { |  | ||||||
| 						<tr class="flex"> |  | ||||||
| 							<td class="border-r-1 grow">{ day.Bookings[bookingI].Timestamp.Format("15:04") }</td> |  | ||||||
| 							<td class="border-r-1 grow">{ day.Bookings[bookingI+1].Timestamp.Format("15:04") }</td> |  | ||||||
| 							<td class="grow">{ day.Bookings[bookingI].BookingType.Name } </td> |  | ||||||
| 						</tr> |  | ||||||
| 					} |  | ||||||
| 					if (day.Absence != models.Absence{}) { |  | ||||||
| 						if len(day.Bookings) > 0 { |  | ||||||
| 						<tr class="border-t-1"> |  | ||||||
| 							<td colspan="2" class="col-span-full">{ day.Absence.AbwesenheitTyp.Name }</td> |  | ||||||
| 						</tr> |  | ||||||
| 						} |  | ||||||
| 						else { |  | ||||||
| 						<tr> |  | ||||||
| 							<td colspan="2" class="col-span-full">{ day.Absence.AbwesenheitTyp.Name }</td> |  | ||||||
| 						</tr> |  | ||||||
| 						} |  | ||||||
|  |  | ||||||
| 					} |  | ||||||
| 				</table> |  | ||||||
| 					</td> |  | ||||||
| 				{{ work, pause, overtime := day.GetAllWorkTimes(e) }} |  | ||||||
| 				@ColorDuration(work, noBorder) |  | ||||||
| 				@ColorDuration(pause, noBorder) |  | ||||||
| 				@ColorDuration(overtime, noBorder + " border-r-0") |  | ||||||
| 				if day.Day.Weekday() == time.Friday { |  | ||||||
| 					<tr> |  | ||||||
| 					<td colspan="7" class="col-span-full bg-neutral-300">Wochenende</td> |  | ||||||
| 					</tr> |  | ||||||
| 				} |  | ||||||
| 				</tr> |  | ||||||
| 			} |  | ||||||
| 		</table> |  | ||||||
| 	</content> |  | ||||||
| 	<div class="p-8 relative flex flex-col gap-4 break-after-page"> |  | ||||||
| 		<div> |  | ||||||
| 			<h1 class="text-2xl font-bold">Kim Mustermensch</h1> |  | ||||||
| 			<p>Zeitraum: <span>{ tsStart.Format("02.01.2006") }</span> - <span>{ tsEnd.Format("02.01.2006") }</span></p> |  | ||||||
| 			<p>Arbeitszeit: <span></span></p> |  | ||||||
| 			<p>Überstunden: <span></span></p> |  | ||||||
| 		</div> |  | ||||||
| 		<table class="*:*:*:border-1 *:text-center border-1 border-collapse"> |  | ||||||
| 			<tr> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">{ kw }</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Kommen</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Gehen</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Arbeitsart</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Stunden</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7">Pause</th> |  | ||||||
| 				<th class="bg-neutral-300 border-neutral-600 w-1/7 border-r-0">Überstunden</th> |  | ||||||
| 			</tr> |  | ||||||
| 			for index, day := range workDays { |  | ||||||
| 				{{ |  | ||||||
| 					if index == len(workDays)-1 { |  | ||||||
| 						noBorder = "border-b-0" |  | ||||||
| 					} |  | ||||||
| 				}} |  | ||||||
| 				<tr> |  | ||||||
| 					<td class={ noBorder }>{ day.Day.Format("02.01.2006") }</td> |  | ||||||
| 					<td colspan="3"> |  | ||||||
| 				<table  class={ "w-full border-collapse" + noBorder }> |  | ||||||
| 					for bookingI := 0; bookingI < len(day.Bookings); bookingI+= 2 { |  | ||||||
| 						<tr class="flex"> |  | ||||||
| 							<td class="border-r-1 grow">{ day.Bookings[bookingI].Timestamp.Format("15:04") }</td> |  | ||||||
| 							<td class="border-r-1 grow">{ day.Bookings[bookingI+1].Timestamp.Format("15:04") }</td> |  | ||||||
| 							<td class="grow">{ day.Bookings[bookingI].BookingType.Name } </td> |  | ||||||
| 						</tr> |  | ||||||
| 					} |  | ||||||
| 					if (day.Absence != models.Absence{}) { |  | ||||||
| 						if len(day.Bookings) > 0 { |  | ||||||
| 						<tr class="border-t-1"> |  | ||||||
| 							<td colspan="2" class="col-span-full">{ day.Absence.AbwesenheitTyp.Name }</td> |  | ||||||
| 						</tr> |  | ||||||
| 						} |  | ||||||
| 						else { |  | ||||||
| 						<tr> |  | ||||||
| 							<td colspan="2" class="col-span-full">{ day.Absence.AbwesenheitTyp.Name }</td> |  | ||||||
| 						</tr> |  | ||||||
| 						} |  | ||||||
|  |  | ||||||
| 					} |  | ||||||
| 				</table> |  | ||||||
| 					</td> |  | ||||||
| 				{{ work, pause, overtime := day.GetAllWorkTimes(e) }} |  | ||||||
| 				@ColorDuration(work, noBorder) |  | ||||||
| 				@ColorDuration(pause, noBorder) |  | ||||||
| 				@ColorDuration(overtime, noBorder + " border-r-0") |  | ||||||
| 				if day.Day.Weekday() == time.Friday { |  | ||||||
| 					<tr> |  | ||||||
| 					<td colspan="7" class="col-span-full bg-neutral-300">Wochenende</td> |  | ||||||
| 					</tr> |  | ||||||
| 				} |  | ||||||
| 				</tr> |  | ||||||
| 			} |  | ||||||
| 		</table> |  | ||||||
| 	</div> |  | ||||||
| } |  | ||||||
|  |  | ||||||
| templ ColorDuration(d time.Duration, classes string){ |  | ||||||
| {{ |  | ||||||
| 	color := "" | 	color := "" | ||||||
| 	if d.Abs() < time.Minute{ | 	if d.Abs() < time.Minute { | ||||||
| 		color = "text-neutral-300" | 		color = "text-neutral-300" | ||||||
| 	} | 	} | ||||||
| }} | 	}} | ||||||
| <p class={ color + " " + classes }>{ helper.FormatDurationFill(d, true) }</p> | 	<p class={ color + " " + classes }>{ helper.FormatDurationFill(d, true) }</p> | ||||||
| } | } | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										29
									
								
								Backend/templates/presencePage.templ
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								Backend/templates/presencePage.templ
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | |||||||
|  | package templates | ||||||
|  |  | ||||||
|  | import "arbeitszeitmessung/models" | ||||||
|  | import "arbeitszeitmessung/helper" | ||||||
|  |  | ||||||
|  | templ TeamPresencePage(teamPresence map[models.User]bool) { | ||||||
|  | 	@Base() | ||||||
|  | 	@headerComponent() | ||||||
|  | 	<div class="grid-main divide-y-1"> | ||||||
|  | 		<div class="grid-sub divide-x-1 bg-neutral-300"> | ||||||
|  | 			<h2 class="grid-cell font-bold uppercase text-xl">Mitarbeiter</h2> | ||||||
|  | 		</div> | ||||||
|  | 		for user, present := range teamPresence { | ||||||
|  | 			<div class="grid-sub"> | ||||||
|  | 				<div class="grid-cell flex flex-row gap-2 col-span-2 md:col-span-1"> | ||||||
|  | 					@timeGaugeComponent(helper.BoolToInt8(present)*100-1, false) | ||||||
|  | 					<p>{ user.Vorname } { user.Name }</p> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="grid-cell col-span-2"> | ||||||
|  | 					if present { | ||||||
|  | 						<span class="text-neutral-500">Anwesend</span> | ||||||
|  | 					} else { | ||||||
|  | 						<span class="text-neutral-500">Abwesend</span> | ||||||
|  | 					} | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		} | ||||||
|  | 	</div> | ||||||
|  | } | ||||||
							
								
								
									
										110
									
								
								Backend/templates/presencePage_templ.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								Backend/templates/presencePage_templ.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | |||||||
|  | // Code generated by templ - DO NOT EDIT. | ||||||
|  |  | ||||||
|  | // templ: version: v0.3.924 | ||||||
|  | package templates | ||||||
|  |  | ||||||
|  | //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||||
|  |  | ||||||
|  | import "github.com/a-h/templ" | ||||||
|  | import templruntime "github.com/a-h/templ/runtime" | ||||||
|  |  | ||||||
|  | import "arbeitszeitmessung/models" | ||||||
|  | import "arbeitszeitmessung/helper" | ||||||
|  |  | ||||||
|  | func TeamPresencePage(teamPresence map[models.User]bool) templ.Component { | ||||||
|  | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
|  | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
|  | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
|  | 			return templ_7745c5c3_CtxErr | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||||
|  | 		if !templ_7745c5c3_IsBuffer { | ||||||
|  | 			defer func() { | ||||||
|  | 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||||
|  | 				if templ_7745c5c3_Err == nil { | ||||||
|  | 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||||
|  | 				} | ||||||
|  | 			}() | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.InitializeContext(ctx) | ||||||
|  | 		templ_7745c5c3_Var1 := templ.GetChildren(ctx) | ||||||
|  | 		if templ_7745c5c3_Var1 == nil { | ||||||
|  | 			templ_7745c5c3_Var1 = templ.NopComponent | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.ClearChildren(ctx) | ||||||
|  | 		templ_7745c5c3_Err = Base().Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = headerComponent().Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<div class=\"grid-main divide-y-1\"><div class=\"grid-sub divide-x-1 bg-neutral-300\"><h2 class=\"grid-cell font-bold uppercase text-xl\">Mitarbeiter</h2></div>") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		for user, present := range teamPresence { | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "<div class=\"grid-sub\"><div class=\"grid-cell flex flex-row gap-2 col-span-2 md:col-span-1\">") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = timeGaugeComponent(helper.BoolToInt8(present)*100-1, false).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "<p>") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			var templ_7745c5c3_Var2 string | ||||||
|  | 			templ_7745c5c3_Var2, templ_7745c5c3_Err = templ.JoinStringErrs(user.Vorname) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/presencePage.templ`, Line: 17, Col: 22} | ||||||
|  | 			} | ||||||
|  | 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var2)) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, " ") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			var templ_7745c5c3_Var3 string | ||||||
|  | 			templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(user.Name) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/presencePage.templ`, Line: 17, Col: 36} | ||||||
|  | 			} | ||||||
|  | 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3)) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "</p></div><div class=\"grid-cell col-span-2\">") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			if present { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "<span class=\"text-neutral-500\">Anwesend</span>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} else { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "<span class=\"text-neutral-500\">Abwesend</span>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, "</div></div>") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "</div>") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var _ = templruntime.GeneratedTemplate | ||||||
| @@ -26,80 +26,140 @@ templ weekPicker(weekStart time.Time) { | |||||||
| 			</svg> | 			</svg> | ||||||
| 		</button> | 		</button> | ||||||
| 	</form> | 	</form> | ||||||
| 	if time.Since(weekStart) < 24*7*time.Hour { |  | ||||||
| 		<p class="text-sm text-red-500">Die Woche kann erst am nächsten Montag gesendet werden!</p> |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
|  |  | ||||||
| templ weekDayComponent(user models.User, day models.WorkDay) { | templ defaultWeekDayComponent(u models.User, day models.IWorkDay) { | ||||||
| 	{{ work, pause := day.GetWorkTimeString() }} |  | ||||||
| 	<div class="flex flex-row gap-2"> | 	<div class="flex flex-row gap-2"> | ||||||
| 		@timeGaugeComponent(day.GetWorkDayProgress(user), false, day.RequiresAction()) | 		@timeGaugeComponent(day.GetDayProgress(u), false) | ||||||
| 		<div class="flex flex-col"> | 		<div class="flex flex-col"> | ||||||
| 			<p class=""><span class="font-bold uppercase hidden md:inline">{ day.Day.Format("Mon") }:</span> { day.Day.Format("02.01.2006") }</p> | 			<p class=""><span class="font-bold uppercase hidden md:inline">{ day.Date().Format("Mon") }:</span> { day.Date().Format("02.01.2006") }</p> | ||||||
| 			if !day.RequiresAction() { | 			if day.IsWorkDay() { | ||||||
| 				<div class="flex flex-row gap-2"> | 				{{ | ||||||
| 					<span class="text-accent">{ work }</span> | 				workDay, _ := day.(*models.WorkDay) | ||||||
| 					<span class="text-neutral-500">{ pause }</span> | 				work, pause, _ := workDay.GetAllWorkTimesReal(u) | ||||||
| 				</div> | 				}} | ||||||
| 				<div class="flex flex-row gap-2 items-center"> | 				if !workDay.RequiresAction() { | ||||||
| 					<svg xmlns="http://www.w3.org/2000/svg" fill="currentColor" class="size-4" viewBox="0 0 16 16"> | 					<div class="flex flex-row gap-2"> | ||||||
| 						<path d="M8 3.5a.5.5 0 0 0-1 0V9a.5.5 0 0 0 .252.434l3.5 2a.5.5 0 0 0 .496-.868L8 8.71z"></path> | 						<span class="text-accent">{ helper.FormatDuration(work) }</span> | ||||||
| 						<path d="M8 16A8 8 0 1 0 8 0a8 8 0 0 0 0 16m7-8A7 7 0 1 1 1 8a7 7 0 0 1 14 0"></path> | 						<span class="text-neutral-500">{ helper.FormatDuration(pause) }</span> | ||||||
| 					</svg> | 					</div> | ||||||
| 					switch  { | 					<div class="flex flex-row gap-2 items-center"> | ||||||
| 						case day.Absence.Datum.Equal(day.Day): | 						<span class="icon-[material-symbols-light--schedule-outline] flex-shrink-0"></span> | ||||||
| 							<p>{ day.Absence.AbwesenheitTyp.Name }</p> | 						switch  { | ||||||
| 						case !day.TimeFrom.Equal(day.TimeTo): | 							case !workDay.TimeFrom.Equal(workDay.TimeTo): | ||||||
| 							<span>{ day.TimeFrom.Format("15:04") }</span> | 								<span>{ workDay.TimeFrom.Format("15:04") }</span> | ||||||
| 							<span>-</span> | 								<span>-</span> | ||||||
| 							<span>{ day.TimeTo.Format("15:04") }</span> | 								<span>{ workDay.TimeTo.Format("15:04") }</span> | ||||||
| 						default: | 							default: | ||||||
| 							<p>Keine Anwesenheit</p> | 								<p>Keine Anwesenheit</p> | ||||||
| 					} | 						} | ||||||
| 				</div> | 					</div> | ||||||
|  | 				} else { | ||||||
|  | 					<p class="text-red-600">Bitte anpassen</p> | ||||||
|  | 				} | ||||||
| 			} else { | 			} else { | ||||||
| 				<p class="text-red-600">Bitte anpassen</p> | 				{{ | ||||||
|  | 					absentDay, _ := day.(*models.Absence) | ||||||
|  | 				}} | ||||||
|  | 				<div>{ absentDay.AbwesenheitTyp.Name } </div> | ||||||
| 			} | 			} | ||||||
| 		</div> | 		</div> | ||||||
| 	</div> | 	</div> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ employeComponent(week models.WorkWeek) { | templ weekDayComponent(user models.User, day models.WorkDay) { | ||||||
|  | 	// {{ work, pause, _ := day.GetAllWorkTimesReal(user) }} | ||||||
|  | 	<div class="flex flex-row gap-2"> | ||||||
|  | 		// @timeGaugeComponent(day.GetWorkDayProgress(user), false, day.RequiresAction()) | ||||||
|  | 		<div class="flex flex-col"> | ||||||
|  | 			if !day.RequiresAction() { | ||||||
|  | 			} | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | } | ||||||
|  |  | ||||||
|  | templ workWeekComponent(week models.WorkWeek, onlyAccept bool) { | ||||||
| 	{{ | 	{{ | ||||||
| 		year, kw := week.WeekStart.ISOWeek() | 		year, kw := week.WeekStart.ISOWeek() | ||||||
| 		progress := (float32(week.Worktime.Hours()) / week.User.ArbeitszeitPerWoche) * 100 | 		progress := (float32(week.WorkTimeVirtual.Hours()) / week.User.ArbeitszeitPerWoche) * 100 | ||||||
| 	}} | 	}} | ||||||
| 	<div class="employeComponent grid-sub responsive lg:divide-x-1 max-md:divide-y-1 @container"> | 	<div class="employeComponent grid-sub responsive lg:divide-x-1 max-md:divide-y-1 @container"> | ||||||
| 		<div class="grid-cell flex flex-col max-md:bg-neutral-300 gap-2"> | 		<div class="grid-cell flex flex-col max-md:bg-neutral-300 gap-2"> | ||||||
|  | 			if !onlyAccept { | ||||||
|  | 				<div class="lg:hidden"> | ||||||
|  | 					@weekPicker(week.WeekStart) | ||||||
|  | 				</div> | ||||||
|  | 			} | ||||||
| 			<p class="font-bold uppercase">{ week.User.Vorname } { week.User.Name }</p> | 			<p class="font-bold uppercase">{ week.User.Vorname } { week.User.Name }</p> | ||||||
| 			<div class="flex flex-row gap-2 col-span-3"> | 			<div class="grid grid-cols-5 gap-2 lg:grid-cols-1"> | ||||||
| 				@timeGaugeComponent(uint8(progress), false, false) | 				if !onlyAccept { | ||||||
| 				<div> | 					<div class="col-span-2"> | ||||||
| 					<p>Arbeitszeit: { fmt.Sprintf("%s", helper.FormatDuration(week.Worktime)) }</p> | 						<span class="flex flex-row gap-2 items-center"> | ||||||
| 					<p>Überstunden: { fmt.Sprintf("%s", helper.FormatDuration(week.Overtime)) }</p> | 							@statusCheckMark(week.CheckStatus(), models.WeekStatusSent) | ||||||
|  | 							Gesendet | ||||||
|  | 						</span> | ||||||
|  | 						<span class="flex flex-row gap-2 items-center"> | ||||||
|  | 							@statusCheckMark(week.CheckStatus(), models.WeekStatusAccepted) | ||||||
|  | 							Akzeptiert | ||||||
|  | 						</span> | ||||||
|  | 					</div> | ||||||
|  | 				} | ||||||
|  | 				<div class="flex flex-row gap-2 col-span-3"> | ||||||
|  | 					@timeGaugeComponent(int8(progress), false) | ||||||
|  | 					<div> | ||||||
|  | 						<p>Arbeitszeit: { fmt.Sprintf("%s", helper.FormatDuration(week.Worktime)) }</p> | ||||||
|  | 						<p>Überstunden: { fmt.Sprintf("%s", helper.FormatDurationFill(week.Overtime, true)) }</p> | ||||||
|  | 					</div> | ||||||
| 				</div> | 				</div> | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
| 		<div class="grid-cell col-span-3 flex flex-col @7xl:grid @7xl:grid-cols-5 gap-2 content-baseline"> | 		<div class="grid-cell col-span-3 flex flex-col @7xl:grid @7xl:grid-cols-5 gap-2 py-4 content-baseline"> | ||||||
| 			for _, day := range week.WorkDays { | 			for _, day := range week.Days { | ||||||
| 				@weekDayComponent(week.User, day) | 				@defaultWeekDayComponent(week.User, day) | ||||||
| 			} | 			} | ||||||
| 		</div> | 		</div> | ||||||
| 		<form class="grid-cell flex flex-col justify-between gap-2" method="post"> | 		<div class="grid-cell flex flex-col gap-2 justify-between"> | ||||||
| 			<p class="text-sm"><span class="">Woche:</span> { fmt.Sprintf("%02d-%d", kw, year) }</p> | 			if onlyAccept { | ||||||
| 			<input type="hidden" name="method" value="accept"/> | 				<p class="text-sm"><span class="">Woche:</span> { fmt.Sprintf("%02d-%d", kw, year) }</p> | ||||||
| 			<input type="hidden" name="user" value={ strconv.Itoa(week.User.PersonalNummer) }/> | 			} else { | ||||||
| 			<input type="hidden" name="week" value={ week.WeekStart.Format(time.DateOnly) }/> | 				<div class="max-md:hidden"> | ||||||
| 			<div class="flex flex-col gap-2"> | 					@weekPicker(week.WeekStart) | ||||||
| 				if week.Status == models.WeekStatusDifferences { | 				</div> | ||||||
| 					<p class="text-red-600 text-sm">Unterschiedliche Arbeitszeit zwischen Abrechnung und individuellen Buchungen</p> | 			} | ||||||
|  | 			<form method="post" class="flex flex-col gap-2"> | ||||||
|  | 				{{ | ||||||
|  | 					week.CheckStatus() | ||||||
|  | 					method := "accept" | ||||||
|  | 					if !onlyAccept { | ||||||
|  | 						method = "send" | ||||||
|  | 					} | ||||||
|  | 				}} | ||||||
|  | 				<input type="hidden" name="method" value={ method }/> | ||||||
|  | 				<input type="hidden" name="user" value={ strconv.Itoa(week.User.PersonalNummer) }/> | ||||||
|  | 				<input type="hidden" name="week" value={ week.WeekStart.Format(time.DateOnly) }/> | ||||||
|  | 				if onlyAccept { | ||||||
|  | 					if week.Status == models.WeekStatusDifferences { | ||||||
|  | 						<p class="text-red-600 text-sm">Unterschiedliche Arbeitszeit zwischen Abrechnung und individuellen Buchungen</p> | ||||||
|  | 					} | ||||||
|  | 					<button type="submit" disabled?={ week.Status == models.WeekStatusDifferences } class="btn">Bestätigen</button> | ||||||
|  | 				} else { | ||||||
|  | 					switch  { | ||||||
|  | 						case week.RequiresAction(): | ||||||
|  | 							<p class="text-sm text-red-500">bitte zuerst Buchungen anpassen</p> | ||||||
|  | 						case time.Since(week.WeekStart) < 24*7*time.Hour: | ||||||
|  | 							<p class="text-sm text-red-500">Die Woche kann erst am nächsten Montag gesendet werden!</p> | ||||||
|  | 						case week.Status == models.WeekStatusNone: | ||||||
|  | 							<p class="text-sm">an Vorgesetzten senden</p> | ||||||
|  | 						case week.Status == models.WeekStatusSent: | ||||||
|  | 							<p class="text-sm">an Vorgesetzten gesendet</p> | ||||||
|  | 						case week.Status == models.WeekStatusAccepted: | ||||||
|  | 							<p class="text-sm">vom Vorgesetzten bestätigt</p> | ||||||
|  | 					} | ||||||
|  | 					<button disabled?={ week.Status < models.WeekStatusSent } type="submit" class="btn">Korrigieren</button> | ||||||
|  | 					<button disabled?={ time.Since(week.WeekStart) < 24*7*time.Hour || week.Status >= models.WeekStatusSent || week.RequiresAction() } type="submit" class="btn">Senden</button> | ||||||
| 				} | 				} | ||||||
| 				<button type="submit" disabled?={ week.Status == models.WeekStatusDifferences } class="btn">Bestätigen</button> | 			</form> | ||||||
| 				// TODO maybe delete function | 		</div> | ||||||
| 				// <button type="button" disabled?={ week.Status < models.WeekStatusDifferences } class="hover:bg-red-600 btn">Antrag löschen</button> |  | ||||||
| 			</div> |  | ||||||
| 		</form> |  | ||||||
| 	</div> | 	</div> | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| // Code generated by templ - DO NOT EDIT. | // Code generated by templ - DO NOT EDIT. | ||||||
|  |  | ||||||
| // templ: version: v0.3.943 | // templ: version: v0.3.924 | ||||||
| package templates | package templates | ||||||
|  |  | ||||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||||
| @@ -113,11 +113,176 @@ func weekPicker(weekStart time.Time) templ.Component { | |||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		if time.Since(weekStart) < 24*7*time.Hour { | 		return nil | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "<p class=\"text-sm text-red-500\">Die Woche kann erst am nächsten Montag gesendet werden!</p>") | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func defaultWeekDayComponent(u models.User, day models.IWorkDay) templ.Component { | ||||||
|  | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
|  | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
|  | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
|  | 			return templ_7745c5c3_CtxErr | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||||
|  | 		if !templ_7745c5c3_IsBuffer { | ||||||
|  | 			defer func() { | ||||||
|  | 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||||
|  | 				if templ_7745c5c3_Err == nil { | ||||||
|  | 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||||
|  | 				} | ||||||
|  | 			}() | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.InitializeContext(ctx) | ||||||
|  | 		templ_7745c5c3_Var6 := templ.GetChildren(ctx) | ||||||
|  | 		if templ_7745c5c3_Var6 == nil { | ||||||
|  | 			templ_7745c5c3_Var6 = templ.NopComponent | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.ClearChildren(ctx) | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "<div class=\"flex flex-row gap-2\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = timeGaugeComponent(day.GetDayProgress(u), false).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, "<div class=\"flex flex-col\"><p class=\"\"><span class=\"font-bold uppercase hidden md:inline\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var7 string | ||||||
|  | 		templ_7745c5c3_Var7, templ_7745c5c3_Err = templ.JoinStringErrs(day.Date().Format("Mon")) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 35, Col: 92} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var7)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 12, ":</span> ") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var8 string | ||||||
|  | 		templ_7745c5c3_Var8, templ_7745c5c3_Err = templ.JoinStringErrs(day.Date().Format("02.01.2006")) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 35, Col: 136} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var8)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, "</p>") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		if day.IsWorkDay() { | ||||||
|  |  | ||||||
|  | 			workDay, _ := day.(*models.WorkDay) | ||||||
|  | 			work, pause, _ := workDay.GetAllWorkTimesReal(u) | ||||||
|  | 			if !workDay.RequiresAction() { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 14, "<div class=\"flex flex-row gap-2\"><span class=\"text-accent\">") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 				var templ_7745c5c3_Var9 string | ||||||
|  | 				templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(helper.FormatDuration(work)) | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 43, Col: 61} | ||||||
|  | 				} | ||||||
|  | 				_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var9)) | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, "</span> <span class=\"text-neutral-500\">") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 				var templ_7745c5c3_Var10 string | ||||||
|  | 				templ_7745c5c3_Var10, templ_7745c5c3_Err = templ.JoinStringErrs(helper.FormatDuration(pause)) | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 44, Col: 67} | ||||||
|  | 				} | ||||||
|  | 				_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var10)) | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "</span></div><div class=\"flex flex-row gap-2 items-center\"><span class=\"icon-[material-symbols-light--schedule-outline] flex-shrink-0\"></span> ") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 				switch { | ||||||
|  | 				case !workDay.TimeFrom.Equal(workDay.TimeTo): | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "<span>") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					var templ_7745c5c3_Var11 string | ||||||
|  | 					templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(workDay.TimeFrom.Format("15:04")) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 50, Col: 48} | ||||||
|  | 					} | ||||||
|  | 					_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11)) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 18, "</span> <span>-</span> <span>") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					var templ_7745c5c3_Var12 string | ||||||
|  | 					templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(workDay.TimeTo.Format("15:04")) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 52, Col: 46} | ||||||
|  | 					} | ||||||
|  | 					_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12)) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "</span>") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 				default: | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "<p>Keine Anwesenheit</p>") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, "</div>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} else { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 22, "<p class=\"text-red-600\">Bitte anpassen</p>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  |  | ||||||
|  | 			absentDay, _ := day.(*models.Absence) | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, "<div>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
|  | 			var templ_7745c5c3_Var13 string | ||||||
|  | 			templ_7745c5c3_Var13, templ_7745c5c3_Err = templ.JoinStringErrs(absentDay.AbwesenheitTyp.Name) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 64, Col: 40} | ||||||
|  | 			} | ||||||
|  | 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var13)) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, "</div>") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, "</div></div>") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		return nil | 		return nil | ||||||
| 	}) | 	}) | ||||||
| @@ -139,148 +304,18 @@ func weekDayComponent(user models.User, day models.WorkDay) templ.Component { | |||||||
| 			}() | 			}() | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.InitializeContext(ctx) | 		ctx = templ.InitializeContext(ctx) | ||||||
| 		templ_7745c5c3_Var6 := templ.GetChildren(ctx) | 		templ_7745c5c3_Var14 := templ.GetChildren(ctx) | ||||||
| 		if templ_7745c5c3_Var6 == nil { | 		if templ_7745c5c3_Var14 == nil { | ||||||
| 			templ_7745c5c3_Var6 = templ.NopComponent | 			templ_7745c5c3_Var14 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		work, pause := day.GetWorkTimeString() | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, "<div class=\"flex flex-row gap-2\"><div class=\"flex flex-col\">") | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, "<div class=\"flex flex-row gap-2\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = timeGaugeComponent(day.GetWorkDayProgress(user), false, day.RequiresAction()).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 12, "<div class=\"flex flex-col\"><p class=\"\"><span class=\"font-bold uppercase hidden md:inline\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		var templ_7745c5c3_Var7 string |  | ||||||
| 		templ_7745c5c3_Var7, templ_7745c5c3_Err = templ.JoinStringErrs(day.Day.Format("Mon")) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 39, Col: 89} |  | ||||||
| 		} |  | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var7)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, ":</span> ") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		var templ_7745c5c3_Var8 string |  | ||||||
| 		templ_7745c5c3_Var8, templ_7745c5c3_Err = templ.JoinStringErrs(day.Day.Format("02.01.2006")) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 39, Col: 130} |  | ||||||
| 		} |  | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var8)) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 14, "</p>") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		if !day.RequiresAction() { | 		if !day.RequiresAction() { | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, "<div class=\"flex flex-row gap-2\"><span class=\"text-accent\">") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 			var templ_7745c5c3_Var9 string |  | ||||||
| 			templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(work) |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 42, Col: 37} |  | ||||||
| 			} |  | ||||||
| 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var9)) |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "</span> <span class=\"text-neutral-500\">") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 			var templ_7745c5c3_Var10 string |  | ||||||
| 			templ_7745c5c3_Var10, templ_7745c5c3_Err = templ.JoinStringErrs(pause) |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 43, Col: 43} |  | ||||||
| 			} |  | ||||||
| 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var10)) |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "</span></div><div class=\"flex flex-row gap-2 items-center\"><svg xmlns=\"http://www.w3.org/2000/svg\" fill=\"currentColor\" class=\"size-4\" viewBox=\"0 0 16 16\"><path d=\"M8 3.5a.5.5 0 0 0-1 0V9a.5.5 0 0 0 .252.434l3.5 2a.5.5 0 0 0 .496-.868L8 8.71z\"></path> <path d=\"M8 16A8 8 0 1 0 8 0a8 8 0 0 0 0 16m7-8A7 7 0 1 1 1 8a7 7 0 0 1 14 0\"></path></svg> ") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 			switch { |  | ||||||
| 			case day.Absence.Datum.Equal(day.Day): |  | ||||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 18, "<p>") |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 				var templ_7745c5c3_Var11 string |  | ||||||
| 				templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(day.Absence.AbwesenheitTyp.Name) |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 52, Col: 43} |  | ||||||
| 				} |  | ||||||
| 				_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11)) |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "</p>") |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 			case !day.TimeFrom.Equal(day.TimeTo): |  | ||||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "<span>") |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 				var templ_7745c5c3_Var12 string |  | ||||||
| 				templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(day.TimeFrom.Format("15:04")) |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 54, Col: 43} |  | ||||||
| 				} |  | ||||||
| 				_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12)) |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, "</span> <span>-</span> <span>") |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 				var templ_7745c5c3_Var13 string |  | ||||||
| 				templ_7745c5c3_Var13, templ_7745c5c3_Err = templ.JoinStringErrs(day.TimeTo.Format("15:04")) |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 56, Col: 41} |  | ||||||
| 				} |  | ||||||
| 				_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var13)) |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 22, "</span>") |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 			default: |  | ||||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, "<p>Keine Anwesenheit</p>") |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, "</div>") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, "<p class=\"text-red-600\">Bitte anpassen</p>") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, "</div></div>") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 27, "</div></div>") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| @@ -288,7 +323,7 @@ func weekDayComponent(user models.User, day models.WorkDay) templ.Component { | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func employeComponent(week models.WorkWeek) templ.Component { | func workWeekComponent(week models.WorkWeek, onlyAccept bool) templ.Component { | ||||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
| @@ -304,144 +339,292 @@ func employeComponent(week models.WorkWeek) templ.Component { | |||||||
| 			}() | 			}() | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.InitializeContext(ctx) | 		ctx = templ.InitializeContext(ctx) | ||||||
| 		templ_7745c5c3_Var14 := templ.GetChildren(ctx) | 		templ_7745c5c3_Var15 := templ.GetChildren(ctx) | ||||||
| 		if templ_7745c5c3_Var14 == nil { | 		if templ_7745c5c3_Var15 == nil { | ||||||
| 			templ_7745c5c3_Var14 = templ.NopComponent | 			templ_7745c5c3_Var15 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
|  |  | ||||||
| 		year, kw := week.WeekStart.ISOWeek() | 		year, kw := week.WeekStart.ISOWeek() | ||||||
| 		progress := (float32(week.Worktime.Hours()) / week.User.ArbeitszeitPerWoche) * 100 | 		progress := (float32(week.WorkTimeVirtual.Hours()) / week.User.ArbeitszeitPerWoche) * 100 | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 27, "<div class=\"employeComponent grid-sub responsive lg:divide-x-1 max-md:divide-y-1 @container\"><div class=\"grid-cell flex flex-col max-md:bg-neutral-300 gap-2\"><p class=\"font-bold uppercase\">") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, "<div class=\"employeComponent grid-sub responsive lg:divide-x-1 max-md:divide-y-1 @container\"><div class=\"grid-cell flex flex-col max-md:bg-neutral-300 gap-2\">") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var15 string | 		if !onlyAccept { | ||||||
| 		templ_7745c5c3_Var15, templ_7745c5c3_Err = templ.JoinStringErrs(week.User.Vorname) | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, "<div class=\"lg:hidden\">") | ||||||
| 		if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 75, Col: 53} | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = weekPicker(week.WeekStart).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 30, "</div>") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var15)) | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 31, "<p class=\"font-bold uppercase\">") | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, " ") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var16 string | 		var templ_7745c5c3_Var16 string | ||||||
| 		templ_7745c5c3_Var16, templ_7745c5c3_Err = templ.JoinStringErrs(week.User.Name) | 		templ_7745c5c3_Var16, templ_7745c5c3_Err = templ.JoinStringErrs(week.User.Vorname) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 75, Col: 72} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 93, Col: 53} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var16)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var16)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, "</p><div class=\"flex flex-row gap-2 col-span-3\">") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 32, " ") | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = timeGaugeComponent(uint8(progress), false, false).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 30, "<div><p>Arbeitszeit: ") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var17 string | 		var templ_7745c5c3_Var17 string | ||||||
| 		templ_7745c5c3_Var17, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%s", helper.FormatDuration(week.Worktime))) | 		templ_7745c5c3_Var17, templ_7745c5c3_Err = templ.JoinStringErrs(week.User.Name) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 79, Col: 78} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 93, Col: 72} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var17)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var17)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 31, "</p><p>Überstunden: ") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 33, "</p><div class=\"grid grid-cols-5 gap-2 lg:grid-cols-1\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		if !onlyAccept { | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 34, "<div class=\"col-span-2\"><span class=\"flex flex-row gap-2 items-center\">") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = statusCheckMark(week.CheckStatus(), models.WeekStatusSent).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 35, "Gesendet</span> <span class=\"flex flex-row gap-2 items-center\">") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = statusCheckMark(week.CheckStatus(), models.WeekStatusAccepted).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 36, "Akzeptiert</span></div>") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 37, "<div class=\"flex flex-row gap-2 col-span-3\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = timeGaugeComponent(int8(progress), false).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 38, "<div><p>Arbeitszeit: ") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var18 string | 		var templ_7745c5c3_Var18 string | ||||||
| 		templ_7745c5c3_Var18, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%s", helper.FormatDuration(week.Overtime))) | 		templ_7745c5c3_Var18, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%s", helper.FormatDuration(week.Worktime))) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 80, Col: 79} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 110, Col: 79} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var18)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var18)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 32, "</p></div></div></div><div class=\"grid-cell col-span-3 flex flex-col @7xl:grid @7xl:grid-cols-5 gap-2 content-baseline\">") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 39, "</p><p>Überstunden: ") | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		for _, day := range week.WorkDays { |  | ||||||
| 			templ_7745c5c3_Err = weekDayComponent(week.User, day).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 33, "</div><form class=\"grid-cell flex flex-col justify-between gap-2\" method=\"post\"><p class=\"text-sm\"><span class=\"\">Woche:</span> ") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var19 string | 		var templ_7745c5c3_Var19 string | ||||||
| 		templ_7745c5c3_Var19, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%02d-%d", kw, year)) | 		templ_7745c5c3_Var19, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%s", helper.FormatDurationFill(week.Overtime, true))) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 90, Col: 85} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 111, Col: 90} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var19)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var19)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 34, "</p><input type=\"hidden\" name=\"method\" value=\"accept\"> <input type=\"hidden\" name=\"user\" value=\"") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 40, "</p></div></div></div></div><div class=\"grid-cell col-span-3 flex flex-col @7xl:grid @7xl:grid-cols-5 gap-2 py-4 content-baseline\">") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var20 string | 		for _, day := range week.Days { | ||||||
| 		templ_7745c5c3_Var20, templ_7745c5c3_Err = templ.JoinStringErrs(strconv.Itoa(week.User.PersonalNummer)) | 			templ_7745c5c3_Err = defaultWeekDayComponent(week.User, day).Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 		if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 92, Col: 82} | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var20)) | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 41, "</div><div class=\"grid-cell flex flex-col gap-2 justify-between\">") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 35, "\"> <input type=\"hidden\" name=\"week\" value=\"") | 		if onlyAccept { | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 42, "<p class=\"text-sm\"><span class=\"\">Woche:</span> ") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			var templ_7745c5c3_Var20 string | ||||||
|  | 			templ_7745c5c3_Var20, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%02d-%d", kw, year)) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 123, Col: 86} | ||||||
|  | 			} | ||||||
|  | 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var20)) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 43, "</p>") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 44, "<div class=\"max-md:hidden\">") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = weekPicker(week.WeekStart).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 45, "</div>") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 46, "<form method=\"post\" class=\"flex flex-col gap-2\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		week.CheckStatus() | ||||||
|  | 		method := "accept" | ||||||
|  | 		if !onlyAccept { | ||||||
|  | 			method = "send" | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 47, "<input type=\"hidden\" name=\"method\" value=\"") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var21 string | 		var templ_7745c5c3_Var21 string | ||||||
| 		templ_7745c5c3_Var21, templ_7745c5c3_Err = templ.JoinStringErrs(week.WeekStart.Format(time.DateOnly)) | 		templ_7745c5c3_Var21, templ_7745c5c3_Err = templ.JoinStringErrs(method) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 93, Col: 80} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 137, Col: 53} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var21)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var21)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 36, "\"><div class=\"flex flex-col gap-2\">") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 48, "\"> <input type=\"hidden\" name=\"user\" value=\"") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		if week.Status == models.WeekStatusDifferences { | 		var templ_7745c5c3_Var22 string | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 37, "<p class=\"text-red-600 text-sm\">Unterschiedliche Arbeitszeit zwischen Abrechnung und individuellen Buchungen</p>") | 		templ_7745c5c3_Var22, templ_7745c5c3_Err = templ.JoinStringErrs(strconv.Itoa(week.User.PersonalNummer)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 138, Col: 83} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var22)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 49, "\"> <input type=\"hidden\" name=\"week\" value=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var23 string | ||||||
|  | 		templ_7745c5c3_Var23, templ_7745c5c3_Err = templ.JoinStringErrs(week.WeekStart.Format(time.DateOnly)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 139, Col: 81} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var23)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 50, "\"> ") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		if onlyAccept { | ||||||
|  | 			if week.Status == models.WeekStatusDifferences { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 51, "<p class=\"text-red-600 text-sm\">Unterschiedliche Arbeitszeit zwischen Abrechnung und individuellen Buchungen</p>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 52, " <button type=\"submit\"") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			if week.Status == models.WeekStatusDifferences { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 53, " disabled") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 54, " class=\"btn\">Bestätigen</button>") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			switch { | ||||||
|  | 			case week.RequiresAction(): | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 55, "<p class=\"text-sm text-red-500\">bitte zuerst Buchungen anpassen</p>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			case time.Since(week.WeekStart) < 24*7*time.Hour: | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 56, "<p class=\"text-sm text-red-500\">Die Woche kann erst am nächsten Montag gesendet werden!</p>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			case week.Status == models.WeekStatusNone: | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 57, "<p class=\"text-sm\">an Vorgesetzten senden</p>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			case week.Status == models.WeekStatusSent: | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 58, "<p class=\"text-sm\">an Vorgesetzten gesendet</p>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			case week.Status == models.WeekStatusAccepted: | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 59, "<p class=\"text-sm\">vom Vorgesetzten bestätigt</p>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 60, " <button") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			if week.Status < models.WeekStatusSent { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 61, " disabled") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 62, " type=\"submit\" class=\"btn\">Korrigieren</button> <button") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			if time.Since(week.WeekStart) < 24*7*time.Hour || week.Status >= models.WeekStatusSent || week.RequiresAction() { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 63, " disabled") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 64, " type=\"submit\" class=\"btn\">Senden</button>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 38, "<button type=\"submit\"") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 65, "</form></div></div>") | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		if week.Status == models.WeekStatusDifferences { |  | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 39, " disabled") |  | ||||||
| 			if templ_7745c5c3_Err != nil { |  | ||||||
| 				return templ_7745c5c3_Err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 40, " class=\"btn\">Bestätigen</button></div></form></div>") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| @@ -465,53 +648,53 @@ func userPresenceComponent(user models.User, present bool) templ.Component { | |||||||
| 			}() | 			}() | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.InitializeContext(ctx) | 		ctx = templ.InitializeContext(ctx) | ||||||
| 		templ_7745c5c3_Var22 := templ.GetChildren(ctx) | 		templ_7745c5c3_Var24 := templ.GetChildren(ctx) | ||||||
| 		if templ_7745c5c3_Var22 == nil { | 		if templ_7745c5c3_Var24 == nil { | ||||||
| 			templ_7745c5c3_Var22 = templ.NopComponent | 			templ_7745c5c3_Var24 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 41, "<div class=\"grid-cell group flex flex-row gap-2\">") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 66, "<div class=\"grid-cell group flex flex-row gap-2\">") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		if present { | 		if present { | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 42, "<div class=\"h-8 bg-accent rounded-md group-hover:text-black md:text-transparent text-center p-1\">Anwesend</div>") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 67, "<div class=\"h-8 bg-accent rounded-md group-hover:text-black md:text-transparent text-center p-1\">Anwesend</div>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 43, "<div class=\"h-8 bg-red-600 rounded-md group-hover:text-white md:text-transparent text-center p-1\">Abwesend</div>") | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 68, "<div class=\"h-8 bg-red-600 rounded-md group-hover:text-white md:text-transparent text-center p-1\">Abwesend</div>") | ||||||
| 			if templ_7745c5c3_Err != nil { | 			if templ_7745c5c3_Err != nil { | ||||||
| 				return templ_7745c5c3_Err | 				return templ_7745c5c3_Err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 44, "<p>") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 69, "<p>") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var23 string | 		var templ_7745c5c3_Var25 string | ||||||
| 		templ_7745c5c3_Var23, templ_7745c5c3_Err = templ.JoinStringErrs(user.Vorname) | 		templ_7745c5c3_Var25, templ_7745c5c3_Err = templ.JoinStringErrs(user.Vorname) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 113, Col: 19} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 173, Col: 19} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var23)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var25)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 45, " ") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 70, " ") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var24 string | 		var templ_7745c5c3_Var26 string | ||||||
| 		templ_7745c5c3_Var24, templ_7745c5c3_Err = templ.JoinStringErrs(user.Name) | 		templ_7745c5c3_Var26, templ_7745c5c3_Err = templ.JoinStringErrs(user.Name) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 113, Col: 33} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/teamComponents.templ`, Line: 173, Col: 33} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var24)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var26)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 46, "</p></div>") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 71, "</p></div>") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -1,115 +1,41 @@ | |||||||
| package templates | package templates | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"arbeitszeitmessung/helper" |  | ||||||
| 	"arbeitszeitmessung/models" | 	"arbeitszeitmessung/models" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"net/url" |  | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| templ inputForm() { | templ lineComponent() { | ||||||
| 	{{ | 	<div class="flex flex-col w-2 py-2 items-center text-accent print:hidden"> | ||||||
| 		urlParams := ctx.Value("urlParams").(url.Values) | 		<svg class="size-2" viewBox="0 0 24 24" fill="currentColor"> | ||||||
| 		user := ctx.Value("user").(models.User) | 			<polygon points="12,2 22,12 12,22 2,12"></polygon> | ||||||
| 	}} | 		</svg> | ||||||
| 	<div class="grid-sub divide-x-1 bg-neutral-300 max-md:flex max-md:flex-col"> | 		<div class="w-[2px] bg-accent flex-grow -my-1"></div> | ||||||
| 		<div class="grid-cell md:col-span-1 max-md:grid grid-cols-2"> | 		<svg class="size-2" viewBox="0 0 24 24" fill="currentColor"> | ||||||
| 			<p class="font-bold uppercase">{ user.Vorname + " " + user.Name }</p> | 			<polygon points="12,2 22,12 12,22 2,12"></polygon> | ||||||
| 			<div class="justify-self-end"> | 		</svg> | ||||||
| 				<p class="text-sm">Überstunden</p> |  | ||||||
| 				<p class="text-accent">{ user.Overtime }</p> |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 		<form id="timeRangeForm" method="GET" class="grid-cell flex flex-row md:col-span-3 gap-2 "> |  | ||||||
| 			@lineComponent() |  | ||||||
| 			<div class="flex flex-col gap-2 justify-between grow-1"> |  | ||||||
| 				<input type="date" value={ urlParams.Get("time_from") } name="time_from" class="w-full bg-neutral-100 placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-0 rounded-md px-3 py-2 transition duration-300 ease focus:outline-none focus:border-neutral-400 hover:border-neutral-300" placeholder="Zeitraum von..."/> |  | ||||||
| 				<input type="date" value={ urlParams.Get("time_to") } name="time_to" class="w-full bg-neutral-100 placeholder:text-neutral-400 text-neutral-700 text-sm border border-neutral-0 rounded-md px-3 py-2 transition duration-300 ease focus:outline-none focus:border-neutral-400 hover:border-neutral-300" placeholder="Zeitraum bis..."/> |  | ||||||
| 			</div> |  | ||||||
| 		</form> |  | ||||||
| 		<div class="grid-cell content-end"> |  | ||||||
| 			<button type="submit" form="timeRangeForm" class="btn bg-neutral-100 hover:bg-neutral-700 color-neutral-700"> |  | ||||||
| 				<p class="">Anzeigen</p> |  | ||||||
| 			</button> |  | ||||||
| 		</div> |  | ||||||
| 	</div> | 	</div> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ dayComponent(workDay models.WorkDay, submitted bool) { | templ changeButtonComponent(id string, workDay bool) { | ||||||
| 	{{ | 	<button class="change-button-component btn w-auto group/button" type="button" onclick={ templ.JSFuncCall("editWorkday", templ.JSExpression("this"), templ.JSExpression("event"), id, workDay) }> | ||||||
| 		work, pause := workDay.GetWorkTimeString() | 		<p class="hidden md:block group-[.edit]/button:hidden">Ändern</p> | ||||||
| 		user := ctx.Value("user").(models.User) | 		<p class="hidden group-[.edit]/button:md:block">Absenden</p> | ||||||
| 		overtime := helper.FormatDuration(workDay.CalcOvertime(user)) |  | ||||||
| 		justify := "" |  | ||||||
| 		if len(workDay.Bookings) <= 1 { |  | ||||||
| 			justify = "justify-content: center" |  | ||||||
| 		} |  | ||||||
| 	}} |  | ||||||
| 	<div class={ "grid-sub divide-x-1 hover:bg-neutral-200 transition-colors", templ.KV("bg-neutral-100", submitted) }> |  | ||||||
| 		<div class="grid-cell md:col-span-1 flex flex-row gap-2"> |  | ||||||
| 			@timeGaugeComponent(workDay.GetWorkDayProgress(ctx.Value("user").(models.User)), workDay.Day.Equal(time.Now().Truncate(24*time.Hour)), workDay.RequiresAction()) |  | ||||||
| 			<div> |  | ||||||
| 				<p class=""><span class="font-bold uppercase hidden md:inline">{ workDay.Day.Format("Mon") }:</span> { workDay.Day.Format("02.01.2006") }</p> |  | ||||||
| 				if work!="" { |  | ||||||
| 					<p class=" text-sm mt-1">Arbeitszeit:</p> |  | ||||||
| 					if (workDay.RequiresAction()) { |  | ||||||
| 						<p class="text-red-600">Bitte anpassen</p> |  | ||||||
| 					} else { |  | ||||||
| 						<p class="text-accent flex flex-row items-center"><span class="icon-[material-symbols-light--nest-clock-farsight-analog-outline]"></span>{ work }</p> |  | ||||||
| 						if pause != "" { |  | ||||||
| 							<p class="text-neutral-500 flex flex-row items-center"><span class="icon-[material-symbols-light--motion-photos-paused-outline]"></span>{ pause }</p> |  | ||||||
| 						} |  | ||||||
| 						if overtime != "" { |  | ||||||
| 							<p class="text-neutral-500 flex flex-row items-center"><span class="icon-[material-symbols-light--more-time]"></span>{ overtime }</p> |  | ||||||
| 						} |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 		<div class="all-booking-component flex flex-row md:col-span-3 gap-2 w-full grid-cell"> |  | ||||||
| 			@lineComponent() |  | ||||||
| 			<form id={ "time-" + workDay.Day.Format("2006-01-02") } class="flex flex-col gap-2 group w-full justify-between" style={ justify } method="post"> |  | ||||||
| 				if (workDay.Absence != models.Absence{}) { |  | ||||||
| 					<p>{ workDay.Absence.AbwesenheitTyp.Name }</p> |  | ||||||
| 				} |  | ||||||
| 				if len(workDay.Bookings) < 1 && (workDay.Absence == models.Absence{}) { |  | ||||||
| 					<p class="text group-[.edit]:hidden">Keine Buchung gefunden. Bitte Arbeitsstunden oder Grund der Abwesenheit eingeben!</p> |  | ||||||
| 					@absenceComponent(workDay) |  | ||||||
| 					@newBookingComponent(workDay) |  | ||||||
| 				} else { |  | ||||||
| 					@absenceComponent(workDay) |  | ||||||
| 					for _, booking := range workDay.Bookings { |  | ||||||
| 						@bookingComponent(booking) |  | ||||||
| 					} |  | ||||||
| 					@newBookingComponent(workDay) |  | ||||||
| 				} |  | ||||||
| 				<input type="hidden" name="action" value="change"/> <!-- default action value for ändern button --> |  | ||||||
| 			</form> |  | ||||||
| 		</div> |  | ||||||
| 		<div class="grid-cell"> |  | ||||||
| 			@changeButtonComponent("time-" + workDay.Day.Format("2006-01-02")) |  | ||||||
| 		</div> |  | ||||||
| 	</div> |  | ||||||
| } |  | ||||||
|  |  | ||||||
| templ changeButtonComponent(id string) { |  | ||||||
| 	<button class="btn w-auto group" type="submit" onclick={ templ.JSFuncCall("editDay", templ.JSExpression("this"), templ.JSExpression("event"), id) }> |  | ||||||
| 		<p class="hidden md:block group-[.edit]:hidden">Ändern</p> |  | ||||||
| 		<p class="hidden group-[.edit]:md:block">Absenden</p> |  | ||||||
| 		<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" fill="currentColor" class="w-4 h-4 md:hidden"> | 		<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" fill="currentColor" class="w-4 h-4 md:hidden"> | ||||||
| 			<path class="group-[.edit]:hidden md:hidden" d="M12.146.146a.5.5 0 0 1 .708 0l3 3a.5.5 0 0 1 0 .708l-10 10a.5.5 0 0 1-.168.11l-5 2a.5.5 0 0 1-.65-.65l2-5a.5.5 0 0 1 .11-.168zM11.207 2.5 13.5 4.793 14.793 3.5 12.5 1.207zm1.586 3L10.5 3.207 4 9.707V10h.5a.5.5 0 0 1 .5.5v.5h.5a.5.5 0 0 1 .5.5v.5h.293zm-9.761 5.175-.106.106-1.528 3.821 3.821-1.528.106-.106A.5.5 0 0 1 5 12.5V12h-.5a.5.5 0 0 1-.5-.5V11h-.5a.5.5 0 0 1-.468-.325"></path> | 			<path class="group-[.edit]/button:hidden md:hidden" d="M12.146.146a.5.5 0 0 1 .708 0l3 3a.5.5 0 0 1 0 .708l-10 10a.5.5 0 0 1-.168.11l-5 2a.5.5 0 0 1-.65-.65l2-5a.5.5 0 0 1 .11-.168zM11.207 2.5 13.5 4.793 14.793 3.5 12.5 1.207zm1.586 3L10.5 3.207 4 9.707V10h.5a.5.5 0 0 1 .5.5v.5h.5a.5.5 0 0 1 .5.5v.5h.293zm-9.761 5.175-.106.106-1.528 3.821 3.821-1.528.106-.106A.5.5 0 0 1 5 12.5V12h-.5a.5.5 0 0 1-.5-.5V11h-.5a.5.5 0 0 1-.468-.325"></path> | ||||||
| 			<path class="hidden group-[.edit]:block md:hidden" d="M12.736 3.97a.733.733 0 0 1 j1.047 0c.286.289.29.756.01 1.05L7.88 12.01a.733.733 0 0 1-1.065.02L3.217 8.384a.757.757 0 0 1 0-1.06.733.733 0 0 1 1.047 0l3.052 3.093 5.4-6.425z"></path> | 			<path class="hidden group-[.edit]/button:block md:hidden" d="M12.736 3.97a.733.733 0 0 1 j1.047 0c.286.289.29.756.01 1.05L7.88 12.01a.733.733 0 0 1-1.065.02L3.217 8.384a.757.757 0 0 1 0-1.06.733.733 0 0 1 1.047 0l3.052 3.093 5.4-6.425z"></path> | ||||||
| 		</svg> | 		</svg> | ||||||
| 	</button> | 	</button> | ||||||
|  | 	<button class="hidden group-[.edit]:flex btn  basis-[content] items-center" onclick={ templ.JSFuncCall("clearEditState") }><span class="size-5 icon-[material-symbols-light--cancel-outline]"></span></button> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ timeGaugeComponent(progress uint8, today bool, warning bool) { | templ timeGaugeComponent(progress int8, today bool) { | ||||||
| 	{{ | 	{{ | ||||||
| 		var bgColor string | 		var bgColor string | ||||||
| 		switch { | 		switch { | ||||||
| 		case (warning): | 		case (0 > progress): | ||||||
| 			bgColor = "bg-red-600" | 			bgColor = "bg-red-600" | ||||||
| 			break | 			break | ||||||
| 		case (progress > 0 && progress < 95): | 		case (progress > 0 && progress < 95): | ||||||
| @@ -135,49 +61,68 @@ templ timeGaugeComponent(progress uint8, today bool, warning bool) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| templ lineComponent() { | templ newAbsenceComponent() { | ||||||
| 	<div class="flex flex-col w-2 py-2 items-center text-accent print:hidden"> | 	<div class="no-booking-component hidden group-[.edit]:flex flex-col gap-2 align-center "> | ||||||
| 		<svg class="size-2" viewBox="0 0 24 24" fill="currentColor"> | 		<button type="button" name="absence" onclick={ templ.JSFuncCall("editWorkday", templ.JSExpression("this"), templ.JSExpression("event"), 0, false) } class="btn border-neutral-500"> | ||||||
| 			<polygon points="12,2 22,12 12,22 2,12"></polygon> | 			Neue Abwesenheit | ||||||
| 		</svg> | 		</button> | ||||||
| 		<div class="w-[2px] bg-accent flex-grow -my-1"></div> |  | ||||||
| 		<svg class="size-2" viewBox="0 0 24 24" fill="currentColor"> |  | ||||||
| 			<polygon points="12,2 22,12 12,22 2,12"></polygon> |  | ||||||
| 		</svg> |  | ||||||
| 	</div> | 	</div> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ absenceComponent(d models.WorkDay) { | templ absenceComponent(a *models.Absence, isKurzarbeit bool) { | ||||||
| 	<div class="no-booking-component hidden group-[.edit]:flex flex-col gap-2 align-center"> | 	{{ | ||||||
| 		<select name="absence" onchange={ templ.JSFuncCall("editAbwesenheit", templ.JSExpression("this"), templ.JSExpression("event")) } class="grow cursor-pointer rounded-md text-neutral-800 p-2 md:px-4 border text-center text-sm transition-colors border-neutral-900" disabled> | 		editBox := "" | ||||||
| 			<option value="0">Abwesenheit?</option> | 		if isKurzarbeit { | ||||||
| 			for _, absence := range models.GetAbsenceTypesCached() { | 			editBox = "edit-box" | ||||||
| 				<option value={ strconv.Itoa(int(absence.Id)) }>{ absence.Name }</option> | 		} | ||||||
|  | 	}} | ||||||
|  | 	<div class={ "flex flex-row items-center gap-2", editBox }> | ||||||
|  | 		<input type="hidden" name="date_from" value={ a.DateFrom.Format(time.DateOnly) }/> | ||||||
|  | 		<input type="hidden" name="date_to" value={ a.DateTo.Format(time.DateOnly) }/> | ||||||
|  | 		<input type="hidden" name="aw_type" value={ a.AbwesenheitTyp.Id }/> | ||||||
|  | 		<input type="hidden" name="aw_id" value={ a.CounterId }/> | ||||||
|  | 		<p class="whitespace-nowrap group-[.edit]:ml-2"> | ||||||
|  | 			{ a.AbwesenheitTyp.Name } | ||||||
|  | 			if a.IsMultiDay() { | ||||||
|  | 				<span class="text-neutral-500">bis { a.DateTo.Format("02.01.2006") }</span> | ||||||
| 			} | 			} | ||||||
| 		</select> | 		</p> | ||||||
|  | 		<div class="w-full"></div> | ||||||
|  | 		if isKurzarbeit { | ||||||
|  | 			<button type="button" onclick={ templ.JSFuncCall("editWorkday", templ.JSExpression("this"), templ.JSExpression("event"), "time-"+a.Date().Format(time.DateOnly), false) } class="hidden btn border-0 rounded-none grow-0 w-auto group-[.edit]:inline">Bearbeiten</button> | ||||||
|  | 		} | ||||||
| 	</div> | 	</div> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ newBookingComponent(d models.WorkDay) { | templ newBookingComponent(d *models.WorkDay) { | ||||||
| 	<div class="new-booking-component hidden group-[.edit]:flex flex-row gap-2 items-center"> | 	<div class="new-booking-component hidden group-[.edit]:flex flex-row gap-2 items-center edit-box border-dashed"> | ||||||
| 		<button name="action" value="add" type="submit" class="hover:text-accent cursor-pointer icon-[material-symbols-light--add-circle-outline]"></button> | 		<input name="timestamp" type="time" value={ time.Now().Format("15:04") } class="text-neutral-700 group-[.edit]:inline hidden bg-neutral-100 text-sm px-3 py-2 cursor-pointer"/> | ||||||
| 		<input name="timestamp" type="time" value={ time.Now().Format("15:04") } class="text-neutral-700 group-[.edit]:inline hidden bg-neutral-100 text-sm border border-neutral-200 rounded-md px-3 py-2 transition duration-300 ease focus:outline-none focus:border-neutral-400 hover:border-neutral-300"/> | 		<input name="date" type="hidden" value={ d.Day.Format(time.DateOnly) }/> | ||||||
| 		<input name="date" type="hidden" value={ d.Day.Format("2006-01-02") }/> | 		<div class="relative"> | ||||||
| 		<select name="check_in_out"> | 			<select class="cursor-pointer appearance-none" name="check_in_out"> | ||||||
| 			<option value="0" disabled>Kommen/Gehen</option> | 				<option value="0" disabled>Kommen/Gehen</option> | ||||||
| 			<option value="3" selected?={ len(d.Bookings) > 0 && d.Bookings[len(d.Bookings)-1].CheckInOut%2 == 0 }>Kommen</option> | 				<option value="3" selected?={ len(d.Bookings) > 0 && d.Bookings[len(d.Bookings)-1].CheckInOut%2 == 0 }>Kommen</option> | ||||||
| 			<option value="4" selected?={ len(d.Bookings) > 0 && d.Bookings[len(d.Bookings)-1].CheckInOut%2 == 1 }>Gehen</option> | 				<option value="4" selected?={ len(d.Bookings) > 0 && d.Bookings[len(d.Bookings)-1].CheckInOut%2 == 1 }>Gehen</option> | ||||||
| 		</select> | 			</select> | ||||||
|  | 			<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.2" stroke="currentColor" class="h-5 w-5 ml-1 absolute right-1 top-[0.125rem] text-slate-700"> | ||||||
|  | 				<path stroke-linecap="round" stroke-linejoin="round" d="M8.25 15 12 18.75 15.75 15m-7.5-6L12 5.25 15.75 9"></path> | ||||||
|  | 			</svg> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="w-full"></div> | ||||||
|  | 		<button name="action" value="add" type="submit" class="hidden btn border-0 rounded-none grow-0 w-auto group-[.edit]:inline"><span class="hidden md:inline">Hinzufügen</span><span class="md:hidden">+</span></button> | ||||||
| 	</div> | 	</div> | ||||||
| } | } | ||||||
|  |  | ||||||
| templ bookingComponent(booking models.Booking) { | templ bookingComponent(booking models.Booking) { | ||||||
| 	<div> | 	<div> | ||||||
| 		<p class="text-neutral-500"> | 		<p class="text-neutral-500 edit-box"> | ||||||
| 			<span class="text-neutral-700 group-[.edit]:hidden inline">{ booking.Timestamp.Format("15:04") }</span> | 			<span class="text-black group-[.edit]:hidden inline">{ booking.Timestamp.Format("15:04") }</span> | ||||||
| 			<input disabled name={ "booking_" + strconv.Itoa(booking.CounterId) } type="time" value={ booking.Timestamp.Format("15:04") } class="text-neutral-700 group-[.edit]:inline hidden bg-neutral-100 text-sm border border-neutral-200 rounded-md px-3 py-2 transition duration-300 ease focus:outline-none focus:border-neutral-400 hover:border-neutral-300"/> | 			<input disabled name={ "booking_" + strconv.Itoa(booking.CounterId) } type="time" value={ booking.Timestamp.Format("15:04") } class="text-neutral-700 group-[.edit]:inline hidden bg-neutral-100 text-sm px-3 py-2 cursor-pointer"/> | ||||||
| 			{ booking.GetBookingType() } | 			{ booking.GetBookingType() } | ||||||
| 		</p> | 		</p> | ||||||
|  | 		if booking.IsSubmittedAndChecked() { | ||||||
|  | 			<p>submitted</p> | ||||||
|  | 		} | ||||||
| 	</div> | 	</div> | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										166
									
								
								Backend/templates/timePage.templ
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								Backend/templates/timePage.templ
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,166 @@ | |||||||
|  | package templates | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"arbeitszeitmessung/helper" | ||||||
|  | 	"arbeitszeitmessung/models" | ||||||
|  | 	"net/url" | ||||||
|  | 	"strconv" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | templ TimePage(workDays []models.WorkDay, lastSub time.Time) { | ||||||
|  | 	{{ | ||||||
|  | 		allDays := ctx.Value("days").([]models.IWorkDay) | ||||||
|  | 	}} | ||||||
|  | 	@Base() | ||||||
|  | 	@headerComponent() | ||||||
|  | 	<div class="grid-main divide-y-1"> | ||||||
|  | 		@inputForm() | ||||||
|  | 		for _, day := range allDays { | ||||||
|  | 			@defaultDayComponent(day) | ||||||
|  | 			if (day.Date().Weekday() == time.Monday) { | ||||||
|  | 				<div class="grid-sub responsive bg-neutral-300 h-2"></div> | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	</div> | ||||||
|  | 	@LegendComponent() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | templ inputForm() { | ||||||
|  | 	{{ | ||||||
|  | 		urlParams := ctx.Value("urlParams").(url.Values) | ||||||
|  | 		user := ctx.Value("user").(models.User) | ||||||
|  | 	}} | ||||||
|  | 	<div class="grid-sub divide-x-1 bg-neutral-300 responsive"> | ||||||
|  | 		<div class="grid-cell md:col-span-1 max-md:grid grid-cols-2"> | ||||||
|  | 			<p class="font-bold uppercase">{ user.Vorname + " " + user.Name }</p> | ||||||
|  | 			<div class="justify-self-end"> | ||||||
|  | 				<p class="text-sm">Überstunden</p> | ||||||
|  | 				<p class="text-accent">{ user.Overtime }</p> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<form id="timeRangeForm" method="GET" class="grid-cell flex flex-row md:col-span-3 gap-2 "> | ||||||
|  | 			@lineComponent() | ||||||
|  | 			<div class="flex flex-col gap-2 justify-between grow-1"> | ||||||
|  | 				<input type="date" value={ urlParams.Get("time_from") } name="time_from" class="btn bg-neutral-100" placeholder="Zeitraum von..."/> | ||||||
|  | 				<input type="date" value={ urlParams.Get("time_to") } name="time_to" class="btn bg-neutral-100" placeholder="Zeitraum bis..."/> | ||||||
|  | 			</div> | ||||||
|  | 		</form> | ||||||
|  | 		<div class="grid-cell content-end"> | ||||||
|  | 			<button type="submit" form="timeRangeForm" class="btn bg-neutral-100 hover:bg-neutral-700 color-neutral-700"> | ||||||
|  | 				<p class="">Anzeigen</p> | ||||||
|  | 			</button> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | 	<form id="absence_form" method="POST" action="/absence" class="grid-sub responsive scroll-m-2 bg-neutral-300 hidden"> | ||||||
|  | 		<input type="hidden" name="aw_id" value=""/> | ||||||
|  | 		<div class="grid-cell border-r-1"><p class="font-bold uppercase">Abwesenheit</p></div> | ||||||
|  | 		<div class="grid-cell"> | ||||||
|  | 			<label class="block mb-1 text-sm text-neutral-700">Abwesenheitsart</label> | ||||||
|  | 			<div class="relative"> | ||||||
|  | 				<select name="aw_type" class="btn appearance-none cursor-pointer bg-neutral-100"> | ||||||
|  | 					for _, absence := range models.GetAbsenceTypesCached() { | ||||||
|  | 						<option value={ strconv.Itoa(int(absence.Id)) }>{ absence.Name }</option> | ||||||
|  | 					} | ||||||
|  | 				</select> | ||||||
|  | 				<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.2" stroke="currentColor" class="h-5 w-5 ml-1 absolute top-2.5 right-2.5 text-slate-700"> | ||||||
|  | 					<path stroke-linecap="round" stroke-linejoin="round" d="M8.25 15 12 18.75 15.75 15m-7.5-6L12 5.25 15.75 9"></path> | ||||||
|  | 				</svg> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="grid-cell"> | ||||||
|  | 			<label class="block mb-1 text-sm text-neutral-700">Abwesenheit ab</label> | ||||||
|  | 			<input name="date_from" type="date" class="btn bg-neutral-100"/> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="grid-cell border-r-1"> | ||||||
|  | 			<label class="block mb-1 text-sm text-neutral-700">Abwesenheit bis</label> | ||||||
|  | 			<input name="date_to" type="date" class="btn bg-neutral-100"/> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="grid-cell flex flex-row items-end"> | ||||||
|  | 			<div class="flex flex-row gap-2 w-full"> | ||||||
|  | 				<button name="action" value="insert" type="submit" class="bg-neutral-100 btn hover:bg-neutral-700">Speichern</button> | ||||||
|  | 				<button name="action" value="delete" type="submit" class="bg-neutral-100 btn hover:bg-red-700 flex basis-[content] items-center"><span class="size-5 icon-[material-symbols-light--delete-outline]"></span></button> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</form> | ||||||
|  | } | ||||||
|  |  | ||||||
|  | templ defaultDayComponent(day models.IWorkDay) { | ||||||
|  | 	{{ | ||||||
|  | 		user := ctx.Value("user").(models.User) | ||||||
|  | 		justify := "justify-center" | ||||||
|  | 		if day.IsWorkDay() && len(day.(*models.WorkDay).Bookings) > 1 { | ||||||
|  | 			justify = "justify-between" | ||||||
|  | 		} | ||||||
|  | 	}} | ||||||
|  | 	<div class={ "grid-sub divide-x-1 hover:bg-neutral-200 transition-colors group" }> | ||||||
|  | 		<div class="grid-cell md:col-span-1 flex flex-row gap-2"> | ||||||
|  | 			@timeGaugeComponent(day.GetDayProgress(user), day.Date().Equal(time.Now().Truncate(24*time.Hour))) | ||||||
|  | 			<div> | ||||||
|  | 				<p> | ||||||
|  | 					<span class="font-bold uppercase hidden md:inline">{ day.Date().Format("Mon") }:</span> { day.Date().Format("02.01.2006") } | ||||||
|  | 				</p> | ||||||
|  | 				if day.IsWorkDay() { | ||||||
|  | 					{{ | ||||||
|  | 						workDay, _ := day.(*models.WorkDay) | ||||||
|  | 						work, pause, overtime := workDay.GetAllWorkTimesReal(user) | ||||||
|  | 					}} | ||||||
|  | 					if day.RequiresAction() { | ||||||
|  | 						<p class="text-red-600">Bitte anpassen</p> | ||||||
|  | 					} else { | ||||||
|  | 						if work > 0 { | ||||||
|  | 							<p class=" text-sm mt-1">Arbeitszeit:</p> | ||||||
|  | 							<p class="text-accent flex flex-row items-center"><span class="icon-[material-symbols-light--schedule-outline]"></span>{ helper.FormatDuration(work) }</p> | ||||||
|  | 						} | ||||||
|  | 						if pause > 0 { | ||||||
|  | 							<p class="text-neutral-500 flex flex-row items-center"><span class="icon-[material-symbols-light--motion-photos-paused-outline]"></span>{ helper.FormatDuration(pause) }</p> | ||||||
|  | 						} | ||||||
|  | 						if overtime != 0 && len(workDay.Bookings) > 0 { | ||||||
|  | 							<p class="text-neutral-500 flex flex-row items-center"> | ||||||
|  | 								<span class="icon-[material-symbols-light--more-time]"></span> | ||||||
|  | 								{ helper.FormatDuration(overtime) } | ||||||
|  | 							</p> | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="all-booking-component grid-cell flex flex-row md:col-span-3 col-span-2 gap-2 w-full"> | ||||||
|  | 			@lineComponent() | ||||||
|  | 			<form id={ "time-" + day.Date().Format(time.DateOnly) } class={ "flex flex-col gap-2 w-full", justify } method="post"> | ||||||
|  | 				if day.IsWorkDay() { | ||||||
|  | 					{{ | ||||||
|  | 					workDay, _ := day.(*models.WorkDay) | ||||||
|  | 					}} | ||||||
|  | 					@newAbsenceComponent() | ||||||
|  | 					if len(workDay.Bookings) < 1 { | ||||||
|  | 						<p class="text group-[.edit]:hidden">Keine Buchung gefunden. Bitte Arbeitsstunden oder Grund der Abwesenheit eingeben!</p> | ||||||
|  | 					} | ||||||
|  | 					if workDay.IsKurzArbeit() { | ||||||
|  | 						@absenceComponent(workDay.GetKurzArbeit(), true) | ||||||
|  | 					} | ||||||
|  | 					for _, booking := range workDay.Bookings { | ||||||
|  | 						@bookingComponent(booking) | ||||||
|  | 					} | ||||||
|  | 					@newBookingComponent(workDay) | ||||||
|  | 				} else { | ||||||
|  | 					{{ | ||||||
|  | 						absentDay, _ := day.(*models.Absence) | ||||||
|  | 					}} | ||||||
|  | 					@absenceComponent(absentDay, false) | ||||||
|  | 				} | ||||||
|  | 				<input type="hidden" name="action" value="change"/> <!-- default action value for ändern button --> | ||||||
|  | 			</form> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="grid-cell flex flex-row gap-2 items-end"> | ||||||
|  | 			@changeButtonComponent("time-"+day.Date().Format(time.DateOnly), day.IsWorkDay()) | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | } | ||||||
|  |  | ||||||
|  | templ absentInput(a models.Absence) { | ||||||
|  | 	<input type="hidden" name="date_from" value={ a.DateFrom.Format(time.DateOnly) }/> | ||||||
|  | 	<input type="hidden" name="date_to" value={ a.DateTo.Format(time.DateOnly) }/> | ||||||
|  | 	<input type="hidden" name="aw_type" value={ a.AbwesenheitTyp.Id }/> | ||||||
|  | 	<input type="hidden" name="aw_id" value={ a.CounterId }/> | ||||||
|  | } | ||||||
							
								
								
									
										563
									
								
								Backend/templates/timePage_templ.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										563
									
								
								Backend/templates/timePage_templ.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,563 @@ | |||||||
|  | // Code generated by templ - DO NOT EDIT. | ||||||
|  |  | ||||||
|  | // templ: version: v0.3.924 | ||||||
|  | package templates | ||||||
|  |  | ||||||
|  | //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||||
|  |  | ||||||
|  | import "github.com/a-h/templ" | ||||||
|  | import templruntime "github.com/a-h/templ/runtime" | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"arbeitszeitmessung/helper" | ||||||
|  | 	"arbeitszeitmessung/models" | ||||||
|  | 	"net/url" | ||||||
|  | 	"strconv" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TimePage(workDays []models.WorkDay, lastSub time.Time) templ.Component { | ||||||
|  | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
|  | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
|  | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
|  | 			return templ_7745c5c3_CtxErr | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||||
|  | 		if !templ_7745c5c3_IsBuffer { | ||||||
|  | 			defer func() { | ||||||
|  | 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||||
|  | 				if templ_7745c5c3_Err == nil { | ||||||
|  | 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||||
|  | 				} | ||||||
|  | 			}() | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.InitializeContext(ctx) | ||||||
|  | 		templ_7745c5c3_Var1 := templ.GetChildren(ctx) | ||||||
|  | 		if templ_7745c5c3_Var1 == nil { | ||||||
|  | 			templ_7745c5c3_Var1 = templ.NopComponent | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.ClearChildren(ctx) | ||||||
|  |  | ||||||
|  | 		allDays := ctx.Value("days").([]models.IWorkDay) | ||||||
|  | 		templ_7745c5c3_Err = Base().Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = headerComponent().Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<div class=\"grid-main divide-y-1\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = inputForm().Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		for _, day := range allDays { | ||||||
|  | 			templ_7745c5c3_Err = defaultDayComponent(day).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, " ") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			if day.Date().Weekday() == time.Monday { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "<div class=\"grid-sub responsive bg-neutral-300 h-2\"></div>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "</div>") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = LegendComponent().Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func inputForm() templ.Component { | ||||||
|  | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
|  | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
|  | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
|  | 			return templ_7745c5c3_CtxErr | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||||
|  | 		if !templ_7745c5c3_IsBuffer { | ||||||
|  | 			defer func() { | ||||||
|  | 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||||
|  | 				if templ_7745c5c3_Err == nil { | ||||||
|  | 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||||
|  | 				} | ||||||
|  | 			}() | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.InitializeContext(ctx) | ||||||
|  | 		templ_7745c5c3_Var2 := templ.GetChildren(ctx) | ||||||
|  | 		if templ_7745c5c3_Var2 == nil { | ||||||
|  | 			templ_7745c5c3_Var2 = templ.NopComponent | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.ClearChildren(ctx) | ||||||
|  |  | ||||||
|  | 		urlParams := ctx.Value("urlParams").(url.Values) | ||||||
|  | 		user := ctx.Value("user").(models.User) | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "<div class=\"grid-sub divide-x-1 bg-neutral-300 responsive\"><div class=\"grid-cell md:col-span-1 max-md:grid grid-cols-2\"><p class=\"font-bold uppercase\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var3 string | ||||||
|  | 		templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(user.Vorname + " " + user.Name) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 36, Col: 66} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "</p><div class=\"justify-self-end\"><p class=\"text-sm\">Überstunden</p><p class=\"text-accent\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var4 string | ||||||
|  | 		templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(user.Overtime) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 39, Col: 42} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "</p></div></div><form id=\"timeRangeForm\" method=\"GET\" class=\"grid-cell flex flex-row md:col-span-3 gap-2 \">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = lineComponent().Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, "<div class=\"flex flex-col gap-2 justify-between grow-1\"><input type=\"date\" value=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var5 string | ||||||
|  | 		templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(urlParams.Get("time_from")) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 45, Col: 57} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var5)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "\" name=\"time_from\" class=\"btn bg-neutral-100\" placeholder=\"Zeitraum von...\"> <input type=\"date\" value=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var6 string | ||||||
|  | 		templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(urlParams.Get("time_to")) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 46, Col: 55} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "\" name=\"time_to\" class=\"btn bg-neutral-100\" placeholder=\"Zeitraum bis...\"></div></form><div class=\"grid-cell content-end\"><button type=\"submit\" form=\"timeRangeForm\" class=\"btn bg-neutral-100 hover:bg-neutral-700 color-neutral-700\"><p class=\"\">Anzeigen</p></button></div></div><form id=\"absence_form\" method=\"POST\" action=\"/absence\" class=\"grid-sub responsive scroll-m-2 bg-neutral-300 hidden\"><input type=\"hidden\" name=\"aw_id\" value=\"\"><div class=\"grid-cell border-r-1\"><p class=\"font-bold uppercase\">Abwesenheit</p></div><div class=\"grid-cell\"><label class=\"block mb-1 text-sm text-neutral-700\">Abwesenheitsart</label><div class=\"relative\"><select name=\"aw_type\" class=\"btn appearance-none cursor-pointer bg-neutral-100\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		for _, absence := range models.GetAbsenceTypesCached() { | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, "<option value=\"") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			var templ_7745c5c3_Var7 string | ||||||
|  | 			templ_7745c5c3_Var7, templ_7745c5c3_Err = templ.JoinStringErrs(strconv.Itoa(int(absence.Id))) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 63, Col: 51} | ||||||
|  | 			} | ||||||
|  | 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var7)) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 12, "\">") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			var templ_7745c5c3_Var8 string | ||||||
|  | 			templ_7745c5c3_Var8, templ_7745c5c3_Err = templ.JoinStringErrs(absence.Name) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 63, Col: 68} | ||||||
|  | 			} | ||||||
|  | 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var8)) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, "</option>") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 14, "</select> <svg xmlns=\"http://www.w3.org/2000/svg\" fill=\"none\" viewBox=\"0 0 24 24\" stroke-width=\"1.2\" stroke=\"currentColor\" class=\"h-5 w-5 ml-1 absolute top-2.5 right-2.5 text-slate-700\"><path stroke-linecap=\"round\" stroke-linejoin=\"round\" d=\"M8.25 15 12 18.75 15.75 15m-7.5-6L12 5.25 15.75 9\"></path></svg></div></div><div class=\"grid-cell\"><label class=\"block mb-1 text-sm text-neutral-700\">Abwesenheit ab</label> <input name=\"date_from\" type=\"date\" class=\"btn bg-neutral-100\"></div><div class=\"grid-cell border-r-1\"><label class=\"block mb-1 text-sm text-neutral-700\">Abwesenheit bis</label> <input name=\"date_to\" type=\"date\" class=\"btn bg-neutral-100\"></div><div class=\"grid-cell flex flex-row items-end\"><div class=\"flex flex-row gap-2 w-full\"><button name=\"action\" value=\"insert\" type=\"submit\" class=\"bg-neutral-100 btn hover:bg-neutral-700\">Speichern</button> <button name=\"action\" value=\"delete\" type=\"submit\" class=\"bg-neutral-100 btn hover:bg-red-700 flex basis-[content] items-center\"><span class=\"size-5 icon-[material-symbols-light--delete-outline]\"></span></button></div></div></form>") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func defaultDayComponent(day models.IWorkDay) templ.Component { | ||||||
|  | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
|  | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
|  | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
|  | 			return templ_7745c5c3_CtxErr | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||||
|  | 		if !templ_7745c5c3_IsBuffer { | ||||||
|  | 			defer func() { | ||||||
|  | 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||||
|  | 				if templ_7745c5c3_Err == nil { | ||||||
|  | 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||||
|  | 				} | ||||||
|  | 			}() | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.InitializeContext(ctx) | ||||||
|  | 		templ_7745c5c3_Var9 := templ.GetChildren(ctx) | ||||||
|  | 		if templ_7745c5c3_Var9 == nil { | ||||||
|  | 			templ_7745c5c3_Var9 = templ.NopComponent | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.ClearChildren(ctx) | ||||||
|  |  | ||||||
|  | 		user := ctx.Value("user").(models.User) | ||||||
|  | 		justify := "justify-center" | ||||||
|  | 		if day.IsWorkDay() && len(day.(*models.WorkDay).Bookings) > 1 { | ||||||
|  | 			justify = "justify-between" | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var10 = []any{"grid-sub divide-x-1 hover:bg-neutral-200 transition-colors group"} | ||||||
|  | 		templ_7745c5c3_Err = templ.RenderCSSItems(ctx, templ_7745c5c3_Buffer, templ_7745c5c3_Var10...) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, "<div class=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var11 string | ||||||
|  | 		templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(templ.CSSClasses(templ_7745c5c3_Var10).String()) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 1, Col: 0} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "\"><div class=\"grid-cell md:col-span-1 flex flex-row gap-2\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = timeGaugeComponent(day.GetDayProgress(user), day.Date().Equal(time.Now().Truncate(24*time.Hour))).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "<div><p><span class=\"font-bold uppercase hidden md:inline\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var12 string | ||||||
|  | 		templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(day.Date().Format("Mon")) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 101, Col: 82} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 18, ":</span> ") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var13 string | ||||||
|  | 		templ_7745c5c3_Var13, templ_7745c5c3_Err = templ.JoinStringErrs(day.Date().Format("02.01.2006")) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 101, Col: 126} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var13)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "</p>") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		if day.IsWorkDay() { | ||||||
|  |  | ||||||
|  | 			workDay, _ := day.(*models.WorkDay) | ||||||
|  | 			work, pause, overtime := workDay.GetAllWorkTimesReal(user) | ||||||
|  | 			if day.RequiresAction() { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "<p class=\"text-red-600\">Bitte anpassen</p>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} else { | ||||||
|  | 				if work > 0 { | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, "<p class=\" text-sm mt-1\">Arbeitszeit:</p><p class=\"text-accent flex flex-row items-center\"><span class=\"icon-[material-symbols-light--schedule-outline]\"></span>") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					var templ_7745c5c3_Var14 string | ||||||
|  | 					templ_7745c5c3_Var14, templ_7745c5c3_Err = templ.JoinStringErrs(helper.FormatDuration(work)) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 113, Col: 155} | ||||||
|  | 					} | ||||||
|  | 					_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var14)) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 22, "</p>") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, " ") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 				if pause > 0 { | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, "<p class=\"text-neutral-500 flex flex-row items-center\"><span class=\"icon-[material-symbols-light--motion-photos-paused-outline]\"></span>") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					var templ_7745c5c3_Var15 string | ||||||
|  | 					templ_7745c5c3_Var15, templ_7745c5c3_Err = templ.JoinStringErrs(helper.FormatDuration(pause)) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 116, Col: 173} | ||||||
|  | 					} | ||||||
|  | 					_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var15)) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, "</p>") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, " ") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 				if overtime != 0 && len(workDay.Bookings) > 0 { | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 27, "<p class=\"text-neutral-500 flex flex-row items-center\"><span class=\"icon-[material-symbols-light--more-time]\"></span> ") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					var templ_7745c5c3_Var16 string | ||||||
|  | 					templ_7745c5c3_Var16, templ_7745c5c3_Err = templ.JoinStringErrs(helper.FormatDuration(overtime)) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 121, Col: 41} | ||||||
|  | 					} | ||||||
|  | 					_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var16)) | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 					templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, "</p>") | ||||||
|  | 					if templ_7745c5c3_Err != nil { | ||||||
|  | 						return templ_7745c5c3_Err | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, "</div></div><div class=\"all-booking-component grid-cell flex flex-row md:col-span-3 col-span-2 gap-2 w-full\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = lineComponent().Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var17 = []any{"flex flex-col gap-2 w-full", justify} | ||||||
|  | 		templ_7745c5c3_Err = templ.RenderCSSItems(ctx, templ_7745c5c3_Buffer, templ_7745c5c3_Var17...) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 30, "<form id=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var18 string | ||||||
|  | 		templ_7745c5c3_Var18, templ_7745c5c3_Err = templ.JoinStringErrs("time-" + day.Date().Format(time.DateOnly)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 130, Col: 56} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var18)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 31, "\" class=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var19 string | ||||||
|  | 		templ_7745c5c3_Var19, templ_7745c5c3_Err = templ.JoinStringErrs(templ.CSSClasses(templ_7745c5c3_Var17).String()) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 1, Col: 0} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var19)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 32, "\" method=\"post\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		if day.IsWorkDay() { | ||||||
|  |  | ||||||
|  | 			workDay, _ := day.(*models.WorkDay) | ||||||
|  | 			templ_7745c5c3_Err = newAbsenceComponent().Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 33, " ") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			if len(workDay.Bookings) < 1 { | ||||||
|  | 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 34, "<p class=\"text group-[.edit]:hidden\">Keine Buchung gefunden. Bitte Arbeitsstunden oder Grund der Abwesenheit eingeben!</p>") | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 35, " ") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			if workDay.IsKurzArbeit() { | ||||||
|  | 				templ_7745c5c3_Err = absenceComponent(workDay.GetKurzArbeit(), true).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			for _, booking := range workDay.Bookings { | ||||||
|  | 				templ_7745c5c3_Err = bookingComponent(booking).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 				if templ_7745c5c3_Err != nil { | ||||||
|  | 					return templ_7745c5c3_Err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 36, " ") | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 			templ_7745c5c3_Err = newBookingComponent(workDay).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  |  | ||||||
|  | 			absentDay, _ := day.(*models.Absence) | ||||||
|  | 			templ_7745c5c3_Err = absenceComponent(absentDay, false).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 			if templ_7745c5c3_Err != nil { | ||||||
|  | 				return templ_7745c5c3_Err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 37, "<input type=\"hidden\" name=\"action\" value=\"change\"><!-- default action value for ändern button --></form></div><div class=\"grid-cell flex flex-row gap-2 items-end\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = changeButtonComponent("time-"+day.Date().Format(time.DateOnly), day.IsWorkDay()).Render(ctx, templ_7745c5c3_Buffer) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 38, "</div></div>") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func absentInput(a models.Absence) templ.Component { | ||||||
|  | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
|  | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
|  | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
|  | 			return templ_7745c5c3_CtxErr | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||||
|  | 		if !templ_7745c5c3_IsBuffer { | ||||||
|  | 			defer func() { | ||||||
|  | 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||||
|  | 				if templ_7745c5c3_Err == nil { | ||||||
|  | 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||||
|  | 				} | ||||||
|  | 			}() | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.InitializeContext(ctx) | ||||||
|  | 		templ_7745c5c3_Var20 := templ.GetChildren(ctx) | ||||||
|  | 		if templ_7745c5c3_Var20 == nil { | ||||||
|  | 			templ_7745c5c3_Var20 = templ.NopComponent | ||||||
|  | 		} | ||||||
|  | 		ctx = templ.ClearChildren(ctx) | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 39, "<input type=\"hidden\" name=\"date_from\" value=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var21 string | ||||||
|  | 		templ_7745c5c3_Var21, templ_7745c5c3_Err = templ.JoinStringErrs(a.DateFrom.Format(time.DateOnly)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 162, Col: 79} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var21)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 40, "\"> <input type=\"hidden\" name=\"date_to\" value=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var22 string | ||||||
|  | 		templ_7745c5c3_Var22, templ_7745c5c3_Err = templ.JoinStringErrs(a.DateTo.Format(time.DateOnly)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 163, Col: 75} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var22)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 41, "\"> <input type=\"hidden\" name=\"aw_type\" value=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var23 string | ||||||
|  | 		templ_7745c5c3_Var23, templ_7745c5c3_Err = templ.JoinStringErrs(a.AbwesenheitTyp.Id) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 164, Col: 64} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var23)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 42, "\"> <input type=\"hidden\" name=\"aw_id\" value=\"") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		var templ_7745c5c3_Var24 string | ||||||
|  | 		templ_7745c5c3_Var24, templ_7745c5c3_Err = templ.JoinStringErrs(a.CounterId) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/timePage.templ`, Line: 165, Col: 54} | ||||||
|  | 		} | ||||||
|  | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var24)) | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 43, "\">") | ||||||
|  | 		if templ_7745c5c3_Err != nil { | ||||||
|  | 			return templ_7745c5c3_Err | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var _ = templruntime.GeneratedTemplate | ||||||
							
								
								
									
										33
									
								
								DB/initdb/01_schema.sql
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										33
									
								
								DB/initdb/01_schema.sql
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							| @@ -22,7 +22,7 @@ COMMENT ON COLUMN "anwesenheit"."geraet_id" IS 'ID des Lesegerätes'; | |||||||
| DROP TABLE IF EXISTS "s_anwesenheit_typen"; | DROP TABLE IF EXISTS "s_anwesenheit_typen"; | ||||||
| CREATE TABLE "s_anwesenheit_typen" ( | CREATE TABLE "s_anwesenheit_typen" ( | ||||||
|     "anwesenheit_id" int2 PRIMARY KEY, |     "anwesenheit_id" int2 PRIMARY KEY, | ||||||
|     "anwesenheit_name" varchar(255) |     "anwesenheit_name" varchar(255) NOT NULL | ||||||
| ); | ); | ||||||
|  |  | ||||||
| -- ---------------------------- | -- ---------------------------- | ||||||
| @@ -40,11 +40,11 @@ CREATE TABLE "s_personal_daten" ( | |||||||
|     "geschlecht" int2, |     "geschlecht" int2, | ||||||
|     "card_uid" varchar(255), |     "card_uid" varchar(255), | ||||||
|     "hauptbeschaeftigungs_ort" int2, |     "hauptbeschaeftigungs_ort" int2, | ||||||
|     "arbeitszeit_per_tag" float4 NOT NULL, |     "arbeitszeit_per_tag" float4, | ||||||
|     "arbeitszeit_per_woche" float4 NOT NULL, |     "arbeitszeit_per_woche" float4, | ||||||
|     "arbeitszeit_min_start" time(6), |     "arbeitszeit_min_start" time(6), | ||||||
|     "arbeitszeit_max_ende" time(6), |     "arbeitszeit_max_ende" time(6), | ||||||
|     "vorgesetzter_pers_nr" int4 NOT NULL |     "vorgesetzter_pers_nr" int4 | ||||||
| ); | ); | ||||||
| COMMENT ON COLUMN "s_personal_daten"."geschlecht" IS '1==weiblich, 2==maennlich, 3==divers'; | COMMENT ON COLUMN "s_personal_daten"."geschlecht" IS '1==weiblich, 2==maennlich, 3==divers'; | ||||||
|  |  | ||||||
| @@ -78,29 +78,34 @@ EXECUTE FUNCTION update_zuletzt_geandert(); | |||||||
| DROP TABLE IF EXISTS "wochen_report"; | DROP TABLE IF EXISTS "wochen_report"; | ||||||
| CREATE TABLE "wochen_report" ( | CREATE TABLE "wochen_report" ( | ||||||
|     "id" serial PRIMARY KEY, |     "id" serial PRIMARY KEY, | ||||||
|     "personal_nummer" int4, |     "personal_nummer" int4 NOT NULL, | ||||||
|     "woche_start" date, |     "woche_start" date NOT NULL, | ||||||
|     "bestaetigt" bool DEFAULT FALSE, |     "bestaetigt" bool DEFAULT FALSE, | ||||||
|     "arbeitszeit" interval, |     "arbeitszeit" interval NOT NULL, | ||||||
|     "ueberstunden" interval, |     "ueberstunden" interval NOT NULL, | ||||||
|  |     "anwesenheiten" int ARRAY, | ||||||
|  |     "abwesenheiten" int ARRAY, | ||||||
|     UNIQUE ("personal_nummer", "woche_start") |     UNIQUE ("personal_nummer", "woche_start") | ||||||
| ); | ); | ||||||
|  |  | ||||||
| DROP TABLE IF EXISTS "abwesenheit"; | DROP TABLE IF EXISTS "abwesenheit"; | ||||||
| CREATE TABLE "abwesenheit" ( | CREATE TABLE "abwesenheit" ( | ||||||
|     "counter_id" bigserial PRIMARY KEY, |     "counter_id" bigserial PRIMARY KEY, | ||||||
|     "card_uid" varchar(255), |     "card_uid" varchar(255) NOT NULL, | ||||||
|     "abwesenheit_typ" int2, |     "abwesenheit_typ" int2 NOT NULL, | ||||||
|     "datum" timestamptz(6) DEFAULT NOW()::DATE |     "datum_from" timestamptz DEFAULT NOW()::DATE NOT NULL, | ||||||
|  |     "datum_to" timestamptz NOT NULL | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
| DROP TABLE IF EXISTS "s_abwesenheit_typen"; | DROP TABLE IF EXISTS "s_abwesenheit_typen"; | ||||||
| CREATE TABLE "s_abwesenheit_typen" ( | CREATE TABLE "s_abwesenheit_typen" ( | ||||||
|     "abwesenheit_id" int2 PRIMARY KEY, |     "abwesenheit_id" int2 PRIMARY KEY NOT NULL, | ||||||
|     "abwesenheit_name" varchar(255), |     "abwesenheit_name" varchar(255) NOT NULL, | ||||||
|     "arbeitszeit_equivalent" float4 |     "arbeitszeit_equivalent" float4 NOT NULL | ||||||
| ); | ); | ||||||
|  |  | ||||||
|  | COMMENT ON COLUMN "s_abwesenheit_typen"."arbeitszeit_equivalent" IS '0=keine Arbeitszeit; 1=Arbeitszeit auffüllen; 2=Arbeitszeit austauschen'; | ||||||
|  |  | ||||||
| -- Adds crypto extension | -- Adds crypto extension | ||||||
|  |  | ||||||
| CREATE EXTENSION IF NOT EXISTS pgcrypto; | CREATE EXTENSION IF NOT EXISTS pgcrypto; | ||||||
|   | |||||||
							
								
								
									
										0
									
								
								DB/initdb/02_sample_data.sql
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										0
									
								
								DB/initdb/02_sample_data.sql
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
								
								
									
										0
									
								
								DB/initdb/03_create_user.sh
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										0
									
								
								DB/initdb/03_create_user.sh
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							| @@ -30,8 +30,11 @@ services: | |||||||
|       NO_CORS: true |       NO_CORS: true | ||||||
|     ports: |     ports: | ||||||
|       - ${EXPOSED_PORT}:8080 |       - ${EXPOSED_PORT}:8080 | ||||||
|  |     volumes: | ||||||
|  |       - ../logs:/app/Backend/logs | ||||||
|     depends_on: |     depends_on: | ||||||
|       - db |       - db | ||||||
|  |  | ||||||
|   swagger: |   swagger: | ||||||
|     image: swaggerapi/swagger-ui |     image: swaggerapi/swagger-ui | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|   | |||||||
| @@ -28,4 +28,6 @@ services: | |||||||
|       - ${EXPOSED_PORT}:8080 |       - ${EXPOSED_PORT}:8080 | ||||||
|     depends_on: |     depends_on: | ||||||
|       - db |       - db | ||||||
|  |     volumes: | ||||||
|  |       - ../logs:/app/Backend/logs | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								Makefile
									
									
									
									
									
								
							| @@ -45,7 +45,7 @@ generateFrontend: | |||||||
|  |  | ||||||
| backend: generateFrontend login_registry | backend: generateFrontend login_registry | ||||||
| 	docker buildx build --platform linux/amd64,linux/arm64 -t ${IMAGE_REGISTRY}/${PACKAGE_OWNER}/arbeitszeitmessung:latest Backend --push | 	docker buildx build --platform linux/amd64,linux/arm64 -t ${IMAGE_REGISTRY}/${PACKAGE_OWNER}/arbeitszeitmessung:latest Backend --push | ||||||
| 	docker buildx build --platform linux/amd64,linux/arm64 -t ${IMAGE_REGISTRY}/${PACKAGE_OWNER}/arbeitszeitmessung:${GIT_COMMIT} Backend --push | 	# docker buildx build --platform linux/amd64,linux/arm64 -t ${IMAGE_REGISTRY}/${PACKAGE_OWNER}/arbeitszeitmessung:${GIT_COMMIT} Backend //--push | ||||||
|  |  | ||||||
| test: | test: | ||||||
| 	$(MAKE) -C Backend test | 	$(MAKE) -C Backend test | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| # Arbeitszeitmessung | # Arbeitszeitmessung | ||||||
|  |  | ||||||
|  | [](https://sonar.letsstein.de/dashboard?id=arbeitszeitmessung) | ||||||
|  |  | ||||||
| bis jetzt ein einfaches Backend mit PostgreSQL Datenbank und GO Webserver um Arbeitszeitbuchungen per HTTP PUT einzufügen | bis jetzt ein einfaches Backend mit PostgreSQL Datenbank und GO Webserver um Arbeitszeitbuchungen per HTTP PUT einzufügen | ||||||
|  |  | ||||||
| ## Installation | ## Installation | ||||||
|   | |||||||
							
								
								
									
										63
									
								
								WIR-typst/main.typ
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								WIR-typst/main.typ
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | |||||||
|  | #set page("a4") | ||||||
|  | #set text(font: "Lato") | ||||||
|  |  | ||||||
|  | = Stunden | ||||||
|  |  | ||||||
|  | == Kim Mustermensch | ||||||
|  |  | ||||||
|  | Zeitraum: 01.10.2025 - 31.10.2025 | ||||||
|  |  | ||||||
|  | Arbeitszeit: 136h 19min | ||||||
|  |  | ||||||
|  | Überstunden: -39h 41min | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | // #show table.cell: it => { | ||||||
|  | //   if it.y == 0 { | ||||||
|  | //     set text(white) | ||||||
|  | //     strong(it) | ||||||
|  | //   } else if it.body == [] { | ||||||
|  | //     // Replace empty cells with 'N/A' | ||||||
|  | //      pad(..it.inset)[0min] | ||||||
|  |      | ||||||
|  | //   } else { | ||||||
|  | //     it | ||||||
|  | //   } | ||||||
|  | // } | ||||||
|  |  | ||||||
|  | #let subgrid(body) = { | ||||||
|  |   table.cell(colspan: 3, inset: 0em)[ | ||||||
|  |     #table( | ||||||
|  |         columns: (1fr, 1fr, 1fr), | ||||||
|  |         gutter: 0em, | ||||||
|  |         stroke: black, | ||||||
|  |         [..#body] | ||||||
|  |     ) | ||||||
|  |   ] | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   "01.09.2025", | ||||||
|  |   "08:07", | ||||||
|  |   "16:28", | ||||||
|  |   "Büro", | ||||||
|  |   "7h 51min", | ||||||
|  |   "30min", | ||||||
|  |   "-9min", | ||||||
|  |     "02.09.2025", | ||||||
|  |     	// return work, pause, overtime | ||||||
|  | table.cell(colspan: 3, inset: 0em)[#table( | ||||||
|  |         columns: (1fr, 1fr, 1fr), | ||||||
|  |         gutter: 0em, | ||||||
|  |         stroke: black, | ||||||
|  |         [08:12], [16:24], [Büro], | ||||||
|  |         [16:30], [17:24], [Homeoffice] | ||||||
|  |       )], | ||||||
|  |   "6h", | ||||||
|  |   "0min", | ||||||
|  |   "-1h 15min" | ||||||
|  |  | ||||||
|  | ) | ||||||
							
								
								
									
										
											BIN
										
									
								
								WIR-typst/template.pdf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								WIR-typst/template.pdf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										2
									
								
								db.sql
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								db.sql
									
									
									
									
									
								
							| @@ -203,7 +203,7 @@ WITH params AS ( | |||||||
|         14::int AS start_days_ago,  -- how many days back to start |         14::int AS start_days_ago,  -- how many days back to start | ||||||
|         0::int  AS end_days_ahead,  -- how many days forward (0 = today) |         0::int  AS end_days_ahead,  -- how many days forward (0 = today) | ||||||
|         0::float AS pause_probability, |         0::float AS pause_probability, | ||||||
|         0.2::float AS absence_probability |         0.0::float AS absence_probability | ||||||
| ), | ), | ||||||
| days AS ( | days AS ( | ||||||
|     SELECT gs::date AS work_date, p.card_uid, p.geraet_id, p.pause_probability, p.absence_probability |     SELECT gs::date AS work_date, p.card_uid, p.geraet_id, p.pause_probability, p.absence_probability | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| -- reverse: modify "s_personal_daten" table | -- reverse: modify "s_personal_daten" table | ||||||
| ALTER TABLE "s_personal_daten" ALTER COLUMN "arbeitszeit_per_woche" DROP NOT NULL, ALTER COLUMN "vorgesetzter_pers_nr" DROP NOT NULL, ALTER COLUMN "arbeitszeit_per_tag" DROP NOT NULL, ALTER COLUMN "nachname" DROP NOT NULL, ALTER COLUMN "vorname" DROP NOT NULL; | ALTER TABLE "s_personal_daten" ALTER COLUMN "nachname" DROP NOT NULL, ALTER COLUMN "vorname" DROP NOT NULL; | ||||||
| -- reverse: modify "anwesenheit" table | -- reverse: modify "anwesenheit" table | ||||||
| ALTER TABLE "anwesenheit" ALTER COLUMN "anwesenheit_typ" DROP NOT NULL, ALTER COLUMN "geraet_id" DROP NOT NULL, ALTER COLUMN "timestamp" DROP NOT NULL; | ALTER TABLE "anwesenheit" ALTER COLUMN "anwesenheit_typ" DROP NOT NULL, ALTER COLUMN "geraet_id" DROP NOT NULL, ALTER COLUMN "timestamp" DROP NOT NULL; | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| -- modify "anwesenheit" table | -- modify "anwesenheit" table | ||||||
| ALTER TABLE "anwesenheit" ALTER COLUMN "timestamp" SET NOT NULL, ALTER COLUMN "geraet_id" SET NOT NULL, ALTER COLUMN "anwesenheit_typ" SET NOT NULL; | ALTER TABLE "anwesenheit" ALTER COLUMN "timestamp" SET NOT NULL, ALTER COLUMN "geraet_id" SET NOT NULL, ALTER COLUMN "anwesenheit_typ" SET NOT NULL; | ||||||
| -- modify "s_personal_daten" table | -- modify "s_personal_daten" table | ||||||
| ALTER TABLE "s_personal_daten" ALTER COLUMN "vorname" SET NOT NULL, ALTER COLUMN "nachname" SET NOT NULL, ALTER COLUMN "arbeitszeit_per_tag" SET NOT NULL, ALTER COLUMN "vorgesetzter_pers_nr" SET NOT NULL, ALTER COLUMN "arbeitszeit_per_woche" SET NOT NULL; | ALTER TABLE "s_personal_daten" ALTER COLUMN "vorname" SET NOT NULL, ALTER COLUMN "nachname" SET NOT NULL; | ||||||
|   | |||||||
| @@ -1,13 +1,19 @@ | |||||||
| ALTER TABLE wochen_report | ALTER TABLE wochen_report | ||||||
|   ALTER COLUMN ueberstunden TYPE interval |   ADD COLUMN ueberstunden_interval interval, | ||||||
|   USING |   ADD COLUMN arbeitszeit_interval interval; | ||||||
|     make_interval( |  | ||||||
|       hours => floor(ueberstunden)::int, | UPDATE wochen_report | ||||||
|       mins  => round((ueberstunden - floor(ueberstunden)) * 60)::int | SET | ||||||
|     ), |   ueberstunden_interval = CASE WHEN ueberstunden IS NULL THEN NULL ELSE (ueberstunden::double precision * INTERVAL '1 hour') END, | ||||||
|   ALTER COLUMN arbeitszeit TYPE interval |   arbeitszeit_interval  = CASE WHEN arbeitszeit  IS NULL THEN NULL ELSE (arbeitszeit::double precision  * INTERVAL '1 hour') END; | ||||||
|   USING |  | ||||||
|     make_interval( | -- when happy, drop old columns and rename new ones | ||||||
|       hours => floor(arbeitszeit)::int, | ALTER TABLE wochen_report | ||||||
|       mins  => round((arbeitszeit - floor(arbeitszeit)) * 60)::int |   DROP COLUMN ueberstunden, | ||||||
|     ); |   DROP COLUMN arbeitszeit; | ||||||
|  |  | ||||||
|  | ALTER TABLE wochen_report | ||||||
|  |   RENAME COLUMN ueberstunden_interval TO ueberstunden; | ||||||
|  |  | ||||||
|  | ALTER TABLE wochen_report | ||||||
|  |   RENAME COLUMN arbeitszeit_interval TO arbeitszeit; | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								migrations/20250916093608_kurzarbeit.down.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								migrations/20250916093608_kurzarbeit.down.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | -- reverse: modify "wochen_report" table | ||||||
|  | ALTER TABLE "wochen_report" ALTER COLUMN "personal_nummer" DROP NOT NULL; | ||||||
|  | -- reverse: modify "s_personal_daten" table | ||||||
|  | ALTER TABLE "s_anwesenheit_typen" ALTER COLUMN "anwesenheit_name" DROP NOT NULL; | ||||||
|  | -- reverse: set comment to column: "arbeitszeit_equivalent" on table: "s_abwesenheit_typen" | ||||||
|  | COMMENT ON COLUMN "s_abwesenheit_typen"."arbeitszeit_equivalent" IS NULL; | ||||||
|  | -- reverse: modify "s_abwesenheit_typen" table | ||||||
|  | ALTER TABLE "s_abwesenheit_typen" ALTER COLUMN "arbeitszeit_equivalent" DROP NOT NULL, ALTER COLUMN "abwesenheit_name" DROP NOT NULL; | ||||||
|  | -- reverse: modify "abwesenheit" table | ||||||
|  | ALTER TABLE "abwesenheit" DROP COLUMN "datum_to", ALTER COLUMN "datum_from" DROP NOT NULL, ALTER COLUMN "abwesenheit_typ" DROP NOT NULL, ALTER COLUMN "card_uid" DROP NOT NULL; | ||||||
|  | ALTER TABLE "abwesenheit" RENAME COLUMN "datum_from" TO "datum"; | ||||||
							
								
								
									
										11
									
								
								migrations/20250916093608_kurzarbeit.up.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								migrations/20250916093608_kurzarbeit.up.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | -- modify "abwesenheit" table | ||||||
|  | ALTER TABLE "abwesenheit" RENAME COLUMN "datum" TO "datum_from"; | ||||||
|  | ALTER TABLE "abwesenheit" ALTER COLUMN "card_uid" SET NOT NULL, ALTER COLUMN "abwesenheit_typ" SET NOT NULL, ALTER COLUMN "datum_from" SET NOT NULL, ADD COLUMN "datum_to" timestamptz; | ||||||
|  | -- modify "s_abwesenheit_typen" table | ||||||
|  | ALTER TABLE "s_abwesenheit_typen" ALTER COLUMN "abwesenheit_name" SET NOT NULL, ALTER COLUMN "arbeitszeit_equivalent" SET NOT NULL; | ||||||
|  | -- set comment to column: "arbeitszeit_equivalent" on table: "s_abwesenheit_typen" | ||||||
|  | COMMENT ON COLUMN "s_abwesenheit_typen"."arbeitszeit_equivalent" IS '0=keine Arbeitszeit; 1=Arbeitszeit auffüllen; 2=Arbeitszeit austauschen'; | ||||||
|  | -- modify "s_anwesenheit_typen" table | ||||||
|  | ALTER TABLE "s_anwesenheit_typen" ALTER COLUMN "anwesenheit_name" SET NOT NULL; | ||||||
|  | -- modify "s_personal_daten" table | ||||||
|  | ALTER TABLE "wochen_report" ALTER COLUMN "personal_nummer" SET NOT NULL; | ||||||
							
								
								
									
										4
									
								
								migrations/20251013212224_buchungs_array.down.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								migrations/20251013212224_buchungs_array.down.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | -- reverse: modify "wochen_report" table | ||||||
|  | ALTER TABLE "wochen_report" DROP COLUMN "abwesenheiten", DROP COLUMN "anwesenheiten", ALTER COLUMN "arbeitszeit" DROP NOT NULL, ALTER COLUMN "ueberstunden" DROP NOT NULL, ALTER COLUMN "woche_start" DROP NOT NULL; | ||||||
|  | -- reverse: modify "abwesenheit" table | ||||||
|  | ALTER TABLE "abwesenheit" ALTER COLUMN "datum_to" DROP NOT NULL; | ||||||
							
								
								
									
										4
									
								
								migrations/20251013212224_buchungs_array.up.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								migrations/20251013212224_buchungs_array.up.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | -- modify "abwesenheit" table | ||||||
|  | ALTER TABLE "abwesenheit" ALTER COLUMN "datum_to" SET NOT NULL; | ||||||
|  | -- modify "wochen_report" table | ||||||
|  | ALTER TABLE "wochen_report" ALTER COLUMN "woche_start" SET NOT NULL, ALTER COLUMN "ueberstunden" SET NOT NULL, ALTER COLUMN "arbeitszeit" SET NOT NULL, ADD COLUMN "anwesenheiten" integer[] NULL, ADD COLUMN "abwesenheiten" integer[] NULL; | ||||||
| @@ -1,7 +1,9 @@ | |||||||
| h1:31y/DB0ca2cm603nNckNvVCwm+XuMaVsoGUB+JvrOKs= | h1:gE7ikkZS7bQbedAjVspQKftSo5ODij2eiQGWbfQEYmI= | ||||||
| 20250901201159_initial.up.sql h1:Mb1RlVdFvcxqU9HrSK6oNeURqFa3O4KzB3rDa+6+3gc= | 20250901201159_initial.up.sql h1:Mb1RlVdFvcxqU9HrSK6oNeURqFa3O4KzB3rDa+6+3gc= | ||||||
| 20250901201250_control_tables.up.sql h1:a5LATgR/CRiC4GsqxkJ94TyJOxeTcW74eCnodIy+c1E= | 20250901201250_control_tables.up.sql h1:a5LATgR/CRiC4GsqxkJ94TyJOxeTcW74eCnodIy+c1E= | ||||||
| 20250901201710_triggers_extension.up.sql h1:z9b6Hk9btE2Ns4mU7B16HjvYBP6EEwHAXVlvPpkn978= | 20250901201710_triggers_extension.up.sql h1:z9b6Hk9btE2Ns4mU7B16HjvYBP6EEwHAXVlvPpkn978= | ||||||
| 20250903221313_overtime.up.sql h1:t/B435ShW5ZEnzC81jRABWVZ5gNm7tPZPnOO6/ZY6ow= | 20250903221313_overtime.up.sql h1:t/B435ShW5ZEnzC81jRABWVZ5gNm7tPZPnOO6/ZY6ow= | ||||||
| 20250903233030_non_null_contraints.up.sql h1:e7d+6ZdpEjPh2cc65N3S06oD2e6diMuG7+klhgYsym8= | 20250903233030_non_null_contraints.up.sql h1:YKeYgazfh+jPyh7hFT/pV+By8eHnk1taXnlgSLyXSA0= | ||||||
| 20250904114004_intervals.up.sql h1:Sz8FIVvvcCIS3aIuSjyzFYLs32fjMcMMHk62shj6Qpw= | 20250904114004_intervals.up.sql h1:gDdN8cJ4xH1vQhAbbhqD5lwdyEO1N9EIqEYkmWGiWIU= | ||||||
|  | 20250916093608_kurzarbeit.up.sql h1:yDAAMLyUXz6b7+MI6XK/HZMPzutKoT2NNNOCjFaqSts= | ||||||
|  | 20251013212224_buchungs_array.up.sql h1:mbhvnwMUkEFFQQ41NC47auqxbtvNkztziWvpLDFm6tA= | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user