@happyvertical/smrt-projects 0.30.0

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 (92) hide show
  1. package/AGENTS.md +31 -0
  2. package/CLAUDE.md +1 -0
  3. package/LICENSE +7 -0
  4. package/README.md +97 -0
  5. package/dist/__smrt-register__.d.ts +2 -0
  6. package/dist/__smrt-register__.d.ts.map +1 -0
  7. package/dist/collections/Issues.d.ts +107 -0
  8. package/dist/collections/Issues.d.ts.map +1 -0
  9. package/dist/collections/Projects.d.ts +90 -0
  10. package/dist/collections/Projects.d.ts.map +1 -0
  11. package/dist/collections/PullRequests.d.ts +107 -0
  12. package/dist/collections/PullRequests.d.ts.map +1 -0
  13. package/dist/collections/Repositories.d.ts +77 -0
  14. package/dist/collections/Repositories.d.ts.map +1 -0
  15. package/dist/constants.d.ts +9 -0
  16. package/dist/constants.d.ts.map +1 -0
  17. package/dist/index.d.ts +14 -0
  18. package/dist/index.d.ts.map +1 -0
  19. package/dist/index.js +2477 -0
  20. package/dist/index.js.map +1 -0
  21. package/dist/manifest.json +4193 -0
  22. package/dist/models/Comment.d.ts +77 -0
  23. package/dist/models/Comment.d.ts.map +1 -0
  24. package/dist/models/Issue.d.ts +200 -0
  25. package/dist/models/Issue.d.ts.map +1 -0
  26. package/dist/models/Label.d.ts +63 -0
  27. package/dist/models/Label.d.ts.map +1 -0
  28. package/dist/models/Project.d.ts +183 -0
  29. package/dist/models/Project.d.ts.map +1 -0
  30. package/dist/models/PullRequest.d.ts +114 -0
  31. package/dist/models/PullRequest.d.ts.map +1 -0
  32. package/dist/models/Repository.d.ts +141 -0
  33. package/dist/models/Repository.d.ts.map +1 -0
  34. package/dist/playground.d.ts +2 -0
  35. package/dist/playground.d.ts.map +1 -0
  36. package/dist/playground.js +129 -0
  37. package/dist/playground.js.map +1 -0
  38. package/dist/prompts.d.ts +2 -0
  39. package/dist/prompts.d.ts.map +1 -0
  40. package/dist/smrt-knowledge.json +1956 -0
  41. package/dist/svelte/components/ApprovalActions.svelte +213 -0
  42. package/dist/svelte/components/ApprovalActions.svelte.d.ts +17 -0
  43. package/dist/svelte/components/ApprovalActions.svelte.d.ts.map +1 -0
  44. package/dist/svelte/components/BulkActions.svelte +224 -0
  45. package/dist/svelte/components/BulkActions.svelte.d.ts +14 -0
  46. package/dist/svelte/components/BulkActions.svelte.d.ts.map +1 -0
  47. package/dist/svelte/components/DurationDisplay.svelte +68 -0
  48. package/dist/svelte/components/DurationDisplay.svelte.d.ts +11 -0
  49. package/dist/svelte/components/DurationDisplay.svelte.d.ts.map +1 -0
  50. package/dist/svelte/components/RejectDialog.svelte +250 -0
  51. package/dist/svelte/components/RejectDialog.svelte.d.ts +17 -0
  52. package/dist/svelte/components/RejectDialog.svelte.d.ts.map +1 -0
  53. package/dist/svelte/components/TimeEntryCard.svelte +294 -0
  54. package/dist/svelte/components/TimeEntryCard.svelte.d.ts +17 -0
  55. package/dist/svelte/components/TimeEntryCard.svelte.d.ts.map +1 -0
  56. package/dist/svelte/components/TimeEntryList.svelte +351 -0
  57. package/dist/svelte/components/TimeEntryList.svelte.d.ts +17 -0
  58. package/dist/svelte/components/TimeEntryList.svelte.d.ts.map +1 -0
  59. package/dist/svelte/components/TimeSummary.svelte +165 -0
  60. package/dist/svelte/components/TimeSummary.svelte.d.ts +19 -0
  61. package/dist/svelte/components/TimeSummary.svelte.d.ts.map +1 -0
  62. package/dist/svelte/components/__tests__/ApprovalActions.test.js +41 -0
  63. package/dist/svelte/components/__tests__/BulkActions.test.js +46 -0
  64. package/dist/svelte/components/__tests__/DurationDisplay.test.js +23 -0
  65. package/dist/svelte/components/__tests__/RejectDialog.test.js +45 -0
  66. package/dist/svelte/components/__tests__/TimeEntryCard.test.js +45 -0
  67. package/dist/svelte/components/__tests__/TimeEntryList.test.js +50 -0
  68. package/dist/svelte/components/__tests__/TimeSummary.test.js +39 -0
  69. package/dist/svelte/components/utils.d.ts +42 -0
  70. package/dist/svelte/components/utils.d.ts.map +1 -0
  71. package/dist/svelte/components/utils.js +43 -0
  72. package/dist/svelte/i18n.d.ts +18 -0
  73. package/dist/svelte/i18n.d.ts.map +1 -0
  74. package/dist/svelte/i18n.js +18 -0
  75. package/dist/svelte/index.d.ts +26 -0
  76. package/dist/svelte/index.d.ts.map +1 -0
  77. package/dist/svelte/index.js +31 -0
  78. package/dist/svelte/playground.d.ts +122 -0
  79. package/dist/svelte/playground.d.ts.map +1 -0
  80. package/dist/svelte/playground.js +114 -0
  81. package/dist/svelte/utils.d.ts +42 -0
  82. package/dist/svelte/utils.d.ts.map +1 -0
  83. package/dist/svelte/utils.js +43 -0
  84. package/dist/types.d.ts +54 -0
  85. package/dist/types.d.ts.map +1 -0
  86. package/dist/types.js +2 -0
  87. package/dist/types.js.map +1 -0
  88. package/dist/ui.d.ts +10 -0
  89. package/dist/ui.d.ts.map +1 -0
  90. package/dist/ui.js +85 -0
  91. package/dist/ui.js.map +1 -0
  92. package/package.json +100 -0
@@ -0,0 +1,213 @@
1
+ <script lang="ts">
2
+ /**
3
+ * ApprovalActions - Status-based action buttons for approval workflow
4
+ * Shows appropriate buttons based on current status
5
+ */
6
+
7
+ import { useI18n } from '@happyvertical/smrt-ui/i18n';
8
+ import { M } from '../i18n.js';
9
+ import type { ApprovalStatus } from './utils.js';
10
+
11
+ const { t } = useI18n();
12
+
13
+ /** Props for ApprovalActions component */
14
+ export interface Props {
15
+ status: ApprovalStatus;
16
+ onsubmit?: () => void;
17
+ onapprove?: () => void;
18
+ onreject?: () => void;
19
+ onedit?: () => void;
20
+ ondelete?: () => void;
21
+ loading?: boolean;
22
+ disabled?: boolean;
23
+ layout?: 'horizontal' | 'vertical';
24
+ }
25
+
26
+ let {
27
+ status,
28
+ onsubmit,
29
+ onapprove,
30
+ onreject,
31
+ onedit,
32
+ ondelete,
33
+ loading = false,
34
+ disabled = false,
35
+ layout = 'horizontal',
36
+ }: Props = $props();
37
+
38
+ // Determine which actions are available based on status
39
+ const canSubmit = $derived(status === 'draft' && onsubmit);
40
+ const canApprove = $derived(status === 'submitted' && onapprove);
41
+ const canReject = $derived(status === 'submitted' && onreject);
42
+ const canEdit = $derived(
43
+ (status === 'draft' || status === 'rejected') && onedit,
44
+ );
45
+ const canDelete = $derived(status === 'draft' && ondelete);
46
+ </script>
47
+
48
+ <div class="approval-actions" class:vertical={layout === 'vertical'}>
49
+ {#if canSubmit}
50
+ <button
51
+ type="button"
52
+ class="btn btn-filled"
53
+ onclick={onsubmit}
54
+ disabled={disabled || loading}
55
+ >
56
+ {loading ? 'Submitting...' : 'Submit for Approval'}
57
+ </button>
58
+ {/if}
59
+
60
+ {#if canApprove}
61
+ <button
62
+ type="button"
63
+ class="btn btn-filled-tonal"
64
+ onclick={onapprove}
65
+ disabled={disabled || loading}
66
+ >
67
+ {loading ? 'Approving...' : 'Approve'}
68
+ </button>
69
+ {/if}
70
+
71
+ {#if canReject}
72
+ <button
73
+ type="button"
74
+ class="btn btn-error"
75
+ onclick={onreject}
76
+ disabled={disabled || loading}
77
+ >
78
+ Reject
79
+ </button>
80
+ {/if}
81
+
82
+ {#if canEdit}
83
+ <button
84
+ type="button"
85
+ class="btn btn-outlined"
86
+ onclick={onedit}
87
+ disabled={disabled || loading}
88
+ >
89
+ Edit
90
+ </button>
91
+ {/if}
92
+
93
+ {#if canDelete}
94
+ <button
95
+ type="button"
96
+ class="btn btn-error-outlined"
97
+ onclick={ondelete}
98
+ disabled={disabled || loading}
99
+ >
100
+ Delete
101
+ </button>
102
+ {/if}
103
+
104
+ {#if status === 'approved'}
105
+ <span class="status-message success">
106
+ {t(M['projects.approval_actions.approved_message'])}
107
+ </span>
108
+ {/if}
109
+
110
+ {#if status === 'rejected'}
111
+ <span class="status-message error">
112
+ {t(M['projects.approval_actions.rejected_message'])}
113
+ </span>
114
+ {/if}
115
+ </div>
116
+
117
+ <style>
118
+ .approval-actions {
119
+ display: flex;
120
+ flex-wrap: wrap;
121
+ gap: 0.75rem;
122
+ align-items: center;
123
+ }
124
+
125
+ .approval-actions.vertical {
126
+ flex-direction: column;
127
+ align-items: stretch;
128
+ }
129
+
130
+ .btn {
131
+ display: inline-flex;
132
+ align-items: center;
133
+ justify-content: center;
134
+ gap: 0.5rem;
135
+ padding: 0.625rem 1.5rem;
136
+ font-size: var(--smrt-typography-label-large-size, 0.875rem);
137
+ font-weight: var(--smrt-typography-label-large-weight, 500);
138
+ letter-spacing: var(--smrt-typography-label-large-tracking, 0.1px);
139
+ border-radius: var(--smrt-radius-full, 9999px);
140
+ border: none;
141
+ cursor: pointer;
142
+ transition: all 0.2s var(--smrt-easing-standard);
143
+ }
144
+
145
+ .btn:disabled {
146
+ opacity: 0.38;
147
+ cursor: not-allowed;
148
+ }
149
+
150
+ .btn-filled {
151
+ background: var(--smrt-color-primary);
152
+ color: var(--smrt-color-on-primary);
153
+ }
154
+
155
+ .btn-filled:hover:not(:disabled) {
156
+ box-shadow: var(--smrt-elevation-1);
157
+ }
158
+
159
+ .btn-filled-tonal {
160
+ background: var(--smrt-color-secondary-container);
161
+ color: var(--smrt-color-on-secondary-container);
162
+ }
163
+
164
+ .btn-filled-tonal:hover:not(:disabled) {
165
+ box-shadow: var(--smrt-elevation-1);
166
+ }
167
+
168
+ .btn-error {
169
+ background: var(--smrt-color-error);
170
+ color: var(--smrt-color-on-error);
171
+ }
172
+
173
+ .btn-error:hover:not(:disabled) {
174
+ box-shadow: var(--smrt-elevation-1);
175
+ }
176
+
177
+ .btn-error-outlined {
178
+ background: transparent;
179
+ border: 1px solid var(--smrt-color-error);
180
+ color: var(--smrt-color-error);
181
+ }
182
+
183
+ .btn-error-outlined:hover:not(:disabled) {
184
+ background: var(--smrt-color-error-container);
185
+ }
186
+
187
+ .btn-outlined {
188
+ background: transparent;
189
+ border: 1px solid var(--smrt-color-outline);
190
+ color: var(--smrt-color-on-surface);
191
+ }
192
+
193
+ .btn-outlined:hover:not(:disabled) {
194
+ background: var(--smrt-color-surface-container-highest);
195
+ }
196
+
197
+ .status-message {
198
+ font-size: var(--smrt-typography-body-medium-size, 0.875rem);
199
+ font-weight: var(--smrt-typography-label-large-weight, 500);
200
+ padding: 0.5rem 1rem;
201
+ border-radius: var(--smrt-radius-small, 8px);
202
+ }
203
+
204
+ .status-message.success {
205
+ background: var(--smrt-color-primary-container);
206
+ color: var(--smrt-color-on-primary-container);
207
+ }
208
+
209
+ .status-message.error {
210
+ background: var(--smrt-color-error-container);
211
+ color: var(--smrt-color-on-error-container);
212
+ }
213
+ </style>
@@ -0,0 +1,17 @@
1
+ import type { ApprovalStatus } from './utils.js';
2
+ /** Props for ApprovalActions component */
3
+ export interface Props {
4
+ status: ApprovalStatus;
5
+ onsubmit?: () => void;
6
+ onapprove?: () => void;
7
+ onreject?: () => void;
8
+ onedit?: () => void;
9
+ ondelete?: () => void;
10
+ loading?: boolean;
11
+ disabled?: boolean;
12
+ layout?: 'horizontal' | 'vertical';
13
+ }
14
+ declare const ApprovalActions: import("svelte").Component<Props, {}, "">;
15
+ type ApprovalActions = ReturnType<typeof ApprovalActions>;
16
+ export default ApprovalActions;
17
+ //# sourceMappingURL=ApprovalActions.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ApprovalActions.svelte.d.ts","sourceRoot":"","sources":["../../../src/svelte/components/ApprovalActions.svelte.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAGjD,0CAA0C;AAC1C,MAAM,WAAW,KAAK;IACpB,MAAM,EAAE,cAAc,CAAC;IACvB,QAAQ,CAAC,EAAE,MAAM,IAAI,CAAC;IACtB,SAAS,CAAC,EAAE,MAAM,IAAI,CAAC;IACvB,QAAQ,CAAC,EAAE,MAAM,IAAI,CAAC;IACtB,MAAM,CAAC,EAAE,MAAM,IAAI,CAAC;IACpB,QAAQ,CAAC,EAAE,MAAM,IAAI,CAAC;IACtB,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,MAAM,CAAC,EAAE,YAAY,GAAG,UAAU,CAAC;CACpC;AAiFD,QAAA,MAAM,eAAe,2CAAwC,CAAC;AAC9D,KAAK,eAAe,GAAG,UAAU,CAAC,OAAO,eAAe,CAAC,CAAC;AAC1D,eAAe,eAAe,CAAC"}
@@ -0,0 +1,224 @@
1
+ <script lang="ts">
2
+ /**
3
+ * BulkActions - Action bar for bulk operations on selected items
4
+ * Appears when items are selected in a list
5
+ */
6
+
7
+ import { useI18n } from '@happyvertical/smrt-ui/i18n';
8
+ import { M } from '../i18n.js';
9
+
10
+ const { t } = useI18n();
11
+
12
+ /** Props for BulkActions component */
13
+ export interface Props {
14
+ selectedCount: number;
15
+ onapprove?: () => void;
16
+ onreject?: () => void;
17
+ ondelete?: () => void;
18
+ onexport?: () => void;
19
+ onclear?: () => void;
20
+ loading?: boolean;
21
+ }
22
+
23
+ let {
24
+ selectedCount,
25
+ onapprove,
26
+ onreject,
27
+ ondelete,
28
+ onexport,
29
+ onclear,
30
+ loading = false,
31
+ }: Props = $props();
32
+
33
+ const visible = $derived(selectedCount > 0);
34
+ </script>
35
+
36
+ {#if visible}
37
+ <div class="bulk-actions">
38
+ <div class="selection-info">
39
+ <span class="count">{selectedCount}</span>
40
+ <span class="label">{selectedCount === 1 ? 'item' : 'items'} selected</span>
41
+ {#if onclear}
42
+ <button type="button" class="clear-btn" onclick={onclear} disabled={loading}>
43
+ Clear
44
+ </button>
45
+ {/if}
46
+ </div>
47
+
48
+ <div class="actions">
49
+ {#if onapprove}
50
+ <button
51
+ type="button"
52
+ class="btn btn-filled"
53
+ onclick={onapprove}
54
+ disabled={loading}
55
+ >
56
+ {t(M['projects.bulk_actions.approve_all'])}
57
+ </button>
58
+ {/if}
59
+
60
+ {#if onreject}
61
+ <button
62
+ type="button"
63
+ class="btn btn-error"
64
+ onclick={onreject}
65
+ disabled={loading}
66
+ >
67
+ {t(M['projects.bulk_actions.reject_all'])}
68
+ </button>
69
+ {/if}
70
+
71
+ {#if onexport}
72
+ <button
73
+ type="button"
74
+ class="btn btn-outlined"
75
+ onclick={onexport}
76
+ disabled={loading}
77
+ >
78
+ Export
79
+ </button>
80
+ {/if}
81
+
82
+ {#if ondelete}
83
+ <button
84
+ type="button"
85
+ class="btn btn-error-outlined"
86
+ onclick={ondelete}
87
+ disabled={loading}
88
+ >
89
+ Delete
90
+ </button>
91
+ {/if}
92
+ </div>
93
+ </div>
94
+ {/if}
95
+
96
+ <style>
97
+ .bulk-actions {
98
+ display: flex;
99
+ justify-content: space-between;
100
+ align-items: center;
101
+ padding: 0.75rem 1rem;
102
+ background: var(--smrt-color-secondary-container);
103
+ border: 1px solid var(--smrt-color-outline-variant);
104
+ border-radius: var(--smrt-radius-medium, 12px);
105
+ margin-bottom: 1rem;
106
+ }
107
+
108
+ .selection-info {
109
+ display: flex;
110
+ align-items: center;
111
+ gap: 0.5rem;
112
+ }
113
+
114
+ .count {
115
+ font-size: var(--smrt-typography-title-medium-size, 1rem);
116
+ font-weight: var(--smrt-typography-title-medium-weight, 500);
117
+ color: var(--smrt-color-on-secondary-container);
118
+ }
119
+
120
+ .label {
121
+ font-size: var(--smrt-typography-body-medium-size, 0.875rem);
122
+ color: var(--smrt-color-on-secondary-container);
123
+ }
124
+
125
+ .clear-btn {
126
+ background: none;
127
+ border: none;
128
+ padding: 0.25rem 0.5rem;
129
+ font-size: var(--smrt-typography-label-medium-size, 0.75rem);
130
+ color: var(--smrt-color-primary);
131
+ cursor: pointer;
132
+ text-decoration: underline;
133
+ }
134
+
135
+ .clear-btn:hover:not(:disabled) {
136
+ color: var(--smrt-color-primary);
137
+ opacity: 0.8;
138
+ }
139
+
140
+ .clear-btn:disabled {
141
+ opacity: 0.38;
142
+ cursor: not-allowed;
143
+ }
144
+
145
+ .actions {
146
+ display: flex;
147
+ gap: 0.5rem;
148
+ }
149
+
150
+ .btn {
151
+ display: inline-flex;
152
+ align-items: center;
153
+ justify-content: center;
154
+ gap: 0.5rem;
155
+ padding: 0.5rem 1rem;
156
+ font-size: var(--smrt-typography-label-large-size, 0.875rem);
157
+ font-weight: var(--smrt-typography-label-large-weight, 500);
158
+ letter-spacing: var(--smrt-typography-label-large-tracking, 0.1px);
159
+ border-radius: var(--smrt-radius-full, 9999px);
160
+ border: none;
161
+ cursor: pointer;
162
+ transition: all 0.2s var(--smrt-easing-standard);
163
+ }
164
+
165
+ .btn:disabled {
166
+ opacity: 0.38;
167
+ cursor: not-allowed;
168
+ }
169
+
170
+ .btn-filled {
171
+ background: var(--smrt-color-primary);
172
+ color: var(--smrt-color-on-primary);
173
+ }
174
+
175
+ .btn-filled:hover:not(:disabled) {
176
+ box-shadow: var(--smrt-elevation-1);
177
+ }
178
+
179
+ .btn-error {
180
+ background: var(--smrt-color-error);
181
+ color: var(--smrt-color-on-error);
182
+ }
183
+
184
+ .btn-error:hover:not(:disabled) {
185
+ box-shadow: var(--smrt-elevation-1);
186
+ }
187
+
188
+ .btn-error-outlined {
189
+ background: transparent;
190
+ border: 1px solid var(--smrt-color-error);
191
+ color: var(--smrt-color-error);
192
+ }
193
+
194
+ .btn-error-outlined:hover:not(:disabled) {
195
+ background: var(--smrt-color-error-container);
196
+ }
197
+
198
+ .btn-outlined {
199
+ background: transparent;
200
+ border: 1px solid var(--smrt-color-outline);
201
+ color: var(--smrt-color-on-surface);
202
+ }
203
+
204
+ .btn-outlined:hover:not(:disabled) {
205
+ background: var(--smrt-color-surface-container-highest);
206
+ }
207
+
208
+ @media (max-width: 640px) {
209
+ .bulk-actions {
210
+ flex-direction: column;
211
+ gap: 0.75rem;
212
+ }
213
+
214
+ .actions {
215
+ width: 100%;
216
+ flex-wrap: wrap;
217
+ }
218
+
219
+ .actions .btn {
220
+ flex: 1;
221
+ min-width: 100px;
222
+ }
223
+ }
224
+ </style>
@@ -0,0 +1,14 @@
1
+ /** Props for BulkActions component */
2
+ export interface Props {
3
+ selectedCount: number;
4
+ onapprove?: () => void;
5
+ onreject?: () => void;
6
+ ondelete?: () => void;
7
+ onexport?: () => void;
8
+ onclear?: () => void;
9
+ loading?: boolean;
10
+ }
11
+ declare const BulkActions: import("svelte").Component<Props, {}, "">;
12
+ type BulkActions = ReturnType<typeof BulkActions>;
13
+ export default BulkActions;
14
+ //# sourceMappingURL=BulkActions.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"BulkActions.svelte.d.ts","sourceRoot":"","sources":["../../../src/svelte/components/BulkActions.svelte.ts"],"names":[],"mappings":"AAWA,sCAAsC;AACtC,MAAM,WAAW,KAAK;IACpB,aAAa,EAAE,MAAM,CAAC;IACtB,SAAS,CAAC,EAAE,MAAM,IAAI,CAAC;IACvB,QAAQ,CAAC,EAAE,MAAM,IAAI,CAAC;IACtB,QAAQ,CAAC,EAAE,MAAM,IAAI,CAAC;IACtB,QAAQ,CAAC,EAAE,MAAM,IAAI,CAAC;IACtB,OAAO,CAAC,EAAE,MAAM,IAAI,CAAC;IACrB,OAAO,CAAC,EAAE,OAAO,CAAC;CACnB;AAmED,QAAA,MAAM,WAAW,2CAAwC,CAAC;AAC1D,KAAK,WAAW,GAAG,UAAU,CAAC,OAAO,WAAW,CAAC,CAAC;AAClD,eAAe,WAAW,CAAC"}
@@ -0,0 +1,68 @@
1
+ <script lang="ts">
2
+ /**
3
+ * DurationDisplay - Formats hours for display
4
+ * Supports decimal (8.5) or HH:MM (8:30) formats
5
+ */
6
+
7
+ import { formatHoursHHMM } from './utils.js';
8
+
9
+ /** Props for DurationDisplay component */
10
+ export interface Props {
11
+ hours: number;
12
+ format?: 'decimal' | 'hhmm';
13
+ size?: 'sm' | 'md' | 'lg';
14
+ showLabel?: boolean;
15
+ }
16
+
17
+ let {
18
+ hours,
19
+ format = 'decimal',
20
+ size = 'md',
21
+ showLabel = false,
22
+ }: Props = $props();
23
+
24
+ // $derived creates a reactive value, not a function - no wrapper or parentheses needed
25
+ const formattedValue = $derived(
26
+ format === 'hhmm' ? formatHoursHHMM(hours) : hours.toFixed(1),
27
+ );
28
+ </script>
29
+
30
+ <span class="duration" class:sm={size === 'sm'} class:lg={size === 'lg'}>
31
+ <span class="value">{formattedValue}</span>
32
+ {#if showLabel || format === 'decimal'}
33
+ <span class="unit">{format === 'hhmm' ? '' : 'h'}</span>
34
+ {/if}
35
+ </span>
36
+
37
+ <style>
38
+ .duration {
39
+ display: inline-flex;
40
+ align-items: baseline;
41
+ gap: 0.125rem;
42
+ font-variant-numeric: tabular-nums;
43
+ font-size: var(--smrt-typography-body-large-size, 1rem);
44
+ }
45
+
46
+ .value {
47
+ font-weight: var(--smrt-typography-title-medium-weight, 500);
48
+ color: var(--smrt-color-on-surface);
49
+ }
50
+
51
+ .unit {
52
+ font-weight: var(--smrt-typography-body-medium-weight, 400);
53
+ color: var(--smrt-color-on-surface-variant);
54
+ }
55
+
56
+ /* Size variants */
57
+ .duration.sm {
58
+ font-size: var(--smrt-typography-body-medium-size, 0.875rem);
59
+ }
60
+
61
+ .duration.sm .value {
62
+ font-weight: var(--smrt-typography-body-medium-weight, 400);
63
+ }
64
+
65
+ .duration.lg {
66
+ font-size: var(--smrt-typography-headline-small-size, 1.5rem);
67
+ }
68
+ </style>
@@ -0,0 +1,11 @@
1
+ /** Props for DurationDisplay component */
2
+ export interface Props {
3
+ hours: number;
4
+ format?: 'decimal' | 'hhmm';
5
+ size?: 'sm' | 'md' | 'lg';
6
+ showLabel?: boolean;
7
+ }
8
+ declare const DurationDisplay: import("svelte").Component<Props, {}, "">;
9
+ type DurationDisplay = ReturnType<typeof DurationDisplay>;
10
+ export default DurationDisplay;
11
+ //# sourceMappingURL=DurationDisplay.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DurationDisplay.svelte.d.ts","sourceRoot":"","sources":["../../../src/svelte/components/DurationDisplay.svelte.ts"],"names":[],"mappings":"AAUA,0CAA0C;AAC1C,MAAM,WAAW,KAAK;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE,SAAS,GAAG,MAAM,CAAC;IAC5B,IAAI,CAAC,EAAE,IAAI,GAAG,IAAI,GAAG,IAAI,CAAC;IAC1B,SAAS,CAAC,EAAE,OAAO,CAAC;CACrB;AA+BD,QAAA,MAAM,eAAe,2CAAwC,CAAC;AAC9D,KAAK,eAAe,GAAG,UAAU,CAAC,OAAO,eAAe,CAAC,CAAC;AAC1D,eAAe,eAAe,CAAC"}