Compare commits
5 Commits
main2
...
34bf29d8bf
| Author | SHA1 | Date | |
|---|---|---|---|
| 34bf29d8bf | |||
|
|
4f3074b1f4 | ||
| 3c7ba1775f | |||
|
|
0a0a5d232c | ||
|
|
58b53c981e |
@@ -46,13 +46,13 @@ const PedigreeTree = ({ dogId, pedigreeData, coi }) => {
|
|||||||
: (isMale ? 'rgba(59,130,246,0.3)' : 'rgba(236,72,153,0.3)')
|
: (isMale ? 'rgba(59,130,246,0.3)' : 'rgba(236,72,153,0.3)')
|
||||||
const ringColor = isRoot ? rootAccent : nodeColor
|
const ringColor = isRoot ? rootAccent : nodeColor
|
||||||
|
|
||||||
const r = isRoot ? 34 : 28
|
const r = isRoot ? 46 : 38
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<g>
|
<g>
|
||||||
{/* Glow halo */}
|
{/* Glow halo */}
|
||||||
<circle
|
<circle
|
||||||
r={r + 10}
|
r={r + 12}
|
||||||
fill={glowColor}
|
fill={glowColor}
|
||||||
style={{ filter: 'blur(6px)' }}
|
style={{ filter: 'blur(6px)' }}
|
||||||
/>
|
/>
|
||||||
@@ -95,25 +95,23 @@ const PedigreeTree = ({ dogId, pedigreeData, coi }) => {
|
|||||||
|
|
||||||
{/* Gender / crown icon */}
|
{/* Gender / crown icon */}
|
||||||
<text
|
<text
|
||||||
fill={isRoot ? '#fff' : '#fff'}
|
fontSize={isRoot ? 28 : 24}
|
||||||
fontSize={isRoot ? 22 : 18}
|
|
||||||
textAnchor="middle"
|
textAnchor="middle"
|
||||||
dy="7"
|
dy="8"
|
||||||
style={{ pointerEvents: 'none', userSelect: 'none' }}
|
style={{ fill: '#ffffff', pointerEvents: 'none', userSelect: 'none' }}
|
||||||
>
|
>
|
||||||
{isRoot ? '👑' : (isMale ? '♂' : '♀')}
|
{isRoot ? '👑' : (isMale ? '♂' : '♀')}
|
||||||
</text>
|
</text>
|
||||||
|
|
||||||
{/* Name label */}
|
{/* Name label */}
|
||||||
<text
|
<text
|
||||||
fill="var(--text-primary, #f5f0e8)"
|
fontSize={isRoot ? 22 : 18}
|
||||||
fontSize={isRoot ? 15 : 13}
|
|
||||||
fontWeight={isRoot ? '700' : '600'}
|
fontWeight={isRoot ? '700' : '600'}
|
||||||
fontFamily="Inter, sans-serif"
|
fontFamily="Inter, sans-serif"
|
||||||
textAnchor="middle"
|
textAnchor="middle"
|
||||||
x="0"
|
x="0"
|
||||||
y={r + 18}
|
y={r + 32}
|
||||||
style={{ pointerEvents: 'none' }}
|
style={{ fill: isRoot ? '#ffffff' : '#f8fafc', pointerEvents: 'none', textShadow: '0 2px 5px rgba(0,0,0,0.9)' }}
|
||||||
>
|
>
|
||||||
{nodeDatum.name}
|
{nodeDatum.name}
|
||||||
</text>
|
</text>
|
||||||
@@ -121,13 +119,12 @@ const PedigreeTree = ({ dogId, pedigreeData, coi }) => {
|
|||||||
{/* Breed label (subtle) */}
|
{/* Breed label (subtle) */}
|
||||||
{breed && (
|
{breed && (
|
||||||
<text
|
<text
|
||||||
fill="var(--text-muted, #8c8472)"
|
fontSize="14"
|
||||||
fontSize="10"
|
|
||||||
fontFamily="Inter, sans-serif"
|
fontFamily="Inter, sans-serif"
|
||||||
textAnchor="middle"
|
textAnchor="middle"
|
||||||
x="0"
|
x="0"
|
||||||
y={r + 31}
|
y={r + 52}
|
||||||
style={{ pointerEvents: 'none' }}
|
style={{ fill: '#cbd5e1', pointerEvents: 'none', textShadow: '0 1px 4px rgba(0,0,0,0.9)' }}
|
||||||
>
|
>
|
||||||
{breed}
|
{breed}
|
||||||
</text>
|
</text>
|
||||||
@@ -136,13 +133,12 @@ const PedigreeTree = ({ dogId, pedigreeData, coi }) => {
|
|||||||
{/* Registration number */}
|
{/* Registration number */}
|
||||||
{nodeDatum.attributes?.registration && (
|
{nodeDatum.attributes?.registration && (
|
||||||
<text
|
<text
|
||||||
fill="var(--text-muted, #8c8472)"
|
fontSize="14"
|
||||||
fontSize="10"
|
|
||||||
fontFamily="Inter, sans-serif"
|
fontFamily="Inter, sans-serif"
|
||||||
textAnchor="middle"
|
textAnchor="middle"
|
||||||
x="0"
|
x="0"
|
||||||
y={r + (breed ? 44 : 31)}
|
y={r + (breed ? 70 : 52)}
|
||||||
style={{ pointerEvents: 'none' }}
|
style={{ fill: '#94a3b8', pointerEvents: 'none', textShadow: '0 1px 4px rgba(0,0,0,0.9)' }}
|
||||||
>
|
>
|
||||||
{nodeDatum.attributes.registration}
|
{nodeDatum.attributes.registration}
|
||||||
</text>
|
</text>
|
||||||
@@ -151,13 +147,12 @@ const PedigreeTree = ({ dogId, pedigreeData, coi }) => {
|
|||||||
{/* Birth year */}
|
{/* Birth year */}
|
||||||
{nodeDatum.attributes?.birth_year && (
|
{nodeDatum.attributes?.birth_year && (
|
||||||
<text
|
<text
|
||||||
fill="var(--text-muted, #8c8472)"
|
fontSize="14"
|
||||||
fontSize="10"
|
|
||||||
fontFamily="Inter, sans-serif"
|
fontFamily="Inter, sans-serif"
|
||||||
textAnchor="middle"
|
textAnchor="middle"
|
||||||
x="0"
|
x="0"
|
||||||
y={r + (breed ? 57 : (nodeDatum.attributes?.registration ? 44 : 31))}
|
y={r + (breed ? 88 : (nodeDatum.attributes?.registration ? 70 : 52))}
|
||||||
style={{ pointerEvents: 'none' }}
|
style={{ fill: '#94a3b8', pointerEvents: 'none', textShadow: '0 1px 4px rgba(0,0,0,0.9)' }}
|
||||||
>
|
>
|
||||||
({nodeDatum.attributes.birth_year})
|
({nodeDatum.attributes.birth_year})
|
||||||
</text>
|
</text>
|
||||||
@@ -232,8 +227,8 @@ const PedigreeTree = ({ dogId, pedigreeData, coi }) => {
|
|||||||
}}
|
}}
|
||||||
orientation="horizontal"
|
orientation="horizontal"
|
||||||
pathFunc="step"
|
pathFunc="step"
|
||||||
separation={{ siblings: 1.6, nonSiblings: 2.2 }}
|
separation={{ siblings: 1.8, nonSiblings: 2.4 }}
|
||||||
nodeSize={{ x: 220, y: 160 }}
|
nodeSize={{ x: 280, y: 200 }}
|
||||||
renderCustomNodeElement={renderCustomNode}
|
renderCustomNodeElement={renderCustomNode}
|
||||||
enableLegacyTransitions
|
enableLegacyTransitions
|
||||||
transitionDuration={300}
|
transitionDuration={300}
|
||||||
|
|||||||
@@ -2,7 +2,8 @@
|
|||||||
position: relative;
|
position: relative;
|
||||||
width: 95vw;
|
width: 95vw;
|
||||||
height: 90vh;
|
height: 90vh;
|
||||||
background: white;
|
background: var(--bg-primary, #1e1e24);
|
||||||
|
border: 1px solid var(--border, #333);
|
||||||
border-radius: 12px;
|
border-radius: 12px;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
@@ -11,7 +12,7 @@
|
|||||||
|
|
||||||
.pedigree-container {
|
.pedigree-container {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
background: linear-gradient(to bottom, #f8fafc 0%, #e2e8f0 100%);
|
background: var(--bg-primary, #1e1e24);
|
||||||
position: relative;
|
position: relative;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
@@ -26,8 +27,8 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
gap: 2rem;
|
gap: 2rem;
|
||||||
padding: 0.75rem 1.5rem;
|
padding: 0.75rem 1.5rem;
|
||||||
background: #f1f5f9;
|
background: var(--bg-elevated, #2a2a35);
|
||||||
border-bottom: 1px solid #e2e8f0;
|
border-bottom: 1px solid var(--border, #333);
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -35,8 +36,8 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 0.5rem;
|
gap: 0.5rem;
|
||||||
font-size: 0.875rem;
|
font-size: 1rem;
|
||||||
color: #475569;
|
color: var(--text-secondary, #a1a1aa);
|
||||||
}
|
}
|
||||||
|
|
||||||
.legend-color {
|
.legend-color {
|
||||||
@@ -57,10 +58,10 @@
|
|||||||
|
|
||||||
.pedigree-info {
|
.pedigree-info {
|
||||||
padding: 0.75rem 1.5rem;
|
padding: 0.75rem 1.5rem;
|
||||||
background: #f8fafc;
|
background: var(--bg-elevated, #2a2a35);
|
||||||
border-top: 1px solid #e2e8f0;
|
border-top: 1px solid var(--border, #333);
|
||||||
font-size: 0.875rem;
|
font-size: 1rem;
|
||||||
color: #64748b;
|
color: var(--text-muted, #a1a1aa);
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -69,7 +70,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.pedigree-info strong {
|
.pedigree-info strong {
|
||||||
color: #334155;
|
color: var(--text-primary, #ffffff);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Override react-d3-tree styles */
|
/* Override react-d3-tree styles */
|
||||||
@@ -94,12 +95,12 @@
|
|||||||
|
|
||||||
.rd3t-label__title {
|
.rd3t-label__title {
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
fill: #1e293b;
|
fill: var(--text-primary, #ffffff);
|
||||||
}
|
}
|
||||||
|
|
||||||
.rd3t-label__attributes {
|
.rd3t-label__attributes {
|
||||||
font-size: 0.875rem;
|
font-size: 1rem;
|
||||||
fill: #64748b;
|
fill: var(--text-muted, #a1a1aa);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Loading state */
|
/* Loading state */
|
||||||
|
|||||||
@@ -124,8 +124,7 @@ function calculateCOI(db, sireId, damId) {
|
|||||||
// 'trial-pairing' as dog IDs and return 404/wrong data.
|
// 'trial-pairing' as dog IDs and return 404/wrong data.
|
||||||
// =====================================================================
|
// =====================================================================
|
||||||
|
|
||||||
// POST /api/pedigree/trial-pairing (alias for /coi)
|
const handleTrialPairing = (req, res) => {
|
||||||
router.post(['/trial-pairing', '/coi'], (req, res) => {
|
|
||||||
try {
|
try {
|
||||||
const { sire_id, dam_id } = req.body;
|
const { sire_id, dam_id } = req.body;
|
||||||
if (!sire_id || !dam_id) {
|
if (!sire_id || !dam_id) {
|
||||||
@@ -156,7 +155,13 @@ router.post(['/trial-pairing', '/coi'], (req, res) => {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
res.status(500).json({ error: error.message });
|
res.status(500).json({ error: error.message });
|
||||||
}
|
}
|
||||||
});
|
};
|
||||||
|
|
||||||
|
// POST /api/pedigree/trial-pairing
|
||||||
|
router.post('/trial-pairing', handleTrialPairing);
|
||||||
|
|
||||||
|
// POST /api/pedigree/coi
|
||||||
|
router.post('/coi', handleTrialPairing);
|
||||||
|
|
||||||
// GET /api/pedigree/:id/coi
|
// GET /api/pedigree/:id/coi
|
||||||
router.get('/:id/coi', (req, res) => {
|
router.get('/:id/coi', (req, res) => {
|
||||||
|
|||||||
26
server/test_app.js
Normal file
26
server/test_app.js
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
const app = require('./index');
|
||||||
|
const http = require('http');
|
||||||
|
|
||||||
|
// Start temporary server
|
||||||
|
const server = http.createServer(app);
|
||||||
|
server.listen(3030, async () => {
|
||||||
|
console.log('Server started on 3030');
|
||||||
|
try {
|
||||||
|
const res = await fetch('http://localhost:3030/api/pedigree/relations/1/2');
|
||||||
|
const text = await res.text();
|
||||||
|
console.log('GET /api/pedigree/relations/1/2 RESPONSE:', res.status, text.substring(0, 150));
|
||||||
|
|
||||||
|
const postRes = await fetch('http://localhost:3030/api/pedigree/trial-pairing', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({ sire_id: 1, dam_id: 2 })
|
||||||
|
});
|
||||||
|
const postText = await postRes.text();
|
||||||
|
console.log('POST /api/pedigree/trial-pairing RESPONSE:', postRes.status, postText.substring(0, 150));
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Fetch error:', err);
|
||||||
|
} finally {
|
||||||
|
server.close();
|
||||||
|
process.exit(0);
|
||||||
|
}
|
||||||
|
});
|
||||||
8
server/test_express.js
Normal file
8
server/test_express.js
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
const express = require('express');
|
||||||
|
const router = express.Router();
|
||||||
|
|
||||||
|
router.post(['/a', '/b'], (req, res) => {
|
||||||
|
res.send('ok');
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log('Started successfully');
|
||||||
Reference in New Issue
Block a user