import { useCallback, useEffect, useMemo, useState } from 'react'
import ReactFlow, { Background, Controls, Edge, Node, Position, useEdgesState, useNodesState } from 'reactflow'
// @ts-ignore: dagre has no bundled types in this context; rely on runtime import
import dagre from 'dagre'
import 'reactflow/dist/style.css'
import { Card, Spin, Tooltip } from 'antd'
import { apiService } from '../services/api'
import { wsService } from '../services/websocket'

interface TaskGraphProps {
  runId: string
  height?: number
}

// 计划结构类型（与后端 plan.json 对齐的最小子集）
interface AttemptDetail { code?: string; result?: string }
interface TaskResult { attempts?: Record<string | number, AttemptDetail>; is_success?: boolean }
interface Subtask { task_id: string; instruction: string; task_type?: string; is_finished?: boolean; is_decomposed?: boolean; subtasks?: Subtask[]; task_result?: TaskResult; dependent_task_ids?: string[] }
interface PlanLike { raw_question?: string; subtasks?: Subtask[]; current_task?: Subtask }

function toGraph(plan: PlanLike | null): { nodes: Node[]; edges: Edge[] } {
  if (!plan || !plan.subtasks) return { nodes: [], edges: [] }

  const g = new dagre.graphlib.Graph()
  g.setGraph({ rankdir: 'LR', nodesep: 40, ranksep: 90 })
  g.setDefaultEdgeLabel(() => ({}))

  const nodes: Node[] = []
  const edges: Edge[] = []
  const taskMap: Record<string, Subtask> = {}

  const addNode = (t: Subtask, parentId?: string, depth: number = 0, index: number = 0) => {
    const id = t.task_id || `${parentId || 't'}-${index}`
    taskMap[id] = t
    const status: 'done' | 'split' | 'pending' = t.is_finished ? 'done' : (t.is_decomposed ? 'split' : 'pending')
    const label = `${id} • ${t.task_type || ''}`.trim()
    const desc = t.instruction || ''
    const isParent = (t.subtasks && t.subtasks.length > 0) || t.is_decomposed

    const bgByRole = isParent ? '#fff1b8' : '#e6f4ff'
    const borderColor = status === 'done' ? '#52c41a' : status === 'split' ? '#faad14' : '#d9d9d9'

    const nodeWidth = 230
    const nodeHeight = 80
    g.setNode(id, { label, width: nodeWidth, height: nodeHeight })
    if (parentId) g.setEdge(parentId, id)

    const children = t.subtasks || []
    children.forEach((c, i) => addNode(c, id, depth + 1, i))

    // 节点对象暂时先 push （位置稍后通过 dagre 布局更新）
    nodes.push({
      id,
      data: { label: (
        <Tooltip title={<div style={{ maxWidth: 420, whiteSpace: 'pre-wrap' }}>{desc}</div>}>
          <div style={{ padding: 6 }}>
            <div style={{ fontWeight: 600 }}>{label}</div>
            <div style={{ fontSize: 12, color: '#595959' }}>{desc.length > 70 ? `${desc.slice(0, 70)}…` : desc}</div>
          </div>
        </Tooltip>
      ) },
      position: { x: 0, y: 0 },
      style: { background: bgByRole, border: `2px solid ${borderColor}`, borderRadius: 10, width: nodeWidth },
      sourcePosition: Position.Right,
      targetPosition: Position.Left,
    })
  }

  plan.subtasks.forEach((t, i) => addNode(t, undefined, 0, i))

  // 依赖边（dependent_task_ids）虚线展示
  Object.values(taskMap).forEach((t: Subtask) => {
    const deps = t.dependent_task_ids || []
    deps.forEach((depId: string) => {
      if (taskMap[depId]) {
        edges.push({ id: `dep:${depId}->${t.task_id}`, source: depId, target: t.task_id, style: { strokeDasharray: '4 2', stroke: '#8c8c8c' } })
      }
    })
  })

  dagre.layout(g)
  // 更新节点坐标为 dagre 结果
  nodes.forEach(n => {
    const layout = g.node(n.id)
    if (layout) {
      n.position = { x: layout.x - layout.width / 2, y: layout.y - layout.height / 2 }
    }
  })

  // 基本父子边（已在 setEdge 内部添加，需生成 Edge 映射）
  g.edges().forEach((e: any) => {
    const id = `${e.v}->${e.w}`
    if (!edges.find(ed => ed.id === id)) {
      edges.push({ id, source: e.v, target: e.w })
    }
  })

  return { nodes, edges }
}

export default function TaskGraph({ runId, height = 360 }: TaskGraphProps) {
  const [plan, setPlan] = useState<PlanLike | null>(null)
  const [loading, setLoading] = useState(false)

  const loadPlan = useCallback(async () => {
    if (!runId) return
    setLoading(true)
    try {
      const text = await apiService.getResultPlan(runId)
      if (!text) { setPlan(null); return }
      try { setPlan(JSON.parse(text)) } catch { setPlan({ raw_question: '', subtasks: [] } as any) }
    } finally { setLoading(false) }
  }, [runId])

  useEffect(() => {
    loadPlan()
    const onWs = (data: any) => {
      if (data && data.type === 'file') {
        const name = (data.name || '').toLowerCase()
        const url = String(data.url || '')
        if (name === 'plan.json' || url.endsWith('/plan.json')) loadPlan()
      }
    }
    if (runId) wsService.connect(runId, onWs)
    return () => wsService.removeCallback(onWs)
  }, [runId, loadPlan])

  const { nodes, edges } = useMemo(() => toGraph(plan), [plan])
  const [nState, setNodes, onNodesChange] = useNodesState(nodes)
  const [eState, setEdges, onEdgesChange] = useEdgesState(edges)

  useEffect(() => { setNodes(nodes); setEdges(edges) }, [nodes, edges, setNodes, setEdges])

  return (
    <Card size="small" title="任务分解图" style={{ marginBottom: 12 }}>
      {loading ? <Spin /> : (
        <div style={{ width: '100%', height }}>
          <ReactFlow nodes={nState} edges={eState} onNodesChange={onNodesChange} onEdgesChange={onEdgesChange} fitView>
            <Background gap={12} />
            <Controls showInteractive={false} />
          </ReactFlow>
        </div>
      )}
    </Card>
  )
}
