mirror of
https://github.com/mauriceboe/TREK.git
synced 2026-06-19 21:31:46 +00:00
c887acddee
Features: - Single Sign-On (OIDC) — login with Google, Apple, Authentik, Keycloak - OpenStreetMap place search as free fallback when no Google API key - Change password in user settings - Delete own account (with last-admin protection) - Last login column in admin user management - SSO badge and provider info in user settings - Google API key "Recommended" badge in admin panel Improvements: - API keys load correctly after page reload - Validate auto-saves keys before testing - Time format respects 12h/24h setting everywhere - Dark mode fixes for popups and backup buttons - Admin stats: removed photos, 4-column layout - Profile picture upload button on avatar overlay - TravelStats duplicate key fix - Backup panel dark mode support
165 lines
4.2 KiB
JavaScript
165 lines
4.2 KiB
JavaScript
import { create } from 'zustand'
|
|
import { authApi } from '../api/client'
|
|
import { connect, disconnect } from '../api/websocket'
|
|
|
|
export const useAuthStore = create((set, get) => ({
|
|
user: null,
|
|
token: localStorage.getItem('auth_token') || null,
|
|
isAuthenticated: !!localStorage.getItem('auth_token'),
|
|
isLoading: false,
|
|
error: null,
|
|
demoMode: localStorage.getItem('demo_mode') === 'true',
|
|
hasMapsKey: false,
|
|
|
|
login: async (email, password) => {
|
|
set({ isLoading: true, error: null })
|
|
try {
|
|
const data = await authApi.login({ email, password })
|
|
localStorage.setItem('auth_token', data.token)
|
|
set({
|
|
user: data.user,
|
|
token: data.token,
|
|
isAuthenticated: true,
|
|
isLoading: false,
|
|
error: null,
|
|
})
|
|
connect(data.token)
|
|
return data
|
|
} catch (err) {
|
|
const error = err.response?.data?.error || 'Anmeldung fehlgeschlagen'
|
|
set({ isLoading: false, error })
|
|
throw new Error(error)
|
|
}
|
|
},
|
|
|
|
register: async (username, email, password) => {
|
|
set({ isLoading: true, error: null })
|
|
try {
|
|
const data = await authApi.register({ username, email, password })
|
|
localStorage.setItem('auth_token', data.token)
|
|
set({
|
|
user: data.user,
|
|
token: data.token,
|
|
isAuthenticated: true,
|
|
isLoading: false,
|
|
error: null,
|
|
})
|
|
connect(data.token)
|
|
return data
|
|
} catch (err) {
|
|
const error = err.response?.data?.error || 'Registrierung fehlgeschlagen'
|
|
set({ isLoading: false, error })
|
|
throw new Error(error)
|
|
}
|
|
},
|
|
|
|
logout: () => {
|
|
disconnect()
|
|
localStorage.removeItem('auth_token')
|
|
set({
|
|
user: null,
|
|
token: null,
|
|
isAuthenticated: false,
|
|
error: null,
|
|
})
|
|
},
|
|
|
|
loadUser: async () => {
|
|
const token = get().token
|
|
if (!token) {
|
|
set({ isLoading: false })
|
|
return
|
|
}
|
|
set({ isLoading: true })
|
|
try {
|
|
const data = await authApi.me()
|
|
set({
|
|
user: data.user,
|
|
isAuthenticated: true,
|
|
isLoading: false,
|
|
})
|
|
connect(token)
|
|
} catch (err) {
|
|
localStorage.removeItem('auth_token')
|
|
set({
|
|
user: null,
|
|
token: null,
|
|
isAuthenticated: false,
|
|
isLoading: false,
|
|
})
|
|
}
|
|
},
|
|
|
|
updateMapsKey: async (key) => {
|
|
try {
|
|
await authApi.updateMapsKey(key)
|
|
set(state => ({
|
|
user: { ...state.user, maps_api_key: key || null }
|
|
}))
|
|
} catch (err) {
|
|
throw new Error(err.response?.data?.error || 'Fehler beim Speichern des API-Schlüssels')
|
|
}
|
|
},
|
|
|
|
updateApiKeys: async (keys) => {
|
|
try {
|
|
const data = await authApi.updateApiKeys(keys)
|
|
set({ user: data.user })
|
|
} catch (err) {
|
|
throw new Error(err.response?.data?.error || 'Fehler beim Speichern der API-Schlüssel')
|
|
}
|
|
},
|
|
|
|
updateProfile: async (profileData) => {
|
|
try {
|
|
const data = await authApi.updateSettings(profileData)
|
|
set({ user: data.user })
|
|
} catch (err) {
|
|
throw new Error(err.response?.data?.error || 'Fehler beim Aktualisieren des Profils')
|
|
}
|
|
},
|
|
|
|
uploadAvatar: async (file) => {
|
|
const formData = new FormData()
|
|
formData.append('avatar', file)
|
|
const data = await authApi.uploadAvatar(formData)
|
|
set(state => ({ user: { ...state.user, avatar_url: data.avatar_url } }))
|
|
return data
|
|
},
|
|
|
|
deleteAvatar: async () => {
|
|
await authApi.deleteAvatar()
|
|
set(state => ({ user: { ...state.user, avatar_url: null } }))
|
|
},
|
|
|
|
setDemoMode: (val) => {
|
|
if (val) localStorage.setItem('demo_mode', 'true')
|
|
else localStorage.removeItem('demo_mode')
|
|
set({ demoMode: val })
|
|
},
|
|
|
|
setHasMapsKey: (val) => set({ hasMapsKey: val }),
|
|
|
|
demoLogin: async () => {
|
|
set({ isLoading: true, error: null })
|
|
try {
|
|
const data = await authApi.demoLogin()
|
|
localStorage.setItem('auth_token', data.token)
|
|
set({
|
|
user: data.user,
|
|
token: data.token,
|
|
isAuthenticated: true,
|
|
isLoading: false,
|
|
demoMode: true,
|
|
error: null,
|
|
})
|
|
connect(data.token)
|
|
return data
|
|
} catch (err) {
|
|
const error = err.response?.data?.error || 'Demo-Login fehlgeschlagen'
|
|
set({ isLoading: false, error })
|
|
throw new Error(error)
|
|
}
|
|
},
|
|
}))
|