marei: headline/colspec keys for SimpleTabular
[kivitendo-erp.git] / templates / print / marei / kiviletter.sty
1 \NeedsTeXFormat{LaTeX2e}
2 \ProvidesPackage{kiviletter}[2020/04/24 Letter Layouts for Kivitendo]
3
4 \newif\if@kivi@infobox
5 \DeclareOption{reffields}{\@kivi@infoboxfalse}
6 \DeclareOption{infobox}{\@kivi@infoboxtrue}
7 \@kivi@infoboxtrue
8
9 \ProcessOptions\relax
10
11
12 \RequirePackage{expl3}
13 \RequirePackage{iftex}
14 \KOMAoptions{fontsize=12pt}
15 % Schriftart, Eingabelayout der Tastatur
16 \ifPDFTeX
17 \RequirePackage[utf8]{inputenc}% Nur notwendig, wenn Basis älter als TL2018
18 \RequirePackage[T1]{fontenc}
19 \RequirePackage{lmodern}
20 \else
21 \RequirePackage{fontspec}
22 \fi
23
24 %\RequirePackage{xltabular}
25 \RequirePackage{tabularx}
26 \RequirePackage{longtable}
27 \RequirePackage{booktabs}
28 \RequirePackage{graphicx}
29
30 \ifPDFTeX
31 \RequirePackage{eurosym}
32 \DeclareUnicodeCharacter{20AC}{\euro}
33 \fi
34
35 \RequirePackage[fromlogo,fromalign=right,
36   firstfoot=false,%Für einheitliche Randeinstellungen
37   refline=nodate,
38         ]{scrletter}
39 \LoadLetterOption{DIN}
40
41 \newkomavar{transaction}
42 \newkomavar[\lieferschein{}~\nr]{delivery}
43 \newkomavar[\angebot{}~\nr]{quote}
44 \newkomavar{orderID}
45 \newkomavar{projectID}
46
47 \usepackage{geometry}
48
49 \ExplSyntaxOn
50 \dim_new:N \g_kivi_margin_dim
51 \dim_gset:Nn \g_kivi_margin_dim {\useplength{toaddrhpos}}
52 \geometry{a4paper,margin=\g_kivi_margin_dim,heightrounded}
53 \savegeometry{kivi.letter@default}
54 %Scratch variables
55 \int_new:N \l_kivi_tmp_int
56 \bool_new:N \l_kivi_tmp_bool
57 \bool_new:N  \g_kivi_TableFoot_bool
58 \dim_new:N \g_kivi_orig@textheight_dim
59 \ExplSyntaxOff
60
61 \newsavebox{\shippingAddressBox}
62
63
64 \DeclareNewLayer[
65 foreground,
66 hoffset=\useplength{toaddrhpos},
67 voffset=\dimexpr\useplength{toaddrvpos}+\useplength{toaddrheight}+4\baselineskip,%sep to shippingaddressbox
68 contents={\usebox\shippingAddressBox}
69 ]{kivitendo.shippingaddress}
70
71
72 \ExplSyntaxOn
73 \AtBeginLetter{\dim_gset:Nn \g_kivi_orig@textheight_dim {\textheight}}
74 \ExplSyntaxOff
75
76 \newpairofpagestyles{kivitendo.letter}{}
77 \renewcommand*{\letterpagestyle}{kivitendo.letter}
78
79 \DeclareNewPageStyleByLayers{kivitendo.letter.PricingTable}{
80         kivitendo.TableHead,
81         kivitendo.TableFoot
82         kivitendo.letter.head.odd,kivitendo.letter.head.even,kivitendo.letter.head.oneside,%
83         kivitendo.letter.foot.odd,kivitendo.letter.foot.even,kivitendo.letter.foot.oneside,%
84 }
85 \DeclareNewPageStyleByLayers{kivitendo.letter.first}{
86         kivitendo.shippingaddress,
87         kivitendo.TableFoot,
88         kivitendo.letter.head.odd,kivitendo.letter.head.even,kivitendo.letter.head.oneside,%
89         kivitendo.letter.foot.odd,kivitendo.letter.foot.even,kivitendo.letter.foot.oneside,%
90 }
91
92 \setkomavar{backaddress}{\firma\ $\cdot$ \strasse\ $\cdot$ \ort}
93
94 \setkomavar{firsthead}{
95         \if@logo
96         \rlap{\usekomavar{fromlogo}}%
97         \fi
98 }
99
100 \@setplength{locwidth}{6cm}
101
102 \ExplSyntaxOn
103 \dim_new:N \l_kivi_tab_desc_leftskip_dim
104
105
106
107 \cs_new:Nn \__kivi_set_colwidth:nn  {
108         \dim_set:cn {l_kivi_tab_#1_dim} {#2}
109 }
110
111
112 \cs_new:Nn \__kivi_initialize_columns: {
113         \clist_map_inline:Nn \g_kivi_pricingtable_col_clist {
114                 \bool_if_exist:cF {l_kivi_col_##1_bool} 
115                 {
116                         \bool_new:c {l_kivi_col_##1_bool}
117                         \dim_new:c {l_kivi_tab_##1_dim}
118                         \keys_define:nn {kivi/PricingTable} {
119                                 ##1 .choice:,
120                                 ##1 / true .code:n = \bool_set_true:c {l_kivi_col_##1_bool},
121                                 ##1 / false .code:n = \bool_set_false:c {l_kivi_col_##1_bool},
122                                 ##1 / unknown .code:n = {
123                                         \bool_set_true:c {l_kivi_col_##1_bool}
124                                         \dim_set:cn {l_kivi_tab_##1_dim} {####1}
125                                 },
126                                 ##1 .default:n = true,
127                                 ##1 .initial:n = true,
128                                 ##1 / header .prop_put:c = {l_kivi_col_##1_prop},
129                                 ##1 / colspec .prop_put:c = {l_kivi_col_##1_prop},
130                         }
131                 }
132         }
133 }
134
135 \clist_new:N \g_kivi_pricingtable_col_clist
136
137 \keys_define:nn {kivi/PricingTable} {
138         columns .code:n = 
139         \clist_gset:Nn \g_kivi_pricingtable_col_clist {#1}
140         \__kivi_initialize_columns:,
141         columns .initial:n = {pos, id, desc, amount, price, pricetotal},
142 }
143
144 % set default values for colwidth
145 \keys_set:nn {kivi/PricingTable} {
146         pos=5ex,
147         id=4em,
148         amount=5em,
149         price=7em,
150         pricetotal=7em,
151 %       desc=auto,
152         pos/header=\position,
153         id/header=\artikelnummer,
154         desc/header=\bezeichnung,
155         amount/header=\menge,
156         price/header=\einzelpreis,
157         pricetotal/header=\gesamtpreis,
158         price / colspec = Price,
159         pricetotal / colspec = Price ,
160 }
161
162 \dim_new:N \g_kivi_tabcolsep_dim
163 \dim_gset:Nn \g_kivi_tabcolsep_dim {.5\tabcolsep}
164
165 \cs_new:Nn \__kivi_calc_desc_column: {
166         \dim_zero:N \l_kivi_tab_desc_leftskip_dim
167         \dim_zero:N \l_kivi_tab_desc_dim
168         \bool_set_false:N \l_tmpa_bool
169         \clist_map_inline:Nn \g_kivi_pricingtable_col_clist {
170                 \tl_if_eq:nnTF {##1} {desc}  {
171                         \dim_set:Nn \l_kivi_tab_desc_dim {
172                                 \textwidth-\l_kivi_tab_desc_leftskip_dim
173                         }
174                         \bool_set_true:N \l_tmpa_bool
175                 
176                 }{
177                         \bool_if:NTF \l_tmpa_bool {
178                                 \bool_if:cT {l_kivi_col_##1_bool} {
179                                         \dim_sub:Nn \l_kivi_tab_desc_dim {
180                                                 \dim_use:c {l_kivi_tab_##1_dim}+2\g_kivi_tabcolsep_dim
181                                         }
182                                 }
183                         }{
184                                 \bool_if:cT {l_kivi_col_##1_bool} {
185                                         \dim_add:Nn \l_kivi_tab_desc_leftskip_dim {
186                                                 \dim_use:c {l_kivi_tab_##1_dim}+2\g_kivi_tabcolsep_dim
187                                         }
188                                 }
189                         }
190                 }
191         }
192 }
193
194 \newcolumntype{P}[1]{>{\raggedleft\arraybackslash}p{#1}<{\__kivi_tab_column_currency:}}
195
196 \RequirePackage{tcolorbox}
197 \tcbuselibrary{breakable, skins}
198
199 \tcb@new@skin{kivi@LT}{base@unbroken,%
200         frame~engine=empty,interior~titled~engine=empty,interior~engine=empty,segmentation~engine=empty,title~engine=empty,%
201         skin~first=kivi@LT@first,skin~middle=kivi@LT@middle,skin~last=kivi@LT@last,
202         underlay~first~and~middle={
203                 \node[anchor=north]  at (interior.north)  {\csname box_use:c\endcsname  {g_kivi_LT@head_box}};
204                 \node[anchor=south]  at (interior.south)  {\csname box_use:c\endcsname  {g_kivi_LT@foot_box}};
205         },
206         underlay~unbroken~and~last={
207         \node[anchor=north]  at (interior.north)  {\csname box_use:c\endcsname  {g_kivi_LT@head_box}};
208         \node[anchor=south]  at (interior.south)  {\csname box_use:c\endcsname  {g_kivi_LT@lastfoot_box}};
209         },
210         boxsep=0pt,
211         boxrule=0pt,
212         left=0pt,
213         right=0pt,
214         bottom=\box_ht:N  \g_kivi_LT@foot_box+\box_dp:N  \g_kivi_LT@foot_box + \aboverulesep,
215         top=\box_ht:N  \g_kivi_LT@head_box+\box_dp:N  \g_kivi_LT@head_box +\belowrulesep,
216         parbox=false,
217 }
218
219 \tcb@new@skin{kivi@LT@first}{base@first,%
220         frame~engine=empty,interior~titled~engine=empty,interior~engine=empty,segmentation~engine=empty,title~engine=empty,%
221         skin~first=kivi@LT@first,skin~middle=kivi@LT@middle,skin~last=kivi@LT@middle,
222 }
223
224 \tcb@new@skin{kivi@LT@middle}{base@middle,%
225         frame~engine=empty,interior~titled~engine=empty,interior~engine=empty,segmentation~engine=empty,title~engine=empty,%
226         skin~first=kivi@LT@middle,skin~middle=kivi@LT@middle,skin~last=kivi@LT@middle,
227 }
228
229 \tcb@new@skin{kivi@LT@last}{base@last,%
230         frame~engine=empty,interior~titled~engine=empty,interior~engine=empty,segmentation~engine=empty,title~engine=empty,%
231         skin~first=kivi@LT@middle,skin~middle=kivi@LT@middle,skin~last=kivi@LT@last,
232 }
233
234 \tcbset{kivi@LT/.style={skin=kivi@LT}}%
235
236
237
238 \seq_new:N \l_kivi_PricingTable_seq
239 \seq_new:N \l_kivi_columns_seq
240 \seq_new:N \g_kivi_extraDescription_seq
241 \newcommand{\FakeTable}[1]{
242         \par
243         \seq_set_split:Nnn \l_kivi_PricingTable_seq {\tabularnewline} {#1}
244         \seq_remove_all:Nn \l_kivi_PricingTable_seq {}
245         \begingroup
246         \let\ExtraDescription\__kivi_addExtraDescription:n
247         \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
248         \seq_map_inline:Nn \l_kivi_PricingTable_seq {
249                 \seq_set_split:Nnn  \l_kivi_columns_seq {&} {##1}
250         \seq_gclear:N \g_kivi_extraDescription_seq
251         \exp_args:Nnx \use:n {\tabular[t]}\g_kivi_Pricing_colspec_tl
252                 \seq_pop_left:NN \__l_FakeTable_columns_seq \l_tmpa_tl
253                 \seq_item:Nn \l_kivi_columns_seq {\l_tmpa_tl}
254                 \seq_map_inline:Nn \__l_FakeTable_columns_seq {
255                         &\seq_item:Nn \l_kivi_columns_seq {####1}
256                 }
257         \endtabular
258         \seq_if_empty:NTF \g_kivi_extraDescription_seq
259         {\par}
260         {\par\nopagebreak
261         \begingroup
262         \setlength{\hsize}{\dimexpr\l_kivi_tab_desc_dim+\l_kivi_tab_desc_leftskip_dim}
263         \setlength{\leftskip}{\l_kivi_tab_desc_leftskip_dim}
264         \usekomafont{extraDescription}
265         \seq_use:Nn \g_kivi_extraDescription_seq {\\}
266         \par\nointerlineskip
267         \endgroup
268         }
269         }
270         \endgroup
271 }
272
273
274 \seq_new:N  \__l_FakeTable_columns_seq
275 \cs_new:Nn \__kivi_setup_FakeTable: {
276         \seq_clear:N \__l_FakeTable_columns_seq
277         \int_zero:N \l_tmpa_int
278         \clist_map_inline:Nn \g_kivi_pricingtable_col_clist {
279                 \int_incr:N \l_tmpa_int
280                 \bool_if:cT {l_kivi_col_##1_bool} {\seq_put_right:Nx \__l_FakeTable_columns_seq {\int_use:N \l_tmpa_int}}
281         }
282 }
283
284 \tl_new:N \g_kivi_Pricing_colspec_tl
285 \tl_gset:Nn \g_kivi_Pricing_colspec_tl {
286         @{}
287         \bool_if:NT \l_kivi_col_pos_bool {p{\l_kivi_tab_pos_dim}}
288         \bool_if:NT \l_kivi_col_id_bool {p{\l_kivi_tab_id_dim}}
289         p{\l_kivi_tab_desc_dim}
290         \bool_if:NT \l_kivi_col_amount_bool {\exp_not:n {>{\raggedleft\arraybackslash}p{\l_kivi_tab_amount_dim}}}
291         \bool_if:NT \l_kivi_col_price_bool {\exp_not:n {>{\raggedleft\arraybackslash}p{\l_kivi_tab_price_dim}<{\__kivi_tab_column_currency:}}}
292         \bool_if:NT \l_kivi_col_pricetotal_bool {\exp_not:n {>{\raggedleft\arraybackslash}p{\l_kivi_tab_pricetotal_dim}<{\__kivi_tab_column_currency:}}}
293         @{}
294 }
295
296 \cs_new:Nn \__kivi_tab_column_currency: {\,\currency}
297 \cs_set:Nn \__kivi_tab_column_header_currency: {}
298 \cs_set_eq:NN \__kivi_tab_column_body_currency:  \__kivi_tab_column_currency:
299
300 \clist_map_inline:nn {head, foot, firsthead, lastfoot} {%TODO reduce
301         \box_new:c {g_kivi_LT@#1_box}
302 }
303
304 \newkomafont{PricingTableHeader}{\bfseries}
305
306 \cs_new:Nn \__kivi_setup_LT_boxes: {
307         \__kivi_calc_desc_column:
308         \hbox_gset:Nn \g_kivi_LT@head_box {
309                 \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
310                 \exp_args:Nnx \use:n {\tabular[b]}\g_kivi_Pricing_colspec_tl
311                 \__kivi_PricingTabular_header:
312                 \endtabular
313         }
314         \hbox_gset:Nn \g_kivi_LT@foot_box {
315                 \raisebox{\depth}{
316                         \begin{tabular*}{\textwidth}{@{\extracolsep{\fill}}r@{}}
317                                 \midrule
318                                 \strut\weiteraufnaechsterseite
319                         \end{tabular*}
320                 }
321         }
322         \hbox_gset:Nn \g_kivi_LT@lastfoot_box {
323                 \raisebox{\dimexpr\depth+\baselineskip}[0pt][0pt]{
324                         \begin{tabular*}{\textwidth}{@{\extracolsep{\fill}}r@{}}
325                                 \bottomrule
326                         \end{tabular*}
327                 }
328         }
329 }
330
331
332 %Macht es sinn hier eine Variante zu machen, in der alle Spalten Belegbar sind?
333 \newenvironment{PricingTotal}{
334         \par
335         \vspace{-\ht\strutbox}
336         \unskip
337         \tabular[t]{@{}p{\dim_eval:n {\linewidth-\l_kivi_tab_pricetotal_dim-2\tabcolsep}}P{\l_kivi_tab_pricetotal_dim}@{}}
338         \midrule
339 }{
340         \endtabular
341 }
342
343
344 \newcommand*\ExtraDescription{
345         \PackageError{kiviletter}{The~command~\string\ExtraDescription\space~may~be~only~used~inside~the~\string\FakeTable\space~environment.}{See~documentation~for~details}
346 }
347
348
349 \cs_new:Nn \__kivi_addExtraDescription:n {\seq_gput_right:Nn \g_kivi_extraDescription_seq {#1}}
350
351 \newenvironment{PricingTabular}[1][]{
352         \begingroup
353         \tl_if_empty:nF {#1} {\keys_set:nn {kivi/PricingTable} {#1}}
354         \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
355         \__kivi_calc_desc_column:
356         \exp_args:Nx \longtable \g_kivi_Pricing_colspec_tl
357         % Tabellenkopf
358         \__kivi_PricingTabular_header:
359         \endhead
360         \midrule
361         \rlap{\makebox[\textwidth][r]{\weiteraufnaechsterseite}}\\
362         \endfoot
363         \bottomrule
364         \endlastfoot
365 }{
366         \endlongtable
367         \endgroup
368 }
369
370 \cs_set:Nn \__kivi_PricingTabular_header: {
371         \toprule
372         \cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_header_currency:
373         \bool_set_false:N \l_tmpa_bool
374         \clist_map_inline:Nn \g_kivi_pricingtable_col_clist     {
375                 \bool_if:cT {l_kivi_col_##1_bool} {
376                         \bool_if:NT \l_tmpa_bool {&}
377                         \bool_set_true:N \l_tmpa_bool
378                         \usekomafont{PricingTableHeader}
379                         \prop_item:cn {l_kivi_col_##1_prop} {header}
380                 }
381         }
382         \cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_body_currency:
383         \\
384         \midrule
385 }
386
387 \RequirePackage{xltabular}
388
389 \keys_define:nn {kivi/SimpleTabular} {
390         colspec .tl_set:N =\l_kivi_SimpleTabular_colspec_tl,
391         colspec .initial:n = {rrX},
392         headline .tl_set:N = \l_kivi_SimpleTabular_headline_tl,
393         headline .initial:n = {\bfseries\position & \bfseries\menge & \bfseries\bezeichnung},
394 }
395
396 \newenvironment{SimpleTabular}[1][]
397 {
398         \tl_if_in:nnTF {#1} {=} {\keys_set:nn {kivi/SimpleTabular} {#1}} {\tl_if_empty:nF {#1} {\tl_set:Nn \l_kivi_SimpleTabular_headline_tl {#1}}}
399         \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
400         \tl_put_right:Nn \l_kivi_SimpleTabular_colspec_tl {@{}}
401         \tl_put_left:Nn \l_kivi_SimpleTabular_colspec_tl {@{}}
402         \exp_args:NnV \xltabular{\linewidth}\l_kivi_SimpleTabular_colspec_tl
403         \toprule
404         \l_kivi_SimpleTabular_headline_tl\\
405         \midrule
406         \endhead
407         \midrule
408         \rlap{\makebox[\textwidth][r]{\weiteraufnaechsterseite}}\\
409         \endfoot
410         \bottomrule
411         \endlastfoot
412         \ignorespaces
413 }{
414         \def\@currenvir{tabularx}
415         \endxltabular
416 }
417
418 %PricingTabular* kann automatisch spalten ignorieren
419 % \begin{PricingTabular*}[id=false]
420 % deaktiviert damit die Spalte der Produktnummer
421 % analog ist dies für pos, amount, price, pricetotal möglich.
422 % Die Spalte der Bezeichnung ist nicht deaktivierbar
423 \newenvironment{PricingTabular*}[1][]{
424         \tl_if_empty:nF {#1} {\keys_set:nn {kivi/PricingTable} {#1}}
425         \__kivi_setup_LT_boxes:
426         \__kivi_setup_FakeTable:
427         \PricingTabularBox\ignorespaces
428 }{\endPricingTabularBox}
429
430 \newtcolorbox{PricingTabularBox}{breakable,skin=kivi@LT}
431
432
433
434 \if@kivi@infobox
435 \setkomavar{location}{
436         \ifkomavarempty{transaction}{}{
437         \bfseries
438         \usekomavar{transaction}
439         }
440         \par
441         \medskip
442         \begin{tabularx}{\useplength{locwidth}}{@{}l<{:}>{\raggedleft\arraybackslash}X@{}}
443                 \usekomavar*{date}&\usekomavar{date}\\
444                 \ifkomavarempty{myref}{}{
445                         \usekomavar*{myref}&\usekomavar{myref}\\
446                 }
447                 \kundennummer&\usekomavar{customer}\\
448                 \ifkomavarempty{yourref}{}{
449                         \usekomavar*{yourref}&\usekomavar{yourref}\\
450                 }
451                 \ifkomavarempty{delivery}{}{
452                         \usekomavar*{delivery}&\usekomavar{delivery}\\
453                 }
454                 \ifkomavarempty{quote}{}{
455                         \usekomavar*{quote}&\usekomavar{quote}\\
456                 }
457                 \ifkomavarempty{orderID}{}{\auftragsnummer&\usekomavar{orderID}\\}
458                 \ifkomavarempty{projectID}{}{\projektnummer&\usekomavar{projectID}\\}
459                 \ansprechpartner&\usekomavar{fromname}
460                 \ifkomavarempty{fromphone}{}{\\\textTelefon&\usekomavar{fromphone}}
461                 \ifkomavarempty{fromemail}{}{\\\textEmail&\usekomavar{fromemail}}
462         \end{tabularx}
463 }
464 \removereffields
465 \AtBeginLetter{
466         \ifdim\ht\shippingAddressBox>\z@
467         \@addtoplength{refvpos}{\dimexpr\ht\shippingAddressBox+\dp\shippingAddressBox}
468         \@addtoplength{refvpos}{4\baselineskip}%sep between address boxes
469         \fi
470 }
471 \ExplSyntaxOff
472 \fi
473
474
475
476 \renewcommand*{\raggedsignature}{\raggedright}
477
478 \newkomafont{extraDescription}{}
479
480 \endinput