330 lines
11 KiB
PHP
330 lines
11 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Template for Warehouse Offer PDF Main Content
|
|
*
|
|
* @var \App\Models\WarehouseOfferModel $offer The offer object
|
|
* @var array $entries Grouped positions array [groupKey => [position1, position2,...]]
|
|
* @var float $subTotal Calculated subtotal of all positions
|
|
* @var string $offerNumber Offer number
|
|
* @var int $offerDate Timestamp of offer creation
|
|
* @var int|null $validUntilDate Timestamp of offer validity end, or null
|
|
* @var bool $includeTax Whether to calculate and show VAT
|
|
* @var float $vatRate The VAT rate (e.g., 0.20 for 20%)
|
|
* @var string $offerText Additional text from the offer record
|
|
*
|
|
* // Bank details are also available but typically used in footer
|
|
* @var string $bank_iban
|
|
* @var string $bank_bic
|
|
* @var string $bank_bank
|
|
* @var string $bank_owner
|
|
*/
|
|
|
|
// Set filename for download (optional, can also be set in controller)
|
|
// $this->setReturnValue(['filename' => ($offerNumber ?? $offer->id) . "_Angebot.pdf"]);
|
|
|
|
// --- Text Elements (Simple German Example - Extend for EN like in Order) ---
|
|
$texts = [
|
|
'DE' => [
|
|
'title' => 'Angebot',
|
|
'offerNumberLabel' => 'Angebotsnr.:',
|
|
'offerDateLabel' => 'Datum:',
|
|
'validUntilLabel' => 'Gültig bis:',
|
|
'pageLabel' => 'Seite', // For page numbering in content if needed
|
|
'table' => [
|
|
'pos' => 'Pos',
|
|
'article' => 'Artikel / Beschreibung',
|
|
// 'description' => 'Beschreibung', // Combined with Article
|
|
'amount' => 'Menge',
|
|
'unit' => 'Einheit',
|
|
'unitPrice' => 'Einzelpreis',
|
|
'totalPrice' => 'Gesamtpreis'
|
|
],
|
|
'summary' => [
|
|
'subTotal' => 'Zwischensumme',
|
|
'vatFormatted' => 'zzgl. {VAT_RATE}% MwSt.', // Placeholder for rate
|
|
'total' => 'Gesamtbetrag',
|
|
'currency' => '€' // Currency symbol
|
|
],
|
|
'notes' => 'Anmerkungen:',
|
|
'defaultOfferText' => 'Vielen Dank für Ihre Anfrage. Es gelten unsere Allgemeinen Geschäftsbedingungen.',
|
|
'taxInfoNet' => '(Alle Preise exkl. MwSt.)', // Info if tax is added at the end
|
|
'taxInfoGross' => '(Alle Preise inkl. MwSt.)', // Info if prices already include tax (less common in B2B offers)
|
|
]
|
|
// Add 'EN' => [...] section if needed
|
|
];
|
|
// Simple language selection (default to DE) - enhance if customer language is known
|
|
$lang = 'DE';
|
|
$text = $texts[$lang];
|
|
$currencySymbol = $text['summary']['currency'];
|
|
|
|
// --- Calculations ---
|
|
$vatAmount = 0;
|
|
$grandTotal = $subTotal;
|
|
if ($includeTax) {
|
|
$vatAmount = $subTotal * $vatRate;
|
|
$grandTotal = $subTotal + $vatAmount;
|
|
}
|
|
|
|
// Format dates
|
|
$formattedOfferDate = date("d.m.Y", $offerDate);
|
|
$formattedValidUntil = $validUntilDate ? date("d.m.Y", $validUntilDate) : date("d.m.Y", strtotime("+14 days", $offerDate));
|
|
|
|
?>
|
|
<!DOCTYPE html>
|
|
<html>
|
|
<head>
|
|
<title><?= $text['title'] ?> <?= $offerNumber ?></title>
|
|
<meta charset="utf-8"/>
|
|
<style>
|
|
body {
|
|
font-family: "Open Sans", sans-serif, Verdana;
|
|
font-size: 10px; /* Slightly smaller base font */
|
|
color: #333;
|
|
}
|
|
h1 {
|
|
text-align: center;
|
|
color: #005384; /* Xinon Blue */
|
|
font-size: 18px;
|
|
margin-bottom: 20px;
|
|
}
|
|
.header-info {
|
|
margin-bottom: 20px;
|
|
font-size: 11px;
|
|
}
|
|
.header-info table {
|
|
width: 100%;
|
|
border-collapse: collapse;
|
|
}
|
|
.header-info td {
|
|
padding: 2px 5px;
|
|
}
|
|
.header-info .label {
|
|
font-weight: bold;
|
|
text-align: right;
|
|
padding-right: 10px;
|
|
width: 100px; /* Fixed width for labels */
|
|
}
|
|
|
|
table#positionsTable {
|
|
width: 100%;
|
|
border-collapse: collapse;
|
|
margin-top: 15px;
|
|
margin-bottom: 15px;
|
|
}
|
|
table#positionsTable th {
|
|
border-bottom: 2px solid #005384;
|
|
padding: 8px 4px;
|
|
text-align: left;
|
|
background-color: #f2f2f2;
|
|
font-size: 10px;
|
|
}
|
|
table#positionsTable td {
|
|
border-bottom: 1px solid #e1e1e1;
|
|
padding: 6px 4px;
|
|
vertical-align: top;
|
|
}
|
|
table#positionsTable tr:last-child td {
|
|
border-bottom: none;
|
|
}
|
|
|
|
/* Column Alignment */
|
|
table#positionsTable th.pos, table#positionsTable td.pos { text-align: center; width: 30px; }
|
|
table#positionsTable th.amount, table#positionsTable td.amount { text-align: right; width: 50px;}
|
|
table#positionsTable th.unit, table#positionsTable td.unit { text-align: center; width: 40px;}
|
|
table#positionsTable th.price, table#positionsTable td.price { text-align: right; width: 80px;}
|
|
table#positionsTable th.total, table#positionsTable td.total { text-align: right; width: 90px;}
|
|
table#positionsTable th.article, table#positionsTable td.article { text-align: left; }
|
|
|
|
.position-group-header td {
|
|
background-color: #e8f0f8; /* Light blue for group headers */
|
|
font-weight: bold;
|
|
border-top: 1px solid #ccc;
|
|
border-bottom: 1px solid #ccc;
|
|
padding: 4px;
|
|
}
|
|
|
|
.article-title {
|
|
font-weight: bold;
|
|
}
|
|
.article-description {
|
|
font-size: 9px; /* Smaller font for description */
|
|
color: #555;
|
|
padding-left: 10px;
|
|
margin-top: 2px;
|
|
}
|
|
|
|
table#summaryTable {
|
|
width: 45%; /* Adjust width as needed */
|
|
float: right; /* Align to the right */
|
|
border-collapse: collapse;
|
|
margin-top: 10px;
|
|
font-size: 11px;
|
|
}
|
|
table#summaryTable td {
|
|
padding: 5px 8px;
|
|
}
|
|
table#summaryTable td.label {
|
|
text-align: right;
|
|
font-weight: bold;
|
|
}
|
|
table#summaryTable td.value {
|
|
text-align: right;
|
|
}
|
|
table#summaryTable tr.grand-total td {
|
|
border-top: 1px solid #333;
|
|
border-bottom: 3px double #333; /* Double line for grand total */
|
|
font-weight: bold;
|
|
font-size: 12px;
|
|
}
|
|
table#summaryTable tr.subtotal td {
|
|
border-top: 1px solid #ccc;
|
|
}
|
|
|
|
.offer-text {
|
|
margin-top: 30px;
|
|
padding-top: 15px;
|
|
border-top: 1px solid #e1e1e1;
|
|
font-size: 10px;
|
|
clear: both; /* Ensure it clears the floated summary table */
|
|
}
|
|
.offer-text p {
|
|
margin: 5px 0;
|
|
}
|
|
.tax-info {
|
|
font-style: italic;
|
|
font-size: 9px;
|
|
margin-top: 5px;
|
|
text-align: right;
|
|
}
|
|
|
|
/* Page break avoidance */
|
|
tr, .position-group-header {
|
|
page-break-inside: avoid;
|
|
}
|
|
h1, .header-info, table#summaryTable, .offer-text {
|
|
page-break-before: auto;
|
|
page-break-after: auto;
|
|
}
|
|
table#positionsTable {
|
|
page-break-inside: auto;
|
|
}
|
|
|
|
|
|
</style>
|
|
</head>
|
|
<body>
|
|
|
|
<h1><?= $text['title'] ?></h1>
|
|
|
|
<div class="header-info">
|
|
<table>
|
|
<tr>
|
|
<td class="label"><?= $text['offerNumberLabel'] ?></td>
|
|
<td><?= htmlspecialchars($offerNumber) ?></td>
|
|
<td class="label"><?= $text['offerDateLabel'] ?></td>
|
|
<td><?= $formattedOfferDate ?></td>
|
|
</tr>
|
|
<tr>
|
|
<td></td><td></td> <td class="label"><?= $text['validUntilLabel'] ?></td>
|
|
<td><?= $formattedValidUntil ?></td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
|
|
<table id="positionsTable">
|
|
<thead>
|
|
<tr>
|
|
<th class="pos"><?= $text['table']['pos'] ?></th>
|
|
<th class="article"><?= $text['table']['article'] ?></th>
|
|
<th class="amount"><?= $text['table']['amount'] ?></th>
|
|
<th class="unit"><?= $text['table']['unit'] ?></th>
|
|
<th class="price"><?= $text['table']['unitPrice'] ?></th>
|
|
<th class="total"><?= $text['table']['totalPrice'] ?></th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<?php
|
|
$posCounter = 0;
|
|
foreach ($entries as $groupName => $positions):
|
|
// Optional: Display group name if it exists
|
|
if (!empty($groupName)): ?>
|
|
<tr class="position-group-header">
|
|
<td colspan="6"><?= htmlspecialchars($groupName) ?></td>
|
|
</tr>
|
|
<?php endif;
|
|
|
|
foreach ($positions as $p):
|
|
$posCounter++;
|
|
?>
|
|
<tr>
|
|
<td class="pos"><?= $posCounter ?></td>
|
|
<td class="article">
|
|
<div class="article-title"><?= htmlspecialchars($p['articleText']) ?></div>
|
|
<?php if (!empty($p['articleDescription'])): ?>
|
|
<div class="article-description"><?= nl2br(htmlspecialchars($p['articleDescription'])) ?></div>
|
|
<?php endif; ?>
|
|
</td>
|
|
<td class="amount"><?= number_format($p['amount'], 2, ',', '.') // Format amount as needed ?></td>
|
|
<td class="unit"><?= htmlspecialchars($p['articleUnit']) ?></td>
|
|
<td class="price"><?= number_format($p['price'], 2, ',', '.') ?> <?= $currencySymbol ?></td>
|
|
<td class="total"><?= number_format($p['totalPrice'], 2, ',', '.') ?> <?= $currencySymbol ?></td>
|
|
</tr>
|
|
<?php endforeach; // End positions loop ?>
|
|
<?php endforeach; // End entries (groups) loop ?>
|
|
|
|
<?php if ($posCounter == 0): // Show message if no positions ?>
|
|
<tr>
|
|
<td colspan="6" style="text-align: center; padding: 20px;">Keine Positionen im Angebot enthalten.</td>
|
|
</tr>
|
|
<?php endif; ?>
|
|
</tbody>
|
|
</table>
|
|
|
|
<table id="summaryTable">
|
|
<tbody>
|
|
<tr class="subtotal">
|
|
<td class="label"><?= $text['summary']['subTotal'] ?>:</td>
|
|
<td class="value"><?= number_format($subTotal, 2, ',', '.') ?> <?= $currencySymbol ?></td>
|
|
</tr>
|
|
<?php if ($includeTax):
|
|
$vatLabel = str_replace('{VAT_RATE}', number_format($vatRate * 100, 0), $text['summary']['vatFormatted']);
|
|
?>
|
|
<tr>
|
|
<td class="label"><?= $vatLabel ?>:</td>
|
|
<td class="value"><?= number_format($vatAmount, 2, ',', '.') ?> <?= $currencySymbol ?></td>
|
|
</tr>
|
|
<tr class="grand-total">
|
|
<td class="label"><?= $text['summary']['total'] ?>:</td>
|
|
<td class="value"><?= number_format($grandTotal, 2, ',', '.') ?> <?= $currencySymbol ?></td>
|
|
</tr>
|
|
<?php else: // If tax not included, Total is same as Subtotal ?>
|
|
<tr class="grand-total">
|
|
<td class="label"><?= $text['summary']['total'] ?>:</td>
|
|
<td class="value"><?= number_format($grandTotal, 2, ',', '.') ?> <?= $currencySymbol ?></td>
|
|
</tr>
|
|
<?php endif; ?>
|
|
</tbody>
|
|
</table>
|
|
|
|
<?php // Tax Information Note
|
|
if ($includeTax) {
|
|
echo '<div class="tax-info" style="clear: both;">' . $text['taxInfoNet'] . '</div>';
|
|
}
|
|
// Add taxInfoGross if your prices *include* tax, which is less common for B2B offers.
|
|
?>
|
|
|
|
|
|
<div class="offer-text">
|
|
<?php if (!empty($offerText)): ?>
|
|
<p><strong><?= $text['notes'] ?></strong></p>
|
|
<div><?= nl2br(htmlspecialchars($offerText)) // Use nl2br to preserve line breaks ?></div>
|
|
<br>
|
|
<?php endif; ?>
|
|
<p><?= $text['defaultOfferText'] // Add default closing text ?></p>
|
|
<p>Zahlungsbedingungen: 14 Tage netto.</p>
|
|
<p>Lieferzeit: nach Vereinbarung.</p>
|
|
</div>
|
|
|
|
</body>
|
|
</html>
|