import React, { useEffect, useState, useRef } from 'react' import FlipMove from 'react-flip-move'; import Chart from 'chart.js/auto' import FunnelTooltip from './funnel-tooltip.js' import ChartDataLabels from 'chartjs-plugin-datalabels' import numberFormatter from '../../util/number-formatter' import Bar from '../bar' import RocketIcon from '../modals/rocket-icon' import * as api from '../../api' import LazyLoader from '../../components/lazy-loader' export default function Funnel(props) { const [loading, setLoading] = useState(true) const [visible, setVisible] = useState(false) const [error, setError] = useState(undefined) const [funnel, setFunnel] = useState(null) const [isSmallScreen, setSmallScreen] = useState(false) const chartRef = useRef(null) const canvasRef = useRef(null) useEffect(() => { if (visible) { setLoading(true) fetchFunnel() .then((res) => { setFunnel(res) setError(undefined) }) .catch((error) => { setError(error) }) .finally(() => { setLoading(false) }) return () => { if (chartRef.current) { chartRef.current.destroy() } } } }, [props.query, props.funnelName, visible, isSmallScreen]) useEffect(() => { if (canvasRef.current && funnel && visible && !isSmallScreen) { initialiseChart() } }, [funnel, visible]) useEffect(() => { const mediaQuery = window.matchMedia('(max-width: 600px)') setSmallScreen(mediaQuery.matches) const handleScreenChange = (e) => { setSmallScreen(e.matches); } mediaQuery.addEventListener("change", handleScreenChange); return () => { mediaQuery.removeEventListener("change", handleScreenChange) } }, []) const isDarkMode = () => { return document.querySelector('html').classList.contains('dark') || false } const getPalette = () => { if (isDarkMode()) { return { dataLabelBackground: 'rgba(25, 30, 56, 0.97)', dataLabelTextColor: 'rgb(243, 244, 246)', visitorsBackground: 'rgb(99, 102, 241)', dropoffBackground: '#2F3949', dropoffStripes: 'rgb(25, 30, 56)', stepNameLegendColor: 'rgb(228, 228, 231)', visitorsLegendClass: 'bg-indigo-500', dropoffLegendClass: 'bg-gray-600', smallBarClass: 'bg-indigo-500' } } else { return { dataLabelBackground: 'rgba(25, 30, 56, 0.97)', dataLabelTextColor: 'rgb(243, 244, 246)', visitorsBackground: 'rgb(99, 102, 241)', dropoffBackground: 'rgb(224, 231, 255)', dropoffStripes: 'rgb(255, 255, 255)', stepNameLegendColor: 'rgb(12, 24, 39)', visitorsLegendClass: 'bg-indigo-500', dropoffLegendClass: 'bg-indigo-100', smallBarClass: 'bg-indigo-300' } } } const formatDataLabel = (visitors, ctx) => { if (ctx.dataset.label === 'Visitors') { const conversionRate = funnel.steps[ctx.dataIndex].conversion_rate return `${formatPercentage(conversionRate)}% \n(${numberFormatter(visitors)} Visitors)` } else { return null } } const calcOffset = (ctx) => { const conversionRate = parseFloat(funnel.steps[ctx.dataIndex].conversion_rate) if (conversionRate > 90) { return -64 } else if (conversionRate > 20) { return -28 } else { return 8 } } const getFunnel = () => { return props.site.funnels.find((funnel) => funnel.name === props.funnelName) } const fetchFunnel = async () => { const funnelMeta = getFunnel() if (typeof funnelMeta === 'undefined') { throw new Error('Could not fetch the funnel. Perhaps it was deleted?') } else { return api.get(`/api/stats/${encodeURIComponent(props.site.domain)}/funnels/${funnelMeta.id}`, props.query) } } const initialiseChart = () => { if (chartRef.current) { chartRef.current.destroy() } const palette = getPalette() const createDiagonalPattern = (color1, color2) => { // create a 10x10 px canvas for the pattern's base shape let shape = document.createElement('canvas') shape.width = 10 shape.height = 10 let c = shape.getContext('2d') c.fillStyle = color1 c.strokeStyle = color2 c.fillRect(0, 0, shape.width, shape.height); c.beginPath() c.moveTo(2, 0) c.lineTo(10, 8) c.stroke() c.beginPath() c.moveTo(0, 8) c.lineTo(2, 10) c.stroke() return c.createPattern(shape, 'repeat') } const labels = funnel.steps.map((step) => step.label) const stepData = funnel.steps.map((step) => step.visitors) const dropOffData = funnel.steps.map((step) => step.dropoff) const ctx = canvasRef.current.getContext("2d") // passing those verbatim to make sure canvas rendering picks them up var fontFamily = 'ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji"' var gradient = ctx.createLinearGradient(900, 0, 900, 900); gradient.addColorStop(1, palette.dropoffBackground); gradient.addColorStop(0, palette.visitorsBackground); const data = { labels: labels, datasets: [ { label: 'Visitors', data: stepData, backgroundColor: gradient, hoverBackgroundColor: gradient, borderRadius: 4, stack: 'Stack 0', }, { label: 'Dropoff', data: dropOffData, backgroundColor: createDiagonalPattern(palette.dropoffBackground, palette.dropoffStripes), hoverBackgroundColor: palette.dropoffBackground, borderRadius: 4, stack: 'Stack 0', }, ], } const config = { plugins: [ChartDataLabels], type: 'bar', data: data, options: { responsive: true, barThickness: 120, plugins: { legend: { display: false, }, tooltip: { enabled: false, mode: 'index', intersect: true, position: 'average', external: FunnelTooltip(palette, funnel) }, datalabels: { formatter: formatDataLabel, anchor: 'end', align: 'end', offset: calcOffset, backgroundColor: palette.dataLabelBackground, color: palette.dataLabelTextColor, borderRadius: 4, clip: true, font: { size: 12, weight: 'normal', lineHeight: 1.6, family: fontFamily }, textAlign: 'center', padding: { top: 8, bottom: 8, right: 8, left: 8 }, }, }, scales: { y: { display: false }, x: { position: 'bottom', display: true, border: { display: false }, grid: { drawBorder: false, display: false }, ticks: { padding: 8, font: { weight: 'bold', family: fontFamily, size: 14 }, color: palette.stepNameLegendColor }, }, }, }, } chartRef.current = new Chart(ctx, config) } const header = () => { return (
{funnel.steps.length}-step funnel • {conversionRate}% conversion rate
{isSmallScreen &&