QuantHive / ui /src /components /MarketPanel.jsx
ARKAISW's picture
Hackathon Final Submission: PettingZoo multi-agent arch, GRPO training, docs
9cb3002
import React, { useMemo } from 'react';
import { motion } from 'framer-motion';
const buildPath = (points, width, height) => {
if (!points.length) {
return '';
}
const prices = points.map((point) => point.price);
const min = Math.min(...prices);
const max = Math.max(...prices);
const range = max - min || 1;
return points
.map((point, index) => {
const x = (index / Math.max(points.length - 1, 1)) * width;
const y = height - ((point.price - min) / range) * height;
return `${index === 0 ? 'M' : 'L'} ${x.toFixed(2)} ${y.toFixed(2)}`;
})
.join(' ');
};
export const MarketPanel = ({ chart, history, trade, engine, lastPriceDelta }) => {
const path = useMemo(() => buildPath(history || [], 250, 90), [history]);
const priceTone = lastPriceDelta < 0 ? '#d95d5d' : '#52b788';
return (
<div className="rounded-[2rem] border border-[#7d5a4f]/15 bg-[#fff8ef]/92 p-6 shadow-[0_28px_50px_rgba(77,44,26,0.12)]">
<div className="flex items-start justify-between gap-4">
<div>
<div className="text-[11px] font-semibold uppercase tracking-[0.34em] text-stone-500">Live Execution</div>
<h3 className="mt-2 text-2xl font-semibold text-stone-800">Market tape</h3>
</div>
<div className="rounded-full bg-white/80 px-3 py-1 text-[10px] font-semibold uppercase tracking-[0.24em] text-stone-500">
{engine?.policy_active ? 'SLM live' : 'Fallback'}
</div>
</div>
<div className="mt-5 rounded-[1.6rem] border border-white/65 bg-white/75 p-4">
<div className="flex items-center justify-between text-[11px] font-semibold uppercase tracking-[0.24em] text-stone-500">
<span>Price</span>
<span>
{lastPriceDelta >= 0 ? '+' : ''}
{lastPriceDelta.toFixed(2)}
</span>
</div>
<div className="mt-3 text-4xl font-semibold tracking-tight text-stone-800">
${chart?.price?.toLocaleString(undefined, { minimumFractionDigits: 2, maximumFractionDigits: 2 })}
</div>
<svg viewBox="0 0 250 90" className="mt-4 h-[90px] w-full overflow-visible">
<path d={path} fill="none" stroke="#edd5c4" strokeWidth="10" strokeLinecap="round" />
<motion.path
d={path}
fill="none"
stroke={priceTone}
strokeWidth="5"
strokeLinecap="round"
initial={{ pathLength: 0 }}
animate={{ pathLength: 1 }}
transition={{ duration: 0.55 }}
/>
</svg>
</div>
<div className="mt-5 grid grid-cols-2 gap-3">
<div className="rounded-[1.4rem] bg-white/75 px-4 py-4">
<div className="text-[10px] uppercase tracking-[0.24em] text-stone-500">Last side</div>
<div className="mt-2 text-lg font-semibold text-stone-800">{trade?.side || 'HOLD'}</div>
</div>
<div className="rounded-[1.4rem] bg-white/75 px-4 py-4">
<div className="text-[10px] uppercase tracking-[0.24em] text-stone-500">Notional</div>
<div className="mt-2 text-lg font-semibold text-stone-800">
${trade?.notional?.toLocaleString(undefined, { maximumFractionDigits: 0 })}
</div>
</div>
</div>
<div className="mt-5 rounded-[1.6rem] border border-white/65 bg-white/80 px-4 py-4">
<div className="text-[10px] font-semibold uppercase tracking-[0.24em] text-stone-500">Decision reason</div>
<div className="mt-2 text-sm leading-6 text-stone-600">{trade?.reason || 'Waiting for the desk to move.'}</div>
</div>
</div>
);
};