@mrck-labs/vanaheim-shared 0.1.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/README.md +56 -0
- package/dist/constants/index.d.mts +42 -0
- package/dist/constants/index.d.ts +42 -0
- package/dist/constants/index.js +147 -0
- package/dist/constants/index.js.map +1 -0
- package/dist/constants/index.mjs +108 -0
- package/dist/constants/index.mjs.map +1 -0
- package/dist/database-BKc0Oj26.d.mts +240 -0
- package/dist/database-BKc0Oj26.d.ts +240 -0
- package/dist/index.d.mts +4 -0
- package/dist/index.d.ts +4 -0
- package/dist/index.js +411 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +343 -0
- package/dist/index.mjs.map +1 -0
- package/dist/types/index.d.mts +135 -0
- package/dist/types/index.d.ts +135 -0
- package/dist/types/index.js +39 -0
- package/dist/types/index.js.map +1 -0
- package/dist/types/index.mjs +12 -0
- package/dist/types/index.mjs.map +1 -0
- package/dist/utils/index.d.mts +159 -0
- package/dist/utils/index.d.ts +159 -0
- package/dist/utils/index.js +288 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/index.mjs +236 -0
- package/dist/utils/index.mjs.map +1 -0
- package/package.json +66 -0
|
@@ -0,0 +1,288 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __export = (target, all) => {
|
|
7
|
+
for (var name in all)
|
|
8
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
9
|
+
};
|
|
10
|
+
var __copyProps = (to, from, except, desc) => {
|
|
11
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
12
|
+
for (let key of __getOwnPropNames(from))
|
|
13
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
14
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
15
|
+
}
|
|
16
|
+
return to;
|
|
17
|
+
};
|
|
18
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
|
+
|
|
20
|
+
// src/utils/index.ts
|
|
21
|
+
var utils_exports = {};
|
|
22
|
+
__export(utils_exports, {
|
|
23
|
+
calculateFocusStats: () => calculateFocusStats,
|
|
24
|
+
calculateLieuBalance: () => calculateLieuBalance,
|
|
25
|
+
calculateMonthlyExpenses: () => calculateMonthlyExpenses,
|
|
26
|
+
calculateMonthlyIncome: () => calculateMonthlyIncome,
|
|
27
|
+
calculateMonthlySavings: () => calculateMonthlySavings,
|
|
28
|
+
calculateSavingsRate: () => calculateSavingsRate,
|
|
29
|
+
formatCurrency: () => formatCurrency,
|
|
30
|
+
formatDate: () => formatDate,
|
|
31
|
+
formatDueDate: () => formatDueDate,
|
|
32
|
+
formatRelativeTime: () => formatRelativeTime,
|
|
33
|
+
formatTime: () => formatTime,
|
|
34
|
+
formatTotalTime: () => formatTotalTime,
|
|
35
|
+
generateId: () => generateId,
|
|
36
|
+
generateRandomColor: () => generateRandomColor,
|
|
37
|
+
generateShortId: () => generateShortId,
|
|
38
|
+
getRepoName: () => getRepoName,
|
|
39
|
+
isNonEmptyString: () => isNonEmptyString,
|
|
40
|
+
isPositiveNumber: () => isPositiveNumber,
|
|
41
|
+
isValidCurrency: () => isValidCurrency,
|
|
42
|
+
isValidEmail: () => isValidEmail,
|
|
43
|
+
isValidFrequency: () => isValidFrequency,
|
|
44
|
+
isValidISODate: () => isValidISODate,
|
|
45
|
+
isValidUrl: () => isValidUrl,
|
|
46
|
+
toMonthlyAmount: () => toMonthlyAmount,
|
|
47
|
+
toYearlyAmount: () => toYearlyAmount,
|
|
48
|
+
truncate: () => truncate
|
|
49
|
+
});
|
|
50
|
+
module.exports = __toCommonJS(utils_exports);
|
|
51
|
+
|
|
52
|
+
// src/utils/formatters.ts
|
|
53
|
+
function formatTime(seconds) {
|
|
54
|
+
const mins = Math.floor(seconds / 60);
|
|
55
|
+
const secs = seconds % 60;
|
|
56
|
+
return `${mins.toString().padStart(2, "0")}:${secs.toString().padStart(2, "0")}`;
|
|
57
|
+
}
|
|
58
|
+
function formatTotalTime(seconds) {
|
|
59
|
+
const hours = Math.floor(seconds / 3600);
|
|
60
|
+
const mins = Math.floor(seconds % 3600 / 60);
|
|
61
|
+
if (hours > 0) {
|
|
62
|
+
return `${hours}h ${mins}m`;
|
|
63
|
+
}
|
|
64
|
+
return `${mins}m`;
|
|
65
|
+
}
|
|
66
|
+
function formatCurrency(amount, currency, locale = "en-CH") {
|
|
67
|
+
return new Intl.NumberFormat(locale, {
|
|
68
|
+
style: "currency",
|
|
69
|
+
currency,
|
|
70
|
+
minimumFractionDigits: 2,
|
|
71
|
+
maximumFractionDigits: 2
|
|
72
|
+
}).format(amount);
|
|
73
|
+
}
|
|
74
|
+
function formatRelativeTime(dateStr) {
|
|
75
|
+
const date = new Date(dateStr);
|
|
76
|
+
const now = /* @__PURE__ */ new Date();
|
|
77
|
+
const diffMs = now.getTime() - date.getTime();
|
|
78
|
+
const diffMins = Math.floor(diffMs / 6e4);
|
|
79
|
+
const diffHours = Math.floor(diffMins / 60);
|
|
80
|
+
const diffDays = Math.floor(diffHours / 24);
|
|
81
|
+
if (diffMins < 1) return "Just now";
|
|
82
|
+
if (diffMins < 60) return `${diffMins}m ago`;
|
|
83
|
+
if (diffHours < 24) return `${diffHours}h ago`;
|
|
84
|
+
if (diffDays < 7) return `${diffDays}d ago`;
|
|
85
|
+
return date.toLocaleDateString();
|
|
86
|
+
}
|
|
87
|
+
function formatDate(dateStr, options = {
|
|
88
|
+
month: "short",
|
|
89
|
+
day: "numeric",
|
|
90
|
+
year: "numeric"
|
|
91
|
+
}) {
|
|
92
|
+
return new Date(dateStr).toLocaleDateString("en-US", options);
|
|
93
|
+
}
|
|
94
|
+
function formatDueDate(dueDate) {
|
|
95
|
+
if (!dueDate) return { text: "", isOverdue: false };
|
|
96
|
+
const due = new Date(dueDate);
|
|
97
|
+
const today = /* @__PURE__ */ new Date();
|
|
98
|
+
today.setHours(0, 0, 0, 0);
|
|
99
|
+
const tomorrow = new Date(today);
|
|
100
|
+
tomorrow.setDate(tomorrow.getDate() + 1);
|
|
101
|
+
const dueDay = new Date(due);
|
|
102
|
+
dueDay.setHours(0, 0, 0, 0);
|
|
103
|
+
const isOverdue = dueDay < today;
|
|
104
|
+
if (dueDay.getTime() === today.getTime()) {
|
|
105
|
+
return { text: "Today", isOverdue: false };
|
|
106
|
+
} else if (dueDay.getTime() === tomorrow.getTime()) {
|
|
107
|
+
return { text: "Tomorrow", isOverdue: false };
|
|
108
|
+
} else if (isOverdue) {
|
|
109
|
+
const daysAgo = Math.ceil(
|
|
110
|
+
(today.getTime() - dueDay.getTime()) / (1e3 * 60 * 60 * 24)
|
|
111
|
+
);
|
|
112
|
+
return { text: `${daysAgo}d overdue`, isOverdue: true };
|
|
113
|
+
} else {
|
|
114
|
+
return {
|
|
115
|
+
text: due.toLocaleDateString("en-US", { month: "short", day: "numeric" }),
|
|
116
|
+
isOverdue: false
|
|
117
|
+
};
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
function truncate(str, maxLength) {
|
|
121
|
+
if (str.length <= maxLength) return str;
|
|
122
|
+
return str.slice(0, maxLength) + "...";
|
|
123
|
+
}
|
|
124
|
+
function getRepoName(url) {
|
|
125
|
+
const match = url.match(/github\.com\/(.+)$/);
|
|
126
|
+
return match ? match[1] : url;
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// src/utils/generators.ts
|
|
130
|
+
function generateId() {
|
|
131
|
+
if (typeof crypto !== "undefined" && typeof crypto.randomUUID === "function") {
|
|
132
|
+
return crypto.randomUUID();
|
|
133
|
+
}
|
|
134
|
+
return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, (c) => {
|
|
135
|
+
const r = Math.random() * 16 | 0;
|
|
136
|
+
const v = c === "x" ? r : r & 3 | 8;
|
|
137
|
+
return v.toString(16);
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
function generateShortId() {
|
|
141
|
+
return `${Date.now()}-${Math.random().toString(36).slice(2, 9)}`;
|
|
142
|
+
}
|
|
143
|
+
function generateRandomColor() {
|
|
144
|
+
const colors = [
|
|
145
|
+
"#ef4444",
|
|
146
|
+
// red
|
|
147
|
+
"#f97316",
|
|
148
|
+
// orange
|
|
149
|
+
"#eab308",
|
|
150
|
+
// yellow
|
|
151
|
+
"#22c55e",
|
|
152
|
+
// green
|
|
153
|
+
"#14b8a6",
|
|
154
|
+
// teal
|
|
155
|
+
"#3b82f6",
|
|
156
|
+
// blue
|
|
157
|
+
"#8b5cf6",
|
|
158
|
+
// violet
|
|
159
|
+
"#ec4899"
|
|
160
|
+
// pink
|
|
161
|
+
];
|
|
162
|
+
return colors[Math.floor(Math.random() * colors.length)];
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
// src/utils/validators.ts
|
|
166
|
+
function isValidEmail(email) {
|
|
167
|
+
const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
|
|
168
|
+
return emailRegex.test(email);
|
|
169
|
+
}
|
|
170
|
+
function isValidUrl(url) {
|
|
171
|
+
try {
|
|
172
|
+
new URL(url);
|
|
173
|
+
return true;
|
|
174
|
+
} catch {
|
|
175
|
+
return false;
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
function isValidISODate(dateStr) {
|
|
179
|
+
const regex = /^\d{4}-\d{2}-\d{2}$/;
|
|
180
|
+
if (!regex.test(dateStr)) return false;
|
|
181
|
+
const date = new Date(dateStr);
|
|
182
|
+
return !isNaN(date.getTime());
|
|
183
|
+
}
|
|
184
|
+
function isValidCurrency(currency) {
|
|
185
|
+
const validCurrencies = ["CHF", "USD", "EUR", "PLN"];
|
|
186
|
+
return validCurrencies.includes(currency);
|
|
187
|
+
}
|
|
188
|
+
function isValidFrequency(frequency) {
|
|
189
|
+
const validFrequencies = ["monthly", "yearly", "6-monthly", "weekly", "one-time"];
|
|
190
|
+
return validFrequencies.includes(frequency);
|
|
191
|
+
}
|
|
192
|
+
function isPositiveNumber(value) {
|
|
193
|
+
return typeof value === "number" && !isNaN(value) && value > 0;
|
|
194
|
+
}
|
|
195
|
+
function isNonEmptyString(value) {
|
|
196
|
+
return typeof value === "string" && value.trim().length > 0;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
// src/constants/index.ts
|
|
200
|
+
var FREQUENCY_MULTIPLIERS = {
|
|
201
|
+
monthly: 12,
|
|
202
|
+
yearly: 1,
|
|
203
|
+
"6-monthly": 2,
|
|
204
|
+
weekly: 52,
|
|
205
|
+
"one-time": 1
|
|
206
|
+
};
|
|
207
|
+
|
|
208
|
+
// src/utils/calculations.ts
|
|
209
|
+
function toYearlyAmount(amount, frequency) {
|
|
210
|
+
const multiplier = FREQUENCY_MULTIPLIERS[frequency] || 1;
|
|
211
|
+
return amount * multiplier;
|
|
212
|
+
}
|
|
213
|
+
function toMonthlyAmount(amount, frequency) {
|
|
214
|
+
const yearly = toYearlyAmount(amount, frequency);
|
|
215
|
+
return yearly / 12;
|
|
216
|
+
}
|
|
217
|
+
function calculateMonthlyExpenses(expenses) {
|
|
218
|
+
return expenses.filter((e) => e.isActive).reduce((total, expense) => {
|
|
219
|
+
const monthly = toMonthlyAmount(
|
|
220
|
+
expense.amount,
|
|
221
|
+
expense.frequency
|
|
222
|
+
);
|
|
223
|
+
const myShare = monthly * expense.sharePercentage / 100;
|
|
224
|
+
return total + myShare;
|
|
225
|
+
}, 0);
|
|
226
|
+
}
|
|
227
|
+
function calculateMonthlyIncome(incomes) {
|
|
228
|
+
return incomes.filter((i) => i.isActive).reduce((total, income) => {
|
|
229
|
+
const monthly = toMonthlyAmount(income.amount, income.frequency);
|
|
230
|
+
return total + monthly;
|
|
231
|
+
}, 0);
|
|
232
|
+
}
|
|
233
|
+
function calculateMonthlySavings(incomes, expenses) {
|
|
234
|
+
return calculateMonthlyIncome(incomes) - calculateMonthlyExpenses(expenses);
|
|
235
|
+
}
|
|
236
|
+
function calculateSavingsRate(incomes, expenses) {
|
|
237
|
+
const income = calculateMonthlyIncome(incomes);
|
|
238
|
+
if (income === 0) return 0;
|
|
239
|
+
const savings = calculateMonthlySavings(incomes, expenses);
|
|
240
|
+
return savings / income * 100;
|
|
241
|
+
}
|
|
242
|
+
function calculateLieuBalance(lieuDays) {
|
|
243
|
+
return lieuDays.reduce((balance, day) => {
|
|
244
|
+
return day.type === "earned" ? balance + 1 : balance - 1;
|
|
245
|
+
}, 0);
|
|
246
|
+
}
|
|
247
|
+
function calculateFocusStats(sessions) {
|
|
248
|
+
const completed = sessions.filter((s) => s.status === "completed");
|
|
249
|
+
const abandoned = sessions.filter((s) => s.status === "abandoned");
|
|
250
|
+
const totalSeconds = completed.reduce((sum, s) => sum + s.actualSeconds, 0);
|
|
251
|
+
const completionRate = sessions.length > 0 ? completed.length / sessions.length * 100 : 0;
|
|
252
|
+
return {
|
|
253
|
+
totalSeconds,
|
|
254
|
+
completedCount: completed.length,
|
|
255
|
+
abandonedCount: abandoned.length,
|
|
256
|
+
completionRate
|
|
257
|
+
};
|
|
258
|
+
}
|
|
259
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
260
|
+
0 && (module.exports = {
|
|
261
|
+
calculateFocusStats,
|
|
262
|
+
calculateLieuBalance,
|
|
263
|
+
calculateMonthlyExpenses,
|
|
264
|
+
calculateMonthlyIncome,
|
|
265
|
+
calculateMonthlySavings,
|
|
266
|
+
calculateSavingsRate,
|
|
267
|
+
formatCurrency,
|
|
268
|
+
formatDate,
|
|
269
|
+
formatDueDate,
|
|
270
|
+
formatRelativeTime,
|
|
271
|
+
formatTime,
|
|
272
|
+
formatTotalTime,
|
|
273
|
+
generateId,
|
|
274
|
+
generateRandomColor,
|
|
275
|
+
generateShortId,
|
|
276
|
+
getRepoName,
|
|
277
|
+
isNonEmptyString,
|
|
278
|
+
isPositiveNumber,
|
|
279
|
+
isValidCurrency,
|
|
280
|
+
isValidEmail,
|
|
281
|
+
isValidFrequency,
|
|
282
|
+
isValidISODate,
|
|
283
|
+
isValidUrl,
|
|
284
|
+
toMonthlyAmount,
|
|
285
|
+
toYearlyAmount,
|
|
286
|
+
truncate
|
|
287
|
+
});
|
|
288
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/utils/index.ts","../../src/utils/formatters.ts","../../src/utils/generators.ts","../../src/utils/validators.ts","../../src/constants/index.ts","../../src/utils/calculations.ts"],"sourcesContent":["/**\n * Utils Index\n *\n * Re-exports all utility functions.\n */\n\nexport * from \"./formatters\";\nexport * from \"./generators\";\nexport * from \"./validators\";\nexport * from \"./calculations\";\n","/**\n * Formatting Utilities\n *\n * Pure functions for formatting data.\n */\n\n/**\n * Format seconds into MM:SS format\n * @example formatTime(125) => \"02:05\"\n */\nexport function formatTime(seconds: number): string {\n const mins = Math.floor(seconds / 60)\n const secs = seconds % 60\n return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`\n}\n\n/**\n * Format seconds into human-readable total time\n * @example formatTotalTime(3700) => \"1h 1m\"\n * @example formatTotalTime(1800) => \"30m\"\n */\nexport function formatTotalTime(seconds: number): string {\n const hours = Math.floor(seconds / 3600)\n const mins = Math.floor((seconds % 3600) / 60)\n if (hours > 0) {\n return `${hours}h ${mins}m`\n }\n return `${mins}m`\n}\n\n/**\n * Format a number as currency\n * @example formatCurrency(1234.56, 'CHF') => \"CHF 1,234.56\"\n */\nexport function formatCurrency(\n amount: number,\n currency: string,\n locale: string = 'en-CH'\n): string {\n return new Intl.NumberFormat(locale, {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n maximumFractionDigits: 2,\n }).format(amount)\n}\n\n/**\n * Format a date as relative time (e.g., \"2h ago\", \"3d ago\")\n */\nexport function formatRelativeTime(dateStr: string): string {\n const date = new Date(dateStr)\n const now = new Date()\n const diffMs = now.getTime() - date.getTime()\n const diffMins = Math.floor(diffMs / 60000)\n const diffHours = Math.floor(diffMins / 60)\n const diffDays = Math.floor(diffHours / 24)\n\n if (diffMins < 1) return 'Just now'\n if (diffMins < 60) return `${diffMins}m ago`\n if (diffHours < 24) return `${diffHours}h ago`\n if (diffDays < 7) return `${diffDays}d ago`\n return date.toLocaleDateString()\n}\n\n/**\n * Format a date string to a readable format\n * @example formatDate('2024-01-15') => \"Jan 15, 2024\"\n */\nexport function formatDate(\n dateStr: string,\n options: Intl.DateTimeFormatOptions = {\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n }\n): string {\n return new Date(dateStr).toLocaleDateString('en-US', options)\n}\n\n/**\n * Format a due date with context (Today, Tomorrow, Overdue, etc.)\n */\nexport function formatDueDate(dueDate: string | null): {\n text: string\n isOverdue: boolean\n} {\n if (!dueDate) return { text: '', isOverdue: false }\n\n const due = new Date(dueDate)\n const today = new Date()\n today.setHours(0, 0, 0, 0)\n\n const tomorrow = new Date(today)\n tomorrow.setDate(tomorrow.getDate() + 1)\n\n const dueDay = new Date(due)\n dueDay.setHours(0, 0, 0, 0)\n\n const isOverdue = dueDay < today\n\n if (dueDay.getTime() === today.getTime()) {\n return { text: 'Today', isOverdue: false }\n } else if (dueDay.getTime() === tomorrow.getTime()) {\n return { text: 'Tomorrow', isOverdue: false }\n } else if (isOverdue) {\n const daysAgo = Math.ceil(\n (today.getTime() - dueDay.getTime()) / (1000 * 60 * 60 * 24)\n )\n return { text: `${daysAgo}d overdue`, isOverdue: true }\n } else {\n return {\n text: due.toLocaleDateString('en-US', { month: 'short', day: 'numeric' }),\n isOverdue: false,\n }\n }\n}\n\n/**\n * Truncate a string with ellipsis\n * @example truncate(\"Hello World\", 5) => \"Hello...\"\n */\nexport function truncate(str: string, maxLength: number): string {\n if (str.length <= maxLength) return str\n return str.slice(0, maxLength) + '...'\n}\n\n/**\n * Extract repo name from GitHub URL\n * @example getRepoName(\"https://github.com/owner/repo\") => \"owner/repo\"\n */\nexport function getRepoName(url: string): string {\n const match = url.match(/github\\.com\\/(.+)$/)\n return match ? match[1] : url\n}\n\n","/**\n * ID and Data Generators\n *\n * Pure functions for generating IDs and data.\n */\n\n/**\n * Generate a unique ID\n * Uses crypto.randomUUID if available, falls back to timestamp-based ID\n *\n * Note: This works in both Node.js and browser environments.\n * React Native needs the fallback since crypto.randomUUID isn't available.\n */\nexport function generateId(): string {\n // Check if crypto.randomUUID is available (Node.js 19+, modern browsers)\n if (\n typeof crypto !== 'undefined' &&\n typeof crypto.randomUUID === 'function'\n ) {\n return crypto.randomUUID()\n }\n\n // Fallback: UUID v4-like implementation\n return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0\n const v = c === 'x' ? r : (r & 0x3) | 0x8\n return v.toString(16)\n })\n}\n\n/**\n * Generate a short ID (timestamp + random)\n * Format: {timestamp}-{random7chars}\n * @example \"1732547123456-k8f3j2m\"\n */\nexport function generateShortId(): string {\n return `${Date.now()}-${Math.random().toString(36).slice(2, 9)}`\n}\n\n/**\n * Generate a random color in hex format\n */\nexport function generateRandomColor(): string {\n const colors = [\n '#ef4444', // red\n '#f97316', // orange\n '#eab308', // yellow\n '#22c55e', // green\n '#14b8a6', // teal\n '#3b82f6', // blue\n '#8b5cf6', // violet\n '#ec4899', // pink\n ]\n return colors[Math.floor(Math.random() * colors.length)]\n}\n\n","/**\n * Validation Utilities\n *\n * Pure functions for validating data.\n */\n\n/**\n * Check if a string is a valid email\n */\nexport function isValidEmail(email: string): boolean {\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/\n return emailRegex.test(email)\n}\n\n/**\n * Check if a string is a valid URL\n */\nexport function isValidUrl(url: string): boolean {\n try {\n new URL(url)\n return true\n } catch {\n return false\n }\n}\n\n/**\n * Check if a string is a valid ISO date (YYYY-MM-DD)\n */\nexport function isValidISODate(dateStr: string): boolean {\n const regex = /^\\d{4}-\\d{2}-\\d{2}$/\n if (!regex.test(dateStr)) return false\n\n const date = new Date(dateStr)\n return !isNaN(date.getTime())\n}\n\n/**\n * Check if a value is a valid currency code\n */\nexport function isValidCurrency(currency: string): boolean {\n const validCurrencies = ['CHF', 'USD', 'EUR', 'PLN']\n return validCurrencies.includes(currency)\n}\n\n/**\n * Check if a value is a valid frequency\n */\nexport function isValidFrequency(frequency: string): boolean {\n const validFrequencies = ['monthly', 'yearly', '6-monthly', 'weekly', 'one-time']\n return validFrequencies.includes(frequency)\n}\n\n/**\n * Validate a positive number\n */\nexport function isPositiveNumber(value: unknown): value is number {\n return typeof value === 'number' && !isNaN(value) && value > 0\n}\n\n/**\n * Validate a non-empty string\n */\nexport function isNonEmptyString(value: unknown): value is string {\n return typeof value === 'string' && value.trim().length > 0\n}\n\n","/**\n * Shared Constants\n *\n * Common constants used across Vanaheim apps.\n */\n\n// ============================================================================\n// Currency\n// ============================================================================\n\nexport const CURRENCIES = ['CHF', 'USD', 'EUR', 'PLN'] as const\nexport type Currency = (typeof CURRENCIES)[number]\n\nexport const CURRENCY_SYMBOLS: Record<Currency, string> = {\n CHF: 'CHF',\n USD: '$',\n EUR: '€',\n PLN: 'zł',\n}\n\nexport const CURRENCY_NAMES: Record<Currency, string> = {\n CHF: 'Swiss Franc',\n USD: 'US Dollar',\n EUR: 'Euro',\n PLN: 'Polish Złoty',\n}\n\n// ============================================================================\n// Frequency\n// ============================================================================\n\nexport const FREQUENCIES = [\n 'monthly',\n 'yearly',\n '6-monthly',\n 'weekly',\n 'one-time',\n] as const\nexport type Frequency = (typeof FREQUENCIES)[number]\n\nexport const FREQUENCY_LABELS: Record<Frequency, string> = {\n monthly: 'Monthly',\n yearly: 'Yearly',\n '6-monthly': 'Every 6 Months',\n weekly: 'Weekly',\n 'one-time': 'One Time',\n}\n\nexport const FREQUENCY_MULTIPLIERS: Record<Frequency, number> = {\n monthly: 12,\n yearly: 1,\n '6-monthly': 2,\n weekly: 52,\n 'one-time': 1,\n}\n\n// ============================================================================\n// Focus\n// ============================================================================\n\nexport const DEFAULT_FOCUS_DURATIONS = [15, 25, 30, 45, 60, 90] as const\nexport type FocusDuration = (typeof DEFAULT_FOCUS_DURATIONS)[number]\n\nexport const FOCUS_STATUS = ['active', 'completed', 'abandoned'] as const\nexport type FocusStatus = (typeof FOCUS_STATUS)[number]\n\n// ============================================================================\n// Linear\n// ============================================================================\n\nexport const LINEAR_PRIORITIES = [0, 1, 2, 3, 4] as const\nexport type LinearPriorityValue = (typeof LINEAR_PRIORITIES)[number]\n\nexport const LINEAR_PRIORITY_COLORS: Record<LinearPriorityValue, string> = {\n 0: '#6b7280', // No priority - gray\n 1: '#ef4444', // Urgent - red\n 2: '#f97316', // High - orange\n 3: '#eab308', // Medium - yellow\n 4: '#3b82f6', // Low - blue\n}\n\n// ============================================================================\n// Cloud Agents\n// ============================================================================\n\nexport const CLOUD_AGENT_STATUSES = [\n 'CREATING',\n 'RUNNING',\n 'FINISHED',\n 'FAILED',\n 'CANCELLED',\n] as const\n\nexport const CLOUD_AGENT_STATUS_EMOJI: Record<string, string> = {\n CREATING: '🔨',\n RUNNING: '⏳',\n FINISHED: '✅',\n FAILED: '❌',\n CANCELLED: '🚫',\n}\n\nexport const CLOUD_AGENT_STATUS_COLORS: Record<string, string> = {\n CREATING: '#3b82f6',\n RUNNING: '#3b82f6',\n FINISHED: '#10b981',\n FAILED: '#ef4444',\n CANCELLED: '#6b7280',\n}\n\n// ============================================================================\n// API URLs\n// ============================================================================\n\nexport const API_URLS = {\n CURSOR_CLOUD: 'https://api.cursor.com',\n LINEAR_GRAPHQL: 'https://api.linear.app/graphql',\n} as const\n\n// ============================================================================\n// Settings Keys\n// ============================================================================\n\nexport const SETTING_KEYS = {\n // AI\n AI_MODEL: 'ai_model',\n AI_REASONING_ENABLED: 'ai_reasoning_enabled',\n\n // API Keys\n OPENAI_API_KEY: 'openai_api_key',\n ANTHROPIC_API_KEY: 'anthropic_api_key',\n CURSOR_API_KEY: 'cursor_api_key',\n LINEAR_API_KEY: 'linear_api_key',\n\n // Google\n GOOGLE_CLIENT_ID: 'google_client_id',\n GOOGLE_CLIENT_SECRET: 'google_client_secret',\n GOOGLE_CALENDAR_TOKENS: 'google_calendar_tokens',\n SELECTED_CALENDAR_IDS: 'selected_calendar_ids',\n} as const\n\nexport type SettingKey = (typeof SETTING_KEYS)[keyof typeof SETTING_KEYS]\n\n","/**\n * Calculation Utilities\n *\n * Pure functions for business logic calculations.\n */\n\nimport type { Expense, Income } from '../types/database'\nimport { FREQUENCY_MULTIPLIERS, type Frequency } from '../constants'\n\n/**\n * Convert an amount to yearly based on frequency\n */\nexport function toYearlyAmount(amount: number, frequency: Frequency): number {\n const multiplier = FREQUENCY_MULTIPLIERS[frequency] || 1\n return amount * multiplier\n}\n\n/**\n * Convert an amount to monthly based on frequency\n */\nexport function toMonthlyAmount(amount: number, frequency: Frequency): number {\n const yearly = toYearlyAmount(amount, frequency)\n return yearly / 12\n}\n\n/**\n * Calculate total expenses (monthly)\n */\nexport function calculateMonthlyExpenses(expenses: Expense[]): number {\n return expenses\n .filter((e) => e.isActive)\n .reduce((total, expense) => {\n const monthly = toMonthlyAmount(\n expense.amount,\n expense.frequency as Frequency\n )\n // Apply share percentage\n const myShare = (monthly * expense.sharePercentage) / 100\n return total + myShare\n }, 0)\n}\n\n/**\n * Calculate total income (monthly)\n */\nexport function calculateMonthlyIncome(incomes: Income[]): number {\n return incomes\n .filter((i) => i.isActive)\n .reduce((total, income) => {\n const monthly = toMonthlyAmount(income.amount, income.frequency as Frequency)\n return total + monthly\n }, 0)\n}\n\n/**\n * Calculate savings (income - expenses)\n */\nexport function calculateMonthlySavings(\n incomes: Income[],\n expenses: Expense[]\n): number {\n return calculateMonthlyIncome(incomes) - calculateMonthlyExpenses(expenses)\n}\n\n/**\n * Calculate savings rate as percentage\n */\nexport function calculateSavingsRate(\n incomes: Income[],\n expenses: Expense[]\n): number {\n const income = calculateMonthlyIncome(incomes)\n if (income === 0) return 0\n\n const savings = calculateMonthlySavings(incomes, expenses)\n return (savings / income) * 100\n}\n\n/**\n * Calculate lieu day balance\n */\nexport function calculateLieuBalance(\n lieuDays: Array<{ type: 'earned' | 'used' }>\n): number {\n return lieuDays.reduce((balance, day) => {\n return day.type === 'earned' ? balance + 1 : balance - 1\n }, 0)\n}\n\n/**\n * Calculate focus time stats for a period\n */\nexport function calculateFocusStats(\n sessions: Array<{ actualSeconds: number; status: string }>\n): {\n totalSeconds: number\n completedCount: number\n abandonedCount: number\n completionRate: number\n} {\n const completed = sessions.filter((s) => s.status === 'completed')\n const abandoned = sessions.filter((s) => s.status === 'abandoned')\n\n const totalSeconds = completed.reduce((sum, s) => sum + s.actualSeconds, 0)\n const completionRate =\n sessions.length > 0 ? (completed.length / sessions.length) * 100 : 0\n\n return {\n totalSeconds,\n completedCount: completed.length,\n abandonedCount: abandoned.length,\n completionRate,\n }\n}\n\n"],"mappings":"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","names":[]}
|
|
@@ -0,0 +1,236 @@
|
|
|
1
|
+
// src/utils/formatters.ts
|
|
2
|
+
function formatTime(seconds) {
|
|
3
|
+
const mins = Math.floor(seconds / 60);
|
|
4
|
+
const secs = seconds % 60;
|
|
5
|
+
return `${mins.toString().padStart(2, "0")}:${secs.toString().padStart(2, "0")}`;
|
|
6
|
+
}
|
|
7
|
+
function formatTotalTime(seconds) {
|
|
8
|
+
const hours = Math.floor(seconds / 3600);
|
|
9
|
+
const mins = Math.floor(seconds % 3600 / 60);
|
|
10
|
+
if (hours > 0) {
|
|
11
|
+
return `${hours}h ${mins}m`;
|
|
12
|
+
}
|
|
13
|
+
return `${mins}m`;
|
|
14
|
+
}
|
|
15
|
+
function formatCurrency(amount, currency, locale = "en-CH") {
|
|
16
|
+
return new Intl.NumberFormat(locale, {
|
|
17
|
+
style: "currency",
|
|
18
|
+
currency,
|
|
19
|
+
minimumFractionDigits: 2,
|
|
20
|
+
maximumFractionDigits: 2
|
|
21
|
+
}).format(amount);
|
|
22
|
+
}
|
|
23
|
+
function formatRelativeTime(dateStr) {
|
|
24
|
+
const date = new Date(dateStr);
|
|
25
|
+
const now = /* @__PURE__ */ new Date();
|
|
26
|
+
const diffMs = now.getTime() - date.getTime();
|
|
27
|
+
const diffMins = Math.floor(diffMs / 6e4);
|
|
28
|
+
const diffHours = Math.floor(diffMins / 60);
|
|
29
|
+
const diffDays = Math.floor(diffHours / 24);
|
|
30
|
+
if (diffMins < 1) return "Just now";
|
|
31
|
+
if (diffMins < 60) return `${diffMins}m ago`;
|
|
32
|
+
if (diffHours < 24) return `${diffHours}h ago`;
|
|
33
|
+
if (diffDays < 7) return `${diffDays}d ago`;
|
|
34
|
+
return date.toLocaleDateString();
|
|
35
|
+
}
|
|
36
|
+
function formatDate(dateStr, options = {
|
|
37
|
+
month: "short",
|
|
38
|
+
day: "numeric",
|
|
39
|
+
year: "numeric"
|
|
40
|
+
}) {
|
|
41
|
+
return new Date(dateStr).toLocaleDateString("en-US", options);
|
|
42
|
+
}
|
|
43
|
+
function formatDueDate(dueDate) {
|
|
44
|
+
if (!dueDate) return { text: "", isOverdue: false };
|
|
45
|
+
const due = new Date(dueDate);
|
|
46
|
+
const today = /* @__PURE__ */ new Date();
|
|
47
|
+
today.setHours(0, 0, 0, 0);
|
|
48
|
+
const tomorrow = new Date(today);
|
|
49
|
+
tomorrow.setDate(tomorrow.getDate() + 1);
|
|
50
|
+
const dueDay = new Date(due);
|
|
51
|
+
dueDay.setHours(0, 0, 0, 0);
|
|
52
|
+
const isOverdue = dueDay < today;
|
|
53
|
+
if (dueDay.getTime() === today.getTime()) {
|
|
54
|
+
return { text: "Today", isOverdue: false };
|
|
55
|
+
} else if (dueDay.getTime() === tomorrow.getTime()) {
|
|
56
|
+
return { text: "Tomorrow", isOverdue: false };
|
|
57
|
+
} else if (isOverdue) {
|
|
58
|
+
const daysAgo = Math.ceil(
|
|
59
|
+
(today.getTime() - dueDay.getTime()) / (1e3 * 60 * 60 * 24)
|
|
60
|
+
);
|
|
61
|
+
return { text: `${daysAgo}d overdue`, isOverdue: true };
|
|
62
|
+
} else {
|
|
63
|
+
return {
|
|
64
|
+
text: due.toLocaleDateString("en-US", { month: "short", day: "numeric" }),
|
|
65
|
+
isOverdue: false
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
function truncate(str, maxLength) {
|
|
70
|
+
if (str.length <= maxLength) return str;
|
|
71
|
+
return str.slice(0, maxLength) + "...";
|
|
72
|
+
}
|
|
73
|
+
function getRepoName(url) {
|
|
74
|
+
const match = url.match(/github\.com\/(.+)$/);
|
|
75
|
+
return match ? match[1] : url;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// src/utils/generators.ts
|
|
79
|
+
function generateId() {
|
|
80
|
+
if (typeof crypto !== "undefined" && typeof crypto.randomUUID === "function") {
|
|
81
|
+
return crypto.randomUUID();
|
|
82
|
+
}
|
|
83
|
+
return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, (c) => {
|
|
84
|
+
const r = Math.random() * 16 | 0;
|
|
85
|
+
const v = c === "x" ? r : r & 3 | 8;
|
|
86
|
+
return v.toString(16);
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
function generateShortId() {
|
|
90
|
+
return `${Date.now()}-${Math.random().toString(36).slice(2, 9)}`;
|
|
91
|
+
}
|
|
92
|
+
function generateRandomColor() {
|
|
93
|
+
const colors = [
|
|
94
|
+
"#ef4444",
|
|
95
|
+
// red
|
|
96
|
+
"#f97316",
|
|
97
|
+
// orange
|
|
98
|
+
"#eab308",
|
|
99
|
+
// yellow
|
|
100
|
+
"#22c55e",
|
|
101
|
+
// green
|
|
102
|
+
"#14b8a6",
|
|
103
|
+
// teal
|
|
104
|
+
"#3b82f6",
|
|
105
|
+
// blue
|
|
106
|
+
"#8b5cf6",
|
|
107
|
+
// violet
|
|
108
|
+
"#ec4899"
|
|
109
|
+
// pink
|
|
110
|
+
];
|
|
111
|
+
return colors[Math.floor(Math.random() * colors.length)];
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
// src/utils/validators.ts
|
|
115
|
+
function isValidEmail(email) {
|
|
116
|
+
const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
|
|
117
|
+
return emailRegex.test(email);
|
|
118
|
+
}
|
|
119
|
+
function isValidUrl(url) {
|
|
120
|
+
try {
|
|
121
|
+
new URL(url);
|
|
122
|
+
return true;
|
|
123
|
+
} catch {
|
|
124
|
+
return false;
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
function isValidISODate(dateStr) {
|
|
128
|
+
const regex = /^\d{4}-\d{2}-\d{2}$/;
|
|
129
|
+
if (!regex.test(dateStr)) return false;
|
|
130
|
+
const date = new Date(dateStr);
|
|
131
|
+
return !isNaN(date.getTime());
|
|
132
|
+
}
|
|
133
|
+
function isValidCurrency(currency) {
|
|
134
|
+
const validCurrencies = ["CHF", "USD", "EUR", "PLN"];
|
|
135
|
+
return validCurrencies.includes(currency);
|
|
136
|
+
}
|
|
137
|
+
function isValidFrequency(frequency) {
|
|
138
|
+
const validFrequencies = ["monthly", "yearly", "6-monthly", "weekly", "one-time"];
|
|
139
|
+
return validFrequencies.includes(frequency);
|
|
140
|
+
}
|
|
141
|
+
function isPositiveNumber(value) {
|
|
142
|
+
return typeof value === "number" && !isNaN(value) && value > 0;
|
|
143
|
+
}
|
|
144
|
+
function isNonEmptyString(value) {
|
|
145
|
+
return typeof value === "string" && value.trim().length > 0;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
// src/constants/index.ts
|
|
149
|
+
var FREQUENCY_MULTIPLIERS = {
|
|
150
|
+
monthly: 12,
|
|
151
|
+
yearly: 1,
|
|
152
|
+
"6-monthly": 2,
|
|
153
|
+
weekly: 52,
|
|
154
|
+
"one-time": 1
|
|
155
|
+
};
|
|
156
|
+
|
|
157
|
+
// src/utils/calculations.ts
|
|
158
|
+
function toYearlyAmount(amount, frequency) {
|
|
159
|
+
const multiplier = FREQUENCY_MULTIPLIERS[frequency] || 1;
|
|
160
|
+
return amount * multiplier;
|
|
161
|
+
}
|
|
162
|
+
function toMonthlyAmount(amount, frequency) {
|
|
163
|
+
const yearly = toYearlyAmount(amount, frequency);
|
|
164
|
+
return yearly / 12;
|
|
165
|
+
}
|
|
166
|
+
function calculateMonthlyExpenses(expenses) {
|
|
167
|
+
return expenses.filter((e) => e.isActive).reduce((total, expense) => {
|
|
168
|
+
const monthly = toMonthlyAmount(
|
|
169
|
+
expense.amount,
|
|
170
|
+
expense.frequency
|
|
171
|
+
);
|
|
172
|
+
const myShare = monthly * expense.sharePercentage / 100;
|
|
173
|
+
return total + myShare;
|
|
174
|
+
}, 0);
|
|
175
|
+
}
|
|
176
|
+
function calculateMonthlyIncome(incomes) {
|
|
177
|
+
return incomes.filter((i) => i.isActive).reduce((total, income) => {
|
|
178
|
+
const monthly = toMonthlyAmount(income.amount, income.frequency);
|
|
179
|
+
return total + monthly;
|
|
180
|
+
}, 0);
|
|
181
|
+
}
|
|
182
|
+
function calculateMonthlySavings(incomes, expenses) {
|
|
183
|
+
return calculateMonthlyIncome(incomes) - calculateMonthlyExpenses(expenses);
|
|
184
|
+
}
|
|
185
|
+
function calculateSavingsRate(incomes, expenses) {
|
|
186
|
+
const income = calculateMonthlyIncome(incomes);
|
|
187
|
+
if (income === 0) return 0;
|
|
188
|
+
const savings = calculateMonthlySavings(incomes, expenses);
|
|
189
|
+
return savings / income * 100;
|
|
190
|
+
}
|
|
191
|
+
function calculateLieuBalance(lieuDays) {
|
|
192
|
+
return lieuDays.reduce((balance, day) => {
|
|
193
|
+
return day.type === "earned" ? balance + 1 : balance - 1;
|
|
194
|
+
}, 0);
|
|
195
|
+
}
|
|
196
|
+
function calculateFocusStats(sessions) {
|
|
197
|
+
const completed = sessions.filter((s) => s.status === "completed");
|
|
198
|
+
const abandoned = sessions.filter((s) => s.status === "abandoned");
|
|
199
|
+
const totalSeconds = completed.reduce((sum, s) => sum + s.actualSeconds, 0);
|
|
200
|
+
const completionRate = sessions.length > 0 ? completed.length / sessions.length * 100 : 0;
|
|
201
|
+
return {
|
|
202
|
+
totalSeconds,
|
|
203
|
+
completedCount: completed.length,
|
|
204
|
+
abandonedCount: abandoned.length,
|
|
205
|
+
completionRate
|
|
206
|
+
};
|
|
207
|
+
}
|
|
208
|
+
export {
|
|
209
|
+
calculateFocusStats,
|
|
210
|
+
calculateLieuBalance,
|
|
211
|
+
calculateMonthlyExpenses,
|
|
212
|
+
calculateMonthlyIncome,
|
|
213
|
+
calculateMonthlySavings,
|
|
214
|
+
calculateSavingsRate,
|
|
215
|
+
formatCurrency,
|
|
216
|
+
formatDate,
|
|
217
|
+
formatDueDate,
|
|
218
|
+
formatRelativeTime,
|
|
219
|
+
formatTime,
|
|
220
|
+
formatTotalTime,
|
|
221
|
+
generateId,
|
|
222
|
+
generateRandomColor,
|
|
223
|
+
generateShortId,
|
|
224
|
+
getRepoName,
|
|
225
|
+
isNonEmptyString,
|
|
226
|
+
isPositiveNumber,
|
|
227
|
+
isValidCurrency,
|
|
228
|
+
isValidEmail,
|
|
229
|
+
isValidFrequency,
|
|
230
|
+
isValidISODate,
|
|
231
|
+
isValidUrl,
|
|
232
|
+
toMonthlyAmount,
|
|
233
|
+
toYearlyAmount,
|
|
234
|
+
truncate
|
|
235
|
+
};
|
|
236
|
+
//# sourceMappingURL=index.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/utils/formatters.ts","../../src/utils/generators.ts","../../src/utils/validators.ts","../../src/constants/index.ts","../../src/utils/calculations.ts"],"sourcesContent":["/**\n * Formatting Utilities\n *\n * Pure functions for formatting data.\n */\n\n/**\n * Format seconds into MM:SS format\n * @example formatTime(125) => \"02:05\"\n */\nexport function formatTime(seconds: number): string {\n const mins = Math.floor(seconds / 60)\n const secs = seconds % 60\n return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`\n}\n\n/**\n * Format seconds into human-readable total time\n * @example formatTotalTime(3700) => \"1h 1m\"\n * @example formatTotalTime(1800) => \"30m\"\n */\nexport function formatTotalTime(seconds: number): string {\n const hours = Math.floor(seconds / 3600)\n const mins = Math.floor((seconds % 3600) / 60)\n if (hours > 0) {\n return `${hours}h ${mins}m`\n }\n return `${mins}m`\n}\n\n/**\n * Format a number as currency\n * @example formatCurrency(1234.56, 'CHF') => \"CHF 1,234.56\"\n */\nexport function formatCurrency(\n amount: number,\n currency: string,\n locale: string = 'en-CH'\n): string {\n return new Intl.NumberFormat(locale, {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n maximumFractionDigits: 2,\n }).format(amount)\n}\n\n/**\n * Format a date as relative time (e.g., \"2h ago\", \"3d ago\")\n */\nexport function formatRelativeTime(dateStr: string): string {\n const date = new Date(dateStr)\n const now = new Date()\n const diffMs = now.getTime() - date.getTime()\n const diffMins = Math.floor(diffMs / 60000)\n const diffHours = Math.floor(diffMins / 60)\n const diffDays = Math.floor(diffHours / 24)\n\n if (diffMins < 1) return 'Just now'\n if (diffMins < 60) return `${diffMins}m ago`\n if (diffHours < 24) return `${diffHours}h ago`\n if (diffDays < 7) return `${diffDays}d ago`\n return date.toLocaleDateString()\n}\n\n/**\n * Format a date string to a readable format\n * @example formatDate('2024-01-15') => \"Jan 15, 2024\"\n */\nexport function formatDate(\n dateStr: string,\n options: Intl.DateTimeFormatOptions = {\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n }\n): string {\n return new Date(dateStr).toLocaleDateString('en-US', options)\n}\n\n/**\n * Format a due date with context (Today, Tomorrow, Overdue, etc.)\n */\nexport function formatDueDate(dueDate: string | null): {\n text: string\n isOverdue: boolean\n} {\n if (!dueDate) return { text: '', isOverdue: false }\n\n const due = new Date(dueDate)\n const today = new Date()\n today.setHours(0, 0, 0, 0)\n\n const tomorrow = new Date(today)\n tomorrow.setDate(tomorrow.getDate() + 1)\n\n const dueDay = new Date(due)\n dueDay.setHours(0, 0, 0, 0)\n\n const isOverdue = dueDay < today\n\n if (dueDay.getTime() === today.getTime()) {\n return { text: 'Today', isOverdue: false }\n } else if (dueDay.getTime() === tomorrow.getTime()) {\n return { text: 'Tomorrow', isOverdue: false }\n } else if (isOverdue) {\n const daysAgo = Math.ceil(\n (today.getTime() - dueDay.getTime()) / (1000 * 60 * 60 * 24)\n )\n return { text: `${daysAgo}d overdue`, isOverdue: true }\n } else {\n return {\n text: due.toLocaleDateString('en-US', { month: 'short', day: 'numeric' }),\n isOverdue: false,\n }\n }\n}\n\n/**\n * Truncate a string with ellipsis\n * @example truncate(\"Hello World\", 5) => \"Hello...\"\n */\nexport function truncate(str: string, maxLength: number): string {\n if (str.length <= maxLength) return str\n return str.slice(0, maxLength) + '...'\n}\n\n/**\n * Extract repo name from GitHub URL\n * @example getRepoName(\"https://github.com/owner/repo\") => \"owner/repo\"\n */\nexport function getRepoName(url: string): string {\n const match = url.match(/github\\.com\\/(.+)$/)\n return match ? match[1] : url\n}\n\n","/**\n * ID and Data Generators\n *\n * Pure functions for generating IDs and data.\n */\n\n/**\n * Generate a unique ID\n * Uses crypto.randomUUID if available, falls back to timestamp-based ID\n *\n * Note: This works in both Node.js and browser environments.\n * React Native needs the fallback since crypto.randomUUID isn't available.\n */\nexport function generateId(): string {\n // Check if crypto.randomUUID is available (Node.js 19+, modern browsers)\n if (\n typeof crypto !== 'undefined' &&\n typeof crypto.randomUUID === 'function'\n ) {\n return crypto.randomUUID()\n }\n\n // Fallback: UUID v4-like implementation\n return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0\n const v = c === 'x' ? r : (r & 0x3) | 0x8\n return v.toString(16)\n })\n}\n\n/**\n * Generate a short ID (timestamp + random)\n * Format: {timestamp}-{random7chars}\n * @example \"1732547123456-k8f3j2m\"\n */\nexport function generateShortId(): string {\n return `${Date.now()}-${Math.random().toString(36).slice(2, 9)}`\n}\n\n/**\n * Generate a random color in hex format\n */\nexport function generateRandomColor(): string {\n const colors = [\n '#ef4444', // red\n '#f97316', // orange\n '#eab308', // yellow\n '#22c55e', // green\n '#14b8a6', // teal\n '#3b82f6', // blue\n '#8b5cf6', // violet\n '#ec4899', // pink\n ]\n return colors[Math.floor(Math.random() * colors.length)]\n}\n\n","/**\n * Validation Utilities\n *\n * Pure functions for validating data.\n */\n\n/**\n * Check if a string is a valid email\n */\nexport function isValidEmail(email: string): boolean {\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/\n return emailRegex.test(email)\n}\n\n/**\n * Check if a string is a valid URL\n */\nexport function isValidUrl(url: string): boolean {\n try {\n new URL(url)\n return true\n } catch {\n return false\n }\n}\n\n/**\n * Check if a string is a valid ISO date (YYYY-MM-DD)\n */\nexport function isValidISODate(dateStr: string): boolean {\n const regex = /^\\d{4}-\\d{2}-\\d{2}$/\n if (!regex.test(dateStr)) return false\n\n const date = new Date(dateStr)\n return !isNaN(date.getTime())\n}\n\n/**\n * Check if a value is a valid currency code\n */\nexport function isValidCurrency(currency: string): boolean {\n const validCurrencies = ['CHF', 'USD', 'EUR', 'PLN']\n return validCurrencies.includes(currency)\n}\n\n/**\n * Check if a value is a valid frequency\n */\nexport function isValidFrequency(frequency: string): boolean {\n const validFrequencies = ['monthly', 'yearly', '6-monthly', 'weekly', 'one-time']\n return validFrequencies.includes(frequency)\n}\n\n/**\n * Validate a positive number\n */\nexport function isPositiveNumber(value: unknown): value is number {\n return typeof value === 'number' && !isNaN(value) && value > 0\n}\n\n/**\n * Validate a non-empty string\n */\nexport function isNonEmptyString(value: unknown): value is string {\n return typeof value === 'string' && value.trim().length > 0\n}\n\n","/**\n * Shared Constants\n *\n * Common constants used across Vanaheim apps.\n */\n\n// ============================================================================\n// Currency\n// ============================================================================\n\nexport const CURRENCIES = ['CHF', 'USD', 'EUR', 'PLN'] as const\nexport type Currency = (typeof CURRENCIES)[number]\n\nexport const CURRENCY_SYMBOLS: Record<Currency, string> = {\n CHF: 'CHF',\n USD: '$',\n EUR: '€',\n PLN: 'zł',\n}\n\nexport const CURRENCY_NAMES: Record<Currency, string> = {\n CHF: 'Swiss Franc',\n USD: 'US Dollar',\n EUR: 'Euro',\n PLN: 'Polish Złoty',\n}\n\n// ============================================================================\n// Frequency\n// ============================================================================\n\nexport const FREQUENCIES = [\n 'monthly',\n 'yearly',\n '6-monthly',\n 'weekly',\n 'one-time',\n] as const\nexport type Frequency = (typeof FREQUENCIES)[number]\n\nexport const FREQUENCY_LABELS: Record<Frequency, string> = {\n monthly: 'Monthly',\n yearly: 'Yearly',\n '6-monthly': 'Every 6 Months',\n weekly: 'Weekly',\n 'one-time': 'One Time',\n}\n\nexport const FREQUENCY_MULTIPLIERS: Record<Frequency, number> = {\n monthly: 12,\n yearly: 1,\n '6-monthly': 2,\n weekly: 52,\n 'one-time': 1,\n}\n\n// ============================================================================\n// Focus\n// ============================================================================\n\nexport const DEFAULT_FOCUS_DURATIONS = [15, 25, 30, 45, 60, 90] as const\nexport type FocusDuration = (typeof DEFAULT_FOCUS_DURATIONS)[number]\n\nexport const FOCUS_STATUS = ['active', 'completed', 'abandoned'] as const\nexport type FocusStatus = (typeof FOCUS_STATUS)[number]\n\n// ============================================================================\n// Linear\n// ============================================================================\n\nexport const LINEAR_PRIORITIES = [0, 1, 2, 3, 4] as const\nexport type LinearPriorityValue = (typeof LINEAR_PRIORITIES)[number]\n\nexport const LINEAR_PRIORITY_COLORS: Record<LinearPriorityValue, string> = {\n 0: '#6b7280', // No priority - gray\n 1: '#ef4444', // Urgent - red\n 2: '#f97316', // High - orange\n 3: '#eab308', // Medium - yellow\n 4: '#3b82f6', // Low - blue\n}\n\n// ============================================================================\n// Cloud Agents\n// ============================================================================\n\nexport const CLOUD_AGENT_STATUSES = [\n 'CREATING',\n 'RUNNING',\n 'FINISHED',\n 'FAILED',\n 'CANCELLED',\n] as const\n\nexport const CLOUD_AGENT_STATUS_EMOJI: Record<string, string> = {\n CREATING: '🔨',\n RUNNING: '⏳',\n FINISHED: '✅',\n FAILED: '❌',\n CANCELLED: '🚫',\n}\n\nexport const CLOUD_AGENT_STATUS_COLORS: Record<string, string> = {\n CREATING: '#3b82f6',\n RUNNING: '#3b82f6',\n FINISHED: '#10b981',\n FAILED: '#ef4444',\n CANCELLED: '#6b7280',\n}\n\n// ============================================================================\n// API URLs\n// ============================================================================\n\nexport const API_URLS = {\n CURSOR_CLOUD: 'https://api.cursor.com',\n LINEAR_GRAPHQL: 'https://api.linear.app/graphql',\n} as const\n\n// ============================================================================\n// Settings Keys\n// ============================================================================\n\nexport const SETTING_KEYS = {\n // AI\n AI_MODEL: 'ai_model',\n AI_REASONING_ENABLED: 'ai_reasoning_enabled',\n\n // API Keys\n OPENAI_API_KEY: 'openai_api_key',\n ANTHROPIC_API_KEY: 'anthropic_api_key',\n CURSOR_API_KEY: 'cursor_api_key',\n LINEAR_API_KEY: 'linear_api_key',\n\n // Google\n GOOGLE_CLIENT_ID: 'google_client_id',\n GOOGLE_CLIENT_SECRET: 'google_client_secret',\n GOOGLE_CALENDAR_TOKENS: 'google_calendar_tokens',\n SELECTED_CALENDAR_IDS: 'selected_calendar_ids',\n} as const\n\nexport type SettingKey = (typeof SETTING_KEYS)[keyof typeof SETTING_KEYS]\n\n","/**\n * Calculation Utilities\n *\n * Pure functions for business logic calculations.\n */\n\nimport type { Expense, Income } from '../types/database'\nimport { FREQUENCY_MULTIPLIERS, type Frequency } from '../constants'\n\n/**\n * Convert an amount to yearly based on frequency\n */\nexport function toYearlyAmount(amount: number, frequency: Frequency): number {\n const multiplier = FREQUENCY_MULTIPLIERS[frequency] || 1\n return amount * multiplier\n}\n\n/**\n * Convert an amount to monthly based on frequency\n */\nexport function toMonthlyAmount(amount: number, frequency: Frequency): number {\n const yearly = toYearlyAmount(amount, frequency)\n return yearly / 12\n}\n\n/**\n * Calculate total expenses (monthly)\n */\nexport function calculateMonthlyExpenses(expenses: Expense[]): number {\n return expenses\n .filter((e) => e.isActive)\n .reduce((total, expense) => {\n const monthly = toMonthlyAmount(\n expense.amount,\n expense.frequency as Frequency\n )\n // Apply share percentage\n const myShare = (monthly * expense.sharePercentage) / 100\n return total + myShare\n }, 0)\n}\n\n/**\n * Calculate total income (monthly)\n */\nexport function calculateMonthlyIncome(incomes: Income[]): number {\n return incomes\n .filter((i) => i.isActive)\n .reduce((total, income) => {\n const monthly = toMonthlyAmount(income.amount, income.frequency as Frequency)\n return total + monthly\n }, 0)\n}\n\n/**\n * Calculate savings (income - expenses)\n */\nexport function calculateMonthlySavings(\n incomes: Income[],\n expenses: Expense[]\n): number {\n return calculateMonthlyIncome(incomes) - calculateMonthlyExpenses(expenses)\n}\n\n/**\n * Calculate savings rate as percentage\n */\nexport function calculateSavingsRate(\n incomes: Income[],\n expenses: Expense[]\n): number {\n const income = calculateMonthlyIncome(incomes)\n if (income === 0) return 0\n\n const savings = calculateMonthlySavings(incomes, expenses)\n return (savings / income) * 100\n}\n\n/**\n * Calculate lieu day balance\n */\nexport function calculateLieuBalance(\n lieuDays: Array<{ type: 'earned' | 'used' }>\n): number {\n return lieuDays.reduce((balance, day) => {\n return day.type === 'earned' ? balance + 1 : balance - 1\n }, 0)\n}\n\n/**\n * Calculate focus time stats for a period\n */\nexport function calculateFocusStats(\n sessions: Array<{ actualSeconds: number; status: string }>\n): {\n totalSeconds: number\n completedCount: number\n abandonedCount: number\n completionRate: number\n} {\n const completed = sessions.filter((s) => s.status === 'completed')\n const abandoned = sessions.filter((s) => s.status === 'abandoned')\n\n const totalSeconds = completed.reduce((sum, s) => sum + s.actualSeconds, 0)\n const completionRate =\n sessions.length > 0 ? (completed.length / sessions.length) * 100 : 0\n\n return {\n totalSeconds,\n completedCount: completed.length,\n abandonedCount: abandoned.length,\n completionRate,\n }\n}\n\n"],"mappings":"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","names":[]}
|