我试图从django中的html模板生成pdf。因此,我几乎使用了在网上找到的基本方法。问题是,我可以让它生成模板的内容,但不是我的媒体目录中的图像。我总是得到以下错误:
在/manager/createpdf/
连接路径位于基本路径组件之外
既然我可以得到一些结果,我假设我的观点没有问题。这是render_to_pdf
def render_to_pdf(template_src, context_dict):
template = get_template(template_src)
html = template.render(context_dict)
result = BytesIO()
pdf = pisa.pisaDocument(BytesIO(html.encode('UTF-8')), result, link_callback=link_callback)
if not pdf.err:
return HttpResponse(result.getvalue(), content_type='application/pdf')
return None
和link_callback:
def link_callback(uri, rel):
result = finders.find(uri)
if result:
if not isinstance(result, (list, tuple)):
result = [result]
result = list(os.path.realpath(path) for path in result)
path=result[0]
else:
sUrl = settings.STATIC_URL
sRoot = settings.STATIC_ROOT
mUrl = settings.MEDIA_URL
mRoot = settings.MEDIA_ROOT
if uri.startswith(mUrl):
path = os.path.join(mRoot, uri.replace(mUrl, ""))
elif uri.startswith(sUrl):
path = os.path.join(sRoot, uri.replace(sUrl, ""))
else:
return uri
# make sure that file exists
if not os.path.isfile(path):
raise Exception( 'media URI must start with %s or %s' % (sUrl, mUrl))
return path
我很确定link_callback不做它的目的。但我的知识太少了,无法弥补。我还假设我正确地配置了媒体目录。我可以在其他视图/模板中访问媒体文件。
帮助是非常感激的,因为我花了几个小时安静在这个问题上…非常感谢所有将在这里做出贡献的人!好的,我找到了!在finder .py中,我检查了find-方法。结果发现find方法只查找静态目录中的文件,而忽略媒体目录。我刚删除了所有的密码,现在能用了。下面是代码:
def link_callback(uri, rel):
sUrl = settings.STATIC_URL
sRoot = settings.STATIC_ROOT
mUrl = settings.MEDIA_URL
mRoot = settings.MEDIA_ROOT
if uri.startswith(mUrl):
path = os.path.join(mRoot, uri.replace(mUrl, ""))
elif uri.startswith(sUrl):
path = os.path.join(sRoot, uri.replace(sUrl, ""))
else:
return uri
if not os.path.isfile(path):
raise Exception( 'media URI must start with %s or %s' % (sUrl, mUrl))
return path
许多开发者都告诉我同样的问题。我将添加一个更复杂的解释。
三言两语
一个快速的答案是,你需要使用绝对路径而不是相对路径(静态函数返回相对)。
<标题>详细回答1。link_callback
函数
link_callback
功能帮助xhtml2pdf
定位和使用静态资产。必须同时处理开发(DEBUG=True
)和生产(DEBUG=False
)环境。
import os
from django.conf import settings
def link_callback(uri, rel):
if os.path.isabs(uri):
return uri
sUrl = settings.STATIC_URL
sRoot = settings.STATIC_ROOT
mUrl = settings.MEDIA_URL
mRoot = settings.MEDIA_ROOT
if uri.startswith(mUrl):
path = os.path.join(mRoot, uri.replace(mUrl, ""))
elif uri.startswith(sUrl):
path = os.path.join(sRoot, uri.replace(sUrl, ""))
else:
return uri
if not os.path.isfile(path):
raise Exception('media URI must start with %s or %s' % (sUrl, mUrl))
return path
2。将HTML呈现为PDF
的函数这个函数将给定的Django HTML模板和上下文转换为PDF响应:
from xhtml2pdf import pisa
from django.template.loader import get_template
from django.http import HttpResponse
import io
def render_to_pdf(template_src, context_dict):
template = get_template(template_src)
html = template.render(context_dict)
response = HttpResponse(content_type='application/pdf')
response['Content-Disposition'] = 'attachment; filename="output.pdf"'
pisa_status = pisa.CreatePDF(html, dest=response, link_callback=link_callback)
if pisa_status.err:
return HttpResponse('We had some errors <pre>' + html + '</pre>')
return response
3。
p>使用render_to_pdf
函数将模板呈现为PDF格式的Django视图:
def generate_pdf(request):
context = {'some_data': 'Hello, World!'}
return render_to_pdf('pdf_template.html', context)
4。HTML模板(pdf_template.html
)
带有静态引用的HTML模板示例:
{% load static %}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>PDF Template</title>
<link rel="stylesheet" href="{% static 'jobs/css/style.css' %}">
</head>
<body>
<h1>Welcome to My PDF</h1>
<p>This is a sample PDF generated using xhtml2pdf.</p>
<img src="{% static 'jobs/images/sample.jpg' %}" alt="Sample Image">
</body>
</html>
5。URL配置
连接generate_pdf
视图到URL:
from django.urls import path
from .views import generate_pdf
urlpatterns = [
path('generate-pdf/', generate_pdf, name='generate_pdf'),
]
<标题>集成步骤:- 将
link_callback
和render_to_pdf
函数放在Django应用中合适的实用程序或视图文件中。 - 创建你想要的HTML模板,嵌入或链接任何必要的静态资产与
{% static %}
标签。 - 设置使用
render_to_pdf
函数生成PDF的视图 - 添加URL路由来触发此视图
- 确保如果
DEBUG
设置为False
,则运行collectstatic
收集STATIC_ROOT
指定位置的所有静态文件。
现在,导航到/generate-pdf/
端点(或您设置的任何URL)将生成基于pdf_template.html
模板的PDF。由于link_callback
函数,静态资产(如CSS和图像)将被正确嵌入。