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.
- package/LICENSE +21 -0
- package/README.md +242 -0
- package/dist/Calculator-BNBRNV4P.js +184 -0
- package/dist/Calculator-BNBRNV4P.js.map +1 -0
- package/dist/Calendar-5EYUVGUU.js +423 -0
- package/dist/Calendar-5EYUVGUU.js.map +1 -0
- package/dist/Checkers-MIAHIKJH.js +214 -0
- package/dist/Checkers-MIAHIKJH.js.map +1 -0
- package/dist/Chess-C5BY45NA.js +190 -0
- package/dist/Chess-C5BY45NA.js.map +1 -0
- package/dist/ConfirmDialog-ZP4AHVUD.js +3 -0
- package/dist/ConfirmDialog-ZP4AHVUD.js.map +1 -0
- package/dist/CurrencyConverter-TYPU2IRF.js +223 -0
- package/dist/CurrencyConverter-TYPU2IRF.js.map +1 -0
- package/dist/Email-JEYYJ3YV.js +1835 -0
- package/dist/Email-JEYYJ3YV.js.map +1 -0
- package/dist/Game2048-3RH3ELRD.js +191 -0
- package/dist/Game2048-3RH3ELRD.js.map +1 -0
- package/dist/GeminiChat-BXLBJFT4.js +184 -0
- package/dist/GeminiChat-BXLBJFT4.js.map +1 -0
- package/dist/Minesweeper-VQGLAZON.js +270 -0
- package/dist/Minesweeper-VQGLAZON.js.map +1 -0
- package/dist/Notepad-YTZRCAXX.js +389 -0
- package/dist/Notepad-YTZRCAXX.js.map +1 -0
- package/dist/PomodoroTimer-HARIJN4S.js +196 -0
- package/dist/PomodoroTimer-HARIJN4S.js.map +1 -0
- package/dist/Spreadsheet-IOKEDNS6.js +446 -0
- package/dist/Spreadsheet-IOKEDNS6.js.map +1 -0
- package/dist/Sudoku-XHLYCEVT.js +197 -0
- package/dist/Sudoku-XHLYCEVT.js.map +1 -0
- package/dist/Tetris-ZHCZYL24.js +243 -0
- package/dist/Tetris-ZHCZYL24.js.map +1 -0
- package/dist/Weather-ROZ7TRNW.js +310 -0
- package/dist/Weather-ROZ7TRNW.js.map +1 -0
- package/dist/apps/index.d.ts +55 -0
- package/dist/apps/index.js +48 -0
- package/dist/apps/index.js.map +1 -0
- package/dist/chunk-5O2KEISQ.js +155 -0
- package/dist/chunk-5O2KEISQ.js.map +1 -0
- package/dist/chunk-D7PYW2QS.js +265 -0
- package/dist/chunk-D7PYW2QS.js.map +1 -0
- package/dist/chunk-GP4Y3VCB.js +806 -0
- package/dist/chunk-GP4Y3VCB.js.map +1 -0
- package/dist/chunk-NSU7OHPC.js +39 -0
- package/dist/chunk-NSU7OHPC.js.map +1 -0
- package/dist/chunk-PDFQNHW7.js +24 -0
- package/dist/chunk-PDFQNHW7.js.map +1 -0
- package/dist/chunk-RFTLYCSF.js +144 -0
- package/dist/chunk-RFTLYCSF.js.map +1 -0
- package/dist/chunk-SVBID2P6.js +142 -0
- package/dist/chunk-SVBID2P6.js.map +1 -0
- package/dist/chunk-TFGOLXGD.js +41 -0
- package/dist/chunk-TFGOLXGD.js.map +1 -0
- package/dist/chunk-WIJ45SYD.js +120 -0
- package/dist/chunk-WIJ45SYD.js.map +1 -0
- package/dist/chunk-WQIS72NL.js +1470 -0
- package/dist/chunk-WQIS72NL.js.map +1 -0
- package/dist/index.d.ts +642 -0
- package/dist/index.js +3443 -0
- package/dist/index.js.map +1 -0
- package/dist/sounds-NT4DEZGD.js +3 -0
- package/dist/sounds-NT4DEZGD.js.map +1 -0
- package/dist/styles.css +174 -0
- package/dist/types-CFIZ1_xt.d.ts +67 -0
- 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"]}
|