Skip to content

Commit 1c0dc7f

Browse files
committed
feat: provide more block themes
1 parent 117ef0c commit 1c0dc7f

File tree

4 files changed

+199
-46
lines changed

4 files changed

+199
-46
lines changed

README.md

+20-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
# ZJU-Project-Report-Template
22

3-
Typst is a new text markup language, considered the successor of LaTeX. This is yet another Zhejiang University project report template written in Typst, mainly designed for the course "Fundamentals of Data Structures (211C0020)" and "Digital Logic Design (211C0060)".
3+
Typst is a new text markup language, considered the successor of LaTeX. This is yet another Zhejiang University project report template written in Typst, mainly designed for the course "Fundamentals of Data Structures (211C0020)" and "Digital Logic Design (211C0060)".
44

55
**Reminder**: Due to requirements of some specific courses, disclosing personal information is prohibited and will be punished. So use this project at your own risk.
66

@@ -18,6 +18,23 @@ Currently, I have provided two themes, corresponding to common lab reports and p
1818

1919
### Fonts
2020

21-
I have configured default fallback options for serif, sans-serif, and monospace fonts, which you can modify using the `font_serif`, `font_sans_serif`, and `font_mono` arguments.
21+
I have configured default fallback options for serif, sans-serif, and monospace fonts, which you can set by modifying `font_serif`, `font_sans_serif`, and `font_mono` parameters.
2222

23-
However, for better rendering experiences, Chineses font families that support arbitrary levels of font weight, such as [`Source Han Serif SC`](https://github.com/adobe-fonts/source-han-serif/releases/tag/2.002R), is recommanded.
23+
However, for better rendering experiences, Chineses font families that support arbitrary levels of font weight, such as [Source Han Serif SC](https://github.com/adobe-fonts/source-han-serif/releases/tag/2.002R), is recommanded.
24+
25+
## Features
26+
27+
### Blocks
28+
29+
We provided a series of block functions to offer an experience that is similar to the callout feature in Obsidian. You can set the theme by modifying the `block_theme` parameter.
30+
31+
```typst
32+
#note(name: [Lagrange Inversion Theorem])[
33+
Given a power serie $F(x)$ and another power series $G(x)$ related by $F(G(x))=G(F(x))=x$, then the nth coefficient of $F(x)$ is
34+
$ [x^n] F(x) = 1/n [x^(-1)] 1/(G^n (x)). $
35+
]
36+
```
37+
38+
Some alias we have provided is as follows: `example`, `proof`, `abstract`, `summary`, `info`, `note`, `tip`, `hint`, `success`, `important`, `help`, `warning`, `attention`, `caution`, `failure`, `danger`, `error`, `bug`, `quote`, `cite`, `experiment`, `question`, `analysis`.
39+
40+
![](./screenshots/themed_blocks.png)

examples/blocks/testblocks.typ

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
#import "../../template.typ": *
2+
3+
#set page(height: 85em)
4+
#show: project.with(
5+
theme: "nocover",
6+
author: "memset0",
7+
)
8+
9+
#let render(theme_name) = [
10+
#state_block_theme.update(theme_name)
11+
= theme: #theme_name
12+
13+
#note(name: [Lagrange Inversion Theorem])[
14+
Given a power serie $F(x)$ and another power series $G(x)$ related by $F(G(x))=G(F(x))=x$, then the nth coefficient of $F(x)$ is
15+
$ [x^n] F(x) = 1/n [x^(-1)] 1/(G^n (x)). $
16+
]
17+
18+
#v(1em)
19+
]
20+
21+
#render("default")
22+
#render("boxed")
23+
#render("float")
24+
#render("thickness")
25+
#render("dashed")
26+
27+
28+
#state_block_theme.update("rounded")
29+
30+
= Otherwise, you will get an error block!
31+
32+
#state_block_theme.update("xxx")
33+
34+
#note[Hello, World!]

screenshots/themed_blocks.png

387 KB
Loading

template.typ

+145-43
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,12 @@
11
#import "@preview/tablex:0.0.8": tablex, colspanx, rowspanx, hlinex, vlinex, cellx
2+
#import "@preview/showybox:2.0.1": showybox
23

34
#let state_course = state("course", none)
45
#let state_author = state("author", none)
56
#let state_school_id = state("school_id", none)
67
#let state_date = state("date", none)
8+
#let state_theme = state("theme", none)
9+
#let state_block_theme = state("block_theme", none)
710

811
#let _underlined_cell(content, color: black) = {
912
tablex(
@@ -24,6 +27,7 @@
2427

2528
#let project(
2629
theme: "project",
30+
block_theme: "default",
2731
course: "<course>",
2832
title: "<title>",
2933
title_size: 3em,
@@ -50,9 +54,6 @@
5054
if (cover_image_size == none) {
5155
cover_image_size = 48%
5256
}
53-
if (language == none) {
54-
language = "cn"
55-
}
5657
} else if (theme == "project") {
5758
if (cover_image_size == none) {
5859
cover_image_size = 50%
@@ -89,6 +90,8 @@
8990
state_author.update(author)
9091
state_school_id.update(school_id)
9192
state_date.update(date)
93+
state_theme.update(theme)
94+
state_block_theme.update(block_theme)
9295

9396
// Cover Page
9497
if (theme == "lab") {
@@ -161,6 +164,8 @@
161164
)
162165
v(4fr)
163166
pagebreak()
167+
} else if (theme == "nocover") {
168+
// no cover page
164169
} else {
165170
set text(fill: red, size: 3em, weight: 900)
166171
align(center)[Theme not found!]
@@ -333,44 +338,141 @@
333338
)
334339
}
335340

336-
#let blockx(it, name: "", color: red, inset: 11pt) = block(
337-
below: 1em, stroke: 0.5pt + color, radius: 3pt,
338-
width: 100%, inset: inset
339-
)[
340-
#place(
341-
top + left,
342-
dy: -6pt - inset, // Account for inset of block
343-
dx: 8pt - inset,
344-
block(fill: white, inset: 2pt)[
345-
#set text(font: "Noto Sans", fill: color)
346-
#strong[#name]
347-
]
348-
)
349-
#set text(fill: color)
350-
#set par(first-line-indent: 0em)
351-
#it
352-
]
341+
#let blockx(
342+
it,
343+
name: "",
344+
color: red,
345+
theme: none,
346+
) = {
347+
context {
348+
let _theme = theme
349+
if (_theme == none) {
350+
_theme = state_block_theme.get()
351+
}
352+
if (_theme == "default") {
353+
let _inset = 0.8em
354+
let _color = color.darken(5%)
355+
v(0.2em)
356+
block(
357+
below: 1em,
358+
stroke: 0.5pt + _color,
359+
radius: 3pt,
360+
width: 100%,
361+
inset: _inset,
362+
)[
363+
#place(
364+
top + left,
365+
dy: -6pt - _inset, // Account for inset of block
366+
dx: 8pt - _inset,
367+
block(fill: white, inset: 2pt)[
368+
#set text(font: "Noto Sans", fill: _color)
369+
#name
370+
]
371+
)
372+
#set text(fill: _color)
373+
#set par(first-line-indent: 0em)
374+
#it
375+
]
376+
} else if (_theme == "boxed") {
377+
showybox(
378+
title: name,
379+
frame: (
380+
border-color: color,
381+
title-color: color.lighten(20%),
382+
body-color: color.lighten(95%),
383+
footer-color: color.lighten(80%)
384+
),
385+
it,
386+
)
387+
} else if (_theme == "float") {
388+
showybox(
389+
title-style: (
390+
boxed-style: (
391+
anchor: (
392+
x: center,
393+
y: horizon
394+
),
395+
radius: (top-left: 10pt, bottom-right: 10pt, rest: 0pt),
396+
)
397+
),
398+
frame: (
399+
title-color: color.darken(5%),
400+
body-color: color.lighten(95%),
401+
footer-color: color.lighten(60%),
402+
border-color: color.darken(20%),
403+
radius: (top-left: 10pt, bottom-right: 10pt, rest: 0pt)
404+
),
405+
title: name,
406+
[
407+
#it
408+
#v(0.25em)
409+
],
410+
)
411+
} else if (_theme == "thickness") {
412+
showybox(
413+
title-style: (
414+
color: color.darken(20%),
415+
sep-thickness: 0pt,
416+
align: center
417+
),
418+
frame: (
419+
title-color: color.lighten(85%),
420+
border-color: color.darken(20%),
421+
thickness: (left: 1pt),
422+
radius: 0pt
423+
),
424+
title: name,
425+
it,
426+
)
427+
} else if (_theme == "dashed") {
428+
showybox(
429+
title: name,
430+
frame: (
431+
border-color: color,
432+
title-color: color,
433+
radius: 0pt,
434+
thickness: 1pt,
435+
body-inset: 1em,
436+
dash: "densely-dash-dotted"
437+
),
438+
it,
439+
)
440+
} else {
441+
block(
442+
width: 100%,
443+
stroke: 0.5pt + red,
444+
inset: 1em,
445+
radius: 5pt,
446+
align(center)[
447+
#set text(fill: red, size: 1.5em)
448+
Unknown block theme!
449+
]
450+
)
451+
}
452+
}
453+
}
454+
353455

354-
#let example(it) = blockx(it, name: "Example", color: gray.darken(60%))
355-
#let proof(it) = blockx(it, name: "Proof", color: rgb(120, 120, 120))
356-
#let abstract(it) = blockx(it, name: "Abstract", color: rgb(0, 133, 143))
357-
#let summary(it) = blockx(it, name: "Summary", color: rgb(0, 133, 143))
358-
#let info(it) = blockx(it, name: "Info", color: rgb(68, 115, 218))
359-
#let note(it) = blockx(it, name: "Note", color: rgb(68, 115, 218))
360-
#let tip(it) = blockx(it, name: "Tip", color: rgb(0, 133, 91))
361-
#let hint(it) = blockx(it, name: "Hint", color: rgb(0, 133, 91))
362-
#let success(it) = blockx(it, name: "Success", color: rgb(62, 138, 0))
363-
#let important(it) = blockx(it, name: "Important", color: rgb(62, 138, 0))
364-
#let help(it) = blockx(it, name: "Help", color: rgb(153, 110, 36))
365-
#let warning(it) = blockx(it, name: "Warning", color: rgb(184, 95, 0))
366-
#let attention(it) = blockx(it, name: "Attention", color: rgb(216, 58, 49))
367-
#let caution(it) = blockx(it, name: "Caution", color: rgb(216, 58, 49))
368-
#let failure(it) = blockx(it, name: "Failure", color: rgb(216, 58, 49))
369-
#let danger(it) = blockx(it, name: "Danger", color: rgb(216, 58, 49))
370-
#let error(it) = blockx(it, name: "Error", color: rgb(216, 58, 49))
371-
#let bug(it) = blockx(it, name: "Bug", color: rgb(204, 51, 153))
372-
#let quote(it) = blockx(it, name: "Quote", color: rgb(132, 90, 231))
373-
#let cite(it) = blockx(it, name: "Cite", color: rgb(132, 90, 231))
374-
#let experiment(it) = blockx(it, name: "Experiment", color: rgb(132, 90, 231))
375-
#let question(it) = blockx(it, name: "Question", color: rgb(132, 90, 231))
376-
#let analysis(it) = blockx(it, name: "Analysis", color: rgb(0, 133, 91))
456+
#let example(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Example") }, color: gray.darken(60%))
457+
#let proof(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Proof") }, color: rgb(120, 120, 120))
458+
#let abstract(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Abstract") }, color: rgb(0, 133, 143))
459+
#let summary(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Summary") }, color: rgb(0, 133, 143))
460+
#let info(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Info") }, color: rgb(68, 115, 218))
461+
#let note(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Note") }, color: rgb(68, 115, 218))
462+
#let tip(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Tip") }, color: rgb(0, 133, 91))
463+
#let hint(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Hint") }, color: rgb(0, 133, 91))
464+
#let success(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Success") }, color: rgb(62, 138, 0))
465+
#let important(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Important") }, color: rgb(62, 138, 0))
466+
#let help(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Help") }, color: rgb(153, 110, 36))
467+
#let warning(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Warning") }, color: rgb(184, 95, 0))
468+
#let attention(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Attention") }, color: rgb(216, 58, 49))
469+
#let caution(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Caution") }, color: rgb(216, 58, 49))
470+
#let failure(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Failure") }, color: rgb(216, 58, 49))
471+
#let danger(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Danger") }, color: rgb(216, 58, 49))
472+
#let error(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Error") }, color: rgb(216, 58, 49))
473+
#let bug(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Bug") }, color: rgb(204, 51, 153))
474+
#let quote(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Quote") }, color: rgb(132, 90, 231))
475+
#let cite(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Cite") }, color: rgb(132, 90, 231))
476+
#let experiment(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Experiment") }, color: rgb(132, 90, 231))
477+
#let question(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Question") }, color: rgb(132, 90, 231))
478+
#let analysis(it, name: none) = blockx(it, name: if (name != none) { name } else { strong("Analysis") }, color: rgb(0, 133, 91))

0 commit comments

Comments
 (0)