@tonguetoquill/collection 0.8.0 → 0.8.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.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@tonguetoquill/collection",
3
- "version": "0.8.0",
3
+ "version": "0.8.1",
4
4
  "repository": {
5
5
  "type": "git",
6
6
  "url": "git+https://github.com/nibsbin/tonguetoquill-collection.git"
@@ -85,9 +85,10 @@ main:
85
85
  description: "Put in your Grade."
86
86
 
87
87
  emergency_contact:
88
- title: Emergency Contact
89
- type: string
90
- description: "Name and phone number of the emergency contact person at the bottom of the form."
88
+ title: Emergency Contacts
89
+ type: markdown
90
+ description: "Emergency contact (name and number)"
91
+
91
92
 
92
93
  cards:
93
94
  itinerary:
@@ -95,6 +96,7 @@ cards:
95
96
  description: "Each card represents one row in the travel itinerary table. Up to 10 rows are supported."
96
97
  ui:
97
98
  hide_body: true
99
+ default_title: Leg
98
100
  fields:
99
101
  date:
100
102
  title: Date
@@ -11,7 +11,7 @@ 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: "John Doe, 999-999-9999"
15
15
  ---
16
16
 
17
17
  ---
@@ -1,4 +1,4 @@
1
- #import "@local/quillmark-helper:0.1.0": data
1
+ #import "@local/quillmark-helper:0.1.0": data, eval-markup
2
2
 
3
3
  #set page(width: 8.5in, height: 11in, margin: 0in)
4
4
  #set text(font: "Arimo", size: 10pt)
@@ -65,4 +65,4 @@
65
65
  #tf(270pt, 680pt)[#data.at("briefer_grade", default: "")]
66
66
 
67
67
  // Emergency Contact
68
- #tf(40pt, 700pt)[#text(size: 10pt, weight: "bold")[EMERGENCY CONTACT:] #text(size: 10pt)[#data.at("emergency_contact", default: "")]]
68
+ #tf(40pt, 700pt)[#text(size: 10pt, weight: "bold")[EMERGENCY CONTACT:] #text(size: 10pt)[#eval-markup(data.at("emergency_contact", default:""))]]
@@ -1,5 +1,5 @@
1
1
  ---
2
- QUILL: daf4392
2
+ QUILL: daf4392@0.1
3
3
 
4
4
  transportation_mode: "other"
5
5
  departure_date: "XX Mar 26"
@@ -11,8 +11,7 @@ briefee_name: "Doe, Jane"
11
11
  briefee_grade: "O-1"
12
12
  briefer_name: ""
13
13
  briefer_grade: ""
14
- emergency_contact_name: "John Doe"
15
- emergency_contact_phone: "999-999-9999"
14
+ emergency_contact: "John Doe, 999-999-9999"
16
15
  ---
17
16
 
18
17
  ---
@@ -1,4 +0,0 @@
1
- // debug.typ (generated � renders form with debug overlays)
2
- #import "../form.typ": form
3
-
4
- #form(debug: true)
@@ -1,4 +0,0 @@
1
- // example.typ (edit this file to fill the form)
2
- #import "../form.typ": form
3
-
4
- #form()
@@ -1,141 +0,0 @@
1
- Quill:
2
- name: daf4392
3
- version: 0.2.0
4
- backend: typst
5
- plate_file: plate.typ
6
- example_file: example.md
7
- description: "DAF Form 4392 - Pre-Departure Safety Briefing (Page 2)"
8
-
9
- main:
10
- ui:
11
- hide_body: true
12
- fields:
13
- transportation_mode:
14
- title: Mode of Transportation
15
- type: string
16
- enum: [pmv, airplane, bus, train, motorcycle, other]
17
- required: true
18
- description: "Matches the check box."
19
-
20
-
21
- departure_date:
22
- title: Departure Date
23
- type: date
24
- required: true
25
- ui:
26
- compact: true
27
- description: "Date of initial departure."
28
-
29
- final_destination:
30
- title: Final Destination
31
- type: string
32
- required: true
33
- ui:
34
- compact: true
35
- description: "Final travel destination."
36
-
37
- dept_flight_num:
38
- title: Departure Flight Number
39
- type: string
40
- ui:
41
- compact: true
42
- description: "Departure flight number. (tenative or confirmed)"
43
-
44
- arrival_flight_num:
45
- title: Arrival Flight Number
46
- type: string
47
- ui:
48
- compact: true
49
- description: "Arrival flight number. (tenative or confirmed)"
50
-
51
- notes:
52
- title: Notes
53
- type: string
54
- description: "Additional notes or remarks for the travel."
55
-
56
- organization:
57
- title: Organization
58
- type: string
59
- required: true
60
- ui:
61
- compact: true
62
- description: "Unit / Organization."
63
-
64
- briefed_date:
65
- title: Date Briefed
66
- type: date
67
- required: true
68
- ui:
69
- compact: true
70
- description: "Put in today's date."
71
-
72
- briefee_name:
73
- title: Individual Receiving Brief (Name)
74
- type: string
75
- required: true
76
- ui:
77
- compact: true
78
- description: "Put in your Name (Last, First)."
79
-
80
- briefee_grade:
81
- title: Individual Receiving Brief (Grade)
82
- type: string
83
- ui:
84
- compact: true
85
- description: "Put in your Grade."
86
-
87
- emergency_contact_name:
88
- title: Emergency Contact Name
89
- type: string
90
- required: true
91
- description: "Put in your Emergency Contact Name."
92
-
93
- emergency_contact_phone:
94
- title: Emergency Contact Phone
95
- type: string
96
- required: true
97
- description: "Put in your Emergency Contact's Phone Number."
98
-
99
- cards:
100
- itinerary:
101
- title: Proposed Travel Itinerary
102
- description: "Each card represents one row in the travel itinerary table. Up to 10 rows are supported."
103
- ui:
104
- hide_body: true
105
- default_title: Leg
106
- fields:
107
- date:
108
- title: Date
109
- type: date
110
- ui:
111
- group: Itinerary Entry
112
- compact: true
113
- description: "Date for the itinerary row."
114
- departure_point:
115
- title: Departure Point
116
- type: string
117
- ui:
118
- group: Itinerary Entry
119
- compact: true
120
- description: "Point of departure."
121
- arrival_point:
122
- title: Arrival Point
123
- type: string
124
- ui:
125
- group: Itinerary Entry
126
- compact: true
127
- description: "Point of arrival."
128
- rest_length:
129
- title: Length of Rest Period
130
- type: string
131
- ui:
132
- group: Itinerary Entry
133
- compact: true
134
- description: "Duration of stay at arrival point"
135
- mileage:
136
- title: Approximate Mileage
137
- type: string
138
- ui:
139
- group: Itinerary Entry
140
- compact: true
141
- description: "Approximate mileage for this leg (leave blank if not applicable, e.g., flights)."
@@ -1,34 +0,0 @@
1
- ---
2
- QUILL: daf4392
3
-
4
- transportation_mode: "other"
5
- departure_date: "XX Mar 26"
6
- final_destination: "123 Sesame St, Fake City, MS"
7
- notes: "Visiting Donald Duck over 3 day weekend."
8
- organization: "333 TRS"
9
- briefed_date: "XX Mar 26"
10
- briefee_name: "Doe, Jane"
11
- briefee_grade: "O-1"
12
- briefer_name: ""
13
- briefer_grade: ""
14
- emergency_contact_name: "John Doe"
15
- emergency_contact_phone: "999-999-9999"
16
- ---
17
-
18
- ---
19
- CARD: itinerary
20
- date: "XX Mar"
21
- departure_point: "Biloxi, MS"
22
- arrival_point: "Fake City, MS"
23
- rest_length: "2 days"
24
- mileage: "250"
25
- ---
26
-
27
- ---
28
- CARD: itinerary
29
- date: "XX Mar"
30
- departure_point: "Fake City, MS"
31
- arrival_point: "Biloxi, MS"
32
- rest_length: "0"
33
- mileage: "0"
34
- ---
@@ -1,9 +0,0 @@
1
- {
2
- "pages": [
3
- {
4
- "width": 612.0,
5
- "height": 792.0
6
- }
7
- ],
8
- "fields": []
9
- }
@@ -1,14 +0,0 @@
1
- // form.typ (generated � do not edit)
2
- #import "lib.typ": render-form
3
-
4
- #let form(
5
- debug: false,
6
-
7
- ) = render-form(
8
- schema: json("FIELDS.json"),
9
- backgrounds: ("page1.png",),
10
- values: (
11
- ,
12
- ),
13
- debug: debug,
14
- )
@@ -1,227 +0,0 @@
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
- }
@@ -1,7 +0,0 @@
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"
@@ -1,68 +0,0 @@
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: "")]]