Django和DjangoCMS视图管理

时间:2019-11-09 19:01:30

标签: django django-views django-urls django-cms

我试图了解Django尤其是DjangoCMS如何管理视图。

我在项目的urls.py中具有以下内容:

urlpatterns = [
    path('admin/', admin.site.urls),
    path('video/', include('video_uploader.urls')),
    path('user/', include('user_accounts.urls')),
    path('', include('cms.urls')),
] + static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)

也就是说,有一些特定的路由,其余的由DjangoCMS(cms.urls)处理。

在其中一个应用程序中(对应于上面的/user/路径),我有以下内容:

from django.urls import path

from . import views

app_name = 'user_accounts'
urlpatterns = [
    path('signup', views.user_signup, name='signup'),
]

此路径的视图如下:

def user_signup(request):
   if request.method == 'POST':
       form = UserCreationForm(request.POST)
       print('Is the form valid?')
       print(form.is_valid())
       if form.is_valid():
           user = form.save()
           login(request, user)
           return redirect('/')
   else:
       form = UserCreationForm()
   return render(request, 'user_accounts/signup.html', {'form': form})

到目前为止,太好了。现在我们来看看有趣的地方。

user_accounts / signup.html

{% extends 'base.html' %}

{% block content %}
<div class="container">
    <h2>Sign up</h2>
    <form method="post" novalidate>
    {% csrf_token %}
    {% include 'includes/form.html' %}
    <button type="submit" class="btn btn-primary">Create an account</button>
    </form>
</div>
{% endblock %}

上面的模板扩展了base.html,它是整个项目的基本模板。换句话说,此注册表单将嵌入到页面的内容块中,以使应用程序具有凝聚力。

在测试注册视图时,我尝试执行以下操作:

class SignUpTest(TestCase):

    def setUp(self):
        url = reverse('user_accounts:signup')
        self.response = self.client.get(url)

    def test_signup_status_code(self):
        self.assertEqual(self.response.status_code, 200)

    def test_signup_url_resolves_signup_view(self):
        view = resolve('/user/signup/')
        print(view)
        print(view.func)
        print(resolve('/video/'))
        print(resolve('/video/').func)
        self.assertIs(view.func, user_signup)

问题是测试失败。失败是因为两个功能不同。

> FAIL: test_signup_url_resolves_signup_view
> (user_accounts.tests.SignUpTest)
> Traceback (most recent call last):   File
> "/home/user-name/sites/project-web/project/user_accounts/tests.py",
> line 20, in test_signup_url_resolves_signup_view
>     self.assertIs(view.func, user_signup) AssertionError: <function details at 0x7f0dc78bf050> is not <function user_signup at
> 0x7f0dc9891200>

它们是什么?

这是我得到的照片。

ResolverMatch(func=cms.views.details, args=(), kwargs={'slug': 'user/signup'}, url_name=pages-details-by-slug, app_names=[], namespaces=[])`
`<function details at 0x7f0dc78bf050>
ResolverMatch(func=video_uploader.views.list_videos, args=(), kwargs={}, url_name=list_videos, app_names=['video_uploader'], namespaces=['video_uploader'])`
`<function list_videos at 0x7f0dc8ecf950>

我正在使用/video/进行打印,因为该应用程序与该应用程序的设置几乎相同。主要区别在于前者尚未扩展base.html模板。看来该视图随后解析为非DjangoCMS视图。

如您所见,访问user/signup/时实际使用的视图是DjangoCMS视图。我不懂为什么。有人可以告诉我为什么吗?在浏览器中,页面看起来和工作正常。但是,我希望该视图成为该应用程序中的视图,而该视图仅使用一些通用模板。

1 个答案:

答案 0 :(得分:1)

CMS网址从此处开始; https://github.com/divio/django-cms/blob/develop/cms/urls.py

最重要的位是details视图; https://github.com/divio/django-cms/blob/develop/cms/views.py#L38

除了一些实用程序功能(here)之外,CMS实质上还尝试从提供的slug(URL路径)获取页面对象。

通常在项目中,诸如video_uploaderuser_accounts之类的应用程序通过apphooksdocs)连接到CMS页面。

您显示的是您的应用程序模板扩展了一个模板,我认为该模板已设置为CMS模板base.html。基本模板是否包括CMS工具栏?我怀疑您是通过扩展模板引起冲突的,因为您曾说过另一个自定义视图不会扩展同一模板。

使用自己的URL将自己的应用程序硬编码到项目中并且不使用CMS apphooks并没有什么错,但是如果这样做,请确保所有内容都隔离起来,确保模板不会插入系统的CMS端。

要回答有关网页链接的问题,可以在模板中使用一些变量分配。例如,我有一个不能保证已安装的图库应用程序;

{% page_url "gallery" as gallery_url %}

这可能不返回任何值,因此可以将其设置为不会执行任何操作的href,或者您可以在条件块中使用它;

{% if gallery_url %}
    <a href="{{ gallery_url }}">
        Gallery
    </a>
{% endif %}