-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmanual.typ
418 lines (352 loc) · 11 KB
/
manual.typ
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
#import "khw.typ": khw, problem as _problem, parts, todo
#set document(
title: [Manual for Kian's Homework Template],
author: "Kian Kasad",
date: datetime.today(),
)
// Show content with khw() so the necessary setup happens
#show: khw
// Override the problem() function to disable outlining
#let problem = _problem.with(outlined: false)
// Title page {{{1
#grid(
columns: (1fr,),
rows: (1fr, 1fr, 1fr, 1fr),
align: center,
[], [
#line(length: 100%)
#text(size: 24pt, weight: "bold")[Kian's Homework Template]
#link("/~https://github.com/kdkasad/typst-homework-template")
Kian Kasad
#datetime.today().display()
#line(length: 100%)
<cover>
]
)
#pagebreak()
// 1}}}
// Layout/format settings {{{1
#set par(justify: true)
#set page(
numbering: "1",
footer: [
#set text(9.5pt)
Kian's Homework Template Manual
#h(1fr)
#context counter(page).display()
],
footer-descent: 40% + 0pt,
)
#set heading(numbering: (..nums) => numbering("1.1", ..nums))
#show heading: (it) => {
block({
if it.numbering != none {
counter(heading).display(it.numbering)
h(1em)
}
it.body
})
}
#set raw(lang: "typc")
// 1}}}
// Function for typesetting argument list items {{{1
#let arg(name, types, description) = {
set terms(indent: 1em)
set raw(lang: none)
let body = {
show raw.where(lang: none): it => text(purple, it)
[(]
types.map(it => raw(it)).join([ | ])
[)]
h(0.5em)
description
}
terms.item(name, body)
}
// 1}}}
// Function to render examples {{{1
#let example(code, caption, render: true) = {
figure(
kind: "example",
supplement: [Example],
caption: caption,
placement: none,
table(
columns: (auto, 1fr,),
align: left + horizon,
fill: (x, y) => if x == 0 { gray.lighten(50%) },
inset: 8pt,
rotate(-90deg, reflow: true)[Markup],
code,
..if render {
(
rotate(-90deg, reflow: true)[Result],
eval(
// Remove imports so our custom problem() doesn't
// get shadowed.
code.text.replace(regex("#import\s+.*\n"), ""),
mode: "markup",
scope: (
parts: parts,
problem: problem,
todo: todo,
)
)
)
} else { () },
) // table
) // figure
}
// 1}}}
// Table of Contents
#outline()
// List of examples
#outline(
title: [Examples],
target: figure.where(kind: "example")
)
#pagebreak()
= Introduction
This project is a document template I created for homework
assignments. It is licensed under the BSD-3-Clause license and is
available at the URL on #link(<cover>)[the cover page].
= Using the template
To use the template, download the file `khw.typ` from this
project's repository and place it in your project/document's
working directory. If your project is a Git repository, you
can use a Git submodule to include the template repository
within your own.
== Document setup
To set up your document, import the `khw()` function and apply
it to all content using a show rule:
#example(render: false,
```typst
#import "khw.typ": khw
#show: khw.with(
title: [My Homework Assignment],
author: "Your Name",
)
```
)[Document setup code.]
The `title` and `author` fields are used to print a title
block on the first page, set the document's metadata using
```typc document()```, and print a header on every page except
the first.
The `khw()` function supports the following optional parameters:
#arg(`title`, ("content", "str", "none"))[
Document title. Defaults to `none`. If `none`, no title
block is printed and no document metadata is set.
]
#arg(`author`, ("content", "str", "none"))[
Document author. Defaults to `none`.
]
#arg(`date`, ("datetime",))[
Document creation date.
Defaults to ```typc datetime.today()```.
]
#arg(`course`, ("content", "string", "none"))[
Course for which the document is being written.
Defaults to `none`.
See also @sec:page-header for details on how the course name is
shortened for use in the page header.
]
The following options do not take effect immediately, but are
used to set default options for the ```typc problem()``` and
```typc parts()``` functions provided by this template.
#arg(`newpages`, ("bool",))[
Whether to start each problem on a new page.
Defaults to `false`.
]
#arg(`problem-prefix`, ("str",))[
The word to place before each problem
number. Defaults to "Problem".
]
#arg(`align-numbers`, ("alignment",))[
How to align problem numbers
within the problem header.
Default is ```typc center + horizon```.
]
#arg(`parts-numbering`, ("str", "function"))[
Numbering to use for
parts of a problem. Takes a value which can be used as the
argument to ```typc numbering()```.
Defaults to `"(a)"`, resulting in parts numbered (a), (b),
(c), etc.
Note that passing a function here will not work (currently).
This appears to be because Typst's `state` type cannot
handle storing functions, but I have not verified this.
]
== Page header <sec:page-header>
On every page except the first, a header will be displayed,
consisting of the course and title on the left and the author
on the right.
The course is preprocessed in a special way: if the `course`
parameter to the `khw()` function was a string, a _course
prefix separator_ and everything after it is removed.
The course prefix separator is anything that matches the
following regular expression: ```regex (:\s+|\s+-+\s+)```.
This allows you to print something like "CS 250: Computer
Architecture" in the title block while still keeping the
header short enough. In this case, only "CS 250" would appear
in the page header.
== Typesetting problems
Use the `problem()` function to typeset a problem. The
function takes a non-optional content argument which can be
used to specify the problem prompt/question.
#example(
```typ
#import "khw.typ": problem
#problem[
#lorem(25)
]
```
)[Typesetting a problem using `problem()`.]
If #link(<problem-arg-outlined>)[the `outlined` parameter] is
not disabled, a first-level heading is created for each
problem. If you need headings within problems, you should
start with second-level headings.
This also means that problems will appear in the table of
contents (if your document has one) and in the table of
contents embedded in the PDF's metadata (a.k.a. bookmarks).
The `problem()` function takes the following optional
arguments:
#arg(`prefix`, ("auto", "str", "content", "none"))[
Specifies the word to place before the problem number.
When `auto`, the value of the ```typc problem-prefix```
argument to the ```typc khw()``` function is used.
Defaults to `auto`.
#example(
```typ
#problem(prefix: "Exercise")[
#lorem(25)
]
```
)[Problem with prefix specified.]
]
#arg(`number`, ("auto", "str", "content"))[
Specifies the number of the problem. When `auto`, problems
are automatically numbered sequentially starting from 1.
Defaults to `auto`.
]
#arg(`points`, ("none", "str", "int", "float", "content"))[
Specifies the point value of the problem. Defaults to
`none`. See @points.
#example(
```typ
#problem(points: 5)[
#lorem(25)
]
```
)[Problem with points value specified.] <points>
]
#arg(`newpage`, ("auto", "bool"))[
Whether to insert a page break before the problem header. If
`auto`, the value of the ```typc newpages``` argument to the
```typc khw()``` function is used. Defaults to `auto`.
]
#arg(`align-number`, ("auto", "alignment"))[
How to align the problem number. When `auto`, the value of
the ```typc align-numbers``` argument to the ```typc khw()```
function is used.
Passing a value of ```typc center + top``` will center the
problem number right below the problem prefix text, so that
it always appears consistent no matter the height of the
problem prompts.
]
#arg(`outlined`, ("bool",))[
Whether this problem shows up in the outline. If `true`, an
invisible ```typc heading()``` is created for this problem,
making it act like a regular heading.
Defaults to `true`.
<problem-arg-outlined>
]
== Typesetting multi-part problems
For problems with multiple parts, there are two places you
might want to typeset the parts: in the prompt and in the
response/solution.
=== In the prompt
The content which makes up the prompt is displayed with
a `set` rule that numbers regular lists using the format
`"(a)"`. You can create a regular numbered list in the prompt
to typeset a multi-part prompt. See @prompt-parts.
#example(
```typ
#problem(points: 10)[
Explain how virtual memory speeds up the following operations:
+ Allocating zero-initialized pages.
+ Spawning child processes.
]
```
)[Multi-part problem prompt.] <prompt-parts>
=== In the response
Since the solution is more likely to include regular numbered
lists, I decided not to just use a `set` rule and to instead
make a function for typesetting parts. This also makes it
quite easy to split up multi-part problems into separate
source files.
Use the `parts()` function to typeset multiple parts in the
solution.
See @parts-function.
#example(
```typ
#parts[
Virtual memory allows for copy-on-write behavior, making
page allocation faster.
][
Only the page table needs to be copied when spawning a child
process, rather than the entire space of mapped memory.
]
```
)[Multi-part problem response.] <parts-function>
Sub-parts can be typeset using a normal numbered list and will
be numbered with lowercase Roman numerals. See @subparts.
#example(
```typ
#parts[
#lorem(5)
+ #lorem(5)
+ #lorem(5)
]
```
)[Solution with parts and sub-parts.] <subparts>
== To-do markers
You can use the `todo` variable to mark things you need to
come back to later.
#example(
```typ
#import "khw.typ": todo
Here is some text. #todo
```
)[The to-do marker.]
= Customization
In order to make more complicated customizations without
having to modify ```none khw.typ```, many of the elements
produced by the document template are labeled. You can use
`show` rules targeting the labeled elements to change their
appearance or override them completely.
#example(
```typ
#show <khw-problem-points>: smallcaps
#show <khw-problem-prompt>: set text(blue)
#show <khw-problem-number>: it => "#" + it
#problem(points: 2)[
#lorem(25)
]
```
)[Customizing the template using labels and `show` rules.]
== List of labels
/ `<khw-problem-block>`: Attached to the entire block
printed at the start of the problem.
/ `<khw-problem-prefix>`: Attached to the problem prefix text.
/ `<khw-problem-number>`: Attached to the problem number. Note
that this labels the text itself, not the grid cell which
contains the number (and controls its alignment).
Unfortunatelly, there is not yet a way in Typst to label the
cell itself.
/ `<khw-problem-points>`: Attached to the text which displays
a problem's point value.
/ `<khw-problem-prompt>`: Attached to the paragraph which
contains the problem prompt.
/ `<khw-part>`: Attached to each element of a `parts()` list.
This labels the `block()` containing the part's content.
// vim: foldmethod=marker foldmarker={{{,}}} sw=2 ts=2 et tw=62