烧瓶管理员记住表格值

时间:2019-07-17 07:53:00

标签: python flask flask-admin

在我的应用程序中,我将“用户”和“帖子”作为模型。每个帖子都有一个用户名外键。当我在帖子模型的顶部创建ModelView时,可以在管理界面中以特定用户的身份创建帖子 如下面的屏幕截图所示

enter image description here

添加帖子并单击“保存并添加另一个”后,“用户”将恢复为“ user1”。如何使表单记住先前的值“ user2”?

我的研究使我相信可以通过修改on_model_change和on_form_prefill并在flask会话中保存先前的值来完成,但这似乎过度设计了这么简单的任务。一定有更简单的方法。

我的代码可以在下面看到

from flask import Flask
from flask_sqlalchemy import SQLAlchemy
import flask_admin
from flask_admin.contrib import sqla

app = Flask(__name__)

db = SQLAlchemy()

admin = flask_admin.Admin(name='Test')


class Users(db.Model):
    """
    Contains users of the database
    """
    user_id = db.Column(db.Integer, primary_key=True)
    username = db.Column(db.String(64), index=True, unique=True, nullable=False)

    def __str__(self):
        return self.username


class Posts(db.Model):
    """
    Contains users of the database
    """
    post_id = db.Column(db.Integer, primary_key=True)
    username = db.Column(db.String(11), db.ForeignKey(Users.username), nullable=False)
    post = db.Column(db.String(256))
    user = db.relation(Users, backref='user')


def build_sample_db():
    db.drop_all()
    db.create_all()
    data = {'user1': 'post1', 'user1': 'post2', 'user2': 'post1'}
    for user, post in data.items():

        u = Users(username=user)
        p = Posts(username=user, post=post)

        db.session.add(u)
        db.session.add(p)
    db.session.commit()

class MyModelView(sqla.ModelView):

    pass


if __name__ == '__main__':

    app.config['SECRET_KEY'] = '123456790'
    app.config['DATABASE_FILE'] = 'sample_db.sqlite'
    app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///database'
    app.config['SQLALCHEMY_ECHO'] = True
    db.init_app(app)

    admin.init_app(app)
    admin.add_view(MyModelView(Posts, db.session))

    with app.app_context():
        build_sample_db()

    # Start app
    app.run(debug=True)

2 个答案:

答案 0 :(得分:4)

我之前遇到过这种情况,并且已经使用2个函数解决了。它非常简单和小巧。

@expose('/edit/', methods=('GET', 'POST'))
def edit_view(self):
    #write your logic to populate the value into html        
    self._template_args["arg_name"] = stored_value
    # in your html find this value to populate it as you need

当用户尝试编辑任何值时,上述函数将使您可以在html中填充值。可以使用存储的值来填充。下面是一个功能,可以帮助您保存上一次编辑的值。

在此class MyModelView(sqla.ModelView):中,您需要添加以下2个功能。

def on_model_change(self, form, model, is_created):
    stored_value = model.user # this is your user name stored
    # get the value of the column from your model and save it 

这是一个两步操作,非常小,不需要很多时间。我现在仅添加了一个框架/伪代码,我将使用实际代码更新此答案,以便您可以重复使用它。

答案 1 :(得分:0)

on_form_prefill不能帮助您解决此问题,因为它仅适用于编辑表单。

  

如lib代码所示-> on_form_prefill =>执行其他操作以   预填写修改表单

命中请求时,将发生以下代码流:

(基类)视图(create_view)-> create_form-> _ create_form_class-> get_create_form-> get_form-> scaffold_form

因此,如果要更改正在加载的创建表单的默认值,我们可以按照上述流程中所述的方法处理该表单来更新它。

因此,我们可以覆盖create_form。

 def create_form(self):
     form = super().create_form()
     # set the user id we want in the form as default.
     form.user_id.default = 'USER_2_ID'
     return form

这是获取表单的默认值。

设置该值,我们将覆盖on_model_change

def on_model_change(self, form, model, is_created):
    # set the USER_ID from this model.user_id
    pass

现在正在遵循通过setter和getter方法共享此数据(USER_ID)的方法,

  1. 我们将其设置为cookie并接受创建请求。
  2. 更新“保存并添加另一个”按钮链接,以在查询字符串中添加user_id。

此数据必须在两个不同的请求之间共享,因此将其存储在应用程序上下文中,g将不起作用。应用程序上下文“不会在请求之间共享”

 def create_form(self):
     form = super().create_form()
     user_id = request.args.get('user_id')
     form.user_id.default = user_id

     # or if you want only one option to be available 
     user_query = self.session.query(User).filter(User.id = user_id).one()
     form.user.query = [user_query]
     return form

如上述答案所述,使用edit_view,create_view也可以用来添加

  

一个简单的选项,但不是一个好的选择,将是在创建时进行查询   这将为您提供最后一个条目(但是这种方法不是最佳的   一个

@expose('/new/', methods=('GET', 'POST'))
def create_view(self):
    model = self.get_model()
    # query the model here to get the latest value of the user_id
    self._template_args['user_id'] = user_id
    return super(YourAdmin, self).details_view()