Merge branch 'dev' into virusdefender-dev

* dev:
  增加比赛倒计时的功能

Conflicts:
	contest/models.py
This commit is contained in:
virusdefender 2015-09-22 13:10:07 +08:00
commit 0e50f7fdc5
12 changed files with 842 additions and 19 deletions

View File

@ -50,7 +50,7 @@ class Contest(models.Model):
# 没有开始 返回1
return CONTEST_NOT_START
elif self.end_time < now():
# 已经结束 返回0
# 已经结束 返回-1
return CONTEST_ENDED
else:
# 正在进行 返回0

View File

@ -463,15 +463,12 @@ class ContestTimeAPIView(APIView):
获取比赛开始或者结束的倒计时返回毫秒数字
"""
def get(self, request):
t = request.GET.get("type", "start")
contest_id = request.GET.get("contest_id", -1)
try:
contest = Contest.objects.get(id=contest_id)
except Contest.DoesNotExist:
return error_response(u"比赛不存在")
if t == "start":
# 距离开始还有多长时间
return success_response(int((contest.start_time - now()).total_seconds() * 1000))
else:
# 距离结束还有多长时间
return success_response(int((contest.end_time - now()).total_seconds() * 1000))
return success_response({"start": int((contest.start_time - now()).total_seconds() * 1000),
"end": int((contest.end_time - now()).total_seconds() * 1000),
"status": contest.status})

View File

@ -89,7 +89,6 @@ urlpatterns = [
url(r'^contests/$', "contest.views.contest_list_page", name="contest_list_page"),
url(r'^contests/(?P<page>\d+)/$', "contest.views.contest_list_page", name="contest_list_page"),
url(r'^contest/(?P<contest_id>\d+)/$', "contest.views.contest_page", name="contest_page"),
url(r'^problem/(?P<problem_id>\d+)/$', "problem.views.problem_page", name="problem_page"),

View File

@ -0,0 +1,14 @@
#timer {
margin: 20px auto;
}
.timer-section {
display: inline-block;
}
.timer-section span {
background-color: #ffffff;
padding: 5px 4px;
border-radius: 5px;
}
.time {
font-weight: bold;
}

View File

@ -2,6 +2,7 @@
@import url("bootstrap/todc-bootstrap.min.css");
@import url("codeMirror/codemirror.css");
@import url("global.css");
@import url("contest/countdown.css");
#language-selector {
width: 130px;

View File

@ -0,0 +1,40 @@
require(["jquery", "jcountdown"], function($, jcountdown){
function getServerTime(){
var contestId = location.pathname.split("/")[2];
var time = 0;
$.ajax({
url: "/api/contest/time/?contest_id=" + contestId,
dataType: "json",
method: "get",
async: false,
success: function(data){
if(!data.code){
time = data.data;
}
}
});
return time;
}
var time = getServerTime();
if(time["status"] == 1){
countdown(time["start"])
}
else if(time["status"] == 0){
countdown(time["end"])
}
function countdown(t){
$("#timer").countdown({
serverDiff: t,
date: "september 21, 2015 21:59",
yearsAndMonths: false,
template: $('#template').html()
}).on("countComplete", function(){
location.reload();
});
}
});

View File

@ -142,9 +142,6 @@ require(["jquery", "codeMirror", "csrfToken", "bsAlert", "ZeroClipboard"],
if(!data.code){
time = data.data;
}
},
error: function(){
time = new Date().getTime();
}
});
return time;
@ -153,13 +150,16 @@ require(["jquery", "codeMirror", "csrfToken", "bsAlert", "ZeroClipboard"],
if(location.href.indexOf("contest") > -1) {
setInterval(function () {
var time = getServerTime();
var minutes = parseInt(time / (1000 * 60));
if(minutes == 0){
bsAlert("比赛即将结束");
}
else if(minutes > 0 && minutes <= 5){
bsAlert("比赛还剩" + minutes.toString() + "分钟");
if(time["status"] == 0){
var minutes = parseInt(time["end"] / (1000 * 60));
if(minutes == 0){
bsAlert("比赛即将结束");
}
else if(minutes > 0 && minutes <= 5){
bsAlert("比赛还剩" + minutes.toString() + "分钟");
}
}
}, 1000 * 60);
}

View File

@ -3,6 +3,7 @@ var require = {
baseUrl: "/static/js/",
paths: {
jquery: "lib/jquery/jquery",
jcountdown: "lib/jcountdown/jcountdown",
avalon: "lib/avalon/avalon",
editor: "utils/editor",
uploader: "utils/uploader",

View File

@ -0,0 +1,727 @@
/*! jCountdown jQuery Plugin - v2.0.0 - 2015-06-28
* https://github.com/tomgrohl/jCountdown/
* Copyright (c) 2015 Tom Ellis; Licensed MIT */
!function(root, factory) {
if (typeof define === 'function' && define.amd) {
define(['jquery'], factory);
} else {
factory(root.jQuery);
}
}(this, function($) {
$.fn.countdown = function( method /*, options*/ ) {
var msPerHr = 3600000,
secPerYear = 31557600,
secPerMonth = 2629800,
secPerWeek = 604800,
secPerDay = 86400,
secPerHr = 3600,
secPerMin = 60,
secPerSec = 1,
rTemplateTokens = /%y|%m|%w|%d|%h|%i|%s|%ty|%tm|%tw|%td|%th|%ti|%ts/g,
rDigitGlobal = /\d/g,
localNumber = function( numToConvert, settings ) {
var arr = numToConvert.toString().match(rDigitGlobal),
localeNumber = "";
$.each( arr, function(i,num) {
num = Number(num);
localeNumber += (""+ settings.digits[num]) || ""+num;
});
return localeNumber;
},
dateNow = function( $this ) {
var now = new Date(), // Default to local time
settings = $this.data("jcdData");
if ( !settings ) {
return new Date();
}
if ( settings.offset !== null ) {
now = getTimezoneDate( settings.offset );
}
now.setMilliseconds(0);
return now;
},
getTimezoneDate = function( offset ) {
// Returns date now based on timezone/offset
var hrs,
dateMS,
curHrs,
tmpDate = new Date();
if ( offset !== null ) {
hrs = offset * msPerHr;
curHrs = tmpDate.getTime() - ( ( -tmpDate.getTimezoneOffset() / 60 ) * msPerHr ) + hrs;
dateMS = tmpDate.setTime( curHrs );
}
return new Date( dateMS );
},
addEventHandlers = function($element, options) {
// Add event handlers where set
if ( options.onStart ) {
$element.on("start.jcdevt", options.onStart );
}
if ( options.onChange ) {
$element.on("change.jcdevt", options.onChange );
}
if ( options.onComplete ) {
$element.on("complete.jcdevt", options.onComplete );
}
if ( options.onPause ) {
$element.on("pause.jcdevt", options.onPause );
}
if ( options.onResume ) {
$element.on("resume.jcdevt", options.onResume );
}
if ( options.onLocaleChange ) {
$element.on("locale.jcdevt", options.onLocaleChange );
}
},
generateTemplate = function( settings ) {
var template = settings.template,
yearsLeft = settings.yearsLeft,
monthsLeft = settings.monthsLeft,
weeksLeft = settings.weeksLeft,
daysLeft = settings.daysLeft,
hrsLeft = settings.hrsLeft,
minsLeft = settings.minsLeft,
secLeft = settings.secLeft,
hideYears = false,
hideMonths = false,
hideWeeks = false,
hideDays = false,
hideHours = false,
hideMins = false;
if (settings.isRTL) {
template = settings.rtlTemplate;
}
if ( settings.omitZero ) {
if ( settings.yearsAndMonths ) {
if( !settings.yearsLeft ) {
hideYears = true;
}
if( !settings.monthsLeft ) {
hideMonths = true;
}
}
if ( settings.weeks && ( ( settings.yearsAndMonths && hideMonths && !settings.weeksLeft ) || ( !settings.yearsAndMonths && !settings.weeksLeft ) ) ) {
hideWeeks = true;
}
if ( hideWeeks && !daysLeft ) {
hideDays = true;
}
if ( hideDays && !hrsLeft ) {
hideHours = true;
}
if ( hideHours && !minsLeft ) {
hideMins = true;
}
}
if ( settings.leadingZero ) {
if ( yearsLeft < 10 ) {
yearsLeft = "0" + yearsLeft;
}
if ( monthsLeft < 10 ) {
monthsLeft = "0" + monthsLeft;
}
if ( weeksLeft < 10 ) {
weeksLeft = "0" + weeksLeft;
}
if ( daysLeft < 10 ) {
daysLeft = "0" + daysLeft;
}
if ( hrsLeft < 10 ) {
hrsLeft = "0" + hrsLeft;
}
if ( minsLeft < 10 ) {
minsLeft = "0" + minsLeft;
}
if ( secLeft < 10 ) {
secLeft = "0" + secLeft;
}
}
yearsLeft = localNumber( yearsLeft, settings );
monthsLeft = localNumber( monthsLeft, settings );
weeksLeft = localNumber( weeksLeft, settings );
daysLeft = localNumber( daysLeft, settings );
hrsLeft = localNumber( hrsLeft, settings );
minsLeft = localNumber( minsLeft, settings );
secLeft = localNumber( secLeft, settings );
if ( settings.yearsAndMonths ) {
if ( !settings.omitZero || !hideYears ) {
template = template.replace('%y', yearsLeft);
template = template.replace('%ty', (yearsLeft == 1 && settings.yearSingularText) ? settings.yearSingularText : settings.yearText);
}
//Only hide months if years is at 0 as well as months
if ( !settings.omitZero || ( !hideYears && monthsLeft ) || ( !hideYears && !hideMonths ) ) {
template = template.replace('%m', monthsLeft);
template = template.replace('%tm', (monthsLeft == 1 && settings.monthSingularText) ? settings.monthSingularText : settings.monthText);
}
}
if ( settings.weeks && !hideWeeks ) {
template = template.replace('%w', weeksLeft);
template = template.replace('%tw', (weeksLeft == 1 && settings.weekSingularText) ? settings.weekSingularText : settings.weekText);
}
if ( !hideDays ) {
template = template.replace('%d', daysLeft);
template = template.replace('%td', (daysLeft == 1 && settings.daySingularText) ? settings.daySingularText : settings.dayText);
}
if ( !hideHours ) {
template = template.replace('%h', hrsLeft);
template = template.replace('%th', (hrsLeft == 1 && settings.hourSingularText) ? settings.hourSingularText : settings.hourText);
}
if ( !hideMins ) {
template = template.replace('%i', minsLeft);
template = template.replace('%ti', (minsLeft == 1 && settings.minSingularText) ? settings.minSingularText : settings.minText);
}
template = template.replace('%s', secLeft);
template = template.replace('%ts', (secLeft == 1 && settings.secSingularText) ? settings.secSingularText : settings.secText);
// Remove un-used tokens
template = template.replace(rTemplateTokens,'');
return template;
},
updateCallback = function() {
var $this = this,
template,
now,
date,
timeLeft,
yearsLeft = 0,
monthsLeft = 0,
weeksLeft = 0,
daysLeft,
hrsLeft,
minsLeft,
secLeft,
time = "",
diff,
extractSection = function( numSecs ) {
var amount;
amount = Math.floor( diff / numSecs );
diff -= amount * numSecs;
return amount;
},
settings = $this.data("jcdData");
// No settings so return
if ( !settings ) {
return false;
}
now = dateNow( $this );
if ( null !== settings.serverDiff ) {
date = new Date( settings.serverDiff + settings.clientdateNow.getTime() );
}
else {
date = settings.dateObj;
}
date.setMilliseconds(0);
timeLeft = ( settings.direction === "down" ) ? date.getTime() - now.getTime() : now.getTime() - date.getTime();
diff = Math.round( timeLeft / 1000 );
daysLeft = extractSection( secPerDay );
hrsLeft = extractSection( secPerHr );
minsLeft = extractSection( secPerMin );
secLeft = extractSection( secPerSec );
if ( settings.yearsAndMonths ) {
//Add days back on so we can calculate years easier
diff += ( daysLeft * secPerDay );
yearsLeft = extractSection( secPerYear );
monthsLeft = extractSection( secPerMonth );
daysLeft = extractSection( secPerDay );
}
if ( settings.weeks ) {
//Add days back on so we can calculate weeks easier
diff += ( daysLeft * secPerDay );
weeksLeft = extractSection( secPerWeek );
daysLeft = extractSection( secPerDay );
}
/**
* The following 3 option should never be used together!
* MAKE them work for any time
*/
if ( settings.hoursOnly || settings.minsOnly || settings.secsOnly )
{
if ( settings.yearsAndMonths ) {
//Add years, months, weeks and days back on so we can calculate dates easier
diff += ( yearsLeft * secPerYear );
diff += ( monthsLeft * secPerMonth );
yearsLeft = monthsLeft = 0;
}
if ( settings.weeks ) {
diff += ( weeksLeft * secPerWeek );
weeksLeft = 0;
}
}
// Assumes you are using dates within a month ( ~ 30 days )
// as years and months aren't taken into account
if ( settings.hoursOnly ) {
// Add days back on
diff += ( daysLeft * secPerDay );
// Add hours back on
diff += ( hrsLeft * secPerHr );
hrsLeft = extractSection( secPerHr );
}
// Assumes you are only using dates in the near future ( <= 24 hours )
// as years and months aren't taken into account
if ( settings.minsOnly ) {
// Add days back on
diff += ( daysLeft * secPerDay );
daysLeft = 0;
// Add hours back on
diff += ( hrsLeft * secPerHr );
hrsLeft = 0;
diff += ( minsLeft * secPerMin );
minsLeft = extractSection( secPerMin );
}
// Assumes you are only using dates in the near future ( <= 60 minutes )
// as years, months and days aren't taken into account
if ( settings.secsOnly ) {
// Add days back on
diff += ( daysLeft * secPerDay );
daysLeft = 0;
// Add hours back on
diff += ( hrsLeft * secPerHr );
hrsLeft = 0;
// Add minutes back on
diff += ( minsLeft * secPerMin );
minsLeft = 0;
// Add seconds back on
diff += secLeft;
secLeft = extractSection( secPerSec );
}
settings.yearsLeft = yearsLeft;
settings.monthsLeft = monthsLeft;
settings.weeksLeft = weeksLeft;
settings.daysLeft = daysLeft;
settings.hrsLeft = hrsLeft;
settings.minsLeft = minsLeft;
settings.secLeft = secLeft;
$this.data("jcdData", settings);
// Check if the countdown has finished
if ( !( settings.direction === "down" && ( now < date || settings.minus ) ) || !( settings.direction === "up" && ( date < now || settings.minus ) ) ) {
//settings.yearsLeft = settings.monthsLeft = settings.weeksLeft = settings.daysLeft = settings.hrsLeft = settings.minsLeft = settings.secLeft = 0;
//settings.hasCompleted = true;
}
// We've got the time sections set so we can now do the templating
if ( ( settings.direction === "down" && ( now < date || settings.minus ) ) || ( settings.direction === "up" && ( date < now || settings.minus ) ) ) {
time = generateTemplate( settings );
}
else {
settings.yearsLeft = settings.monthsLeft = settings.weeksLeft = settings.daysLeft = settings.hrsLeft = settings.minsLeft = settings.secLeft = 0;
time = generateTemplate( settings );
settings.hasCompleted = true;
}
$this.html( time ).triggerMulti("change.jcdevt,countChange", [settings]);
if ( settings.hasCompleted ) {
$this.triggerMulti("complete.jcdevt,countComplete");
clearInterval( settings.timer );
}
},
methods = {
init: function( options ) {
var opts = $.extend( {}, $.fn.countdown.defaults, options ),
testDate,
testString,
settings = {};
return this.each(function() {
var $this = $(this),
intervalCallback;
// If this element already has a countdown timer
// just change the settings
if ( $this.data("jcdData") ) {
$this.countdown("changeSettings", options, true);
opts = $this.data("jcdData");
}
if ( opts.date === null && opts.dataAttr === null ) {
$.error("No Date passed to jCountdown. date option is required.");
return true;
}
if ( opts.date ) {
testString = opts.date;
} else {
testString = $this.data(opts.dataAttr);
}
testDate = new Date(testString);
if( testDate.toString() === "Invalid Date" ) {
$.error("Invalid Date passed to jCountdown: " + testString);
}
// Setup any event handlers
addEventHandlers($this, options);
// Create a settings object based off the plugin options
settings = $.extend({}, opts);
// Cache DOM elements for templating
settings.dom = {};
settings.dom.$time = $("<"+settings.timeWrapElement+">").addClass(settings.timeWrapClass);
settings.dom.$text = $("<"+settings.textWrapElement+">").addClass(settings.textWrapClass);
settings.clientdateNow = new Date();
settings.clientdateNow.setMilliseconds(0);
settings.originalHTML = $this.html();
settings.dateObj = new Date( testString );
settings.dateObj.setMilliseconds(0);
settings.hasCompleted = false;
settings.timer = 0;
settings.yearsLeft = settings.monthsLeft = settings.weeksLeft = settings.daysLeft = settings.hrsLeft = settings.minsLeft = settings.secLeft = 0;
settings.difference = null;
// Create a callback for the interval
intervalCallback = $.proxy(updateCallback, $this);
settings.timer = setInterval(intervalCallback, settings.updateTime);
$this.data("jcdData", settings).triggerMulti("start.jcdevt,countStart", [settings]);
intervalCallback();
});
},
changeSettings: function( options, internal ) {
// Like resume but with resetting/changing options
return this.each(function() {
var $this = $(this),
settings,
testDate,
func = $.proxy( updateCallback, $this );
if ( !$this.data("jcdData") ) {
return true;
}
settings = $.extend( {}, $this.data("jcdData"), options );
if ( options.hasOwnProperty("date") ) {
testDate = new Date(options.date);
if ( testDate.toString() === "Invalid Date" ) {
$.error("Invalid Date passed to jCountdown: " + options.date);
}
}
settings.completed = false;
settings.dateObj = new Date( options.date );
// Clear the timer, as it might not be needed
clearInterval( settings.timer );
$this.off(".jcdevt").data("jcdData", settings);
// As this can be accessed via the init method as well,
// we need to check how this method is being accessed
if ( !internal ) {
addEventHandlers($this, settings);
settings.timer = setInterval( func, settings.updateTime );
$this.data("jcdData", settings);
func(); //Needs to run straight away when changing settings
}
settings = null;
});
},
resume: function() {
// Resumes a countdown timer
return this.each(function() {
var $this = $(this),
settings = $this.data("jcdData"),
func = $.proxy( updateCallback, $this );
if ( !settings ) {
return true;
}
$this.data("jcdData", settings).triggerMulti("resume.jcdevt,countResume", [settings] );
// We only want to resume a countdown that hasn't finished
if ( !settings.hasCompleted ) {
settings.timer = setInterval( func, settings.updateTime );
if ( settings.stopwatch && settings.direction === "up" ) {
var t = dateNow( $this ).getTime() - settings.pausedAt.getTime(),
d = new Date();
d.setTime( settings.dateObj.getTime() + t );
settings.dateObj = d; //This is internal date
}
func();
}
});
},
pause: function() {
// Pause a countdown timer
return this.each(function() {
var $this = $(this),
settings = $this.data("jcdData");
if ( !settings ) {
return true;
}
if ( settings.stopwatch ) {
settings.pausedAt = dateNow( $this );
}
// Clear interval (Will be started on resume)
clearInterval( settings.timer );
// Trigger pause event handler
$this.data("jcdData", settings).triggerMulti("pause.jcdevt,countPause", [settings] );
});
},
complete: function() {
return this.each(function() {
var $this = $(this),
settings = $this.data("jcdData");
if ( !settings ) {
return true;
}
// Clear timer
clearInterval( settings.timer );
settings.hasCompleted = true;
// Update setting, trigger complete event handler, then unbind all events
// We don"t delete the settings in case they need to be checked later on
$this.data("jcdData", settings).triggerMulti("complete.jcdevt,countComplete", [settings]);
});
},
destroy: function() {
return this.each(function() {
var $this = $(this),
settings = $this.data("jcdData");
if ( !settings ) {
return true;
}
// Clear timer
clearInterval( settings.timer );
// Unbind all events, remove data and put DOM Element back to its original state (HTML wise)
$this.off(".jcdevt").removeData("jcdData").html( settings.originalHTML );
});
},
getSettings: function( name ) {
var $this = $(this),
settings = $this.data("jcdData");
// If an individual setting is required
if ( name && settings ) {
// If it exists, return it
if ( settings.hasOwnProperty( name ) ) {
return settings[name];
}
return undefined;
}
// Return all settings or undefined
return settings;
},
changeLocale: function( locale ) {
var $this = $(this),
settings = $this.data("jcdData");
// If no locale exists error and return false
if ( !$.fn.countdown.locale[locale] ) {
$.error("Locale '" + locale + "' does not exist");
return false;
}
$.extend( settings, $.fn.countdown.locale[locale] );
$this.data("jcdData", settings).triggerMulti("locale.jcdevt,localeChange", [settings]);
return true;
}
};
if( methods[ method ] ) {
return methods[ method ].apply( this, Array.prototype.slice.call( arguments, 1 ) );
} else if ( typeof method === "object" || !method ) {
return methods.init.apply( this, arguments );
} else {
$.error("Method "+ method +" does not exist in the jCountdown Plugin");
}
};
// Expose defaults so we can easily override settings
// Useful for locale plugins
$.fn.countdown.defaults = {
date: null,
dataAttr: null,
updateTime: 1000,
yearText: 'years',
monthText: 'months',
weekText: 'weeks',
dayText: '天',
hourText: '小时',
minText: '分',
secText: '秒',
yearSingularText: 'year',
monthSingularText: 'month',
weekSingularText: 'week',
daySingularText: 'day',
hourSingularText: 'hour',
minSingularText: 'min',
secSingularText: 'sec',
digits : [0,1,2,3,4,5,6,7,8,9],
isRTL: false,
minus: false,
onStart: null,
onChange: null,
onComplete: null,
onResume: null,
onPause: null,
onLocaleChange: null,
leadingZero: false,
offset: null,
serverDiff:null,
hoursOnly: false,
minsOnly: false,
secsOnly: false,
weeks: false,
hours: false,
yearsAndMonths: false,
direction: "down",
stopwatch: false,
omitZero: false,
rtlTemplate: '%ts %s : %ti %i : %th %h : %tm %m : %ty %y',
template: '%y %ty : %m %tm : %h %th : %i %ti : %s %ts'
};
// Locale cache
$.fn.countdown.locale = [];
// Store default english locale so we can switch easier
$.fn.countdown.locale.en = {
yearText: 'years',
monthText: 'months',
weekText: 'weeks',
dayText: 'days',
hourText: 'hours',
minText: 'mins',
secText: 'sec',
yearSingularText: 'year',
monthSingularText: 'month',
weekSingularText: 'week',
daySingularText: 'day',
hourSingularText: 'hour',
minSingularText: 'min',
secSingularText: 'sec',
isRTL: false
};
//
$.fn.triggerMulti = function( eventTypes, extraParameters ) {
var events = eventTypes.split(",");
return this.each(function() {
var $this = $(this);
for ( var i = 0; i < events.length; i++) {
$this.trigger( events[i], extraParameters );
}
});
};
});

View File

@ -1,6 +1,8 @@
{% load contest %}
<h2 class="text-center">{{ contest.title }}</h2>
<div id="timer" class="text-center"></div>
<hr>
<div>
<table class="table table-bordered">

View File

@ -20,4 +20,26 @@
</ul>
{% include "oj/contest/_contest_header.html" %}
</div>
{% endblock %}
{% endblock %}
{% block js_block %}
<script type="x-template" id="template">
{% if contest.status %}
<strong><span>距离比赛开始还有:</span></strong>
{% else %}
<strong><span>距离比赛结束还有:</span></strong>
{% endif %}
<div class="day-timer timer-section">
<span class="time">%d</span> <span>%td</span>
</div>
<div class="hour-timer timer-section">
<span class="time">%h</span> : <span>%th</span>
</div>
<div class="min-timer timer-section">
<span class="time">%i</span> : <span>%ti</span>
</div>
<div class="second-timer timer-section">
<span class="time">%s</span> : <span>%ts</span>
</div>
</script>
<script src="/static/js/app/oj/contest/contestCountdown.js" defer="defer"></script>
{% endblock %}

View File

@ -39,5 +39,25 @@
</div>
{% endblock %}
{% block js_block %}
<script type="x-template" id="template">
{% if contest.status %}
<strong><span>距离比赛开始还有:</span></strong>
{% else %}
<strong><span>距离比赛结束还有:</span></strong>
{% endif %}
<div class="day-timer timer-section">
<span class="time">%d</span> <span>%td</span>
</div>
<div class="hour-timer timer-section">
<span class="time">%h</span> : <span>%th</span>
</div>
<div class="min-timer timer-section">
<span class="time">%i</span> : <span>%ti</span>
</div>
<div class="second-timer timer-section">
<span class="time">%s</span> : <span>%ts</span>
</div>
</script>
<script src="/static/js/app/oj/contest/contestPassword.js"></script>
<script src="/static/js/app/oj/contest/contestCountdown.js" defer="defer"></script>
{% endblock %}