commit eb6585bb6ee2628122ab6ee798b36029df8812a4 Author: Fabian Kowalski Date: Wed Feb 12 14:05:01 2025 +0100 first commit diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..0fe5e26 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +backend/node_modules +backend/package-lock.json \ No newline at end of file diff --git a/backend/dane.json b/backend/dane.json new file mode 100644 index 0000000..bacd50c --- /dev/null +++ b/backend/dane.json @@ -0,0 +1,42 @@ +{ + "1A": [ + { + "uczen": { + "imie": "Jan", + "nazwisko": "Kowalski", + "Zadanie domowe": 10, + "Aktywność": 3, + "Sprawdzian": 4 + } + }, + { + "uczen": { + "imie": "Anna", + "nazwisko": "Nowak", + "Zadanie domowe": 2, + "Aktywność": 1, + "Sprawdzian": 0 + } + } + ], + "2B": [ + { + "uczen": { + "imie": "Piotr", + "nazwisko": "Wiśniewski", + "Zadanie domowe": 10, + "Aktywność": 0, + "Sprawdzian": 0 + } + }, + { + "uczen": { + "imie": "Maria", + "nazwisko": "Kowalczyk", + "Zadanie domowe": 3, + "Aktywność": 2, + "Sprawdzian": 1 + } + } + ] +} \ No newline at end of file diff --git a/backend/kryteria.json b/backend/kryteria.json new file mode 100644 index 0000000..0959157 --- /dev/null +++ b/backend/kryteria.json @@ -0,0 +1,5 @@ +[ + "Zadanie domowe", + "Aktywność", + "Sprawdzian" +] \ No newline at end of file diff --git a/backend/package.json b/backend/package.json new file mode 100644 index 0000000..913f2e0 --- /dev/null +++ b/backend/package.json @@ -0,0 +1,8 @@ +{ + "dependencies": { + "cors": "^2.8.5", + "express": "^4.21.2", + "fs": "^0.0.1-security", + "path": "^0.12.7" + } +} diff --git a/backend/server.js b/backend/server.js new file mode 100644 index 0000000..c6a88bd --- /dev/null +++ b/backend/server.js @@ -0,0 +1,67 @@ +const express = require("express"); +const fs = require("fs"); +const path = require("path"); +const cors = require("cors"); + +const app = express(); +const PORT = 4000; + +// Ścieżki do plików JSON +const UCZNIOWIE_FILE = path.join(__dirname, "uczniowie.json"); +const KRYTERIA_FILE = path.join(__dirname, "kryteria.json"); +const DANE_FILE = path.join(__dirname, "dane.json"); + +// Middleware do obsługi JSON i CORS +app.use(express.json()); +app.use(cors()); + +// Endpoint do pobierania listy uczniów +app.get("/uczniowie", (req, res) => { + fs.readFile(UCZNIOWIE_FILE, "utf8", (err, data) => { + if (err) { + console.error("Błąd odczytu uczniów:", err); + return res.status(500).send("Błąd odczytu uczniów."); + } + res.json(JSON.parse(data)); + }); +}); + +// Endpoint do pobierania listy kryteriów +app.get("/kryteria", (req, res) => { + fs.readFile(KRYTERIA_FILE, "utf8", (err, data) => { + if (err) { + console.error("Błąd odczytu kryteriów:", err); + return res.status(500).send("Błąd odczytu kryteriów."); + } + res.json(JSON.parse(data)); + }); +}); + +// Endpoint do pobierania danych uczniów +app.get("/dane", (req, res) => { + fs.readFile(DANE_FILE, "utf8", (err, data) => { + if (err) { + console.error("Błąd odczytu danych:", err); + return res.status(500).send("Błąd odczytu danych."); + } + res.json(JSON.parse(data)); + }); +}); + +// Endpoint do zapisywania danych uczniów +app.post("/dane", (req, res) => { + const newData = req.body; + + fs.writeFile(DANE_FILE, JSON.stringify(newData, null, 2), err => { + if (err) { + console.error("Błąd zapisywania danych:", err); + return res.status(500).send("Błąd zapisywania danych."); + } + res.send("Dane zapisane pomyślnie."); + }); +}); + +// Uruchomienie serwera +app.listen(PORT, () => { + console.log(`Serwer API działa na http://localhost:${PORT}`); +}); \ No newline at end of file diff --git a/backend/uczniowie.json b/backend/uczniowie.json new file mode 100644 index 0000000..96740bd --- /dev/null +++ b/backend/uczniowie.json @@ -0,0 +1,12 @@ +{ + "klasy": { + "1A": [ + { "imie": "Jan", "nazwisko": "Kowalski" }, + { "imie": "Anna", "nazwisko": "Nowak" } + ], + "2B": [ + { "imie": "Piotr", "nazwisko": "Wiśniewski" }, + { "imie": "Maria", "nazwisko": "Kowalczyk" } + ] + } +} \ No newline at end of file diff --git a/index.html b/index.html new file mode 100644 index 0000000..225cfd9 --- /dev/null +++ b/index.html @@ -0,0 +1,26 @@ + + + + + + Zarządzanie Uczniami + + + +
+

Zarządzanie Uczniami

+ + + + + + +
+ + + +
+ + + + \ No newline at end of file diff --git a/script.js b/script.js new file mode 100644 index 0000000..f14dcc4 --- /dev/null +++ b/script.js @@ -0,0 +1,121 @@ +let uczniowie = {}; +let kryteria = []; +let dane = {}; + +document.addEventListener("DOMContentLoaded", () => { + const classSelect = document.getElementById("class-select"); + const studentsList = document.getElementById("students-list"); + const saveButton = document.getElementById("save-button"); + + // Ładowanie danych z serwera + Promise.all([ + fetch("http://localhost:4000/uczniowie").then(response => response.json()), + fetch("http://localhost:4000/kryteria").then(response => response.json()), + fetch("http://localhost:4000/dane").then(response => response.json()) + ]) + .then(([uczniowieData, kryteriaData, daneData]) => { + uczniowie = uczniowieData.klasy; + kryteria = kryteriaData; + dane = daneData; + + populateClassSelect(); + }) + .catch(error => console.error("Błąd ładowania danych:", error)); + + // Wypełnij listę klas + function populateClassSelect() { + Object.keys(uczniowie).forEach(klasa => { + const option = document.createElement("option"); + option.value = klasa; + option.textContent = klasa; + classSelect.appendChild(option); + }); + + // Wyświetl pierwszą klasę domyślnie + if (classSelect.options.length > 0) { + classSelect.value = classSelect.options[0].value; + renderStudents(classSelect.value); + } + } + + // Aktualizuj listę uczniów po wyborze klasy + classSelect.addEventListener("change", () => { + const selectedClass = classSelect.value; + renderStudents(selectedClass); + }); + + // Renderuj uczniów + function renderStudents(className) { + studentsList.innerHTML = ""; + const students = uczniowie[className]; + + students.forEach(student => { + const studentKey = `${className}-${student.imie}-${student.nazwisko}`; + const details = document.createElement("details"); + const summary = document.createElement("summary"); + + summary.textContent = `${student.imie} ${student.nazwisko}`; + details.appendChild(summary); + + const criteriaList = document.createElement("div"); + criteriaList.className = "criteria-list"; + + kryteria.forEach(criteria => { + const criteriaItem = document.createElement("div"); + criteriaItem.className = "criteria-item"; + + const label = document.createElement("span"); + label.textContent = criteria; + + const input = document.createElement("input"); + input.type = "number"; + input.min = "0"; + + // Wyszukaj dane ucznia w pliku `dane.json` + const savedStudent = dane[className]?.find(s => s.uczen.imie === student.imie && s.uczen.nazwisko === student.nazwisko); + input.value = savedStudent?.uczen[criteria] || 0; + + input.addEventListener("input", () => { + if (!savedStudent) { + dane[className] = dane[className] || []; + dane[className].push({ + uczen: { + imie: student.imie, + nazwisko: student.nazwisko + } + }); + } + + const currentStudent = dane[className].find(s => s.uczen.imie === student.imie && s.uczen.nazwisko === student.nazwisko); + currentStudent.uczen[criteria] = parseInt(input.value) || 0; + }); + + criteriaItem.appendChild(label); + criteriaItem.appendChild(input); + criteriaList.appendChild(criteriaItem); + }); + + details.appendChild(criteriaList); + studentsList.appendChild(details); + }); + } + + // Zapisz dane na zewnętrznym serwerze + saveButton.addEventListener("click", () => { + fetch("http://localhost:4000/dane", { + method: "POST", + headers: { + "Content-Type": "application/json" + }, + body: JSON.stringify(dane) + }) + .then(response => { + if (response.ok) { + alert("Dane zostały zapisane!"); + } else { + alert("Wystąpił błąd podczas zapisywania danych."); + } + }) + .catch(error => console.error("Błąd zapisywania danych:", error)); + }); +}); \ No newline at end of file diff --git a/style.css b/style.css new file mode 100644 index 0000000..dd9840e --- /dev/null +++ b/style.css @@ -0,0 +1,84 @@ +body { + font-family: Arial, sans-serif; + margin: 20px; + background-color: #f4f4f9; +} + +.container { + max-width: 800px; + margin: 0 auto; + padding: 20px; + background: #fff; + border-radius: 8px; + box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1); +} + +h1 { + text-align: center; + color: #333; +} + +label { + display: block; + margin-top: 20px; + font-weight: bold; +} + +select { + width: 100%; + padding: 10px; + margin-top: 5px; + border: 1px solid #ccc; + border-radius: 4px; +} + +#students-list { + margin-top: 20px; +} + +details { + margin-bottom: 10px; + padding: 10px; + border: 1px solid #ddd; + border-radius: 4px; + background-color: #f9f9f9; +} + +summary { + font-weight: bold; + cursor: pointer; + outline: none; +} + +.criteria-list { + margin-left: 20px; +} + +.criteria-item { + display: flex; + align-items: center; + margin: 5px 0; +} + +.criteria-item input { + width: 50px; + margin-left: 10px; + padding: 5px; + border: 1px solid #ccc; + border-radius: 4px; +} + +button { + display: block; + margin: 20px auto; + padding: 10px 20px; + background-color: #007bff; + color: #fff; + border: none; + border-radius: 4px; + cursor: pointer; +} + +button:hover { + background-color: #0056b3; +} \ No newline at end of file