mirror of
https://github.com/renbaoshuo/S2OJ.git
synced 2025-01-24 18:21:53 +00:00
1756 lines
57 KiB
JavaScript
1756 lines
57 KiB
JavaScript
// locale
|
||
uojLocaleData = {
|
||
"username": {
|
||
"en": "Username",
|
||
"zh-cn": "用户名"
|
||
},
|
||
"contests::total score": {
|
||
"en": "Score",
|
||
"zh-cn": "总分"
|
||
},
|
||
"contests::n participants": {
|
||
"en": function(n) {
|
||
return n + " participant" + (n <= 1 ? '' : 's');
|
||
},
|
||
"zh-cn": function(n) {
|
||
return "共 " + n + " 名参赛者";
|
||
}
|
||
},
|
||
"click-zan::good": {
|
||
"en": "Good",
|
||
"zh-cn": "好评"
|
||
},
|
||
"click-zan::bad": {
|
||
"en": "Bad",
|
||
"zh-cn": "差评"
|
||
},
|
||
"editor::language": {
|
||
"en": "Language",
|
||
"zh-cn": "语言"
|
||
},
|
||
"editor::browse": {
|
||
"en": "Browse",
|
||
"zh-cn": "浏览"
|
||
},
|
||
"editor::upload source": {
|
||
"en": "Source",
|
||
"zh-cn": "来源"
|
||
},
|
||
"editor::upload by editor": {
|
||
"en": "Editor",
|
||
"zh-cn": "编辑器"
|
||
},
|
||
"editor::upload from local": {
|
||
"en": "Local file",
|
||
"zh-cn": "本地文件"
|
||
}
|
||
};
|
||
|
||
function uojLocale(name) {
|
||
locale = $.cookie('uoj_locale');
|
||
if (uojLocaleData[name] === undefined) {
|
||
return '';
|
||
}
|
||
if (uojLocaleData[name][locale] === undefined) {
|
||
locale = 'zh-cn';
|
||
}
|
||
val = uojLocaleData[name][locale];
|
||
if (!$.isFunction(val)) {
|
||
return val;
|
||
} else {
|
||
var args = [];
|
||
for (var i = 1; i < arguments.length; i++) {
|
||
args.push(arguments[i]);
|
||
}
|
||
return val.apply(this, args);
|
||
}
|
||
}
|
||
|
||
// utility
|
||
function strToDate(str) {
|
||
var a = str.split(/[^0-9]/);
|
||
return new Date(
|
||
parseInt(a[0]),
|
||
parseInt(a[1]) - 1,
|
||
parseInt(a[2]),
|
||
parseInt(a[3]),
|
||
parseInt(a[4]),
|
||
parseInt(a[5]),
|
||
0);
|
||
}
|
||
function dateToStr(date) {
|
||
return date.getFullYear() + '-' + (date.getMonth() + 1) + '-' + date.getDate() + ' ' + date.getHours() + ':' + date.getMinutes() + ':' + date.getSeconds();
|
||
}
|
||
function toFilledStr(o, f, l) {
|
||
var s = o.toString();
|
||
while (s.length < l) {
|
||
s = f.toString() + s;
|
||
}
|
||
return s;
|
||
}
|
||
function getPenaltyTimeStr(x) {
|
||
var ss = toFilledStr(x % 60, '0', 2);
|
||
x = Math.floor(x / 60);
|
||
var mm = toFilledStr(x % 60, '0', 2);
|
||
x = Math.floor(x / 60);
|
||
var hh = x.toString();
|
||
return hh + ':' + mm + ':' + ss;
|
||
}
|
||
|
||
function htmlspecialchars(str)
|
||
{
|
||
var s = "";
|
||
if (str.length == 0) return "";
|
||
s = str.replace(/&/g, "&");
|
||
s = s.replace(/</g, "<");
|
||
s = s.replace(/>/g, ">");
|
||
s = s.replace(/"/g, """);
|
||
return s;
|
||
}
|
||
|
||
function getColOfScore(score) {
|
||
if (score == 0) {
|
||
return ColorConverter.toStr(ColorConverter.toRGB(new HSV(0, 100, 80)));
|
||
} else if (score == 100) {
|
||
return ColorConverter.toStr(ColorConverter.toRGB(new HSV(120, 100, 80)));
|
||
} else {
|
||
return ColorConverter.toStr(ColorConverter.toRGB(new HSV(30 + score * 60 / 100, 100, 90)));
|
||
}
|
||
}
|
||
|
||
function getUserLink(username, realname, color) {
|
||
if (!username) {
|
||
return '';
|
||
}
|
||
var text = username;
|
||
var style = '';
|
||
if (username.charAt(0) == '@') {
|
||
username = username.substr(1);
|
||
}
|
||
if (realname) {
|
||
text = text + ' <span class="uoj-realname d-inline-block">(' + realname + ')</span>';
|
||
}
|
||
if (color) {
|
||
style += 'color: ' + color + ';';
|
||
}
|
||
return '<a class="uoj-username" href="' + uojHome + '/user/' + username + '" ' + 'style="' + style + '">' + text + '</a>';
|
||
}
|
||
function getUserSpan(username, realname, color) {
|
||
if (!username) {
|
||
return '';
|
||
}
|
||
var text = username;
|
||
var style = '';
|
||
if (username.charAt(0) == '@') {
|
||
username = username.substr(1);
|
||
}
|
||
if (realname) {
|
||
text = text + ' <span class="uoj-realname d-inline-block">(' + realname + ')</span>';
|
||
}
|
||
if (color) {
|
||
style += 'color: ' + color + ';';
|
||
}
|
||
return '<span class="uoj-username" ' + 'style="' + style + '">' + text + '</span>';
|
||
}
|
||
|
||
function replaceWithHighlightUsername() {
|
||
var username = $(this).text();
|
||
var realname = $(this).data("realname");
|
||
var color = $(this).data("color");
|
||
|
||
if ($(this).data("link") != 0) {
|
||
$(this).replaceWith(getUserLink(username, realname, color));
|
||
} else {
|
||
$(this).replaceWith(getUserSpan(username, realname, color));
|
||
}
|
||
}
|
||
|
||
$.fn.uoj_honor = function() {
|
||
return this.each(function() {
|
||
var honor = $(this).text();
|
||
var realname = $(this).data("realname");
|
||
if (realname) {
|
||
honor = honor + ' (' + realname + ')';
|
||
}
|
||
$(this).css('color', '#007bff').html(honor);
|
||
});
|
||
}
|
||
|
||
function showErrorHelp(name, err) {
|
||
if (err) {
|
||
$('#div-' + name).addClass('has-validation has-error');
|
||
$('#div-' + name).addClass('is-invalid');
|
||
$('#input-' + name).addClass('is-invalid');
|
||
$('#help-' + name).text(err);
|
||
return false;
|
||
} else {
|
||
$('#div-' + name).removeClass('has-validation has-error');
|
||
$('#div-' + name).removeClass('is-invalid');
|
||
$('#input-' + name).removeClass('is-invalid');
|
||
$('#help-' + name).text('');
|
||
return true;
|
||
}
|
||
}
|
||
function getFormErrorAndShowHelp(name, val) {
|
||
var err = val($('#input-' + name).val());
|
||
return showErrorHelp(name, err);
|
||
}
|
||
|
||
function validateSettingPassword(str) {
|
||
if (str.length < 6) {
|
||
return '密码长度不应小于6。';
|
||
} else if (! /^[!-~]+$/.test(str)) {
|
||
return '密码应只包含可见ASCII字符。';
|
||
} else if (str != $('#input-confirm_password').val()) {
|
||
return '两次输入的密码不一致。';
|
||
} else {
|
||
return '';
|
||
}
|
||
}
|
||
function validatePassword(str) {
|
||
if (str.length < 6) {
|
||
return '密码长度不应小于6。';
|
||
} else if (! /^[!-~]+$/.test(str)) {
|
||
return '密码应只包含可见ASCII字符。';
|
||
} else {
|
||
return '';
|
||
}
|
||
}
|
||
function validateEmail(str) {
|
||
if (str.length > 50) {
|
||
return '电子邮箱地址太长。';
|
||
} else if (! /^(.+)@(.+)$/.test(str)) {
|
||
return '电子邮箱地址非法。';
|
||
} else {
|
||
return '';
|
||
}
|
||
}
|
||
function validateUsername(str) {
|
||
if (str.length == 0) {
|
||
return '用户名不能为空。';
|
||
} else if (! /^[a-zA-Z0-9_]+$/.test(str)) {
|
||
return '用户名应只包含大小写英文字母、数字和下划线。';
|
||
} else {
|
||
return '';
|
||
}
|
||
}
|
||
function validateQQ(str) {
|
||
if (str.length < 5) {
|
||
return 'QQ的长度不应小于5。';
|
||
} else if (str.length > 15) {
|
||
return 'QQ的长度不应大于15。';
|
||
} else if (/\D/.test(str)) {
|
||
return 'QQ应只包含0~9的数字。';
|
||
} else {
|
||
return '';
|
||
}
|
||
}
|
||
function validateMotto(str) {
|
||
if (str.length > 1024) {
|
||
return '不能超过 1024 个字符。';
|
||
} else {
|
||
return '';
|
||
}
|
||
}
|
||
|
||
// tags
|
||
$.fn.uoj_problem_tag = function() {
|
||
return this.each(function() {
|
||
$(this).attr('href', uojHome + '/problems?tag=' + encodeURIComponent($(this).text()));
|
||
});
|
||
}
|
||
$.fn.uoj_list_tag = function() {
|
||
return this.each(function() {
|
||
$(this).attr('href', uojHome + '/lists?tag=' + encodeURIComponent($(this).text()));
|
||
});
|
||
}
|
||
$.fn.uoj_blog_tag = function() {
|
||
return this.each(function() {
|
||
$(this).attr('href', uojBlogUrl + '/archive?tag=' + encodeURIComponent($(this).text()));
|
||
});
|
||
}
|
||
|
||
// click zan
|
||
function click_zan(zan_id, zan_type, zan_delta, node) {
|
||
var loading_node = $('<div class="uoj-click-zan-block text-muted">loading...</div>');
|
||
$(node).replaceWith(loading_node);
|
||
$.post(zan_link + '/click-zan', {
|
||
id : zan_id,
|
||
delta : zan_delta,
|
||
type : zan_type,
|
||
}, function(ret) {
|
||
$(loading_node).replaceWith($(ret).click_zan_block());
|
||
}).fail(function() {
|
||
$(loading_node).replaceWith('<div class="uoj-click-zan-block text-danger">failed</div>');
|
||
});
|
||
}
|
||
|
||
$.fn.click_zan_block = function() {
|
||
return this.each(function() {
|
||
var id = $(this).data('id');
|
||
var type = $(this).data('type');
|
||
var val = parseInt($(this).data('val'));
|
||
var cnt = parseInt($(this).data('cnt'));
|
||
if (isNaN(cnt)) {
|
||
return;
|
||
}
|
||
|
||
var up_icon_html = '<i class="bi bi-hand-thumbs-up"></i>';
|
||
var down_icon_html = '<i class="bi bi-hand-thumbs-down"></i>';
|
||
|
||
if (val == 1) {
|
||
$(this).addClass('uoj-click-zan-block-cur-up');
|
||
up_icon_html = '<i class="bi bi-hand-thumbs-up-fill"></i>';
|
||
} else if (val == 0) {
|
||
$(this).addClass('uoj-click-zan-block-cur-zero');
|
||
} else if (val == -1) {
|
||
$(this).addClass('uoj-click-zan-block-cur-down');
|
||
down_icon_html = '<i class="bi bi-hand-thumbs-down-fill"></i>';
|
||
} else {
|
||
return;
|
||
}
|
||
if (cnt > 0) {
|
||
$(this).addClass('uoj-click-zan-block-positive');
|
||
} else if (cnt == 0) {
|
||
$(this).addClass('uoj-click-zan-block-neutral');
|
||
} else {
|
||
$(this).addClass('uoj-click-zan-block-negative');
|
||
}
|
||
|
||
var node = this;
|
||
var up_node = $('<a href="#" class="uoj-click-zan-up">'+up_icon_html+uojLocale('click-zan::good')+'</a>').click(function(e) {
|
||
e.preventDefault();
|
||
click_zan(id, type, 1, node);
|
||
});
|
||
var down_node = $('<a href="#" class="uoj-click-zan-down">'+down_icon_html+uojLocale('click-zan::bad')+'</a>').click(function(e) {
|
||
e.preventDefault();
|
||
click_zan(id, type, -1, node);
|
||
});
|
||
|
||
$(this)
|
||
.append(up_node)
|
||
.append(down_node)
|
||
.append($('<span class="uoj-click-zan-cnt">[<strong>' + (cnt > 0 ? '+' + cnt : cnt) + '</strong>]</span>'));
|
||
});
|
||
}
|
||
|
||
// count down
|
||
function getCountdownStr(t, font_size, color = true) {
|
||
var x = Math.floor(t);
|
||
var ss = toFilledStr(x % 60, '0', 2);
|
||
x = Math.floor(x / 60);
|
||
var mm = toFilledStr(x % 60, '0', 2);
|
||
x = Math.floor(x / 60);
|
||
var hh = x.toString();
|
||
|
||
var res = '<span style="font-size:' + font_size + '">';
|
||
res += '<span '
|
||
if (color) res += ' style="color:' + getColOfScore(Math.min(t / 10800 * 100, 100)) + '" ';
|
||
res += ' >' + hh + '</span>';
|
||
res += ':';
|
||
res += '<span '
|
||
if (color) res += ' style="color:' + getColOfScore(mm / 60 * 100) + '" ';
|
||
res += ' >' + mm + '</span>';
|
||
res += ':';
|
||
res += '<span ';
|
||
if (color) res += ' style="color:' + getColOfScore(ss / 60 * 100) + '" ';
|
||
res +=' >' + ss + '</span>';
|
||
res += '</span>'
|
||
return res;
|
||
}
|
||
|
||
$.fn.countdown = function(rest, callback, font_size = '30px', color = true) {
|
||
return this.each(function() {
|
||
var start = new Date().getTime();
|
||
var cur_rest = rest != undefined ? rest : parseInt($(this).data('rest'));
|
||
var cur = this;
|
||
var countdown = function() {
|
||
var passed = Math.floor((new Date().getTime() - start) / 1000);
|
||
if (passed >= cur_rest) {
|
||
$(cur).html(getCountdownStr(0, font_size, color));
|
||
if (callback != undefined) {
|
||
callback();
|
||
}
|
||
} else {
|
||
$(cur).html(getCountdownStr(cur_rest - passed, font_size, color));
|
||
setTimeout(countdown, 1000);
|
||
}
|
||
}
|
||
countdown();
|
||
});
|
||
};
|
||
|
||
// update_judgement_status
|
||
update_judgement_status_list = []
|
||
function update_judgement_status_details(id) {
|
||
update_judgement_status_list.push(id);
|
||
};
|
||
|
||
$(document).ready(function() {
|
||
function update() {
|
||
$.get("/submission-status-details", {
|
||
get: update_judgement_status_list
|
||
},
|
||
function(data) {
|
||
for (var i = 0; i < update_judgement_status_list.length; i++) {
|
||
$("#status_details_" + update_judgement_status_list[i]).html(data[i].html);
|
||
if (data[i].judged) {
|
||
location.reload();
|
||
}
|
||
}
|
||
}, 'json').always(
|
||
function() {
|
||
setTimeout(update, 500);
|
||
}
|
||
);
|
||
}
|
||
if (update_judgement_status_list.length > 0) {
|
||
setTimeout(update, 500);
|
||
}
|
||
});
|
||
|
||
// highlight
|
||
$.fn.uoj_highlight = function() {
|
||
return $(this).each(function() {
|
||
$(this).find("span.uoj-username, span[data-uoj-username]").each(replaceWithHighlightUsername);
|
||
$(this).find(".uoj-honor").uoj_honor();
|
||
$(this).find(".uoj-score").each(function() {
|
||
var score = parseFloat($(this).text());
|
||
var maxscore = parseFloat($(this).data('max'));
|
||
if (isNaN(score)) {
|
||
return;
|
||
}
|
||
if (isNaN(maxscore)) {
|
||
$(this).css("color", getColOfScore(score));
|
||
} else {
|
||
$(this).css("color", getColOfScore(score / maxscore * 100));
|
||
}
|
||
});
|
||
$(this).find(".uoj-status").each(function() {
|
||
var success = parseInt($(this).data("success"));
|
||
if(isNaN(success)){
|
||
return;
|
||
}
|
||
if (success == 1) {
|
||
$(this).css("color", ColorConverter.toStr(ColorConverter.toRGB(new HSV(120, 100, 80))));
|
||
}
|
||
else {
|
||
$(this).css("color", ColorConverter.toStr(ColorConverter.toRGB(new HSV(0, 100, 100))));
|
||
}
|
||
});
|
||
$(this).find(".uoj-problem-tag").uoj_problem_tag();
|
||
$(this).find(".uoj-list-tag").uoj_list_tag();
|
||
$(this).find(".uoj-blog-tag").uoj_blog_tag();
|
||
$(this).find(".uoj-click-zan-block").click_zan_block();
|
||
$(this).find(".countdown").countdown();
|
||
$(this).find(".uoj-readmore").readmore({
|
||
moreLink: '<a href="#" class="text-right">more...</a>',
|
||
lessLink: '<a href="#" class="text-right">close</a>',
|
||
});
|
||
});
|
||
};
|
||
|
||
$(document).ready(function() {
|
||
$('body').uoj_highlight();
|
||
});
|
||
|
||
// contest notice
|
||
function checkNotice(lastTime) {
|
||
$.post(uojHome + '/check-notice', {
|
||
last_time : lastTime
|
||
},
|
||
function(data) {
|
||
if (data === null) {
|
||
return;
|
||
}
|
||
setTimeout(function() {
|
||
checkNotice(data.time);
|
||
}, 60000);
|
||
for (var i = 0; i < data.msg.length; i++) {
|
||
alert(data.msg[i]);
|
||
}
|
||
},
|
||
'json'
|
||
).fail(function() {
|
||
setTimeout(function() {
|
||
checkNotice(lastTime);
|
||
}, 60000);
|
||
});
|
||
}
|
||
|
||
// long table
|
||
$.fn.long_table = function(data, cur_page, header_row, get_row_str, config) {
|
||
return this.each(function() {
|
||
var table_div = this;
|
||
|
||
$(table_div).html('');
|
||
|
||
var page_len = config.page_len != undefined ? config.page_len : 10;
|
||
|
||
if (!config.echo_full) {
|
||
var n_rows = data.length;
|
||
var n_pages = Math.max(Math.ceil(n_rows / page_len), 1);
|
||
if (cur_page == undefined) {
|
||
cur_page = 1;
|
||
}
|
||
if (cur_page < 1) {
|
||
cur_page = 1;
|
||
} else if (cur_page > n_pages) {
|
||
cur_page = n_pages;
|
||
}
|
||
var cur_start = (cur_page - 1) * page_len;
|
||
} else {
|
||
var n_rows = data.length;
|
||
var n_pages = 1;
|
||
cur_page = 1;
|
||
var cur_start = (cur_page - 1) * page_len;
|
||
}
|
||
|
||
var div_classes = config.div_classes != undefined ? config.div_classes : ['table-responsive'];
|
||
var table_classes = config.table_classes != undefined ? config.table_classes : ['table', 'table-bordered', 'table-hover', 'table-striped', 'table-text-center'];
|
||
|
||
var now_cnt = 0;
|
||
var tbody = $('<tbody />')
|
||
for (var i = 0; i < page_len && cur_start + i < n_rows; i++) {
|
||
now_cnt++;
|
||
if (config.get_row_index) {
|
||
tbody.append(get_row_str(data[cur_start + i], cur_start + i));
|
||
} else {
|
||
tbody.append(get_row_str(data[cur_start + i]));
|
||
}
|
||
}
|
||
if (now_cnt == 0) {
|
||
tbody.append('<tr><td colspan="233">无</td></tr>');
|
||
}
|
||
|
||
$(table_div).append(
|
||
$('<div class="' + div_classes.join(' ') + '" />').append(
|
||
(typeof config.print_before_table === 'function' ? config.print_before_table() : ''),
|
||
$('<table class="' + table_classes.join(' ') + '" />').append(
|
||
$('<thead>' + header_row + '</thead>')
|
||
).append(
|
||
tbody
|
||
),
|
||
(typeof config.print_after_table === 'function' ? config.print_after_table() : '')
|
||
)
|
||
);
|
||
|
||
var get_page_li = function(p, h) {
|
||
if (p == -1) {
|
||
return $('<li class="page-item"></li>').addClass('disabled').append($('<a class="page-link"></a>').append(h));
|
||
}
|
||
|
||
var li = $('<li class="page-item"></li>');
|
||
if (p == cur_page) {
|
||
li.addClass('active');
|
||
}
|
||
li.append(
|
||
$('<a class="page-link"></a>').attr('href', '#' + table_div.id).append(h).click(function(e) {
|
||
if (config.prevent_focus_on_click) {
|
||
e.preventDefault();
|
||
}
|
||
$(table_div).long_table(data, p, header_row, get_row_str, config);
|
||
})
|
||
);
|
||
return li;
|
||
};
|
||
|
||
if (n_pages > 1) {
|
||
var pagination = $('<ul class="pagination top-buffer-no bot-buffer-sm justify-content-center"></ul>');
|
||
if (cur_page > 1) {
|
||
pagination.append(get_page_li(1, '<i class="bi bi-chevron-double-left"></i>'));
|
||
pagination.append(get_page_li(cur_page - 1, '<i class="bi bi-chevron-left"></i>'));
|
||
} else {
|
||
pagination.append(get_page_li(-1, '<i class="bi bi-chevron-double-left"></i>'));
|
||
pagination.append(get_page_li(-1, '<i class="bi bi-chevron-left"></i>'));
|
||
}
|
||
var max_extend = config.max_extend != undefined ? config.max_extend : 5;
|
||
for (var i = Math.max(cur_page - max_extend, 1); i <= Math.min(cur_page + max_extend, n_pages); i++) {
|
||
pagination.append(get_page_li(i, i.toString()));
|
||
}
|
||
if (cur_page < n_pages) {
|
||
pagination.append(get_page_li(cur_page + 1, '<i class="bi bi-chevron-right"></i>'));
|
||
pagination.append(get_page_li(n_pages, '<i class="bi bi-chevron-double-right"></i>'));
|
||
} else {
|
||
pagination.append(get_page_li(-1, '<i class="bi bi-chevron-right"></i>'));
|
||
pagination.append(get_page_li(-1, '<i class="bi bi-chevron-double-right"></i>'));
|
||
}
|
||
$(table_div).append($('<div class="text-center"></div>').append(pagination));
|
||
}
|
||
});
|
||
};
|
||
|
||
// monaco editor
|
||
function require_monaco(config, callback) {
|
||
window.require = {
|
||
paths: {
|
||
vs: '/js/monaco-editor/min/vs',
|
||
},
|
||
"vs/nls": {
|
||
availableLanguages: {
|
||
'*': 'zh-cn',
|
||
},
|
||
},
|
||
};
|
||
$LAB.script('/js/monaco-editor/min/vs/loader.js').wait()
|
||
.script('/js/monaco-editor/min/vs/editor/editor.main.nls.js').wait()
|
||
.script('/js/monaco-editor/min/vs/editor/editor.main.nls.zh-cn.js').wait()
|
||
.script('/js/monaco-editor/min/vs/editor/editor.main.js').wait(function() {
|
||
$LAB.script('/js/monaco-themes.js').wait(callback);
|
||
});
|
||
}
|
||
|
||
function get_monaco_mode(lang) {
|
||
switch (lang) {
|
||
case 'C++':
|
||
case 'C++11':
|
||
case 'C++17':
|
||
case 'C++20':
|
||
case 'C++98':
|
||
case 'C++03':
|
||
return 'cpp';
|
||
case 'C':
|
||
return 'c';
|
||
case 'Python2':
|
||
case 'Python2.7':
|
||
case 'Python3':
|
||
return 'python';
|
||
case 'Pascal':
|
||
return 'pascal';
|
||
case 'Java8':
|
||
case 'Java11':
|
||
case 'Java17':
|
||
return 'java';
|
||
case 'text':
|
||
return 'text';
|
||
default:
|
||
return 'text';
|
||
}
|
||
}
|
||
|
||
// auto save
|
||
function autosave_locally(interval, name, target) {
|
||
if (typeof(Storage) === "undefined") {
|
||
console.log('autosave_locally: Sorry! No Web Storage support..');
|
||
return;
|
||
}
|
||
var url = window.location.href;
|
||
var hp = url.indexOf('#');
|
||
var uri = hp == -1 ? url : url.substr(0, hp);
|
||
var full_name = name + '@' + uri;
|
||
|
||
target.val(localStorage.getItem(full_name));
|
||
var save = function() {
|
||
localStorage.setItem(full_name, target.val());
|
||
setTimeout(save, interval);
|
||
};
|
||
setTimeout(save, interval);
|
||
}
|
||
|
||
function autosave_locally_monaco(interval, name, monaco_instance) {
|
||
if (typeof(Storage) === "undefined") {
|
||
console.log('autosave_locally_monaco: Sorry! No Web Storage support..');
|
||
return;
|
||
}
|
||
var url = window.location.href;
|
||
var hp = url.indexOf('#');
|
||
var uri = hp == -1 ? url : url.substring(0, hp);
|
||
var full_name = name + '@' + uri;
|
||
|
||
monaco_instance.getModel().setValue(localStorage.getItem(full_name));
|
||
var save = function() {
|
||
localStorage.setItem(full_name, monaco_instance.getModel().getValue());
|
||
setTimeout(save, interval);
|
||
};
|
||
setTimeout(save, interval);
|
||
}
|
||
|
||
// source code form group
|
||
$.fn.source_code_form_group = function(name, text, langs_options_html) {
|
||
return this.each(function() {
|
||
var input_upload_type_editor_id = 'input-' + name + '_upload_type_editor';
|
||
var input_upload_type_file_id = 'input-' + name + '_upload_type_file';
|
||
var input_upload_type_name = name + '_upload_type';
|
||
var input_language_id = 'input-' + name + '_language';
|
||
var input_language_name = name + '_language';
|
||
var input_editor_id = 'input-' + name + '_editor';
|
||
var input_editor_name = name + '_editor';
|
||
var input_file_id = 'input-' + name + '_file';
|
||
var input_file_name = name + '_file';
|
||
var spinner_id = 'spinner-' + name + '_editor';
|
||
var div_help_language_id = 'div-help-' + name + '_language';
|
||
var div_editor_id = 'div-' + name + '_editor';
|
||
var div_file_id = 'div-' + name + '_file';
|
||
var help_file_id = 'help-' + name + '_file';
|
||
|
||
var input_language =
|
||
$('<select id="' + input_language_id + '" name="' + input_language_name + '" class="form-select form-select-sm d-inline-block"/>')
|
||
.html(langs_options_html);
|
||
var input_upload_type_editor = $('<input class="form-check-input" type="radio" id="' + input_upload_type_editor_id + '" name="' + input_upload_type_name + '" value="editor" />');
|
||
var input_upload_type_file = $('<input class="form-check-input" type="radio" id="' + input_upload_type_file_id + '" name="' + input_upload_type_name + '" value="file" />');
|
||
var input_file = $('<input type="file" class="form-control" id="' + input_file_id + '" name="' + input_file_name + '" />');
|
||
var div_editor =
|
||
$('<div id="' + div_editor_id + '" style="height: 350px" />')
|
||
.append(
|
||
$('<div id="' + spinner_id + '" class="border d-flex justify-content-center align-items-center" style="width: 100%; height: 350px;" />')
|
||
.append('<div class="spinner-border text-muted" style="width: 3rem; height: 3rem;" />')
|
||
);
|
||
var div_file =
|
||
$('<div id="' + div_file_id + '" />')
|
||
.append(input_file)
|
||
.append($('<span class="help-block" id="' + help_file_id + '"></span>'))
|
||
|
||
var div_help_language = $('<div id="' + div_help_language_id + '" class="text-warning mb-2">');
|
||
|
||
var show_help_lang = function() {
|
||
if ($(this).val().startsWith('Java')) {
|
||
div_help_language.text('注意:Java 程序源代码中不应指定所在的 package。我们会在源代码中找到第一个被定义的类并以它的 main 函数为程序入口点。');
|
||
} else {
|
||
div_help_language.text('');
|
||
}
|
||
};
|
||
|
||
var monaco_editor_instance = null;
|
||
var monaco_editor_init = function() {
|
||
require_monaco({}, function() {
|
||
if (monaco_editor_instance != null) {
|
||
return;
|
||
}
|
||
|
||
$(div_editor).html('');
|
||
|
||
var mode = get_monaco_mode(input_language.val());
|
||
|
||
monaco_editor_instance = monaco.editor.create(div_editor[0], {
|
||
language: mode,
|
||
automaticLayout: true,
|
||
fontSize: "14px",
|
||
});
|
||
|
||
$('#' + spinner_id).css('display', 'none !important');
|
||
$(div_editor).addClass('border overflow-hidden').show();
|
||
autosave_locally_monaco(2000, name, monaco_editor_instance);
|
||
|
||
$('#' + input_editor_id).val(monaco_editor_instance.getModel().getValue());
|
||
monaco_editor_instance.onDidChangeModelContent(function () {
|
||
$('#' + input_editor_id).val(monaco_editor_instance.getModel().getValue());
|
||
});
|
||
|
||
input_language.change(function() {
|
||
monaco.editor.setModelLanguage(monaco_editor_instance.getModel(), get_monaco_mode(input_language.val()));
|
||
});
|
||
});
|
||
}
|
||
|
||
var save_prefer_upload_type = function(type) {
|
||
$.cookie('uoj_source_code_form_group_preferred_upload_type', type, { expires: 7, path: '/' });
|
||
};
|
||
|
||
var prefer_upload_type = $.cookie('uoj_source_code_form_group_preferred_upload_type');
|
||
if (prefer_upload_type === null) {
|
||
prefer_upload_type = 'editor';
|
||
}
|
||
if (prefer_upload_type == 'file') {
|
||
input_upload_type_file[0].checked = true;
|
||
div_editor.css('display', 'none');
|
||
} else {
|
||
input_upload_type_editor[0].checked = true;
|
||
div_file.css('display', 'none');
|
||
}
|
||
|
||
input_language.each(show_help_lang);
|
||
input_language.change(show_help_lang);
|
||
|
||
input_upload_type_editor.click(function() {
|
||
div_editor.show('fast');
|
||
div_file.hide('fast');
|
||
save_prefer_upload_type('editor');
|
||
});
|
||
input_upload_type_file.click(function() {
|
||
div_file.show('fast');
|
||
div_editor.hide('fast');
|
||
save_prefer_upload_type('file');
|
||
});
|
||
|
||
$(this).append(
|
||
$('<div class="row mb-2 align-items-center"/>')
|
||
.append($('<div class="col-sm-4 text-start">' + text + '</div>'))
|
||
.append(
|
||
$('<div class="col-sm-4 row align-items-center"/>')
|
||
.append($('<div class="col-auto" />').append('<label class="col-form-label px-1' +' " for="' + input_language_id + '">' + uojLocale('editor::language') + '</label>'))
|
||
.append($('<div class="col-auto" />').append(input_language))
|
||
)
|
||
.append($('<div class="col-sm-4 text-end"/>')
|
||
.append(uojLocale('editor::upload source') + ': ')
|
||
.append($('<div class="form-check d-inline-block">')
|
||
.append($('<label for="' + input_upload_type_editor_id + '" />')
|
||
.append(input_upload_type_editor)
|
||
.append(' ' + uojLocale('editor::upload by editor'))
|
||
)
|
||
)
|
||
.append($('<div class="form-check d-inline-block ms-3"/>')
|
||
.append(input_upload_type_file)
|
||
.append($('<label for="' + input_upload_type_file_id +'" />')
|
||
.append(' ' + uojLocale('editor::upload from local'))
|
||
)
|
||
))
|
||
)
|
||
.append(div_help_language)
|
||
.append(div_editor)
|
||
.append(div_file)
|
||
.append($('<input type="hidden" name="' + input_editor_name + '" id="' + input_editor_id + '"/>'));
|
||
|
||
var check_monaco_editor_init = function() {
|
||
if (div_editor.is(':visible')) {
|
||
monaco_editor_init();
|
||
} else {
|
||
setTimeout(check_monaco_editor_init, 1);
|
||
}
|
||
}
|
||
check_monaco_editor_init();
|
||
});
|
||
}
|
||
|
||
// text file form group
|
||
$.fn.text_file_form_group = function(name, text) {
|
||
return this.each(function() {
|
||
var input_upload_type_editor_id = 'input-' + name + '_upload_type_editor';
|
||
var input_upload_type_file_id = 'input-' + name + '_upload_type_file';
|
||
var input_upload_type_name = name + '_upload_type';
|
||
var input_editor_id = 'input-' + name + '_editor';
|
||
var input_editor_name = name + '_editor';
|
||
var input_file_id = 'input-' + name + '_file';
|
||
var input_file_name = name + '_file';
|
||
var spinner_id = 'spinner-' + name + '_editor';
|
||
var div_editor_id = 'div-' + name + '_editor';
|
||
var div_file_id = 'div-' + name + '_file';
|
||
|
||
var help_file_id = 'help-' + name + '_file';
|
||
|
||
var input_upload_type_editor = $('<input class="form-check-input" type="radio" id="' + input_upload_type_editor_id + '" name="' + input_upload_type_name + '" value="editor" />');
|
||
var input_upload_type_file = $('<input class="form-check-input" type="radio" id="' + input_upload_type_file_id + '" name="' + input_upload_type_name + '" value="file" />');
|
||
var input_file = $('<input type="file" class="form-control" id="' + input_file_id + '" name="' + input_file_name + '" />');
|
||
|
||
var div_editor = $('<div id="' + div_editor_id + '" style="height: 350px" />')
|
||
.append(
|
||
$('<div id="' + spinner_id + '" class="border d-flex justify-content-center align-items-center" style="width: 100%; height: 350px;" />')
|
||
.append('<div class="spinner-border text-muted" style="width: 3rem; height: 3rem;" />')
|
||
);
|
||
var div_file =
|
||
$('<div id="' + div_file_id + '" />')
|
||
.append(input_file)
|
||
.append($('<span class="help-block" id="' + help_file_id + '"></span>'))
|
||
|
||
var monaco_editor_instance = null;
|
||
var monaco_editor_init = function() {
|
||
require_monaco({}, function() {
|
||
if (monaco_editor_instance != null) {
|
||
return;
|
||
}
|
||
|
||
$(div_editor).html('');
|
||
|
||
monaco_editor_instance = monaco.editor.create(div_editor[0], {
|
||
language: 'text',
|
||
automaticLayout: true,
|
||
fontSize: "14px",
|
||
});
|
||
|
||
$('#' + spinner_id).css('display', 'none !important');
|
||
$(div_editor).addClass('border overflow-hidden').show();
|
||
autosave_locally_monaco(2000, name, monaco_editor_instance);
|
||
|
||
$('#' + input_editor_id).val(monaco_editor_instance.getModel().getValue());
|
||
monaco_editor_instance.onDidChangeModelContent(function () {
|
||
$('#' + input_editor_id).val(monaco_editor_instance.getModel().getValue());
|
||
});
|
||
});
|
||
}
|
||
|
||
var save_prefer_upload_type = function(type) {
|
||
$.cookie('uoj_text_file_form_group_preferred_upload_type', type, { expires: 7, path: '/' });
|
||
};
|
||
|
||
var prefer_upload_type = $.cookie('uoj_text_file_form_group_preferred_upload_type');
|
||
if (prefer_upload_type === null) {
|
||
prefer_upload_type = 'editor';
|
||
}
|
||
if (prefer_upload_type == 'file') {
|
||
input_upload_type_file[0].checked = true;
|
||
div_editor.css('display', 'none');
|
||
} else {
|
||
input_upload_type_editor[0].checked = true;
|
||
div_file.css('display', 'none');
|
||
}
|
||
|
||
input_upload_type_editor.click(function() {
|
||
div_editor.show('fast');
|
||
div_file.hide('fast');
|
||
save_prefer_upload_type('editor');
|
||
});
|
||
input_upload_type_file.click(function() {
|
||
div_file.show('fast');
|
||
div_editor.hide('fast');
|
||
save_prefer_upload_type('file');
|
||
});
|
||
|
||
$(this)
|
||
.append($('<div class="row justify-content-between mb-2" />')
|
||
.append($('<div class="col text-start">' + text + '</div>'))
|
||
.append($('<div class="col text-end"/>')
|
||
.append(uojLocale('editor::upload source') + ': ')
|
||
.append($('<div class="form-check d-inline-block">')
|
||
.append($('<label for="' + input_upload_type_editor_id + '" />')
|
||
.append(input_upload_type_editor)
|
||
.append(' ' + uojLocale('editor::upload by editor'))
|
||
)
|
||
)
|
||
.append($('<div class="form-check d-inline-block ms-3"/>')
|
||
.append(input_upload_type_file)
|
||
.append($('<label for="' + input_upload_type_file_id +'" />')
|
||
.append(' ' + uojLocale('editor::upload from local'))
|
||
)
|
||
))
|
||
)
|
||
.append(div_editor)
|
||
.append(div_file)
|
||
.append($('<input type="hidden" name="' + input_editor_name + '" id="' + input_editor_id + '"/>'));
|
||
|
||
var check_monaco_editor_init = function() {
|
||
if (div_editor.is(':visible')) {
|
||
monaco_editor_init();
|
||
} else {
|
||
setTimeout(check_monaco_editor_init, 1);
|
||
}
|
||
}
|
||
check_monaco_editor_init();
|
||
});
|
||
}
|
||
|
||
// remote judge submit type group
|
||
$.fn.remote_submit_type_group = function(oj, pid, url, submit_type) {
|
||
return this.each(function() {
|
||
var input_submit_type_bot_id = 'input-submit_type_bot';
|
||
var input_submit_type_my_id = 'input-submit_type_my';
|
||
var input_submit_type_archive_id = 'input-submit_type_archive';
|
||
var div_submit_type_bot_id = 'div-submit_type_bot';
|
||
var div_submit_type_my_id = 'div-submit_type_my';
|
||
var div_submit_type_archive_id = 'div-submit_type_archive';
|
||
|
||
var input_submit_type_bot = $('<input class="form-check-input" type="radio" name="answer_remote_submit_type" id="' + input_submit_type_bot_id + '" value="bot" />');
|
||
var input_submit_type_my = $('<input class="form-check-input" type="radio" name="answer_remote_submit_type" id="' + input_submit_type_my_id + '" value="my" />');
|
||
var input_submit_type_archive = $('<input class="form-check-input" type="radio" name="answer_remote_submit_type" id="' + input_submit_type_archive_id + '" value="archive" />');
|
||
var input_my_account_data = $('<input type="hidden" name="answer_remote_account_data" value="" />');
|
||
|
||
var my_account_validation_status = $('<span />').append('<span class="text-secondary">待验证</span>');
|
||
var my_account_validation_btn = $('<button type="button" class="btn btn-secondary btn-sm ms-2">验证</button>');
|
||
|
||
var validate_my_account = function(data) {
|
||
my_account_validation_btn.html('<span class="spinner-border spinner-border-sm" role="status"></span>');
|
||
my_account_validation_btn.addClass('disabled');
|
||
|
||
$.ajax({
|
||
type: 'POST',
|
||
url: '/api/remote_judge/custom_account_validator?type=' + oj,
|
||
data: data,
|
||
success: function(res) {
|
||
my_account_validation_btn.html('验证');
|
||
my_account_validation_btn.removeClass('disabled');
|
||
|
||
console.log('Validation status', res);
|
||
|
||
if (res.ok) {
|
||
my_account_validation_status.html('<span class="text-success">可用</span>');
|
||
} else {
|
||
my_account_validation_status.html('<span class="text-danger">不可用</span>');
|
||
}
|
||
},
|
||
error: function() {
|
||
my_account_validation_btn.html('验证');
|
||
my_account_validation_btn.removeClass('disabled');
|
||
my_account_validation_status.html('<span class="text-secondary">待验证</span>');
|
||
},
|
||
dataType: 'json',
|
||
});
|
||
};
|
||
|
||
var div_submit_type_bot = $('<div id="' + div_submit_type_bot_id + '" />')
|
||
.append('<div class="mt-3">将使用公用账号提交本题。</div>');
|
||
var div_submit_type_my = $('<div id="' + div_submit_type_my_id + '" />')
|
||
.append($('<div class="mt-3" />')
|
||
.append('<span>将使用您的账号提交本题。</span>')
|
||
.append('<span>配置方法请查阅 <a href="https://sjzezoj.com/blog/baoshuo/post/717" target="_blank">使用教程</a>。</span>')
|
||
);
|
||
var div_submit_type_archive = $('<div id="' + div_submit_type_archive_id + '" />')
|
||
.append($('<div class="mt-3" />')
|
||
.append('<span>将从您给定的提交记录中抓取评测结果。</span>')
|
||
.append('<span>配置方法请查阅 <a href="https://sjzezoj.com/blog/baoshuo/post/717" target="_blank">使用教程</a>。</span>')
|
||
).append(
|
||
$('<div class="row mt-3 align-items-center" />')
|
||
.append($('<div class="col-sm-2" />').append('<label for="input-answer_remote_submission_id" class="col-form-label">提交记录 ID</label>'))
|
||
.append($('<div class="col-sm-4" />').append('<input id="input-answer_remote_submission_id" name="answer_remote_submission_id" class="form-control font-monospace" autocomplete="off" />'))
|
||
.append($('<div class="col-sm-6" />').append($('<div class="form-text mt-0" />').append('请填入远程 OJ 上的提交记录 ID。')))
|
||
);
|
||
var div_account_data = $('<div class="border px-3 py-2 mt-3" />')
|
||
.append($('<div class="mt-2" />').append('<span class="fs-6 fw-bold">远程账号信息</span>'))
|
||
.append($('<div class="mt-3" />')
|
||
.append('<span>账号状态:</span>')
|
||
.append(my_account_validation_status)
|
||
.append(my_account_validation_btn)
|
||
);
|
||
|
||
if ('localStorage' in window) {
|
||
var prefer_submit_type = localStorage.getItem('uoj_remote_judge_save_prefer_submit_type__' + oj) || null;
|
||
var save_prefer_submit_type = function(type) {
|
||
localStorage.setItem('uoj_remote_judge_save_prefer_submit_type__' + oj, type);
|
||
}
|
||
} else {
|
||
var prefer_submit_type = null;
|
||
var save_prefer_submit_type = function(type) {};
|
||
}
|
||
|
||
input_submit_type_bot.click(function() {
|
||
div_account_data.hide('fast');
|
||
div_submit_type_my.hide('fast');
|
||
div_submit_type_archive.hide('fast');
|
||
div_submit_type_bot.show('fast');
|
||
$('#form-group-answer_answer').show('fast');
|
||
save_prefer_submit_type('bot');
|
||
});
|
||
input_submit_type_my.click(function() {
|
||
div_submit_type_bot.hide('fast');
|
||
div_submit_type_archive.hide('fast');
|
||
div_submit_type_my.show('fast');
|
||
div_account_data.show('fast');
|
||
$('#form-group-answer_answer').show('fast');
|
||
save_prefer_submit_type('my');
|
||
});
|
||
input_submit_type_archive.click(function() {
|
||
div_submit_type_bot.hide('fast');
|
||
div_submit_type_my.hide('fast');
|
||
div_submit_type_archive.show('fast');
|
||
div_account_data.show('fast');
|
||
$('#form-group-answer_answer').hide('fast');
|
||
save_prefer_submit_type('archive');
|
||
});
|
||
|
||
if (submit_type[0] == 'bot') {
|
||
div_account_data.hide();
|
||
div_submit_type_my.hide();
|
||
div_submit_type_archive.hide();
|
||
div_submit_type_bot.show();
|
||
$('#form-group-answer_answer').show();
|
||
input_submit_type_bot[0].checked = true;
|
||
} else if (submit_type[0] == 'my') {
|
||
div_submit_type_bot.hide();
|
||
div_submit_type_my.show();
|
||
div_submit_type_archive.hide();
|
||
div_account_data.show();
|
||
$('#form-group-answer_answer').show();
|
||
input_submit_type_my[0].checked = true;
|
||
} else if (submit_type[0] == 'archive') {
|
||
div_submit_type_bot.hide();
|
||
div_submit_type_my.hide();
|
||
div_submit_type_archive.show();
|
||
div_account_data.show();
|
||
$('#form-group-answer_answer').hide();
|
||
input_submit_type_archive[0].checked = true;
|
||
}
|
||
|
||
if (submit_type.indexOf('bot') == -1) {
|
||
input_submit_type_bot.attr('disabled', 'disabled');
|
||
} else if (prefer_submit_type == 'bot') {
|
||
div_account_data.hide();
|
||
div_submit_type_my.hide();
|
||
div_submit_type_archive.hide();
|
||
div_submit_type_bot.show();
|
||
$('#form-group-answer_answer').show();
|
||
input_submit_type_bot[0].checked = true;
|
||
input_submit_type_my[0].checked = false;
|
||
input_submit_type_archive[0].checked = false;
|
||
}
|
||
|
||
if (submit_type.indexOf('my') == -1) {
|
||
input_submit_type_my.attr('disabled', 'disabled');
|
||
} else if (prefer_submit_type == 'my') {
|
||
div_submit_type_bot.hide();
|
||
div_submit_type_my.show();
|
||
div_submit_type_archive.hide();
|
||
div_account_data.show();
|
||
$('#form-group-answer_answer').show();
|
||
input_submit_type_bot[0].checked = false;
|
||
input_submit_type_my[0].checked = true;
|
||
input_submit_type_archive[0].checked = false;
|
||
}
|
||
|
||
if (submit_type.indexOf('archive') == -1) {
|
||
input_submit_type_archive.attr('disabled', 'disabled');
|
||
} else if (prefer_submit_type == 'archive') {
|
||
div_submit_type_bot.hide();
|
||
div_submit_type_my.hide();
|
||
div_submit_type_archive.show();
|
||
div_account_data.show();
|
||
$('#form-group-answer_answer').hide();
|
||
input_submit_type_bot[0].checked = false;
|
||
input_submit_type_my[0].checked = false;
|
||
input_submit_type_archive[0].checked = true;
|
||
}
|
||
|
||
if (oj == 'luogu') {
|
||
var luogu_account_data = {"_uid": "", "__client_id": ""};
|
||
var input_luogu_uid = $('<input class="form-control font-monospace" type="text" name="luogu_uid" id="input-luogu_uid" autocomplete="off" />');
|
||
var input_luogu_client_id = $('<input class="form-control font-monospace" type="text" name="luogu_client_id" id="input-luogu_client_id" autocomplete="off" />');
|
||
|
||
if ('localStorage' in window) {
|
||
try {
|
||
var luogu_account_data_str = localStorage.getItem('uoj_remote_judge_luogu_account_data');
|
||
if (luogu_account_data_str) {
|
||
luogu_account_data = JSON.parse(luogu_account_data_str);
|
||
}
|
||
} catch (e) {}
|
||
|
||
var save_luogu_account_data = function() {
|
||
localStorage.setItem('uoj_remote_judge_luogu_account_data', JSON.stringify(luogu_account_data));
|
||
}
|
||
} else {
|
||
var save_luogu_account_data = function() {};
|
||
}
|
||
|
||
input_luogu_uid.change(function() {
|
||
luogu_account_data._uid = $(this).val();
|
||
input_my_account_data.val(JSON.stringify(luogu_account_data));
|
||
save_luogu_account_data();
|
||
my_account_validation_status.html('<span class="text-secondary">待验证</span>');
|
||
});
|
||
|
||
input_luogu_client_id.change(function() {
|
||
luogu_account_data.__client_id = $(this).val();
|
||
input_my_account_data.val(JSON.stringify(luogu_account_data));
|
||
save_luogu_account_data();
|
||
my_account_validation_status.html('<span class="text-secondary">待验证</span>');
|
||
});
|
||
|
||
my_account_validation_btn.click(function() {
|
||
validate_my_account({
|
||
type: 'luogu',
|
||
_uid: input_luogu_uid.val(),
|
||
__client_id: input_luogu_client_id.val(),
|
||
});
|
||
});
|
||
|
||
input_my_account_data.val(JSON.stringify(luogu_account_data));
|
||
input_luogu_uid.val(luogu_account_data._uid);
|
||
input_luogu_client_id.val(luogu_account_data.__client_id);
|
||
|
||
if (luogu_account_data._uid && luogu_account_data.__client_id){
|
||
validate_my_account({
|
||
type: 'luogu',
|
||
_uid: luogu_account_data._uid,
|
||
__client_id: luogu_account_data.__client_id,
|
||
});
|
||
}
|
||
|
||
div_account_data.append(
|
||
$('<div class="row mt-3 align-items-center" />')
|
||
.append($('<div class="col-sm-2" />').append('<label for="input-luogu_uid" class="col-form-label">_uid</label>'))
|
||
.append($('<div class="col-sm-4" />').append(input_luogu_uid))
|
||
.append($('<div class="col-sm-6" />').append($('<div class="form-text mt-0" />').append('请填入 Cookie 中的 <code>_uid</code>。')))
|
||
).append(
|
||
$('<div class="row mt-3 align-items-center" />')
|
||
.append($('<div class="col-sm-2" />').append('<label for="input-luogu_client_id" class="col-form-label">__client_id</label>'))
|
||
.append($('<div class="col-sm-4" />').append(input_luogu_client_id))
|
||
.append($('<div class="col-sm-6" />').append($('<div class="form-text mt-0" />').append('请填入 Cookie 中的 <code>__client_id</code>。')))
|
||
).append(input_my_account_data);
|
||
} else if (oj == 'codeforces') {
|
||
var codeforces_account_data = {"JSESSIONID": ""};
|
||
var input_codeforces_jsessionid = $('<input class="form-control font-monospace" type="text" name="codeforces_jsessionid" id="input-codeforces_jsessionid" autocomplete="off" />');
|
||
|
||
if ('localStorage' in window) {
|
||
try {
|
||
var codeforces_account_data_str = localStorage.getItem('uoj_remote_judge_codeforces_account_data');
|
||
if (codeforces_account_data_str) {
|
||
codeforces_account_data = JSON.parse(codeforces_account_data_str);
|
||
}
|
||
} catch (e) {}
|
||
|
||
var save_codeforces_account_data = function() {
|
||
localStorage.setItem('uoj_remote_judge_codeforces_account_data', JSON.stringify(codeforces_account_data));
|
||
}
|
||
}
|
||
|
||
input_codeforces_jsessionid.change(function() {
|
||
codeforces_account_data.JSESSIONID = $(this).val();
|
||
input_my_account_data.val(JSON.stringify(codeforces_account_data));
|
||
save_codeforces_account_data();
|
||
my_account_validation_status.html('<span class="text-secondary">待验证</span>');
|
||
});
|
||
|
||
my_account_validation_btn.click(function() {
|
||
validate_my_account({
|
||
type: 'codeforces',
|
||
JSESSIONID: input_codeforces_jsessionid.val(),
|
||
});
|
||
});
|
||
|
||
input_my_account_data.val(JSON.stringify(codeforces_account_data));
|
||
input_codeforces_jsessionid.val(codeforces_account_data.JSESSIONID);
|
||
|
||
if (codeforces_account_data.JSESSIONID) {
|
||
validate_my_account({
|
||
type: 'codeforces',
|
||
JSESSIONID: codeforces_account_data.JSESSIONID,
|
||
});
|
||
}
|
||
|
||
div_account_data.append(
|
||
$('<div class="row mt-3 align-items-center" />')
|
||
.append($('<div class="col-sm-2" />').append('<label for="input-codeforces_jsessionid" class="col-form-label">JSESSIONID</label>'))
|
||
.append($('<div class="col-sm-4" />').append(input_codeforces_jsessionid))
|
||
.append($('<div class="col-sm-6" />').append($('<div class="form-text mt-0" />').append('请填入 Cookie 中的 <code>JSESSIONID</code>。')))
|
||
).append(input_my_account_data);
|
||
} else if (oj == 'loj') {
|
||
var loj_account_data = {username: "", token: ""};
|
||
var input_loj_token = $('<input class="form-control font-monospace" type="text" name="loj_token" id="input-loj_token" autocomplete="off" />');
|
||
|
||
if ('localStorage' in window) {
|
||
try {
|
||
var loj_account_data_str = localStorage.getItem('uoj_remote_judge_loj_account_data');
|
||
if (loj_account_data_str) {
|
||
loj_account_data = JSON.parse(loj_account_data_str);
|
||
}
|
||
} catch (e) {}
|
||
|
||
var save_loj_account_data = function() {
|
||
localStorage.setItem('uoj_remote_judge_loj_account_data', JSON.stringify(loj_account_data));
|
||
}
|
||
} else {
|
||
var save_loj_account_data = function() {};
|
||
}
|
||
|
||
input_loj_token.change(function() {
|
||
loj_account_data.token = $(this).val();
|
||
input_my_account_data.val(JSON.stringify(loj_account_data));
|
||
save_loj_account_data();
|
||
my_account_validation_status.html('<span class="text-secondary">待验证</span>');
|
||
});
|
||
|
||
my_account_validation_btn.click(function() {
|
||
validate_my_account({
|
||
type: 'loj',
|
||
token: input_loj_token.val(),
|
||
});
|
||
});
|
||
|
||
input_my_account_data.val(JSON.stringify(loj_account_data));
|
||
input_loj_token.val(loj_account_data.token);
|
||
|
||
if (loj_account_data.token) {
|
||
validate_my_account({
|
||
type: 'loj',
|
||
token: loj_account_data.token,
|
||
});
|
||
}
|
||
|
||
div_account_data.append(
|
||
$('<div class="row mt-3 align-items-center" />')
|
||
.append($('<div class="col-sm-2" />').append('<label for="input-loj_token" class="col-form-label">Token</label>'))
|
||
.append($('<div class="col-sm-4" />').append(input_loj_token))
|
||
.append($('<div class="col-sm-6" />').append($('<div class="form-text mt-0" />').append('请前往 <a href="https://loj.ac" target="_blank">LibreOJ</a> 登录账号,然后输入在控制台中运行 <code>console.log(JSON.parse(localStorage.appState).token)</code> 的输出结果。')))
|
||
).append(input_my_account_data);
|
||
}
|
||
|
||
$(this).append(
|
||
$('<div class="mt-3" />').append(
|
||
$('<div class="form-check d-inline-block" />')
|
||
.append(input_submit_type_bot)
|
||
.append($('<label class="form-check-label" for="' + input_submit_type_bot_id + '" />').append(' 公用账号'))
|
||
).append(
|
||
$('<div class="form-check d-inline-block ms-3" />')
|
||
.append(input_submit_type_my)
|
||
.append($('<label class="form-check-label" for="' + input_submit_type_my_id + '" />').append(' 自有账号'))
|
||
).append(
|
||
$('<div class="form-check d-inline-block ms-3" />')
|
||
.append(input_submit_type_archive)
|
||
.append($('<label class="form-check-label" for="' + input_submit_type_archive_id + '" />').append(' 归档'))
|
||
)
|
||
)
|
||
.append(div_submit_type_bot)
|
||
.append(div_submit_type_my)
|
||
.append(div_submit_type_archive)
|
||
.append(div_account_data);
|
||
});
|
||
}
|
||
|
||
// problem_configure: print problem.conf
|
||
$.fn.problem_conf_preview = function(problem_conf) {
|
||
return $(this).each(function() {
|
||
var keys = Object.keys(problem_conf);
|
||
var res = '';
|
||
|
||
for (var i = 0; i < keys.length; i++) {
|
||
var key = keys[i];
|
||
var value = problem_conf[key];
|
||
|
||
if (!value) {
|
||
continue;
|
||
}
|
||
|
||
if (key == 'use_builtin_judger' && value == 'off') {
|
||
continue;
|
||
}
|
||
|
||
if (key == 'use_builtin_checker' && value == 'ownchk') {
|
||
continue;
|
||
}
|
||
|
||
res += key + ' ' + value + '\n';
|
||
}
|
||
|
||
$(this).html('<pre class="bg-light mb-0 p-3"><code>' + res + '</code></pre>');
|
||
});
|
||
}
|
||
|
||
// problem_configure: point scores
|
||
$.fn.problem_configure_point_scores = function(problem_conf) {
|
||
return $(this).each(function() {
|
||
var _this = this;
|
||
var n_tests = parseInt(problem_conf['n_tests']);
|
||
|
||
$(this).empty();
|
||
|
||
if (isNaN(n_tests) || n_tests <= 0) {
|
||
$(this).html('不可用。');
|
||
|
||
return;
|
||
}
|
||
|
||
for (var i = 1; i <= n_tests; i++) {
|
||
var input_point_score = $('<input class="form-control form-control-sm uoj-problem-configure-point-score-input" type="number" name="point_score_' + i + '" id="input-point_score_' + i + '" min="0" max="100" />');
|
||
|
||
if (problem_conf['point_score_' + i]) {
|
||
input_point_score.val(problem_conf['point_score_' + i]);
|
||
}
|
||
|
||
(function(i){
|
||
input_point_score.change(function() {
|
||
problem_conf['point_score_' + i] = $(this).val();
|
||
$('#problem-conf-preview').problem_conf_preview(problem_conf);
|
||
});
|
||
})(i);
|
||
|
||
$(this).append(
|
||
$('<div class="col-sm-6" />').append(
|
||
$('<div class="" />')
|
||
.append($('<div class="" />').append('<label for="input-point_score_' + i + '" class="col-form-label col-form-label-sm">测试点 #' + i + '</label>'))
|
||
.append($('<div class="" />').append(input_point_score))
|
||
)
|
||
);
|
||
}
|
||
|
||
$('.uoj-problem-configure-point-score-input', this).change(function() {
|
||
var full_score = 100;
|
||
var rest_tests = parseInt(problem_conf['n_tests'] || '10');
|
||
var score_type = problem_conf['score_type'] || 'int';
|
||
|
||
$('.uoj-problem-configure-point-score-input', _this).each(function() {
|
||
var point_score = parseInt($(this).val());
|
||
if (!isNaN(point_score)) {
|
||
full_score -= point_score;
|
||
rest_tests--;
|
||
}
|
||
});
|
||
|
||
$('.uoj-problem-configure-point-score-input', _this).each(function() {
|
||
if ($(this).val() == '') {
|
||
var val = full_score / rest_tests;
|
||
|
||
if (score_type == 'int') {
|
||
val = Math.floor(val);
|
||
} else {
|
||
var decimal_places = parseInt(score_type.substring(5));
|
||
|
||
val = val.toFixed(decimal_places);
|
||
}
|
||
|
||
$(this).attr('placeholder', val);
|
||
}
|
||
});
|
||
});
|
||
|
||
$('.uoj-problem-configure-point-score-input', this).first().trigger('change');
|
||
});
|
||
};
|
||
|
||
// problem_configure: subtasks
|
||
$.fn.problem_configure_subtasks = function(problem_conf) {
|
||
return $(this).each(function() {
|
||
var _this = this;
|
||
var n_subtasks = parseInt(problem_conf['n_subtasks'] || '0');
|
||
|
||
$(this).empty();
|
||
|
||
if (isNaN(n_subtasks)) {
|
||
$(this).html('不可用。');
|
||
|
||
return;
|
||
}
|
||
|
||
var input_n_subtasks = $('<input class="form-control" type="number" name="n_subtasks" id="input-n_subtasks" />');
|
||
var div_subtasks = $('<div class="list-group list-group-flush border-top" />');
|
||
|
||
if (n_subtasks) {
|
||
input_n_subtasks.val(n_subtasks);
|
||
}
|
||
|
||
$(this).append(
|
||
$('<div class="m-3" />').append(
|
||
$('<div class="row" />').append(
|
||
$('<div class="col-4" />').append('<label for="input-n_subtasks" class="col-form-label">子任务数</label>')
|
||
).append(
|
||
$('<div class="col-8" />').append(input_n_subtasks)
|
||
)
|
||
)
|
||
).append(div_subtasks);
|
||
|
||
input_n_subtasks.change(function() {
|
||
div_subtasks.empty();
|
||
|
||
var n_subtasks = parseInt(input_n_subtasks.val() || '0');
|
||
var n_tests = parseInt(problem_conf['n_tests'] || '10');
|
||
problem_conf['n_subtasks'] = input_n_subtasks.val();
|
||
|
||
for (var i = 1; i <= n_subtasks; i++) {
|
||
var input_subtask_type = $('<select class="form-select form-select-sm" name="subtask_type_' + i + '" id="input-subtask_type_' + i + '" />');
|
||
var input_subtask_end = $('<input class="form-control form-control-sm uoj-problem-configure-subtask-end-input" type="number" name="subtask_end_' + i + '" id="input-subtask_end_' + i + '" min="0" max="' + n_tests + '" />');
|
||
var input_subtask_score = $('<input class="form-control form-control-sm uoj-problem-configure-subtask-score-input" type="number" name="subtask_score_' + i + '" id="input-subtask_score_' + i + '" min="0" max="100" />');
|
||
var input_subtask_used_time_type = $('<select class="form-select form-select-sm" name="subtask_used_time_type_' + i + '" id="input-subtask_used_time_type_' + i + '" />');
|
||
|
||
input_subtask_type
|
||
.append($('<option value="packed" />').text('错一个就零分'))
|
||
.append($('<option value="min" />').text('取所有测试点中的最小值'));
|
||
input_subtask_used_time_type
|
||
.append($('<option value="sum" />').text('全部相加'))
|
||
.append($('<option value="max" />').text('取所有测试点中的最大值'));
|
||
|
||
(function(i) {
|
||
input_subtask_type.change(function() {
|
||
problem_conf['subtask_type_' + i] = $(this).val();
|
||
$('#problem-conf-preview').problem_conf_preview(problem_conf);
|
||
});
|
||
|
||
input_subtask_end.change(function() {
|
||
problem_conf['subtask_end_' + i] = $(this).val();
|
||
$('#problem-conf-preview').problem_conf_preview(problem_conf);
|
||
})
|
||
|
||
input_subtask_score.change(function() {
|
||
problem_conf['subtask_score_' + i] = $(this).val();
|
||
$('#problem-conf-preview').problem_conf_preview(problem_conf);
|
||
});
|
||
|
||
input_subtask_used_time_type.change(function() {
|
||
problem_conf['subtask_used_time_type_' + i] = $(this).val();
|
||
$('#problem-conf-preview').problem_conf_preview(problem_conf);
|
||
});
|
||
})(i);
|
||
|
||
if (problem_conf['subtask_type_' + i]) {
|
||
input_subtask_type.val(problem_conf['subtask_type_' + i]);
|
||
}
|
||
|
||
if (problem_conf['subtask_end_' + i]) {
|
||
input_subtask_end.val(problem_conf['subtask_end_' + i]);
|
||
}
|
||
|
||
if (problem_conf['subtask_score_' + i]) {
|
||
input_subtask_score.val(problem_conf['subtask_score_' + i]);
|
||
}
|
||
|
||
if (problem_conf['subtask_used_time_type_' + i]) {
|
||
input_subtask_used_time_type.val(problem_conf['subtask_used_time_type_' + i]);
|
||
}
|
||
|
||
div_subtasks.append(
|
||
$('<div class="list-group-item" />').append(
|
||
$('<div class="fw-bold" />').text('Subtask #' + i)
|
||
).append(
|
||
$('<div />').append(
|
||
$('<div class="row mt-2" />').append(
|
||
$('<div class="col-sm-6" />').append('<label for="input-subtask_type_' + i + '" class="col-form-label col-form-label-sm">评分类型</label>')
|
||
).append(
|
||
$('<div class="col-sm-6" />').append(input_subtask_type)
|
||
)
|
||
).append(
|
||
$('<div class="row mt-2" />').append(
|
||
$('<div class="col-sm-6" />').append('<label for="input-subtask_end_' + i + '" class="col-form-label col-form-label-sm">最后一个测试点的编号</label>')
|
||
).append(
|
||
$('<div class="col-sm-6" />').append(input_subtask_end)
|
||
)
|
||
).append(
|
||
$('<div class="row mt-2" />').append(
|
||
$('<div class="col-sm-6" />').append('<label for="input-subtask_score_' + i + '" class="col-form-label col-form-label-sm">分数</label>')
|
||
).append(
|
||
$('<div class="col-sm-6" />').append(input_subtask_score)
|
||
)
|
||
).append(
|
||
$('<div class="row mt-2" />').append(
|
||
$('<div class="col-sm-6" />').append('<label for="input-subtask_used_time_type_' + i + '" class="col-form-label col-form-label-sm">程序用时统计方式</label>')
|
||
).append(
|
||
$('<div class="col-sm-6" />').append(input_subtask_used_time_type)
|
||
)
|
||
)
|
||
)
|
||
);
|
||
}
|
||
|
||
$('.uoj-problem-configure-subtask-score-input', _this).change(function() {
|
||
var full_score = 100;
|
||
var rest_subtasks = parseInt(problem_conf['n_subtasks'] || '10');
|
||
var score_type = problem_conf['score_type'] || 'int';
|
||
|
||
$('.uoj-problem-configure-subtask-score-input', _this).each(function() {
|
||
var subtask_score = parseInt($(this).val());
|
||
|
||
if (!isNaN(subtask_score)) {
|
||
full_score -= subtask_score;
|
||
rest_subtasks--;
|
||
}
|
||
});
|
||
|
||
$('.uoj-problem-configure-subtask-score-input', _this).each(function() {
|
||
if ($(this).val() == '') {
|
||
var val = full_score / rest_subtasks;
|
||
|
||
if (score_type == 'int') {
|
||
val = Math.floor(val);
|
||
} else {
|
||
var decimal_places = parseInt(score_type.substring(5));
|
||
|
||
val = val.toFixed(decimal_places);
|
||
}
|
||
|
||
$(this).attr('placeholder', val);
|
||
}
|
||
});
|
||
});
|
||
|
||
$('.uoj-problem-configure-subtask-score-input', _this).first().trigger('change');
|
||
$('#problem-conf-preview').problem_conf_preview(problem_conf);
|
||
});
|
||
|
||
input_n_subtasks.trigger('change');
|
||
});
|
||
};
|
||
|
||
// custom test
|
||
function custom_test_onsubmit(response_text, div_result, url) {
|
||
if (response_text != '') {
|
||
$(div_result).html('<div class="text-danger">' + response_text + '</div>');
|
||
return;
|
||
}
|
||
var update = function() {
|
||
var can_next = true;
|
||
$.get(url,
|
||
function(data) {
|
||
if (data.judged === undefined) {
|
||
$(div_result).html('<div class="text-danger">error</div>');
|
||
} else {
|
||
var judge_status = $('<table class="table table-bordered table-text-center"><tr class="info">' + data.html + '</tr></table>');
|
||
$(div_result).empty();
|
||
$(div_result).append(judge_status);
|
||
if (data.judged) {
|
||
var judge_result = $(data.result);
|
||
judge_result.css('display', 'none');
|
||
$(div_result).append(judge_result);
|
||
judge_status.hide(500);
|
||
judge_result.slideDown(500);
|
||
can_next = false;
|
||
}
|
||
}
|
||
}, 'json')
|
||
.always(function() {
|
||
if (can_next) {
|
||
setTimeout(update, 500);
|
||
}
|
||
});
|
||
};
|
||
setTimeout(update, 500);
|
||
}
|
||
|
||
// comment
|
||
function showCommentReplies(id, replies) {
|
||
var toggleFormReply = function(from, text) {
|
||
if (text == undefined) {
|
||
text = '';
|
||
}
|
||
|
||
var p = '#comment-body-' + id;
|
||
var q = '#div-form-reply';
|
||
var r = '#input-reply_comment';
|
||
var t = '#input-reply_id';
|
||
if ($(q).data('from') != from) {
|
||
$(q).data('from', from);
|
||
$(q).hide('fast', function() {
|
||
$(this).appendTo(p).show('fast', function() {
|
||
$(t).val(id);
|
||
$(r).val(text).focus();
|
||
});
|
||
});
|
||
|
||
} else if ($(q).css('display') != 'none') {
|
||
$(q).appendTo(p).hide('fast');
|
||
} else {
|
||
$(q).appendTo(p).show('fast', function() {
|
||
$(t).val(id);
|
||
$(r).val(text).focus();
|
||
});
|
||
}
|
||
}
|
||
|
||
$('#reply-to-' + id).click(function(e) {
|
||
e.preventDefault();
|
||
toggleFormReply(id);
|
||
});
|
||
|
||
if (replies.length == 0) {
|
||
return;
|
||
}
|
||
|
||
$("#replies-" + id).long_table(
|
||
replies,
|
||
1,
|
||
'<tr>' +
|
||
'<th>评论回复</th>' +
|
||
'</tr>',
|
||
function(reply) {
|
||
return $('<tr id="' + 'comment-' + reply.id + '" />').append(
|
||
$('<td />').append(
|
||
$('<div class="comment-content">' + getUserLink(reply.poster, reply.poster_realname, reply.poster_username_color) + ':' + reply.content + '</div>')
|
||
).append(
|
||
$('<ul class="text-end mb-0 list-inline" />').append(
|
||
'<li class="list-inline-item small text-muted">' + reply.post_time + '</li>'
|
||
).append(
|
||
$('<li class="list-inline-item" />').append(
|
||
$('<a href="#">回复</a>').click(function (e) {
|
||
e.preventDefault();
|
||
toggleFormReply(reply.id, '回复 @' + reply.poster + ':');
|
||
})
|
||
)
|
||
)
|
||
)
|
||
).uoj_highlight();
|
||
}, {
|
||
table_classes: ['table'],
|
||
page_len: 5,
|
||
prevent_focus_on_click: true
|
||
}
|
||
);
|
||
}
|
||
|
||
// PDF
|
||
$(document).ready(function() {
|
||
$('div[data-pdf]').each(function() {
|
||
$(this).append(
|
||
$('<div class="uoj-pdf-loading-spinner border d-flex flex-column justify-content-center align-items-center" style="width: 100%; height: 350px;" />')
|
||
.append('<div class="spinner-border text-muted" style="width: 3rem; height: 3rem;" />')
|
||
.append('<div class="mt-3">Loading PDF...</div>')
|
||
);
|
||
});
|
||
|
||
if ($('div[data-pdf]').length > 0) {
|
||
$LAB.script('/js/pdf.js').wait(function() {
|
||
pdfjsLib.GlobalWorkerOptions.workerSrc = '/js/pdf.worker.js';
|
||
|
||
// Support HiDPI-screens.
|
||
var outputScale = window.devicePixelRatio || 1;
|
||
|
||
$('div[data-pdf]').each(function() {
|
||
var _this = $(this);
|
||
var pdf_src = $(this).data('src');
|
||
|
||
$(this).css('width', '100%').css('height', '100%');
|
||
|
||
var task = pdfjsLib.getDocument(pdf_src);
|
||
var id = 'pdf_' + task.docId;
|
||
|
||
$(this).attr('id', id + '_container');
|
||
|
||
task.promise.then(function(pdf) {
|
||
for (var i = 1; i <= pdf.numPages; i++) {
|
||
$(_this).append('<canvas id="' + id + '_page_' + i + '" class="pdf-page-canvas"></canvas>');
|
||
|
||
pdf.getPage(i).then(function(page) {
|
||
var viewport = page.getViewport({
|
||
scale: 2.5,
|
||
});
|
||
|
||
var canvas = document.getElementById(id + '_page_' + page.pageNumber);
|
||
|
||
canvas.height = Math.floor(viewport.height * outputScale);
|
||
canvas.width = Math.floor(viewport.width * outputScale);
|
||
|
||
var transform = outputScale !== 1
|
||
? [outputScale, 0, 0, outputScale, 0, 0]
|
||
: null;
|
||
|
||
page.render({
|
||
canvasContext: canvas.getContext('2d'),
|
||
viewport: viewport,
|
||
transform: transform,
|
||
});
|
||
});
|
||
}
|
||
|
||
$('.uoj-pdf-loading-spinner', _this).remove();
|
||
});
|
||
});
|
||
});
|
||
}
|
||
});
|
||
|
||
// Tooltip
|
||
$(document).ready(function() {
|
||
bootstrap.Tooltip.jQueryInterface.call($('[data-bs-toggle="tooltip"]'));
|
||
});
|
||
|
||
// Popovers
|
||
$(document).ready(function() {
|
||
bootstrap.Popover.jQueryInterface.call($('[data-bs-toggle="popover"]'));
|
||
});
|
||
|
||
// Copy button
|
||
$(document).ready(function() {
|
||
$('.markdown-body pre, .copy-button-container pre').each(function () {
|
||
var thisEl = $(this);
|
||
|
||
$(this).wrap(
|
||
$('<div class="wrapped-copy-button-container" style="position: relative;"></div>')
|
||
).parent().prepend(
|
||
$(
|
||
'<div style="position: absolute; right: 0; top: 0; margin-top: 0.75rem; margin-right: 0.75rem; font-size: 0.85em;"></div>'
|
||
).append(
|
||
$('<button style="position: relative; background: transparent; border: 0;"></button>')
|
||
.click(function () {
|
||
navigator.clipboard
|
||
.writeText($(thisEl).text())
|
||
.then(() => {
|
||
$(this).html('<i class="bi bi-check2 text-success"></i>');
|
||
|
||
setTimeout(() => {
|
||
$(this).html('<i class="bi bi-clipboard text-muted"></i>');
|
||
}, 1000);
|
||
})
|
||
.catch(() => {
|
||
$(this).html('<i class="bi bi-x-lg text-danger"></i>');
|
||
|
||
setTimeout(() => {
|
||
$(this).html('<i class="bi bi-clipboard text-muted"></i>');
|
||
}, 1000);
|
||
});
|
||
})
|
||
.append('<i class="bi bi-clipboard text-muted"></i>')
|
||
)
|
||
);
|
||
});
|
||
});
|