intelligent-system-design-language 0.3.21 → 0.3.23

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 (116) hide show
  1. package/.claude/agents/langium-language-designer.md +38 -38
  2. package/.claude/agents/typescript-vscode-expert.md +29 -29
  3. package/.claude/agents/ui-ux-designer.md +36 -36
  4. package/.claude/settings.local.json +33 -33
  5. package/.idea/inspectionProfiles/Project_Default.xml +6 -6
  6. package/.idea/isdl.iml +13 -13
  7. package/.idea/modules.xml +8 -8
  8. package/.idea/vcs.xml +6 -6
  9. package/.idea/watcherTasks.xml +3 -3
  10. package/.vscodeignore +18 -18
  11. package/LICENSE +673 -673
  12. package/README.md +86 -86
  13. package/bin/cli.js +4 -4
  14. package/bin/lsp.js +8 -8
  15. package/out/_backgrounds.scss +91 -91
  16. package/out/_handlebars.scss +497 -497
  17. package/out/_isdlStyles.scss +1444 -1381
  18. package/out/_vuetifyOverrides.scss +425 -425
  19. package/out/_vuetifyStyles.scss +31957 -31957
  20. package/out/cli/components/_backgrounds.scss +91 -91
  21. package/out/cli/components/_handlebars.scss +497 -497
  22. package/out/cli/components/_isdlStyles.scss +1444 -1381
  23. package/out/cli/components/_vuetifyOverrides.scss +425 -425
  24. package/out/cli/components/_vuetifyStyles.scss +31957 -31957
  25. package/out/cli/components/active-effect-sheet-generator.js +453 -453
  26. package/out/cli/components/chat-card-generator.js +654 -651
  27. package/out/cli/components/chat-card-generator.js.map +1 -1
  28. package/out/cli/components/css-generator.js +4 -4
  29. package/out/cli/components/damage-roll-generator.js +160 -160
  30. package/out/cli/components/datamodel-generator.js +264 -257
  31. package/out/cli/components/datamodel-generator.js.map +1 -1
  32. package/out/cli/components/derived-data-generator.js +923 -923
  33. package/out/cli/components/hotbar-drop-hook-generator.js +82 -82
  34. package/out/cli/components/init-hook-generator.js +495 -495
  35. package/out/cli/components/language-generator.js +1 -1
  36. package/out/cli/components/language-generator.js.map +1 -1
  37. package/out/cli/components/measured-template-preview.js +221 -221
  38. package/out/cli/components/method-generator.js +979 -887
  39. package/out/cli/components/method-generator.js.map +1 -1
  40. package/out/cli/components/ready-hook-generator.js +404 -404
  41. package/out/cli/components/token-generator.js +116 -116
  42. package/out/cli/components/vue/base-components/vue-attribute.js +138 -138
  43. package/out/cli/components/vue/base-components/vue-boolean.js +64 -64
  44. package/out/cli/components/vue/base-components/vue-calculator.js +93 -93
  45. package/out/cli/components/vue/base-components/vue-damage-application.js +356 -356
  46. package/out/cli/components/vue/base-components/vue-damage-bonuses.js +165 -165
  47. package/out/cli/components/vue/base-components/vue-damage-resistances.js +196 -196
  48. package/out/cli/components/vue/base-components/vue-damage-track.js +121 -121
  49. package/out/cli/components/vue/base-components/vue-date-time.js +42 -42
  50. package/out/cli/components/vue/base-components/vue-dice.js +98 -98
  51. package/out/cli/components/vue/base-components/vue-die.js +73 -73
  52. package/out/cli/components/vue/base-components/vue-document-choice.js +149 -149
  53. package/out/cli/components/vue/base-components/vue-document-choices.js +179 -179
  54. package/out/cli/components/vue/base-components/vue-document-link.js +60 -60
  55. package/out/cli/components/vue/base-components/vue-extended-choice.js +88 -88
  56. package/out/cli/components/vue/base-components/vue-inventory.js +519 -519
  57. package/out/cli/components/vue/base-components/vue-macro-choice.js +138 -138
  58. package/out/cli/components/vue/base-components/vue-measured-template.js +530 -530
  59. package/out/cli/components/vue/base-components/vue-money.js +483 -483
  60. package/out/cli/components/vue/base-components/vue-number.js +174 -174
  61. package/out/cli/components/vue/base-components/vue-paperdoll.js +43 -43
  62. package/out/cli/components/vue/base-components/vue-parent-property-reference.js +76 -76
  63. package/out/cli/components/vue/base-components/vue-prosemirror.js +18 -18
  64. package/out/cli/components/vue/base-components/vue-resource.js +136 -136
  65. package/out/cli/components/vue/base-components/vue-roll-visualizer.js +286 -109
  66. package/out/cli/components/vue/base-components/vue-roll-visualizer.js.map +1 -1
  67. package/out/cli/components/vue/base-components/vue-self-property-reference.js +62 -62
  68. package/out/cli/components/vue/base-components/vue-string-choice.js +98 -98
  69. package/out/cli/components/vue/base-components/vue-string-choices.js +203 -203
  70. package/out/cli/components/vue/base-components/vue-string.js +60 -60
  71. package/out/cli/components/vue/base-components/vue-text-field.js +53 -53
  72. package/out/cli/components/vue/base-components/vue-tracker.js +431 -431
  73. package/out/cli/components/vue/vue-action-component-generator.js +64 -64
  74. package/out/cli/components/vue/vue-active-effect-sheet-generator.js +856 -856
  75. package/out/cli/components/vue/vue-datatable-sheet-class-generator.js +292 -292
  76. package/out/cli/components/vue/vue-datatable2-component-generator.js +824 -824
  77. package/out/cli/components/vue/vue-document-creation-app.js +121 -121
  78. package/out/cli/components/vue/vue-document-creation-sheet.js +94 -94
  79. package/out/cli/components/vue/vue-generator.js +40 -40
  80. package/out/cli/components/vue/vue-mixin.js +296 -296
  81. package/out/cli/components/vue/vue-pinned-datatable-component-generator.js +260 -260
  82. package/out/cli/components/vue/vue-prompt-generator.js +91 -76
  83. package/out/cli/components/vue/vue-prompt-generator.js.map +1 -1
  84. package/out/cli/components/vue/vue-prompt-sheet-class-generator.js +317 -317
  85. package/out/cli/components/vue/vue-sheet-application-generator.js +1177 -1167
  86. package/out/cli/components/vue/vue-sheet-application-generator.js.map +1 -1
  87. package/out/cli/components/vue/vue-sheet-class-generator.js +510 -510
  88. package/out/cli/generator.js +438 -433
  89. package/out/cli/generator.js.map +1 -1
  90. package/out/extension/github/githubAuthProvider.js +71 -29
  91. package/out/extension/github/githubAuthProvider.js.map +1 -1
  92. package/out/extension/github/githubGistManager.js +4 -3
  93. package/out/extension/github/githubGistManager.js.map +1 -1
  94. package/out/extension/github/githubManager.js +40 -38
  95. package/out/extension/github/githubManager.js.map +1 -1
  96. package/out/extension/github/githubQuickActions.js +120 -120
  97. package/out/extension/github/system-workflow.yml +47 -47
  98. package/out/extension/main.cjs +909 -532
  99. package/out/extension/main.cjs.map +3 -3
  100. package/out/extension/package.json +419 -419
  101. package/out/language/generated/ast.js +51 -2
  102. package/out/language/generated/ast.js.map +1 -1
  103. package/out/language/generated/grammar.js +14240 -13991
  104. package/out/language/generated/grammar.js.map +1 -1
  105. package/out/language/intelligent-system-design-language-validator.js +32 -2
  106. package/out/language/intelligent-system-design-language-validator.js.map +1 -1
  107. package/out/language/isdl-scope-provider.js +14 -1
  108. package/out/language/isdl-scope-provider.js.map +1 -1
  109. package/out/language/main.cjs +913 -569
  110. package/out/language/main.cjs.map +3 -3
  111. package/out/package.json +419 -419
  112. package/out/progressbar.min.js +6 -6
  113. package/out/styles.scss +762 -747
  114. package/out/test/validating/diagnostics.test.js +40 -0
  115. package/out/test/validating/diagnostics.test.js.map +1 -1
  116. package/package.json +419 -419
@@ -7,437 +7,437 @@ export default function generateTrackerComponent(destination) {
7
7
  if (!fs.existsSync(generatedFileDir)) {
8
8
  fs.mkdirSync(generatedFileDir, { recursive: true });
9
9
  }
10
- const fileNode = expandToNode `
11
- <script setup>
12
- import { ref, computed, inject, watchEffect } from "vue";
13
-
14
- const props = defineProps({
15
- label: String,
16
- systemPath: String,
17
- context: Object,
18
- visibility: String,
19
- editMode: Boolean,
20
- primaryColor: String,
21
- secondaryColor: String,
22
- tertiaryColor: String,
23
- trackerStyle: String,
24
- icon: String,
25
- hideMin: Boolean,
26
- disableMin: Boolean,
27
- disableValue: Boolean,
28
- disableMax: Boolean,
29
- segments: Number,
30
- isHealth: Boolean,
31
- isWounds: Boolean
32
- });
33
-
34
- const document = inject("rawDocument");
35
-
36
- // Vuetify's up/down stepper buttons update the model without firing a
37
- // native change event, so Foundry's submitOnChange form handler never
38
- // persists them. When the value changes while focus is NOT on a text
39
- // input (i.e. a stepper click, not typing), persist directly. Typing
40
- // still persists via the input's native change on blur/enter.
41
- // ('document' is the injected Foundry document; DOM access uses window.)
42
- const persistOnStep = (path, newValue) => {
43
- if (document && window.document.activeElement?.tagName !== 'INPUT') {
44
- document.update({ [path]: newValue });
45
- }
46
- };
47
-
48
- const isHidden = computed(() => {
49
- if (props.visibility === "hidden") {
50
- return true;
51
- }
52
- if (props.visibility === "gmOnly") {
53
- return !game.user.isGM;
54
- }
55
- if (props.visibility === "secret") {
56
- const isGm = game.user.isGM;
57
- const isOwner = document.getUserLevel(game.user) === CONST.DOCUMENT_OWNERSHIP_LEVELS.OWNER;
58
- return !isGm && !isOwner;
59
- }
60
- if (props.visibility === "edit") {
61
- return !props.editMode;
62
- }
63
- if (props.visibility === "play") {
64
- return props.editMode;
65
- }
66
-
67
- // Default to visible
68
- return false;
69
- });
70
-
71
- const isDisabled = (type) => {
72
- const disabledStates = ["readonly", "locked"];
73
- if (disabledStates.includes(props.visibility)) {
74
- return true;
75
- }
76
- if (props.visibility === "gmEdit") {
77
- const isGm = game.user.isGM;
78
- const isEditMode = props.editMode;
79
- return !isGm && !isEditMode;
80
- }
81
-
82
- if (props.visibility === "unlocked") {
83
- return false;
84
- }
85
-
86
- // Default to enabled while in editMode
87
- if (type == "value") return false;
88
- return !props.editMode;
89
- };
90
-
91
- const min = computed({
92
- get: () => foundry.utils.getProperty(props.context, props.systemPath + ".min") ?? 0,
93
- set: (newValue) => foundry.utils.setProperty(props.context, props.systemPath + ".min", newValue)
94
- });
95
-
96
- const value = computed({
97
- get: () => foundry.utils.getProperty(props.context, props.systemPath + ".value"),
98
- set: (newValue) => foundry.utils.setProperty(props.context, props.systemPath + ".value", newValue)
99
- });
100
-
101
- const temp = computed({
102
- get: () => foundry.utils.getProperty(props.context, props.systemPath + ".temp") ?? 0,
103
- set: (newValue) => foundry.utils.setProperty(props.context, props.systemPath + ".temp", newValue)
104
- });
105
-
106
- const max = computed({
107
- get: () => foundry.utils.getProperty(props.context, props.systemPath + ".max") ?? 10000,
108
- set: (newValue) => foundry.utils.setProperty(props.context, props.systemPath + ".max", newValue)
109
- });
110
-
111
- const barMax = computed(() => {
112
- const totalValue = Math.floor(value.value + (temp.value ?? 0));
113
- const floorMax = Math.floor(max.value);
114
- if (totalValue > floorMax) return totalValue;
115
- return floorMax;
116
- });
117
-
118
- const circleValue = computed(() => {
119
- // We need to calculate the percentage of the value in relation to the min to max range
120
- const percentage = (value.value - min.value) / (max.value - min.value);
121
- return Math.round(percentage * 100);
122
- });
123
-
124
- const refill = () => value.value = max.value;
125
- const empty = () => value.value = 0;
126
-
127
- const filledIcon = computed(() => {
128
- if (!props.icon) return "fa-solid fa-star";
129
- return "fa-solid " + props.icon;
130
- });
131
- const emptyIcon = computed(() => {
132
- if (!props.icon) return "fa-regular fa-star";
133
- return "fa-regular " + props.icon;
134
- });
135
-
136
- const mainColor = computed(() => {
137
- // If this is health, use a scale of red (0) to green (max). Wounds should be reverse. Otherwise, use primary.
138
-
139
- if (props.isHealth) {
140
- const pct = (value.value - (min.value ?? 0)) / ((max.value ?? 0) - (min.value ?? 0));
141
- // Use the number === 0 logic for health
142
- const red = Math.round(255 * (1 - (pct / 2)));
143
- const green = Math.round(255 * pct);
144
- const blue = 0;
145
- return \`rgb(\${red}, \${green}, \${blue})\`;
146
- }
147
- if (props.isWounds) {
148
- const pct = (value.value - (min.value ?? 0)) / ((max.value ?? 0) - (min.value ?? 0));
149
- // Use the else logic for wounds
150
- const red = Math.round(255 * (0.5 * pct));
151
- const green = Math.round(255 * (0.7 * pct));
152
- const blue = Math.round(255 * (0.5 + (pct / 2)));
153
- return \`rgb(\${red}, \${green}, \${blue})\`;
154
- }
155
- return props.primaryColor;
156
- });
157
-
158
- const expanded = ref(false);
159
-
160
- const expandIcon = computed(() => {
161
- return expanded.value ? "fa-solid fa-caret-up" : "fa-solid fa-caret-down";
162
- });
163
-
164
- const displayText = computed(() => {
165
- if (temp.value > 0) {
166
- return value.value + " / " + max.value + " (+" + temp.value + ")";
167
- }
168
- return value.value + " / " + max.value;
169
- });
170
-
171
- const circularText = computed(() => {
172
- return value.value + " / " + max.value;
173
- });
174
-
175
- const add = () => {
176
- if (props.disableValue || isDisabled('value')) return;
177
- if (value.value < max.value) {
178
- value.value++;
179
- }
180
- }
181
-
182
- const remove = () => {
183
- if (props.disableValue || isDisabled('value')) return;
184
- if (value.value > min.value) {
185
- value.value--;
186
- }
187
- }
188
-
189
- const getSegmentFill = (segmentIndex) => {
190
- const filled = value.value;
191
- const tempFilled = filled + (temp?.value ?? 0);
192
- const segmentStart = (segmentIndex - 1) * props.segments;
193
- const segmentEnd = segmentIndex * props.segments;
194
-
195
- const primaryFill = Math.min(Math.max(filled - segmentStart, 0), props.segments);
196
- const tempFill = Math.min(Math.max(tempFilled - segmentStart, 0), props.segments);
197
-
198
- const primaryPct = (primaryFill / props.segments) * 100;
199
- const tempPct = (tempFill / props.segments) * 100;
200
-
201
- const fill = \`linear-gradient(
202
- to right,
203
- \${mainColor.value} 0%,
204
- \${mainColor.value} \${primaryPct}%,
205
- \${props.tertiaryColor} \${primaryPct}%,
206
- \${props.tertiaryColor} \${tempPct}%,
207
- transparent \${tempPct}%
208
- )\`;
209
-
210
- const segmentLines = \`repeating-linear-gradient(
211
- to right,
212
- \${props.secondaryColor} 0,
213
- \${props.secondaryColor} 1px,
214
- transparent 1px,
215
- transparent calc(100% / \${props.segments})
216
- )\`;
217
-
218
- return \`\${segmentLines}, \${fill}\`;
219
- };
220
-
221
- const size = 100;
222
- const radius = size / 2 - 2;
223
-
224
- function describeSlice(index, total, r, center) {
225
- const anglePer = (2 * Math.PI) / total;
226
- const startAngle = anglePer * index - Math.PI / 2;
227
- const endAngle = startAngle + anglePer;
228
-
229
- const x1 = center + r * Math.cos(startAngle);
230
- const y1 = center + r * Math.sin(startAngle);
231
- const x2 = center + r * Math.cos(endAngle);
232
- const y2 = center + r * Math.sin(endAngle);
233
-
234
- const largeArcFlag = anglePer > Math.PI ? 1 : 0;
235
-
236
- return \`
237
- M \${center} \${center}
238
- L \${x1} \${y1}
239
- A \${r} \${r} 0 \${largeArcFlag} 1 \${x2} \${y2}
240
- Z
241
- \`;
242
- }
243
-
244
- const getLabel = computed(() => {
245
- const localized = game.i18n.localize(props.label);
246
- if (props.icon) {
247
- return \`<i class="fa-solid \${props.icon}"></i> \${localized}\`;
248
- }
249
- return localized;
250
- });
251
- </script>
252
-
253
- <template>
254
- <v-input v-model="value" :class="[trackerStyle, 'isdl-tracker']" v-if="!isHidden">
255
- <template #default>
256
- <v-field
257
- class="v-field--active"
258
- density="compact"
259
- variant="outlined"
260
- >
261
- <template #label>
262
- <span v-html="getLabel" />
263
- </template>
264
- <template #append-inner>
265
- <v-icon
266
- :icon="expandIcon"
267
- @click.stop="expanded = !expanded"
268
- class="v-select__menu-icon"
269
- />
270
- </template>
271
- <div class="tracker-content flexcol">
272
- <div class="d-flex tracker-inner-content">
273
- <v-progress-linear
274
- v-if="trackerStyle == 'bar'"
275
- :height="18"
276
- :color="mainColor"
277
- bg-color="#92aed9"
278
- rounded
279
- :model-value="value"
280
- min="0"
281
- :data-tooltip="displayText"
282
- :max="barMax"
283
- :buffer-value="value + temp"
284
- buffer-opacity="1"
285
- :buffer-color="tertiaryColor"
286
- >
287
- <template v-slot:default>
288
- {{ displayText }}
289
- </template>
290
- </v-progress-linear>
291
-
292
- <v-progress-circular
293
- v-if="trackerStyle == 'dial'"
294
- :model-value="circleValue"
295
- :rotate="360"
296
- :size="100"
297
- :width="15"
298
- :data-tooltip="displayText"
299
- :color="mainColor">
300
-
301
- <template v-slot:default> {{ circularText }} </template>
302
- </v-progress-circular>
303
-
304
- <div v-if="trackerStyle == 'icons'" class="d-flex flex-row" @click.stop="add" @contextmenu.prevent.stop="remove" style="overflow-x: auto; overflow-y: hidden;">
305
- <v-icon v-if="value > 0" v-for="i in Math.floor(value)" :key="i" :icon="filledIcon" :color="mainColor" style="margin-right: 0.25rem; width: 25px;" :data-tooltip="displayText" />
306
- <v-icon v-if="temp > 0" v-for="i in Math.floor(temp)" :key="i + value" :icon="filledIcon" :color="tertiaryColor" style="margin-right: 0.25rem; width: 25px;" :data-tooltip="displayText" />
307
- <v-icon v-if="Math.floor(max - value - temp) > 0" v-for="i in Math.floor(max - value - temp)" :key="i + temp + value" :icon="emptyIcon" :color="secondaryColor" style="margin-right: 0.25rem; width: 25px;" :data-tooltip="displayText" />
308
- </div>
309
-
310
- <div v-if="trackerStyle == 'slashes'" class="d-flex flex-row" @click.stop="add" @contextmenu.prevent.stop="remove" style="overflow-x: scroll; padding-left: 0.5rem; padding-right: 0.5rem;">
311
- <div
312
- v-for="i in barMax"
313
- :data-tooltip="displayText"
314
- :key="i"
315
- :style="{
316
- flex: 1,
317
- minWidth: '5px',
318
- flexShrink: 0,
319
- height: '30px',
320
- backgroundColor: i <= value ? mainColor : (i <= value + temp ? tertiaryColor : 'transparent'),
321
- border: i <= value ? 'none' : '2px solid ' + secondaryColor,
322
- transform: 'skewX(-20deg)',
323
- borderRadius: '2px',
324
- marginRight: '0.25rem'
325
- }"
326
- />
327
- </div>
328
-
329
- <div v-if="trackerStyle == 'segmented'" class="d-flex flex-row" @click.stop="add" @contextmenu.prevent.stop="remove">
330
- <div
331
- v-for="i in Math.ceil(barMax / segments)"
332
- :key="i"
333
- :data-tooltip="displayText"
334
- :style="{
335
- flex: 1,
336
- minWidth: '15px',
337
- flexShrink: 0,
338
- height: '30px',
339
- border: '2px solid ' + secondaryColor,
340
- borderRadius: '2px',
341
- marginRight: '0.25rem',
342
- background: getSegmentFill(i)
343
- }"
344
- />
345
- </div>
346
-
347
- <svg
348
- v-if="trackerStyle === 'clock'"
349
- :width="size"
350
- :height="size"
351
- :viewBox="\`0 0 \${size} \${size}\`"
352
- @click.stop="add"
353
- @contextmenu.prevent.stop="remove"
354
- :data-tooltip="displayText"
355
- style="width: auto;"
356
- >
357
- <g v-for="i in barMax" :key="i">
358
- <path
359
- :d="describeSlice(i - 1, barMax, radius, size / 2)"
360
- :fill="i <= value ? mainColor : (i <= value + temp ? tertiaryColor: 'transparent')"
361
- :stroke="secondaryColor"
362
- stroke-width="2"
363
- />
364
- </g>
365
- </svg>
366
-
367
- <v-number-input v-if="trackerStyle == 'plain'" :model-value="value" @update:model-value="(v) => { value = v; persistOnStep(systemPath + '.value', v); }" :name="systemPath" :min="min" :max="max" :disabled="isDisabled('value') || disableValue" type="number" variant="outlined" density="compact" hide-details="true"></v-number-input>
368
- </div>
369
- <v-expand-transition>
370
- <div v-show="expanded" style="margin-top: 1rem;">
371
- <div class="d-flex flex-row">
372
- <v-number-input
373
- :model-value="value"
374
- @update:model-value="(v) => { value = v; persistOnStep(systemPath + '.value', v); }"
375
- :name="systemPath + '.value'"
376
- label="Value"
377
- controlVariant="stacked"
378
- density="compact"
379
- variant="outlined"
380
- class="flex-grow-1 slim-number"
381
- style="min-width: 70px;"
382
- :hide-details="true"
383
- :tile="true"
384
- :disabled="isDisabled('value') || disableValue"
385
- />
386
- <v-number-input
387
- :model-value="temp"
388
- @update:model-value="(v) => { temp = v; persistOnStep(systemPath + '.temp', v); }"
389
- :name="systemPath + '.temp'"
390
- label="Temp"
391
- controlVariant="stacked"
392
- density="compact"
393
- variant="outlined"
394
- class="flex-grow-1"
395
- style="min-width: 70px; margin-right: 0.5rem;"
396
- :hide-details="true"
397
- :tile="true"
398
- :disabled="isDisabled('value') || disableValue"
399
- />
400
- <v-btn size="small" icon="fa-solid fa-battery-empty" @click="empty" :disabled="isDisabled('value') || disableValue" data-tooltip="Empty" :color="secondaryColor" />
401
- </div>
402
- <div class="d-flex flex-row" style="margin-top: 1rem;">
403
- <v-number-input
404
- :model-value="min"
405
- @update:model-value="(v) => { min = v; persistOnStep(systemPath + '.min', v); }"
406
- :name="systemPath + '.min'"
407
- label="Min"
408
- controlVariant="stacked"
409
- density="compact"
410
- variant="outlined"
411
- class="flex-grow-1 slim-number"
412
- style="min-width: 70px;"
413
- :hide-details="true"
414
- :tile="true"
415
- :disabled="isDisabled('min') || disableMin"
416
- v-if="!hideMin"
417
- />
418
- <v-number-input
419
- :model-value="max"
420
- @update:model-value="(v) => { max = v; persistOnStep(systemPath + '.max', v); }"
421
- :name="systemPath + '.max'"
422
- label="Max"
423
- controlVariant="stacked"
424
- density="compact"
425
- variant="outlined"
426
- class="flex-grow-1"
427
- style="min-width: 70px; margin-right: 0.5rem;"
428
- :hide-details="true"
429
- :tile="true"
430
- :disabled="isDisabled('max') || disableMax"
431
- />
432
- <v-btn size="small" icon="fa-solid fa-battery-full" @click="refill" :disabled="isDisabled('value') || disableValue" data-tooltip="Refill" :color="secondaryColor" />
433
- </div>
434
- </div>
435
- </v-expand-transition>
436
- </div>
437
- </v-field>
438
- </template>
439
- </v-input>
440
- </template>
10
+ const fileNode = expandToNode `
11
+ <script setup>
12
+ import { ref, computed, inject, watchEffect } from "vue";
13
+
14
+ const props = defineProps({
15
+ label: String,
16
+ systemPath: String,
17
+ context: Object,
18
+ visibility: String,
19
+ editMode: Boolean,
20
+ primaryColor: String,
21
+ secondaryColor: String,
22
+ tertiaryColor: String,
23
+ trackerStyle: String,
24
+ icon: String,
25
+ hideMin: Boolean,
26
+ disableMin: Boolean,
27
+ disableValue: Boolean,
28
+ disableMax: Boolean,
29
+ segments: Number,
30
+ isHealth: Boolean,
31
+ isWounds: Boolean
32
+ });
33
+
34
+ const document = inject("rawDocument");
35
+
36
+ // Vuetify's up/down stepper buttons update the model without firing a
37
+ // native change event, so Foundry's submitOnChange form handler never
38
+ // persists them. When the value changes while focus is NOT on a text
39
+ // input (i.e. a stepper click, not typing), persist directly. Typing
40
+ // still persists via the input's native change on blur/enter.
41
+ // ('document' is the injected Foundry document; DOM access uses window.)
42
+ const persistOnStep = (path, newValue) => {
43
+ if (document && window.document.activeElement?.tagName !== 'INPUT') {
44
+ document.update({ [path]: newValue });
45
+ }
46
+ };
47
+
48
+ const isHidden = computed(() => {
49
+ if (props.visibility === "hidden") {
50
+ return true;
51
+ }
52
+ if (props.visibility === "gmOnly") {
53
+ return !game.user.isGM;
54
+ }
55
+ if (props.visibility === "secret") {
56
+ const isGm = game.user.isGM;
57
+ const isOwner = document.getUserLevel(game.user) === CONST.DOCUMENT_OWNERSHIP_LEVELS.OWNER;
58
+ return !isGm && !isOwner;
59
+ }
60
+ if (props.visibility === "edit") {
61
+ return !props.editMode;
62
+ }
63
+ if (props.visibility === "play") {
64
+ return props.editMode;
65
+ }
66
+
67
+ // Default to visible
68
+ return false;
69
+ });
70
+
71
+ const isDisabled = (type) => {
72
+ const disabledStates = ["readonly", "locked"];
73
+ if (disabledStates.includes(props.visibility)) {
74
+ return true;
75
+ }
76
+ if (props.visibility === "gmEdit") {
77
+ const isGm = game.user.isGM;
78
+ const isEditMode = props.editMode;
79
+ return !isGm && !isEditMode;
80
+ }
81
+
82
+ if (props.visibility === "unlocked") {
83
+ return false;
84
+ }
85
+
86
+ // Default to enabled while in editMode
87
+ if (type == "value") return false;
88
+ return !props.editMode;
89
+ };
90
+
91
+ const min = computed({
92
+ get: () => foundry.utils.getProperty(props.context, props.systemPath + ".min") ?? 0,
93
+ set: (newValue) => foundry.utils.setProperty(props.context, props.systemPath + ".min", newValue)
94
+ });
95
+
96
+ const value = computed({
97
+ get: () => foundry.utils.getProperty(props.context, props.systemPath + ".value"),
98
+ set: (newValue) => foundry.utils.setProperty(props.context, props.systemPath + ".value", newValue)
99
+ });
100
+
101
+ const temp = computed({
102
+ get: () => foundry.utils.getProperty(props.context, props.systemPath + ".temp") ?? 0,
103
+ set: (newValue) => foundry.utils.setProperty(props.context, props.systemPath + ".temp", newValue)
104
+ });
105
+
106
+ const max = computed({
107
+ get: () => foundry.utils.getProperty(props.context, props.systemPath + ".max") ?? 10000,
108
+ set: (newValue) => foundry.utils.setProperty(props.context, props.systemPath + ".max", newValue)
109
+ });
110
+
111
+ const barMax = computed(() => {
112
+ const totalValue = Math.floor(value.value + (temp.value ?? 0));
113
+ const floorMax = Math.floor(max.value);
114
+ if (totalValue > floorMax) return totalValue;
115
+ return floorMax;
116
+ });
117
+
118
+ const circleValue = computed(() => {
119
+ // We need to calculate the percentage of the value in relation to the min to max range
120
+ const percentage = (value.value - min.value) / (max.value - min.value);
121
+ return Math.round(percentage * 100);
122
+ });
123
+
124
+ const refill = () => value.value = max.value;
125
+ const empty = () => value.value = 0;
126
+
127
+ const filledIcon = computed(() => {
128
+ if (!props.icon) return "fa-solid fa-star";
129
+ return "fa-solid " + props.icon;
130
+ });
131
+ const emptyIcon = computed(() => {
132
+ if (!props.icon) return "fa-regular fa-star";
133
+ return "fa-regular " + props.icon;
134
+ });
135
+
136
+ const mainColor = computed(() => {
137
+ // If this is health, use a scale of red (0) to green (max). Wounds should be reverse. Otherwise, use primary.
138
+
139
+ if (props.isHealth) {
140
+ const pct = (value.value - (min.value ?? 0)) / ((max.value ?? 0) - (min.value ?? 0));
141
+ // Use the number === 0 logic for health
142
+ const red = Math.round(255 * (1 - (pct / 2)));
143
+ const green = Math.round(255 * pct);
144
+ const blue = 0;
145
+ return \`rgb(\${red}, \${green}, \${blue})\`;
146
+ }
147
+ if (props.isWounds) {
148
+ const pct = (value.value - (min.value ?? 0)) / ((max.value ?? 0) - (min.value ?? 0));
149
+ // Use the else logic for wounds
150
+ const red = Math.round(255 * (0.5 * pct));
151
+ const green = Math.round(255 * (0.7 * pct));
152
+ const blue = Math.round(255 * (0.5 + (pct / 2)));
153
+ return \`rgb(\${red}, \${green}, \${blue})\`;
154
+ }
155
+ return props.primaryColor;
156
+ });
157
+
158
+ const expanded = ref(false);
159
+
160
+ const expandIcon = computed(() => {
161
+ return expanded.value ? "fa-solid fa-caret-up" : "fa-solid fa-caret-down";
162
+ });
163
+
164
+ const displayText = computed(() => {
165
+ if (temp.value > 0) {
166
+ return value.value + " / " + max.value + " (+" + temp.value + ")";
167
+ }
168
+ return value.value + " / " + max.value;
169
+ });
170
+
171
+ const circularText = computed(() => {
172
+ return value.value + " / " + max.value;
173
+ });
174
+
175
+ const add = () => {
176
+ if (props.disableValue || isDisabled('value')) return;
177
+ if (value.value < max.value) {
178
+ value.value++;
179
+ }
180
+ }
181
+
182
+ const remove = () => {
183
+ if (props.disableValue || isDisabled('value')) return;
184
+ if (value.value > min.value) {
185
+ value.value--;
186
+ }
187
+ }
188
+
189
+ const getSegmentFill = (segmentIndex) => {
190
+ const filled = value.value;
191
+ const tempFilled = filled + (temp?.value ?? 0);
192
+ const segmentStart = (segmentIndex - 1) * props.segments;
193
+ const segmentEnd = segmentIndex * props.segments;
194
+
195
+ const primaryFill = Math.min(Math.max(filled - segmentStart, 0), props.segments);
196
+ const tempFill = Math.min(Math.max(tempFilled - segmentStart, 0), props.segments);
197
+
198
+ const primaryPct = (primaryFill / props.segments) * 100;
199
+ const tempPct = (tempFill / props.segments) * 100;
200
+
201
+ const fill = \`linear-gradient(
202
+ to right,
203
+ \${mainColor.value} 0%,
204
+ \${mainColor.value} \${primaryPct}%,
205
+ \${props.tertiaryColor} \${primaryPct}%,
206
+ \${props.tertiaryColor} \${tempPct}%,
207
+ transparent \${tempPct}%
208
+ )\`;
209
+
210
+ const segmentLines = \`repeating-linear-gradient(
211
+ to right,
212
+ \${props.secondaryColor} 0,
213
+ \${props.secondaryColor} 1px,
214
+ transparent 1px,
215
+ transparent calc(100% / \${props.segments})
216
+ )\`;
217
+
218
+ return \`\${segmentLines}, \${fill}\`;
219
+ };
220
+
221
+ const size = 100;
222
+ const radius = size / 2 - 2;
223
+
224
+ function describeSlice(index, total, r, center) {
225
+ const anglePer = (2 * Math.PI) / total;
226
+ const startAngle = anglePer * index - Math.PI / 2;
227
+ const endAngle = startAngle + anglePer;
228
+
229
+ const x1 = center + r * Math.cos(startAngle);
230
+ const y1 = center + r * Math.sin(startAngle);
231
+ const x2 = center + r * Math.cos(endAngle);
232
+ const y2 = center + r * Math.sin(endAngle);
233
+
234
+ const largeArcFlag = anglePer > Math.PI ? 1 : 0;
235
+
236
+ return \`
237
+ M \${center} \${center}
238
+ L \${x1} \${y1}
239
+ A \${r} \${r} 0 \${largeArcFlag} 1 \${x2} \${y2}
240
+ Z
241
+ \`;
242
+ }
243
+
244
+ const getLabel = computed(() => {
245
+ const localized = game.i18n.localize(props.label);
246
+ if (props.icon) {
247
+ return \`<i class="fa-solid \${props.icon}"></i> \${localized}\`;
248
+ }
249
+ return localized;
250
+ });
251
+ </script>
252
+
253
+ <template>
254
+ <v-input v-model="value" :class="[trackerStyle, 'isdl-tracker']" v-if="!isHidden">
255
+ <template #default>
256
+ <v-field
257
+ class="v-field--active"
258
+ density="compact"
259
+ variant="outlined"
260
+ >
261
+ <template #label>
262
+ <span v-html="getLabel" />
263
+ </template>
264
+ <template #append-inner>
265
+ <v-icon
266
+ :icon="expandIcon"
267
+ @click.stop="expanded = !expanded"
268
+ class="v-select__menu-icon"
269
+ />
270
+ </template>
271
+ <div class="tracker-content flexcol">
272
+ <div class="d-flex tracker-inner-content">
273
+ <v-progress-linear
274
+ v-if="trackerStyle == 'bar'"
275
+ :height="18"
276
+ :color="mainColor"
277
+ bg-color="#92aed9"
278
+ rounded
279
+ :model-value="value"
280
+ min="0"
281
+ :data-tooltip="displayText"
282
+ :max="barMax"
283
+ :buffer-value="value + temp"
284
+ buffer-opacity="1"
285
+ :buffer-color="tertiaryColor"
286
+ >
287
+ <template v-slot:default>
288
+ {{ displayText }}
289
+ </template>
290
+ </v-progress-linear>
291
+
292
+ <v-progress-circular
293
+ v-if="trackerStyle == 'dial'"
294
+ :model-value="circleValue"
295
+ :rotate="360"
296
+ :size="100"
297
+ :width="15"
298
+ :data-tooltip="displayText"
299
+ :color="mainColor">
300
+
301
+ <template v-slot:default> {{ circularText }} </template>
302
+ </v-progress-circular>
303
+
304
+ <div v-if="trackerStyle == 'icons'" class="d-flex flex-row" @click.stop="add" @contextmenu.prevent.stop="remove" style="overflow-x: auto; overflow-y: hidden;">
305
+ <v-icon v-if="value > 0" v-for="i in Math.floor(value)" :key="i" :icon="filledIcon" :color="mainColor" style="margin-right: 0.25rem; width: 25px;" :data-tooltip="displayText" />
306
+ <v-icon v-if="temp > 0" v-for="i in Math.floor(temp)" :key="i + value" :icon="filledIcon" :color="tertiaryColor" style="margin-right: 0.25rem; width: 25px;" :data-tooltip="displayText" />
307
+ <v-icon v-if="Math.floor(max - value - temp) > 0" v-for="i in Math.floor(max - value - temp)" :key="i + temp + value" :icon="emptyIcon" :color="secondaryColor" style="margin-right: 0.25rem; width: 25px;" :data-tooltip="displayText" />
308
+ </div>
309
+
310
+ <div v-if="trackerStyle == 'slashes'" class="d-flex flex-row" @click.stop="add" @contextmenu.prevent.stop="remove" style="overflow-x: scroll; padding-left: 0.5rem; padding-right: 0.5rem;">
311
+ <div
312
+ v-for="i in barMax"
313
+ :data-tooltip="displayText"
314
+ :key="i"
315
+ :style="{
316
+ flex: 1,
317
+ minWidth: '5px',
318
+ flexShrink: 0,
319
+ height: '30px',
320
+ backgroundColor: i <= value ? mainColor : (i <= value + temp ? tertiaryColor : 'transparent'),
321
+ border: i <= value ? 'none' : '2px solid ' + secondaryColor,
322
+ transform: 'skewX(-20deg)',
323
+ borderRadius: '2px',
324
+ marginRight: '0.25rem'
325
+ }"
326
+ />
327
+ </div>
328
+
329
+ <div v-if="trackerStyle == 'segmented'" class="d-flex flex-row" @click.stop="add" @contextmenu.prevent.stop="remove">
330
+ <div
331
+ v-for="i in Math.ceil(barMax / segments)"
332
+ :key="i"
333
+ :data-tooltip="displayText"
334
+ :style="{
335
+ flex: 1,
336
+ minWidth: '15px',
337
+ flexShrink: 0,
338
+ height: '30px',
339
+ border: '2px solid ' + secondaryColor,
340
+ borderRadius: '2px',
341
+ marginRight: '0.25rem',
342
+ background: getSegmentFill(i)
343
+ }"
344
+ />
345
+ </div>
346
+
347
+ <svg
348
+ v-if="trackerStyle === 'clock'"
349
+ :width="size"
350
+ :height="size"
351
+ :viewBox="\`0 0 \${size} \${size}\`"
352
+ @click.stop="add"
353
+ @contextmenu.prevent.stop="remove"
354
+ :data-tooltip="displayText"
355
+ style="width: auto;"
356
+ >
357
+ <g v-for="i in barMax" :key="i">
358
+ <path
359
+ :d="describeSlice(i - 1, barMax, radius, size / 2)"
360
+ :fill="i <= value ? mainColor : (i <= value + temp ? tertiaryColor: 'transparent')"
361
+ :stroke="secondaryColor"
362
+ stroke-width="2"
363
+ />
364
+ </g>
365
+ </svg>
366
+
367
+ <v-number-input v-if="trackerStyle == 'plain'" :model-value="value" @update:model-value="(v) => { value = v; persistOnStep(systemPath + '.value', v); }" :name="systemPath" :min="min" :max="max" :disabled="isDisabled('value') || disableValue" type="number" variant="outlined" density="compact" hide-details="true"></v-number-input>
368
+ </div>
369
+ <v-expand-transition>
370
+ <div v-show="expanded" style="margin-top: 1rem;">
371
+ <div class="d-flex flex-row">
372
+ <v-number-input
373
+ :model-value="value"
374
+ @update:model-value="(v) => { value = v; persistOnStep(systemPath + '.value', v); }"
375
+ :name="systemPath + '.value'"
376
+ label="Value"
377
+ controlVariant="stacked"
378
+ density="compact"
379
+ variant="outlined"
380
+ class="flex-grow-1 slim-number"
381
+ style="min-width: 70px;"
382
+ :hide-details="true"
383
+ :tile="true"
384
+ :disabled="isDisabled('value') || disableValue"
385
+ />
386
+ <v-number-input
387
+ :model-value="temp"
388
+ @update:model-value="(v) => { temp = v; persistOnStep(systemPath + '.temp', v); }"
389
+ :name="systemPath + '.temp'"
390
+ label="Temp"
391
+ controlVariant="stacked"
392
+ density="compact"
393
+ variant="outlined"
394
+ class="flex-grow-1"
395
+ style="min-width: 70px; margin-right: 0.5rem;"
396
+ :hide-details="true"
397
+ :tile="true"
398
+ :disabled="isDisabled('value') || disableValue"
399
+ />
400
+ <v-btn size="small" icon="fa-solid fa-battery-empty" @click="empty" :disabled="isDisabled('value') || disableValue" data-tooltip="Empty" :color="secondaryColor" />
401
+ </div>
402
+ <div class="d-flex flex-row" style="margin-top: 1rem;">
403
+ <v-number-input
404
+ :model-value="min"
405
+ @update:model-value="(v) => { min = v; persistOnStep(systemPath + '.min', v); }"
406
+ :name="systemPath + '.min'"
407
+ label="Min"
408
+ controlVariant="stacked"
409
+ density="compact"
410
+ variant="outlined"
411
+ class="flex-grow-1 slim-number"
412
+ style="min-width: 70px;"
413
+ :hide-details="true"
414
+ :tile="true"
415
+ :disabled="isDisabled('min') || disableMin"
416
+ v-if="!hideMin"
417
+ />
418
+ <v-number-input
419
+ :model-value="max"
420
+ @update:model-value="(v) => { max = v; persistOnStep(systemPath + '.max', v); }"
421
+ :name="systemPath + '.max'"
422
+ label="Max"
423
+ controlVariant="stacked"
424
+ density="compact"
425
+ variant="outlined"
426
+ class="flex-grow-1"
427
+ style="min-width: 70px; margin-right: 0.5rem;"
428
+ :hide-details="true"
429
+ :tile="true"
430
+ :disabled="isDisabled('max') || disableMax"
431
+ />
432
+ <v-btn size="small" icon="fa-solid fa-battery-full" @click="refill" :disabled="isDisabled('value') || disableValue" data-tooltip="Refill" :color="secondaryColor" />
433
+ </div>
434
+ </div>
435
+ </v-expand-transition>
436
+ </div>
437
+ </v-field>
438
+ </template>
439
+ </v-input>
440
+ </template>
441
441
  `;
442
442
  fs.writeFileSync(generatedFilePath, toString(fileNode));
443
443
  }