Files
sbv/frontend/src/components/SettingsModal.jsx
T

180 lines
5.3 KiB
React

import { useState, useEffect } from 'react'
import axios from 'axios'
const API_BASE = import.meta.env.VITE_API_URL || 'http://localhost:8085/api'
const MESSAGE_LIMIT_OPTIONS = [
{ value: 100, label: '100' },
{ value: 1000, label: '1,000' },
{ value: 10000, label: '10,000' },
{ value: 100000, label: '100,000' },
{ value: 200000, label: '200,000' },
{ value: 500000, label: '500,000' },
]
function SettingsModal({ show, onClose, onSettingsUpdated }) {
const [settings, setSettings] = useState({
conversations: {
show_calls: true,
message_limit: 100000
}
})
const [loading, setLoading] = useState(true)
const [saving, setSaving] = useState(false)
const [error, setError] = useState('')
useEffect(() => {
if (show) {
fetchSettings()
}
}, [show])
const fetchSettings = async () => {
try {
setLoading(true)
const response = await axios.get(`${API_BASE}/settings`)
setSettings(response.data)
} catch (err) {
console.error('Failed to fetch settings:', err)
setError('Failed to load settings')
} finally {
setLoading(false)
}
}
const handleSave = async () => {
try {
setSaving(true)
setError('')
await axios.put(`${API_BASE}/settings`, settings)
if (onSettingsUpdated) {
onSettingsUpdated(settings)
}
onClose()
} catch (err) {
console.error('Failed to save settings:', err)
setError('Failed to save settings')
} finally {
setSaving(false)
}
}
const handleToggleShowCalls = () => {
setSettings({
...settings,
conversations: {
...settings.conversations,
show_calls: !settings.conversations.show_calls
}
})
}
const handleMessageLimitChange = (e) => {
setSettings({
...settings,
conversations: {
...settings.conversations,
message_limit: parseInt(e.target.value, 10)
}
})
}
if (!show) return null
return (
<div className="modal show d-block" tabIndex="-1" style={{ backgroundColor: 'rgba(0,0,0,0.5)' }}>
<div className="modal-dialog modal-dialog-centered">
<div className="modal-content">
<div className="modal-header">
<h5 className="modal-title">Settings</h5>
<button type="button" className="btn-close" onClick={onClose} disabled={saving}></button>
</div>
<div className="modal-body">
{loading ? (
<div className="text-center py-4">
<div className="spinner-border" role="status">
<span className="visually-hidden">Loading...</span>
</div>
</div>
) : (
<>
{error && (
<div className="alert alert-danger" role="alert">
{error}
</div>
)}
<h6 className="mb-3">Conversations</h6>
<div className="form-check form-switch mb-3">
<input
className="form-check-input"
type="checkbox"
id="showCallsToggle"
checked={settings.conversations.show_calls}
onChange={handleToggleShowCalls}
disabled={saving}
/>
<label className="form-check-label" htmlFor="showCallsToggle">
Show calls in conversation list
</label>
<div className="form-text">
When enabled, phone calls will appear in the conversation list alongside messages.
</div>
</div>
<div className="mb-3">
<label htmlFor="messageLimitSelect" className="form-label">Messages to load</label>
<select
className="form-select"
id="messageLimitSelect"
value={settings.conversations.message_limit || 100000}
onChange={handleMessageLimitChange}
disabled={saving}
>
{MESSAGE_LIMIT_OPTIONS.map(opt => (
<option key={opt.value} value={opt.value}>{opt.label}</option>
))}
</select>
<div className="form-text">
Maximum number of messages shown when opening a conversation.
</div>
</div>
</>
)}
</div>
<div className="modal-footer">
<button
type="button"
className="btn btn-secondary"
onClick={onClose}
disabled={saving}
>
Cancel
</button>
<button
type="button"
className="btn btn-primary"
onClick={handleSave}
disabled={loading || saving}
>
{saving ? (
<>
<span className="spinner-border spinner-border-sm me-2" role="status" aria-hidden="true"></span>
Saving...
</>
) : (
'Save Changes'
)}
</button>
</div>
</div>
</div>
</div>
)
}
export default SettingsModal