mirror of
https://github.com/supabase/supabase.git
synced 2026-05-08 09:50:33 -04:00
56de26fe22
This PR migrates the whole monorepo to use Tailwind v4: - Removed `@tailwindcss/container-queries` plugin since it's included by default in v4, - Bump all instances of Tailwind to v4. Made minimal changes to the shared config to remove non-supported features (`alpha` mentions), - Migrate all apps to be compatible with v4 configs, - Fix the `typography.css` import in 3 apps, - Add missing rules which were included by default in v3, - Run `pnpm dlx @tailwindcss/upgrade` on all apps, which renames a lot of classes - Rename all misnamed classes according to https://tailwindcss.com/docs/upgrade-guide#renamed-utilities in all apps. --------- Co-authored-by: Jordi Enric <jordi.err@gmail.com>
146 lines
6.0 KiB
TypeScript
146 lines
6.0 KiB
TypeScript
import { ChevronDown, ChevronRight } from 'lucide-react'
|
|
import { useState } from 'react'
|
|
import { cn, Tooltip, TooltipContent, TooltipTrigger } from 'ui'
|
|
|
|
import { parseDetailLines } from './ExplainVisualizer.parser'
|
|
import { RowCountIndicator } from './ExplainVisualizer.RowCountIndicator'
|
|
import type { ExplainNode } from './ExplainVisualizer.types'
|
|
import { formatNodeDuration, getScanBarColor, getScanBorderColor } from './ExplainVisualizer.utils'
|
|
|
|
interface ExplainNodeRowProps {
|
|
node: ExplainNode
|
|
depth: number
|
|
/** Maximum duration across all nodes, used to calculate bar width as % */
|
|
maxDuration: number
|
|
}
|
|
|
|
export function ExplainNodeRow({ node, depth, maxDuration }: ExplainNodeRowProps) {
|
|
const [isExpanded, setIsExpanded] = useState(false)
|
|
const hasChildren = node.children.length > 0
|
|
const hasDetails = Boolean(node.details?.trim())
|
|
const canExpand = hasDetails
|
|
|
|
const detailLines = parseDetailLines(node.details)
|
|
const indentPx = depth * 24
|
|
|
|
// Calculate duration and bar width as % of max duration
|
|
const duration = node.actualTime ? node.actualTime.end - node.actualTime.start : 0
|
|
const hasTimingData = node.actualTime && duration > 0
|
|
const barWidthPercent = maxDuration > 0 ? (duration / maxDuration) * 100 : 0
|
|
const barColorClass = getScanBarColor(node.operation)
|
|
const borderColorClass = getScanBorderColor(node.operation)
|
|
|
|
return (
|
|
<>
|
|
{/* Wrapper for group hover */}
|
|
<div className="group">
|
|
{/* Main row */}
|
|
<div
|
|
className={cn(
|
|
'flex items-stretch border-l-4 transition-colors bg-studio group-hover:bg-surface-100/50',
|
|
borderColorClass
|
|
)}
|
|
>
|
|
{/* Left section: expand button + operation info */}
|
|
<div
|
|
className="flex items-center gap-3 px-4 py-3 shrink-0 min-w-[400px]"
|
|
style={{ paddingLeft: `${16 + indentPx}px` }}
|
|
>
|
|
{/* Expand/collapse button */}
|
|
<button
|
|
type="button"
|
|
onClick={() => canExpand && setIsExpanded(!isExpanded)}
|
|
disabled={!canExpand}
|
|
className={cn(
|
|
'flex items-center justify-center w-5 h-5 rounded-sm border border-border-muted shrink-0',
|
|
canExpand ? 'hover:bg-surface-200 cursor-pointer' : 'opacity-30 cursor-default'
|
|
)}
|
|
aria-label={isExpanded ? 'Collapse details' : 'Expand details'}
|
|
>
|
|
{isExpanded ? (
|
|
<ChevronDown size={12} className="text-foreground-light" />
|
|
) : (
|
|
<ChevronRight size={12} className="text-foreground-light" />
|
|
)}
|
|
</button>
|
|
|
|
{/* Operation name and cost info */}
|
|
<div className="flex items-center gap-2 font-mono text-xs min-w-0">
|
|
<span className="text-foreground uppercase font-medium whitespace-nowrap">
|
|
{node.operation}
|
|
</span>
|
|
<span className="text-foreground-muted whitespace-nowrap">
|
|
(cost {node.cost?.end?.toFixed(1) ?? '-'}, estimated{' '}
|
|
{node.rows?.toLocaleString() ?? '?'} {node.rows === 1 ? 'row' : 'rows'})
|
|
</span>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Right section: duration bar visualization */}
|
|
<div className="flex-1 relative min-h-[43px] flex items-center">
|
|
{hasTimingData && (
|
|
<>
|
|
{/* Duration bar - width represents % of slowest operation */}
|
|
<div
|
|
className={cn('absolute left-0 top-0 h-full', barColorClass)}
|
|
style={{ width: `${barWidthPercent}%` }}
|
|
/>
|
|
{/* Duration and row count info */}
|
|
<div className="relative flex items-center gap-2 font-mono text-xs whitespace-nowrap px-3">
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<span className="text-foreground-light cursor-help">
|
|
{formatNodeDuration(duration)}
|
|
</span>
|
|
</TooltipTrigger>
|
|
<TooltipContent side="top" className="max-w-xs font-sans">
|
|
<p className="font-medium">Execution time: {formatNodeDuration(duration)}</p>
|
|
<p className="text-foreground-lighter text-xs mt-1">
|
|
This is how long this operation took to execute. The bar width shows this as
|
|
a percentage of the slowest operation ({Math.round(barWidthPercent)}%) —
|
|
wider bars indicate where more time is spent.
|
|
</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
<span className="text-foreground-muted">/</span>
|
|
<RowCountIndicator
|
|
actualRows={node.actualRows}
|
|
estimatedRows={node.rows}
|
|
rowsRemovedByFilter={node.rowsRemovedByFilter}
|
|
/>
|
|
</div>
|
|
</>
|
|
)}
|
|
</div>
|
|
</div>
|
|
|
|
{/* Expanded details section */}
|
|
{isExpanded && detailLines.length > 0 && (
|
|
<div
|
|
className={cn(
|
|
'border-t-border-muted border-t border-l-4 bg-studio group-hover:bg-surface-100/50',
|
|
borderColorClass
|
|
)}
|
|
style={{ paddingLeft: `${16 + indentPx + 32}px` }}
|
|
>
|
|
<div className="px-0 py-3 space-y-2 font-mono text-xs">
|
|
{detailLines.map((line, idx) => (
|
|
<div key={idx} className="flex items-start gap-1">
|
|
{line.label && <span className="text-foreground-muted">{line.label}</span>}
|
|
<span className="text-foreground-light break-all">{line.value}</span>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{/* Render children recursively */}
|
|
{hasChildren &&
|
|
node.children.map((child, idx) => (
|
|
<ExplainNodeRow key={idx} node={child} depth={depth + 1} maxDuration={maxDuration} />
|
|
))}
|
|
</>
|
|
)
|
|
}
|