Reformat using black
This commit is contained in:
parent
efee4ad081
commit
a87fb49918
221 changed files with 19127 additions and 7310 deletions
|
@ -16,8 +16,14 @@ from reversion_compare.admin import CompareVersionAdmin
|
|||
from django_ace import AceWidget
|
||||
from judge.models import Contest, ContestProblem, ContestSubmission, Profile, Rating
|
||||
from judge.ratings import rate_contest
|
||||
from judge.widgets import AdminHeavySelect2MultipleWidget, AdminHeavySelect2Widget, AdminPagedownWidget, \
|
||||
AdminSelect2MultipleWidget, AdminSelect2Widget, HeavyPreviewAdminPageDownWidget
|
||||
from judge.widgets import (
|
||||
AdminHeavySelect2MultipleWidget,
|
||||
AdminHeavySelect2Widget,
|
||||
AdminPagedownWidget,
|
||||
AdminSelect2MultipleWidget,
|
||||
AdminSelect2Widget,
|
||||
HeavyPreviewAdminPageDownWidget,
|
||||
)
|
||||
|
||||
|
||||
class AdminHeavySelect2Widget(AdminHeavySelect2Widget):
|
||||
|
@ -28,159 +34,252 @@ class AdminHeavySelect2Widget(AdminHeavySelect2Widget):
|
|||
|
||||
class ContestTagForm(ModelForm):
|
||||
contests = ModelMultipleChoiceField(
|
||||
label=_('Included contests'),
|
||||
label=_("Included contests"),
|
||||
queryset=Contest.objects.all(),
|
||||
required=False,
|
||||
widget=AdminHeavySelect2MultipleWidget(data_view='contest_select2'))
|
||||
widget=AdminHeavySelect2MultipleWidget(data_view="contest_select2"),
|
||||
)
|
||||
|
||||
|
||||
class ContestTagAdmin(admin.ModelAdmin):
|
||||
fields = ('name', 'color', 'description', 'contests')
|
||||
list_display = ('name', 'color')
|
||||
fields = ("name", "color", "description", "contests")
|
||||
list_display = ("name", "color")
|
||||
actions_on_top = True
|
||||
actions_on_bottom = True
|
||||
form = ContestTagForm
|
||||
|
||||
if AdminPagedownWidget is not None:
|
||||
formfield_overrides = {
|
||||
TextField: {'widget': AdminPagedownWidget},
|
||||
TextField: {"widget": AdminPagedownWidget},
|
||||
}
|
||||
|
||||
def save_model(self, request, obj, form, change):
|
||||
super(ContestTagAdmin, self).save_model(request, obj, form, change)
|
||||
obj.contests.set(form.cleaned_data['contests'])
|
||||
obj.contests.set(form.cleaned_data["contests"])
|
||||
|
||||
def get_form(self, request, obj=None, **kwargs):
|
||||
form = super(ContestTagAdmin, self).get_form(request, obj, **kwargs)
|
||||
if obj is not None:
|
||||
form.base_fields['contests'].initial = obj.contests.all()
|
||||
form.base_fields["contests"].initial = obj.contests.all()
|
||||
return form
|
||||
|
||||
|
||||
class ContestProblemInlineForm(ModelForm):
|
||||
class Meta:
|
||||
widgets = {'problem': AdminHeavySelect2Widget(data_view='problem_select2')}
|
||||
widgets = {"problem": AdminHeavySelect2Widget(data_view="problem_select2")}
|
||||
|
||||
|
||||
class ContestProblemInline(admin.TabularInline):
|
||||
model = ContestProblem
|
||||
verbose_name = _('Problem')
|
||||
verbose_name_plural = 'Problems'
|
||||
fields = ('problem', 'points', 'partial', 'is_pretested', 'max_submissions', 'output_prefix_override', 'order',
|
||||
'rejudge_column')
|
||||
readonly_fields = ('rejudge_column',)
|
||||
verbose_name = _("Problem")
|
||||
verbose_name_plural = "Problems"
|
||||
fields = (
|
||||
"problem",
|
||||
"points",
|
||||
"partial",
|
||||
"is_pretested",
|
||||
"max_submissions",
|
||||
"output_prefix_override",
|
||||
"order",
|
||||
"rejudge_column",
|
||||
)
|
||||
readonly_fields = ("rejudge_column",)
|
||||
form = ContestProblemInlineForm
|
||||
|
||||
def rejudge_column(self, obj):
|
||||
if obj.id is None:
|
||||
return ''
|
||||
return format_html('<a class="button rejudge-link" href="{}">Rejudge</a>',
|
||||
reverse('admin:judge_contest_rejudge', args=(obj.contest.id, obj.id)))
|
||||
rejudge_column.short_description = ''
|
||||
return ""
|
||||
return format_html(
|
||||
'<a class="button rejudge-link" href="{}">Rejudge</a>',
|
||||
reverse("admin:judge_contest_rejudge", args=(obj.contest.id, obj.id)),
|
||||
)
|
||||
|
||||
rejudge_column.short_description = ""
|
||||
|
||||
|
||||
class ContestForm(ModelForm):
|
||||
def __init__(self, *args, **kwargs):
|
||||
super(ContestForm, self).__init__(*args, **kwargs)
|
||||
if 'rate_exclude' in self.fields:
|
||||
if "rate_exclude" in self.fields:
|
||||
if self.instance and self.instance.id:
|
||||
self.fields['rate_exclude'].queryset = \
|
||||
Profile.objects.filter(contest_history__contest=self.instance).distinct()
|
||||
self.fields["rate_exclude"].queryset = Profile.objects.filter(
|
||||
contest_history__contest=self.instance
|
||||
).distinct()
|
||||
else:
|
||||
self.fields['rate_exclude'].queryset = Profile.objects.none()
|
||||
self.fields['banned_users'].widget.can_add_related = False
|
||||
self.fields['view_contest_scoreboard'].widget.can_add_related = False
|
||||
self.fields["rate_exclude"].queryset = Profile.objects.none()
|
||||
self.fields["banned_users"].widget.can_add_related = False
|
||||
self.fields["view_contest_scoreboard"].widget.can_add_related = False
|
||||
|
||||
def clean(self):
|
||||
cleaned_data = super(ContestForm, self).clean()
|
||||
cleaned_data['banned_users'].filter(current_contest__contest=self.instance).update(current_contest=None)
|
||||
cleaned_data["banned_users"].filter(
|
||||
current_contest__contest=self.instance
|
||||
).update(current_contest=None)
|
||||
|
||||
class Meta:
|
||||
widgets = {
|
||||
'authors': AdminHeavySelect2MultipleWidget(data_view='profile_select2'),
|
||||
'curators': AdminHeavySelect2MultipleWidget(data_view='profile_select2'),
|
||||
'testers': AdminHeavySelect2MultipleWidget(data_view='profile_select2'),
|
||||
'private_contestants': AdminHeavySelect2MultipleWidget(data_view='profile_select2',
|
||||
attrs={'style': 'width: 100%'}),
|
||||
'organizations': AdminHeavySelect2MultipleWidget(data_view='organization_select2'),
|
||||
'tags': AdminSelect2MultipleWidget,
|
||||
'banned_users': AdminHeavySelect2MultipleWidget(data_view='profile_select2',
|
||||
attrs={'style': 'width: 100%'}),
|
||||
'view_contest_scoreboard': AdminHeavySelect2MultipleWidget(data_view='profile_select2',
|
||||
attrs={'style': 'width: 100%'}),
|
||||
"authors": AdminHeavySelect2MultipleWidget(data_view="profile_select2"),
|
||||
"curators": AdminHeavySelect2MultipleWidget(data_view="profile_select2"),
|
||||
"testers": AdminHeavySelect2MultipleWidget(data_view="profile_select2"),
|
||||
"private_contestants": AdminHeavySelect2MultipleWidget(
|
||||
data_view="profile_select2", attrs={"style": "width: 100%"}
|
||||
),
|
||||
"organizations": AdminHeavySelect2MultipleWidget(
|
||||
data_view="organization_select2"
|
||||
),
|
||||
"tags": AdminSelect2MultipleWidget,
|
||||
"banned_users": AdminHeavySelect2MultipleWidget(
|
||||
data_view="profile_select2", attrs={"style": "width: 100%"}
|
||||
),
|
||||
"view_contest_scoreboard": AdminHeavySelect2MultipleWidget(
|
||||
data_view="profile_select2", attrs={"style": "width: 100%"}
|
||||
),
|
||||
}
|
||||
|
||||
if HeavyPreviewAdminPageDownWidget is not None:
|
||||
widgets['description'] = HeavyPreviewAdminPageDownWidget(preview=reverse_lazy('contest_preview'))
|
||||
widgets["description"] = HeavyPreviewAdminPageDownWidget(
|
||||
preview=reverse_lazy("contest_preview")
|
||||
)
|
||||
|
||||
|
||||
class ContestAdmin(CompareVersionAdmin):
|
||||
fieldsets = (
|
||||
(None, {'fields': ('key', 'name', 'authors', 'curators', 'testers')}),
|
||||
(_('Settings'), {'fields': ('is_visible', 'use_clarifications', 'hide_problem_tags', 'scoreboard_visibility',
|
||||
'run_pretests_only', 'points_precision')}),
|
||||
(_('Scheduling'), {'fields': ('start_time', 'end_time', 'time_limit')}),
|
||||
(_('Details'), {'fields': ('description', 'og_image', 'logo_override_image', 'tags', 'summary')}),
|
||||
(_('Format'), {'fields': ('format_name', 'format_config', 'problem_label_script')}),
|
||||
(_('Rating'), {'fields': ('is_rated', 'rate_all', 'rating_floor', 'rating_ceiling', 'rate_exclude')}),
|
||||
(_('Access'), {'fields': ('access_code', 'is_private', 'private_contestants', 'is_organization_private',
|
||||
'organizations', 'view_contest_scoreboard')}),
|
||||
(_('Justice'), {'fields': ('banned_users',)}),
|
||||
(None, {"fields": ("key", "name", "authors", "curators", "testers")}),
|
||||
(
|
||||
_("Settings"),
|
||||
{
|
||||
"fields": (
|
||||
"is_visible",
|
||||
"use_clarifications",
|
||||
"hide_problem_tags",
|
||||
"scoreboard_visibility",
|
||||
"run_pretests_only",
|
||||
"points_precision",
|
||||
)
|
||||
},
|
||||
),
|
||||
(_("Scheduling"), {"fields": ("start_time", "end_time", "time_limit")}),
|
||||
(
|
||||
_("Details"),
|
||||
{
|
||||
"fields": (
|
||||
"description",
|
||||
"og_image",
|
||||
"logo_override_image",
|
||||
"tags",
|
||||
"summary",
|
||||
)
|
||||
},
|
||||
),
|
||||
(
|
||||
_("Format"),
|
||||
{"fields": ("format_name", "format_config", "problem_label_script")},
|
||||
),
|
||||
(
|
||||
_("Rating"),
|
||||
{
|
||||
"fields": (
|
||||
"is_rated",
|
||||
"rate_all",
|
||||
"rating_floor",
|
||||
"rating_ceiling",
|
||||
"rate_exclude",
|
||||
)
|
||||
},
|
||||
),
|
||||
(
|
||||
_("Access"),
|
||||
{
|
||||
"fields": (
|
||||
"access_code",
|
||||
"is_private",
|
||||
"private_contestants",
|
||||
"is_organization_private",
|
||||
"organizations",
|
||||
"view_contest_scoreboard",
|
||||
)
|
||||
},
|
||||
),
|
||||
(_("Justice"), {"fields": ("banned_users",)}),
|
||||
)
|
||||
list_display = ('key', 'name', 'is_visible', 'is_rated', 'start_time', 'end_time', 'time_limit', 'user_count')
|
||||
search_fields = ('key', 'name')
|
||||
list_display = (
|
||||
"key",
|
||||
"name",
|
||||
"is_visible",
|
||||
"is_rated",
|
||||
"start_time",
|
||||
"end_time",
|
||||
"time_limit",
|
||||
"user_count",
|
||||
)
|
||||
search_fields = ("key", "name")
|
||||
inlines = [ContestProblemInline]
|
||||
actions_on_top = True
|
||||
actions_on_bottom = True
|
||||
form = ContestForm
|
||||
change_list_template = 'admin/judge/contest/change_list.html'
|
||||
filter_horizontal = ['rate_exclude']
|
||||
date_hierarchy = 'start_time'
|
||||
change_list_template = "admin/judge/contest/change_list.html"
|
||||
filter_horizontal = ["rate_exclude"]
|
||||
date_hierarchy = "start_time"
|
||||
|
||||
def get_actions(self, request):
|
||||
actions = super(ContestAdmin, self).get_actions(request)
|
||||
|
||||
if request.user.has_perm('judge.change_contest_visibility') or \
|
||||
request.user.has_perm('judge.create_private_contest'):
|
||||
for action in ('make_visible', 'make_hidden'):
|
||||
if request.user.has_perm(
|
||||
"judge.change_contest_visibility"
|
||||
) or request.user.has_perm("judge.create_private_contest"):
|
||||
for action in ("make_visible", "make_hidden"):
|
||||
actions[action] = self.get_action(action)
|
||||
|
||||
return actions
|
||||
|
||||
def get_queryset(self, request):
|
||||
queryset = Contest.objects.all()
|
||||
if request.user.has_perm('judge.edit_all_contest'):
|
||||
if request.user.has_perm("judge.edit_all_contest"):
|
||||
return queryset
|
||||
else:
|
||||
return queryset.filter(Q(authors=request.profile) | Q(curators=request.profile)).distinct()
|
||||
return queryset.filter(
|
||||
Q(authors=request.profile) | Q(curators=request.profile)
|
||||
).distinct()
|
||||
|
||||
def get_readonly_fields(self, request, obj=None):
|
||||
readonly = []
|
||||
if not request.user.has_perm('judge.contest_rating'):
|
||||
readonly += ['is_rated', 'rate_all', 'rate_exclude']
|
||||
if not request.user.has_perm('judge.contest_access_code'):
|
||||
readonly += ['access_code']
|
||||
if not request.user.has_perm('judge.create_private_contest'):
|
||||
readonly += ['is_private', 'private_contestants', 'is_organization_private', 'organizations']
|
||||
if not request.user.has_perm('judge.change_contest_visibility'):
|
||||
readonly += ['is_visible']
|
||||
if not request.user.has_perm('judge.contest_problem_label'):
|
||||
readonly += ['problem_label_script']
|
||||
if not request.user.has_perm("judge.contest_rating"):
|
||||
readonly += ["is_rated", "rate_all", "rate_exclude"]
|
||||
if not request.user.has_perm("judge.contest_access_code"):
|
||||
readonly += ["access_code"]
|
||||
if not request.user.has_perm("judge.create_private_contest"):
|
||||
readonly += [
|
||||
"is_private",
|
||||
"private_contestants",
|
||||
"is_organization_private",
|
||||
"organizations",
|
||||
]
|
||||
if not request.user.has_perm("judge.change_contest_visibility"):
|
||||
readonly += ["is_visible"]
|
||||
if not request.user.has_perm("judge.contest_problem_label"):
|
||||
readonly += ["problem_label_script"]
|
||||
return readonly
|
||||
|
||||
def save_model(self, request, obj, form, change):
|
||||
# `is_visible` will not appear in `cleaned_data` if user cannot edit it
|
||||
if form.cleaned_data.get('is_visible') and not request.user.has_perm('judge.change_contest_visibility'):
|
||||
if not form.cleaned_data['is_private'] and not form.cleaned_data['is_organization_private']:
|
||||
if form.cleaned_data.get("is_visible") and not request.user.has_perm(
|
||||
"judge.change_contest_visibility"
|
||||
):
|
||||
if (
|
||||
not form.cleaned_data["is_private"]
|
||||
and not form.cleaned_data["is_organization_private"]
|
||||
):
|
||||
raise PermissionDenied
|
||||
if not request.user.has_perm('judge.create_private_contest'):
|
||||
if not request.user.has_perm("judge.create_private_contest"):
|
||||
raise PermissionDenied
|
||||
|
||||
super().save_model(request, obj, form, change)
|
||||
|
||||
# We need this flag because `save_related` deals with the inlines, but does not know if we have already rescored
|
||||
self._rescored = False
|
||||
if form.changed_data and any(f in form.changed_data for f in ('format_config', 'format_name')):
|
||||
if form.changed_data and any(
|
||||
f in form.changed_data for f in ("format_config", "format_name")
|
||||
):
|
||||
self._rescore(obj.key)
|
||||
self._rescored = True
|
||||
|
||||
|
@ -188,10 +287,10 @@ class ContestAdmin(CompareVersionAdmin):
|
|||
super().save_related(request, form, formsets, change)
|
||||
# Only rescored if we did not already do so in `save_model`
|
||||
if not self._rescored and any(formset.has_changed() for formset in formsets):
|
||||
self._rescore(form.cleaned_data['key'])
|
||||
self._rescore(form.cleaned_data["key"])
|
||||
|
||||
def has_change_permission(self, request, obj=None):
|
||||
if not request.user.has_perm('judge.edit_own_contest'):
|
||||
if not request.user.has_perm("judge.edit_own_contest"):
|
||||
return False
|
||||
if obj is None:
|
||||
return True
|
||||
|
@ -199,76 +298,113 @@ class ContestAdmin(CompareVersionAdmin):
|
|||
|
||||
def _rescore(self, contest_key):
|
||||
from judge.tasks import rescore_contest
|
||||
|
||||
transaction.on_commit(rescore_contest.s(contest_key).delay)
|
||||
|
||||
def make_visible(self, request, queryset):
|
||||
if not request.user.has_perm('judge.change_contest_visibility'):
|
||||
queryset = queryset.filter(Q(is_private=True) | Q(is_organization_private=True))
|
||||
if not request.user.has_perm("judge.change_contest_visibility"):
|
||||
queryset = queryset.filter(
|
||||
Q(is_private=True) | Q(is_organization_private=True)
|
||||
)
|
||||
count = queryset.update(is_visible=True)
|
||||
self.message_user(request, ungettext('%d contest successfully marked as visible.',
|
||||
'%d contests successfully marked as visible.',
|
||||
count) % count)
|
||||
make_visible.short_description = _('Mark contests as visible')
|
||||
self.message_user(
|
||||
request,
|
||||
ungettext(
|
||||
"%d contest successfully marked as visible.",
|
||||
"%d contests successfully marked as visible.",
|
||||
count,
|
||||
)
|
||||
% count,
|
||||
)
|
||||
|
||||
make_visible.short_description = _("Mark contests as visible")
|
||||
|
||||
def make_hidden(self, request, queryset):
|
||||
if not request.user.has_perm('judge.change_contest_visibility'):
|
||||
queryset = queryset.filter(Q(is_private=True) | Q(is_organization_private=True))
|
||||
if not request.user.has_perm("judge.change_contest_visibility"):
|
||||
queryset = queryset.filter(
|
||||
Q(is_private=True) | Q(is_organization_private=True)
|
||||
)
|
||||
count = queryset.update(is_visible=True)
|
||||
self.message_user(request, ungettext('%d contest successfully marked as hidden.',
|
||||
'%d contests successfully marked as hidden.',
|
||||
count) % count)
|
||||
make_hidden.short_description = _('Mark contests as hidden')
|
||||
self.message_user(
|
||||
request,
|
||||
ungettext(
|
||||
"%d contest successfully marked as hidden.",
|
||||
"%d contests successfully marked as hidden.",
|
||||
count,
|
||||
)
|
||||
% count,
|
||||
)
|
||||
|
||||
make_hidden.short_description = _("Mark contests as hidden")
|
||||
|
||||
def get_urls(self):
|
||||
return [
|
||||
url(r'^rate/all/$', self.rate_all_view, name='judge_contest_rate_all'),
|
||||
url(r'^(\d+)/rate/$', self.rate_view, name='judge_contest_rate'),
|
||||
url(r'^(\d+)/judge/(\d+)/$', self.rejudge_view, name='judge_contest_rejudge'),
|
||||
url(r"^rate/all/$", self.rate_all_view, name="judge_contest_rate_all"),
|
||||
url(r"^(\d+)/rate/$", self.rate_view, name="judge_contest_rate"),
|
||||
url(
|
||||
r"^(\d+)/judge/(\d+)/$", self.rejudge_view, name="judge_contest_rejudge"
|
||||
),
|
||||
] + super(ContestAdmin, self).get_urls()
|
||||
|
||||
def rejudge_view(self, request, contest_id, problem_id):
|
||||
queryset = ContestSubmission.objects.filter(problem_id=problem_id).select_related('submission')
|
||||
queryset = ContestSubmission.objects.filter(
|
||||
problem_id=problem_id
|
||||
).select_related("submission")
|
||||
for model in queryset:
|
||||
model.submission.judge(rejudge=True)
|
||||
|
||||
self.message_user(request, ungettext('%d submission was successfully scheduled for rejudging.',
|
||||
'%d submissions were successfully scheduled for rejudging.',
|
||||
len(queryset)) % len(queryset))
|
||||
return HttpResponseRedirect(reverse('admin:judge_contest_change', args=(contest_id,)))
|
||||
self.message_user(
|
||||
request,
|
||||
ungettext(
|
||||
"%d submission was successfully scheduled for rejudging.",
|
||||
"%d submissions were successfully scheduled for rejudging.",
|
||||
len(queryset),
|
||||
)
|
||||
% len(queryset),
|
||||
)
|
||||
return HttpResponseRedirect(
|
||||
reverse("admin:judge_contest_change", args=(contest_id,))
|
||||
)
|
||||
|
||||
def rate_all_view(self, request):
|
||||
if not request.user.has_perm('judge.contest_rating'):
|
||||
if not request.user.has_perm("judge.contest_rating"):
|
||||
raise PermissionDenied()
|
||||
with transaction.atomic():
|
||||
with connection.cursor() as cursor:
|
||||
cursor.execute('TRUNCATE TABLE `%s`' % Rating._meta.db_table)
|
||||
cursor.execute("TRUNCATE TABLE `%s`" % Rating._meta.db_table)
|
||||
Profile.objects.update(rating=None)
|
||||
for contest in Contest.objects.filter(is_rated=True, end_time__lte=timezone.now()).order_by('end_time'):
|
||||
for contest in Contest.objects.filter(
|
||||
is_rated=True, end_time__lte=timezone.now()
|
||||
).order_by("end_time"):
|
||||
rate_contest(contest)
|
||||
return HttpResponseRedirect(reverse('admin:judge_contest_changelist'))
|
||||
return HttpResponseRedirect(reverse("admin:judge_contest_changelist"))
|
||||
|
||||
def rate_view(self, request, id):
|
||||
if not request.user.has_perm('judge.contest_rating'):
|
||||
if not request.user.has_perm("judge.contest_rating"):
|
||||
raise PermissionDenied()
|
||||
contest = get_object_or_404(Contest, id=id)
|
||||
if not contest.is_rated or not contest.ended:
|
||||
raise Http404()
|
||||
with transaction.atomic():
|
||||
contest.rate()
|
||||
return HttpResponseRedirect(request.META.get('HTTP_REFERER', reverse('admin:judge_contest_changelist')))
|
||||
return HttpResponseRedirect(
|
||||
request.META.get("HTTP_REFERER", reverse("admin:judge_contest_changelist"))
|
||||
)
|
||||
|
||||
def get_form(self, request, obj=None, **kwargs):
|
||||
form = super(ContestAdmin, self).get_form(request, obj, **kwargs)
|
||||
if 'problem_label_script' in form.base_fields:
|
||||
if "problem_label_script" in form.base_fields:
|
||||
# form.base_fields['problem_label_script'] does not exist when the user has only view permission
|
||||
# on the model.
|
||||
form.base_fields['problem_label_script'].widget = AceWidget('lua', request.profile.ace_theme)
|
||||
|
||||
perms = ('edit_own_contest', 'edit_all_contest')
|
||||
form.base_fields['curators'].queryset = Profile.objects.filter(
|
||||
Q(user__is_superuser=True) |
|
||||
Q(user__groups__permissions__codename__in=perms) |
|
||||
Q(user__user_permissions__codename__in=perms),
|
||||
form.base_fields["problem_label_script"].widget = AceWidget(
|
||||
"lua", request.profile.ace_theme
|
||||
)
|
||||
|
||||
perms = ("edit_own_contest", "edit_all_contest")
|
||||
form.base_fields["curators"].queryset = Profile.objects.filter(
|
||||
Q(user__is_superuser=True)
|
||||
| Q(user__groups__permissions__codename__in=perms)
|
||||
| Q(user__user_permissions__codename__in=perms),
|
||||
).distinct()
|
||||
return form
|
||||
|
||||
|
@ -276,29 +412,48 @@ class ContestAdmin(CompareVersionAdmin):
|
|||
class ContestParticipationForm(ModelForm):
|
||||
class Meta:
|
||||
widgets = {
|
||||
'contest': AdminSelect2Widget(),
|
||||
'user': AdminHeavySelect2Widget(data_view='profile_select2'),
|
||||
"contest": AdminSelect2Widget(),
|
||||
"user": AdminHeavySelect2Widget(data_view="profile_select2"),
|
||||
}
|
||||
|
||||
|
||||
class ContestParticipationAdmin(admin.ModelAdmin):
|
||||
fields = ('contest', 'user', 'real_start', 'virtual', 'is_disqualified')
|
||||
list_display = ('contest', 'username', 'show_virtual', 'real_start', 'score', 'cumtime', 'tiebreaker')
|
||||
actions = ['recalculate_results']
|
||||
fields = ("contest", "user", "real_start", "virtual", "is_disqualified")
|
||||
list_display = (
|
||||
"contest",
|
||||
"username",
|
||||
"show_virtual",
|
||||
"real_start",
|
||||
"score",
|
||||
"cumtime",
|
||||
"tiebreaker",
|
||||
)
|
||||
actions = ["recalculate_results"]
|
||||
actions_on_bottom = actions_on_top = True
|
||||
search_fields = ('contest__key', 'contest__name', 'user__user__username')
|
||||
search_fields = ("contest__key", "contest__name", "user__user__username")
|
||||
form = ContestParticipationForm
|
||||
date_hierarchy = 'real_start'
|
||||
date_hierarchy = "real_start"
|
||||
|
||||
def get_queryset(self, request):
|
||||
return super(ContestParticipationAdmin, self).get_queryset(request).only(
|
||||
'contest__name', 'contest__format_name', 'contest__format_config',
|
||||
'user__user__username', 'real_start', 'score', 'cumtime', 'tiebreaker', 'virtual',
|
||||
return (
|
||||
super(ContestParticipationAdmin, self)
|
||||
.get_queryset(request)
|
||||
.only(
|
||||
"contest__name",
|
||||
"contest__format_name",
|
||||
"contest__format_config",
|
||||
"user__user__username",
|
||||
"real_start",
|
||||
"score",
|
||||
"cumtime",
|
||||
"tiebreaker",
|
||||
"virtual",
|
||||
)
|
||||
)
|
||||
|
||||
def save_model(self, request, obj, form, change):
|
||||
super().save_model(request, obj, form, change)
|
||||
if form.changed_data and 'is_disqualified' in form.changed_data:
|
||||
if form.changed_data and "is_disqualified" in form.changed_data:
|
||||
obj.set_disqualified(obj.is_disqualified)
|
||||
|
||||
def recalculate_results(self, request, queryset):
|
||||
|
@ -306,17 +461,26 @@ class ContestParticipationAdmin(admin.ModelAdmin):
|
|||
for participation in queryset:
|
||||
participation.recompute_results()
|
||||
count += 1
|
||||
self.message_user(request, ungettext('%d participation recalculated.',
|
||||
'%d participations recalculated.',
|
||||
count) % count)
|
||||
recalculate_results.short_description = _('Recalculate results')
|
||||
self.message_user(
|
||||
request,
|
||||
ungettext(
|
||||
"%d participation recalculated.",
|
||||
"%d participations recalculated.",
|
||||
count,
|
||||
)
|
||||
% count,
|
||||
)
|
||||
|
||||
recalculate_results.short_description = _("Recalculate results")
|
||||
|
||||
def username(self, obj):
|
||||
return obj.user.username
|
||||
username.short_description = _('username')
|
||||
username.admin_order_field = 'user__user__username'
|
||||
|
||||
username.short_description = _("username")
|
||||
username.admin_order_field = "user__user__username"
|
||||
|
||||
def show_virtual(self, obj):
|
||||
return obj.virtual or '-'
|
||||
show_virtual.short_description = _('virtual')
|
||||
show_virtual.admin_order_field = 'virtual'
|
||||
return obj.virtual or "-"
|
||||
|
||||
show_virtual.short_description = _("virtual")
|
||||
show_virtual.admin_order_field = "virtual"
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue