1 \NeedsTeXFormat{LaTeX2e}
2 \ProvidesPackage{kiviletter}[2020/04/24 Letter Layouts for Kivitendo]
5 \DeclareOption{reffields}{\@kivi@infoboxfalse}
6 \DeclareOption{infobox}{\@kivi@infoboxtrue}
12 \RequirePackage{expl3}
13 \RequirePackage{iftex}
14 \KOMAoptions{fontsize=12pt}
15 % Schriftart, Eingabelayout der Tastatur
17 \RequirePackage[utf8]{inputenc}% Nur notwendig, wenn Basis älter als TL2018
18 \RequirePackage[T1]{fontenc}
19 \RequirePackage{lmodern}
21 \RequirePackage{fontspec}
24 %\RequirePackage{xltabular}
25 \RequirePackage{tabularx}
26 \RequirePackage{longtable}
27 \RequirePackage{booktabs}
28 \RequirePackage{graphicx}
31 \RequirePackage{eurosym}
32 \DeclareUnicodeCharacter{20AC}{\euro}
35 \RequirePackage[fromlogo,fromalign=right,
36 firstfoot=false,%Für einheitliche Randeinstellungen
39 \LoadLetterOption{DIN}
41 \newkomavar{transaction}
42 \newkomavar[\lieferschein{}~\nr]{delivery}
43 \newkomavar[\angebot{}~\nr]{quote}
45 \newkomavar{projectID}
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}
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
61 \newsavebox{\shippingAddressBox}
66 hoffset=\useplength{toaddrhpos},
67 voffset=\dimexpr\useplength{toaddrvpos}+\useplength{toaddrheight}+4\baselineskip,%sep to shippingaddressbox
68 contents={\usebox\shippingAddressBox}
69 ]{kivitendo.shippingaddress}
73 \AtBeginLetter{\dim_gset:Nn \g_kivi_orig@textheight_dim {\textheight}}
76 \newpairofpagestyles{kivitendo.letter}{}
77 \renewcommand*{\letterpagestyle}{kivitendo.letter}
79 \DeclareNewPageStyleByLayers{kivitendo.letter.PricingTable}{
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,%
85 \DeclareNewPageStyleByLayers{kivitendo.letter.first}{
86 kivitendo.shippingaddress,
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,%
92 \setkomavar{backaddress}{\firma\ $\cdot$ \strasse\ $\cdot$ \ort}
94 \setkomavar{firsthead}{
96 \rlap{\usekomavar{fromlogo}}%
100 \@setplength{locwidth}{6cm}
103 \dim_new:N \l_kivi_tab_desc_leftskip_dim
107 \cs_new:Nn \__kivi_set_colwidth:nn {
108 \dim_set:cn {l_kivi_tab_#1_dim} {#2}
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}
116 \bool_new:c {l_kivi_col_##1_bool}
117 \dim_new:c {l_kivi_tab_##1_dim}
118 \keys_define:nn {kivi/PricingTable} {
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}
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},
135 \clist_new:N \g_kivi_pricingtable_col_clist
137 \keys_define:nn {kivi/PricingTable} {
139 \clist_gset:Nn \g_kivi_pricingtable_col_clist {#1}
140 \__kivi_initialize_columns:,
141 columns .initial:n = {pos, id, desc, amount, price, pricetotal},
144 % set default values for colwidth
145 \keys_set:nn {kivi/PricingTable} {
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 ,
162 \dim_new:N \g_kivi_tabcolsep_dim
163 \dim_gset:Nn \g_kivi_tabcolsep_dim {.5\tabcolsep}
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
174 \bool_set_true:N \l_tmpa_bool
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
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
194 \newcolumntype{P}[1]{>{\raggedleft\arraybackslash}p{#1}<{\__kivi_tab_column_currency:}}
196 \RequirePackage{tcolorbox}
197 \tcbuselibrary{breakable, skins}
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}};
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}};
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,
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,
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,
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,
234 \tcbset{kivi@LT/.style={skin=kivi@LT}}%
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]{
243 \seq_set_split:Nnn \l_kivi_PricingTable_seq {\tabularnewline} {#1}
244 \seq_remove_all:Nn \l_kivi_PricingTable_seq {}
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}
258 \seq_if_empty:NTF \g_kivi_extraDescription_seq
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 {\\}
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}}
284 \tl_new:N \g_kivi_Pricing_colspec_tl
285 \tl_gset:Nn \g_kivi_Pricing_colspec_tl {
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:}}}
296 \cs_new:Nn \__kivi_tab_column_currency: {\,\currency}
297 \def\tabcurrency{\__kivi_tab_column_currency:}
298 \cs_set:Nn \__kivi_tab_column_header_currency: {}
299 \cs_set_eq:NN \__kivi_tab_column_body_currency: \__kivi_tab_column_currency:
301 \clist_map_inline:nn {head, foot, firsthead, lastfoot} {%TODO reduce
302 \box_new:c {g_kivi_LT@#1_box}
305 \newkomafont{PricingTableHeader}{\bfseries}
307 \cs_new:Nn \__kivi_setup_LT_boxes: {
308 \__kivi_calc_desc_column:
309 \hbox_gset:Nn \g_kivi_LT@head_box {
310 \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
311 \exp_args:Nnx \use:n {\tabular[b]}\g_kivi_Pricing_colspec_tl
312 \__kivi_PricingTabular_header:
315 \hbox_gset:Nn \g_kivi_LT@foot_box {
317 \begin{tabular*}{\textwidth}{@{\extracolsep{\fill}}r@{}}
319 \strut\weiteraufnaechsterseite
323 \hbox_gset:Nn \g_kivi_LT@lastfoot_box {
324 \raisebox{\dimexpr\depth+\baselineskip}[0pt][0pt]{
325 \begin{tabular*}{\textwidth}{@{\extracolsep{\fill}}r@{}}
333 %Macht es sinn hier eine Variante zu machen, in der alle Spalten Belegbar sind?
334 \newenvironment{PricingTotal}{
336 \vspace{-\ht\strutbox}
338 \tabular[t]{@{}p{\dim_eval:n {\linewidth-\l_kivi_tab_pricetotal_dim-2\tabcolsep}}P{\l_kivi_tab_pricetotal_dim}@{}}
345 \newcommand*\ExtraDescription{
346 \PackageError{kiviletter}{The~command~\string\ExtraDescription\space~may~be~only~used~inside~the~\string\FakeTable\space~environment.}{See~documentation~for~details}
350 \cs_new:Nn \__kivi_addExtraDescription:n {\seq_gput_right:Nn \g_kivi_extraDescription_seq {#1}}
352 \newenvironment{PricingTabular}[1][]{
354 \tl_if_empty:nF {#1} {\keys_set:nn {kivi/PricingTable} {#1}}
355 \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
356 \__kivi_calc_desc_column:
357 \exp_args:Nx \longtable \g_kivi_Pricing_colspec_tl
359 \__kivi_PricingTabular_header:
362 \rlap{\makebox[\textwidth][r]{\weiteraufnaechsterseite}}\\
371 \cs_set:Nn \__kivi_PricingTabular_header: {
373 \cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_header_currency:
374 \bool_set_false:N \l_tmpa_bool
375 \clist_map_inline:Nn \g_kivi_pricingtable_col_clist {
376 \bool_if:cT {l_kivi_col_##1_bool} {
377 \bool_if:NT \l_tmpa_bool {&}
378 \bool_set_true:N \l_tmpa_bool
379 \usekomafont{PricingTableHeader}
380 \prop_item:cn {l_kivi_col_##1_prop} {header}
383 \cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_body_currency:
388 \RequirePackage{xltabular}
390 \keys_define:nn {kivi/SimpleTabular} {
391 colspec .tl_set:N =\l_kivi_SimpleTabular_colspec_tl,
392 colspec .initial:n = {rrX},
393 headline .tl_set:N = \l_kivi_SimpleTabular_headline_tl,
394 headline .initial:n = {\bfseries\position & \bfseries\menge & \bfseries\bezeichnung},
397 \newenvironment{SimpleTabular}[1][]
399 \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}}}
400 \setlength{\tabcolsep}{\g_kivi_tabcolsep_dim}
401 \tl_put_right:Nn \l_kivi_SimpleTabular_colspec_tl {@{}}
402 \tl_put_left:Nn \l_kivi_SimpleTabular_colspec_tl {@{}}
403 \exp_args:NnV \xltabular{\linewidth}\l_kivi_SimpleTabular_colspec_tl
405 \cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_header_currency:
406 \l_kivi_SimpleTabular_headline_tl
408 \noalign{\cs_gset_eq:NN \__kivi_tab_column_currency: \__kivi_tab_column_body_currency:}
412 \rlap{\makebox[\textwidth][r]{\weiteraufnaechsterseite}}\\
418 \def\@currenvir{tabularx}
422 %PricingTabular* kann automatisch spalten ignorieren
423 % \begin{PricingTabular*}[id=false]
424 % deaktiviert damit die Spalte der Produktnummer
425 % analog ist dies für pos, amount, price, pricetotal möglich.
426 % Die Spalte der Bezeichnung ist nicht deaktivierbar
427 \newenvironment{PricingTabular*}[1][]{
428 \tl_if_empty:nF {#1} {\keys_set:nn {kivi/PricingTable} {#1}}
429 \__kivi_setup_LT_boxes:
430 \__kivi_setup_FakeTable:
431 \PricingTabularBox\ignorespaces
432 }{\endPricingTabularBox}
434 \newtcolorbox{PricingTabularBox}{breakable,skin=kivi@LT}
439 \setkomavar{location}{
440 \ifkomavarempty{transaction}{}{
442 \usekomavar{transaction}
446 \begin{tabularx}{\useplength{locwidth}}{@{}l<{:}>{\raggedleft\arraybackslash}X@{}}
447 \usekomavar*{date}&\usekomavar{date}\\
448 \ifkomavarempty{myref}{}{
449 \usekomavar*{myref}&\usekomavar{myref}\\
451 \kundennummer&\usekomavar{customer}\\
452 \ifkomavarempty{yourref}{}{
453 \usekomavar*{yourref}&\usekomavar{yourref}\\
455 \ifkomavarempty{delivery}{}{
456 \usekomavar*{delivery}&\usekomavar{delivery}\\
458 \ifkomavarempty{quote}{}{
459 \usekomavar*{quote}&\usekomavar{quote}\\
461 \ifkomavarempty{orderID}{}{\auftragsnummer&\usekomavar{orderID}\\}
462 \ifkomavarempty{projectID}{}{\projektnummer&\usekomavar{projectID}\\}
463 \ansprechpartner&\usekomavar{fromname}
464 \ifkomavarempty{fromphone}{}{\\\textTelefon&\usekomavar{fromphone}}
465 \ifkomavarempty{fromemail}{}{\\\textEmail&\usekomavar{fromemail}}
470 \ifdim\ht\shippingAddressBox>\z@
471 \@addtoplength{refvpos}{\dimexpr\ht\shippingAddressBox+\dp\shippingAddressBox}
472 \@addtoplength{refvpos}{4\baselineskip}%sep between address boxes
480 \renewcommand*{\raggedsignature}{\raggedright}
482 \newkomafont{extraDescription}{}