【前端功能点】柱状/条形图实现

确定实现的目标

  • 主标题、副标题
  • 分类色块
  • x/y轴(轴、辅助线、分割点、分割点对应的值)
  • 柱状(包含动画)

实现思路

  1. 通过ref获取盒子的宽高、绘制对象ctx
  2. 标题、轴数据通过props获取
  3. 标题、副标题、分类色块根据数据直接绘制即可
  4. 轴分割点、轴辅助线、轴分割点对应的值、柱状需要依赖盒子的宽高
  5. 监听元素尺寸变化,变化则会重新绘制图形

具体实现(只写组件内部逻辑)

  1. 获取宽高、绘制对象
 // 定义保存外层盒子的ref
 const boxRef = React.useRef<HTMLDivElement>(null);
 // 定义保存canvas元素的ref
 const canvasRef = React.useRef<HTMLCanvasElement>(null);
 // 保存对应的值
 React.useEffect(() => {
    if (canvasRef.current && boxRef.current) {
      setConfig({
        w: boxRef.current.offsetWidth,
        h: boxRef.current.offsetHeight,
        ctx: canvasRef.current.getContext('2d'),
      });
    }
 }, []);
 return (
     <div ref={boxRef} style={{ width: '100%', height: '100%' }}>
      <canvas ref={canvasRef} width={w} height={h} />
    </div>
 )
  1. 绘制标题、副标题、色块
// 绘制标题
const drawText = (c, t) => {
    if (t && t.text) {
      c.font = t.font;
      c.fillStyle = t.fillStyle;
      c.textBaseline = t.textBaseline;
      c.fillText(`${t.text}`, t.x, t.y);
    }
};
// 绘制圆角矩形
const drawThumb = React.useCallback((c, d) => {
    if (!c || !Array.isArray(d)) return;
    d.forEach((it) => {
      if (it.thumbnail) {
        const { x, y, w, h, r } = it.thumbnail;
        c.beginPath();
        c.fillStyle = it.bg;
        drawRoundedRect(c, x, y, w, h, r);
        c.fill();
        if (it.text && it.text.text) {
          c.font = it.text.font;
          c.fillStyle = it.text.fillStyle;
          c.fillText(`${it.text.text}`, it.text.x, it.text.y);
        }
      }
    });
}, []);
React.useEffect(() => {
    if (ctx) {
      // 绘制标题
      drawText(ctx, title);
      // 绘制副标题
      drawText(ctx, subTitle);
      // 绘制圆角矩形
      drawThumb(ctx, data);
    }
}, [ctx, w, h]);
  1. 绘制坐标轴以及辅助线和分割点
// 绘制坐标轴分割点
const drawSplitPoint = ({ c, d, btm, left, right, axs }) => {
    const xwt = right - left;
    let len;
    let step;
    let splitPoint;
    let endX;
    let endY;
    let endY2;
    if (!d) {
      // y 轴的兼容
    } else {
      len = d.length;
      step = xwt / len;
      endX = right - 2;
      endY = btm;
      endY2 = btm + 4;
      splitPoint = axs.splitPoint || (axs.x && axs.x.splitPoint);
    }
    drawLine(c, splitPoint);
    for (let i = 0; i <= len; i++) {
      c.beginPath();
      if (i === len) {
        c.moveTo(endX, endY);
        c.lineTo(endX, endY2);
      } else {
        c.fillText(`${d[i]}`, left + (i + 0.25) * step, endY + 10);
        c.moveTo(left + 1 + i * step, endY);
        c.lineTo(left + 1 + i * step, endY2);
      }
      c.stroke();
      c.closePath();
    }
    return { step };
};
// 绘制坐标轴辅助线
const drawAssistLine = ({ c, d, btm, left, right, top, axs }) => {
    const yht = btm - top;
    let len;
    let step;
    let assist;
    let endY;
    let space;
    let valRate;
    if (!d) {
      assist = axs.assist || (axs.y && axs.y.assist);
      space = axs.space || (axs.y && axs.y.space);
      len = Math.ceil(maxVal / space);
      step = yht / len;
      endY = top + 1;
      valRate = step / space;
    } else {
      // x轴的兼容
    }
    drawLine(c, assist);
    for (let i = 0; i <= len; i++) {
      c.beginPath();
      const text = `${i * space}`;
      ctx.fillText(text, left - 10, btm - i * step - 5);
      if (i === len) {
        ctx.moveTo(left, endY);
        ctx.lineTo(right, endY);
      } else if (i !== 0) {
        ctx.moveTo(left, btm - i * step);
        ctx.lineTo(right, btm - i * step);
      }
      c.stroke();
      c.closePath();
    }
    return { valRate };
};
React.useEffect(() => {
    if (ctx) {
      // 绘制标题
      drawText(ctx, title);
      // 绘制副标题
      drawText(ctx, subTitle);
      // 绘制圆角矩形
      drawThumb(ctx, data);
      // 绘制x轴
      drawAxis(ctx, axis, w, h, 'x');
      // 绘制y轴
      const { btm, left, right, top } = drawAxis(ctx, axis, w, h, 'y');
      // 绘制x轴分割点
      const { step: step_x } = drawSplitPoint({
        c: ctx,
        d: xData,
        btm,
        left,
        axs: axis,
        right,
      });
      // 绘制y轴辅助线和分界值
      const { valRate } = drawAssistLine({
        c: ctx,
        d: null,
        btm,
        left,
        axs: axis,
        right,
        top,
      });
    }
}, [ctx, w, h]);
  1. 绘制柱状
 // 绘制柱状
 const drawBar = (step_x, left, btm, valRate) => {
    const zw = (step_x - 6) / data.length;
    let y_h = 0.1;
    const drawZhu = (x, v, y_idx, idx) => {
      const yh = y_h * v;
      ctx.fillStyle = data[y_idx].bg;
      ctx.beginPath();
      drawRoundedRect(ctx, x, btm - yh, zw, yh, 4);
      ctx.fill();
      if (y_h < 1) {
        if (y_idx === data.length - 1 && idx === data[y_idx].values.length - 1) {
          // 刚好走完一轮
          if (y_h > 0.99) {
            y_h = 1;
          } else {
            y_h += 0.1 - y_h / 10;
          }
        }
        requestAnimationFrame(() => drawZhu(x, v, y_idx, idx));
      }
    };
    for (let i = 0; i < data.length; i++) {
      const values = data[i].values;
      for (let j = 0; j < values.length; j++) {
        const value = values[j] * valRate;
        const x = left + 2 + j * step_x + i * (zw + 2);
        drawZhu(x, value, i, j);
      }
    }
};
React.useEffect(() => {
    if (ctx) {
      // 绘制标题
      drawText(ctx, title);
      // 绘制副标题
      drawText(ctx, subTitle);
      // 绘制x轴
      drawAxis(ctx, axis, w, h, 'x');
      // 绘制y轴
      const { btm, left, right, top } = drawAxis(ctx, axis, w, h, 'y');
      // 绘制圆角矩形
      drawThumb(ctx, data);
      // 绘制x轴分割点
      const { step: step_x } = drawSplitPoint({
        c: ctx,
        d: xData,
        btm,
        left,
        axs: axis,
        right,
      });
      // 绘制y轴辅助线和分界值
      const { valRate } = drawAssistLine({
        c: ctx,
        d: null,
        btm,
        left,
        axs: axis,
        right,
        top,
      });
      // 绘制柱状
      drawBar(step_x, left, btm, valRate);
    }
}, [ctx, w, h]);

全部代码

types.ts

export interface ChartBarText {
  font?: string;
  fillStyle?: string;
  // eslint-disable-next-line no-undef
  textBaseline?: CanvasTextBaseline;
  text?: string;
  x?: number;
  y?: number;
}

export interface Line {
  strokeStyle?: string;
  fillStyle?: string;
  lineWidth?: number;
  textAlign?: string;
  lineCap?: string;
  lineDash?: number[];
}

export interface AxisBase {
  strokeStyle?: string;
  assist?: Line;
  splitPoint?: Line;
  space?: number;
}

export interface Axis extends AxisBase {
  x?: AxisBase;
  y?: AxisBase;
  margin?: {
    left?: number;
    right?: number;
    top?: number;
    bottom?: number;
    x?: number;
    y?: number;
  };
}

export interface ChartBarData {
  name: string;
  bg: string;
  values: number[];
  text?: ChartBarText;
  thumbnail?: {
    x?: number;
    y?: number;
    w?: number;
    h?: number;
    r?: number;
  };
}

export interface ChartBarProps {
  data: ChartBarData[];
  xData: (string | number)[];
  axis: Axis;
  title?: ChartBarText;
  subTitle?: ChartBarText;
}

utils.ts

/**
 * @description debounce 防抖函数
 * @param {func} func 需要防抖的函数
 * @param {Number} delay 时间
 */
export const debounce = (func: Func, delay: number = 500) => {
  let timer: any = null;
  return () => {
    if (timer) clearTimeout(timer);
    timer = setTimeout(() => {
      if (typeof func === 'function') func();
      clearTimeout(timer);
    }, delay);
  };
};
/**
 * 绘制圆角
 * @param ctx 绘制对象
 * @param x
 * @param y
 * @param w
 * @param h
 * @param r
 */
export const drawRoundedRect = (ctx, x, y, w, h, r) => {
  if (typeof ctx.roundRect === 'function') {
    ctx.roundRect(x, y, w, h, r);
  } else {
    ctx.beginPath();
    ctx.moveTo(x + r, y);
    ctx.lineTo(x + w - r, y);
    ctx.quadraticCurveTo(x + w, y, x + w, y + r);
    ctx.lineTo(x + w, y + h - r);
    ctx.quadraticCurveTo(x + w, y + h, x + w - r, y + h);
    ctx.lineTo(x + r, y + h);
    ctx.quadraticCurveTo(x, y + h, x, y + h - r);
    ctx.lineTo(x, y + r);
    ctx.quadraticCurveTo(x, y, x + r, y);
    ctx.closePath();
  }
};

hooks.ts

import { type RefObject, useEffect } from 'react';
import { debounce } from '../utils';

/**
 * useListenDomSize 监听节点尺寸变化
 * @param dom 要监听尺寸变化的节点
 * @param callback 尺寸变化时的回调
 */
export const useListenDomSize = (
  elRef: RefObject<HTMLElement>,
  callback?: (p?: any) => void,
  time = 3000,
) => {
  useEffect(() => {
    if (elRef.current && typeof callback === 'function') {
      if (window.ResizeObserver) {
        const cb = debounce(callback, time);
        const domObserver = new ResizeObserver(cb);
        domObserver.observe(elRef.current);
        return () => {
          if (elRef.current) {
            domObserver.unobserve(elRef.current);
            domObserver.disconnect();
          }
        };
      }
    }
    return () => null;
  }, [elRef, callback]);
};

index.tsx

import React from 'react';
import type { ChartBarProps } from './types';
import { useListenDomSize } from './hooks';
import { drawRoundedRect } from './utils';
const ChartBar: React.FunctionComponent<ChartBarProps> = (props) => {
const { xData, data, title, subTitle, axis } = props;
const boxRef = React.useRef<HTMLDivElement>(null);
const canvasRef = React.useRef<HTMLCanvasElement>(null);
const [{ w, h, ctx }, setConfig] = React.useState({ w: null, h: null, ctx: null });
useListenDomSize(
boxRef,
React.useCallback(() => {
if (boxRef.current) {
setConfig((o) => ({
...o,
w: boxRef.current.offsetWidth,
h: boxRef.current.offsetHeight,
}));
}
}, []),
300,
);
React.useEffect(() => {
if (canvasRef.current) {
setConfig((o) => ({
...o,
ctx: canvasRef.current.getContext('2d'),
}));
}
}, []);
// 绘制标题
const drawText = (c, t) => {
if (t && t.text) {
c.font = t.font;
c.fillStyle = t.fillStyle;
c.textBaseline = t.textBaseline;
c.fillText(`${t.text}`, t.x, t.y);
}
};
// 绘制坐标轴
const drawAxis = (c, axs, width, height, t) => {
if (axs && width && height) {
c.beginPath();
const left = axs.margin.x || axs.margin.left;
const right = width - (axs.margin.x || axs.margin.right);
const btm = height - (axs.margin.y || axs.margin.bottom);
let top;
if (t === 'x') {
c.strokeStyle = axs.strokeStyle || axs.x.strokeStyle;
c.moveTo(left, btm);
c.lineTo(right, btm);
} else {
c.strokeStyle = axs.strokeStyle || axs.y.strokeStyle;
top = axs.margin.y || axs.margin.top;
c.moveTo(left, top);
c.lineTo(left, btm);
}
c.stroke();
c.closePath();
return { btm, left, right, top };
}
return {};
};
// 绘制圆角矩形
const drawThumb = React.useCallback((c, d) => {
if (!c || !Array.isArray(d))) return;
d.forEach((it) => {
if (it.thumbnail) {
const { x, y, w, h, r } = it.thumbnail;
c.beginPath();
c.fillStyle = it.bg;
drawRoundedRect(c, x, y, w, h, r);
c.fill();
if (it.text && it.text.text) {
c.font = it.text.font;
c.fillStyle = it.text.fillStyle;
c.fillText(`${it.text.text}`, it.text.x, it.text.y);
}
}
});
}, []);
// 最大值
const maxVal = React.useMemo(() => {
let values = [];
for (let j = 0, len = data.length; j < len; j++) {
values = [...values, ...data[j].values];
}
const v = Math.max(...values);
return v;
}, [data]);
// 绘制线条
const drawLine = (c, o) => {
if (o) {
c.beginPath();
c.strokeStyle = o.strokeStyle;
c.fillStyle = o.fillStyle;
c.lineWidth = o.lineWidth;
c.textAlign = o.textAlign;
c.lineCap = o.lineCap;
if (o.lineDash) {
ctx.setLineDash(o.lineDash);
}
}
};
// 绘制坐标轴分割点
const drawSplitPoint = ({ c, d, btm, left, right, axs }) => {
const xwt = right - left;
let len;
let step;
let splitPoint;
let endX;
let endY;
let endY2;
if (!d) {
// y 轴的兼容
} else {
len = d.length;
step = xwt / len;
endX = right - 2;
endY = btm;
endY2 = btm + 4;
splitPoint = axs.splitPoint || (axs.x && axs.x.splitPoint);
}
drawLine(c, splitPoint);
for (let i = 0; i <= len; i++) {
c.beginPath();
if (i === len) {
c.moveTo(endX, endY);
c.lineTo(endX, endY2);
} else {
c.fillText(`${d[i]}`, left + (i + 0.25) * step, endY + 10);
c.moveTo(left + 1 + i * step, endY);
c.lineTo(left + 1 + i * step, endY2);
}
c.stroke();
c.closePath();
}
return { step };
};
// 绘制坐标轴辅助线
const drawAssistLine = ({ c, d, btm, left, right, top, axs }) => {
const yht = btm - top;
let len;
let step;
let assist;
let endY;
let space;
let valRate;
if (!d) {
assist = axs.assist || (axs.y && axs.y.assist);
space = axs.space || (axs.y && axs.y.space);
len = Math.ceil(maxVal / space);
step = yht / len;
endY = top + 1;
valRate = step / space;
} else {
// x轴的兼容
}
drawLine(c, assist);
for (let i = 0; i <= len; i++) {
c.beginPath();
const text = `${i * space}`;
ctx.fillText(text, left - 10, btm - i * step - 5);
if (i === len) {
ctx.moveTo(left, endY);
ctx.lineTo(right, endY);
} else if (i !== 0) {
ctx.moveTo(left, btm - i * step);
ctx.lineTo(right, btm - i * step);
}
c.stroke();
c.closePath();
}
return { valRate };
};
// 绘制柱状
const drawBar = (step_x, left, btm, valRate) => {
const zw = (step_x - 6) / data.length;
let y_h = 0.1;
const drawZhu = (x, v, y_idx, idx) => {
const yh = y_h * v;
ctx.fillStyle = data[y_idx].bg;
ctx.beginPath();
drawRoundedRect(ctx, x, btm - yh, zw, yh, 4);
ctx.fill();
if (y_h < 1) {
if (y_idx === data.length - 1 && idx === data[y_idx].values.length - 1) {
// 刚好走完一轮
if (y_h > 0.99) {
y_h = 1;
} else {
y_h += 0.1 - y_h / 10;
}
}
requestAnimationFrame(() => drawZhu(x, v, y_idx, idx));
}
};
for (let i = 0; i < data.length; i++) {
const values = data[i].values;
for (let j = 0; j < values.length; j++) {
const value = values[j] * valRate;
const x = left + 2 + j * step_x + i * (zw + 2);
drawZhu(x, value, i, j);
}
}
};
React.useEffect(() => {
if (ctx) {
// 绘制标题
drawText(ctx, title);
// 绘制副标题
drawText(ctx, subTitle);
// 绘制x轴
drawAxis(ctx, axis, w, h, 'x');
// 绘制y轴
const { btm, left, right, top } = drawAxis(ctx, axis, w, h, 'y');
// 绘制圆角矩形
drawThumb(ctx, data);
// 绘制x轴分割点
const { step: step_x } = drawSplitPoint({
c: ctx,
d: xData,
btm,
left,
axs: axis,
right,
});
// 绘制y轴辅助线和分界值
const { valRate } = drawAssistLine({
c: ctx,
d: null,
btm,
left,
axs: axis,
right,
top,
});
// 绘制柱状
drawBar(step_x, left, btm, valRate);
}
}, [ctx, w, h]);
return (
<div ref={boxRef} style={{ width: '100%', height: '100%' }}>
<canvas ref={canvasRef} width={w} height={h} />
</div>
);
};
export default ChartBar;

demo

import { ChartBar } from '@esy-ui';
// x轴的值
const xData = [
'1月',
'2月',
'3月',
'4月',
'5月',
'6月',
'7月',
'8月',
'9月',
'10月',
'11月',
'12月',
];
// 每个分类柱状的高度值
const data = [
{
name: '蒸发量',
bg: 'skyblue',
thumbnail: {
x: 150,
y: 10,
w: 24,
h: 16,
r: 4,
},
text: {
text: '蒸发量',
fillStyle: 'skyblue',
font: '14px Arial',
x: 185,
y: 12,
},
values: [
2.0, 4.9, 7.0, 23.2, 25.6, 76.7, 135.6, 162.2, 32.6, 210, 6.4, 200,
],
},
{
name: '降水量',
bg: 'pink',
thumbnail: {
x: 250,
y: 10,
w: 24,
h: 16,
r: 4,
},
text: {
text: '降水量',
fillStyle: 'pink',
font: '14px Arial',
x: 285,
y: 12,
},
values: [
2.6, 5.9, 9.0, 26.4, 28.7, 70.7, 175.6, 182.2, 48.7, 18.8, 6.0, 250,
],
},
];
// 标题
const title = {
font: '24px bold Arial',
fillStyle: '#222',
textBaseline: 'top',
x: 10,
y: 10,
text: '这是标题',
};
// 副标题
const subTitle = {
font: '14px Arial',
fillStyle: '#ccc',
textBaseline: 'top',
x: 10,
y: 50,
text: '这是副标题',
};
// 坐标轴
const axis = {
x: {
splitPoint: {
strokeStyle: 'black',
fillStyle: 'black',
lineWidth: 1,
},
},
y: {
assist: {
strokeStyle: 'black',
fillStyle: 'black',
lineWidth: 0.2,
textAlign: 'right',
lineCap: 'round',
lineDash: [5, 5],
},
space: 50,
},
strokeStyle: 'blue',
margin: {
x: 50,
top: 120,
bottom: 50,
},
};
export default () => {
return (
<div style={{ width: '100%', height: '32rem' }}>
<ChartBar
xData={xData}
data={data}
title={title}
subTitle={subTitle}
axis={axis}
/>
</div>
);
};

最终效果图(代码执行会有动画)

【前端功能点】柱状/条形图实现

总结

  • 目前功能单一,没有平均值、极值、鼠标以上等效果
  • 主要是提供一种思路,思路确定后实现起来就会简单很多
  • 尺寸发生变化时是通过ResizeObserver实现的,有兼容性问题

原文链接:https://juejin.cn/post/7356652717393215540 作者:洛城赋

(0)
上一篇 2024年4月13日 上午10:48
下一篇 2024年4月13日 上午10:58

相关推荐

发表回复

登录后才能评论