1 \NeedsTeXFormat{LaTeX2e}
2 \ProvidesPackage{kiviletter}[2020/04/24 Letter Layouts for Kivitendo]
6 \DeclareOption{reffields}{\@kivi@infoboxfalse}
7 \DeclareOption{infobox}{\@kivi@infoboxtrue}
8 \DeclareOption{nofooter}{\@kivi@footerfalse}
9 \DeclareOption{footer}{\@kivi@footertrue}
13 \DeclareOption*{\PassOptionsToPackage{\CurrentOption}{scrletter}}
18 \RequirePackage{expl3}
19 \RequirePackage{iftex}
20 \KOMAoptions{fontsize=12pt}
21 % Schriftart, Eingabelayout der Tastatur
23 \RequirePackage[utf8]{inputenc}% Nur notwendig, wenn Basis älter als TL2018
24 \RequirePackage[T1]{fontenc}
25 \RequirePackage{lmodern}
27 \RequirePackage{fontspec}
30 \RequirePackage{xltabular}
31 \RequirePackage{booktabs}
32 \RequirePackage{graphicx}
35 \RequirePackage{eurosym}
36 \DeclareUnicodeCharacter{20AC}{\euro}
39 \RequirePackage[fromlogo,fromalign=right,
40 firstfoot=false,%Für einheitliche Randeinstellungen
43 \LoadLetterOption{DIN}
45 \newkomavar{transaction}
46 \newkomavar[\lieferschein{}~\nr]{delivery}
47 \newkomavar[\angebot{}~\nr]{quote}
49 \newkomavar{projectID}
54 \dim_new:N \g_kivi_margin_dim
55 \dim_gset:Nn \g_kivi_margin_dim {\useplength{toaddrhpos}}
56 \geometry{a4paper,margin=\g_kivi_margin_dim,heightrounded}
58 \int_new:N \l_kivi_tmp_int
59 \bool_new:N \l_kivi_tmp_bool
60 \bool_new:N \g_kivi_TableFoot_bool
61 \dim_new:N \g_kivi_orig@textheight_dim
64 \newsavebox{\shippingAddressBox}
69 hoffset=\useplength{toaddrhpos},
70 voffset=\dimexpr\useplength{toaddrvpos}+\useplength{toaddrheight}+4\baselineskip,%sep to shippingaddressbox
71 contents={\usebox\shippingAddressBox}
72 ]{kivitendo.shippingaddress}
74 \newpairofpagestyles{kivitendo.letter}{}
76 \renewcommand*{\letterpagestyle}{kivitendo.letter}
78 \DeclareNewPageStyleByLayers{kivitendo.letter.first}{
79 kivitendo.shippingaddress,
80 plain.kivitendo.letter.head.odd,plain.kivitendo.letter.head.even,plain.kivitendo.letter.head.oneside,%
81 plain.kivitendo.letter.foot.odd,plain.kivitendo.letter.foot.even,plain.kivitendo.letter.foot.oneside,%
84 \setkomavar{backaddress}{\firma\ $\cdot$ \strasse\ $\cdot$ \ort}
86 \setkomavar{firsthead}{
88 \rlap{\usekomavar{fromlogo}}%
92 \@setplength{locwidth}{6cm}
95 \dim_new:N \l_kivi_tab_desc_leftskip_dim
99 \cs_new:Nn \__kivi_set_colwidth:nn {
100 \dim_set:cn {l_kivi_tab_#1_dim} {#2}
104 \cs_new:Nn \__kivi_initialize_columns: {
105 \clist_map_inline:Nn \g_kivi_pricingtable_col_clist {
106 \bool_if_exist:cF {l_kivi_col_##1_bool}
108 \bool_new:c {l_kivi_col_##1_bool}
109 \dim_new:c {l_kivi_tab_##1_dim}
110 \keys_define:nn {kivi/PricingTable} {
112 ##1 / true .code:n = \bool_set_true:c {l_kivi_col_##1_bool},
113 ##1 / false .code:n = \bool_set_false:c {l_kivi_col_##1_bool},
114 ##1 / unknown .code:n = {
115 \bool_set_true:c {l_kivi_col_##1_bool}
116 \dim_set:cn {l_kivi_tab_##1_dim} {####1}
118 ##1 .default:n = true,
119 ##1 .initial:n = true,
120 ##1 / header .prop_put:c = {l_kivi_col_##1_prop},
121 ##1 / colspec .prop_put:c = {l_kivi_col_##1_prop},
127 \clist_new:N \g_kivi_pricingtable_col_clist
129 \keys_define:nn {kivi/PricingTable} {
131 \clist_gset:Nn \g_kivi_pricingtable_col_clist {#1}
132 \__kivi_initialize_columns:,
133 columns .initial:n = {pos, id, desc, amount, price, pricetotal},
136 % set default values for colwidth
137 \keys_set:nn {kivi/PricingTable} {
144 pos/header=\position,
145 id/header=\artikelnummer,
146 desc/header=\bezeichnung,
147 amount/header=\menge,
148 price/header=\einzelpreis,
149 pricetotal/header=\gesamtpreis,
150 price / colspec = Price,
151 pricetotal / colspec = Price ,
154 \dim_new:N \g_kivi_tabcolsep_dim
155 \dim_gset:Nn \g_kivi_tabcolsep_dim {.5\tabcolsep}
157 \cs_new:Nn \__kivi_calc_desc_column: {
158 \dim_zero:N \l_kivi_tab_desc_leftskip_dim
159 \dim_zero:N \l_kivi_tab_desc_dim
160 \bool_set_false:N \l_tmpa_bool
161 \clist_map_inline:Nn \g_kivi_pricingtable_col_clist {
162 \tl_if_eq:nnTF {##1} {desc} {
163 \dim_set:Nn \l_kivi_tab_desc_dim {
164 \textwidth-\l_kivi_tab_desc_leftskip_dim
166 \bool_set_true:N \l_tmpa_bool
169 \bool_if:NTF \l_tmpa_bool {
170 \bool_if:cT {l_kivi_col_##1_bool} {
171 \dim_sub:Nn \l_kivi_tab_desc_dim {
172 \dim_use:c {l_kivi_tab_##1_dim}+2\g_kivi_tabcolsep_dim
176 \bool_if:cT {l_kivi_col_##1_bool} {
177 \dim_add:Nn \l_kivi_tab_desc_leftskip_dim {
178 \dim_use:c {l_kivi_tab_##1_dim}+2\g_kivi_tabcolsep_dim
186 \newcolumntype{P}[1]{>{\raggedleft\arraybackslash}p{#1}<{\__kivi_tab_column_currency:}}
188 \RequirePackage{tcolorbox}
189 \tcbuselibrary{breakable, skins}
191 \tcb@new@skin{kivi@LT}{base@unbroken,%
192 frame~engine=empty,interior~titled~engine=empty,interior~engine=empty,segmentation~engine=empty,title~engine=empty,%
193 skin~first=kivi@LT@first,skin~middle=kivi@LT@middle,skin~last=kivi@LT@last,
194 underlay~first~and~middle={
195 \node[anchor=north] at (interior.north) {\csname box_use:c\endcsname {g_kivi_LT@head_box}};
196 \node[anchor=south] at (interior.south) {\csname box_use:c\endcsname {g_kivi_LT@foot_box}};
198 underlay~unbroken~and~last={
199 \node[anchor=north] at (interior.north) {\csname box_use:c\endcsname {g_kivi_LT@head_box}};
200 \node[anchor=south] at (interior.south) {\csname box_use:c\endcsname {g_kivi_LT@lastfoot_box}};
206 bottom=\box_ht:N \g_kivi_LT@foot_box+\box_dp:N \g_kivi_LT@foot_box + \aboverulesep,
207 top=\box_ht:N \g_kivi_LT@head_box+\box_dp:N \g_kivi_LT@head_box +\belowrulesep,
211 \tcb@new@skin{kivi@LT@first}{base@first,%
212 frame~engine=empty,interior~titled~engine=empty,interior~engine=empty,segmentation~engine=empty,title~engine=empty,%
213 skin~first=kivi@LT@first,skin~middle=kivi@LT@middle,skin~last=kivi@LT@middle,
216 \tcb@new@skin{kivi@LT@middle}{base@middle,%
217 frame~engine=empty,interior~titled~engine=empty,interior~engine=empty,segmentation~engine=empty,title~engine=empty,%
218 skin~first=kivi@LT@middle,skin~middle=kivi@LT@middle,skin~last=kivi@LT@middle,
221 \tcb@new@skin{kivi@LT@last}{base@last,%
222 frame~engine=empty,interior~titled~engine=empty,interior~engine=empty,segmentation~engine=empty,title~engine=empty,%
223 skin~first=kivi@LT@middle,skin~middle=kivi@LT@middle,skin~last=kivi@LT@last,
226 \tcbset{kivi@LT/.style={skin=kivi@LT}}%
230 \seq_new:N \l_kivi_PricingTable_seq
231 \seq_new:N \l_kivi_columns_seq
232 \seq_new:N \g_kivi_extraDescription_seq
233 \newcommand{\FakeTable}[1]{
235 \seq_set_split:Nnn \l_kivi_PricingTable_seq {\tabularnewline} {#1}
236 \seq_remove_all:Nn \l_kivi_PricingTable_seq {}
238 \setlength{\parskip}{\c_zero_dim}
239 \let\ExtraDescription\__kivi_addExtraDescription:n
240 \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
241 \seq_map_inline:Nn \l_kivi_PricingTable_seq {
242 \seq_set_split:Nnn \l_kivi_columns_seq {&} {##1}
243 \seq_gclear:N \g_kivi_extraDescription_seq
244 \exp_args:Nnx \use:n {\tabular[t]}\g_kivi_Pricing_colspec_tl
245 \seq_pop_left:NN \__l_FakeTable_columns_seq \l_tmpa_tl
246 \seq_item:Nn \l_kivi_columns_seq {\l_tmpa_tl}
247 \seq_map_inline:Nn \__l_FakeTable_columns_seq {
248 &\seq_item:Nn \l_kivi_columns_seq {####1}
251 \seq_if_empty:NTF \g_kivi_extraDescription_seq
255 \setlength{\hsize}{\dimexpr\l_kivi_tab_desc_dim+\l_kivi_tab_desc_leftskip_dim}
256 \setlength{\leftskip}{\l_kivi_tab_desc_leftskip_dim}
257 \usekomafont{extraDescription}
258 \seq_use:Nn \g_kivi_extraDescription_seq {\\}
267 \seq_new:N \__l_FakeTable_columns_seq
268 \cs_new:Nn \__kivi_setup_FakeTable: {
269 \seq_clear:N \__l_FakeTable_columns_seq
270 \int_zero:N \l_tmpa_int
271 \clist_map_inline:Nn \g_kivi_pricingtable_col_clist {
272 \int_incr:N \l_tmpa_int
273 \bool_if:cT {l_kivi_col_##1_bool} {\seq_put_right:Nx \__l_FakeTable_columns_seq {\int_use:N \l_tmpa_int}}
277 \tl_new:N \g_kivi_Pricing_colspec_tl
278 \tl_gset:Nn \g_kivi_Pricing_colspec_tl {
280 \bool_if:NT \l_kivi_col_pos_bool {p{\l_kivi_tab_pos_dim}}
281 \bool_if:NT \l_kivi_col_id_bool {p{\l_kivi_tab_id_dim}}
282 p{\l_kivi_tab_desc_dim}
283 \bool_if:NT \l_kivi_col_amount_bool {\exp_not:n {>{\raggedleft\arraybackslash}p{\l_kivi_tab_amount_dim}}}
284 \bool_if:NT \l_kivi_col_price_bool {\exp_not:n {>{\raggedleft\arraybackslash}p{\l_kivi_tab_price_dim}<{\__kivi_tab_column_currency:}}}
285 \bool_if:NT \l_kivi_col_pricetotal_bool {\exp_not:n {>{\raggedleft\arraybackslash}p{\l_kivi_tab_pricetotal_dim}<{\__kivi_tab_column_currency:}}}
289 \cs_new:Nn \__kivi_tab_column_currency: {\,\currency}
290 \def\tabcurrency{\__kivi_tab_column_currency:}
291 \cs_set:Nn \__kivi_tab_column_header_currency: {}
292 \cs_set_eq:NN \__kivi_tab_column_body_currency: \__kivi_tab_column_currency:
294 \clist_map_inline:nn {head, foot, firsthead, lastfoot} {%TODO reduce
295 \box_new:c {g_kivi_LT@#1_box}
298 \newkomafont{PricingTableHeader}{\bfseries}
300 \cs_new:Nn \__kivi_setup_LT_boxes: {
301 \__kivi_calc_desc_column:
302 \hbox_gset:Nn \g_kivi_LT@head_box {
303 \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
304 \exp_args:Nnx \use:n {\tabular[b]}\g_kivi_Pricing_colspec_tl
305 \__kivi_PricingTabular_header:
308 \hbox_gset:Nn \g_kivi_LT@foot_box {
310 \begin{tabular*}{\textwidth}{@{\extracolsep{\fill}}r@{}}
312 \strut\weiteraufnaechsterseite
316 \hbox_gset:Nn \g_kivi_LT@lastfoot_box {
317 \raisebox{\dimexpr\depth+\baselineskip}[0pt][0pt]{
318 \begin{tabular*}{\textwidth}{@{\extracolsep{\fill}}r@{}}
326 %Macht es sinn hier eine Variante zu machen, in der alle Spalten Belegbar sind?
327 \newenvironment{PricingTotal}{
330 \tabular[t]{@{}p{\dim_eval:n {\linewidth-\l_kivi_tab_pricetotal_dim-2\tabcolsep}}P{\l_kivi_tab_pricetotal_dim}@{}}
337 \newcommand*\ExtraDescription{
338 \PackageError{kiviletter}{The~command~\string\ExtraDescription\space~may~be~only~used~inside~the~\string\FakeTable\space~environment.}{See~documentation~for~details}
342 \cs_new:Nn \__kivi_addExtraDescription:n {\seq_gput_right:Nn \g_kivi_extraDescription_seq {#1}}
344 \newenvironment{PricingTabular}[1][]{
346 \dim_set:Nn \parskip {\c_zero_dim}
347 \tl_if_empty:nF {#1} {\keys_set:nn {kivi/PricingTable} {#1}}
348 \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
349 \__kivi_calc_desc_column:
350 \exp_args:Nx \longtable \g_kivi_Pricing_colspec_tl
352 \__kivi_PricingTabular_header:
355 \rlap{\makebox[\textwidth][r]{\weiteraufnaechsterseite}}\\
364 \cs_set:Nn \__kivi_PricingTabular_header: {
366 \cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_header_currency:
367 \bool_set_false:N \l_tmpa_bool
368 \clist_map_inline:Nn \g_kivi_pricingtable_col_clist {
369 \bool_if:cT {l_kivi_col_##1_bool} {
370 \bool_if:NT \l_tmpa_bool {&}
371 \bool_set_true:N \l_tmpa_bool
372 \usekomafont{PricingTableHeader}
373 \prop_item:cn {l_kivi_col_##1_prop} {header}
376 \cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_body_currency:
382 \keys_define:nn {kivi/SimpleTabular} {
383 colspec .tl_set:N =\l_kivi_SimpleTabular_colspec_tl,
384 colspec .initial:n = {rrX},
385 headline .tl_set:N = \l_kivi_SimpleTabular_headline_tl,
386 headline .initial:n = {\bfseries\position & \bfseries\menge & \bfseries\bezeichnung},
389 \newcommand*{\SetupSimpleTabular}[1]{\keys_set:nn {kivi/SimpleTabular} {#1}}
390 \newcommand*{\SetupPricingTabular}[1]{\keys_set:nn {kivi/PricingTable} {#1}}
392 \newenvironment{SimpleTabular}[1][]
394 \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}}}
395 \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
396 \dim_set:Nn \parskip {\c_zero_dim}
397 \tl_put_right:Nn \l_kivi_SimpleTabular_colspec_tl {@{}}
398 \tl_put_left:Nn \l_kivi_SimpleTabular_colspec_tl {@{}}
399 \exp_args:NnV \xltabular{\linewidth}\l_kivi_SimpleTabular_colspec_tl
401 \cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_header_currency:
402 \l_kivi_SimpleTabular_headline_tl
404 \noalign{\cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_body_currency:}
408 \rlap{\makebox[\textwidth][r]{\weiteraufnaechsterseite}}\\
414 \def\@currenvir{tabularx}
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 \dim_set:Nn \parskip {\c_zero_dim}
428 \PricingTabularBox\ignorespaces
429 }{\endPricingTabularBox}
431 \newtcolorbox{PricingTabularBox}{breakable,skin=kivi@LT}
436 \setkomavar{location}{
437 \Ifkomavarempty{transaction}{}{
439 \usekomavar{transaction}
443 \begin{tabularx}{\useplength{locwidth}}{@{}l<{:}>{\raggedleft\arraybackslash}X@{}}
444 \usekomavar*{date}&\usekomavar{date}\\
445 \Ifkomavarempty{myref}{}{
446 \usekomavar*{myref}&\usekomavar{myref}\\
448 \kundennummer&\usekomavar{customer}\\
449 \Ifkomavarempty{yourref}{}{
450 \usekomavar*{yourref}&\usekomavar{yourref}\\
452 \Ifkomavarempty{delivery}{}{
453 \usekomavar*{delivery}&\usekomavar{delivery}\\
455 \Ifkomavarempty{quote}{}{
456 \usekomavar*{quote}&\usekomavar{quote}\\
458 \Ifkomavarempty{orderID}{}{\auftragsnummer&\usekomavar{orderID}\\}
459 \Ifkomavarempty{projectID}{}{\projektnummer&\usekomavar{projectID}\\}
460 \ansprechpartner&\usekomavar{fromname}
461 \Ifkomavarempty{fromphone}{}{\\\textTelefon&\usekomavar{fromphone}}
462 \Ifkomavarempty{fromemail}{}{\\\textEmail&\usekomavar{fromemail}}
467 \ifdim\ht\shippingAddressBox>\z@
468 \@addtoplength{refvpos}{\dimexpr\ht\shippingAddressBox+\dp\shippingAddressBox}
469 \@addtoplength{refvpos}{4\baselineskip}%sep between address boxes
475 %Fallback for older KOMA-Script-Versions
476 \cs_if_exist:NF \Ifstr {\let\Ifstr\ifstr}
477 \cs_if_exist:NF \Ifkomavarempty {\let\Ifkomavarempty\ifkomavarempty}
479 %Definitionen für die insettings.tex
481 \newcommand*{\setupIdentpath}[1]{
482 \int_set:Nn \l_kivi_tmp_int {1}
483 \bool_set_true:N \l_kivi_tmp_bool
484 \bool_while_do:Nn \l_kivi_tmp_bool {
485 \file_if_exist:nTF {firma\int_use:N \l_kivi_tmp_int/ident.tex}
487 \exp_args:Nf \str_if_in:nnTF {#1} {Firma\int_use:N \l_kivi_tmp_int}
489 \newcommand*{\identpath}{firma\int_use:N \l_kivi_tmpa_int}
490 \bool_set_false:N \l_kivi_tmp_bool
492 {\int_incr:N \l_kivi_tmp_int}
495 \bool_set_false:N \l_kivi_tmp_bool
496 \newcommand*{\identpath}{firma}
501 \newcommand*{\setupCurrencyConfig}[2]{
502 \tl_new:N \g_kivi_currency_tl
503 \exp_args:Nf \str_if_in:nnT {#2} {USD} {\tl_gset:Nn \g_kivi_currency_tl {usd}}
504 \exp_args:Nf \str_if_in:nnT {#2} {CHF} {\tl_gset:Nn \g_kivi_currency_tl {chf}}
505 \exp_args:Nf \str_if_in:nnT {#2} {EUR} {\tl_gset:Nn \g_kivi_currency_tl {euro}}
506 \tl_if_empty:NT \g_kivi_currency_tl {
507 \tl_gset:Nn \g_kivi_currency_tl {default}
508 \edef \currency {\tl_to_str:N \lxcurrency}
510 \input{#1/\g_kivi_currency_tl _account.tex}
516 \renewcommand*{\raggedsignature}{\raggedright}
518 \newkomafont{extraDescription}{}