diff options
| author | Adam Mathes <adam@adammathes.com> | 2026-02-16 19:37:50 -0800 |
|---|---|---|
| committer | Adam Mathes <adam@adammathes.com> | 2026-02-16 19:40:10 -0800 |
| commit | cba29e6aae637b04ff6eaf28f74bc15b6242b9ea (patch) | |
| tree | 226753a3fcd18a6d45089dafcb1ee72557140aa8 /frontend/src/components/Settings.tsx | |
| parent | cb6d0c9e330c27ff85ff065c2ea6dd1a756cbf6d (diff) | |
| download | neko-cba29e6aae637b04ff6eaf28f74bc15b6242b9ea.tar.gz neko-cba29e6aae637b04ff6eaf28f74bc15b6242b9ea.tar.bz2 neko-cba29e6aae637b04ff6eaf28f74bc15b6242b9ea.zip | |
Remove legacy V2 React frontend and update build/test scripts to focus on Vanilla JS (V3)
Diffstat (limited to 'frontend/src/components/Settings.tsx')
| -rw-r--r-- | frontend/src/components/Settings.tsx | 236 |
1 files changed, 0 insertions, 236 deletions
diff --git a/frontend/src/components/Settings.tsx b/frontend/src/components/Settings.tsx deleted file mode 100644 index 3dab77f..0000000 --- a/frontend/src/components/Settings.tsx +++ /dev/null @@ -1,236 +0,0 @@ -import React, { useEffect, useState } from 'react'; -import type { Feed } from '../types'; -import './Settings.css'; -import { apiFetch } from '../utils'; - -interface SettingsProps { - fontTheme?: string; - setFontTheme?: (t: string) => void; -} - -export default function Settings({ fontTheme, setFontTheme }: SettingsProps) { - const [feeds, setFeeds] = useState<Feed[]>([]); - /* ... existing state ... */ - const [newFeedUrl, setNewFeedUrl] = useState(''); - const [loading, setLoading] = useState(false); - const [error, setError] = useState<string | null>(null); - - const [importFile, setImportFile] = useState<File | null>(null); - - /* ... existing fetchFeeds ... */ - const fetchFeeds = React.useCallback(() => { - setLoading(true); - apiFetch('/api/feed/') - .then((res) => { - if (!res.ok) throw new Error('Failed to fetch feeds'); - return res.json(); - }) - .then((data) => { - setFeeds(data); - setLoading(false); - }) - .catch((err) => { - setError(err.message); - setLoading(false); - }); - }, []); - - useEffect(() => { - // eslint-disable-next-line - fetchFeeds(); - }, [fetchFeeds]); - - /* ... existing handlers ... */ - const handleAddFeed = (e: React.FormEvent) => { - e.preventDefault(); - if (!newFeedUrl) return; - - setLoading(true); - apiFetch('/api/feed/', { - method: 'POST', - headers: { 'Content-Type': 'application/json' }, - body: JSON.stringify({ url: newFeedUrl }), - }) - .then((res) => { - if (!res.ok) throw new Error('Failed to add feed'); - return res.json(); - }) - .then(() => { - setNewFeedUrl(''); - fetchFeeds(); - }) - .catch((err) => { - setError(err.message); - setLoading(false); - }); - }; - - const handleDeleteFeed = (id: number) => { - if (!globalThis.confirm('Are you sure you want to delete this feed?')) return; - - setLoading(true); - apiFetch(`/api/feed/${id}`, { - method: 'DELETE', - }) - .then((res) => { - if (!res.ok) throw new Error('Failed to delete feed'); - setFeeds(feeds.filter((f) => f._id !== id)); - setLoading(false); - }) - .catch((err) => { - setError(err.message); - setLoading(false); - }); - }; - - const handleImport = (e: React.FormEvent) => { - e.preventDefault(); - if (!importFile) return; - - setLoading(true); - const formData = new FormData(); - formData.append('file', importFile); - formData.append('format', 'opml'); - - apiFetch('/api/import', { - method: 'POST', - body: formData, - }) - .then((res) => { - if (!res.ok) throw new Error('Failed to import feeds'); - return res.json(); - }) - .then(() => { - setImportFile(null); - fetchFeeds(); - alert('Import successful!'); - }) - .catch((err) => { - setError(err.message); - setLoading(false); - }); - }; - - const handleCrawl = () => { - setLoading(true); - apiFetch('/api/crawl', { - method: 'POST', - }) - .then((res) => { - if (!res.ok) throw new Error('Failed to start crawl'); - return res.json(); - }) - .then(() => { - setLoading(false); - alert('Crawl started!'); - }) - .catch((err) => { - setError(err.message); - setLoading(false); - }); - }; - - return ( - <div className="settings-page variant-glass"> - <h2>Settings</h2> - - {setFontTheme && ( - <div className="appearance-section"> - <h3>Appearance</h3> - <div className="font-selector"> - <label htmlFor="font-theme-select">Font Theme:</label> - <select - id="font-theme-select" - value={fontTheme || 'default'} - onChange={(e) => setFontTheme(e.target.value)} - className="font-select" - > - <option value="default">Default</option> - <option value="serif">Serif</option> - <option value="sans">Sans-Serif</option> - <option value="mono">Monospace</option> - </select> - </div> - </div> - )} - - <div className="add-feed-section"> - <h3>Add New Feed</h3> - <form onSubmit={handleAddFeed} className="add-feed-form"> - <input - type="url" - value={newFeedUrl} - onChange={(e) => setNewFeedUrl(e.target.value)} - placeholder="https://example.com/feed.xml" - required - className="feed-input" - disabled={loading} - /> - <button type="submit" disabled={loading}> - Add Feed - </button> - </form> - </div> - - <div className="import-export-section"> - <div className="import-section"> - <h3>Import Feeds (OPML)</h3> - <form onSubmit={handleImport} className="import-form"> - <input - type="file" - accept=".opml,.xml,.txt" - aria-label="Import Feeds" - onChange={(e) => setImportFile(e.target.files?.[0] || null)} - className="file-input" - disabled={loading} - /> - <button type="submit" disabled={!importFile || loading}> - Import - </button> - </form> - </div> - - <div className="export-section"> - <h3>Export Feeds</h3> - <div className="export-buttons"> - <a href="/api/export/opml" className="export-btn">OPML</a> - <a href="/api/export/text" className="export-btn">Text</a> - <a href="/api/export/json" className="export-btn">JSON</a> - </div> - </div> - - <div className="crawl-section"> - <h3>Actions</h3> - <button onClick={handleCrawl} disabled={loading} className="crawl-btn"> - Crawl All Feeds Now - </button> - </div> - </div> - - {error && <p className="error-message">{error}</p>} - - <div className="feed-list-section"> - <h3>Manage Feeds</h3> - {loading && <p>Loading...</p>} - <ul className="settings-feed-list"> - {feeds.map((feed) => ( - <li key={feed._id} className="settings-feed-item"> - <div className="feed-info"> - <span className="feed-title">{feed.title || '(No Title)'}</span> - <span className="feed-url">{feed.url}</span> - </div> - <button - onClick={() => handleDeleteFeed(feed._id)} - className="delete-btn" - disabled={loading} - title="Delete Feed" - > - Delete - </button> - </li> - ))} - </ul> - </div> - </div> - ); -} |
