Compare commits

...

11 Commits

Author SHA1 Message Date
17b008a674 Merge pull request 'stroke fix' (#55) from pedigree-update into master
Reviewed-on: #55
2026-03-11 15:49:55 -05:00
jason
9b3210a81e stroke fix 2026-03-11 15:49:46 -05:00
81357e87ae Merge pull request 'halo effect' (#54) from pedigree-update into master
Reviewed-on: #54
2026-03-11 15:41:57 -05:00
jason
8abd5e2db6 halo effect 2026-03-11 15:41:30 -05:00
a63617d9c0 Merge pull request 'remove shadow' (#53) from pedigree-update into master
Reviewed-on: #53
2026-03-11 15:37:49 -05:00
jason
7195aaecfc remove shadow 2026-03-11 15:37:38 -05:00
34bf29d8bf Merge pull request 'text update' (#52) from pedigree-update into master
Reviewed-on: #52
2026-03-11 15:33:49 -05:00
jason
4f3074b1f4 text update 2026-03-11 15:33:23 -05:00
3c7ba1775f Merge pull request 'ped changes' (#51) from pedigree-update into master
Reviewed-on: #51
2026-03-11 15:27:42 -05:00
jason
0a0a5d232c ped changes 2026-03-11 15:26:35 -05:00
jason
58b53c981e feat: Add pedigree routes for COI calculation, direct relation checks, and ancestral/descendant trees. 2026-03-11 14:48:59 -05:00
5 changed files with 83 additions and 43 deletions

View File

@@ -46,15 +46,15 @@ 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 — kept within the circle so it doesn't bleed onto text labels */}
<circle <circle
r={r + 10} r={r - 4}
fill={glowColor} fill={glowColor}
style={{ filter: 'blur(6px)' }} style={{ filter: 'blur(4px)' }}
/> />
{/* Outer ring */} {/* Outer ring */}
@@ -95,25 +95,25 @@ 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' }} stroke="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' }} stroke="none"
style={{ fill: isRoot ? '#ffffff' : '#f8fafc', pointerEvents: 'none' }}
> >
{nodeDatum.name} {nodeDatum.name}
</text> </text>
@@ -121,13 +121,13 @@ 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' }} stroke="none"
style={{ fill: '#cbd5e1', pointerEvents: 'none' }}
> >
{breed} {breed}
</text> </text>
@@ -136,13 +136,13 @@ 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' }} stroke="none"
style={{ fill: '#94a3b8', pointerEvents: 'none' }}
> >
{nodeDatum.attributes.registration} {nodeDatum.attributes.registration}
</text> </text>
@@ -151,13 +151,13 @@ 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' }} stroke="none"
style={{ fill: '#94a3b8', pointerEvents: 'none' }}
> >
({nodeDatum.attributes.birth_year}) ({nodeDatum.attributes.birth_year})
</text> </text>
@@ -232,8 +232,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}

View File

@@ -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 */

View File

@@ -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
View 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
View 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');