feat: 多班级版 v2.0 - Go后端重写 + 43轮代码审查

- 后端从 Python FastAPI 重写为 Go Gin(端口 56789)
- 多班级完全隔离
- 超级管理员独立登录
- 课代表作业管理、排行榜分项排行
- 角色加减分上下限可配置
- 家长改密功能(可开关)
- 周度/月度重置功能
- MySQL 5.7 兼容
- 43轮代码审查+全部修复
- Apache 2.0 许可证
This commit is contained in:
2026-06-22 10:06:10 +08:00
parent 4084afc53c
commit d6dec878bd
214 changed files with 12622 additions and 9725 deletions

View File

@@ -1,11 +1,11 @@
<?php
/**
* 班级操行分管理系统 - 公共头部
* 多班级版班级管理系统 - 公共头部
*
* 开发者: Canglan
* 联系方式: admin@sea-studio.top
* 版权归属: Sea Network Technology Studio
* 许可证: MIT License
* 许可证: Apache License 2.0
*
* 版权所有 © Sea Network Technology Studio
*/
@@ -17,6 +17,8 @@ if (!isset($_SESSION)) {
$current_page = basename($_SERVER['PHP_SELF'], '.php');
$user_type = $_SESSION['user_type'] ?? '';
$role = $_SESSION['role'] ?? '';
$class_id = $_SESSION['class_id'] ?? null;
$class_name = $_SESSION['class_name'] ?? '';
$page_title = $page_title ?? '首页';
?>
<!DOCTYPE html>
@@ -24,7 +26,7 @@ $page_title = $page_title ?? '首页';
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=no">
<title><?php echo SITE_NAME; ?> - <?php echo $page_title; ?></title>
<title><?php echo htmlspecialchars(SITE_NAME, ENT_QUOTES, 'UTF-8'); ?> - <?php echo htmlspecialchars($page_title); ?></title>
<link rel="stylesheet" href="/assets/css/style.css">
<?php if ($user_type === 'admin'): ?>
<link rel="stylesheet" href="/assets/css/admin.css">
@@ -32,8 +34,11 @@ $page_title = $page_title ?? '首页';
</head>
<body>
<div class="header">
<h1><?php echo SITE_NAME; ?></h1>
<h1><?php echo htmlspecialchars(SITE_NAME, ENT_QUOTES, 'UTF-8'); ?></h1>
<div class="header-info">
<?php if ($class_name): ?>
<span class="class-name" id="className"><?php echo htmlspecialchars($class_name); ?></span>
<?php endif; ?>
<span class="user-name" id="userName"><?php echo htmlspecialchars($_SESSION['real_name'] ?? ''); ?></span>
<?php if ($role): ?>
<span class="user-role">(<?php echo htmlspecialchars($role); ?>)</span>
@@ -42,38 +47,46 @@ $page_title = $page_title ?? '首页';
</div>
</div>
<script>
window.API_BASE_URL = '<?php echo API_BASE_URL; ?>';
window.JWT_STORAGE_KEY = '<?php echo JWT_STORAGE_KEY; ?>';
window.USER_STORAGE_KEY = '<?php echo USER_STORAGE_KEY; ?>';
window.API_BASE_URL = <?php echo json_encode(API_BASE_URL); ?>;
window.JWT_STORAGE_KEY = <?php echo json_encode(JWT_STORAGE_KEY); ?>;
window.USER_STORAGE_KEY = <?php echo json_encode(USER_STORAGE_KEY); ?>;
window.CLASS_ID = <?php echo $class_id ? $class_id : 'null'; ?>;
window.CLASS_NAME = <?php echo json_encode($class_name); ?>;
</script>
<script>
// 从后端API步加载扣分规则配置
// 从后端API步加载扣分规则配置(优先加载班级级配置)
(function() {
var xhr = new XMLHttpRequest();
xhr.open('GET', window.API_BASE_URL + '/api/config/deduction-rules', false); // 同步请求
try {
xhr.send();
if (xhr.status === 200) {
var resp = JSON.parse(xhr.responseText);
if (resp.success && resp.data) {
window.DEDUCTION_HOMEWORK_NOT_SUBMIT = resp.data.DEDUCTION_HOMEWORK_NOT_SUBMIT;
window.DEDUCTION_HOMEWORK_LATE = resp.data.DEDUCTION_HOMEWORK_LATE;
window.DEDUCTION_ATTENDANCE_ABSENT = resp.data.DEDUCTION_ATTENDANCE_ABSENT;
window.DEDUCTION_ATTENDANCE_LATE = resp.data.DEDUCTION_ATTENDANCE_LATE;
window.DEDUCTION_ATTENDANCE_LEAVE = resp.data.DEDUCTION_ATTENDANCE_LEAVE;
window.STUDENT_INITIAL_POINTS = resp.data.STUDENT_INITIAL_POINTS;
}
}
} catch(e) {
// API加载失败时使用默认值
window.DEDUCTION_HOMEWORK_NOT_SUBMIT = 2;
window.DEDUCTION_HOMEWORK_LATE = 1;
window.DEDUCTION_ATTENDANCE_ABSENT = 3;
window.DEDUCTION_ATTENDANCE_LATE = 1;
window.DEDUCTION_ATTENDANCE_LEAVE = 0;
window.STUDENT_INITIAL_POINTS = 60;
// 先设置默认值,避免异步加载期间其他脚本读取到 undefined
window.DEDUCTION_HOMEWORK_NOT_SUBMIT = 2;
window.DEDUCTION_HOMEWORK_LATE = 1;
window.DEDUCTION_ATTENDANCE_ABSENT = 3;
window.DEDUCTION_ATTENDANCE_LATE = 1;
window.DEDUCTION_ATTENDANCE_LEAVE = 0;
window.STUDENT_INITIAL_POINTS = 60;
var token = localStorage.getItem(window.JWT_STORAGE_KEY) || '';
var apiUrl = window.API_BASE_URL + '/api/config/deduction-rules';
if (window.CLASS_ID) {
apiUrl += '?class_id=' + window.CLASS_ID;
}
fetch(apiUrl, {
headers: token ? { 'Authorization': 'Bearer ' + token } : {}
}).then(function(resp) {
return resp.json();
}).then(function(data) {
if (data.success && data.data) {
window.DEDUCTION_HOMEWORK_NOT_SUBMIT = data.data.DEDUCTION_HOMEWORK_NOT_SUBMIT;
window.DEDUCTION_HOMEWORK_LATE = data.data.DEDUCTION_HOMEWORK_LATE;
window.DEDUCTION_ATTENDANCE_ABSENT = data.data.DEDUCTION_ATTENDANCE_ABSENT;
window.DEDUCTION_ATTENDANCE_LATE = data.data.DEDUCTION_ATTENDANCE_LATE;
window.DEDUCTION_ATTENDANCE_LEAVE = data.data.DEDUCTION_ATTENDANCE_LEAVE;
window.STUDENT_INITIAL_POINTS = data.data.STUDENT_INITIAL_POINTS;
}
}).catch(function() {
// API加载失败时保留默认值
});
})();
</script>
<script src="/assets/js/common.js"></script>
<script src="/assets/js/modules/utils.js"></script>
<script src="/assets/js/modules/utils.js"></script>
<div class="container">