// set some styles #let style(it) = { show heading: set text(font: "New Computer Modern") show link: underline it } // transform md-similes to actual symbols #let smartypants(it) = { // smartypants and latex compatibility show "--": [#sym.dash.en] show "---": [#sym.dash.em] show "\&": [#sym.amp] it } // Choose the compiled language through cli by doing // // $ typst compile --input lang=de cv.typ // #let lang = { if "lang" in sys.inputs and sys.inputs.lang == "de" { "de" } else { "en" } } #let _columns_3(left_body, center_body, right_body) = { block[ #box(width: 1fr)[ #align(left)[#left_body] ] #box(width: 1fr)[ #align(center)[#center_body] ] #box(width: 1fr)[ #align(right)[#right_body] ] ] } #let header(about, columns: (1.5fr, 1fr, 1fr)) = { [= #about.fullname] let contact_fields = ( for c in about.contact { if "link" in c { ([#c.icon ~ #link(c.link)[#c.text]],) } else { ([#c.icon ~ #c.text],) } } ) grid( columns: columns, gutter: 5pt, ..contact_fields ) } #let subdued(body) = { block(inset: 5%, width: 85%, text(fill: luma(150), body)) } #let entry(item: ()) = { if "title" in item { [*#item.title.at(lang)*] } if "place" in item { if "title" in item { [, ] } [_#item.place.at(lang)_] } [#h(1fr)] if "date" in item { [ _#item.date.at(lang)_ \ ] } if "bullets" in item { for bullet in item.bullets { [- #bullet.at(lang)] } } if "publication" in item { subdued[#item.publication.at(lang) \ ] } if "abstract" in item { subdued[#item.abstract.at(lang) \ ] } } #let horizon_line() = { v(-3pt) line(length: 100%) v(-5pt) } #let section_header(title) = { [== #title] horizon_line() }; #let section(title: "Section", entries: (), body) = { section_header(title) if body == none or body == [] { for e in entries { entry(item: e) } } else { body } }; // TODO: make it _return_ the data, not display it on its own #let by_client(experience: ()) = { let by_client = (:) for item in experience { let client = item.place.at(lang) if client not in by_client { by_client.insert(client, ()) } by_client.at(client).push((title: item.title.at(lang), date: item.date.at(lang))) } for (client, jobs) in by_client { [*#client*:] for j in jobs { [- #j.title #h(1fr) #j.date] } } } #let by_experience_type(type: (), experience: ()) = { let by_ty = (:) for (id, desc) in type { let matching_exp_items = experience.filter(item => int(item.typeid) == int(id)) if matching_exp_items.len() == 0 { return } [=== _#desc.at(lang)_] by_client(experience: matching_exp_items) } } // Slightly re-styled entry with PLACE first and TITLE second #let education_entry(item: ()) = { assert( "place" in item and "title" in item and "date" in item, message: "Education items require place, program and date.", ) [*#item.place.at(lang)*, #item.title.at(lang) #h(1fr)] [ _#item.date.at(lang)_ \ ] } // Restyled entry with PLACE not emphasized like usual, and no date but an abstract #let thesis_entry(item: ()) = { assert("title" in item and "place" in item, message: "Thesis items require type and title.") [*#item.title.at(lang)* #item.place.at(lang) #h(1fr)] [#par(item.abstract.at(lang))] } // skill-specific entry, changing its style for sidebar #let skill_item(item: (), is_sidebar: false) = { let side_list(body) = if is_sidebar { list(body) } else { par(body) } for skill in item { side_list({ [*#skill.name.at(lang)*] if is_sidebar [\ ] else [ (] for (i, v) in skill.items.enumerate() { [#v.at(lang)] if i < skill.items.len() - 1 { [, ] } } if not is_sidebar [)] }) } }