我正在尝试将模型中的某些对象呈现为HTML。我最初以为这很简单(很可能是这样),但是我遇到了许多错误。
在这个项目中,我建立了一些可以正常工作的API视图(HintsListApiView
和HintsRudView
可以正常工作)。但是,理想情况下,我希望使用API生成一个常规的-HTMLAPIView
样式的只读HTML页面,然后可以根据需要设置样式。我正在尝试使用TemplateHTMLRenderer
,但遇到了错误。我想要的只是让text属性显示在HTML中。实际文本各只有一个句子。
这些是我的文件:
models.py:
from django.db import models
from rest_framework.reverse import reverse as api_reverse
class Hints(models.Model):
text = models.TextField(max_length=255)
author = models.CharField(max_length=20)
timestamp = models.DateTimeField(auto_now_add=True)
def __str__(self):
return str(self.text)
def timestamp_pretty(self):
return self.timestamp.strftime('%b %d %Y')
def get_api_url(self, request=None):
return api_reverse("api-hints1:hints-rud", kwargs={'pk': self.pk}, request=request)
views.py:
class HTMLAPIView(generics.RetrieveAPIView):
lookup_field = 'pk'
serializer_class = HTMLSerializer
renderer_classes = (TemplateHTMLRenderer,)
def get_queryset(self):
queryset = Hints.objects.value('text')
return Response({'queryset': queryset}, template_name='base.html')
class HintsListApiView(mixins.CreateModelMixin, generics.ListAPIView):
lookup_field = 'pk'
serializer_class = HintsSerializer
def get_queryset(self):
qs = Hints.objects.all()
query = self.request.GET.get("q")
if query is not None:
qs = qs.filter(
Q(text__icontains=query)|
Q(author__icontains=query)
).distinct()
return qs
def post(self, request, *args, **kwargs):
return self.create(request, *args, **kwargs)
def get_serializer_context(self, *args, **kwargs):
return {"request": self.request}
class HintsRudView(generics.RetrieveUpdateDestroyAPIView):
lookup_field = 'pk'
serializer_class = HintsSerializer
def get_queryset(self):
return Hints.objects.all()
def get_serializer_context(self, *args, **kwargs):
return {"request": self.request}
serializers.py:
class HintsSerializer(serializers.ModelSerializer):
url = serializers.SerializerMethodField(read_only=True)
class Meta:
model = Hints
fields = [
'url',
'id',
'text',
'author',
'timestamp_pretty'
]
read_only_fields = ['timestamp_pretty', 'id']
def get_url(self, obj):
request = self.context.get("request")
return obj.get_api_url(request=request)
class HTMLSerializer(serializers.ModelSerializer):
class Meta:
model = Hints
fields = [
'text',
]
read_only_fields = ['text',]
root urls.py:
from django.contrib import admin
from django.conf.urls import url, include
from rest_framework import routers, serializers, viewsets
urlpatterns = [
url('admin/', admin.site.urls),
url(r'^api/hints/', include(('hints1.api.urls', 'api'), namespace='api-hints1')),
url(r'^api-auth/', include('rest_framework.urls')),
]
urls.py:
from .views import HintsRudView, HintsListApiView, HTMLAPIView
from . import views
from django.contrib import admin
from django.conf.urls import url, include
from rest_framework import routers, serializers, viewsets
urlpatterns = [
url(r'^(?P<pk>\d+)$', HintsRudView.as_view(), name='hints-rud'),
url(r'^$', HintsListApiView.as_view(), name='hints-list'),
url(r'^html/', HTMLAPIView.as_view(), name='html' )
]
我遇到的错误多种多样,目前我在AttributeError
遇到/api/hints/html/
Manager
对象没有属性“值”。
我尝试过使用和不使用序列化程序(因为在文档中提到TemplateHTMLRenderer
不需要一个)。我认为问题出在view.py
和get_queryset
函数之内。我尝试了各种方法,但会遇到其他错误,例如
TypeError上下文必须是字典,而不是QuerySet。
我们将不胜感激! 谢谢!
答案 0 :(得分:0)
所以我设法做到了自己想要的。但是,老实说,我不确定它为什么能工作,因此,如果有人可以提供说明,请这样做。
我将views.py中的HTMLAPIView更改为Viewset:
class HTMLAPIView(viewsets.ViewSet):
renderer_classes = [TemplateHTMLRenderer]
template_name = 'base.html'
serializer_class = HTMLSerializer
def list(self, request):
queryset = Hints.objects.order_by('pk')
return Response({'queryset': queryset})
然后我的urls.py中出现错误,但能够通过将字典部分包含在.as_view()
url(r'^html/', HTMLAPIView.as_view({'get': 'list'}), name='html' )
我不确定为什么这样做是因为它会根据需要从模型返回我的text
属性,但是我看不到在哪里指定了正确的属性。仅仅是因为它是第一个吗?
我也尝试注释掉序列化器,但它仍然可以正常工作,所以我得出结论,这不是导致结果的原因。