我希望有一个动态导航菜单,如果用户当前未登录则显示“登录”,如果用户已登录则显示“登出”。
我正在使用类似于以下内容的代码:
import flask
import flask_nav
import flask_nav.elements as fne
frontend = flask.Blueprint('frontend', __name__)
application = flask.Flask(__name__)
mySess = flask_session.Session()
flask_appconfig.AppConfig(application)
flask_bootstrap.Bootstrap(application)
application.register_blueprint(frontend)
application.config['BOOTSTRAP_SERVE_LOCAL'] = True
application.config['SSL'] = True
application.secret_key = SECRET_KEY
application.config['SESSION_TYPE'] = SESSION_TYPE
mySess.init_app(application)
nav = flask_nav.Nav()
class CustomRenderer(flask_bootstrap.nav.BootstrapRenderer):
def visit_Navbar(self, node):
nav_tag = super(CustomRenderer, self).visit_Navbar(node)
nav_tag['class'] = 'navbar navbar-default navbar-fixed-top'
return nav_tag
flask_nav.register_renderer(application, 'custom', CustomRenderer)
nav.init_app(application)
@nav.navigation()
def top_nav():
items = [ fne.View('Home', '.index') ]
if 'google_token' in flask.session:
items.append(fne.View('Logout', '.logout'))
elif 'auth_url' in flask.session:
items.append(fne.View('Login', flask.session['auth_url']))
else:
items.append(fne.View('Login', '.login'))
items.append(fne.View('About', '.about'))
items.append(fne.View('Contact', '.contact'))
items.append(fne.View('Shop', '.shop'))
items.append(fne.View('Help & Feedback', '.help'))
return fne.Navbar('', *items)
nav.register_element('frontend_top', top_nav())
不幸的是,Flask会话变量超出了nav对象的范围,因此我无法从top_nav中访问flask.session。
当我在我的应用程序之外创建用于访问flask会话的任何独立函数时,我遇到了同样的困难,例如
def user_is_logged_in():
if 'google_token' in flask.session:
return True
else:
return False
return False
这些函数给出了预期的错误“RuntimeError:在请求上下文之外工作。”
出于安全原因,我不想在我的application.py代码中为用户使用全局变量,因此多人可以同时访问应用程序而不会出错。我相信SESSION应该存储用户当前是否登录。
如何让我的flask_nav.Nav()看到我的应用程序的flask.session?
答案 0 :(得分:1)
Flask-Login会让您的生活更轻松。它提供了current_user
指向当前用户,并且用户对象具有is_authenticated
属性:
from flask_login import current_user
...
@nav.navigation()
def top_nav():
...
if current_user.is_authenticated:
items.append(View("Logout", ".logout"))
else:
items.append(View("Login", ".login"))
初始化Flask-Login的代码如下:
from flask import Flask
from flask_login import LoginManager, UserMixin
app = Flask(__name__)
login_manager = LoginManager(app)
# The user model
class User(db.Model, UserMixin):
...
@login_manager.user_loader
def load_user(user_id):
return User.get(user_id)
查看the documentation了解更多详情。
答案 1 :(得分:1)
flask_nav
在应用程序生命周期的某个阶段注册扩展。
当应用程序中存在请求上下文时,您可以覆盖template_global的注册。
排除常见导航项目。
nav = Nav()
# registers the "top" menubar
navitems = [
View('Widgits, Inc.', 'index'),
View('Our Mission', 'about'),
]
设置一个功能,根据会话中的值
返回相应的视图/链接def with_user_session_action(items):
return (
items
+ [ View('Login', 'login') if not session.get('logged') else View('Logout', 'logout')]
)
在委派给nav.register_element的函数中使用它
def register_element(nav, navitems):
navitems = with_user_session_action(navitems)
return nav.register_element('top',
Navbar(*navitems)
)
取代render_template以始终传递计算出的导航
_render_template = render_template
def render_template(*args, **kwargs):
register_element(nav, navitems)
return _render_template(*args, nav=nav.elems, **kwargs)
<强>加成:强>
您可以缓存计算导航以进行登录/注销,这样就不会为每种情况计算一次。