Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
1) <?php
2) /*
3) This file belongs to the Webinterface of schokokeks.org Hosting
4)
5) Written 2008-2018 by schokokeks.org Hosting, namely
6) Bernd Wurst <bernd@schokokeks.org>
7) Hanno Böck <hanno@schokokeks.org>
8)
9) To the extent possible under law, the author(s) have dedicated all copyright and related and neighboring rights to this software to the public domain worldwide. This software is distributed without any warranty.
10)
11) You should have received a copy of the CC0 Public Domain Dedication along with this software. If not, see
12) http://creativecommons.org/publicdomain/zero/1.0/
13)
14) Nevertheless, in case you use a significant part of this code, we ask (but not require, see the license) that you keep the authors' names in place and return your changes to the public. We would be especially happy if you tell us what you're going to do with this code.
15) */
16)
17) require_once('session/start.php');
18) require_once('invoice.php');
19) require_once('inc/security.php');
20)
21) require_role(ROLE_CUSTOMER);
22)
|
Umstellung von filter_input...
Bernd Wurst authored 5 years ago
|
23) $invoice_id = (int) $_GET['id'];
|
Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
24)
25) $items = invoice_items($invoice_id);
26)
27) $invoice = invoice_details($invoice_id);
28)
29)
30) $outformat = 'html';
31) if (isset($_REQUEST['out']) && $_REQUEST['out'] == 'pdf') {
32) $outformat = 'pdf';
33) }
34)
35)
36) // Footer
37) $footer = '<footer>
38) <table width="100%">
39) <tr><td>schokokeks.org GbR<br>
40) Bernd Wurst / Johannes Böck<br>
41) www.schokokeks.org<br>
42) root@schokokeks.org</td>
43)
44) <td>Steuernummer 51072/01109<br>
45) Finanzamt Backnang<br>
46) USt-ID: DE255720588</td>
47)
48) <td>Volksbank Backnang<br>
49) IBAN: DE91 6029 1120 0041 5120 06<br>
50) BIC: GENODES1VBK<br>
51) (Kto: 41512 006 / BLZ: 602 911 20)</td>
52) </tr>
53) </table></footer>';
54)
55)
56)
57)
58) $html = '<html><head><title>Rechnung Nr. '.$invoice_id.'</title>
59) <style type="text/css">
60)
61) @media (min-width: 22cm) {
62) body {
63) margin: 2cm;
64) margin-top: 0.5cm;
65) width: 17cm;
66) }
67) }
68)
69) @media print,dompdf {
70) html {
71) margin: 0;
72) padding: 0;
73) }
74) @page {
75) margin: 0;
76) padding: 0;
77) }
78) body {
79) margin: 2cm;
80) margin-bottom: 3cm;
81) }
82) }
83)
84) body {
85) font-family: "DejaVu Sans", sans-serif;
86) font-size: 8pt;
87) padding: 0;
88) text-align: justify;
89) }
90)
91) table, th, td, tr {
92) font-size: 8pt;
93) margin: 0;
94) padding: 0;
95) border: none;
96) border-collapse: collapse;
97) vertical-align: top;
98) }
99) th {
100) text-align: left;
101) border-bottom: 1px solid black;
102) }
103)
104) @media print,dompdf {
105) #header,
106) footer {
107) position: fixed;
108) left: 0;
109) right: 0;
110) font-size: 8pt;
111) }
112) }
113)
114) #header {
115) top: 0;
116) }
117)
118) footer {
119) border-top: 1px solid #000;
120) }
121)
122) @media print,dompdf {
123) footer {
124) position: absolute;
125) bottom: -2cm;
126) }
127) }
128)
129) #header table,
130) footer table {
131) width: 100%;
132) border-collapse: collapse;
133) border: none;
134) }
135)
136) #header td,
137) footer td {
138) padding: 0;
139) width: 33%;
140) }
141)
142) .page-number {
143) text-align: center;
144) }
145)
146) .page-number:before {
147) content: "Seite " counter(page);
148) }
149)
150) #addressfield {
151) float: left;
152) position: relative;
153) top: 3cm;
154) left: 0cm;
155) width: 8.5cm;
156) min-height: 5cm;
157) }
158) #addressfield p {
159) margin-left: 0.5cm;
160) margin-top: 0.3cm;
161) font-size: 11pt;
162) }
163)
164) #addressfieldheader {
165) font-size: 8pt;
166) padding-left: 0.2cm;
167) width: 100%;
168) border-bottom: 0.1pt solid black;
169) }
170)
171)
172) #rightcolumn {
173) float: right;
174) width: 6cm;
175) }
176)
177)
178)
179) #rightcolumn img {
180) position: relative;
181) top: 0;
182) width: 4.08cm;
183) height: 3cm;
184) margin: 0;
185) padding: 0;
186) }
187)
188) td {
189) font-size: 8pt;
190) }
191)
192) p {
193) margin-bottom: 0.6cm;
194) }
195)
196) tr.even {
197) background-color: #eee;
198) }
199)
200)
201) </style>
202) </head>
203) <body>
204) ';
205) // Folding Marks
206) $html .= '
207) <div style="position: absolute; left: -2cm; top: 8.5cm; width: 0.5cm; height: 1px; border-top: 1px solid black;"></div>
208) <div style="position: absolute; left: -2cm; top: 19cm; width: 0.5cm; height: 1px; border-top: 1px solid black;"></div>
209) <div style="position: absolute; left: -2cm; top: 12.85cm; width: 0.6cm; height: 1px; border-top: 1px solid black;"></div>
210) ';
211)
212) if ($outformat == 'pdf') {
213) $html .= $footer;
214) }
215)
216) $address = invoice_address($invoice['kunde']);
217)
218) // Address Field
219) $html .= '<div id="addressfield">
220) <div id="addressfieldheader">schokokeks.org · Köchersberg 32 · 71540 Murrhardt</div>
|
codingstyle fixes
Hanno authored 5 years ago
|
221) <p>'.($address['company'] != null ? $address['company'].'<br>' : '').$address['name'].'<br>
|
Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
222) '.$address['address'].'<br>
223) '.($address['country'] != 'DE' ? $address['country'].'-' : '').$address['zip'].' '.$address['city'].'
224) </p>
225) </div>';
226) // Right col
227) $html .= '<div id="rightcolumn">
228) <img src="'.($outformat == 'pdf' ? '.' : '../..').'/themes/default/images/schokokeks.png">
229) <p style="margin-bottom: 0.2cm;"><strong>schokokeks.org GbR</strong><br>
230) Bernd Wurst / Johannes Böck<br>
231) Köchersberg 32<br>
232) 71540 Murrhardt</p>
233) <p>Tel: 07192-936432<br>
234) E-Mail: root@schokokeks.org</p>
235) </div>';
236)
237) // Caption / Invoice-Details
238) $html .= '
239) <table style="width: 100%; clear: both;">
240) <tr><td style="width: 11cm; font-size: 11pt; font-weight: bold;">Rechnung</td><td>
241) <table>
242) <tr><td colspan="2"><strong>Bei Fragen bitte immer angeben:</strong></td></tr>
243) <tr><td>Rechnungsdatum:</td><td align="right">'.date("d. m. Y", strtotime($invoice['datum'])).'</td></tr>
244) <tr><td>Rechnungsnummer:</td><td align="right">'.$invoice_id.'</td></tr>
245) <tr><td>Kundennummer:</td><td align="right">'.$invoice['kunde'].'</td></tr>
246) </table>
247) </td></tr>
248) </table>';
249)
250) $anrede = 'Sehr geehrte Damen und Herren';
251)
252) $parts = explode(' ', $address['name']);
253) $nachname = array_pop($parts);
254)
255) if ($address['name']) {
256) if ($address['salutation'] == 'Herr') {
257) $anrede = 'Sehr geehrter Herr '.$nachname;
258) } elseif ($address['salutation'] == 'Frau') {
259) $anrede = 'Sehr geehrte Frau '.$nachname;
260) }
261) }
262)
263) // Salutation
264) $html .= '<p>'.$anrede.',<br>
265) hiermit stellen wir die nachfolgend genannten Posten in Rechnung.</p>';
266)
267)
268) // Table
269) $html .= '<table style="width: 100%;">
270) <tr style="border-bottom: 1px solid black;">
271) <th style="width: 1cm; text-align: center;">Anz.</th>
|
Erlaube Admins das Abfragen...
Bernd Wurst authored 5 years ago
|
272) <th style="width: 2cm;"> </th>
|
Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
273) <th>Beschreibung</th>
274) <th style="width: 2.5cm; text-align: center;">Einzelpreis</th>
275) <th style="width: 3cm; text-align: center;">Gesamtpreis</th></tr>';
276)
277) $vattype = null;
278) // An der ersten Zeile entscheidet sich, ob die gesamte Rechnung als Netto- oder Bruttorechnung erstellt wird
|
Codingstyle PSR12 + array s...
Hanno Böck authored 3 years ago
|
279) $einzelsummen = [];
|
Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
280) $summe = 0.0;
281)
282) $odd = true;
283) foreach ($items as $item) {
284) if ($vattype == 'gross' && $item['brutto'] == 0) {
|
codingstyle fixes
Hanno authored 5 years ago
|
285) system_failure('Mixed gross and net positions');
|
Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
286) } elseif ($vattype == 'net' && $item['brutto'] == 1) {
|
codingstyle fixes
Hanno authored 5 years ago
|
287) system_failure('Mixed gross and net positions');
|
Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
288) } else {
289) $vattype = ($item['brutto'] == 1 ? 'gross' : 'net');
290) }
291)
292) $anzahl = $item['anzahl'];
293) if (round($anzahl, 0) == $anzahl) {
294) $anzahl = round($anzahl, 0);
295) }
296) $desc = $item['beschreibung'];
297) if ($item['enddatum'] == null) {
298) $desc .= '<br />(Leistungsdatum: '.$item['datum'].')';
299) } else {
300) $desc .= '<br />(Leistungszeitraum: '.$item['datum'].' - '.$item['enddatum'].')';
301) }
302) $epreis = $item['betrag'];
303) if ($item['brutto'] == 0) {
304) $epreis = $epreis * (1 + ($item['mwst'] / 100));
305) }
306) $einheit = ($item['einheit'] ? $item['einheit'] : '');
307) $gesamt = $epreis * $item['anzahl'];
308) $epreis = $epreis;
309)
310) if (array_key_exists($item['mwst'], $einzelsummen)) {
311) $einzelsummen[$item['mwst']]['net'] += $gesamt / (1 + ($item['mwst'] / 100));
312) $einzelsummen[$item['mwst']]['vat'] += $gesamt / (1 + ($item['mwst'] / 100)) * ($item['mwst'] / 100);
313) $einzelsummen[$item['mwst']]['gross'] += $gesamt;
314) } else {
|
Codingstyle PSR12 + array s...
Hanno Böck authored 3 years ago
|
315) $einzelsummen[$item['mwst']] = ['net' => $gesamt / (1 + ($item['mwst'] / 100)),
|
Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
316) 'vat' => $gesamt / (1 + ($item['mwst'] / 100)) * ($item['mwst'] / 100),
|
Codingstyle PSR12 + array s...
Hanno Böck authored 3 years ago
|
317) 'gross' => $gesamt, ];
|
Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
318) }
319) $summe += $gesamt;
320)
321) $html .= "<tr class='".($odd ? 'odd' : 'even')."'><td style='text-align: right;'>{$anzahl}</td><td>{$einheit}</td>";
322) $html .= "<td>{$desc}</td>";
323) $html .= "<td style='text-align: right;'>".number_format($epreis, 2, ',', '.')." €</td><td style='text-align: right;'>".number_format($gesamt, 2, ',', '.')." €</td></tr>\n";
324) $odd = ! $odd;
325) }
326) $html .= '<tr><td> </td></tr>';
327)
328) foreach ($einzelsummen as $percent => $sums) {
329) $html .= '<tr><td colspan="4" style="text-align: right;">Nettobetrag ('.number_format($percent, 1, ',', '.').'% MwSt):</td><td style="text-align: right;">'.number_format($sums['net'], 2, ',', '.').' €</td></tr>
330) <tr><td colspan="4" style="text-align: right;">MwSt-Betrag '.number_format($percent, 1, ',', '.').'%:</td><td style="text-align: right;">'.number_format($sums['vat'], 2, ',', '.').' €</td></tr>';
331) if (count($einzelsummen) > 1) {
|
codingstyle fixes
Hanno authored 5 years ago
|
332) $html .= '<tr><td colspan="4" style="text-align: right;">Brutto-Teilbetrag '.number_format($percent, 1, ',', '.').'% MwSt:</td><td style="text-align: right;">'.number_format($sums['gross'], 2, ',', '.').' €</td></tr>';
|
Erzeuge HTML als Eingabe fü...
Bernd Wurst authored 5 years ago
|
333) }
334) }
335) $html .= '<tr style="font-weight: bold;"><td colspan="4" style="text-align: right;">Rechnungsbetrag:</td><td style="text-align: right;">'.number_format($summe, 2, ',', '.').' €</td></tr>
336) </table>';
337)
338) // Disclaimer
339) if ($invoice['abbuchung'] == 1) {
340) $sepamandat = get_sepamandat($invoice['sepamandat']);
341) $iban = substr($sepamandat['iban'], 0, 8) . '**********' . substr($sepamandat['iban'], -4);
342) $display_iban = $iban;
343) for ($i = strlen($iban)-(strlen($iban)%4) ; $i != 0 ; $i -= 4) {
344) $display_iban = substr($display_iban, 0, $i) . ' ' . substr($display_iban, $i);
345) }
346) $html .= '<p><strong>Bitte nicht überweisen!</strong> Der fällige Betrag wird gemäß dem von Ihnen erteilten
347) Lastschrift-Mandat in wenigen Tagen vom Konto mit der IBAN '.$display_iban.' bei der '.$sepamandat['bankname'].'
348) (BIC: '.$sepamandat['bic'].') abgebucht. Diese Kontodaten beruhen auf dem Mandat Nr. '.$sepamandat['mandatsreferenz'].'
349) vom '.$sepamandat['erteilt'].'. Unsere Gläubiger-ID lautet '.$sepamandat['glaeubiger_id'].'.</p>';
350) } else {
351) $html .= '<p>Bitte begleichen Sie diese Rechnung umgehend nach Erhalt ohne Abzüge auf das unten angegebene Konto. Geben Sie
352) im Verwendungszweck Ihrer Überweisung bitte die Rechnungsnummer '.$invoice_id.' an, damit Ihre Buchung korrekt
353) zugeordnet werden kann.</p>';
354) }
355)
356)
357) $html .= '<p>Wir danken Ihnen, dass Sie unser Angebot in Anspruch genommen haben und hoffen weiterhin auf eine gute
358) Zusammenarbeit. Dieser Rechnung liegen die Allgemeinen Geschäftsbedingungen zum Zeitpunkt des
359) Rechnungsdatums zugrunde, die Sie unter https://www.schokokeks.org/agb abrufen können.</p>';
360)
361)
362) if ($outformat == 'html') {
363) $html .= $footer;
364) }
365)
366) $html .= '</body>
367) </html>';
368)
369) // Composer's auto-loading functionality
370) require "vendor/autoload.php";
371)
372) use Dompdf\Dompdf;
373)
374) //generate some PDFs!
375) $dompdf = new DOMPDF(); //if you use namespaces you may use new \DOMPDF()
376) $dompdf->setPaper('A4', 'portrait');
377) $dompdf->loadHtml($html);
378) $dompdf->render();
379)
380) if ($outformat == 'pdf') {
|
Codingstyle PSR12 + array s...
Hanno Böck authored 3 years ago
|
381) $dompdf->stream("sample.pdf", ["Attachment"=>0]);
|