Accept Merge Request #45 : (sxw-dev -> dev)
Merge Request: 更新后台announcement的功能 Created By: @esp Accepted By: @esp URL: https://coding.net/u/virusdefender/p/qduoj/git/merge/45
This commit is contained in:
39
account/migrations/0001_initial.py
Normal file
39
account/migrations/0001_initial.py
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
from django.db import models, migrations
|
||||||
|
import account.models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='User',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(verbose_name='ID', serialize=False, auto_created=True, primary_key=True)),
|
||||||
|
('password', models.CharField(max_length=128, verbose_name='password')),
|
||||||
|
('last_login', models.DateTimeField(null=True, verbose_name='last login', blank=True)),
|
||||||
|
('username', models.CharField(unique=True, max_length=30)),
|
||||||
|
('real_name', models.CharField(max_length=30, null=True, blank=True)),
|
||||||
|
('email', models.EmailField(max_length=254, null=True, blank=True)),
|
||||||
|
('create_time', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('admin_type', models.IntegerField(default=0)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'db_table': 'user',
|
||||||
|
},
|
||||||
|
managers=[
|
||||||
|
('objects', account.models.UserManager()),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='AdminGroup',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(verbose_name='ID', serialize=False, auto_created=True, primary_key=True)),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
]
|
||||||
0
account/migrations/__init__.py
Normal file
0
account/migrations/__init__.py
Normal file
@@ -11,4 +11,4 @@ class AdminTemplateView(APIView):
|
|||||||
try:
|
try:
|
||||||
return HttpResponse(open(path).read(), content_type="text/html")
|
return HttpResponse(open(path).read(), content_type="text/html")
|
||||||
except IOError:
|
except IOError:
|
||||||
return HttpResponse(u"模板不存在", content_type="text/html")
|
raise Http404
|
||||||
|
|||||||
@@ -25,3 +25,7 @@ img {
|
|||||||
label {
|
label {
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[ms-controller] {
|
||||||
|
display: none
|
||||||
|
}
|
||||||
@@ -1,17 +1,23 @@
|
|||||||
require(["jquery", "avalon", "csrf", "bs_alert", "editor", "validation"], function ($, avalon, csrfHeader, bs_alert, editor) {
|
require(["jquery", "avalon", "csrf", "bs_alert", "editor", "validation"], function ($, avalon, csrfHeader, bs_alert, editor) {
|
||||||
announcementEditor = editor("#editor"); //创建新建公告的内容编辑器
|
announcementEditor = editor("#editor"); //创建新建公告的内容编辑器
|
||||||
editAnnouncementEditor = null;
|
editAnnouncementEditor = editor("#editAnnouncementEditor");
|
||||||
|
|
||||||
|
avalon.vmodels.announcement = null;
|
||||||
|
|
||||||
if (!avalon.vmodels.announcement) // 防止模式重新定义
|
|
||||||
{
|
|
||||||
// avalon:定义模式 announcement
|
// avalon:定义模式 announcement
|
||||||
vm = avalon.define({
|
avalon.ready(function () {
|
||||||
|
var vm = avalon.define({
|
||||||
$id: "announcement",
|
$id: "announcement",
|
||||||
|
//通用变量
|
||||||
announcement: [], // 公告列表数据项
|
announcement: [], // 公告列表数据项
|
||||||
previous_page: 0, // 之前的页数
|
previous_page: 0, // 之前的页数
|
||||||
next_page: 0, // 之后的页数
|
next_page: 0, // 之后的页数
|
||||||
page: 1, // 当前页数
|
page: 1, // 当前页数
|
||||||
isEditing: 0, // 正在编辑的公告的ID, 为零说明未在编辑
|
isEditing: 0, // 正在编辑的公告的ID, 为零说明未在编辑
|
||||||
|
page_count: 1, // 总页数
|
||||||
|
visableOnly: false, //仅显示可见公告
|
||||||
|
// 编辑
|
||||||
|
announcementVisible: 0,
|
||||||
getState: function (el) { //获取公告当前状态,显示
|
getState: function (el) { //获取公告当前状态,显示
|
||||||
if (el.visible)
|
if (el.visible)
|
||||||
return "可见";
|
return "可见";
|
||||||
@@ -21,14 +27,14 @@ require(["jquery", "avalon", "csrf", "bs_alert", "editor", "validation"], functi
|
|||||||
getNext: function (el) {
|
getNext: function (el) {
|
||||||
if (!vm.next_page)
|
if (!vm.next_page)
|
||||||
return;
|
return;
|
||||||
getPageData(++(vm.page));
|
getPageData(vm.page + 1);
|
||||||
},
|
},
|
||||||
getPrevious: function (el) {
|
getPrevious: function (el) {
|
||||||
if (!vm.previous_page)
|
if (!vm.previous_page)
|
||||||
return;
|
return;
|
||||||
getPageData(--(vm.page));
|
getPageData(vm.page - 1);
|
||||||
},
|
},
|
||||||
getBtnClass: function (btn) {
|
getBtnClass: function (btn) { //上一页/下一页按钮启用禁用逻辑
|
||||||
if (btn) {
|
if (btn) {
|
||||||
return vm.next_page ? "btn btn-primary" : "btn btn-primary disabled";
|
return vm.next_page ? "btn btn-primary" : "btn btn-primary disabled";
|
||||||
}
|
}
|
||||||
@@ -39,11 +45,8 @@ require(["jquery", "avalon", "csrf", "bs_alert", "editor", "validation"], functi
|
|||||||
},
|
},
|
||||||
enEdit: function (el) { //点击编辑按钮的事件,显示/隐藏编辑区
|
enEdit: function (el) { //点击编辑按钮的事件,显示/隐藏编辑区
|
||||||
$("#newTitle").val(el.title);
|
$("#newTitle").val(el.title);
|
||||||
if (!editAnnouncementEditor) //初始化编辑器
|
|
||||||
editAnnouncementEditor = editor("#editAnnouncementEditor");
|
|
||||||
editAnnouncementEditor.setValue(el.content);
|
editAnnouncementEditor.setValue(el.content);
|
||||||
if (el.visible == false)
|
vm.announcementVisible = el.visible;
|
||||||
$("#hidden").attr("checked", true);
|
|
||||||
if (vm.isEditing == el.id)
|
if (vm.isEditing == el.id)
|
||||||
vm.isEditing = 0;
|
vm.isEditing = 0;
|
||||||
else
|
else
|
||||||
@@ -54,21 +57,25 @@ require(["jquery", "avalon", "csrf", "bs_alert", "editor", "validation"], functi
|
|||||||
vm.isEditing = 0;
|
vm.isEditing = 0;
|
||||||
},
|
},
|
||||||
submitChange: function () { // 处理编辑公告提交事件,顺便验证字段为空
|
submitChange: function () { // 处理编辑公告提交事件,顺便验证字段为空
|
||||||
var title = $("#newTitle").val(), content = editAnnouncementEditor.getValue(), visible = true;
|
var title = $("#newTitle").val(), content = editAnnouncementEditor.getValue();
|
||||||
if ($("#hidden").attr("checked") == true)
|
|
||||||
visible = false;
|
|
||||||
if (title != "") {
|
if (title != "") {
|
||||||
if (content != "") {
|
if (content != "") {
|
||||||
$.ajax({ //发送修改公告请求
|
$.ajax({ //发送修改公告请求
|
||||||
beforeSend: csrfHeader,
|
beforeSend: csrfHeader,
|
||||||
url: "/api/edit_announcements/",
|
url: "/api/admin/announcement/",
|
||||||
dataType: "json",
|
dataType: "json",
|
||||||
method: "post",
|
method: "put",
|
||||||
data: {id: vm.isEditing, title: title, content: content, visible: visible},
|
data: {
|
||||||
|
id: vm.isEditing,
|
||||||
|
title: title,
|
||||||
|
content: content,
|
||||||
|
visible: vm.announcementVisible
|
||||||
|
},
|
||||||
success: function (data) {
|
success: function (data) {
|
||||||
if (!data.code) {
|
if (!data.code) {
|
||||||
bs_alert("修改成功");
|
bs_alert("修改成功");
|
||||||
vm.isEditing = 0;
|
vm.isEditing = 0;
|
||||||
|
getPageData(1);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
bs_alert(data.data);
|
bs_alert(data.data);
|
||||||
@@ -83,24 +90,31 @@ require(["jquery", "avalon", "csrf", "bs_alert", "editor", "validation"], functi
|
|||||||
bs_alert("公告标题不能为空");
|
bs_alert("公告标题不能为空");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
vm.$watch("visableOnly", function () {
|
||||||
|
getPageData(1);
|
||||||
|
});
|
||||||
|
|
||||||
avalon.scan();
|
avalon.scan();
|
||||||
|
|
||||||
getPageData(1); //公告列表初始化
|
getPageData(1); //公告列表初始化
|
||||||
vm.page = 1;
|
|
||||||
vm.isEditing = 0;
|
|
||||||
//Ajax get数据
|
//Ajax get数据
|
||||||
function getPageData(page) {
|
function getPageData(page) {
|
||||||
|
var visible = '';
|
||||||
|
if (vm.visableOnly == true)
|
||||||
|
visible = "&visible=true";
|
||||||
$.ajax({
|
$.ajax({
|
||||||
beforeSend: csrfHeader,
|
beforeSend: csrfHeader,
|
||||||
url: "/api/announcements/?paging=true&page=" + page + "&page_size=10",
|
url: "/api/announcements/?paging=true&page=" + page + "&page_size=10" + visible,
|
||||||
dataType: "json",
|
dataType: "json",
|
||||||
method: "get",
|
method: "get",
|
||||||
success: function (data) {
|
success: function (data) {
|
||||||
if (!data.code) {
|
if (!data.code) {
|
||||||
vm.announcement = data.data.results;
|
vm.announcement = data.data.results;
|
||||||
|
vm.page_count = data.data.total_page;
|
||||||
vm.previous_page = data.data.previous_page;
|
vm.previous_page = data.data.previous_page;
|
||||||
vm.next_page = data.data.next_page;
|
vm.next_page = data.data.next_page;
|
||||||
|
vm.page = page;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
bs_alert(data.data);
|
bs_alert(data.data);
|
||||||
@@ -150,5 +164,7 @@ require(["jquery", "avalon", "csrf", "bs_alert", "editor", "validation"], functi
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
});
|
});
|
||||||
@@ -103,7 +103,7 @@
|
|||||||
<img src="/static/img/loading.gif" id="loading-gif">
|
<img src="/static/img/loading.gif" id="loading-gif">
|
||||||
<!-- custom body begin -->
|
<!-- custom body begin -->
|
||||||
|
|
||||||
<div class='col-md-8' ms-include-src="template_url" data-include-rendered="hide_loading"></div>
|
<div ms-include-src="template_url" data-include-rendered="hide_loading"></div>
|
||||||
|
|
||||||
<!-- custom body end -->
|
<!-- custom body end -->
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
67
template/admin/announcement/announcement.html
Normal file
67
template/admin/announcement/announcement.html
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
<style></style>
|
||||||
|
|
||||||
|
<div ms-controller="announcement" class="col-md-9">
|
||||||
|
<h1>Announcement</h1>
|
||||||
|
<table class="table table-striped">
|
||||||
|
<tr>
|
||||||
|
<th>编号</th>
|
||||||
|
<th>标题</th>
|
||||||
|
<th>创建时间</th>
|
||||||
|
<th>更新时间</th>
|
||||||
|
<th>创建者</th>
|
||||||
|
<th>状态</th>
|
||||||
|
<th>操作</th>
|
||||||
|
</tr>
|
||||||
|
<tr ms-repeat="announcement">
|
||||||
|
<td>{{el.id}}</td>
|
||||||
|
<td>{{el.title}}</td>
|
||||||
|
<td>{{el.create_time|date("yyyy-MM-dd HH:mm:ss")}}</td>
|
||||||
|
<td>{{el.last_update_time|date("yyyy-MM-dd HH:mm:ss")}}</td>
|
||||||
|
<td>{{el.created_by.username}}</td>
|
||||||
|
<td>{{getState(el)}}</td>
|
||||||
|
<td>
|
||||||
|
<button class="btn-sm btn-info" ms-click="enEdit(el)">编辑</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
<div class="form-group">
|
||||||
|
<label>仅显示可见 <input ms-duplex-checked="visableOnly" type="checkbox"/></label>
|
||||||
|
</div>
|
||||||
|
<div class="text-right">
|
||||||
|
页数:{{page}}/{{page_count}}
|
||||||
|
<botton ms-attr-class="getBtnClass(0)" ms-click="getPrevious">上一页</botton>
|
||||||
|
<botton ms-attr-class="getBtnClass(1)" ms-click="getNext">下一页</botton>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div ms-visible="isEditing">
|
||||||
|
<h3>编辑公告</h3>
|
||||||
|
<div class="form-group"><label for="title">标题</label>
|
||||||
|
<input name="title" type="text" class="form-control" id="newTitle" placeholder="公告标题" value=""></div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label>内容</label>
|
||||||
|
<textarea id="editAnnouncementEditor"></textarea>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label>可见 <input ms-duplex-checked="announcementVisible" type="checkbox"/></label>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<button ms-click="submitChange()" class="btn btn-primary">提交</button>
|
||||||
|
|
||||||
|
<button ms-click="disEdit()" class="btn btn-danger">取消</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<h3>添加公告</h3>
|
||||||
|
|
||||||
|
<form id="announcement-form">
|
||||||
|
<div class="form-group"><label for="title">标题</label>
|
||||||
|
<input name="title" type="text" class="form-control" id="title" placeholder="公告标题"></div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label>内容</label>
|
||||||
|
<textarea id="editor" placeholder="公告内容"></textarea>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<button type="submit" class="btn btn-primary">提交</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
<script src="/static/js/app/admin/announcement/announcement.js"></script>
|
||||||
@@ -1,6 +1,4 @@
|
|||||||
<div ms-controller="add_contest">
|
<div ms-controller="add_contest">
|
||||||
|
|
||||||
|
|
||||||
<form id="add-contest-form">
|
<form id="add-contest-form">
|
||||||
<div class="col-md-9">
|
<div class="col-md-9">
|
||||||
<div class="col-md-12">
|
<div class="col-md-12">
|
||||||
|
|||||||
@@ -70,7 +70,11 @@ def paginate(request, query_set, object_serializer):
|
|||||||
except Exception:
|
except Exception:
|
||||||
return error_response(u"参数错误")
|
return error_response(u"参数错误")
|
||||||
|
|
||||||
data = {"results": object_serializer(current_page, many=True).data, "previous_page": None, "next_page": None}
|
data = {"results": object_serializer(current_page, many=True).data,
|
||||||
|
"previous_page": None,
|
||||||
|
"next_page": None,
|
||||||
|
"count": paginator.count,
|
||||||
|
"total_page": paginator.num_pages}
|
||||||
|
|
||||||
try:
|
try:
|
||||||
data["previous_page"] = current_page.previous_page_number()
|
data["previous_page"] = current_page.previous_page_number()
|
||||||
|
|||||||
@@ -51,14 +51,18 @@ class PaginatorTest(APITestCase):
|
|||||||
self.assertEqual(response.data, {"code": 1, "data": u"参数错误"})
|
self.assertEqual(response.data, {"code": 1, "data": u"参数错误"})
|
||||||
|
|
||||||
def test_correct_paginate(self):
|
def test_correct_paginate(self):
|
||||||
response = self.client.get(self.url + "?paging=true&limit=1&page_size=1&page=1")
|
response = self.client.get(self.url + "?paging=true&page_size=1&page=1")
|
||||||
self.assertEqual(response.data["code"], 0)
|
self.assertEqual(response.data["code"], 0)
|
||||||
self.assertEqual(response.data["data"]["previous_page"], None)
|
self.assertEqual(response.data["data"]["previous_page"], None)
|
||||||
self.assertEqual(response.data["data"]["next_page"], 2)
|
self.assertEqual(response.data["data"]["next_page"], 2)
|
||||||
self.assertEqual(len(response.data["data"]["results"]), 1)
|
self.assertEqual(len(response.data["data"]["results"]), 1)
|
||||||
|
self.assertEqual(response.data["data"]["count"], 2)
|
||||||
|
self.assertEqual(response.data["data"]["total_page"], 2)
|
||||||
|
|
||||||
response = self.client.get(self.url + "?paging=true&limit=1&page_size=2&page=1")
|
response = self.client.get(self.url + "?paging=true&page_size=2&page=1")
|
||||||
self.assertEqual(response.data["code"], 0)
|
self.assertEqual(response.data["code"], 0)
|
||||||
self.assertEqual(response.data["data"]["previous_page"], None)
|
self.assertEqual(response.data["data"]["previous_page"], None)
|
||||||
self.assertEqual(response.data["data"]["next_page"], None)
|
self.assertEqual(response.data["data"]["next_page"], None)
|
||||||
self.assertEqual(len(response.data["data"]["results"]), 2)
|
self.assertEqual(len(response.data["data"]["results"]), 2)
|
||||||
|
self.assertEqual(response.data["data"]["count"], 2)
|
||||||
|
self.assertEqual(response.data["data"]["total_page"], 1)
|
||||||
|
|||||||
Reference in New Issue
Block a user