7- Flask构建弹幕微电影网站-网站页面搭建
已上线演示地址: http://movie.mtianyan.cn
项目源码地址:https://github.com/mtianyan/movie_project
前端搭建
实现前台后台html布局页面搭建
前台页面源码链接:
https://github.com/mtianyan/movie_project_html
-
将整个前台页面源码放入我们的staic/tpl目录下。
-
打开tpl/2-movie/index.html.
点击右上角在浏览器里预览网页。可以看到首页。
data:image/s3,"s3://crabby-images/96086/96086bac195481ef03731f0878ef3e19c1f09cf7" alt=""
网页整体分为:
- 幻灯片banner展示
- 标签的筛选
- 卡片的电影展览
- 分页效果
- 顶部的导航
- 底部的版权信息。
- 搜索
- 搜索结果的分页。点击播放
- 播放界面:电影介绍,评论。评论列表分页
- 会员登录注册界面
- 会员中心:修改资料,密码。评论记录修改
- 收藏电影。
前台布局搭建:
- 静态文件的引入
{{ url_for('static',filename='文件路径')}}
- 定义路由:
{{ url_for('模块名.视图名',变量=参数)}}
- 定义数据块:
{%block 数据块名称%} .... {% endblock %}
- 首先在templated/home目录下创建home.html
网站共用的部分是顶部和底部。
tpl/2-movie/nav.html就是我们需要的导航和底部。
将10-13行link标签修改为静态文件
<link rel="shortcut icon" href="{{ url_for('static',filename='base/images/logo.png') }}">
<link rel="stylesheet" href="{{ url_for('static',filename='base/css/bootstrap.min.css') }}">
<link rel="stylesheet" href="{{ url_for('static',filename='base/css/bootstrap-movie.css') }}">
<link rel="stylesheet" href="{{ url_for('static',filename='base/css/animate.css') }}">
<img src="{{ url_for('static',filename='base/images/logo.png') }}" style="height:30px;">
<script src="{{ url_for('static',filename='base/js/jquery.min.js') }}"></script>
<script src="{{ url_for('static',filename='base/js/bootstrap.min.js') }}"></script>
<script src="{{ url_for('static',filename='base/js/jquery.singlePageNav.min.js') }}"></script>
<script src="{{ url_for('static',filename='base/js/wow.min.js') }}"></script>
<script src="{{ url_for('static',filename='lazyload/jquery.lazyload.min.js') }}"></script>
将原本的引用外部的css/图片/js的部分。改写为static静态目录下文件名。
filename 应该是staic之后的相对地址。
在如下的内容部分定义content块。
<!--内容-->
<div class="container" style="margin-top:76px">
{% block content %}{% endblock %}
</div>
<!--内容-->
打开home模块的视图处理器(views.py):
- 定义我们的路由。
home/view,py
from . import home
from flask import render_template
@home.route("/")
def index():
return render_template("home/index.html")
此时我们该去创建我们的index.html.(home/index.html)
{% extends "home.html" %}
{% block %}
<h1>helloworld</h1>
{% endblock %}
继承home.html,然后写一个块。
打开入口脚本。运行manage.py
data:image/s3,"s3://crabby-images/22e59/22e59fe9faa50785822d938d891babd1e6d62099" alt=""
改为:继承home/重写content块
{% extends "home.html" %}
{% block content %}
<h1>helloworld</h1>
{% endblock %}
报错:
jinja2.exceptions.TemplateNotFound
jinja2.exceptions.TemplateNotFound: home.html
因为我们的路径有问题:
{% extends "home/home.html" %}
{% block content %}
<h1>helloworld</h1>
{% endblock %}
将static/tpl/static/base剪切到static/
data:image/s3,"s3://crabby-images/04a2b/04a2bd507514bb80b2f6158d7706d57b8a4d111b" alt=""
此时再去运行manage.py
访问我们可以看到首页的效果
data:image/s3,"s3://crabby-images/88c58/88c58faadeeb5ed7ea8202c291972d3a4707ee9c" alt=""
会员登录页面的搭建
需要掌握的知识点:
#登录
@home.route("/login/")
def login():
return render_template("home/login.html")
#退出
@home.route("/logout/")
def logout():
return redirect(url_for('home.login'))
上述登录和退出分别采用return渲染的模板以及直接重定向。
编码
app\home\views.py:
添加代码
from flask import render_template,url_for,redirect
#登录
@home.route("/login/")
def login():
return render_template("home/login.html")
#退出
@home.route("/logout/")
def logout():
return redirect(url_for('home.login'))
#重定向到home模块下的登录。
- 在app\templates\home下新建login.html并将app/static/tpl/2-movie/login.html中内容注释的部分复制出来贴进去。
data:image/s3,"s3://crabby-images/27867/27867bb10bfbf858afdbf11081d9da92be0426b5" alt=""
- 修改app/templates/home/home.html的登录退出按钮的href
<a class="curlink" href="{{url_for('home.index')}}"><span class="glyphicon glyphicon-film"></span> 电影</a>
<a class="curlink" href="{{ url_for('home.login')}}"><span class="glyphicon glyphicon-log-in"></span> 登录</a>
<a class="curlink" href="{{ url_for('home.logout')}}"><span class="glyphicon glyphicon-log-out"></span> 退出</a>
注意:传入的是一个'home.login'
不要把单引号漏了
修改完成后进入manage.py点击run。访问
可以看到登录界面已经好了http://127.0.0.1:5000/login/
data:image/s3,"s3://crabby-images/3e013/3e01364cd9bae2d63ac7b4033677c3f880ded115" alt=""
点击登录和点击退出。也都能正确的跳转到登录。
会员注册页面的搭建
编码
定义home/views.py:
# 会员注册
@home.route("/register/")
def regist():
return render_template("home/register.html")
- 在app\templates\home下新建register.html并将app/static/tpl/2-movie/register.html中注释表明为内容的部分复制出来贴进去。
这里笔者的强迫症犯了,请自行将所有的regist 改为register写全。
此时我们点击电影helloworld,点击登录,注册,分布展示对应界面。点击退出,会跳转到登录。
此处为笔者上次前去考研中断项目的地址
对应提交commit: 项目中断重启初始化: 首页路由,登录,注册已完成。
会员中心界面搭建
基础语句概要
# 会员中心
@home.route("/user/")
# 修改密码
@home.route("/pwd/")
# 评论记录
@home.route("/comments/")
# 登录日志
@home.route("/loginlog/")
# 收藏
@home.route("/moviecol/")
实际编码与模板创建
@home.route("/user/")
def user():
"""
用户中心
"""
return render_template("home/user.html")
@home.route("/pwd/")
def pwd():
"""
修改密码
"""
return render_template("home/pwd.html")
@home.route("/comments/")
def comments():
"""
评论记录
"""
return render_template("home/comments.html")
@home.route("/loginlog/")
def loginlog():
"""
登录日志
"""
return render_template("home/loginlog.html")
@home.route("/moviecol/")
def moviecol():
"""
收藏电影
"""
return render_template("home/moviecol.html")
然后分别创建对应的template
data:image/s3,"s3://crabby-images/e8a70/e8a7032d3be94efdc320bbfa50d04b7e2804f056" alt=""
定义会员中心界面。
data:image/s3,"s3://crabby-images/0d2cf/0d2cfcd87c26069821317f5e39b6d889137b88dc" alt=""
可以看到整个微电影用户中心的左侧是一个菜单。这个菜单是用户中心的这些功能共用的。
新建menu.html
将app/static/tpl/2-movie/user.html中的菜单拷贝过来。
data:image/s3,"s3://crabby-images/17230/17230453590b43b779d191b301367e6acd74d30a" alt=""
修改我们的url链接
data:image/s3,"s3://crabby-images/8b99c/8b99cb61c0258b4cb08c15bcf4b5fc57f4f5839d" alt=""
编写user.html
data:image/s3,"s3://crabby-images/7355d/7355d08aa1d35d9151bdbe2801e77fbfa0bbb00c" alt=""
其中的contaier部分为app/static/tpl/2-movie/user.html中的内容部分
可能报错(已解决):
AssertionError: View function mapping is overwriting an existing endpoint function: home.pwd
说明视图中定义了两个重名的函数。
data:image/s3,"s3://crabby-images/af3a8/af3a869f5c7b40947a8c42d7b5951b20960356f4" alt=""
包含menu页面。
此时home.html中通往会员的url还没有打通,前往修改。
data:image/s3,"s3://crabby-images/e0e69/e0e69951edcf72d0b8212084f542ff8f59215309" alt=""
可以看到user中还有和我们的home页面之余的自己独特的东西
data:image/s3,"s3://crabby-images/82796/827965af1ffece371f25ca5abc6fa7969c271349" alt=""
将home中的style合并一下。
data:image/s3,"s3://crabby-images/6f086/6f0861d40f9e36011b5f2de5a909de5e8102b5f7" alt=""
在home.html中定义一个css 数据块,用于其儿子们重写该数据块。
data:image/s3,"s3://crabby-images/698e1/698e1216c419d4b816209e121ee8a5f7ee4cafef" alt=""
前往user.html中重写
将app/static/tpl/2-movie/user.html中的style部分拷贝过来
data:image/s3,"s3://crabby-images/13506/13506e446a60477ba47b225d12b75c886a6044b3" alt=""
data:image/s3,"s3://crabby-images/2b765/2b7657ed22f8d5cf443e09682fb4960921702c97" alt=""
将user收缩至如图结构,覆盖到pwd comments loginlog moviecol
将其中的col-md-9 分别用/tpl/2-movie/中的这部分代替。
将每个页面内的style覆盖css数据块
找猫画虎,自行完成环节。
实现菜单激活某一个具体项
为menu中的各项添加id
data:image/s3,"s3://crabby-images/6f358/6f358613de1976ca354fe82c60dc40d561f5ea65" alt=""
在home.html中添加数据块block js,因为我们想在menu中使用jQuery动态添加class
data:image/s3,"s3://crabby-images/ef550/ef55076a26309825c4dccee61b8bc394b3c0149b" alt=""
user.html中重新改block
data:image/s3,"s3://crabby-images/834c9/834c9a94ac5cab18c7d121a2ea1f77284486b651" alt=""
然后左眼看着图中顺序,右手去其他文件中分别m2,m3
data:image/s3,"s3://crabby-images/90773/90773d0eedf4520b37c28503c2b8590b706a7eb1" alt=""
此时并没有实现激活样式。控制台查看发现是lazyload没加载
将tpl中 lazyload拷贝到static
data:image/s3,"s3://crabby-images/e9240/e9240f7c750fde9c39afb78d117f5337aa3ce904" alt=""
出现两个激活,是因为忘记删除默认的user active 前往menu中删除即可。
点击登录注册暂时设置为跳转会员中心。
将user.html替换为
data:image/s3,"s3://crabby-images/751d4/751d4621522bd67b97138a78aad323a509aa142f" alt=""
提交commit:前台: 会员中心搭建
提交的代码有一些问题:
敲重点: jinja2 urlfor中括号里面必须单引号包裹。不能直接填home.user
jinja2.exceptions.UndefinedError
jinja2.exceptions.UndefinedError: 'home' is undefined
电影列表页面搭建
# 列表
@home.route("/")
def index():
return render_template("home/index.html")
# 动画
@home.route("/animation/")
def animation():
return render_template("home/animation.html")
新建layout.html 复制home.html 到 layout
将内容部分。
data:image/s3,"s3://crabby-images/86fef/86fef2d96ed153c0a9ee5005bba2a7aa91618c1d" alt=""
只保留content
打开index.html继承layout
data:image/s3,"s3://crabby-images/56803/568034bb7e46f02e87fc23710b38b8b2ebedb651" alt=""
将tpl/index.html中的电影列表部分粘贴到content当中。
将html收至如图形状
data:image/s3,"s3://crabby-images/77d85/77d85b75c6c10fa54381c028462f5ad268eea582" alt=""
只保留前四个col-md-3,其余先删除了。
将热门电影也拷过来。可以看到热门电影指向一个iframe。animation
将tpl/animation的代码复制到home/animation(新建)中
将其配套的静态文件,在static目录下新建一个anim目录存放。
data:image/s3,"s3://crabby-images/4c4a2/4c4a207a1e545d8326ca5f24b050781ad4b15cce" alt=""
将animation.html中的静态文件自行进行修改。
data:image/s3,"s3://crabby-images/570bb/570bb45bef30c1bc634c1ddf84a29dc03b299609" alt=""
配置首页指向animation的url
只保留一个col-md-3使用for循环创建12个。
data:image/s3,"s3://crabby-images/d7432/d7432e31c09158be6166b2769ee4744b8db46c2a" alt=""
这时运行项目,可能遇到的报错。
- 静态文件的格式,单引号没有成对添加
- 图片不显示,是忘记添加.jpg后缀
列表的holder.js没有显示
data:image/s3,"s3://crabby-images/47e45/47e45b9c092796f9c90cb7c02fc223aac1f2d609" alt=""
此时在layout中添加对于holder.js的支持。
data:image/s3,"s3://crabby-images/cd934/cd93432b497d8afce8aaa787e175994963b4ed89" alt=""
电影搜索页面搭建
@home.route("/search/")
def search():
"""
搜索
"""
return render_template("home/search.html")
创建search.html,内容如下
data:image/s3,"s3://crabby-images/82fcd/82fcdbf928684ee2896ae56593060608812714b8" alt=""
因为home.html的content外围已经有一个container div了,所以将tpl/search.html中的row部分拷进来。
data:image/s3,"s3://crabby-images/8fd61/8fd61b69651f820f7d535e21cb2be7f7b3d04e11" alt=""
data:image/s3,"s3://crabby-images/b13e6/b13e60f990725eb0c35a7c8bd4fc08e5fd11b2cf" alt=""
将div media 删除到剩下一个。使用for进行填充。
data:image/s3,"s3://crabby-images/a3aef/a3aef3dc713a651db64d4bfca6eda0c6fb2c73f1" alt=""
前往home模板中为搜索添加url
data:image/s3,"s3://crabby-images/7f44f/7f44f0204bcf495d84ecaf7b32e7c2e9b4906376" alt=""
同时修改layout和home两个html中的search url 链接
如果发现holder.js没有显示图片站位,前去添加图片占位
data:image/s3,"s3://crabby-images/4d8c8/4d8c8f72ff85738a953bb8acf71c68ea3c47e45d" alt=""
电影详情页面搭建
@home.route("/play/")
def play():
"""
播放
"""
return render_template("home/play.html")
新建play.html
data:image/s3,"s3://crabby-images/792f9/792f92eb5aea6f50660f2f9854192727b63c645d" alt=""
跟搜索差不多照猫画虎做出来。
将头部中的play页面独有的css内容拷贝过来,用css数据块包裹
data:image/s3,"s3://crabby-images/dfc87/dfc87ce3f5ceb44f51111c5a204890a4226ec451" alt=""
将带有播放页面的静态资源都分布放入css 和 js 块中。
将url路径改好为url_for格式,此时可以发现我们的那些static下并没有这些。
将tpl中静态资源复制过去。
报错:
jinja2.exceptions.TemplateSyntaxError
jinja2.exceptions.TemplateSyntaxError: expected token ',', got 'static'
一般原因都是urlfor标签未正常
将home中的.html的跳转全部改成url for
将index中的播放同理修改
data:image/s3,"s3://crabby-images/f14d8/f14d801253587738e9abcbe38aa989c3cbc752a2" alt=""
搜索中的电影播放与收藏中的电影播放。
收藏中电影项使用for range 大法处理
404页面搭建
@app.errorhandler(404)
def page_not_found(error):
"""
404
"""
return render_template("home/404.html"), 404
data:image/s3,"s3://crabby-images/f0c67/f0c6706fab306da97c8b9f51ca5d52e97b09930f" alt=""
新建404页面。将tpl下404复制过来
将static文件全部自行修改。
输入错误地址依然提示not found 是因为我们依然打开的是debug模式。
以上是我在django中的理解。flask中debug模式下仍然可以404
修改404中 index的url链接。
对应的commit提交:
第五章完结