react-os-shell 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (65) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +242 -0
  3. package/dist/Calculator-BNBRNV4P.js +184 -0
  4. package/dist/Calculator-BNBRNV4P.js.map +1 -0
  5. package/dist/Calendar-5EYUVGUU.js +423 -0
  6. package/dist/Calendar-5EYUVGUU.js.map +1 -0
  7. package/dist/Checkers-MIAHIKJH.js +214 -0
  8. package/dist/Checkers-MIAHIKJH.js.map +1 -0
  9. package/dist/Chess-C5BY45NA.js +190 -0
  10. package/dist/Chess-C5BY45NA.js.map +1 -0
  11. package/dist/ConfirmDialog-ZP4AHVUD.js +3 -0
  12. package/dist/ConfirmDialog-ZP4AHVUD.js.map +1 -0
  13. package/dist/CurrencyConverter-TYPU2IRF.js +223 -0
  14. package/dist/CurrencyConverter-TYPU2IRF.js.map +1 -0
  15. package/dist/Email-JEYYJ3YV.js +1835 -0
  16. package/dist/Email-JEYYJ3YV.js.map +1 -0
  17. package/dist/Game2048-3RH3ELRD.js +191 -0
  18. package/dist/Game2048-3RH3ELRD.js.map +1 -0
  19. package/dist/GeminiChat-BXLBJFT4.js +184 -0
  20. package/dist/GeminiChat-BXLBJFT4.js.map +1 -0
  21. package/dist/Minesweeper-VQGLAZON.js +270 -0
  22. package/dist/Minesweeper-VQGLAZON.js.map +1 -0
  23. package/dist/Notepad-YTZRCAXX.js +389 -0
  24. package/dist/Notepad-YTZRCAXX.js.map +1 -0
  25. package/dist/PomodoroTimer-HARIJN4S.js +196 -0
  26. package/dist/PomodoroTimer-HARIJN4S.js.map +1 -0
  27. package/dist/Spreadsheet-IOKEDNS6.js +446 -0
  28. package/dist/Spreadsheet-IOKEDNS6.js.map +1 -0
  29. package/dist/Sudoku-XHLYCEVT.js +197 -0
  30. package/dist/Sudoku-XHLYCEVT.js.map +1 -0
  31. package/dist/Tetris-ZHCZYL24.js +243 -0
  32. package/dist/Tetris-ZHCZYL24.js.map +1 -0
  33. package/dist/Weather-ROZ7TRNW.js +310 -0
  34. package/dist/Weather-ROZ7TRNW.js.map +1 -0
  35. package/dist/apps/index.d.ts +55 -0
  36. package/dist/apps/index.js +48 -0
  37. package/dist/apps/index.js.map +1 -0
  38. package/dist/chunk-5O2KEISQ.js +155 -0
  39. package/dist/chunk-5O2KEISQ.js.map +1 -0
  40. package/dist/chunk-D7PYW2QS.js +265 -0
  41. package/dist/chunk-D7PYW2QS.js.map +1 -0
  42. package/dist/chunk-GP4Y3VCB.js +806 -0
  43. package/dist/chunk-GP4Y3VCB.js.map +1 -0
  44. package/dist/chunk-NSU7OHPC.js +39 -0
  45. package/dist/chunk-NSU7OHPC.js.map +1 -0
  46. package/dist/chunk-PDFQNHW7.js +24 -0
  47. package/dist/chunk-PDFQNHW7.js.map +1 -0
  48. package/dist/chunk-RFTLYCSF.js +144 -0
  49. package/dist/chunk-RFTLYCSF.js.map +1 -0
  50. package/dist/chunk-SVBID2P6.js +142 -0
  51. package/dist/chunk-SVBID2P6.js.map +1 -0
  52. package/dist/chunk-TFGOLXGD.js +41 -0
  53. package/dist/chunk-TFGOLXGD.js.map +1 -0
  54. package/dist/chunk-WIJ45SYD.js +120 -0
  55. package/dist/chunk-WIJ45SYD.js.map +1 -0
  56. package/dist/chunk-WQIS72NL.js +1470 -0
  57. package/dist/chunk-WQIS72NL.js.map +1 -0
  58. package/dist/index.d.ts +642 -0
  59. package/dist/index.js +3443 -0
  60. package/dist/index.js.map +1 -0
  61. package/dist/sounds-NT4DEZGD.js +3 -0
  62. package/dist/sounds-NT4DEZGD.js.map +1 -0
  63. package/dist/styles.css +174 -0
  64. package/dist/types-CFIZ1_xt.d.ts +67 -0
  65. package/package.json +76 -0
@@ -0,0 +1,155 @@
1
+ import { useState, useRef, useCallback, useEffect } from 'react';
2
+
3
+ // src/hooks/useGoogleAuth.ts
4
+ var SCOPES = [
5
+ "https://www.googleapis.com/auth/gmail.readonly",
6
+ "https://www.googleapis.com/auth/gmail.compose",
7
+ "https://www.googleapis.com/auth/gmail.send",
8
+ "https://www.googleapis.com/auth/gmail.modify",
9
+ "https://www.googleapis.com/auth/calendar.readonly",
10
+ "https://www.googleapis.com/auth/calendar.events",
11
+ "https://www.googleapis.com/auth/generative-language.retriever"
12
+ ].join(" ");
13
+ var TOKEN_KEY = "google_access_token";
14
+ var TOKEN_EXPIRY_KEY = "google_token_expiry";
15
+ var USER_KEY = "google_user_info";
16
+ var CLIENT_ID_KEY = "google_oauth_client_id";
17
+ var gisLoaded = false;
18
+ var gisLoadPromise = null;
19
+ function loadGisScript() {
20
+ if (gisLoaded) return Promise.resolve();
21
+ if (gisLoadPromise) return gisLoadPromise;
22
+ gisLoadPromise = new Promise((resolve, reject) => {
23
+ const script = document.createElement("script");
24
+ script.src = "https://accounts.google.com/gsi/client";
25
+ script.async = true;
26
+ script.defer = true;
27
+ script.onload = () => {
28
+ gisLoaded = true;
29
+ resolve();
30
+ };
31
+ script.onerror = () => reject(new Error("Failed to load Google Identity Services"));
32
+ document.head.appendChild(script);
33
+ });
34
+ return gisLoadPromise;
35
+ }
36
+ function isTokenValid() {
37
+ const expiry = localStorage.getItem(TOKEN_EXPIRY_KEY);
38
+ if (!expiry) return false;
39
+ return Date.now() < parseInt(expiry, 10) - 6e4;
40
+ }
41
+ function getGoogleAccessToken() {
42
+ if (!isTokenValid()) return null;
43
+ return localStorage.getItem(TOKEN_KEY);
44
+ }
45
+ function getGoogleClientId() {
46
+ return localStorage.getItem(CLIENT_ID_KEY) || "";
47
+ }
48
+ function useGoogleAuth() {
49
+ const [accessToken, setAccessToken] = useState(() => isTokenValid() ? localStorage.getItem(TOKEN_KEY) : null);
50
+ const [user, setUser] = useState(() => {
51
+ try {
52
+ const u = localStorage.getItem(USER_KEY);
53
+ return u ? JSON.parse(u) : null;
54
+ } catch {
55
+ return null;
56
+ }
57
+ });
58
+ const [loading, setLoading] = useState(false);
59
+ const [error, setError] = useState(null);
60
+ const clientRef = useRef(null);
61
+ const clientId = getGoogleClientId();
62
+ const hasClientId = !!clientId;
63
+ const setClientId = useCallback((id) => {
64
+ localStorage.setItem(CLIENT_ID_KEY, id);
65
+ window.dispatchEvent(new Event("google-client-id-changed"));
66
+ }, []);
67
+ const fetchUserInfo = useCallback(async (token) => {
68
+ try {
69
+ const res = await fetch("https://www.googleapis.com/oauth2/v2/userinfo", {
70
+ headers: { Authorization: `Bearer ${token}` }
71
+ });
72
+ if (res.ok) {
73
+ const data = await res.json();
74
+ const userInfo = { email: data.email, name: data.name, picture: data.picture };
75
+ localStorage.setItem(USER_KEY, JSON.stringify(userInfo));
76
+ setUser(userInfo);
77
+ }
78
+ } catch {
79
+ }
80
+ }, []);
81
+ const handleTokenResponse = useCallback((response) => {
82
+ if (response.error) {
83
+ setError(response.error);
84
+ setLoading(false);
85
+ return;
86
+ }
87
+ const token = response.access_token;
88
+ const expiresIn = response.expires_in || 3600;
89
+ localStorage.setItem(TOKEN_KEY, token);
90
+ localStorage.setItem(TOKEN_EXPIRY_KEY, String(Date.now() + expiresIn * 1e3));
91
+ setAccessToken(token);
92
+ setError(null);
93
+ setLoading(false);
94
+ fetchUserInfo(token);
95
+ }, [fetchUserInfo]);
96
+ useEffect(() => {
97
+ if (!clientId) return;
98
+ loadGisScript().then(() => {
99
+ const google = window.google;
100
+ if (!google?.accounts?.oauth2) return;
101
+ clientRef.current = google.accounts.oauth2.initTokenClient({
102
+ client_id: clientId,
103
+ scope: SCOPES,
104
+ callback: handleTokenResponse
105
+ });
106
+ }).catch((err) => setError(err.message));
107
+ }, [clientId, handleTokenResponse]);
108
+ const connect = useCallback(() => {
109
+ if (!clientRef.current) {
110
+ setError("Google client not initialized. Check your Client ID.");
111
+ return;
112
+ }
113
+ setLoading(true);
114
+ setError(null);
115
+ clientRef.current.requestAccessToken({ prompt: "consent" });
116
+ }, []);
117
+ const disconnect = useCallback(() => {
118
+ const token = localStorage.getItem(TOKEN_KEY);
119
+ if (token) {
120
+ const google = window.google;
121
+ if (google?.accounts?.oauth2) {
122
+ google.accounts.oauth2.revoke(token);
123
+ }
124
+ }
125
+ localStorage.removeItem(TOKEN_KEY);
126
+ localStorage.removeItem(TOKEN_EXPIRY_KEY);
127
+ localStorage.removeItem(USER_KEY);
128
+ setAccessToken(null);
129
+ setUser(null);
130
+ }, []);
131
+ useEffect(() => {
132
+ const interval = setInterval(() => {
133
+ if (accessToken && !isTokenValid()) {
134
+ setAccessToken(null);
135
+ }
136
+ }, 3e4);
137
+ return () => clearInterval(interval);
138
+ }, [accessToken]);
139
+ return {
140
+ isConnected: !!accessToken && isTokenValid(),
141
+ user,
142
+ accessToken,
143
+ loading,
144
+ error,
145
+ connect,
146
+ disconnect,
147
+ getClientId: () => clientId,
148
+ setClientId,
149
+ hasClientId
150
+ };
151
+ }
152
+
153
+ export { getGoogleAccessToken, useGoogleAuth };
154
+ //# sourceMappingURL=chunk-5O2KEISQ.js.map
155
+ //# sourceMappingURL=chunk-5O2KEISQ.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/hooks/useGoogleAuth.ts"],"names":[],"mappings":"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","file":"chunk-5O2KEISQ.js","sourcesContent":["/**\n * Google OAuth2 hook for Gmail, Calendar, and Gemini access.\n *\n * Uses Google Identity Services (GIS) to get an access token with combined scopes.\n * Requires a Google Cloud OAuth2 Client ID configured in System Settings.\n *\n * Scopes requested:\n * - Gmail: read, compose, send, modify\n * - Calendar: read, write events\n * - Gemini: generative language (via Vertex AI or Google AI)\n */\n\nimport { useState, useEffect, useCallback, useRef } from 'react';\n\nconst SCOPES = [\n 'https://www.googleapis.com/auth/gmail.readonly',\n 'https://www.googleapis.com/auth/gmail.compose',\n 'https://www.googleapis.com/auth/gmail.send',\n 'https://www.googleapis.com/auth/gmail.modify',\n 'https://www.googleapis.com/auth/calendar.readonly',\n 'https://www.googleapis.com/auth/calendar.events',\n 'https://www.googleapis.com/auth/generative-language.retriever',\n].join(' ');\n\nconst TOKEN_KEY = 'google_access_token';\nconst TOKEN_EXPIRY_KEY = 'google_token_expiry';\nconst USER_KEY = 'google_user_info';\nconst CLIENT_ID_KEY = 'google_oauth_client_id';\n\ninterface GoogleUser {\n email: string;\n name: string;\n picture: string;\n}\n\ninterface GoogleAuthState {\n isConnected: boolean;\n user: GoogleUser | null;\n accessToken: string | null;\n loading: boolean;\n error: string | null;\n connect: () => void;\n disconnect: () => void;\n getClientId: () => string;\n setClientId: (id: string) => void;\n hasClientId: boolean;\n}\n\n// Load GIS script\nlet gisLoaded = false;\nlet gisLoadPromise: Promise<void> | null = null;\n\nfunction loadGisScript(): Promise<void> {\n if (gisLoaded) return Promise.resolve();\n if (gisLoadPromise) return gisLoadPromise;\n\n gisLoadPromise = new Promise((resolve, reject) => {\n const script = document.createElement('script');\n script.src = 'https://accounts.google.com/gsi/client';\n script.async = true;\n script.defer = true;\n script.onload = () => { gisLoaded = true; resolve(); };\n script.onerror = () => reject(new Error('Failed to load Google Identity Services'));\n document.head.appendChild(script);\n });\n return gisLoadPromise;\n}\n\nfunction isTokenValid(): boolean {\n const expiry = localStorage.getItem(TOKEN_EXPIRY_KEY);\n if (!expiry) return false;\n return Date.now() < parseInt(expiry, 10) - 60000; // 1 min buffer\n}\n\nexport function getGoogleAccessToken(): string | null {\n if (!isTokenValid()) return null;\n return localStorage.getItem(TOKEN_KEY);\n}\n\nexport function getGoogleClientId(): string {\n return localStorage.getItem(CLIENT_ID_KEY) || '';\n}\n\nexport default function useGoogleAuth(): GoogleAuthState {\n const [accessToken, setAccessToken] = useState<string | null>(() => isTokenValid() ? localStorage.getItem(TOKEN_KEY) : null);\n const [user, setUser] = useState<GoogleUser | null>(() => {\n try { const u = localStorage.getItem(USER_KEY); return u ? JSON.parse(u) : null; } catch { return null; }\n });\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n const clientRef = useRef<any>(null);\n\n const clientId = getGoogleClientId();\n const hasClientId = !!clientId;\n\n const setClientId = useCallback((id: string) => {\n localStorage.setItem(CLIENT_ID_KEY, id);\n window.dispatchEvent(new Event('google-client-id-changed'));\n }, []);\n\n // Fetch user info from Google\n const fetchUserInfo = useCallback(async (token: string) => {\n try {\n const res = await fetch('https://www.googleapis.com/oauth2/v2/userinfo', {\n headers: { Authorization: `Bearer ${token}` },\n });\n if (res.ok) {\n const data = await res.json();\n const userInfo: GoogleUser = { email: data.email, name: data.name, picture: data.picture };\n localStorage.setItem(USER_KEY, JSON.stringify(userInfo));\n setUser(userInfo);\n }\n } catch { /* ignore */ }\n }, []);\n\n const handleTokenResponse = useCallback((response: any) => {\n if (response.error) {\n setError(response.error);\n setLoading(false);\n return;\n }\n const token = response.access_token;\n const expiresIn = response.expires_in || 3600;\n localStorage.setItem(TOKEN_KEY, token);\n localStorage.setItem(TOKEN_EXPIRY_KEY, String(Date.now() + expiresIn * 1000));\n setAccessToken(token);\n setError(null);\n setLoading(false);\n fetchUserInfo(token);\n }, [fetchUserInfo]);\n\n // Initialize GIS client\n useEffect(() => {\n if (!clientId) return;\n loadGisScript().then(() => {\n const google = (window as any).google;\n if (!google?.accounts?.oauth2) return;\n clientRef.current = google.accounts.oauth2.initTokenClient({\n client_id: clientId,\n scope: SCOPES,\n callback: handleTokenResponse,\n });\n }).catch(err => setError(err.message));\n }, [clientId, handleTokenResponse]);\n\n const connect = useCallback(() => {\n if (!clientRef.current) {\n setError('Google client not initialized. Check your Client ID.');\n return;\n }\n setLoading(true);\n setError(null);\n clientRef.current.requestAccessToken({ prompt: 'consent' });\n }, []);\n\n const disconnect = useCallback(() => {\n const token = localStorage.getItem(TOKEN_KEY);\n if (token) {\n // Revoke token\n const google = (window as any).google;\n if (google?.accounts?.oauth2) {\n google.accounts.oauth2.revoke(token);\n }\n }\n localStorage.removeItem(TOKEN_KEY);\n localStorage.removeItem(TOKEN_EXPIRY_KEY);\n localStorage.removeItem(USER_KEY);\n setAccessToken(null);\n setUser(null);\n }, []);\n\n // Check token expiry periodically\n useEffect(() => {\n const interval = setInterval(() => {\n if (accessToken && !isTokenValid()) {\n setAccessToken(null);\n }\n }, 30000);\n return () => clearInterval(interval);\n }, [accessToken]);\n\n return {\n isConnected: !!accessToken && isTokenValid(),\n user,\n accessToken,\n loading,\n error,\n connect,\n disconnect,\n getClientId: () => clientId,\n setClientId,\n hasClientId,\n };\n}\n"]}
@@ -0,0 +1,265 @@
1
+ // src/utils/sounds.ts
2
+ var audioCtx = null;
3
+ function getCtx() {
4
+ if (!audioCtx) audioCtx = new AudioContext();
5
+ return audioCtx;
6
+ }
7
+ function tone(freq, dur, type = "sine", vol = 0.15) {
8
+ try {
9
+ const ctx = getCtx();
10
+ const osc = ctx.createOscillator();
11
+ const gain = ctx.createGain();
12
+ osc.type = type;
13
+ osc.frequency.value = freq;
14
+ gain.gain.value = vol;
15
+ gain.gain.exponentialRampToValueAtTime(1e-3, ctx.currentTime + dur);
16
+ osc.connect(gain);
17
+ gain.connect(ctx.destination);
18
+ osc.start();
19
+ osc.stop(ctx.currentTime + dur);
20
+ } catch {
21
+ }
22
+ }
23
+ var classic = {
24
+ label: "Classic",
25
+ success: () => {
26
+ tone(523, 0.1, "sine", 0.1);
27
+ setTimeout(() => tone(659, 0.1, "sine", 0.1), 100);
28
+ setTimeout(() => tone(784, 0.15, "sine", 0.1), 200);
29
+ },
30
+ error: () => {
31
+ tone(200, 0.15, "sawtooth", 0.08);
32
+ setTimeout(() => tone(180, 0.15, "sawtooth", 0.08), 150);
33
+ },
34
+ notification: () => {
35
+ tone(880, 0.08, "sine", 0.12);
36
+ setTimeout(() => tone(1100, 0.12, "sine", 0.1), 100);
37
+ },
38
+ startup: () => {
39
+ tone(392, 0.15, "sine", 0.08);
40
+ setTimeout(() => tone(523, 0.15, "sine", 0.08), 150);
41
+ setTimeout(() => tone(659, 0.15, "sine", 0.08), 300);
42
+ setTimeout(() => tone(784, 0.25, "sine", 0.1), 450);
43
+ },
44
+ logout: () => {
45
+ tone(784, 0.15, "sine", 0.08);
46
+ setTimeout(() => tone(659, 0.15, "sine", 0.08), 150);
47
+ setTimeout(() => tone(523, 0.15, "sine", 0.08), 300);
48
+ setTimeout(() => tone(392, 0.25, "sine", 0.06), 450);
49
+ },
50
+ click: () => {
51
+ tone(800, 0.05, "square", 0.05);
52
+ },
53
+ timerDone: () => {
54
+ for (let i = 0; i < 3; i++) {
55
+ setTimeout(() => tone(1e3, 0.15, "sine", 0.12), i * 300);
56
+ setTimeout(() => tone(800, 0.15, "sine", 0.1), i * 300 + 150);
57
+ }
58
+ }
59
+ };
60
+ var minimal = {
61
+ label: "Minimal",
62
+ success: () => {
63
+ tone(1200, 0.06, "sine", 0.08);
64
+ },
65
+ error: () => {
66
+ tone(300, 0.1, "triangle", 0.06);
67
+ },
68
+ notification: () => {
69
+ tone(900, 0.06, "sine", 0.08);
70
+ },
71
+ startup: () => {
72
+ tone(600, 0.12, "sine", 0.06);
73
+ setTimeout(() => tone(900, 0.15, "sine", 0.06), 120);
74
+ },
75
+ click: () => {
76
+ tone(1e3, 0.03, "sine", 0.03);
77
+ },
78
+ timerDone: () => {
79
+ tone(1e3, 0.2, "sine", 0.1);
80
+ setTimeout(() => tone(1e3, 0.2, "sine", 0.1), 400);
81
+ },
82
+ logout: () => {
83
+ tone(900, 0.1, "sine", 0.06);
84
+ setTimeout(() => tone(600, 0.15, "sine", 0.05), 120);
85
+ }
86
+ };
87
+ var retro = {
88
+ label: "Retro",
89
+ success: () => {
90
+ tone(440, 0.08, "square", 0.08);
91
+ setTimeout(() => tone(550, 0.08, "square", 0.08), 80);
92
+ setTimeout(() => tone(660, 0.08, "square", 0.08), 160);
93
+ setTimeout(() => tone(880, 0.12, "square", 0.1), 240);
94
+ },
95
+ error: () => {
96
+ tone(150, 0.2, "square", 0.06);
97
+ setTimeout(() => tone(100, 0.3, "square", 0.06), 200);
98
+ },
99
+ notification: () => {
100
+ tone(660, 0.06, "square", 0.08);
101
+ setTimeout(() => tone(880, 0.06, "square", 0.08), 80);
102
+ setTimeout(() => tone(660, 0.06, "square", 0.08), 160);
103
+ },
104
+ startup: () => {
105
+ [262, 330, 392, 523].forEach((f, i) => setTimeout(() => tone(f, 0.1, "square", 0.07), i * 100));
106
+ },
107
+ click: () => {
108
+ tone(600, 0.03, "square", 0.06);
109
+ },
110
+ timerDone: () => {
111
+ for (let i = 0; i < 4; i++) setTimeout(() => tone(880, 0.1, "square", 0.1), i * 200);
112
+ },
113
+ logout: () => {
114
+ [523, 392, 330, 262].forEach((f, i) => setTimeout(() => tone(f, 0.1, "square", 0.07), i * 100));
115
+ }
116
+ };
117
+ var soft = {
118
+ label: "Soft",
119
+ success: () => {
120
+ tone(700, 0.2, "sine", 0.06);
121
+ setTimeout(() => tone(900, 0.25, "sine", 0.06), 200);
122
+ },
123
+ error: () => {
124
+ tone(250, 0.2, "sine", 0.05);
125
+ },
126
+ notification: () => {
127
+ tone(600, 0.15, "triangle", 0.06);
128
+ setTimeout(() => tone(800, 0.2, "triangle", 0.05), 180);
129
+ },
130
+ startup: () => {
131
+ tone(400, 0.3, "sine", 0.05);
132
+ setTimeout(() => tone(500, 0.3, "sine", 0.05), 300);
133
+ setTimeout(() => tone(600, 0.4, "sine", 0.06), 600);
134
+ },
135
+ click: () => {
136
+ tone(500, 0.04, "triangle", 0.03);
137
+ },
138
+ timerDone: () => {
139
+ tone(700, 0.3, "triangle", 0.08);
140
+ setTimeout(() => tone(700, 0.3, "triangle", 0.08), 500);
141
+ setTimeout(() => tone(900, 0.4, "triangle", 0.08), 1e3);
142
+ },
143
+ logout: () => {
144
+ tone(600, 0.3, "sine", 0.05);
145
+ setTimeout(() => tone(400, 0.4, "sine", 0.04), 300);
146
+ }
147
+ };
148
+ var arcade = {
149
+ label: "Arcade",
150
+ success: () => {
151
+ [523, 659, 784, 1047].forEach((f, i) => setTimeout(() => tone(f, 0.06, "square", 0.07), i * 60));
152
+ },
153
+ error: () => {
154
+ tone(100, 0.3, "sawtooth", 0.06);
155
+ setTimeout(() => tone(80, 0.4, "sawtooth", 0.05), 200);
156
+ },
157
+ notification: () => {
158
+ tone(1047, 0.05, "square", 0.08);
159
+ setTimeout(() => tone(784, 0.05, "square", 0.08), 60);
160
+ setTimeout(() => tone(1047, 0.08, "square", 0.1), 120);
161
+ },
162
+ startup: () => {
163
+ [262, 330, 392, 523, 659, 784].forEach((f, i) => setTimeout(() => tone(f, 0.08, "square", 0.06), i * 80));
164
+ },
165
+ click: () => {
166
+ tone(1200, 0.02, "square", 0.05);
167
+ },
168
+ timerDone: () => {
169
+ for (let i = 0; i < 5; i++) setTimeout(() => tone(1200, 0.05, "square", 0.1), i * 120);
170
+ },
171
+ logout: () => {
172
+ [784, 659, 523, 392, 262].forEach((f, i) => setTimeout(() => tone(f, 0.06, "square", 0.06), i * 60));
173
+ }
174
+ };
175
+ var silent = {
176
+ label: "Silent",
177
+ success: () => {
178
+ },
179
+ error: () => {
180
+ },
181
+ notification: () => {
182
+ },
183
+ startup: () => {
184
+ },
185
+ click: () => {
186
+ },
187
+ timerDone: () => {
188
+ },
189
+ logout: () => {
190
+ }
191
+ };
192
+ var SOUND_PACKS = { classic, minimal, retro, soft, arcade, silent };
193
+ var SOUND_PACK_KEYS = Object.keys(SOUND_PACKS);
194
+ var SOUND_TYPES = ["click", "success", "error", "notification", "startup", "timerDone", "logout"];
195
+ var SOUND_TYPE_LABELS = {
196
+ click: "Click",
197
+ success: "Success",
198
+ error: "Error",
199
+ notification: "Notification",
200
+ startup: "Startup",
201
+ timerDone: "Timer",
202
+ logout: "Logout"
203
+ };
204
+ var STORAGE_KEY = "erp_sound_config";
205
+ function soundsEnabled() {
206
+ return localStorage.getItem("erp_sounds") !== "false";
207
+ }
208
+ function toggleSounds() {
209
+ const next = !soundsEnabled();
210
+ localStorage.setItem("erp_sounds", String(next));
211
+ return next;
212
+ }
213
+ function getSoundConfig() {
214
+ try {
215
+ const stored = JSON.parse(localStorage.getItem(STORAGE_KEY) || "");
216
+ if (stored && typeof stored === "object") return { click: "classic", success: "classic", error: "classic", notification: "classic", startup: "classic", timerDone: "classic", logout: "classic", ...stored };
217
+ } catch {
218
+ }
219
+ return { click: "classic", success: "classic", error: "classic", notification: "classic", startup: "classic", timerDone: "classic", logout: "classic" };
220
+ }
221
+ function setSoundForType(soundType, packKey) {
222
+ const config = getSoundConfig();
223
+ config[soundType] = packKey;
224
+ localStorage.setItem(STORAGE_KEY, JSON.stringify(config));
225
+ }
226
+ function setAllSounds(packKey) {
227
+ const config = {};
228
+ SOUND_TYPES.forEach((t) => {
229
+ config[t] = packKey;
230
+ });
231
+ localStorage.setItem(STORAGE_KEY, JSON.stringify(config));
232
+ }
233
+ function getPackForType(soundType) {
234
+ const config = getSoundConfig();
235
+ return SOUND_PACKS[config[soundType]] || classic;
236
+ }
237
+ function playClick() {
238
+ if (soundsEnabled()) getPackForType("click").click();
239
+ }
240
+ function playSuccess() {
241
+ if (soundsEnabled()) getPackForType("success").success();
242
+ }
243
+ function playError() {
244
+ if (soundsEnabled()) getPackForType("error").error();
245
+ }
246
+ function playNotification() {
247
+ if (soundsEnabled()) getPackForType("notification").notification();
248
+ }
249
+ function playStartup() {
250
+ if (soundsEnabled()) getPackForType("startup").startup();
251
+ }
252
+ function playTimerDone() {
253
+ if (soundsEnabled()) getPackForType("timerDone").timerDone();
254
+ }
255
+ function playLogout() {
256
+ if (soundsEnabled()) getPackForType("logout").logout();
257
+ }
258
+ function previewSound(packKey, sound) {
259
+ const pack = SOUND_PACKS[packKey];
260
+ if (pack && typeof pack[sound] === "function") pack[sound]();
261
+ }
262
+
263
+ export { SOUND_PACKS, SOUND_PACK_KEYS, SOUND_TYPES, SOUND_TYPE_LABELS, getSoundConfig, playClick, playError, playLogout, playNotification, playStartup, playSuccess, playTimerDone, previewSound, setAllSounds, setSoundForType, soundsEnabled, toggleSounds };
264
+ //# sourceMappingURL=chunk-D7PYW2QS.js.map
265
+ //# sourceMappingURL=chunk-D7PYW2QS.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/utils/sounds.ts"],"names":[],"mappings":"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","file":"chunk-D7PYW2QS.js","sourcesContent":["/**\n * Custom sound effects using Web Audio API.\n * Multiple sound packs — user can choose in Customization > Desktop.\n */\n\nlet audioCtx: AudioContext | null = null;\n\nfunction getCtx(): AudioContext {\n if (!audioCtx) audioCtx = new AudioContext();\n return audioCtx;\n}\n\nfunction tone(freq: number, dur: number, type: OscillatorType = 'sine', vol = 0.15) {\n try {\n const ctx = getCtx();\n const osc = ctx.createOscillator();\n const gain = ctx.createGain();\n osc.type = type;\n osc.frequency.value = freq;\n gain.gain.value = vol;\n gain.gain.exponentialRampToValueAtTime(0.001, ctx.currentTime + dur);\n osc.connect(gain);\n gain.connect(ctx.destination);\n osc.start();\n osc.stop(ctx.currentTime + dur);\n } catch { /* Audio not available */ }\n}\n\nfunction delay(ms: number) { return new Promise(r => setTimeout(r, ms)); }\n\n// ── Sound Pack Definitions ──\n\ninterface SoundPack {\n label: string;\n success: () => void;\n error: () => void;\n notification: () => void;\n startup: () => void;\n click: () => void;\n timerDone: () => void;\n logout: () => void;\n}\n\nconst classic: SoundPack = {\n label: 'Classic',\n success: () => { tone(523, 0.1, 'sine', 0.1); setTimeout(() => tone(659, 0.1, 'sine', 0.1), 100); setTimeout(() => tone(784, 0.15, 'sine', 0.1), 200); },\n error: () => { tone(200, 0.15, 'sawtooth', 0.08); setTimeout(() => tone(180, 0.15, 'sawtooth', 0.08), 150); },\n notification: () => { tone(880, 0.08, 'sine', 0.12); setTimeout(() => tone(1100, 0.12, 'sine', 0.1), 100); },\n startup: () => { tone(392, 0.15, 'sine', 0.08); setTimeout(() => tone(523, 0.15, 'sine', 0.08), 150); setTimeout(() => tone(659, 0.15, 'sine', 0.08), 300); setTimeout(() => tone(784, 0.25, 'sine', 0.1), 450); },\n logout: () => { tone(784, 0.15, 'sine', 0.08); setTimeout(() => tone(659, 0.15, 'sine', 0.08), 150); setTimeout(() => tone(523, 0.15, 'sine', 0.08), 300); setTimeout(() => tone(392, 0.25, 'sine', 0.06), 450); },\n click: () => { tone(800, 0.05, 'square', 0.05); },\n timerDone: () => { for (let i = 0; i < 3; i++) { setTimeout(() => tone(1000, 0.15, 'sine', 0.12), i * 300); setTimeout(() => tone(800, 0.15, 'sine', 0.1), i * 300 + 150); } },\n};\n\nconst minimal: SoundPack = {\n label: 'Minimal',\n success: () => { tone(1200, 0.06, 'sine', 0.08); },\n error: () => { tone(300, 0.1, 'triangle', 0.06); },\n notification: () => { tone(900, 0.06, 'sine', 0.08); },\n startup: () => { tone(600, 0.12, 'sine', 0.06); setTimeout(() => tone(900, 0.15, 'sine', 0.06), 120); },\n click: () => { tone(1000, 0.03, 'sine', 0.03); },\n timerDone: () => { tone(1000, 0.2, 'sine', 0.1); setTimeout(() => tone(1000, 0.2, 'sine', 0.1), 400); },\n logout: () => { tone(900, 0.1, 'sine', 0.06); setTimeout(() => tone(600, 0.15, 'sine', 0.05), 120); },\n};\n\nconst retro: SoundPack = {\n label: 'Retro',\n success: () => { tone(440, 0.08, 'square', 0.08); setTimeout(() => tone(550, 0.08, 'square', 0.08), 80); setTimeout(() => tone(660, 0.08, 'square', 0.08), 160); setTimeout(() => tone(880, 0.12, 'square', 0.1), 240); },\n error: () => { tone(150, 0.2, 'square', 0.06); setTimeout(() => tone(100, 0.3, 'square', 0.06), 200); },\n notification: () => { tone(660, 0.06, 'square', 0.08); setTimeout(() => tone(880, 0.06, 'square', 0.08), 80); setTimeout(() => tone(660, 0.06, 'square', 0.08), 160); },\n startup: () => { [262, 330, 392, 523].forEach((f, i) => setTimeout(() => tone(f, 0.1, 'square', 0.07), i * 100)); },\n click: () => { tone(600, 0.03, 'square', 0.06); },\n timerDone: () => { for (let i = 0; i < 4; i++) setTimeout(() => tone(880, 0.1, 'square', 0.1), i * 200); },\n logout: () => { [523, 392, 330, 262].forEach((f, i) => setTimeout(() => tone(f, 0.1, 'square', 0.07), i * 100)); },\n};\n\nconst soft: SoundPack = {\n label: 'Soft',\n success: () => { tone(700, 0.2, 'sine', 0.06); setTimeout(() => tone(900, 0.25, 'sine', 0.06), 200); },\n error: () => { tone(250, 0.2, 'sine', 0.05); },\n notification: () => { tone(600, 0.15, 'triangle', 0.06); setTimeout(() => tone(800, 0.2, 'triangle', 0.05), 180); },\n startup: () => { tone(400, 0.3, 'sine', 0.05); setTimeout(() => tone(500, 0.3, 'sine', 0.05), 300); setTimeout(() => tone(600, 0.4, 'sine', 0.06), 600); },\n click: () => { tone(500, 0.04, 'triangle', 0.03); },\n timerDone: () => { tone(700, 0.3, 'triangle', 0.08); setTimeout(() => tone(700, 0.3, 'triangle', 0.08), 500); setTimeout(() => tone(900, 0.4, 'triangle', 0.08), 1000); },\n logout: () => { tone(600, 0.3, 'sine', 0.05); setTimeout(() => tone(400, 0.4, 'sine', 0.04), 300); },\n};\n\nconst arcade: SoundPack = {\n label: 'Arcade',\n success: () => { [523, 659, 784, 1047].forEach((f, i) => setTimeout(() => tone(f, 0.06, 'square', 0.07), i * 60)); },\n error: () => { tone(100, 0.3, 'sawtooth', 0.06); setTimeout(() => tone(80, 0.4, 'sawtooth', 0.05), 200); },\n notification: () => { tone(1047, 0.05, 'square', 0.08); setTimeout(() => tone(784, 0.05, 'square', 0.08), 60); setTimeout(() => tone(1047, 0.08, 'square', 0.1), 120); },\n startup: () => { [262, 330, 392, 523, 659, 784].forEach((f, i) => setTimeout(() => tone(f, 0.08, 'square', 0.06), i * 80)); },\n click: () => { tone(1200, 0.02, 'square', 0.05); },\n timerDone: () => { for (let i = 0; i < 5; i++) setTimeout(() => tone(1200, 0.05, 'square', 0.1), i * 120); },\n logout: () => { [784, 659, 523, 392, 262].forEach((f, i) => setTimeout(() => tone(f, 0.06, 'square', 0.06), i * 60)); },\n};\n\nconst silent: SoundPack = {\n label: 'Silent',\n success: () => {}, error: () => {}, notification: () => {},\n startup: () => {}, click: () => {}, timerDone: () => {}, logout: () => {},\n};\n\nexport const SOUND_PACKS: Record<string, SoundPack> = { classic, minimal, retro, soft, arcade, silent };\n\n/** All pack keys including silent */\nexport const SOUND_PACK_KEYS = Object.keys(SOUND_PACKS) as string[];\n\n// ── Sound types ──\nexport const SOUND_TYPES = ['click', 'success', 'error', 'notification', 'startup', 'timerDone', 'logout'] as const;\nexport type SoundType = typeof SOUND_TYPES[number];\n\nexport const SOUND_TYPE_LABELS: Record<SoundType, string> = {\n click: 'Click', success: 'Success', error: 'Error',\n notification: 'Notification', startup: 'Startup', timerDone: 'Timer', logout: 'Logout',\n};\n\n// ── State: per-sound pack selection ──\n\nconst STORAGE_KEY = 'erp_sound_config';\n\nexport function soundsEnabled(): boolean {\n return localStorage.getItem('erp_sounds') !== 'false';\n}\n\nexport function toggleSounds(): boolean {\n const next = !soundsEnabled();\n localStorage.setItem('erp_sounds', String(next));\n return next;\n}\n\n/** Get the per-sound configuration: { click: 'classic', success: 'retro', ... } */\nexport function getSoundConfig(): Record<SoundType, string> {\n try {\n const stored = JSON.parse(localStorage.getItem(STORAGE_KEY) || '');\n if (stored && typeof stored === 'object') return { click: 'classic', success: 'classic', error: 'classic', notification: 'classic', startup: 'classic', timerDone: 'classic', logout: 'classic', ...stored };\n } catch {}\n return { click: 'classic', success: 'classic', error: 'classic', notification: 'classic', startup: 'classic', timerDone: 'classic', logout: 'classic' };\n}\n\n/** Set the pack for a specific sound type */\nexport function setSoundForType(soundType: SoundType, packKey: string) {\n const config = getSoundConfig();\n config[soundType] = packKey;\n localStorage.setItem(STORAGE_KEY, JSON.stringify(config));\n}\n\n/** Set all sounds to a specific pack */\nexport function setAllSounds(packKey: string) {\n const config: Record<string, string> = {};\n SOUND_TYPES.forEach(t => { config[t] = packKey; });\n localStorage.setItem(STORAGE_KEY, JSON.stringify(config));\n}\n\nfunction getPackForType(soundType: SoundType): SoundPack {\n const config = getSoundConfig();\n return SOUND_PACKS[config[soundType]] || classic;\n}\n\n// ── Public API ──\n\nexport function playClick() { if (soundsEnabled()) getPackForType('click').click(); }\nexport function playSuccess() { if (soundsEnabled()) getPackForType('success').success(); }\nexport function playError() { if (soundsEnabled()) getPackForType('error').error(); }\nexport function playNotification() { if (soundsEnabled()) getPackForType('notification').notification(); }\nexport function playStartup() { if (soundsEnabled()) getPackForType('startup').startup(); }\nexport function playTimerDone() { if (soundsEnabled()) getPackForType('timerDone').timerDone(); }\nexport function playLogout() { if (soundsEnabled()) getPackForType('logout').logout(); }\n\n/** Preview a specific sound from a specific pack (ignores enabled state) */\nexport function previewSound(packKey: string, sound: SoundType) {\n const pack = SOUND_PACKS[packKey];\n if (pack && typeof pack[sound] === 'function') (pack[sound] as () => void)();\n}\n"]}