@mrck-labs/vanaheim-shared 0.2.0 → 0.3.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.
- package/dist/atoms/index.js.map +1 -1
- package/dist/atoms/index.mjs.map +1 -1
- package/dist/constants/index.d.mts +53 -1
- package/dist/constants/index.d.ts +53 -1
- package/dist/constants/index.js +102 -0
- package/dist/constants/index.js.map +1 -1
- package/dist/constants/index.mjs +96 -0
- package/dist/constants/index.mjs.map +1 -1
- package/dist/{database-BKc0Oj26.d.mts → database-3Vv5PNDa.d.mts} +90 -6
- package/dist/{database-BKc0Oj26.d.ts → database-3Vv5PNDa.d.ts} +90 -6
- package/dist/date/index.d.mts +4 -0
- package/dist/date/index.d.ts +4 -0
- package/dist/date/index.js.map +1 -1
- package/dist/date/index.mjs.map +1 -1
- package/dist/index.d.mts +3 -3
- package/dist/index.d.ts +3 -3
- package/dist/index.js +192 -1
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +179 -1
- package/dist/index.mjs.map +1 -1
- package/dist/types/index.d.mts +1 -1
- package/dist/types/index.d.ts +1 -1
- package/dist/utils/index.d.mts +50 -3
- package/dist/utils/index.d.ts +50 -3
- package/dist/utils/index.js +90 -1
- package/dist/utils/index.js.map +1 -1
- package/dist/utils/index.mjs +83 -1
- package/dist/utils/index.mjs.map +1 -1
- package/package.json +1 -1
package/dist/utils/index.js
CHANGED
|
@@ -26,15 +26,18 @@ __export(utils_exports, {
|
|
|
26
26
|
calculateMonthlyIncome: () => calculateMonthlyIncome,
|
|
27
27
|
calculateMonthlySavings: () => calculateMonthlySavings,
|
|
28
28
|
calculateSavingsRate: () => calculateSavingsRate,
|
|
29
|
+
calculateWeeklyProgress: () => calculateWeeklyProgress,
|
|
29
30
|
formatCurrency: () => formatCurrency,
|
|
30
31
|
formatDate: () => formatDate,
|
|
31
32
|
formatDueDate: () => formatDueDate,
|
|
33
|
+
formatDuration: () => formatDuration,
|
|
32
34
|
formatRelativeTime: () => formatRelativeTime,
|
|
33
35
|
formatTime: () => formatTime,
|
|
34
36
|
formatTotalTime: () => formatTotalTime,
|
|
35
37
|
generateId: () => generateId,
|
|
36
38
|
generateRandomColor: () => generateRandomColor,
|
|
37
39
|
generateShortId: () => generateShortId,
|
|
40
|
+
getFrequencyDescription: () => getFrequencyDescription,
|
|
38
41
|
getRepoName: () => getRepoName,
|
|
39
42
|
isNonEmptyString: () => isNonEmptyString,
|
|
40
43
|
isPositiveNumber: () => isPositiveNumber,
|
|
@@ -43,6 +46,10 @@ __export(utils_exports, {
|
|
|
43
46
|
isValidFrequency: () => isValidFrequency,
|
|
44
47
|
isValidISODate: () => isValidISODate,
|
|
45
48
|
isValidUrl: () => isValidUrl,
|
|
49
|
+
parseFrequencyConfig: () => parseFrequencyConfig,
|
|
50
|
+
shouldDoOnDate: () => shouldDoOnDate,
|
|
51
|
+
shouldDoToday: () => shouldDoToday,
|
|
52
|
+
stringifyFrequencyConfig: () => stringifyFrequencyConfig,
|
|
46
53
|
toMonthlyAmount: () => toMonthlyAmount,
|
|
47
54
|
toYearlyAmount: () => toYearlyAmount,
|
|
48
55
|
truncate: () => truncate
|
|
@@ -63,7 +70,19 @@ function formatTotalTime(seconds) {
|
|
|
63
70
|
}
|
|
64
71
|
return `${mins}m`;
|
|
65
72
|
}
|
|
66
|
-
function
|
|
73
|
+
function formatDuration(seconds) {
|
|
74
|
+
const hours = Math.floor(seconds / 3600);
|
|
75
|
+
const minutes = Math.floor(seconds % 3600 / 60);
|
|
76
|
+
const secs = seconds % 60;
|
|
77
|
+
if (hours > 0) {
|
|
78
|
+
return `${hours}h ${minutes}m`;
|
|
79
|
+
} else if (minutes > 0) {
|
|
80
|
+
return `${minutes}m${secs > 0 ? ` ${secs}s` : ""}`;
|
|
81
|
+
} else {
|
|
82
|
+
return `${secs}s`;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
function formatCurrency(amount, currency = "CHF", locale = "de-CH") {
|
|
67
86
|
return new Intl.NumberFormat(locale, {
|
|
68
87
|
style: "currency",
|
|
69
88
|
currency,
|
|
@@ -256,6 +275,69 @@ function calculateFocusStats(sessions) {
|
|
|
256
275
|
completionRate
|
|
257
276
|
};
|
|
258
277
|
}
|
|
278
|
+
|
|
279
|
+
// src/utils/health.ts
|
|
280
|
+
function parseFrequencyConfig(config) {
|
|
281
|
+
if (!config) return null;
|
|
282
|
+
try {
|
|
283
|
+
return JSON.parse(config);
|
|
284
|
+
} catch {
|
|
285
|
+
return null;
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
function stringifyFrequencyConfig(config) {
|
|
289
|
+
if (!config) return null;
|
|
290
|
+
return JSON.stringify(config);
|
|
291
|
+
}
|
|
292
|
+
function getFrequencyDescription(habit) {
|
|
293
|
+
const config = parseFrequencyConfig(habit.frequencyConfig);
|
|
294
|
+
switch (habit.frequencyType) {
|
|
295
|
+
case "daily":
|
|
296
|
+
return "Every day";
|
|
297
|
+
case "specific_days": {
|
|
298
|
+
if (!config?.days) return "Specific days";
|
|
299
|
+
const dayNames = ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"];
|
|
300
|
+
return config.days.map((d) => dayNames[d]).join(", ");
|
|
301
|
+
}
|
|
302
|
+
case "times_per_week":
|
|
303
|
+
return `${config?.target || 1}x per week`;
|
|
304
|
+
case "times_per_month":
|
|
305
|
+
return `${config?.target || 1}x per month`;
|
|
306
|
+
default:
|
|
307
|
+
return "Unknown";
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
function shouldDoOnDate(habit, date) {
|
|
311
|
+
const dayOfWeek = date.getDay();
|
|
312
|
+
switch (habit.frequencyType) {
|
|
313
|
+
case "daily":
|
|
314
|
+
return true;
|
|
315
|
+
case "specific_days": {
|
|
316
|
+
const config = parseFrequencyConfig(habit.frequencyConfig);
|
|
317
|
+
return config?.days?.includes(dayOfWeek) ?? false;
|
|
318
|
+
}
|
|
319
|
+
case "times_per_week":
|
|
320
|
+
case "times_per_month":
|
|
321
|
+
return true;
|
|
322
|
+
default:
|
|
323
|
+
return true;
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
function shouldDoToday(habit) {
|
|
327
|
+
return shouldDoOnDate(habit, /* @__PURE__ */ new Date());
|
|
328
|
+
}
|
|
329
|
+
function calculateWeeklyProgress(habit, weekCompletionCount) {
|
|
330
|
+
const config = parseFrequencyConfig(habit.frequencyConfig);
|
|
331
|
+
const target = config?.target ?? 1;
|
|
332
|
+
const completed = weekCompletionCount;
|
|
333
|
+
return {
|
|
334
|
+
habit,
|
|
335
|
+
target,
|
|
336
|
+
completed,
|
|
337
|
+
remaining: Math.max(0, target - completed),
|
|
338
|
+
isComplete: completed >= target
|
|
339
|
+
};
|
|
340
|
+
}
|
|
259
341
|
// Annotate the CommonJS export names for ESM import in node:
|
|
260
342
|
0 && (module.exports = {
|
|
261
343
|
calculateFocusStats,
|
|
@@ -264,15 +346,18 @@ function calculateFocusStats(sessions) {
|
|
|
264
346
|
calculateMonthlyIncome,
|
|
265
347
|
calculateMonthlySavings,
|
|
266
348
|
calculateSavingsRate,
|
|
349
|
+
calculateWeeklyProgress,
|
|
267
350
|
formatCurrency,
|
|
268
351
|
formatDate,
|
|
269
352
|
formatDueDate,
|
|
353
|
+
formatDuration,
|
|
270
354
|
formatRelativeTime,
|
|
271
355
|
formatTime,
|
|
272
356
|
formatTotalTime,
|
|
273
357
|
generateId,
|
|
274
358
|
generateRandomColor,
|
|
275
359
|
generateShortId,
|
|
360
|
+
getFrequencyDescription,
|
|
276
361
|
getRepoName,
|
|
277
362
|
isNonEmptyString,
|
|
278
363
|
isPositiveNumber,
|
|
@@ -281,6 +366,10 @@ function calculateFocusStats(sessions) {
|
|
|
281
366
|
isValidFrequency,
|
|
282
367
|
isValidISODate,
|
|
283
368
|
isValidUrl,
|
|
369
|
+
parseFrequencyConfig,
|
|
370
|
+
shouldDoOnDate,
|
|
371
|
+
shouldDoToday,
|
|
372
|
+
stringifyFrequencyConfig,
|
|
284
373
|
toMonthlyAmount,
|
|
285
374
|
toYearlyAmount,
|
|
286
375
|
truncate
|
package/dist/utils/index.js.map
CHANGED
|
@@ -1 +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 * Uses Swiss German locale for authentic Swiss formatting (apostrophe as thousands separator)\n * @example formatCurrency(1234.56, 'CHF') => \"CHF 1'234.56\"\n */\nexport function formatCurrency(\n amount: number,\n currency: string,\n locale: string = 'de-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":[]}
|
|
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","../../src/utils/health.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\";\nexport * from \"./health\";\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 duration in seconds to human-readable format (includes seconds)\n * @example formatDuration(3661) => \"1h 1m\"\n * @example formatDuration(125) => \"2m 5s\"\n * @example formatDuration(45) => \"45s\"\n */\nexport function formatDuration(seconds: number): string {\n const hours = Math.floor(seconds / 3600)\n const minutes = Math.floor((seconds % 3600) / 60)\n const secs = seconds % 60\n\n if (hours > 0) {\n return `${hours}h ${minutes}m`\n } else if (minutes > 0) {\n return `${minutes}m${secs > 0 ? ` ${secs}s` : ''}`\n } else {\n return `${secs}s`\n }\n}\n\n/**\n * Format a number as currency\n * Uses Swiss German locale for authentic Swiss formatting (apostrophe as thousands separator)\n * @example formatCurrency(1234.56, 'CHF') => \"CHF 1'234.56\"\n * @example formatCurrency(1234.56) => \"CHF 1'234.56\" (defaults to CHF)\n */\nexport function formatCurrency(\n amount: number,\n currency: string = 'CHF',\n locale: string = 'de-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 * Frequency options for dropdowns/toggle groups\n * Format: { value: Frequency, label: string }\n */\nexport const FREQUENCY_OPTIONS = [\n { value: \"monthly\" as const, label: \"Monthly\" },\n { value: \"yearly\" as const, label: \"Yearly\" },\n { value: \"6-monthly\" as const, label: \"Every 6 Months\" },\n { value: \"weekly\" as const, label: \"Weekly\" },\n { value: \"one-time\" as const, label: \"One Time\" },\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// Health & Habits\n// ============================================================================\n\nexport const HEALTH_FREQUENCY_TYPES = [\n \"daily\",\n \"specific_days\",\n \"times_per_week\",\n \"times_per_month\",\n] as const;\n\nexport const HEALTH_FREQUENCY_LABELS: Record<string, string> = {\n daily: \"Daily\",\n specific_days: \"Specific Days\",\n times_per_week: \"Times per Week\",\n times_per_month: \"Times per Month\",\n};\n\nexport const DAY_OPTIONS = [\n { value: 0, label: \"Sun\" },\n { value: 1, label: \"Mon\" },\n { value: 2, label: \"Tue\" },\n { value: 3, label: \"Wed\" },\n { value: 4, label: \"Thu\" },\n { value: 5, label: \"Fri\" },\n { value: 6, label: \"Sat\" },\n] as const;\n\n/**\n * Preset colors for categories, labels, and other UI elements\n */\nexport const PRESET_COLORS = [\n \"#10B981\", // Emerald\n \"#3B82F6\", // Blue\n \"#F59E0B\", // Amber\n \"#EF4444\", // Red\n \"#8B5CF6\", // Purple\n \"#EC4899\", // Pink\n \"#14B8A6\", // Teal\n \"#F97316\", // Orange\n \"#6366F1\", // Indigo\n \"#84CC16\", // Lime\n];\n\n/**\n * Preset icons/emojis for categories and UI elements\n */\nexport const PRESET_ICONS = [\n \"📝\", // Writing/Notes\n \"💻\", // Coding/Tech\n \"🎨\", // Design/Art\n \"📊\", // Analytics/Data\n \"🔧\", // Tools/Settings\n \"📚\", // Learning/Docs\n \"💡\", // Ideas/Lightbulb\n \"🚀\", // Launch/Start\n \"⚡\", // Fast/Energy\n \"🎯\", // Target/Goal\n \"💊\", // Health/Medicine\n \"🥗\", // Food/Diet\n \"💧\", // Water/Hydration\n \"🏃\", // Exercise/Running\n \"😴\", // Sleep/Rest\n \"🧘\", // Meditation/Mindfulness\n \"💪\", // Strength/Fitness\n \"🍎\", // Health/Food\n \"🔗\", // Links/Connections\n];\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 * 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","/**\n * Health & Habits Utilities\n *\n * Shared utility functions for the health & habits module.\n * Used by both desktop and mobile apps.\n */\n\nimport type { HealthHabit, HealthFrequencyConfig } from \"../types/database\";\n\n// ============================================================================\n// Frequency Config Helpers\n// ============================================================================\n\n/**\n * Parse frequency config from JSON string\n */\nexport function parseFrequencyConfig(\n config: string | null\n): HealthFrequencyConfig | null {\n if (!config) return null;\n try {\n return JSON.parse(config);\n } catch {\n return null;\n }\n}\n\n/**\n * Stringify frequency config to JSON\n */\nexport function stringifyFrequencyConfig(\n config: HealthFrequencyConfig | null\n): string | null {\n if (!config) return null;\n return JSON.stringify(config);\n}\n\n// ============================================================================\n// Frequency Description\n// ============================================================================\n\n/**\n * Get human-readable frequency description for a habit\n */\nexport function getFrequencyDescription(habit: HealthHabit): string {\n const config = parseFrequencyConfig(habit.frequencyConfig);\n\n switch (habit.frequencyType) {\n case \"daily\":\n return \"Every day\";\n case \"specific_days\": {\n if (!config?.days) return \"Specific days\";\n const dayNames = [\"Sun\", \"Mon\", \"Tue\", \"Wed\", \"Thu\", \"Fri\", \"Sat\"];\n return config.days.map((d) => dayNames[d]).join(\", \");\n }\n case \"times_per_week\":\n return `${config?.target || 1}x per week`;\n case \"times_per_month\":\n return `${config?.target || 1}x per month`;\n default:\n return \"Unknown\";\n }\n}\n\n// ============================================================================\n// Schedule Helpers\n// ============================================================================\n\n/**\n * Check if habit should be done on a specific date based on frequency\n */\nexport function shouldDoOnDate(habit: HealthHabit, date: Date): boolean {\n const dayOfWeek = date.getDay(); // 0 = Sunday\n\n switch (habit.frequencyType) {\n case \"daily\":\n return true;\n case \"specific_days\": {\n const config = parseFrequencyConfig(habit.frequencyConfig);\n return config?.days?.includes(dayOfWeek) ?? false;\n }\n case \"times_per_week\":\n case \"times_per_month\":\n // These are flexible - always show as available\n return true;\n default:\n return true;\n }\n}\n\n/**\n * Check if habit should be done today\n */\nexport function shouldDoToday(habit: HealthHabit): boolean {\n return shouldDoOnDate(habit, new Date());\n}\n\n// ============================================================================\n// Weekly Progress\n// ============================================================================\n\nexport interface WeeklyProgressItem {\n habit: HealthHabit;\n target: number;\n completed: number;\n remaining: number;\n isComplete: boolean;\n}\n\n/**\n * Calculate weekly progress for a habit\n */\nexport function calculateWeeklyProgress(\n habit: HealthHabit,\n weekCompletionCount: number\n): WeeklyProgressItem {\n const config = parseFrequencyConfig(habit.frequencyConfig);\n const target = config?.target ?? 1;\n const completed = weekCompletionCount;\n\n return {\n habit,\n target,\n completed,\n remaining: Math.max(0, target - completed),\n isComplete: completed >= target,\n };\n}\n"],"mappings":"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","names":[]}
|
package/dist/utils/index.mjs
CHANGED
|
@@ -12,7 +12,19 @@ function formatTotalTime(seconds) {
|
|
|
12
12
|
}
|
|
13
13
|
return `${mins}m`;
|
|
14
14
|
}
|
|
15
|
-
function
|
|
15
|
+
function formatDuration(seconds) {
|
|
16
|
+
const hours = Math.floor(seconds / 3600);
|
|
17
|
+
const minutes = Math.floor(seconds % 3600 / 60);
|
|
18
|
+
const secs = seconds % 60;
|
|
19
|
+
if (hours > 0) {
|
|
20
|
+
return `${hours}h ${minutes}m`;
|
|
21
|
+
} else if (minutes > 0) {
|
|
22
|
+
return `${minutes}m${secs > 0 ? ` ${secs}s` : ""}`;
|
|
23
|
+
} else {
|
|
24
|
+
return `${secs}s`;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
function formatCurrency(amount, currency = "CHF", locale = "de-CH") {
|
|
16
28
|
return new Intl.NumberFormat(locale, {
|
|
17
29
|
style: "currency",
|
|
18
30
|
currency,
|
|
@@ -205,6 +217,69 @@ function calculateFocusStats(sessions) {
|
|
|
205
217
|
completionRate
|
|
206
218
|
};
|
|
207
219
|
}
|
|
220
|
+
|
|
221
|
+
// src/utils/health.ts
|
|
222
|
+
function parseFrequencyConfig(config) {
|
|
223
|
+
if (!config) return null;
|
|
224
|
+
try {
|
|
225
|
+
return JSON.parse(config);
|
|
226
|
+
} catch {
|
|
227
|
+
return null;
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
function stringifyFrequencyConfig(config) {
|
|
231
|
+
if (!config) return null;
|
|
232
|
+
return JSON.stringify(config);
|
|
233
|
+
}
|
|
234
|
+
function getFrequencyDescription(habit) {
|
|
235
|
+
const config = parseFrequencyConfig(habit.frequencyConfig);
|
|
236
|
+
switch (habit.frequencyType) {
|
|
237
|
+
case "daily":
|
|
238
|
+
return "Every day";
|
|
239
|
+
case "specific_days": {
|
|
240
|
+
if (!config?.days) return "Specific days";
|
|
241
|
+
const dayNames = ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"];
|
|
242
|
+
return config.days.map((d) => dayNames[d]).join(", ");
|
|
243
|
+
}
|
|
244
|
+
case "times_per_week":
|
|
245
|
+
return `${config?.target || 1}x per week`;
|
|
246
|
+
case "times_per_month":
|
|
247
|
+
return `${config?.target || 1}x per month`;
|
|
248
|
+
default:
|
|
249
|
+
return "Unknown";
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
function shouldDoOnDate(habit, date) {
|
|
253
|
+
const dayOfWeek = date.getDay();
|
|
254
|
+
switch (habit.frequencyType) {
|
|
255
|
+
case "daily":
|
|
256
|
+
return true;
|
|
257
|
+
case "specific_days": {
|
|
258
|
+
const config = parseFrequencyConfig(habit.frequencyConfig);
|
|
259
|
+
return config?.days?.includes(dayOfWeek) ?? false;
|
|
260
|
+
}
|
|
261
|
+
case "times_per_week":
|
|
262
|
+
case "times_per_month":
|
|
263
|
+
return true;
|
|
264
|
+
default:
|
|
265
|
+
return true;
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
function shouldDoToday(habit) {
|
|
269
|
+
return shouldDoOnDate(habit, /* @__PURE__ */ new Date());
|
|
270
|
+
}
|
|
271
|
+
function calculateWeeklyProgress(habit, weekCompletionCount) {
|
|
272
|
+
const config = parseFrequencyConfig(habit.frequencyConfig);
|
|
273
|
+
const target = config?.target ?? 1;
|
|
274
|
+
const completed = weekCompletionCount;
|
|
275
|
+
return {
|
|
276
|
+
habit,
|
|
277
|
+
target,
|
|
278
|
+
completed,
|
|
279
|
+
remaining: Math.max(0, target - completed),
|
|
280
|
+
isComplete: completed >= target
|
|
281
|
+
};
|
|
282
|
+
}
|
|
208
283
|
export {
|
|
209
284
|
calculateFocusStats,
|
|
210
285
|
calculateLieuBalance,
|
|
@@ -212,15 +287,18 @@ export {
|
|
|
212
287
|
calculateMonthlyIncome,
|
|
213
288
|
calculateMonthlySavings,
|
|
214
289
|
calculateSavingsRate,
|
|
290
|
+
calculateWeeklyProgress,
|
|
215
291
|
formatCurrency,
|
|
216
292
|
formatDate,
|
|
217
293
|
formatDueDate,
|
|
294
|
+
formatDuration,
|
|
218
295
|
formatRelativeTime,
|
|
219
296
|
formatTime,
|
|
220
297
|
formatTotalTime,
|
|
221
298
|
generateId,
|
|
222
299
|
generateRandomColor,
|
|
223
300
|
generateShortId,
|
|
301
|
+
getFrequencyDescription,
|
|
224
302
|
getRepoName,
|
|
225
303
|
isNonEmptyString,
|
|
226
304
|
isPositiveNumber,
|
|
@@ -229,6 +307,10 @@ export {
|
|
|
229
307
|
isValidFrequency,
|
|
230
308
|
isValidISODate,
|
|
231
309
|
isValidUrl,
|
|
310
|
+
parseFrequencyConfig,
|
|
311
|
+
shouldDoOnDate,
|
|
312
|
+
shouldDoToday,
|
|
313
|
+
stringifyFrequencyConfig,
|
|
232
314
|
toMonthlyAmount,
|
|
233
315
|
toYearlyAmount,
|
|
234
316
|
truncate
|
package/dist/utils/index.mjs.map
CHANGED
|
@@ -1 +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 * Uses Swiss German locale for authentic Swiss formatting (apostrophe as thousands separator)\n * @example formatCurrency(1234.56, 'CHF') => \"CHF 1'234.56\"\n */\nexport function formatCurrency(\n amount: number,\n currency: string,\n locale: string = 'de-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":[]}
|
|
1
|
+
{"version":3,"sources":["../../src/utils/formatters.ts","../../src/utils/generators.ts","../../src/utils/validators.ts","../../src/constants/index.ts","../../src/utils/calculations.ts","../../src/utils/health.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 duration in seconds to human-readable format (includes seconds)\n * @example formatDuration(3661) => \"1h 1m\"\n * @example formatDuration(125) => \"2m 5s\"\n * @example formatDuration(45) => \"45s\"\n */\nexport function formatDuration(seconds: number): string {\n const hours = Math.floor(seconds / 3600)\n const minutes = Math.floor((seconds % 3600) / 60)\n const secs = seconds % 60\n\n if (hours > 0) {\n return `${hours}h ${minutes}m`\n } else if (minutes > 0) {\n return `${minutes}m${secs > 0 ? ` ${secs}s` : ''}`\n } else {\n return `${secs}s`\n }\n}\n\n/**\n * Format a number as currency\n * Uses Swiss German locale for authentic Swiss formatting (apostrophe as thousands separator)\n * @example formatCurrency(1234.56, 'CHF') => \"CHF 1'234.56\"\n * @example formatCurrency(1234.56) => \"CHF 1'234.56\" (defaults to CHF)\n */\nexport function formatCurrency(\n amount: number,\n currency: string = 'CHF',\n locale: string = 'de-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 * Frequency options for dropdowns/toggle groups\n * Format: { value: Frequency, label: string }\n */\nexport const FREQUENCY_OPTIONS = [\n { value: \"monthly\" as const, label: \"Monthly\" },\n { value: \"yearly\" as const, label: \"Yearly\" },\n { value: \"6-monthly\" as const, label: \"Every 6 Months\" },\n { value: \"weekly\" as const, label: \"Weekly\" },\n { value: \"one-time\" as const, label: \"One Time\" },\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// Health & Habits\n// ============================================================================\n\nexport const HEALTH_FREQUENCY_TYPES = [\n \"daily\",\n \"specific_days\",\n \"times_per_week\",\n \"times_per_month\",\n] as const;\n\nexport const HEALTH_FREQUENCY_LABELS: Record<string, string> = {\n daily: \"Daily\",\n specific_days: \"Specific Days\",\n times_per_week: \"Times per Week\",\n times_per_month: \"Times per Month\",\n};\n\nexport const DAY_OPTIONS = [\n { value: 0, label: \"Sun\" },\n { value: 1, label: \"Mon\" },\n { value: 2, label: \"Tue\" },\n { value: 3, label: \"Wed\" },\n { value: 4, label: \"Thu\" },\n { value: 5, label: \"Fri\" },\n { value: 6, label: \"Sat\" },\n] as const;\n\n/**\n * Preset colors for categories, labels, and other UI elements\n */\nexport const PRESET_COLORS = [\n \"#10B981\", // Emerald\n \"#3B82F6\", // Blue\n \"#F59E0B\", // Amber\n \"#EF4444\", // Red\n \"#8B5CF6\", // Purple\n \"#EC4899\", // Pink\n \"#14B8A6\", // Teal\n \"#F97316\", // Orange\n \"#6366F1\", // Indigo\n \"#84CC16\", // Lime\n];\n\n/**\n * Preset icons/emojis for categories and UI elements\n */\nexport const PRESET_ICONS = [\n \"📝\", // Writing/Notes\n \"💻\", // Coding/Tech\n \"🎨\", // Design/Art\n \"📊\", // Analytics/Data\n \"🔧\", // Tools/Settings\n \"📚\", // Learning/Docs\n \"💡\", // Ideas/Lightbulb\n \"🚀\", // Launch/Start\n \"⚡\", // Fast/Energy\n \"🎯\", // Target/Goal\n \"💊\", // Health/Medicine\n \"🥗\", // Food/Diet\n \"💧\", // Water/Hydration\n \"🏃\", // Exercise/Running\n \"😴\", // Sleep/Rest\n \"🧘\", // Meditation/Mindfulness\n \"💪\", // Strength/Fitness\n \"🍎\", // Health/Food\n \"🔗\", // Links/Connections\n];\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 * 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","/**\n * Health & Habits Utilities\n *\n * Shared utility functions for the health & habits module.\n * Used by both desktop and mobile apps.\n */\n\nimport type { HealthHabit, HealthFrequencyConfig } from \"../types/database\";\n\n// ============================================================================\n// Frequency Config Helpers\n// ============================================================================\n\n/**\n * Parse frequency config from JSON string\n */\nexport function parseFrequencyConfig(\n config: string | null\n): HealthFrequencyConfig | null {\n if (!config) return null;\n try {\n return JSON.parse(config);\n } catch {\n return null;\n }\n}\n\n/**\n * Stringify frequency config to JSON\n */\nexport function stringifyFrequencyConfig(\n config: HealthFrequencyConfig | null\n): string | null {\n if (!config) return null;\n return JSON.stringify(config);\n}\n\n// ============================================================================\n// Frequency Description\n// ============================================================================\n\n/**\n * Get human-readable frequency description for a habit\n */\nexport function getFrequencyDescription(habit: HealthHabit): string {\n const config = parseFrequencyConfig(habit.frequencyConfig);\n\n switch (habit.frequencyType) {\n case \"daily\":\n return \"Every day\";\n case \"specific_days\": {\n if (!config?.days) return \"Specific days\";\n const dayNames = [\"Sun\", \"Mon\", \"Tue\", \"Wed\", \"Thu\", \"Fri\", \"Sat\"];\n return config.days.map((d) => dayNames[d]).join(\", \");\n }\n case \"times_per_week\":\n return `${config?.target || 1}x per week`;\n case \"times_per_month\":\n return `${config?.target || 1}x per month`;\n default:\n return \"Unknown\";\n }\n}\n\n// ============================================================================\n// Schedule Helpers\n// ============================================================================\n\n/**\n * Check if habit should be done on a specific date based on frequency\n */\nexport function shouldDoOnDate(habit: HealthHabit, date: Date): boolean {\n const dayOfWeek = date.getDay(); // 0 = Sunday\n\n switch (habit.frequencyType) {\n case \"daily\":\n return true;\n case \"specific_days\": {\n const config = parseFrequencyConfig(habit.frequencyConfig);\n return config?.days?.includes(dayOfWeek) ?? false;\n }\n case \"times_per_week\":\n case \"times_per_month\":\n // These are flexible - always show as available\n return true;\n default:\n return true;\n }\n}\n\n/**\n * Check if habit should be done today\n */\nexport function shouldDoToday(habit: HealthHabit): boolean {\n return shouldDoOnDate(habit, new Date());\n}\n\n// ============================================================================\n// Weekly Progress\n// ============================================================================\n\nexport interface WeeklyProgressItem {\n habit: HealthHabit;\n target: number;\n completed: number;\n remaining: number;\n isComplete: boolean;\n}\n\n/**\n * Calculate weekly progress for a habit\n */\nexport function calculateWeeklyProgress(\n habit: HealthHabit,\n weekCompletionCount: number\n): WeeklyProgressItem {\n const config = parseFrequencyConfig(habit.frequencyConfig);\n const target = config?.target ?? 1;\n const completed = weekCompletionCount;\n\n return {\n habit,\n target,\n completed,\n remaining: Math.max(0, target - completed),\n isComplete: completed >= target,\n };\n}\n"],"mappings":"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","names":[]}
|