我有两个单独的基于类的视图,并希望保持其功能,并让两个CBV指向同一个模板(我试图将两个单独的表单放入一个页面)。
更具体地说,我正在尝试子类化/组合this email view和这个password change view,以便我可以让它们指向同一个模板,因为我最终希望两个表单在同一页面上。
我试图通过将它们子类化到我自己的视图中来做到这一点:
class MyEmailUpdateView(LoginRequiredMixin, EmailView):
template_name = 'account/account_settings'
success_url = reverse_lazy('settings')
def form_valid(self, form):
return super(SettingsUpdateView, self).form_valid(form)
class MyPasswordUpdateView(LoginRequiredMixin, PasswordChangeView):
template_name = 'account/account_settings'
success_url = reverse_lazy('settings')
def form_valid(self, form):
return super(SettingsUpdateView, self).form_valid(form)
但我现在发现由于错误,一个接一个地说,除非我手动将它传入(success_url,方法等),否则父类中的任何内容实际上都没有转移到我的自定义类。即便如此,我正在子类化的原始类中的代码也指向其他地方。 那么,当组合这两个视图时,我是否需要将所有原始代码复制到我的自定义子类视图中?
这是实现它的正确方法吗?我如何结合这两个视图?我最终想找到一种方法,在我自己的应用程序中将它们的两个表单放在一个页面上。是否有可能使用library's provided templates更简单的方法来实现这一目标?
我想你可以使用django中的默认类来实现相同的结果。据我所知,我得到了这样的场景,我们有两个django形式,我们需要它在同一个模板中使用,如果这是我们可以使用来自django.contrib.auth.views
的LoginView的场景,它有几个可自定义的选项,就像你可以提供额外的形式像这样
class LoginUserView(LoginView):
authentication_form = LoginForm
extra_context = {"register_form": RegistrationForm}
template_name = 'accounts/index.html'
它将使用get context数据方法来更新您将能够在模板中获得的表单并相应地使用。如果您不希望使用这样的代码,您仍然可以像这样使用它
class MyEmailUpdateView(LoginRequiredMixin, EmailView):
form_class = EmailForm
template_name = 'account/account_settings'
success_url = reverse_lazy('settings')
def get_context_data(self, **kwargs):
context = super(MyEmailUpdateView, self).get_context_data(**kwargs)
context['password_reset_form'] = ResetPasswordForm
return context
def form_valid(self, form):
return super(MyEmailUpdateView, self).form_valid(form)
然后,您可以根据您的要求处理有效的表格。如果您需要任何其他要求,希望有助于取回。
您可以使用Django Multi Form View在视图中组合多种形式
安装后你可以像下面这样使用它:
class MultiFView(MultiFormView):
form_classes = {
'email_form' : AddEmailForm,
'change_password_form' : ChangePasswordForm
}
record_id = None
template_name = 'web/multi.html'
def forms_valid(self, forms):
email = forms['email_form'].save(commit=False)
email.save()
return super(MultiFView, self).forms_valid(forms)
并在模板中:
{{ forms.email_form.as_p }}
{{ forms.change_password_form.as_p }}
这可以通过为视图实现(种类)部分更新方法来解决。
工具:
首先保护您的敏感数据:
sensitive_variables
装饰:
如果代码中的函数(视图或任何常规回调)使用易受包含敏感信息的局部变量,则可以使用sensitive_variables装饰器阻止这些变量的值包含在错误报告中sensitive_post_parameters()
装饰:
如果您的一个视图收到一个HttpRequest对象,其POST参数容易包含敏感信息,您可以使用sensitive_post_parameters装饰器阻止这些参数的值包含在错误报告中之后创建一些代码:
ModelForm
创建一个包含您感兴趣的特定字段的表单。不要包含password
字段,因为我们将以不同的方式添加它以满足我们的需求:
myapp/forms.py
:
class PartialUpdateForm(forms.ModelForm):
new_password = forms.CharField(
required=False, widget=forms.widgets.PasswordInput
)
class Meta:
model = MyUser
fields = ('email', 'other_fields',...)
如果你想保留除exclude
之前的所有其他字段,你可以使用fields
而不是password
:ex。 exclude = ('password',)
UpdateView
来处理喧嚣并覆盖它的form_valid
方法来处理部分更新:
myapp/views.py
:
class MyUpdateView(UpdateView):
template_name = 'account/account_settings'
form_class = PartialUpdateForm
success_url = reverse_lazy('settings')
@sensitive_variables('new_password')
@sensitive_post_parameters('new_password')
def form_valid(self, form):
clean = form.cleaned_data
new_password = clean.get('new_password')
if new_password:
form.instance.password = hash_password(new_password)
return super().form_valid(form)
myapp/urls.py
:
...
url(r'^your/pattern/$', MyUpdateView.as_view()),
这样你就可以处理:
使用此解决方案中的代码:Django: Only update fields that have been changed in UpdateView