我试图了解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视图。我不懂为什么。有人可以告诉我为什么吗?在浏览器中,页面看起来和工作正常。但是,我希望该视图成为该应用程序中的视图,而该视图仅使用一些通用模板。
答案 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路径)获取页面对象。
通常在django-cms项目中,诸如video_uploader
和user_accounts
之类的应用程序通过apphooks
(docs)连接到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 %}