S2OJ/web/app/controllers/group_manage.php

500 lines
15 KiB
PHP
Raw Normal View History

2022-09-26 03:26:17 +00:00
<?php
2022-11-10 00:16:48 +00:00
requirePHPLib('form');
requirePHPLib('judger');
requirePHPLib('data');
2022-09-26 03:26:17 +00:00
2022-11-11 00:20:33 +00:00
Auth::check() || redirectToLogin();
UOJGroup::init(UOJRequest::get('id')) || UOJResponse::page404();
UOJGroup::cur()->userCanManage(Auth::user()) || UOJResponse::page403();
2022-11-10 00:16:48 +00:00
2022-11-11 00:20:33 +00:00
$cur_tab = UOJRequest::get('tab', 'is_string', 'profile');
2022-11-10 00:16:48 +00:00
$tabs_info = [
'profile' => [
'name' => '基本信息',
2022-11-11 00:20:33 +00:00
'url' => '/group/' . UOJGroup::info('id') . '/manage/profile',
2022-11-10 00:16:48 +00:00
],
'assignments' => [
'name' => '作业管理',
2022-11-11 00:20:33 +00:00
'url' => '/group/' . UOJGroup::info('id') . '/manage/assignments',
2022-11-10 00:16:48 +00:00
],
'users' => [
'name' => '用户管理',
2022-11-11 00:20:33 +00:00
'url' => '/group/' . UOJGroup::info('id') . '/manage/users',
2022-11-10 00:16:48 +00:00
]
];
if (!isset($tabs_info[$cur_tab])) {
2022-11-11 00:20:33 +00:00
UOJResponse::page404();
2022-11-10 00:16:48 +00:00
}
if ($cur_tab == 'profile') {
2023-01-15 00:48:08 +00:00
$update_profile_form = new UOJForm('update_profile');
$update_profile_form->addInput('name', [
'label' => '名称',
2023-02-14 10:52:31 +00:00
'default_value' => HTML::unescape(UOJGroup::info('title')),
2023-01-15 00:48:08 +00:00
'validator_php' => function ($title, &$vdata) {
2022-11-10 00:16:48 +00:00
if ($title == '') {
return '名称不能为空';
}
if (strlen($title) > 100) {
return '名称过长';
}
2022-09-26 03:26:17 +00:00
2022-11-10 00:16:48 +00:00
$title = HTML::escape($title);
if ($title === '') {
return '无效编码';
}
$vdata['title'] = $title;
return '';
},
2023-01-15 00:48:08 +00:00
]);
$update_profile_form->addCheckboxes('is_hidden', [
'div_class' => 'mt-3',
'label' => '可见性',
'label_class' => 'me-3',
'options' => [
0 => '公开',
1 => '隐藏',
],
'select_class' => 'd-inline-block',
'option_div_class' => 'form-check d-inline-block ms-2',
'default_value' => UOJGroup::info('is_hidden'),
]);
$update_profile_form->addTextArea('announcement', [
'div_class' => 'mt-3',
'label' => '公告',
'input_class' => 'form-control font-monospace',
'default_value' => UOJGroup::info('announcement'),
'help' => '公告支持 Markdown 语法。',
'validator_php' => function ($announcement, &$vdata) {
2022-11-10 00:16:48 +00:00
if (strlen($announcement) > 3000) {
return '公告过长';
}
$vdata['announcement'] = $announcement;
return '';
},
2023-01-15 00:48:08 +00:00
]);
2022-11-11 00:20:33 +00:00
$update_profile_form->handle = function ($vdata) {
DB::update([
"update `groups`",
"set", [
"title" => $vdata['title'],
"is_hidden" => $_POST['is_hidden'],
"announcement" => $vdata['announcement'],
],
"where", [
"id" => UOJGroup::info('id'),
],
]);
2022-11-10 00:16:48 +00:00
dieWithJsonData(['status' => 'success', 'message' => '修改成功']);
};
$update_profile_form->setAjaxSubmit(<<<EOD
2023-01-15 00:48:08 +00:00
function(res) {
if (res.status === 'success') {
$('#result-alert')
.html('小组信息修改成功!')
.addClass('alert-success')
.removeClass('alert-danger')
.show();
} else {
$('#result-alert')
.html('小组信息修改失败。' + (res.message || ''))
.removeClass('alert-success')
.addClass('alert-danger')
.show();
}
2022-09-26 03:26:17 +00:00
2023-01-15 00:48:08 +00:00
$(window).scrollTop(0);
}
EOD);
$update_profile_form->config['submit_button']['class'] = 'btn btn-secondary';
$update_profile_form->config['submit_button']['text'] = '更新';
2022-11-10 00:16:48 +00:00
$update_profile_form->runAtServer();
} elseif ($cur_tab == 'assignments') {
if (isset($_POST['submit-remove_assignment']) && $_POST['submit-remove_assignment'] == 'remove_assignment') {
2022-11-11 00:20:33 +00:00
$list_id = UOJRequest::post('list_id');
2022-11-10 00:16:48 +00:00
2022-11-11 00:20:33 +00:00
$list = UOJGroupAssignment::query($list_id);
if (!$list || !$list->valid()) {
dieWithAlert('题单不合法。');
2022-11-10 00:16:48 +00:00
}
2022-10-21 12:42:48 +00:00
2022-11-11 00:20:33 +00:00
DB::delete([
"delete from groups_assignments",
"where", [
"list_id" => $list->info['id'],
"group_id" => UOJGroup::info('id'),
],
]);
2022-11-10 00:16:48 +00:00
dieWithAlert('移除成功!');
}
2023-01-15 00:48:08 +00:00
$add_new_assignment_form = new UOJForm('add_new_assignment');
$add_new_assignment_form->addInput('new_assignment_list_id', [
'label' => '题单 ID',
'validator_php' => function ($list_id, &$vdata) {
2022-10-21 12:42:48 +00:00
if (!validateUInt($list_id)) {
2022-11-10 00:16:48 +00:00
return '题单 ID 不合法';
2022-09-26 03:26:17 +00:00
}
2022-11-10 00:16:48 +00:00
$list = UOJList::query($list_id);
if (!$list) {
return '题单不存在';
2022-09-26 03:26:17 +00:00
}
2022-11-10 00:16:48 +00:00
if ($list->info['is_hidden']) {
return '题单是隐藏的';
}
2022-09-26 03:26:17 +00:00
2022-11-11 00:20:33 +00:00
if (UOJGroup::cur()->hasAssignment($list)) {
2022-11-10 00:16:48 +00:00
return '该题单已经在作业中';
}
2022-09-26 03:26:17 +00:00
2022-11-10 00:16:48 +00:00
$vdata['list_id'] = $list_id;
return '';
},
2023-01-15 00:48:08 +00:00
]);
$add_new_assignment_form->addInput('new_assignment_end_time', [
'label' => '截止时间',
'default_value' => UOJTime::time2str((new DateTime())->add(new DateInterval("P7D"))->setTime(22, 30, 0)),
'validator_php' => function ($end_time, &$vdata) {
2022-11-10 00:16:48 +00:00
try {
$vdata['end_time'] = new DateTime($end_time);
} catch (Exception $e) {
return '无效时间格式';
}
return '';
},
2023-01-15 00:48:08 +00:00
]);
2022-11-11 00:20:33 +00:00
$add_new_assignment_form->handle = function (&$vdata) {
DB::insert([
"insert into groups_assignments",
DB::bracketed_fields(["group_id", "list_id", "end_time"]),
"values", DB::tuple([
UOJGroup::info('id'),
$vdata['list_id'],
$vdata['end_time']->format('Y-m-d H:i:s'),
]),
]);
2022-11-10 00:16:48 +00:00
dieWithJsonData([
'status' => 'success',
'message' => '题单 #' . $vdata['list_id'] . ' 已经被添加到作业列表中,结束时间为 ' . $vdata['end_time']->format('Y-m-d H:i:s') . '。'
]);
};
2023-01-15 00:48:08 +00:00
$add_new_assignment_form->config['submit_button']['class'] = 'btn btn-secondary';
$add_new_assignment_form->config['submit_button']['text'] = '添加';
2022-11-10 00:16:48 +00:00
$add_new_assignment_form->setAjaxSubmit(<<<EOD
2023-01-15 00:48:08 +00:00
function(res) {
if (res.status === 'success') {
$('#result-alert')
.html('作业添加成功!' + (res.message || ''))
.addClass('alert-success')
.removeClass('alert-danger')
.show();
} else {
$('#result-alert')
.html('作业添加失败。' + (res.message || ''))
.removeClass('alert-success')
.addClass('alert-danger')
.show();
}
2022-09-26 03:26:17 +00:00
2023-01-15 00:48:08 +00:00
$(window).scrollTop(0);
}
EOD);
2022-11-10 00:16:48 +00:00
$add_new_assignment_form->runAtServer();
$hidden_time = new DateTime();
2022-11-11 00:20:33 +00:00
$hidden_time->sub(new DateInterval('P3D'));
2022-11-10 00:16:48 +00:00
} elseif ($cur_tab == 'users') {
if (isset($_POST['submit-remove_user']) && $_POST['submit-remove_user'] == 'remove_user') {
2022-11-11 00:20:33 +00:00
$user = UOJUser::query(UOJRequest::post('remove_username'));
2022-11-10 00:16:48 +00:00
2022-11-11 00:20:33 +00:00
if (!$user) {
2022-11-10 00:16:48 +00:00
dieWithAlert('用户不存在。');
}
2022-11-11 00:20:33 +00:00
if (!UOJGroup::cur()->hasUser($user)) {
2022-11-10 00:16:48 +00:00
dieWithAlert('该用户不在小组中。');
}
2022-10-22 11:24:48 +00:00
2022-11-11 00:20:33 +00:00
DB::delete([
"delete from groups_users",
"where", [
"username" => $user['username'],
"group_id" => UOJGroup::info('id'),
],
]);
2022-11-10 00:16:48 +00:00
dieWithAlert('移除成功!');
}
2023-01-15 00:48:08 +00:00
$add_new_user_form = new UOJForm('add_new_user');
$add_new_user_form->addInput('new_username', [
'label' => '用户名',
'validator_php' => function ($username, &$vdata) {
2022-11-11 00:20:33 +00:00
$user = UOJUser::query($username);
2022-09-26 03:26:17 +00:00
2022-11-11 00:20:33 +00:00
if (!$user) {
return '用户不存在。';
2022-09-26 03:26:17 +00:00
}
2022-11-11 00:20:33 +00:00
if (UOJGroup::cur()->hasUser($user)) {
2022-11-10 00:16:48 +00:00
return '该用户已经在小组中';
2022-09-26 03:26:17 +00:00
}
2022-11-11 00:20:33 +00:00
$vdata['username'] = $user['username'];
2022-09-26 03:26:17 +00:00
2022-11-10 00:16:48 +00:00
return '';
},
2023-01-15 00:48:08 +00:00
]);
$add_new_user_form->config['submit_button']['class'] = 'btn btn-secondary';
$add_new_user_form->config['submit_button']['text'] = '添加';
2022-11-11 00:20:33 +00:00
$add_new_user_form->handle = function (&$vdata) {
DB::insert([
"insert into groups_users",
DB::bracketed_fields(["group_id", "username"]),
"values",
DB::tuple([
UOJGroup::info('id'),
$vdata['username']
]),
]);
2022-09-26 03:26:17 +00:00
2022-11-10 00:16:48 +00:00
dieWithJsonData(['status' => 'success', 'message' => '已将用户名为 ' . $vdata['username'] . ' 的用户添加到本小组。']);
};
$add_new_user_form->setAjaxSubmit(<<<EOD
2023-01-15 00:48:08 +00:00
function(res) {
if (res.status === 'success') {
$('#result-alert')
.html('用户添加成功!' + (res.message || ''))
.addClass('alert-success')
.removeClass('alert-danger')
.show();
} else {
$('#result-alert')
.html('用户添加失败。' + (res.message || ''))
.removeClass('alert-success')
.addClass('alert-danger')
.show();
}
2022-09-26 03:26:17 +00:00
2023-01-15 00:48:08 +00:00
$(window).scrollTop(0);
}
EOD);
2022-11-10 00:16:48 +00:00
$add_new_user_form->runAtServer();
}
?>
2022-11-11 00:20:33 +00:00
<?php echoUOJPageHeader('管理 - ' . UOJGroup::info('title')); ?>
2022-09-26 03:26:17 +00:00
2022-11-06 02:26:21 +00:00
<h1 class="d-block d-md-inline-block">
2022-11-11 00:20:33 +00:00
<?= UOJGroup::info('title') ?>
<small class="fs-5">(ID: #<?= UOJGroup::info('id') ?>)</small>
2022-10-21 12:42:48 +00:00
管理
2022-09-26 03:26:17 +00:00
</h1>
2022-10-21 12:42:48 +00:00
<div class="row mt-4">
2022-11-10 00:16:48 +00:00
<!-- left col -->
<div class="col-md-3">
<?= HTML::navListGroup($tabs_info, $cur_tab) ?>
2022-09-26 04:09:54 +00:00
2022-11-11 00:20:33 +00:00
<?=
UOJGroup::cur()->getLink([
'class' => 'btn btn-light d-block mt-2 w-100 text-start text-primary uoj-back-btn',
'text' => '<i class="bi bi-arrow-left"></i> 返回',
]);
?>
2022-10-21 12:42:48 +00:00
</div>
2022-11-10 00:16:48 +00:00
<!-- end left col -->
<!-- right col -->
<div class="col-md-9">
<?php if ($cur_tab == 'profile') : ?>
<div class="card mt-3 mt-md-0">
<div class="card-body">
<div id="result-alert" class="alert" role="alert" style="display: none"></div>
<div class="row row-cols-1 row-cols-md-2">
<div class="col">
<?= $update_profile_form->printHTML() ?>
</div>
<div class="col">
<h5>注意事项</h5>
<ul class="mb-0">
<li>隐藏的小组无法被普通用户查看,即使该用户属于本小组。</li>
<li>公告支持 Markdown 语法,但不支持添加数学公式。</li>
</ul>
</div>
</div>
</div>
</div>
<?php elseif ($cur_tab == 'assignments') : ?>
<div class="card mt-3 mt-md-0">
<div class="card-header">
<ul class="nav nav-tabs card-header-tabs" role="tablist">
<li class="nav-item">
<a class="nav-link active" href="#assignments" data-bs-toggle="tab" data-bs-target="#assignments">作业列表</a>
</li>
<li class="nav-item">
<a class="nav-link" href="#add-assignment" data-bs-toggle="tab" data-bs-target="#add-assignment">添加作业</a>
</li>
</ul>
</div>
<div class="card-body">
<div class="tab-content">
<div class="tab-pane active" id="assignments">
<?php
echoLongTable(
['*'],
'groups_assignments',
2022-11-11 00:20:33 +00:00
["group_id" => UOJGroup::info('id')],
2022-11-10 00:16:48 +00:00
'order by end_time desc, list_id desc',
<<<EOD
2022-11-11 00:20:33 +00:00
<tr>
<th style="width:4em" class="text-center">题单 ID</th>
<th style="width:12em">标题</th>
<th style="width:4em">状态</th>
<th style="width:8em">结束时间</th>
<th style="width:8em">操作</th>
</tr>
EOD,
function ($row) use ($hidden_time) {
$assignment = UOJGroupAssignment::query($row['list_id']);
echo HTML::tag_begin('tr');
echo HTML::tag('td', ['class' => 'text-center'], $assignment->info['id']);
echo HTML::tag_begin('td');
echo $assignment->getLink();
if ($assignment->info['is_hidden']) {
2022-11-10 00:16:48 +00:00
echo ' <span class="badge text-bg-danger"><i class="bi bi-eye-slash-fill"></i> ', UOJLocale::get('hidden'), '</span> ';
}
2022-11-11 00:20:33 +00:00
echo HTML::tag_end('td');
if ($assignment->info['end_time'] < $hidden_time) {
echo HTML::tag('td', ['class' => 'text-secondary'], '已隐藏');
} elseif ($assignment->info['end_time'] < UOJTime::$time_now) {
echo HTML::tag('td', ['class' => 'text-danger'], '已结束');
2022-11-10 00:16:48 +00:00
} else {
2022-11-11 00:20:33 +00:00
echo HTML::tag('td', ['class' => 'text-success'], '进行中');
2022-11-10 00:16:48 +00:00
}
2022-11-11 00:20:33 +00:00
echo HTML::tag('td', [], $assignment->info['end_time_str']);
2022-11-10 00:16:48 +00:00
echo '<td>';
2022-11-11 00:20:33 +00:00
echo ' <a class="text-decoration-none d-inline-block align-middle" href="/list/', $assignment->info['id'], '/manage">编辑</a> ';
echo ' <form class="d-inline-block" method="POST" onsubmit=\'return confirm("你真的要移除这份作业(题单 #', $assignment->info['id'], ')吗?移除作业不会删除题单。")\'>'
. HTML::hiddenToken()
. '<input type="hidden" name="list_id" value="' . $assignment->info['id'] . '">'
2022-11-10 00:16:48 +00:00
. '<button class="btn btn-link text-danger text-decoration-none p-0" type="submit" name="submit-remove_assignment" value="remove_assignment">移除</button>'
. '</form>';
echo '</td>';
echo '</tr>';
},
[
2022-11-12 07:39:17 +00:00
'page_len' => 10,
2022-11-10 00:16:48 +00:00
'div_classes' => ['table-responsive'],
'table_classes' => ['table', 'align-middle'],
]
);
?>
</div>
<div class="tab-pane" id="add-assignment">
<div id="result-alert" class="alert" role="alert" style="display: none"></div>
<div class="row row-cols-1 row-cols-md-2">
<div class="col">
<?php $add_new_assignment_form->printHTML() ?>
</div>
<div class="col">
<h5>注意事项</h5>
<ul class="mt-0">
<li>要被添加为作业的题单必须是公开的。</li>
<li>请为学生预留合理的完成作业的时间。</li>
<li>排行榜将在结束后停止更新。</li>
<li>如需延长结束时间请删除后再次添加,排行数据不会丢失。</li>
<li>作业结束七天后将会自动在小组主页中隐藏,但仍可直接通过 URL 访问。</li>
</ul>
</div>
</div>
</div>
2022-10-21 12:42:48 +00:00
</div>
</div>
</div>
2022-11-10 00:16:48 +00:00
<?php elseif ($cur_tab == 'users') : ?>
<div class="card mt-3 mt-md-0">
<div class="card-header">
<ul class="nav nav-tabs card-header-tabs" role="tablist">
<li class="nav-item">
<a class="nav-link active" href="#users" data-bs-toggle="tab" data-bs-target="#users">用户列表</a>
</li>
<li class="nav-item">
<a class="nav-link" href="#add-user" data-bs-toggle="tab" data-bs-target="#add-user">添加用户</a>
</li>
</ul>
</div>
<div class="card-body">
<div class="tab-content">
<div class="tab-pane active" id="users">
<?php
echoLongTable(
['*'],
'groups_users',
2022-11-11 00:20:33 +00:00
["group_id" => UOJGroup::info('id')],
2022-11-10 00:16:48 +00:00
'order by username asc',
<<<EOD
2022-11-11 00:20:33 +00:00
<tr>
<th>用户名</th>
<th>操作</th>
</tr>
EOD,
function ($row) {
echo HTML::tag_begin('tr');
echo HTML::tag('td', [], UOJUser::getLink($row['username']));
2022-11-10 00:16:48 +00:00
echo '<td>';
echo '<form class="d-inline-block" method="POST" onsubmit=\'return confirm("你真的要从小组中移除这个用户吗?")\'>'
2022-11-12 07:39:17 +00:00
. HTML::hiddenToken()
2022-11-10 00:16:48 +00:00
. '<input type="hidden" name="remove_username" value="' . $row['username'] . '">'
. '<button class="btn btn-link text-danger text-decoration-none p-0" type="submit" name="submit-remove_user" value="remove_user">移除</button>'
. '</form>';
echo '</td>';
2022-11-11 00:20:33 +00:00
echo HTML::tag_end('tr');
2022-11-10 00:16:48 +00:00
},
[
2022-11-12 07:39:17 +00:00
'page_len' => 10,
2022-11-10 00:16:48 +00:00
'div_classes' => ['table-responsive'],
'table_classes' => ['table', 'align-middle'],
]
);
?>
</div>
<div class="tab-pane" id="add-user">
<div id="result-alert" class="alert" role="alert" style="display: none"></div>
<div class="row row-cols-1 row-cols-md-2">
<div class="col">
<?php $add_new_user_form->printHTML() ?>
</div>
<div class="col">
<h5>注意事项</h5>
<ul class="mb-0">
<li>添加用户前请确认用户名是否正确以免带来不必要的麻烦。</li>
<li>用户被添加到小组后将自动被加入组内的所有作业排行中。</li>
</ul>
</div>
</div>
</div>
2022-10-21 12:42:48 +00:00
</div>
</div>
</div>
2022-11-10 00:16:48 +00:00
<?php endif ?>
2022-10-21 12:42:48 +00:00
</div>
</div>
2022-09-26 04:09:54 +00:00
2022-09-26 03:26:17 +00:00
<?php echoUOJPageFooter() ?>