Django reportlab插入分页符html端

时间:2013-03-09 03:23:48

标签: django pdf reportlab

我正在尝试在使用reportlab生成的pdf中的表后插入分页符,我使用以下函数生成pdf:

def render_to_pdf(template_src, context_dict):
  template = get_template(template_src)
  context = Context(context_dict)
  html  = template.render(context)
  result = StringIO.StringIO()
  pdf = pisa.pisaDocument(StringIO.StringIO(html.encode("ISO-8859-1")), dest=result, link_callback=fetch_resources)
  if not pdf.err:
    return result.getvalue()
  return HttpResponse('We had some errors<pre>%s</pre>' % escape(html))
def fetch_resources(uri, rel):
    return os.path.join(MEDIA_ROOT, uri.replace(MEDIA_URL, ""))

我从这种方式调用函数:

@login_required(login_url=reverse('accounts:login_box'))
def quote_pdf(request, quote_id):
    data_pdf = {}
    quote = get_object_or_404(Quote, id=quote_id)
    data_pdf['pagesize'] = 'letter'
    data_pdf['quote'] = quote
    pdf = render_to_pdf('rents/quote_pdf.html', data_pdf)
    return HttpResponse(pdf, mimetype='application/pdf')

这是我的模板(quote_pdf.html)

{% load humanize compress verbatim %}

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
        "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
    <title>Cotización No. {{ quote.id }} Flasherz.co/alquiler</title>
    <style type="text/css">
        @page {
            size: {{ pagesize }};
            margin: 1cm;
            @frame footer {
                -pdf-frame-content: footerContent;
                bottom: 0cm;
                margin-left: 0cm;
                margin-right: 0cm;
                height: 0cm;
            }
        }
    </style>
    <style type="text/css">
        body {
            font-size: 12px;
            line-height: 13px;
        }
        div#pdf-header {
            display: block;
        }
        div#pdf-header h2 {
            display: block;
            text-align: center;
            font-weight: normal;
            line-height: 35px;
            font-size: 18px;
        }
        div#pdf-quote-info {
            display: block;
            margin-bottom: 20px;
        }
        p {
            margin: 0;
        }
        table {
            border-top: 1px solid #eee;
        }
        table td, table th {
            border-right: 1px solid #eee;
            border-bottom: 1px solid #eee;
            padding: 5px;
            border-left: 1px solid #eee;
        }
        table .price {
            text-align: right;
        }
        table th {
            padding: 5px 0;
            font-size: 13px;
            text-transform: uppercase;
            color: black;
            text-align: left;
            background-color: #eee;
            padding-left: 5px;
        }
        table p.description {
            color: #555;
            font-size: 11px;
        }
        table#quote-info {
            border: none;
        }
        table#quote-info td, table#quote-info th {
            border: none;
            padding: 0;
        }
        table td.quantity {
            text-align: center;
        }
    </style>
</head>
<body>
<div id="pdf-header">
    <div style="width: 100%; text-align: center;">
        <img src="{{ STATIC_URL }}img/quotes-logo.jpeg" alt="Flasherz.co Alquiler" width="600" height="126" />
        <div style="display: block; text-align: center; margin: 10px 0;">
            <h2>Cotización No. {{ quote.id }}, {{ quote.client_name }}</h2>
        </div>
    </div>
</div>
<div id="pdf-quote-info">
    <table id="quote-info" width="70%" border="none" cellpadding="3" cellspacing="0">
        <tr>
            <td width="25%">Fecha:</td>
            <td width="75%">{{ quote.created|date:'l j' }} de {{ quote.created|date:'F' }} de {{ quote.created|date:'Y' }}</td>
        </tr>
        {% if quote.client_name %}
            <tr>
                <td>Cliente:</td>
                <td>{{ quote.client_name }}</td>
            </tr>
        {% endif %}
        {% if quote.client_email %}
            <tr>
                <td>Correo:</td>
                <td>{{ quote.client_email }}</td>
            </tr>
        {% endif %}
        {% if quote.client_address %}
            <tr>
                <td>Dirección:</td>
                <td>{{ quote.client_address }}</td>
            </tr>
        {% endif %}
        {% if quote.client_phone %}
            <tr>
                <td>Teléfono:</td>
                <td>{{ quote.client_phone }}</td>
            </tr>
        {% endif %}
        <tr>
            <td>Cantidad de días:</td>
            <td>{{ quote.rental_days }}</td>
        </tr>
    </table>
    </div>
</div>
<div id="pdf-quote-table">
    <table id="quote-table" cellpadding="0" cellspacing="0" width="100%">
        {% for category in quote.categories.all %}
            <tbody>
            <tr>
                <th colspan="4" class="category-header">{{ category.category.name }}</th>
            </tr>
            {% for item in category.items.all %}
                <tr>
                    <td class="quantity" width="10%">
                        <p>{{ item.quantity }}</p>
                    </td>
                    <td class="name" width="50%">
                        <p class="name">{{ item.name }}</p>
                        {% if item.content %}<p class="description">{{ item.content }}</p>{% endif %}
                    </td>
                    <td class="price" width="20%">
                        <p>${{ item.price|intcomma }}</p>
                    </td>
                    <td class="price total-price" width="20%">
                        <p>${{ item.total_price|intcomma }}</p>
                    </td>
                </tr>
            {% endfor %}
            </tbody>
        {% endfor %}
        <tbody id="others">
        <tr>
            <th colspan="4" class="category-header">Seguro del 10%</th>
        </tr>
        <tr>
            <td class="quantity"></td>
            <td class="name">Seguro 10%</td>
            <td class="price"></td>
            <td class="price total-price">
                <p>${{ quote.get_insurance_price|intcomma }}</p>
            </td>
        </tr>
        </tbody>
        {% if quote.discount %}
            <tbody id="others">
            <tr>
                <th colspan="4" class="category-header">Descuento</th>
            </tr>
            <tr>
                <td class="quantity">
                </td>
                <td class="name">Descuento</td>
                <td class="price"></td>
                <td class="price total-price">
                    <p>${{ quote.discount|intcomma }}</p>
                </td>
            </tr>
            </tbody>
        {% endif %}
        <tbody id="totals">
        <tr>
            <th colspan="4" class="category-header">Total</th>
        </tr>
        <tr>
            <td class="fake" colspan="2"></td>
            <td class="name">Subtotal</td>
            <td class="price">
                <p>${{ quote.get_subtotal|intcomma }}</p>
            </td>
        </tr>
        <tr>
            <td class="fake" colspan="2"></td>
            <td class="name">Total días</td>
            <td class="price">
                <p>${{ quote.get_total_days|intcomma }}</p>
            </td>
        </tr>
        <tr>
            <td class="fake" colspan="2"></td>
            <td class="name">Total</td>
            <td class="price">
                <p><strong>${{ quote.get_total|intcomma }}</strong></p>
            </td>
        </tr>
        </tbody>
    </table>
</div>
</body>
</html>

我不知道在表格之后插入分页符以在下一页插入“条款和条件”的方式。

此外,我对pdf中的图像有问题,渲染时没有人出现。

感谢您的帮助

3 个答案:

答案 0 :(得分:12)

根据我在Pisa's documentation(以及this forum post)中学到的内容,您可以使用page-break-afterpage-break-before等标准CSS代码。比萨还有一些自己的标签,比如pdf:nextpage。要使用它,只需在您希望分页的地方插入以下代码块(div是必需的,因此Pisa的HTML5解析器不会尝试解释它们):

<div>
    <pdf:nextpage /> 
</div> 

答案 1 :(得分:9)

我找到了正确的方法,我需要在我的pdf中插入一些特定的样式

<style type="text/css">
    @page {
        size: {{ pagesize }};
        margin: 1cm;
        @frame footer {
            -pdf-frame-content: footerContent;
            bottom: 0cm;
            margin-left: 0cm;
            margin-right: 0cm;
            height: 0cm;
        }
    }
    .page-break{
        page-break-after: always;
    }
</style>

然后将一个带有.page-break类的元素放在你希望在pdf中有分页符的位置。

答案 2 :(得分:1)

我对css方法没有运气。

只需在您希望分页的地方的template.html中插入这段代码即可。这个对我有用。我不确定,但我认为它必须在div中。

<div> 
   <pdf:nextpage /> 
</div>