@tonguetoquill/collection 0.6.1 → 0.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/package.json +1 -1
  2. package/quills/daf1206/0.1.0/Quill.yaml +52 -0
  3. package/quills/daf1206/0.1.0/assets/arimo-v35-latin-700.ttf +0 -0
  4. package/quills/daf1206/0.1.0/assets/arimo-v35-latin-700italic.ttf +0 -0
  5. package/quills/daf1206/0.1.0/assets/arimo-v35-latin-italic.ttf +0 -0
  6. package/quills/daf1206/0.1.0/assets/arimo-v35-latin-regular.ttf +0 -0
  7. package/quills/daf1206/0.1.0/example.md +19 -0
  8. package/quills/daf1206/0.1.0/packages/typst-daf1206/FIELDS.json +158 -0
  9. package/quills/daf1206/0.1.0/packages/typst-daf1206/form.pdf +0 -0
  10. package/quills/daf1206/0.1.0/packages/typst-daf1206/form.typ +36 -0
  11. package/quills/daf1206/0.1.0/packages/typst-daf1206/lib.typ +270 -0
  12. package/quills/daf1206/0.1.0/packages/typst-daf1206/page1.png +0 -0
  13. package/quills/daf1206/0.1.0/packages/typst-daf1206/page2.png +0 -0
  14. package/quills/daf1206/0.1.0/packages/typst-daf1206/typst.toml +7 -0
  15. package/quills/daf1206/0.1.0/plate.typ +28 -0
  16. package/quills/daf1206/assets/Figtree-Bold.ttf +0 -0
  17. package/quills/daf1206/assets/Figtree-Italic.ttf +0 -0
  18. package/quills/daf1206/assets/Figtree-Regular.ttf +0 -0
  19. package/quills/daf1206/assets/GNU General Public License.txt +340 -0
  20. package/quills/daf1206/assets/NimbusRomNo9L-Med.otf +0 -0
  21. package/quills/daf1206/assets/NimbusRomNo9L-MedIta.otf +0 -0
  22. package/quills/daf1206/assets/NimbusRomNo9L-Reg.otf +0 -0
  23. package/quills/daf1206/assets/NimbusRomNo9L-RegIta.otf +0 -0
  24. package/quills/daf4392/0.1.0/Quill.yaml +6 -11
  25. package/quills/daf4392/0.1.0/packages/daf4392page2_pkg/out/debug.typ +4 -0
  26. package/quills/daf4392/0.1.0/packages/daf4392page2_pkg/out/example.typ +4 -0
  27. package/quills/daf4392/0.1.0/packages/daf4392page2_pkg/page1.png +0 -0
  28. package/quills/daf4392/0.1.0/plate.typ +1 -1
  29. package/quills/daf4392/0.2.0/Quill.yaml +140 -0
  30. package/quills/daf4392/0.2.0/assets/arimo-v35-latin-700.ttf +0 -0
  31. package/quills/daf4392/0.2.0/assets/arimo-v35-latin-700italic.ttf +0 -0
  32. package/quills/daf4392/0.2.0/assets/arimo-v35-latin-italic.ttf +0 -0
  33. package/quills/daf4392/0.2.0/assets/arimo-v35-latin-regular.ttf +0 -0
  34. package/quills/daf4392/0.2.0/assets/page1.png +0 -0
  35. package/quills/daf4392/0.2.0/example.md +34 -0
  36. package/quills/daf4392/0.2.0/packages/daf4392page2_pkg/FIELDS.json +9 -0
  37. package/quills/daf4392/0.2.0/packages/daf4392page2_pkg/form.typ +14 -0
  38. package/quills/daf4392/0.2.0/packages/daf4392page2_pkg/lib.typ +227 -0
  39. package/quills/daf4392/0.2.0/packages/daf4392page2_pkg/typst.toml +7 -0
  40. package/quills/daf4392/0.2.0/plate.typ +68 -0
  41. package/templates/daf1206.md +31 -0
  42. package/templates/daf4392.md +2 -1
  43. package/templates/templates.json +7 -0
@@ -0,0 +1,227 @@
1
+ // Formalizer Engine – rendering engine
2
+ // Renders pixel-perfect PDF form replicas from a PyMuPDF-extracted schema.
3
+
4
+ /// Render a single field's content overlay.
5
+ ///
6
+ /// - field-type (str): normalised lowercase type
7
+ /// - value: user-supplied value for this field (or none)
8
+ /// - width (length): field width
9
+ /// - height (length): field height
10
+ /// - field (dictionary): raw field entry from the schema
11
+ #let render-field(field-type, value, width, height, field) = {
12
+ if field-type == "text" {
13
+ if value != none and str(value) != "" {
14
+ context {
15
+ let default-size = height * 0.6
16
+ let min-size = 6pt
17
+ let x-inset = 1.5pt
18
+ let m = measure(text(size: default-size, str(value)))
19
+ let scale = calc.min(1.0, (width - 2 * x-inset) / m.width)
20
+ let final-size = calc.max(min-size, default-size * scale)
21
+ box(
22
+ width: width,
23
+ height: height,
24
+ clip: true,
25
+ inset: (x: x-inset),
26
+ align(left + horizon, text(size: final-size, str(value))),
27
+ )
28
+ }
29
+ }
30
+ } else if field-type == "checkbox" {
31
+ if value == true {
32
+ box(
33
+ width: width,
34
+ height: height,
35
+ align(center + horizon, text(size: height * 0.8, "✓")),
36
+ )
37
+ }
38
+ } else if field-type == "radio" {
39
+ // value is true when this specific button is the selected one
40
+ // (resolved at group level before calling this helper)
41
+ if value == true {
42
+ let dot-r = calc.min(width, height) * 0.3
43
+ box(
44
+ width: width,
45
+ height: height,
46
+ align(center + horizon, circle(radius: dot-r, fill: black)),
47
+ )
48
+ }
49
+ } else if field-type == "combobox" or field-type == "listbox" {
50
+ let display = if value != none { str(value) } else { "" }
51
+ // Resolve export value → display label when options are present
52
+ if field.at("options", default: none) != none and value != none {
53
+ for opt in field.options {
54
+ if str(opt.at(0)) == str(value) {
55
+ display = str(opt.at(1))
56
+ }
57
+ }
58
+ }
59
+ if display != "" {
60
+ context {
61
+ let default-size = height * 0.6
62
+ let min-size = 6pt
63
+ let x-inset = 2pt
64
+ let m = measure(text(size: default-size, display))
65
+ let scale = calc.min(1.0, (width - 2 * x-inset) / m.width)
66
+ let final-size = calc.max(min-size, default-size * scale)
67
+ box(
68
+ width: width,
69
+ height: height,
70
+ clip: true,
71
+ inset: (x: x-inset),
72
+ align(left + horizon, text(size: final-size, display)),
73
+ )
74
+ }
75
+ }
76
+ }
77
+ }
78
+
79
+ /// Determine whether a single radio button should appear selected.
80
+ ///
81
+ /// Strategy:
82
+ /// 1. If the field carries an `export_value` key, match against it.
83
+ /// 2. Otherwise fall back to matching the 0-based index within the group
84
+ /// (stringified) against the supplied value.
85
+ #let radio-button-selected(field, group-value, index-in-group) = {
86
+ if group-value == none { return false }
87
+ let ev = field.at("export_value", default: none)
88
+ if ev != none {
89
+ return str(ev) == str(group-value)
90
+ }
91
+ // Fallback: match against the stringified index
92
+ str(index-in-group) == str(group-value)
93
+ }
94
+
95
+ /// Draw a debug overlay rectangle with a label for a field.
96
+ ///
97
+ /// - field-type (str): normalised lowercase type
98
+ /// - name (str): field name
99
+ /// - width (length): field width
100
+ /// - height (length): field height
101
+ #let debug-overlay(field-type, name, width, height) = {
102
+ let color = if field-type == "text" {
103
+ rgb(0, 0, 255, 40%)
104
+ } else if field-type == "checkbox" {
105
+ rgb(0, 128, 0, 40%)
106
+ } else if field-type == "radio" {
107
+ rgb(255, 165, 0, 40%)
108
+ } else if field-type == "combobox" {
109
+ rgb(128, 0, 128, 40%)
110
+ } else if field-type == "listbox" {
111
+ rgb(0, 128, 128, 40%)
112
+ } else {
113
+ rgb(128, 128, 128, 40%)
114
+ }
115
+
116
+ let stroke-color = if field-type == "text" {
117
+ rgb(0, 0, 255)
118
+ } else if field-type == "checkbox" {
119
+ rgb(0, 128, 0)
120
+ } else if field-type == "radio" {
121
+ rgb(255, 165, 0)
122
+ } else if field-type == "combobox" {
123
+ rgb(128, 0, 128)
124
+ } else if field-type == "listbox" {
125
+ rgb(0, 128, 128)
126
+ } else {
127
+ rgb(128, 128, 128)
128
+ }
129
+
130
+ // Insert zero-width spaces after _ and - so the label can wrap
131
+ let breakable-name = name.replace("_", "_\u{200B}").replace("-", "-\u{200B}")
132
+ let label = breakable-name + " [" + field-type + "]"
133
+
134
+ box(width: width, height: height, {
135
+ // Semi-transparent colored background
136
+ rect(width: 100%, height: 100%, fill: color, stroke: 0.5pt + stroke-color)
137
+ // Label in top-left — uses block so text wraps within field width
138
+ place(
139
+ top + left,
140
+ dx: 1pt,
141
+ dy: 1pt,
142
+ block(
143
+ width: calc.max(width - 2pt, 10pt),
144
+ fill: white,
145
+ inset: (x: 2pt, y: 1pt),
146
+ radius: 2pt,
147
+ stroke: 0.3pt + stroke-color,
148
+ breakable: false,
149
+ text(size: 5pt, fill: stroke-color, weight: "bold", label),
150
+ ),
151
+ )
152
+ })
153
+ }
154
+
155
+ /// Main entry point.
156
+ ///
157
+ /// - schema (dictionary): result of `json("FIELDS.json")`
158
+ /// - backgrounds (array): list of image paths / bytes, one per page
159
+ /// - values (dictionary): field name → value; omit to render blank
160
+ /// - debug (bool): when true, draw colored overlays on each field
161
+ #let render-form(schema: none, backgrounds: (), values: (:), debug: false) = {
162
+ assert(schema != none, message: "render-form: `schema` is required")
163
+ assert(backgrounds.len() > 0, message: "render-form: `backgrounds` is required")
164
+ let pages = schema.pages
165
+ let fields = schema.fields
166
+
167
+ // Track how many radio buttons we have seen per group so far
168
+ let radio-counters = (:)
169
+
170
+ for (i, page-info) in pages.enumerate() {
171
+ let page-num = i + 1
172
+ let page-fields = fields.filter(f => f.page == page-num)
173
+
174
+ let bg = backgrounds.at(i)
175
+ let pw = page-info.width * 1pt
176
+ let ph = page-info.height * 1pt
177
+
178
+ page(
179
+ width: pw,
180
+ height: ph,
181
+ margin: 0pt,
182
+ )[
183
+ // Background image (the original PDF page rasterised as PNG)
184
+ #place(top + left, image(bg, width: pw, height: ph, fit: "stretch"))
185
+
186
+ // Field overlays
187
+ #for field in page-fields.filter(f => ("text", "checkbox", "radio", "combobox", "listbox").contains(lower(
188
+ f.type,
189
+ ))) {
190
+ let x = field.bbox.at(0) * 1pt
191
+ let y = field.bbox.at(1) * 1pt
192
+ let w = (field.bbox.at(2) - field.bbox.at(0)) * 1pt
193
+ let h = (field.bbox.at(3) - field.bbox.at(1)) * 1pt
194
+
195
+ let field-type = lower(field.type)
196
+ let val = values.at(field.name, default: none)
197
+
198
+ // --- Radio: resolve group-level value to per-button boolean ---
199
+ if field-type == "radio" {
200
+ let group-name = field.name
201
+ let idx = radio-counters.at(group-name, default: 0)
202
+ radio-counters.insert(group-name, idx + 1)
203
+ val = radio-button-selected(field, val, idx)
204
+ }
205
+
206
+ place(
207
+ top + left,
208
+ dx: x,
209
+ dy: y,
210
+ render-field(field-type, val, w, h, field),
211
+ )
212
+
213
+ if debug {
214
+ place(
215
+ top + left,
216
+ dx: x,
217
+ dy: y,
218
+ debug-overlay(field-type, field.name, w, h),
219
+ )
220
+ }
221
+
222
+ // Zero-width fence to break PDF viewer text-selection grouping
223
+ place(top + left, dx: x, dy: y, text(size: 0.001pt, "\u{FEFF}"))
224
+ }
225
+ ]
226
+ }
227
+ }
@@ -0,0 +1,7 @@
1
+ [package]
2
+ name = "daf4392page2-pkg"
3
+ version = "0.1.0"
4
+ entrypoint = "form.typ"
5
+ authors = ["formalizer"]
6
+ license = "Apache-2.0"
7
+ description = "Fillable form package generated by formalizer"
@@ -0,0 +1,68 @@
1
+ #import "@local/quillmark-helper:0.1.0": data
2
+
3
+ #set page(width: 8.5in, height: 11in, margin: 0in)
4
+ #set text(font: "Arimo", size: 10pt)
5
+
6
+ // Background
7
+ #place(image("assets/page1.png", width: 100%, height: 100%))
8
+
9
+ #let tf(dx, dy, content) = place(dx: dx, dy: dy, content)
10
+ #let check(dx, dy, is_checked) = if is_checked { place(dx: dx, dy: dy, text(weight: "bold", size: 12pt)[✓]) }
11
+
12
+ // Mode of Transportation
13
+ #let mode = data.at("transportation_mode", default: "")
14
+ #check(40pt, 67pt, mode == "pmv")
15
+ #check(173pt, 67pt, mode == "airplane")
16
+ #check(243pt, 67pt, mode == "bus")
17
+ #check(286pt, 67pt, mode == "train")
18
+ #check(336pt, 67pt, mode == "motorcycle")
19
+ #check(411pt, 67pt, mode == "other")
20
+
21
+ // Fields
22
+ #tf(40pt, 100pt)[#data.at("departure_date", default: "")]
23
+ #tf(123pt, 100pt)[#data.at("final_destination", default: "")]
24
+
25
+ // Itinerary Rows (via CARDS)
26
+ #{
27
+ let row = 0
28
+ let dy-start = 160pt
29
+ let dy-step = 40pt
30
+ if "CARDS" in data {
31
+ for card in data.CARDS {
32
+ if card.CARD == "itinerary" and row < 10 {
33
+ let dy = dy-start + (row * dy-step)
34
+ tf(80pt, dy)[#card.at("date", default: "")]
35
+ tf(135pt, dy)[#card.at("departure_point", default: "")]
36
+ tf(295pt, dy)[#card.at("arrival_point", default: "")]
37
+ tf(450pt, dy)[#card.at("rest_length", default: "")]
38
+ tf(515pt, dy)[#card.at("mileage", default: "")]
39
+ row = row + 1
40
+ }
41
+ }
42
+ }
43
+ }
44
+
45
+ // Flight Info
46
+ #{
47
+ let d-flight = str(data.at("dept_flight_num", default: ""))
48
+ if d-flight != "" { tf(350pt, 545pt)[#text(weight: "bold")[Dept Flight:] #d-flight] }
49
+
50
+ let a-flight = str(data.at("arrival_flight_num", default: ""))
51
+ if a-flight != "" { tf(450pt, 545pt)[#text(weight: "bold")[Arr Flight:] #a-flight] }
52
+ }
53
+
54
+ // Notes
55
+ #tf(40pt, 565pt)[#block(width: 530pt)[#data.at("notes", default: "")]]
56
+
57
+ // Acknowledgements
58
+ #tf(40pt, 620pt)[#data.at("organization", default: "")]
59
+ #tf(500pt, 620pt)[#data.at("briefed_date", default: "")]
60
+
61
+ #tf(40pt, 650pt)[#data.at("briefee_name", default: "")]
62
+ #tf(270pt, 650pt)[#data.at("briefee_grade", default: "")]
63
+
64
+ #tf(40pt, 680pt)[#data.at("briefer_name", default: "")]
65
+ #tf(270pt, 680pt)[#data.at("briefer_grade", default: "")]
66
+
67
+ // Emergency Contact
68
+ #tf(40pt, 700pt)[#text(size: 10pt, weight: "bold")[EMERGENCY CONTACT:] #text(size: 10pt)[#data.at("emergency_contact_name", default: ""): #data.at("emergency_contact_phone", default: "")]]
@@ -0,0 +1,31 @@
1
+ ---
2
+ QUILL: daf1206@0.1.0
3
+
4
+ # EDIT: Award name (e.g., "AIR AND SPACE ACHIEVEMENT MEDAL")
5
+ AWARD: "AWARD NAME"
6
+ # EDIT: Category (If Applicable)
7
+ CATEGORY: ""
8
+ # EDIT: Award Period (e.g., "1 Jan 24 - 31 Dec 24")
9
+ AWARD_PERIOD: "1 Jan 2X - 31 Dec 2X"
10
+ # EDIT: Rank/Name of Nominee (First, Middle Initial, Last)
11
+ RANKNAME_OF_NOMINEE: "LAST, FIRST M."
12
+ # EDIT: MAJCOM/FLDCOM/FOA/DRU
13
+ MAJCOM_FLDCOM_FOA_OR_DRU: ""
14
+ # EDIT: DAFSC/Duty Title
15
+ DAFSCDUTY_TITLE: ""
16
+ # EDIT: Nominee's Telephone (DSN/Commercial)
17
+ NOMINEES_TELEPHONE: ""
18
+ # EDIT: Unit/Office Symbol/Street Address/Base/State/Zip Code
19
+ UNITOFFICE_SYMBOL: ""
20
+ # EDIT: Rank/Name of Unit Commander & Telephone
21
+ UNIT_COMMANDER: ""
22
+
23
+ # EDIT: Specific Accomplishments (Page 1)
24
+ ACCOMPLISHMENTS: |-
25
+ -
26
+ -
27
+
28
+ # EDIT: Specific Accomplishments (Continued - Page 2)
29
+ ACCOMPLISHMENTS_CONTINUED: |-
30
+ -
31
+ ---
@@ -11,7 +11,8 @@ briefee_name: "Doe, Jane"
11
11
  briefee_grade: "O-1"
12
12
  briefer_name: ""
13
13
  briefer_grade: ""
14
- emergency_contact: "John Doe: 999-999-9999"
14
+ emergency_contact_name: "John Doe"
15
+ emergency_contact_phone: "999-999-9999"
15
16
  ---
16
17
 
17
18
  ---
@@ -61,5 +61,12 @@
61
61
  "description": "Pre-Departure Safety Briefing (Page 2)",
62
62
  "file": "daf4392.md",
63
63
  "production": true
64
+ },
65
+ {
66
+ "id": "daf-form-1206",
67
+ "name": "DAF Form 1206",
68
+ "description": "Nomination for Award",
69
+ "file": "daf1206.md",
70
+ "production": true
64
71
  }
65
72
  ]