rfunds-online/components/nodes/OutcomeNode.tsx

300 lines
14 KiB
TypeScript

'use client'
import { memo, useState, useCallback, useMemo } from 'react'
import { Handle, Position } from '@xyflow/react'
import type { NodeProps } from '@xyflow/react'
import type { OutcomeNodeData } from '@/lib/types'
import { useConnectionState } from '../ConnectionContext'
function OutcomeNode({ data, selected, id }: NodeProps) {
const nodeData = data as OutcomeNodeData
const { label, description, fundingReceived, fundingTarget, status } = nodeData
const [isExpanded, setIsExpanded] = useState(false)
const connectingFrom = useConnectionState()
const progress = fundingTarget > 0 ? Math.min(100, (fundingReceived / fundingTarget) * 100) : 0
const isFunded = fundingReceived >= fundingTarget
const isPartial = fundingReceived > 0 && fundingReceived < fundingTarget
// Handle highlighting: outcome targets glow when dragging from spending or stream handles
const isTargetHighlighted = useMemo(() => {
if (!connectingFrom || connectingFrom.nodeId === id) return false
const handleId = connectingFrom.handleId
if (handleId === 'spending-out') return true
if (handleId === 'stream-out') return true
return false
}, [connectingFrom, id])
const statusColors = {
'not-started': { bg: 'bg-slate-100', text: 'text-slate-600', border: 'border-slate-300' },
'in-progress': { bg: 'bg-blue-100', text: 'text-blue-700', border: 'border-blue-300' },
'completed': { bg: 'bg-emerald-100', text: 'text-emerald-700', border: 'border-emerald-300' },
'blocked': { bg: 'bg-red-100', text: 'text-red-700', border: 'border-red-300' },
}
const colors = statusColors[status] || statusColors['not-started']
const handleDoubleClick = useCallback((e: React.MouseEvent) => {
e.stopPropagation()
setIsExpanded(true)
}, [])
const handleCloseExpanded = useCallback(() => {
setIsExpanded(false)
}, [])
return (
<>
<div
className={`
bg-white rounded-lg shadow-lg border-2 min-w-[200px] max-w-[240px]
transition-all duration-200
${selected ? 'border-pink-500 shadow-pink-200' : 'border-slate-200'}
`}
onDoubleClick={handleDoubleClick}
>
<Handle
type="target"
position={Position.Top}
className={`!w-4 !h-4 !bg-pink-500 !border-2 !border-white !-top-2 transition-all ${
isTargetHighlighted ? '!animate-pulse !ring-2 !ring-blue-400 !scale-125' : ''
}`}
/>
<div className="px-3 py-2 border-b border-slate-100 bg-gradient-to-r from-pink-50 to-purple-50 rounded-t-md">
<div className="flex items-center gap-2">
<div className="w-6 h-6 bg-pink-500 rounded flex items-center justify-center flex-shrink-0">
<svg className="w-4 h-4 text-white" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
</div>
<div className="min-w-0">
<span className="font-semibold text-slate-800 text-sm truncate block">{label}</span>
{nodeData.source?.type === 'rvote' && nodeData.source.rvoteSpaceSlug && nodeData.source.rvoteProposalId ? (
<a
href={`https://rvote.online/s/${nodeData.source.rvoteSpaceSlug}/proposal/${nodeData.source.rvoteProposalId}`}
target="_blank"
rel="noopener noreferrer"
className="text-[9px] text-violet-500 font-medium hover:text-violet-700 hover:underline inline-flex items-center gap-0.5 transition-colors"
onClick={(e) => e.stopPropagation()}
>
rVote &bull; score +{nodeData.source.rvoteProposalScore ?? 0}
<svg className="w-2.5 h-2.5" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M10 6H6a2 2 0 00-2 2v10a2 2 0 002 2h10a2 2 0 002-2v-4M14 4h6m0 0v6m0-6L10 14" />
</svg>
</a>
) : nodeData.source?.type === 'rvote' ? (
<span className="text-[9px] text-violet-500 font-medium">
rVote &bull; score +{nodeData.source.rvoteProposalScore ?? 0}
</span>
) : null}
</div>
</div>
</div>
<div className="p-3 space-y-2">
{description && (
<p className="text-xs text-slate-500 line-clamp-2">{description}</p>
)}
<div className="flex items-center justify-between">
<span className={`text-[10px] px-2 py-0.5 rounded-full font-medium uppercase tracking-wide ${colors.bg} ${colors.text}`}>
{status.replace('-', ' ')}
</span>
{isFunded && (
<svg className="w-4 h-4 text-emerald-500" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clipRule="evenodd" />
</svg>
)}
</div>
<div>
<div className="flex justify-between items-center mb-1">
<span className="text-[10px] text-slate-500 uppercase tracking-wide">Funding</span>
<span className="text-xs font-mono text-slate-700">
${Math.floor(fundingReceived).toLocaleString()} / ${fundingTarget.toLocaleString()}
</span>
</div>
<div className="h-2 bg-slate-100 rounded-full overflow-hidden">
<div
className={`h-full transition-all duration-500 ${
isFunded ? 'bg-emerald-500' : isPartial ? 'bg-blue-500' : 'bg-slate-300'
}`}
style={{ width: `${progress}%` }}
/>
</div>
<div className="text-right mt-0.5">
<span className="text-[10px] font-medium text-slate-500">{progress.toFixed(0)}%</span>
</div>
</div>
<div className="text-center">
<span className="text-[8px] text-slate-400">Double-click for details</span>
</div>
</div>
</div>
{/* Expanded Detail Modal */}
{isExpanded && (
<div
className="fixed inset-0 bg-black/50 flex items-center justify-center z-50"
onClick={handleCloseExpanded}
>
<div
className="bg-white rounded-2xl shadow-2xl p-6 min-w-[400px] max-w-lg max-h-[90vh] overflow-y-auto"
onClick={(e) => e.stopPropagation()}
>
{/* Header */}
<div className="flex items-center justify-between mb-4">
<div className="flex items-center gap-3">
<div className="w-8 h-8 bg-pink-500 rounded-lg flex items-center justify-center flex-shrink-0">
<svg className="w-5 h-5 text-white" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
</div>
<div>
<h3 className="text-lg font-bold text-slate-800">{label}</h3>
<div className="flex items-center gap-2">
<span className={`text-[10px] px-2 py-0.5 rounded-full font-medium uppercase tracking-wide ${colors.bg} ${colors.text}`}>
{status.replace('-', ' ')}
</span>
{nodeData.source?.type === 'rvote' && nodeData.source.rvoteSpaceSlug && nodeData.source.rvoteProposalId && (
<a
href={`https://rvote.online/s/${nodeData.source.rvoteSpaceSlug}/proposal/${nodeData.source.rvoteProposalId}`}
target="_blank"
rel="noopener noreferrer"
className="text-[10px] text-violet-500 font-medium hover:text-violet-700 hover:underline inline-flex items-center gap-0.5"
>
rVote
<svg className="w-2.5 h-2.5" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M10 6H6a2 2 0 00-2 2v10a2 2 0 002 2h10a2 2 0 002-2v-4M14 4h6m0 0v6m0-6L10 14" />
</svg>
</a>
)}
</div>
</div>
</div>
<button
onClick={handleCloseExpanded}
className="text-slate-400 hover:text-slate-600"
>
<svg className="w-5 h-5" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" />
</svg>
</button>
</div>
{/* Description */}
{description && (
<div className="mb-4 p-3 bg-slate-50 rounded-xl">
<span className="text-[10px] text-slate-500 uppercase tracking-wide block mb-1">Description</span>
<p className="text-sm text-slate-700">{description}</p>
</div>
)}
{/* Funding Progress */}
<div className="mb-4 p-3 bg-gradient-to-r from-pink-50 to-purple-50 rounded-xl">
<span className="text-[10px] text-slate-500 uppercase tracking-wide block mb-2">Funding Progress</span>
<div className="flex justify-between items-center mb-2">
<span className="text-2xl font-bold font-mono text-slate-800">
${Math.floor(fundingReceived).toLocaleString()}
</span>
<span className="text-sm text-slate-500">
/ ${fundingTarget.toLocaleString()}
</span>
</div>
<div className="h-3 bg-white/60 rounded-full overflow-hidden">
<div
className={`h-full transition-all duration-500 rounded-full ${
isFunded ? 'bg-emerald-500' : isPartial ? 'bg-blue-500' : 'bg-slate-300'
}`}
style={{ width: `${progress}%` }}
/>
</div>
<div className="text-right mt-1">
<span className="text-xs font-medium text-slate-600">{progress.toFixed(1)}%</span>
</div>
</div>
{/* rVote Details */}
{nodeData.source?.type === 'rvote' && (
<div className="mb-4 p-3 bg-violet-50 rounded-xl">
<span className="text-[10px] text-violet-600 uppercase tracking-wide block mb-2">rVote Details</span>
<div className="space-y-1.5 text-xs">
<div className="flex justify-between">
<span className="text-slate-500">Score</span>
<span className="font-mono text-violet-700 font-medium">+{nodeData.source.rvoteProposalScore ?? 0}</span>
</div>
{nodeData.source.rvoteProposalStatus && (
<div className="flex justify-between">
<span className="text-slate-500">Status</span>
<span className="text-violet-700 font-medium capitalize">{nodeData.source.rvoteProposalStatus}</span>
</div>
)}
{nodeData.source.rvoteSpaceSlug && nodeData.source.rvoteProposalId && (
<a
href={`https://rvote.online/s/${nodeData.source.rvoteSpaceSlug}/proposal/${nodeData.source.rvoteProposalId}`}
target="_blank"
rel="noopener noreferrer"
className="block text-center mt-2 px-3 py-1.5 bg-violet-600 text-white rounded-lg text-xs font-medium hover:bg-violet-500 transition-colors"
>
View on rVote
</a>
)}
</div>
</div>
)}
{/* Milestones */}
{nodeData.milestones && nodeData.milestones.length > 0 && (
<div className="mb-4 p-3 bg-emerald-50 rounded-xl">
<span className="text-[10px] text-emerald-600 uppercase tracking-wide block mb-2">Milestones</span>
<div className="space-y-1.5">
{nodeData.milestones.map((milestone, i) => (
<div key={i} className="flex items-center gap-2 text-xs">
{milestone.completedAt ? (
<svg className="w-3.5 h-3.5 text-emerald-500 flex-shrink-0" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clipRule="evenodd" />
</svg>
) : (
<div className="w-3.5 h-3.5 rounded-full border-2 border-slate-300 flex-shrink-0" />
)}
<span className={milestone.completedAt ? 'text-slate-500 line-through' : 'text-slate-700'}>
{milestone.label}
</span>
</div>
))}
</div>
</div>
)}
{/* Completion Info */}
{nodeData.completedAt && (
<div className="mb-4 p-3 bg-emerald-50 rounded-xl flex items-center gap-2">
<svg className="w-5 h-5 text-emerald-500" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clipRule="evenodd" />
</svg>
<span className="text-sm text-emerald-700 font-medium">
Completed {new Date(nodeData.completedAt).toLocaleDateString()}
</span>
</div>
)}
{/* Close Button */}
<div className="flex justify-center mt-4">
<button
onClick={handleCloseExpanded}
className="px-6 py-2 bg-slate-800 text-white rounded-lg hover:bg-slate-700 transition-colors font-medium"
>
Close
</button>
</div>
</div>
</div>
)}
</>
)
}
export default memo(OutcomeNode)