271 lines
9.3 KiB
TypeScript
271 lines
9.3 KiB
TypeScript
'use client'
|
|
|
|
import { useState } from 'react'
|
|
|
|
export function TradingTools() {
|
|
const [activeTool, setActiveTool] = useState<'calculator' | 'alerts' | 'notes'>('calculator')
|
|
|
|
return (
|
|
<div className="space-y-4">
|
|
{/* Tool Navigation */}
|
|
<div className="flex gap-2">
|
|
{[
|
|
{ id: 'calculator', label: '🧮 Position Calculator', count: 0 },
|
|
{ id: 'alerts', label: '🔔 Trade Alerts', count: 0 },
|
|
{ id: 'notes', label: '📝 Trade Notes', count: 0 },
|
|
].map(tool => (
|
|
<button
|
|
key={tool.id}
|
|
onClick={() => setActiveTool(tool.id as any)}
|
|
className={`px-4 py-2 rounded-lg text-sm font-medium transition ${
|
|
activeTool === tool.id
|
|
? 'bg-brand-pink text-white'
|
|
: 'bg-white/10 text-white/70 hover:bg-white/20'
|
|
}`}
|
|
>
|
|
{tool.label}
|
|
</button>
|
|
))}
|
|
</div>
|
|
|
|
{/* Position Calculator */}
|
|
{activeTool === 'calculator' && <PositionCalculator />}
|
|
|
|
{/* Trade Alerts */}
|
|
{activeTool === 'alerts' && <TradeAlerts />}
|
|
|
|
{/* Trade Notes */}
|
|
{activeTool === 'notes' && <TradeNotes />}
|
|
</div>
|
|
)
|
|
}
|
|
|
|
function PositionCalculator() {
|
|
const [form, setForm] = useState({
|
|
accountSize: '10000',
|
|
riskPercent: '2',
|
|
stopLossPercent: '1',
|
|
leverage: '1',
|
|
entryPrice: '',
|
|
direction: 'long' as 'long' | 'short',
|
|
})
|
|
|
|
const calculate = () => {
|
|
const account = parseFloat(form.accountSize) || 0
|
|
const riskPct = parseFloat(form.riskPercent) || 0
|
|
const slPct = parseFloat(form.stopLossPercent) || 0
|
|
const entry = parseFloat(form.entryPrice) || 0
|
|
const lev = parseFloat(form.leverage) || 1
|
|
|
|
const riskAmount = account * (riskPct / 100)
|
|
const positionSize = lev * (riskAmount / (slPct / 100))
|
|
const margin = positionSize / lev
|
|
const potentialProfit = positionSize * (slPct / 100) * 2 // 2:1
|
|
const potentialLoss = riskAmount
|
|
|
|
return {
|
|
riskAmount,
|
|
positionSize,
|
|
margin,
|
|
potentialProfit,
|
|
potentialLoss,
|
|
rr: '2:1',
|
|
}
|
|
}
|
|
|
|
const result = calculate()
|
|
|
|
return (
|
|
<div className="border border-white/20 rounded-lg p-4 bg-white/5">
|
|
<h3 className="text-lg font-bold mb-4">🧮 Position Size Calculator</h3>
|
|
|
|
<div className="grid grid-cols-2 gap-4">
|
|
<div>
|
|
<label className="block text-sm text-white/70 mb-1">Account Size ($)</label>
|
|
<input
|
|
type="number"
|
|
value={form.accountSize}
|
|
onChange={e => setForm({...form, accountSize: e.target.value})}
|
|
className="w-full bg-white/10 border border-white/20 rounded-lg px-3 py-2 text-white"
|
|
/>
|
|
</div>
|
|
|
|
<div>
|
|
<label className="block text-sm text-white/70 mb-1">Risk Per Trade (%)</label>
|
|
<input
|
|
type="number"
|
|
value={form.riskPercent}
|
|
onChange={e => setForm({...form, riskPercent: e.target.value})}
|
|
className="w-full bg-white/10 border border-white/20 rounded-lg px-3 py-2 text-white"
|
|
/>
|
|
</div>
|
|
|
|
<div>
|
|
<label className="block text-sm text-white/70 mb-1">Stop Loss (%)</label>
|
|
<input
|
|
type="number"
|
|
value={form.stopLossPercent}
|
|
onChange={e => setForm({...form, stopLossPercent: e.target.value})}
|
|
className="w-full bg-white/10 border border-white/20 rounded-lg px-3 py-2 text-white"
|
|
/>
|
|
</div>
|
|
|
|
<div>
|
|
<label className="block text-sm text-white/70 mb-1">Leverage</label>
|
|
<input
|
|
type="number"
|
|
value={form.leverage}
|
|
onChange={e => setForm({...form, leverage: e.target.value})}
|
|
className="w-full bg-white/10 border border-white/20 rounded-lg px-3 py-2 text-white"
|
|
/>
|
|
</div>
|
|
|
|
<div>
|
|
<label className="block text-sm text-white/70 mb-1">Entry Price</label>
|
|
<input
|
|
type="number"
|
|
value={form.entryPrice}
|
|
onChange={e => setForm({...form, entryPrice: e.target.value})}
|
|
placeholder="0.00"
|
|
className="w-full bg-white/10 border border-white/20 rounded-lg px-3 py-2 text-white"
|
|
/>
|
|
</div>
|
|
|
|
<div>
|
|
<label className="block text-sm text-white/70 mb-1">Direction</label>
|
|
<div className="flex gap-2">
|
|
<button
|
|
onClick={() => setForm({...form, direction: 'long'})}
|
|
className={`flex-1 py-2 rounded-lg ${
|
|
form.direction === 'long' ? 'bg-green-500' : 'bg-white/10'
|
|
} text-white`}
|
|
>
|
|
LONG
|
|
</button>
|
|
<button
|
|
onClick={() => setForm({...form, direction: 'short'})}
|
|
className={`flex-1 py-2 rounded-lg ${
|
|
form.direction === 'short' ? 'bg-red-500' : 'bg-white/10'
|
|
} text-white`}
|
|
>
|
|
SHORT
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Results */}
|
|
<div className="mt-6 grid grid-cols-2 md:grid-cols-4 gap-3">
|
|
<div className="p-3 bg-white/5 rounded-lg text-center">
|
|
<p className="text-xs text-white/50">Max Risk</p>
|
|
<p className="text-xl font-bold text-red-400">${result.riskAmount.toFixed(2)}</p>
|
|
</div>
|
|
<div className="p-3 bg-white/5 rounded-lg text-center">
|
|
<p className="text-xs text-white/50">Position Size</p>
|
|
<p className="text-xl font-bold text-white">${result.positionSize.toFixed(2)}</p>
|
|
</div>
|
|
<div className="p-3 bg-white/5 rounded-lg text-center">
|
|
<p className="text-xs text-white/50">Required Margin</p>
|
|
<p className="text-xl font-bold text-blue-400">${result.margin.toFixed(2)}</p>
|
|
</div>
|
|
<div className="p-3 bg-white/5 rounded-lg text-center">
|
|
<p className="text-xs text-white/50">Potential Profit</p>
|
|
<p className="text-xl font-bold text-green-400">+${result.potentialProfit.toFixed(2)}</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
function TradeAlerts() {
|
|
const [alerts, setAlerts] = useState([
|
|
{ id: '1', pair: 'BTC/USD', type: 'SL Hit', price: 66500, time: '2h ago', triggered: true },
|
|
{ id: '2', pair: 'ETH/USD', type: 'TP Hit', price: 3200, time: '5h ago', triggered: true },
|
|
])
|
|
|
|
return (
|
|
<div className="border border-white/20 rounded-lg p-4 bg-white/5">
|
|
<h3 className="text-lg font-bold mb-4">🔔 Trade Alerts</h3>
|
|
|
|
{alerts.length === 0 ? (
|
|
<p className="text-white/50 text-center py-8">No alerts yet</p>
|
|
) : (
|
|
<div className="space-y-2">
|
|
{alerts.map(alert => (
|
|
<div
|
|
key={alert.id}
|
|
className={`p-3 rounded-lg border flex justify-between items-center ${
|
|
alert.triggered
|
|
? 'bg-green-500/10 border-green-500/30'
|
|
: 'bg-yellow-500/10 border-yellow-500/30'
|
|
}`}
|
|
>
|
|
<div>
|
|
<span className="font-medium">{alert.pair}</span>
|
|
<span className={`ml-2 text-xs px-2 py-0.5 rounded ${
|
|
alert.triggered ? 'bg-green-500/20 text-green-400' : 'bg-yellow-500/20 text-yellow-400'
|
|
}`}>
|
|
{alert.type}
|
|
</span>
|
|
</div>
|
|
<div className="text-right">
|
|
<p className="font-medium">${alert.price.toLocaleString()}</p>
|
|
<p className="text-xs text-white/50">{alert.time}</p>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
)
|
|
}
|
|
|
|
function TradeNotes() {
|
|
const [notes, setNotes] = useState<{id: string, date: string, content: string, tradeId?: string}[]>([])
|
|
const [newNote, setNewNote] = useState('')
|
|
|
|
const addNote = () => {
|
|
if (!newNote.trim()) return
|
|
setNotes([{ id: Date.now().toString(), date: new Date().toISOString(), content: newNote }, ...notes])
|
|
setNewNote('')
|
|
}
|
|
|
|
return (
|
|
<div className="border border-white/20 rounded-lg p-4 bg-white/5">
|
|
<h3 className="text-lg font-bold mb-4">📝 Trade Notes</h3>
|
|
|
|
{/* Add Note */}
|
|
<div className="mb-4">
|
|
<textarea
|
|
value={newNote}
|
|
onChange={e => setNewNote(e.target.value)}
|
|
placeholder="Write a note about your trade..."
|
|
className="w-full bg-white/10 border border-white/20 rounded-lg px-3 py-2 text-white placeholder-white/50 h-24 resize-none"
|
|
/>
|
|
<button
|
|
onClick={addNote}
|
|
className="mt-2 px-4 py-2 bg-brand-pink rounded-lg text-white font-medium hover:bg-brand-pink/80"
|
|
>
|
|
Add Note
|
|
</button>
|
|
</div>
|
|
|
|
{/* Notes List */}
|
|
{notes.length === 0 ? (
|
|
<p className="text-white/50 text-center py-8">No notes yet. Start journaling your trades!</p>
|
|
) : (
|
|
<div className="space-y-2 max-h-[400px] overflow-y-auto">
|
|
{notes.map(note => (
|
|
<div key={note.id} className="p-3 bg-white/5 rounded-lg border border-white/10">
|
|
<p className="text-sm text-white/80">{note.content}</p>
|
|
<p className="text-xs text-white/50 mt-2">
|
|
{new Date(note.date).toLocaleDateString()} • {new Date(note.date).toLocaleTimeString()}
|
|
</p>
|
|
</div>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
)
|
|
}
|