UIPackage

Graph Chart

Vue chart
Edit on GitHub

Network / graph chart wrapper around Apache ECharts. Force-directed, circular, or manual layouts; directed or undirected edges; optional category coloring. Useful for service maps, social graphs, knowledge bases.

Installation

$ npx shadcn-vue@latest add https://uipkge.dev/r/vue/graph-chart.json

Or with the named registry: npx shadcn-vue@latest add @uipkge/graph-chart

Examples

Props

Name Type / Values Default Required
nodes GraphNode[] required
links GraphLink[] required
categories

Optional category labels rendered in the legend.

string[] optional
layout

on a ring, `none` lets you place nodes manually via `x`/`y`.

'force''circular''none'
'force' optional
roam

Allow click-and-drag pan + scroll zoom. Default false.

boolean false optional
directed

Draw arrowheads on the target end. Default true.

boolean true optional
height number | string 380 optional
option any optional
class string optional

Schema

Type aliases from this item's source — use them to shape the data you pass in.

GraphNode
interface GraphNode {
  name: string
  /** Optional category index (paints with chart-N colour). */
  category?: number
  /** Optional fixed marker size. Defaults to 28. */
  symbolSize?: number
}
GraphLink
interface GraphLink {
  source: string
  target: string
  /** Optional edge value (shows up in the tooltip + sizes the line on weighted layouts). */
  value?: number
}

Dependencies

Used by

Files (3)

  • app/components/ui/charts/graph-chart/GraphChart.vue 3.4 kB
    <script setup lang="ts">
    import { computed } from 'vue'
    import { use } from 'echarts/core'
    import { CanvasRenderer } from 'echarts/renderers'
    import { GraphChart as EChartsGraphChart } from 'echarts/charts'
    import { TooltipComponent, LegendComponent } from 'echarts/components'
    import VChart from 'vue-echarts'
    import { cn } from '@/lib/utils'
    import {
      chartColors,
      chartAxisColor,
      chartTextColor,
      chartTooltipBg,
      chartTooltipBorder,
      chartTooltipText,
    } from '../useChartTheme'
    
    use([CanvasRenderer, EChartsGraphChart, TooltipComponent, LegendComponent])
    
    interface GraphNode {
      name: string
      /** Optional category index (paints with chart-N colour). */
      category?: number
      /** Optional fixed marker size. Defaults to 28. */
      symbolSize?: number
    }
    interface GraphLink {
      source: string
      target: string
      /** Optional edge value (shows up in the tooltip + sizes the line on weighted layouts). */
      value?: number
    }
    
    interface Props {
      nodes: GraphNode[]
      links: GraphLink[]
      /** Optional category labels rendered in the legend. */
      categories?: string[]
      /** Layout engine. `force` is force-directed (default), `circular` arranges
       *  on a ring, `none` lets you place nodes manually via `x`/`y`. */
      layout?: 'force' | 'circular' | 'none'
      /** Allow click-and-drag pan + scroll zoom. Default false. */
      roam?: boolean
      /** Draw arrowheads on the target end. Default true. */
      directed?: boolean
      height?: number | string
      option?: any
      class?: string
    }
    
    const props = withDefaults(defineProps<Props>(), {
      layout: 'force',
      roam: false,
      directed: true,
      height: 380,
    })
    
    const mergedOption = computed(() => {
      const series = [
        {
          type: 'graph',
          layout: props.layout,
          roam: props.roam,
          symbolSize: 28,
          label: { show: true, fontSize: 11, color: chartTextColor.value },
          edgeSymbol: props.directed ? (['none', 'arrow'] as [string, string]) : (['none', 'none'] as [string, string]),
          edgeSymbolSize: [0, 6],
          force: { repulsion: 220, edgeLength: 90 },
          lineStyle: { color: chartAxisColor.value, curveness: 0.15, width: 1 },
          emphasis: { focus: 'adjacency' as const, lineStyle: { width: 2 } },
          categories: props.categories?.map((name) => ({ name })),
          data: props.nodes.map((n) => ({
            ...n,
            itemStyle:
              typeof n.category === 'number'
                ? { color: chartColors.value[n.category % chartColors.value.length] }
                : undefined,
          })),
          links: props.links,
        },
      ]
    
      const userOption: any = props.option ?? {}
      const { series: userSeries, ...userRest } = userOption
      const mergedSeries = Array.isArray(userSeries) ? series.map((s, i) => ({ ...s, ...(userSeries[i] ?? {}) })) : series
    
      return {
        color: chartColors.value,
        tooltip: {
          trigger: 'item',
          backgroundColor: chartTooltipBg.value,
          borderColor: chartTooltipBorder.value,
          textStyle: { color: chartTooltipText.value, fontSize: 12 },
        },
        legend: props.categories?.length
          ? {
              bottom: 0,
              icon: 'circle',
              itemWidth: 8,
              itemHeight: 8,
              textStyle: { fontSize: 11, color: chartTextColor.value },
            }
          : undefined,
        series: mergedSeries,
        ...userRest,
      }
    })
    </script>
    
    <template>
      <div
        :style="{ height: /^\d+$/.test(String(height)) ? `${height}px` : String(height) }"
        :class="cn('w-full', props.class)"
      >
        <VChart :option="mergedOption" :autoresize="true" class="size-full" />
      </div>
    </template>
  • app/components/ui/charts/graph-chart/index.ts 0.1 kB
    export { default as GraphChart } from './GraphChart.vue'
  • app/components/ui/charts/useChartTheme.ts 6.4 kB
    import { computed, ref, type ComputedRef } from 'vue'
    
    // Chart palette is driven by Tailwind v4 CSS variables (`--chart-1`..`--chart-5`,
    // `--muted-foreground`, `--border`, `--popover`, etc.) so dark/light flips
    // happen automatically when the consumer toggles their theme class. The
    // values resolve at runtime via `getComputedStyle`, so they pick up whatever
    // the consumer set in their own `tailwind.css` -- no fork required.
    //
    // We bump `themeKey` whenever `<html>` class/style changes (the typical
    // shadcn dark-mode pivot) so every consuming `computed` re-resolves and
    // downstream ECharts options re-paint.
    
    const themeKey = ref(0)
    
    if (typeof window !== 'undefined') {
      // Bump once on the first paint so post-hydration getComputedStyle reads
      // the *resolved* CSS values (during SSR-built bundles the very first
      // computed pass returns the fallbacks below).
      requestAnimationFrame(() => themeKey.value++)
      new MutationObserver(() => themeKey.value++).observe(document.documentElement, {
        attributes: true,
        attributeFilter: ['class', 'style', 'data-theme'],
      })
    }
    
    // Lazy canvas context used to normalize any CSS color string (including
    // `oklch(...)`, `oklab(...)`, `color(display-p3 ...)`) into a hex / rgba
    // string ECharts' canvas renderer can consume. Without this, code that
    // does `color + '40'` (8-digit hex alpha trick) produces invalid color
    // strings like `oklch(...)40` and the canvas API throws.
    let _hexCanvas: CanvasRenderingContext2D | null = null
    function toHex(cssColor: string): string {
      if (typeof document === 'undefined') return cssColor
      if (!_hexCanvas) {
        _hexCanvas = document.createElement('canvas').getContext('2d')
      }
      if (!_hexCanvas) return cssColor
      // Reset, then assign; the browser normalizes whatever it accepted into
      // the canonical hex/rgba form when read back.
      _hexCanvas.fillStyle = '#000'
      _hexCanvas.fillStyle = cssColor
      return _hexCanvas.fillStyle as string
    }
    
    // Convert any CSS color (hex, rgb, oklch, color()) + alpha 0..1 to a
    // canvas-safe rgba(r,g,b,a). `colorString + '40'` (8-digit hex alpha)
    // only works when `colorString` is `#rrggbb`; once tokens resolve to
    // oklch() post-hydration the gradient stops break and the canvas paint
    // throws every frame. Stay defensive and always return rgba.
    export function toRgba(cssColor: string, alpha: number): string {
      if (typeof document === 'undefined') return cssColor
      if (!_hexCanvas) {
        _hexCanvas = document.createElement('canvas').getContext('2d')
      }
      if (!_hexCanvas) return cssColor
      _hexCanvas.fillStyle = '#000'
      _hexCanvas.fillStyle = cssColor
      const normalized = _hexCanvas.fillStyle as string
      if (normalized.startsWith('#') && normalized.length === 7) {
        const r = parseInt(normalized.slice(1, 3), 16)
        const g = parseInt(normalized.slice(3, 5), 16)
        const b = parseInt(normalized.slice(5, 7), 16)
        return `rgba(${r},${g},${b},${alpha})`
      }
      if (normalized.startsWith('rgba(')) {
        return normalized.replace(/,\s*[\d.]+\s*\)$/, `,${alpha})`)
      }
      if (normalized.startsWith('rgb(')) {
        return normalized.replace(/^rgb\(/, 'rgba(').replace(/\)$/, `,${alpha})`)
      }
      // Canvas refused to parse this color -- ship the original string and
      // let ECharts complain (better than crashing the paint loop).
      return cssColor
    }
    
    function resolveVar(name: string, fallback: string): string {
      if (typeof window === 'undefined') return fallback
      const v = getComputedStyle(document.documentElement).getPropertyValue(name).trim()
      if (!v) return fallback
      return toHex(v)
    }
    
    // SSR / pre-hydration fallback palette. Hex values picked to roughly
    // match the shadcn Neutral defaults in `tailwind.css` so the first paint
    // doesn't flicker.
    const CHART_FALLBACK = ['#f59e0b', '#14b8a6', '#3b82f6', '#f97316', '#eab308']
    
    export const chartColors: ComputedRef<string[]> = computed(() => {
      themeKey.value
      return Array.from({ length: 5 }, (_, i) => resolveVar(`--chart-${i + 1}`, CHART_FALLBACK[i]!))
    })
    
    export const chartTextColor: ComputedRef<string> = computed(() => {
      themeKey.value
      return resolveVar('--muted-foreground', '#888888')
    })
    
    export const chartAxisColor: ComputedRef<string> = computed(() => {
      themeKey.value
      return resolveVar('--border', '#e5e5e5')
    })
    
    export const chartSplitLineColor: ComputedRef<string> = computed(() => {
      themeKey.value
      return resolveVar('--border', '#f0f0f0')
    })
    
    export const chartTooltipBg: ComputedRef<string> = computed(() => {
      themeKey.value
      return resolveVar('--popover', 'rgba(255,255,255,0.96)')
    })
    
    export const chartTooltipBorder: ComputedRef<string> = computed(() => {
      themeKey.value
      return resolveVar('--border', '#e5e5e5')
    })
    
    export const chartTooltipText: ComputedRef<string> = computed(() => {
      themeKey.value
      return resolveVar('--popover-foreground', '#333333')
    })
    
    // Two-level deep merge for ECharts option blocks (xAxis, yAxis, grid,
    // tooltip, legend, singleAxis, parallel, etc.). The top-level keys merge
    // shallowly, but one nested level (axisLabel, axisLine, splitLine, etc.)
    // merges shallowly too so a consumer passing `xAxis: { axisLabel: { fontSize: 9 } }`
    // doesn't blow away the wrapper's `color` + base font defaults on the same
    // axisLabel block. Arrays + primitives replace outright.
    //
    // This is the merge strategy the chart wrappers use to fold `props.option`
    // onto their computed base option without forcing consumers to spell out
    // every default they want to preserve.
    export function mergeOptionBlock<T extends Record<string, any>>(base: T, user: Partial<T> | undefined): T {
      if (!user) return base
      const out: any = { ...base }
      for (const k of Object.keys(user)) {
        const bv = (base as any)[k]
        const uv = (user as any)[k]
        if (
          bv != null &&
          uv != null &&
          typeof bv === 'object' &&
          typeof uv === 'object' &&
          !Array.isArray(bv) &&
          !Array.isArray(uv)
        ) {
          out[k] = { ...bv, ...uv }
        } else {
          out[k] = uv
        }
      }
      return out
    }
    
    // Default gauge stoplight: teal (safe) -> amber (warning) -> red (danger).
    // Pulled off saturated green and onto teal so the gauge ties back to the
    // dashboard palette; red is kept as the universal "limit reached" cue.
    // GaugeChart consumes this via its `thresholds` prop default; consumers
    // pass their own array to override. Static because gauges have semantic
    // meaning (green safe / red danger) that we deliberately don't theme-flip.
    export const gaugeThresholds: [number, string][] = [
      [0.6, '#14b8a6'],
      [0.85, '#f59e0b'],
      [1, '#dc2626'],
    ]

Raw manifest: https://uipkge.dev/r/vue/graph-chart.json