已上线演示地址: http://movie.mtianyan.cn
项目源码地址:https://github.com/mtianyan/movie_project
标签管理
- 模型:Tag
- 表单: TagForm
- 请求方法: GET POST
- 访问控制:
@admin_login_req
前往forms定义TagForm
class TagForm(FlaskForm):
name = StringField(
label="名称",
validators=[
DataRequired("标签名不能为空")
],
description="标签",
render_kw={
"class": "form-control",
"id": "input_name",
"placeholder": "请输入标签名称!"
}
)
submit = SubmitField(
'添加',
render_kw={
"class": "btn btn-primary",
}
)
admin/views.py:
form = TagForm()
return render_template("admin/tag_add.html", form=form)
实例化出一个Tagform,然后将form传递到前台去。
下面在html中使用form
data:image/s3,"s3://crabby-images/e17fa/e17fa905fe7db8d8777c514d430e7db78e9662db" alt="mark"
在提交前加上csrf token
data:image/s3,"s3://crabby-images/81bfc/81bfca1dc76d7259b092fcfeac859d18e2cfce4b" alt="mark"
数据库入库操作之前:html中定义展示错误信息
data:image/s3,"s3://crabby-images/94afb/94afb6c3356054931feeb784c10caa3cfc1ee263" alt="mark"
此时验证没有标红的提示信息。是因为没有定义我们的method方法。
data:image/s3,"s3://crabby-images/202bb/202bb744e47b9374b2736fb3029dad80d28a598b" alt="mark"
Method Not Allowed
The method is not allowed for the requested URL.
这是因为虽然在form中定义了method,但是在views中没有接受。
@admin.route("/tag/add/", methods=["GET", "POST"])
标签不能重名
data = form.data
tag = Tag.query.filter_by(name=data["name"]).count()
# 说明已经有这个标签了
if tag == 1:
flash("标签已存在", "err")
return redirect(url_for("admin.tag_add"))
tag = Tag(
name= data["name"]
)
db.session.add(tag)
db.session.commit()
flash("标签添加成功", "ok")
为flash进行分类,第二个参数指定flash信息的分类
data:image/s3,"s3://crabby-images/581e2/581e22fe0dff924090e868271c7125f8f8335585" alt="mark"
添加弹窗alert
data:image/s3,"s3://crabby-images/4f0e7/4f0e7ba40b66fae59cf45320aad8a41f31b6c164" alt="mark"
标签的分页展示
前往views中的标签列表进行查询以及分页。
@admin.route("/tag/list/<int:page>/", methods= ["GET"])
@admin_login_req
def tag_list(page=None):
"""
标签列表
"""
if page is None:
page = 1
page_data = Tag.query.order_by(
Tag.addtime.desc()
).paginate(page=page, per_page=1)
return render_template("admin/tag_list.html", page_data=page_data)
html中
data:image/s3,"s3://crabby-images/8b892/8b8929b5bf512ca80f5bd099282f34215af3e4f2" alt="mark"
修改gird中标签列表的url,为其添加参数page =1
data:image/s3,"s3://crabby-images/da30a/da30a4c70287d92387b89acff38c875fea724b94" alt="mark"
data:image/s3,"s3://crabby-images/d1ea2/d1ea21b34980246e4cec7f44d4c79ddcdc60c4f8" alt="mark"
将页码信息拷贝走。
data:image/s3,"s3://crabby-images/38b3e/38b3e46451ac9b70b7c542bbd61c0d2458e315ee" alt="mark"
并使用macro进行页码的动态显示
然后在要使用分页的html文件中import之后调用pg.page方法。传入分页后的数据。以及名称
data:image/s3,"s3://crabby-images/cf571/cf571f86ca3d7582f5439d7cd2714b4f84f5fa8c" alt="mark"
data:image/s3,"s3://crabby-images/e0ca9/e0ca9d6aee56dcf7740b6e2552892bb265fd8371" alt="mark"
标签的删除
@admin.route("/tag/del/<int:id>/", methods= ["GET"])
@admin_login_req
def tag_del(id=None):
"""
标签删除
"""
# filter_by在查不到或多个的时候并不会报错,get会报错。
tag = Tag.query.filter_by(id=id).first_or_404()
db.session.delete(tag)
db.session.commit()
flash("标签<<{0}>>删除成功".format(tag.name), "ok")
return redirect(url_for("admin.tag_list", page=1))
将tag add中的alert弹框拷贝一份粘贴到tag list中接收闪现。
data:image/s3,"s3://crabby-images/077eb/077eb3928ed31799fff5e93fcd44d9ae25f832e1" alt="mark"
for msg in 和endfor忘了复制过来记得自己加。
为删除字符添加对应的处理view函数
data:image/s3,"s3://crabby-images/8ed98/8ed981e8e81dc11f031276e30ce95b8c9389f94c" alt="mark"
编辑标签
@admin.route("/tag/edit/<int:id>", methods=["GET", "POST"])
@admin_login_req
def tag_edit(id=None):
"""
标签编辑
"""
form = TagForm()
form.submit.label.text = "编辑"
tag = Tag.query.get_or_404(id)
if form.validate_on_submit():
data = form.data
tag_count = Tag.query.filter_by(name=data["name"]).count()
# 说明已经有这个标签了,此时向添加一个与其他标签重名的标签。
if tag.name != data["name"] and tag_count == 1:
flash("标签已存在", "err")
return redirect(url_for("admin.tag_edit", id=tag.id))
tag.name = data["name"]
db.session.add(tag)
db.session.commit()
flash("标签修改成功", "ok")
redirect(url_for("admin.tag_edit", id=tag.id))
return render_template("admin/tag_edit.html", form=form, tag=tag)
创建一个tag edit的html将tag add的内容拷贝进来
- 修改汉字将添加标签修改为编辑标签。
data:image/s3,"s3://crabby-images/6e13a/6e13af411b99c867571e9074863802a46fc81a67" alt="mark"
然后为from.name添加上value值。
在taglist中修改指向tag edit的路由
此处对于submit字段的text修改,必须放在redirect之前。
data:image/s3,"s3://crabby-images/5c838/5c838a7fc4d414bdb448b0f878a808b97a197a45" alt="mark"
电影管理
- 模型: Movie
- 表单: MovieForm
- 请求方法: GET POST
- 访问控制: @admin_login_req
创建添加电影的表单
class MovieForm(FlaskForm):
title = StringField(
label="片名",
validators=[
DataRequired("片名不能为空!")
],
description="片名",
render_kw={
"class": "form-control",
"id": "input_title",
"placeholder": "请输入片名!"
}
)
url = FileField(
label="文件",
validators=[
DataRequired("请上传文件!")
],
description="文件",
)
info = TextAreaField(
label="简介",
validators=[
DataRequired("简介不能为空!")
],
description="简介",
render_kw={
"class": "form-control",
"rows": 10
}
)
logo = FileField(
label="封面",
validators=[
DataRequired("请上传封面!")
],
description="封面",
)
star = SelectField(
label="星级",
validators=[
DataRequired("请选择星级!")
],
# star的数据类型
coerce=int,
choices=[(1, "1星"), (2, "2星"), (3, "3星"), (4, "4星"), (5, "5星")],
description="星级",
render_kw={
"class": "form-control",
}
)
# 标签要在数据库中查询已存在的标签
tag_id = SelectField(
label="标签",
validators=[
DataRequired("请选择标签!")
],
coerce=int,
# 通过列表生成器生成列表
choices=[(v.id, v.name) for v in Tag.query.all()],
description="标签",
render_kw={
"class": "form-control",
}
)
area = StringField(
label="地区",
validators=[
DataRequired("请输入地区!")
],
description="地区",
render_kw={
"class": "form-control",
"placeholder": "请输入地区!"
}
)
length = StringField(
label="片长",
validators=[
DataRequired("片长不能为空!")
],
description="片长",
render_kw={
"class": "form-control",
"placeholder": "请输入片长!"
}
)
release_time = StringField(
label="上映时间",
validators=[
DataRequired("上映时间不能为空!")
],
description="上映时间",
render_kw={
"class": "form-control",
"placeholder": "请选择上映时间!",
"id": "input_release_time"
}
)
submit = SubmitField(
'添加',
render_kw={
"class": "btn btn-primary",
}
)
将movie & movieforms 导入到views里面
def movie_add():
"""
添加电影页面
"""
form = MovieForm()
return render_template("admin/movie_add.html",form=form)
下面展示在html中怎么用这个。
data:image/s3,"s3://crabby-images/353bb/353bbffacf56eb6f3c8f53afaf23bc6ec213c565" alt="mark"
找猫画虎同理可得的将这些修改完。
添加csrf_token,以及为form添加method = post
因为这里我们需要上传的是文件:
enctype = "multipart/form-data"
书写逻辑验证
- 添加错误信息:
data:image/s3,"s3://crabby-images/48bb6/48bb6f4d07eab1c76b99da29f874deb69ac87cce" alt="mark"
照猫画虎自行为所有字段添加错误信息
Method Not Allowed
The method is not allowed for the requested URL.
@admin.route("/movie/add/", methods=["GET", "POST"])
- 进行上传操作
首先打开init.py定义一个文件上传的路径。
app.config["UP_DIR"] = os.path.join(os.path.abspath(os.path.dirname(__file__)), "static/uploads/")
获取上传的数据,
from app import db, app
from werkzeug.utils import secure_filename
# 修改文件名称
def change_filename(filename):
fileinfo = os.path.splitext(filename)
filename = datetime.now().strftime("%Y%m%d%H%M%S") + str(uuid.uuid4().hex) + fileinfo[-1]
return filename
@admin.route("/movie/add/", methods=["GET", "POST"])
@admin_login_req
def movie_add():
"""
添加电影页面
"""
form = MovieForm()
if form.validate_on_submit():
data = form.data
file_url = secure_filename(form.url.data.filename)
file_logo = secure_filename(form.logo.data.filename)
if not os.path.exists(app.config["UP_DIR"]):
# 创建一个多级目录
os.makedirs(app.config["UP_DIR"])
os.chmod(app.config["UP_DIR"], "rw")
url = change_filename(file_url)
logo = change_filename(file_logo)
# 保存
form.url.data.save(app.config["UP_DIR"]+url)
form.logo.data.save(app.config["UP_DIR"] + logo)
# url,logo为上传视频,图片之后获取到的地址
movie = Movie(
title= data["title"],
url = url,
info = data["info"],
logo = logo,
star= data["star"],
playnum=0,
commentnum=0,
tag_id=data["tag_id"],
area=data["area"],
release_time=data["release_time"],
length=data["length"]
)
db.session.add(movie)
db.session.commit()
flash("添加电影成功!", "ok")
return redirect(url_for('admin.movie_add'))
return render_template("admin/movie_add.html", form=form)
添加消息闪现。
data:image/s3,"s3://crabby-images/1061b/1061bed5e55e49c91538df6f62d0b8086864bf8d" alt="mark"
builtins.AttributeError
AttributeError: module 'app' has no attribute 'config'
说明引入的app不是我们需要的app
import app
# 改为
from app import app
报错:
builtins.TypeError
TypeError: an integer is required (got type str)
star= int(data["star"]),
tag_id=int(data["tag_id"]),
检查是否为整型然后提交。
data:image/s3,"s3://crabby-images/5effc/5effc8fb2be6763e6e2238ded14b694f331d3e81" alt="mark"
网友评论