before optimization

This commit is contained in:
2026-03-25 19:42:42 +01:00
parent fda3e3ecbf
commit 3ce4d206d7
7 changed files with 107 additions and 47 deletions
+2
View File
@@ -14,3 +14,5 @@ The React Compiler is not enabled on this template because of its impact on dev
## Expanding the ESLint configuration ## Expanding the ESLint configuration
If you are developing a production application, we recommend using TypeScript with type-aware lint rules enabled. Check out the [TS template](https://github.com/vitejs/vite/tree/main/packages/create-vite/template-react-ts) for information on how to integrate TypeScript and [`typescript-eslint`](https://typescript-eslint.io) in your project. If you are developing a production application, we recommend using TypeScript with type-aware lint rules enabled. Check out the [TS template](https://github.com/vitejs/vite/tree/main/packages/create-vite/template-react-ts) for information on how to integrate TypeScript and [`typescript-eslint`](https://typescript-eslint.io) in your project.
## Start Frontend: npm run dev
+13 -4
View File
@@ -3,19 +3,28 @@ import { CardContext } from '../../store/CardContext';
import PrintingRow from '../PrintingRow/PrintingRow'; import PrintingRow from '../PrintingRow/PrintingRow';
function CardRow({ card }) { function CardRow({ card }) {
const { expandedCardId, setExpandedCardId } = useContext(CardContext); const { expandedCardId, setExpandedCardId, ownedAmounts } = useContext(CardContext);
const isExpanded = expandedCardId === card.id; const isExpanded = expandedCardId === card.id;
const toggleExpand = () => setExpandedCardId(isExpanded ? null : card.id); const toggleExpand = () => setExpandedCardId(isExpanded ? null : card.id);
// Calculate total owned across all printings
const totalOwned = card.printings?.reduce((sum, p) => {
const owned = ownedAmounts[card.id]?.[p.set_id] ?? p.amount_owned ?? 0;
return sum + owned;
}, 0) ?? 0;
return ( return (
<div style={{ borderBottom: '1px solid #eee', padding: '0.5rem' }}> <div style={{ borderBottom: '1px solid #eee', padding: '0.5rem' }}>
<div <div
style={{ cursor: 'pointer', display: 'flex', justifyContent: 'space-between' }} style={{ cursor: 'pointer', display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}
onClick={toggleExpand} onClick={toggleExpand}
> >
<span>{card.name}</span> <span>{card.name}</span>
<span>{card.type}</span> <div style={{ display: 'flex', gap: '1rem', alignItems: 'center' }}>
<span>{card.type}</span>
<span>Total owned: {totalOwned}</span>
</div>
</div> </div>
{isExpanded && card.printings?.length > 0 && ( {isExpanded && card.printings?.length > 0 && (
@@ -33,4 +42,4 @@ function CardRow({ card }) {
); );
} }
export default CardRow; export default CardRow;
+7 -25
View File
@@ -1,16 +1,14 @@
import React, { useContext, useState } from 'react'; import React, { useContext, useState, memo } from 'react';
import { CardContext } from '../../store/CardContext'; import { CardContext } from '../../store/CardContext';
function PrintingRow({ card_id, printing }) { function PrintingRow({ card_id, printing }) {
const { ownedAmounts, updateAmount } = useContext(CardContext); const { ownedAmounts, updateAmount } = useContext(CardContext);
const [loading, setLoading] = useState(false); const [loading, setLoading] = useState(false);
// Current amount from context, fallback to DB value const currentAmount = ownedAmounts[card_id]?.[printing.set_id]?.[printing.rarity_id] ?? printing.amount_owned ?? 0;
const currentAmount = ownedAmounts[card_id]?.[printing.set_id] ?? printing.amount_owned ?? 0;
const updateBackend = async (newAmount) => { const updateBackend = async (newAmount) => {
const { set_id, rarity_id } = printing; const { set_id, rarity_id } = printing;
if (card_id == null || set_id == null || rarity_id == null) return; if (card_id == null || set_id == null || rarity_id == null) return;
setLoading(true); setLoading(true);
@@ -18,38 +16,22 @@ function PrintingRow({ card_id, printing }) {
const response = await fetch('http://localhost:3000/collection/amount', { const response = await fetch('http://localhost:3000/collection/amount', {
method: 'PUT', method: 'PUT',
headers: { 'Content-Type': 'application/json' }, headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ body: JSON.stringify({ card_id, set_id, rarity_id, amount_owned: newAmount })
card_id,
set_id,
rarity_id,
amount_owned: newAmount
})
}); });
if (response.ok) { if (response.ok) {
updateAmount(card_id, set_id, newAmount); updateAmount(card_id, set_id, rarity_id, newAmount);
} }
} catch (err) {
// silently fail or handle elsewhere
} finally { } finally {
setLoading(false); setLoading(false);
} }
}; };
const increment = () => updateBackend(currentAmount + 1); const increment = () => updateBackend(currentAmount + 1);
const decrement = () => { const decrement = () => { if (currentAmount > 0) updateBackend(currentAmount - 1); };
if (currentAmount > 0) updateBackend(currentAmount - 1);
};
return ( return (
<div <div style={{ display: 'flex', justifyContent: 'space-between', padding: '0.25rem 0', opacity: loading ? 0.6 : 1 }}>
style={{
display: 'flex',
justifyContent: 'space-between',
padding: '0.25rem 0',
opacity: loading ? 0.6 : 1
}}
>
<span>{printing.set_name} {printing.rarity_name}</span> <span>{printing.set_name} {printing.rarity_name}</span>
<div> <div>
<button onClick={decrement} disabled={loading || currentAmount === 0}></button> <button onClick={decrement} disabled={loading || currentAmount === 0}></button>
@@ -60,4 +42,4 @@ function PrintingRow({ card_id, printing }) {
); );
} }
export default PrintingRow; export default memo(PrintingRow);
+40
View File
@@ -0,0 +1,40 @@
import React from 'react';
function SearchBar({ searchTerm, setSearchTerm }) {
return (
<div style={{ position: 'relative', width: '100%' }}>
<input
type="text"
placeholder="Search by card name..."
value={searchTerm}
onChange={e => setSearchTerm(e.target.value)}
style={{
width: '100%',
padding: '0.5rem 2rem 0.5rem 0.5rem',
borderRadius: '4px',
border: '1px solid #ccc',
boxSizing: 'border-box'
}}
/>
{searchTerm && (
<span
onClick={() => setSearchTerm('')}
style={{
position: 'absolute',
right: '0.5rem',
top: '50%',
transform: 'translateY(-50%)',
cursor: 'pointer',
fontSize: '1rem',
color: '#888',
userSelect: 'none'
}}
>
×
</span>
)}
</div>
);
}
export default SearchBar;
+4 -5
View File
@@ -55,17 +55,16 @@ body {
} }
#root { #root {
width: 1126px; width: 100%;
max-width: 100%; margin: 0;
margin: 0 auto; text-align: left; /* optional: better for layout apps */
text-align: center;
border-inline: 1px solid var(--border);
min-height: 100svh; min-height: 100svh;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
box-sizing: border-box; box-sizing: border-box;
} }
h1, h1,
h2 { h2 {
font-family: var(--heading); font-family: var(--heading);
+27 -3
View File
@@ -1,14 +1,30 @@
import React, { useEffect, useState, useContext } from 'react'; import React, { useEffect, useState, useContext } from 'react';
import CardRow from '../components/CardRow/CardRow'; import CardRow from '../components/CardRow/CardRow';
import SearchBar from '../components/SearchBar/SearchBar';
import { fetchCards } from '../services/api'; import { fetchCards } from '../services/api';
import { CardContext } from '../store/CardContext'; import { CardContext } from '../store/CardContext';
// Debounce hook
function useDebouncedValue(value, delay = 250) {
const [debounced, setDebounced] = useState(value);
useEffect(() => {
const handler = setTimeout(() => setDebounced(value), delay);
return () => clearTimeout(handler);
}, [value, delay]);
return debounced;
}
function HomePage() { function HomePage() {
const [cards, setCards] = useState([]); const [cards, setCards] = useState([]);
const [searchTerm, setSearchTerm] = useState('');
const [loading, setLoading] = useState(true); const [loading, setLoading] = useState(true);
const [error, setError] = useState(null); const [error, setError] = useState(null);
const { expandedCardId, cardImages, setCardImage } = useContext(CardContext); const { expandedCardId, cardImages, setCardImage } = useContext(CardContext);
const debouncedSearchTerm = useDebouncedValue(searchTerm, 250);
useEffect(() => { useEffect(() => {
fetchCards() fetchCards()
.then(data => setCards(data)) .then(data => setCards(data))
@@ -33,12 +49,20 @@ function HomePage() {
const expandedCard = cards.find(c => c.id === expandedCardId); const expandedCard = cards.find(c => c.id === expandedCardId);
// Filter + sort using debounced search
const filteredCards = cards
.filter(card =>
card.name.toLowerCase().includes(debouncedSearchTerm.toLowerCase())
)
.sort((a, b) => a.name.localeCompare(b.name));
return ( return (
<div style={{ display: 'flex', height: '100vh' }}> <div style={{ display: 'flex', height: '100vh' }}>
{/* Left panel: card list */} {/* Left panel: card list */}
<div style={{ flex: 1, borderRight: '1px solid #ccc', padding: '1rem', overflowY: 'auto' }}> <div style={{ flex: 2, borderRight: '1px solid #ccc', padding: '1rem', overflowY: 'auto' }}>
<h2>Card List</h2> <h2>Card List</h2>
{cards.map(card => ( <SearchBar searchTerm={searchTerm} setSearchTerm={setSearchTerm} />
{filteredCards.map(card => (
<CardRow key={card.id} card={card} /> <CardRow key={card.id} card={card} />
))} ))}
</div> </div>
@@ -64,4 +88,4 @@ function HomePage() {
); );
} }
export default HomePage; export default HomePage;
+14 -10
View File
@@ -3,29 +3,33 @@ import React, { createContext, useState } from 'react';
export const CardContext = createContext(); export const CardContext = createContext();
export function CardProvider({ children }) { export function CardProvider({ children }) {
const [expandedCardId, setExpandedCardId] = useState(null); // ownedAmounts structure:
// { [card_id]: { "[set_id]-[rarity_id]": amount, ... }, ... }
const [ownedAmounts, setOwnedAmounts] = useState({}); const [ownedAmounts, setOwnedAmounts] = useState({});
const [cardImages, setCardImages] = useState({}); // cache loaded images const [expandedCardId, setExpandedCardId] = useState(null);
const [cardImages, setCardImages] = useState({});
// Use set_id as the key to match backend const updateAmount = (card_id, key, amount) => {
const updateAmount = (cardId, setId, value) => {
setOwnedAmounts(prev => ({ setOwnedAmounts(prev => ({
...prev, ...prev,
[cardId]: { ...(prev[cardId] || {}), [setId]: value } [card_id]: {
...prev[card_id],
[key]: amount
}
})); }));
}; };
const setCardImage = (cardId, imageData) => { const setCardImage = (card_id, blob) => {
setCardImages(prev => ({ ...prev, [cardId]: imageData })); setCardImages(prev => ({ ...prev, [card_id]: blob }));
}; };
return ( return (
<CardContext.Provider <CardContext.Provider
value={{ value={{
expandedCardId,
setExpandedCardId,
ownedAmounts, ownedAmounts,
updateAmount, updateAmount,
expandedCardId,
setExpandedCardId,
cardImages, cardImages,
setCardImage setCardImage
}} }}
@@ -33,4 +37,4 @@ export function CardProvider({ children }) {
{children} {children}
</CardContext.Provider> </CardContext.Provider>
); );
} }