// 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>') ) ); }); });