v0.5测试

This commit is contained in:
2026-04-14 11:35:56 +08:00
parent 2f271b1dc4
commit bf5b2516e2
17 changed files with 646 additions and 439 deletions

View File

@@ -20,7 +20,7 @@ if (!isset($_SESSION['user_id']) || $_SESSION['user_type'] !== 'admin') {
$page_title = '作业管理';
$role = $_SESSION['role'] ?? '';
if (!in_array($role, ['班主任', '科代表'])) {
if (!in_array($role, ['班主任', '学习委员'])) {
header('Location: /admin/dashboard.php');
exit();
}
@@ -28,142 +28,152 @@ if (!in_array($role, ['班主任', '科代表'])) {
include __DIR__ . '/../includes/header.php';
?>
<div class="nav">
<a href="/admin/dashboard.php" class="nav-item">首页</a>
<a href="/admin/students.php" class="nav-item">学生管理</a>
<?php if ($role === '班主任' || $role === '班长'): ?>
<a href="/admin/conduct.php" class="nav-item">操行分管理</a>
<?php endif; ?>
<a href="/admin/homework.php" class="nav-item active">作业管理</a>
<?php if ($role === '班主任' || $role === '考勤委员'): ?>
<a href="/admin/attendance.php" class="nav-item">考勤管理</a>
<?php endif; ?>
<?php if ($role === '班主任'): ?>
<a href="/admin/subjects.php" class="nav-item">科目管理</a>
<a href="/admin/admins.php" class="nav-item">管理员管理</a>
<?php endif; ?>
<a href="/admin/history.php" class="nav-item">历史记录</a>
<a href="/admin/password.php" class="nav-item">修改密码</a>
</div>
<?php include __DIR__ . '/../includes/nav.php'; ?>
<div class="container">
<div class="card">
<div class="action-bar">
<?php if ($role === '班主任'): ?>
<button class="btn btn-primary" onclick="showAddAssignmentModal()">发布作业</button>
<?php endif; ?>
<div class="action-buttons">
<button class="btn btn-primary" onclick="showBatchPointsModal()">批量加减分</button>
</div>
</div>
<div class="table-wrapper">
<table class="table">
<thead>
<tr>
<th><input type="checkbox" id="selectAll" onclick="toggleSelectAll()"></th>
<th>学号</th>
<th>姓名</th>
<th>当前操行分</th>
<th>操作</th>
</tr>
</thead>
<tbody id="studentList"></tbody>
</table>
</div>
<div id="assignmentsList"></div>
</div>
</div>
<!-- 发布作业模态框 -->
<div id="addAssignmentModal" class="modal">
<div id="batchPointsModal" class="modal">
<div class="modal-content">
<div class="modal-header">
<h3>发布作业</h3>
<button class="modal-close" onclick="closeModal('addAssignmentModal')">&times;</button>
<h3>批量加减分</h3>
<button class="modal-close" onclick="closeModal('batchPointsModal')">&times;</button>
</div>
<form onsubmit="event.preventDefault(); submitAddAssignment()">
<form onsubmit="event.preventDefault(); handleSubmitPoints()">
<div class="form-group">
<label>科目</label>
<select id="assignmentSubjectId" required></select>
<label>已选学生</label>
<div id="selectedStudentsCount" class="selected-info">未选择学生</div>
</div>
<div class="form-group">
<label>作业标题</label>
<input type="text" id="assignmentTitle" required>
<label>扣分类型</label>
<div class="deduction-types">
<button type="button" class="btn btn-sm" onclick="selectDeductionType(-window.DEDUCTION_HOMEWORK_NOT_SUBMIT, '未交作业')">未交作业(-<span class="hw-not-submit"></span>分)</button>
<button type="button" class="btn btn-sm" onclick="selectDeductionType(-window.DEDUCTION_HOMEWORK_LATE, '迟交作业')">迟交作业(-<span class="hw-late"></span>分)</button>
<button type="button" class="btn btn-sm" onclick="selectDeductionType(0, '')">自定义</button>
</div>
</div>
<div class="form-group">
<label>作业描述</label>
<textarea id="assignmentDescription" rows="3"></textarea>
<label>分数变动</label>
<input type="number" id="pointsChange" required min="-3" max="3" step="1" placeholder="正数加分,负数扣分">
<small>每次加减分不超过<span class="hw-max"></span>分</small>
</div>
<div class="form-group">
<label>截止日期</label>
<input type="date" id="assignmentDeadline" required>
<label>原因</label>
<textarea id="pointsReason" rows="3" required placeholder="请输入加减分原因"></textarea>
</div>
<div class="modal-footer">
<button type="submit" class="btn btn-primary">发布</button>
<button type="button" class="btn" onclick="closeModal('addAssignmentModal')">取消</button>
<button type="submit" class="btn btn-primary">确认提交</button>
<button type="button" class="btn" onclick="closeModal('batchPointsModal')">取消</button>
</div>
</form>
</div>
</div>
<script>
async function loadAssignments() {
const res = await apiGet('/api/admin/homework/assignments');
if (res && res.success) {
let html = '';
for (const assignment of res.data.assignments) {
html += `
<div class="card assignment-card">
<div class="assignment-header">
<div><span class="assignment-title">${escapeHtml(assignment.title)}</span> <span class="assignment-meta">(${escapeHtml(assignment.subject_name)})</span></div>
<div class="assignment-meta">截止: ${assignment.deadline}</div>
</div>
<div class="table-wrapper">
<table class="table">
<thead><tr><th>学生</th><th>状态</th><th>备注</th><th>操作</th></tr></thead>
<tbody id="submissions-${assignment.assignment_id}"><tr><td colspan="4" class="loading">加载中...</td></tr></tbody>
</table>
</div>
</div>
`;
}
if (res.data.assignments.length === 0) {
html = '<div style="text-align:center;padding:40px;">暂无作业</div>';
}
document.getElementById('assignmentsList').innerHTML = html;
for (const assignment of res.data.assignments) {
await loadSubmissions(assignment.assignment_id);
}
}
}
let selectedStudentIds = [];
async function loadSubmissions(assignmentId) {
const res = await apiGet(`/api/admin/homework/submissions/${assignmentId}`);
// 初始化扣分配置
const hwMaxPoints = window.HOMEWORK_MAX_POINTS || 3;
const hwNotSubmit = window.DEDUCTION_HOMEWORK_NOT_SUBMIT || 2;
const hwLate = window.DEDUCTION_HOMEWORK_LATE || 1;
// 更新页面中的配置值显示
document.querySelectorAll('.hw-not-submit').forEach(el => el.textContent = hwNotSubmit);
document.querySelectorAll('.hw-late').forEach(el => el.textContent = hwLate);
document.querySelectorAll('.hw-max').forEach(el => el.textContent = hwMaxPoints);
// 更新输入框的 min/max
document.getElementById('pointsChange').setAttribute('min', -hwMaxPoints);
document.getElementById('pointsChange').setAttribute('max', hwMaxPoints);
async function loadStudents() {
const res = await apiGet('/api/admin/students');
if (res && res.success) {
let html = '';
res.data.submissions.forEach(sub => {
res.data.students.forEach(student => {
html += `<tr>
<td>${escapeHtml(sub.student_name)}</td>
<td>${getStatusBadge(sub.status, 'homework')}</td>
<td>${escapeHtml(sub.comments || '-')}</td>
<td>
<select id="status-${sub.submission_id}" class="status-select">
<option value="submitted" ${sub.status === 'submitted' ? 'selected' : ''}>已提交</option>
<option value="not_submitted" ${sub.status === 'not_submitted' ? 'selected' : ''}>未提交</option>
<option value="late" ${sub.status === 'late' ? 'selected' : ''}>迟交</option>
</select>
<label><input type="checkbox" id="deduct-${sub.submission_id}"> 同时扣分</label>
<button class="btn btn-sm btn-primary" onclick="updateSubmission(${sub.submission_id})">更新</button>
</td>
<td><input type="checkbox" class="student-checkbox" data-id="${student.student_id}"></td>
<td>${escapeHtml(student.student_no)}</td>
<td>${escapeHtml(student.name)}</td>
<td>${student.total_points}</td>
<td><button class="btn btn-sm btn-primary" onclick="showSinglePointsModal(${student.student_id}, '${escapeHtml(student.name)}')">加减分</button></td>
</tr>`;
});
document.getElementById(`submissions-${assignmentId}`).innerHTML = html;
if (res.data.students.length === 0) {
html = '<tr><td colspan="5" style="text-align:center;">暂无学生数据</td></tr>';
}
document.getElementById('studentList').innerHTML = html;
}
}
async function updateSubmission(submissionId) {
const status = document.getElementById(`status-${submissionId}`).value;
const applyDeduction = document.getElementById(`deduct-${submissionId}`).checked;
const res = await apiPut('/api/admin/homework/submission', {
submission_id: submissionId,
status: status,
apply_deduction: applyDeduction
function showSinglePointsModal(studentId, studentName) {
selectedStudentIds = [studentId];
document.getElementById('selectedStudentsCount').innerHTML = `${studentName} (1人)`;
document.getElementById('pointsChange').value = '';
document.getElementById('pointsReason').value = '';
document.getElementById('batchPointsModal').style.display = 'flex';
}
function showBatchPointsModal() {
selectedStudentIds = [];
document.querySelectorAll('.student-checkbox:checked').forEach(cb => {
selectedStudentIds.push(parseInt(cb.dataset.id));
});
if (res && res.success) {
showToast('更新成功');
loadAssignments();
if (selectedStudentIds.length === 0) {
showToast('请先选择学生', 'warning');
return;
}
document.getElementById('selectedStudentsCount').innerHTML = `已选择 ${selectedStudentIds.length} 名学生`;
document.getElementById('pointsChange').value = '';
document.getElementById('pointsReason').value = '';
document.getElementById('batchPointsModal').style.display = 'flex';
}
function selectDeductionType(points, reason) {
document.getElementById('pointsChange').value = points;
if (points !== 0) {
document.getElementById('pointsReason').value = reason;
} else {
showToast(res?.message || '更新失败', 'error');
document.getElementById('pointsReason').value = '';
document.getElementById('pointsReason').focus();
}
}
loadAssignments();
function handleSubmitPoints() {
const pointsChange = parseInt(document.getElementById('pointsChange').value);
if (isNaN(pointsChange) || pointsChange === 0) {
showToast('请输入有效的加减分值', 'warning');
return;
}
if (Math.abs(pointsChange) > hwMaxPoints) {
showToast(`每次加减分不超过${hwMaxPoints}分`, 'warning');
return;
}
submitBatchPoints();
}
loadStudents();
loadStudents();
</script>
<script src="/assets/js/admin.js"></script>