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,362 +7,362 @@ export default function generateDamageApplicationComponent(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, watch } from "vue";
13
-
14
- const props = defineProps({
15
- rollData: {
16
- type: Object,
17
- required: true,
18
- default: () => ({
19
- value: 0,
20
- isDamageRoll: false,
21
- damageType: null,
22
- damageColor: null,
23
- damageIcon: null,
24
- metadata: {}
25
- })
26
- },
27
- visible: {
28
- type: Boolean,
29
- default: false
30
- }
31
- });
32
-
33
- const emit = defineEmits(['close', 'apply']);
34
-
35
- // Internal state
36
- const selectedTargetType = ref('selected');
37
- const selectedMultiplier = ref(1);
38
- const showDialog = ref(false);
39
-
40
- // Available multipliers
41
- const multipliers = [
42
- { value: 0.25, label: '¼x' },
43
- { value: 0.5, label: '½x' },
44
- { value: 1, label: '1x' },
45
- { value: 1.5, label: '1.5x' },
46
- { value: 2, label: '2x' },
47
- { value: 3, label: '3x' },
48
- { value: 4, label: '4x' }
49
- ];
50
-
51
- // Target type options
52
- const targetTypes = [
53
- { value: 'targeted', label: 'Targeted', icon: 'fa-solid fa-bullseye' },
54
- { value: 'selected', label: 'Selected', icon: 'fa-solid fa-expand' }
55
- ];
56
-
57
- // Watch for visibility changes
58
- watch(() => props.visible, (newValue) => {
59
- showDialog.value = newValue;
60
- });
61
-
62
- // Computed values
63
- const finalDamage = computed(() => {
64
- return Math.floor(props.rollData.value * selectedMultiplier.value);
65
- });
66
-
67
- const targets = computed(() => {
68
- const targetType = selectedTargetType.value;
69
- const targetList = targetType === 'targeted'
70
- ? [...game.user.targets]
71
- : (canvas?.tokens?.controlled ?? []);
72
-
73
- return targetList;
74
- });
75
-
76
- const hasTargets = computed(() => targets.value.length > 0);
77
-
78
- const damageTypeDisplay = computed(() => {
79
- if (!props.rollData.isDamageRoll || !props.rollData.damageType) {
80
- return null;
81
- }
82
-
83
- return {
84
- type: props.rollData.damageType,
85
- color: props.rollData.damageColor || '#ffffff',
86
- icon: props.rollData.damageIcon
87
- };
88
- });
89
-
90
- // Methods
91
- const closeDialog = () => {
92
- showDialog.value = false;
93
- emit('close');
94
- };
95
-
96
- const applyDamage = (type = 'damage') => {
97
- if (!hasTargets.value) {
98
- ui.notifications.warn(game.i18n.localize(\`NOTIFICATIONS.\${selectedTargetType.value === 'targeted' ? 'NoTokenTargeted' : 'NoTokenSelected'}\`));
99
- return;
100
- }
101
-
102
- const context = {
103
- amount: type === 'healing' ? -finalDamage.value : finalDamage.value,
104
- multiplier: selectedMultiplier.value,
105
- targetType: selectedTargetType.value,
106
- damageType: props.rollData.isDamageRoll ? props.rollData.damageType : null,
107
- damageMetadata: props.rollData.isDamageRoll ? props.rollData.metadata : {},
108
- color: props.rollData.isDamageRoll ? props.rollData.damageColor : null,
109
- icon: props.rollData.isDamageRoll ? props.rollData.damageIcon : null,
110
- isDamageRoll: props.rollData.isDamageRoll
111
- };
112
-
113
- emit('apply', {
114
- type,
115
- targets: targets.value,
116
- context
117
- });
118
-
119
- closeDialog();
120
- };
121
-
122
- // Expose data for the Application sheet
123
- const getApplicationData = () => {
124
- return {
125
- targets: targets.value,
126
- context: {
127
- amount: finalDamage.value,
128
- multiplier: selectedMultiplier.value,
129
- targetType: selectedTargetType.value
130
- }
131
- };
132
- };
133
-
134
- // Expose the method to the parent component
135
- const rawSheet = inject("rawSheet", null);
136
- if (rawSheet && typeof rawSheet === 'object') {
137
- rawSheet.getApplicationData = getApplicationData;
138
- }
139
-
140
- const applyHealing = () => applyDamage('healing');
141
- const applyTemp = () => applyDamage('temp');
142
-
143
- const selectTargetType = (type) => {
144
- selectedTargetType.value = type;
145
- game.settings.set(game.system.id, 'userTargetDamageApplicationType', type);
146
- };
147
-
148
- const selectMultiplier = (multiplier) => {
149
- selectedMultiplier.value = multiplier;
150
- };
151
-
152
- // Initialize from settings
153
- const initializeSettings = () => {
154
- const allowTargeting = game.settings.get(game.system.id, 'allowTargetDamageApplication');
155
- let targetType = game.settings.get(game.system.id, 'userTargetDamageApplicationType');
156
-
157
- if (!allowTargeting && targetType !== 'selected') {
158
- targetType = 'selected';
159
- game.settings.set(game.system.id, 'userTargetDamageApplicationType', 'selected');
160
- }
161
-
162
- selectedTargetType.value = targetType;
163
- };
164
-
165
- // Initialize on mount
166
- initializeSettings();
167
- </script>
168
-
169
- <template>
170
- <v-dialog
171
- v-model="showDialog"
172
- max-width="500"
173
- persistent
174
- @click:outside="closeDialog"
175
- >
176
- <v-card>
177
- <v-card-title class="d-flex align-center">
178
- <v-icon class="mr-2">mdi-sword-cross</v-icon>
179
- Apply Damage
180
- <v-spacer></v-spacer>
181
- <v-btn
182
- icon="mdi-close"
183
- variant="text"
184
- size="small"
185
- @click="closeDialog"
186
- ></v-btn>
187
- </v-card-title>
188
-
189
- <v-card-text>
190
- <!-- Damage Info -->
191
- <div class="damage-info mb-4">
192
- <div class="d-flex align-center mb-2">
193
- <span class="text-h6 mr-2">{{ props.rollData.value }}</span>
194
- <span v-if="damageTypeDisplay" class="damage-type-badge">
195
- <v-icon
196
- v-if="damageTypeDisplay.icon"
197
- :icon="damageTypeDisplay.icon"
198
- :color="damageTypeDisplay.color"
199
- size="small"
200
- class="mr-1"
201
- ></v-icon>
202
- <span :style="{ color: damageTypeDisplay.color }">
203
- {{ damageTypeDisplay.type }}
204
- </span>
205
- </span>
206
- </div>
207
-
208
- <!-- Damage metadata -->
209
- <div v-if="props.rollData.isDamageRoll && Object.keys(props.rollData.metadata).length > 0"
210
- class="damage-metadata">
211
- <v-chip
212
- v-for="(value, key) in props.rollData.metadata"
213
- :key="key"
214
- size="small"
215
- variant="outlined"
216
- class="mr-1 mb-1"
217
- >
218
- {{ key }}: {{ value }}
219
- </v-chip>
220
- </div>
221
- </div>
222
-
223
- <!-- Target Selection -->
224
- <div class="target-selection mb-4">
225
- <h4 class="mb-2">Target Type</h4>
226
- <v-btn-toggle
227
- v-model="selectedTargetType"
228
- mandatory
229
- variant="outlined"
230
- divided
231
- >
232
- <v-btn
233
- v-for="targetType in targetTypes"
234
- :key="targetType.value"
235
- :value="targetType.value"
236
- @click="selectTargetType(targetType.value)"
237
- >
238
- <v-icon :icon="targetType.icon" class="mr-1"></v-icon>
239
- {{ targetType.label }}
240
- </v-btn>
241
- </v-btn-toggle>
242
-
243
- <div class="mt-2">
244
- <v-alert
245
- v-if="!hasTargets"
246
- type="warning"
247
- density="compact"
248
- class="mb-2"
249
- >
250
- No tokens {{ selectedTargetType === 'targeted' ? 'targeted' : 'selected' }}
251
- </v-alert>
252
- <v-chip
253
- v-else
254
- color="success"
255
- size="small"
256
- >
257
- {{ targets.length }} target{{ targets.length !== 1 ? 's' : '' }}
258
- </v-chip>
259
- </div>
260
- </div>
261
-
262
- <!-- Multiplier Selection -->
263
- <div class="multiplier-selection mb-4">
264
- <h4 class="mb-2">Damage Multiplier</h4>
265
- <v-btn-toggle
266
- v-model="selectedMultiplier"
267
- mandatory
268
- variant="outlined"
269
- divided
270
- >
271
- <v-btn
272
- v-for="multiplier in multipliers"
273
- :key="multiplier.value"
274
- :value="multiplier.value"
275
- @click="selectMultiplier(multiplier.value)"
276
- size="small"
277
- >
278
- {{ multiplier.label }}
279
- </v-btn>
280
- </v-btn-toggle>
281
- </div>
282
-
283
- <!-- Final Damage Display -->
284
- <div class="final-damage mb-4">
285
- <v-card variant="outlined">
286
- <v-card-text class="text-center">
287
- <div class="text-h5">
288
- Final Damage: {{ finalDamage }}
289
- </div>
290
- <div v-if="selectedMultiplier !== 1" class="text-caption">
291
- ({{ props.rollData.value }} × {{ selectedMultiplier }})
292
- </div>
293
- </v-card-text>
294
- </v-card>
295
- </div>
296
- </v-card-text>
297
-
298
- <v-card-actions>
299
- <v-btn
300
- color="error"
301
- variant="elevated"
302
- prepend-icon="mdi-sword"
303
- :disabled="!hasTargets"
304
- @click="applyDamage('damage')"
305
- >
306
- Apply Damage
307
- </v-btn>
308
-
309
- <v-btn
310
- color="success"
311
- variant="elevated"
312
- prepend-icon="mdi-medical-bag"
313
- :disabled="!hasTargets"
314
- @click="applyHealing"
315
- >
316
- Apply Healing
317
- </v-btn>
318
-
319
- <v-btn
320
- color="info"
321
- variant="elevated"
322
- prepend-icon="mdi-heart"
323
- :disabled="!hasTargets"
324
- @click="applyTemp"
325
- >
326
- Temp HP
327
- </v-btn>
328
-
329
- <v-spacer></v-spacer>
330
-
331
- <v-btn
332
- variant="text"
333
- @click="closeDialog"
334
- >
335
- Cancel
336
- </v-btn>
337
- </v-card-actions>
338
- </v-card>
339
- </v-dialog>
340
- </template>
341
-
342
- <style scoped>
343
- .damage-type-badge {
344
- display: inline-flex;
345
- align-items: center;
346
- padding: 2px 8px;
347
- border-radius: 12px;
348
- background-color: rgba(0, 0, 0, 0.1);
349
- font-size: 0.875rem;
350
- }
351
-
352
- .damage-metadata {
353
- margin-top: 8px;
354
- }
355
-
356
- .target-selection, .multiplier-selection {
357
- border: 1px solid rgba(0, 0, 0, 0.12);
358
- border-radius: 4px;
359
- padding: 12px;
360
- }
361
-
362
- .final-damage {
363
- margin-top: 16px;
364
- }
365
- </style>
10
+ const fileNode = expandToNode `
11
+ <script setup>
12
+ import { ref, computed, inject, watch } from "vue";
13
+
14
+ const props = defineProps({
15
+ rollData: {
16
+ type: Object,
17
+ required: true,
18
+ default: () => ({
19
+ value: 0,
20
+ isDamageRoll: false,
21
+ damageType: null,
22
+ damageColor: null,
23
+ damageIcon: null,
24
+ metadata: {}
25
+ })
26
+ },
27
+ visible: {
28
+ type: Boolean,
29
+ default: false
30
+ }
31
+ });
32
+
33
+ const emit = defineEmits(['close', 'apply']);
34
+
35
+ // Internal state
36
+ const selectedTargetType = ref('selected');
37
+ const selectedMultiplier = ref(1);
38
+ const showDialog = ref(false);
39
+
40
+ // Available multipliers
41
+ const multipliers = [
42
+ { value: 0.25, label: '¼x' },
43
+ { value: 0.5, label: '½x' },
44
+ { value: 1, label: '1x' },
45
+ { value: 1.5, label: '1.5x' },
46
+ { value: 2, label: '2x' },
47
+ { value: 3, label: '3x' },
48
+ { value: 4, label: '4x' }
49
+ ];
50
+
51
+ // Target type options
52
+ const targetTypes = [
53
+ { value: 'targeted', label: 'Targeted', icon: 'fa-solid fa-bullseye' },
54
+ { value: 'selected', label: 'Selected', icon: 'fa-solid fa-expand' }
55
+ ];
56
+
57
+ // Watch for visibility changes
58
+ watch(() => props.visible, (newValue) => {
59
+ showDialog.value = newValue;
60
+ });
61
+
62
+ // Computed values
63
+ const finalDamage = computed(() => {
64
+ return Math.floor(props.rollData.value * selectedMultiplier.value);
65
+ });
66
+
67
+ const targets = computed(() => {
68
+ const targetType = selectedTargetType.value;
69
+ const targetList = targetType === 'targeted'
70
+ ? [...game.user.targets]
71
+ : (canvas?.tokens?.controlled ?? []);
72
+
73
+ return targetList;
74
+ });
75
+
76
+ const hasTargets = computed(() => targets.value.length > 0);
77
+
78
+ const damageTypeDisplay = computed(() => {
79
+ if (!props.rollData.isDamageRoll || !props.rollData.damageType) {
80
+ return null;
81
+ }
82
+
83
+ return {
84
+ type: props.rollData.damageType,
85
+ color: props.rollData.damageColor || '#ffffff',
86
+ icon: props.rollData.damageIcon
87
+ };
88
+ });
89
+
90
+ // Methods
91
+ const closeDialog = () => {
92
+ showDialog.value = false;
93
+ emit('close');
94
+ };
95
+
96
+ const applyDamage = (type = 'damage') => {
97
+ if (!hasTargets.value) {
98
+ ui.notifications.warn(game.i18n.localize(\`NOTIFICATIONS.\${selectedTargetType.value === 'targeted' ? 'NoTokenTargeted' : 'NoTokenSelected'}\`));
99
+ return;
100
+ }
101
+
102
+ const context = {
103
+ amount: type === 'healing' ? -finalDamage.value : finalDamage.value,
104
+ multiplier: selectedMultiplier.value,
105
+ targetType: selectedTargetType.value,
106
+ damageType: props.rollData.isDamageRoll ? props.rollData.damageType : null,
107
+ damageMetadata: props.rollData.isDamageRoll ? props.rollData.metadata : {},
108
+ color: props.rollData.isDamageRoll ? props.rollData.damageColor : null,
109
+ icon: props.rollData.isDamageRoll ? props.rollData.damageIcon : null,
110
+ isDamageRoll: props.rollData.isDamageRoll
111
+ };
112
+
113
+ emit('apply', {
114
+ type,
115
+ targets: targets.value,
116
+ context
117
+ });
118
+
119
+ closeDialog();
120
+ };
121
+
122
+ // Expose data for the Application sheet
123
+ const getApplicationData = () => {
124
+ return {
125
+ targets: targets.value,
126
+ context: {
127
+ amount: finalDamage.value,
128
+ multiplier: selectedMultiplier.value,
129
+ targetType: selectedTargetType.value
130
+ }
131
+ };
132
+ };
133
+
134
+ // Expose the method to the parent component
135
+ const rawSheet = inject("rawSheet", null);
136
+ if (rawSheet && typeof rawSheet === 'object') {
137
+ rawSheet.getApplicationData = getApplicationData;
138
+ }
139
+
140
+ const applyHealing = () => applyDamage('healing');
141
+ const applyTemp = () => applyDamage('temp');
142
+
143
+ const selectTargetType = (type) => {
144
+ selectedTargetType.value = type;
145
+ game.settings.set(game.system.id, 'userTargetDamageApplicationType', type);
146
+ };
147
+
148
+ const selectMultiplier = (multiplier) => {
149
+ selectedMultiplier.value = multiplier;
150
+ };
151
+
152
+ // Initialize from settings
153
+ const initializeSettings = () => {
154
+ const allowTargeting = game.settings.get(game.system.id, 'allowTargetDamageApplication');
155
+ let targetType = game.settings.get(game.system.id, 'userTargetDamageApplicationType');
156
+
157
+ if (!allowTargeting && targetType !== 'selected') {
158
+ targetType = 'selected';
159
+ game.settings.set(game.system.id, 'userTargetDamageApplicationType', 'selected');
160
+ }
161
+
162
+ selectedTargetType.value = targetType;
163
+ };
164
+
165
+ // Initialize on mount
166
+ initializeSettings();
167
+ </script>
168
+
169
+ <template>
170
+ <v-dialog
171
+ v-model="showDialog"
172
+ max-width="500"
173
+ persistent
174
+ @click:outside="closeDialog"
175
+ >
176
+ <v-card>
177
+ <v-card-title class="d-flex align-center">
178
+ <v-icon class="mr-2">mdi-sword-cross</v-icon>
179
+ Apply Damage
180
+ <v-spacer></v-spacer>
181
+ <v-btn
182
+ icon="mdi-close"
183
+ variant="text"
184
+ size="small"
185
+ @click="closeDialog"
186
+ ></v-btn>
187
+ </v-card-title>
188
+
189
+ <v-card-text>
190
+ <!-- Damage Info -->
191
+ <div class="damage-info mb-4">
192
+ <div class="d-flex align-center mb-2">
193
+ <span class="text-h6 mr-2">{{ props.rollData.value }}</span>
194
+ <span v-if="damageTypeDisplay" class="damage-type-badge">
195
+ <v-icon
196
+ v-if="damageTypeDisplay.icon"
197
+ :icon="damageTypeDisplay.icon"
198
+ :color="damageTypeDisplay.color"
199
+ size="small"
200
+ class="mr-1"
201
+ ></v-icon>
202
+ <span :style="{ color: damageTypeDisplay.color }">
203
+ {{ damageTypeDisplay.type }}
204
+ </span>
205
+ </span>
206
+ </div>
207
+
208
+ <!-- Damage metadata -->
209
+ <div v-if="props.rollData.isDamageRoll && Object.keys(props.rollData.metadata).length > 0"
210
+ class="damage-metadata">
211
+ <v-chip
212
+ v-for="(value, key) in props.rollData.metadata"
213
+ :key="key"
214
+ size="small"
215
+ variant="outlined"
216
+ class="mr-1 mb-1"
217
+ >
218
+ {{ key }}: {{ value }}
219
+ </v-chip>
220
+ </div>
221
+ </div>
222
+
223
+ <!-- Target Selection -->
224
+ <div class="target-selection mb-4">
225
+ <h4 class="mb-2">Target Type</h4>
226
+ <v-btn-toggle
227
+ v-model="selectedTargetType"
228
+ mandatory
229
+ variant="outlined"
230
+ divided
231
+ >
232
+ <v-btn
233
+ v-for="targetType in targetTypes"
234
+ :key="targetType.value"
235
+ :value="targetType.value"
236
+ @click="selectTargetType(targetType.value)"
237
+ >
238
+ <v-icon :icon="targetType.icon" class="mr-1"></v-icon>
239
+ {{ targetType.label }}
240
+ </v-btn>
241
+ </v-btn-toggle>
242
+
243
+ <div class="mt-2">
244
+ <v-alert
245
+ v-if="!hasTargets"
246
+ type="warning"
247
+ density="compact"
248
+ class="mb-2"
249
+ >
250
+ No tokens {{ selectedTargetType === 'targeted' ? 'targeted' : 'selected' }}
251
+ </v-alert>
252
+ <v-chip
253
+ v-else
254
+ color="success"
255
+ size="small"
256
+ >
257
+ {{ targets.length }} target{{ targets.length !== 1 ? 's' : '' }}
258
+ </v-chip>
259
+ </div>
260
+ </div>
261
+
262
+ <!-- Multiplier Selection -->
263
+ <div class="multiplier-selection mb-4">
264
+ <h4 class="mb-2">Damage Multiplier</h4>
265
+ <v-btn-toggle
266
+ v-model="selectedMultiplier"
267
+ mandatory
268
+ variant="outlined"
269
+ divided
270
+ >
271
+ <v-btn
272
+ v-for="multiplier in multipliers"
273
+ :key="multiplier.value"
274
+ :value="multiplier.value"
275
+ @click="selectMultiplier(multiplier.value)"
276
+ size="small"
277
+ >
278
+ {{ multiplier.label }}
279
+ </v-btn>
280
+ </v-btn-toggle>
281
+ </div>
282
+
283
+ <!-- Final Damage Display -->
284
+ <div class="final-damage mb-4">
285
+ <v-card variant="outlined">
286
+ <v-card-text class="text-center">
287
+ <div class="text-h5">
288
+ Final Damage: {{ finalDamage }}
289
+ </div>
290
+ <div v-if="selectedMultiplier !== 1" class="text-caption">
291
+ ({{ props.rollData.value }} × {{ selectedMultiplier }})
292
+ </div>
293
+ </v-card-text>
294
+ </v-card>
295
+ </div>
296
+ </v-card-text>
297
+
298
+ <v-card-actions>
299
+ <v-btn
300
+ color="error"
301
+ variant="elevated"
302
+ prepend-icon="mdi-sword"
303
+ :disabled="!hasTargets"
304
+ @click="applyDamage('damage')"
305
+ >
306
+ Apply Damage
307
+ </v-btn>
308
+
309
+ <v-btn
310
+ color="success"
311
+ variant="elevated"
312
+ prepend-icon="mdi-medical-bag"
313
+ :disabled="!hasTargets"
314
+ @click="applyHealing"
315
+ >
316
+ Apply Healing
317
+ </v-btn>
318
+
319
+ <v-btn
320
+ color="info"
321
+ variant="elevated"
322
+ prepend-icon="mdi-heart"
323
+ :disabled="!hasTargets"
324
+ @click="applyTemp"
325
+ >
326
+ Temp HP
327
+ </v-btn>
328
+
329
+ <v-spacer></v-spacer>
330
+
331
+ <v-btn
332
+ variant="text"
333
+ @click="closeDialog"
334
+ >
335
+ Cancel
336
+ </v-btn>
337
+ </v-card-actions>
338
+ </v-card>
339
+ </v-dialog>
340
+ </template>
341
+
342
+ <style scoped>
343
+ .damage-type-badge {
344
+ display: inline-flex;
345
+ align-items: center;
346
+ padding: 2px 8px;
347
+ border-radius: 12px;
348
+ background-color: rgba(0, 0, 0, 0.1);
349
+ font-size: 0.875rem;
350
+ }
351
+
352
+ .damage-metadata {
353
+ margin-top: 8px;
354
+ }
355
+
356
+ .target-selection, .multiplier-selection {
357
+ border: 1px solid rgba(0, 0, 0, 0.12);
358
+ border-radius: 4px;
359
+ padding: 12px;
360
+ }
361
+
362
+ .final-damage {
363
+ margin-top: 16px;
364
+ }
365
+ </style>
366
366
  `.appendNewLine();
367
367
  fs.writeFileSync(generatedFilePath, toString(fileNode));
368
368
  }