先看实现的效果:
主要功能点:支持传入分组的组数,支持传入不同的颜色色条,可拖动左侧滑块改变占比,可通过输入右侧input框实现更改占比,支持百分比和数值两种显示模式
直接上代码:注释写的很详细了:
import React, { useEffect, useState } from 'react';
import styles from './index.less';
import { CaretLeftOutlined } from '@ant-design/icons';
import { InputNumber } from 'antd';
import Slider from 'rc-slider';
import 'rc-slider/assets/index.css';
import { bignumberFc } from '@/utils/bigNumber';
type colorBlockProps = {
groupCount: number; //分的组数
colorGroupArrayHex: string[]; //颜色色块的固定数组
type: 'percent' | 'value'; //百分比或数值的显示模式
min?: number; //数值显示模式的最小值
max?: number; //数值显示模式的最大值
currentColorBlockInfo?: {
//保存的当前的组件的信息,还原时候需要
sliderValues: number[];
numberValues: number[];
groupRangeArrs: number[][];
};
// 最大值最小值变化的flag,由于我组件外层会有多个地方触发最大最小值的变化,防止重复渲染,所以传入一个flag去监听,而不是监听min和max
minmaxChangeFlag?: string;
};
const ColorBlock: React.FC<colorBlockProps> = (props: colorBlockProps) => {
const { groupCount, colorGroupArrayHex, type, min, max, currentColorBlockInfo, minmaxChangeFlag } = props;
// 滑动条组件的values,实现是利用从从0到1,表示百分比,同时也是百分比模式下input框的显示值
const [sliderValues, setSliderValues] = useState<number[]>([]);
// 数值类型的数组,它的值与sliderValues与min max 是一一对应的,是数值模式下input框的显示值
const [numberValues, setNumberValues] = useState<number[]>([]);
// input输入框的层级数组,聚焦的时候把当前的input框的z值调大,防止被遮挡
const [zIndexArray, setZIndexArray] = useState<number[]>([]);
// 拖动滑块的响应函数
const handleSliderChange = (values: number[] | number) => {
let valueArray = values as number[];
if (valueArray.includes(0) && valueArray.includes(1)) {
setSliderValues(valueArray);
}
};
// 更改inputNumber延迟生效定时器
let timer: any = null;
// input输入框手动输入的响应(百分比显示模式)
const handlePercentInputChange = (index: number, value: any) => {
const newSliderValues = [...sliderValues];
if (typeof value === 'number') {
// 处理成整数并且取消百分比模式
const newValue = Math.round(value) / 100;
if (newValue < newSliderValues[index + 1] && newValue > newSliderValues[index - 1]) {
newSliderValues[index] = newValue;
}
if (timer) clearTimeout(timer);
timer = setTimeout(() => {
setSliderValues(newSliderValues);
}, 500);
}
};
// input输入框手动输入的响应(数值显示模式)
const handleValueInputChange = (index: number, value: any) => {
const newSliderValues = [...sliderValues];
const newNumberValues = [...numberValues];
if (typeof value === 'number' && max !== undefined && min !== undefined) {
// 首先判断输入的范围是否在相邻两个值的步长范围内
if (
value < newNumberValues[index + 1] - (max - min) / 100 &&
value > newNumberValues[index - 1] + (max - min) / 100
) {
// 算出新的占比
newSliderValues[index] = (value - min) / (max - min);
}
if (timer) clearTimeout(timer);
timer = setTimeout(() => {
setSliderValues(newSliderValues);
}, 500);
}
};
// 聚焦的input输入框永远在最上方,防止被遮盖看不到信息
const handleInputOnFocus = (index: number) => {
let newZIndexArray = [...zIndexArray];
for (let i: number = 0; i < newZIndexArray.length; i++) {
if (i === index) {
newZIndexArray[i] = 9999;
} else newZIndexArray[i] = 1000;
}
setZIndexArray(newZIndexArray);
};
// 初始化函数,若最大值最值,或分组的组数,或显示模式变化,则重新初始化
useEffect(() => {
const initSliderValues: number[] = [];
const initZIndexArray: number[] = [];
for (let i: number = 0; i < groupCount + 1; i++) {
initSliderValues.push(Math.round((1 / groupCount) * i * 100) / 100);
initZIndexArray.push(1000);
}
setSliderValues(initSliderValues);
setZIndexArray(initZIndexArray);
}, [minmaxChangeFlag, groupCount, type]);
// 因数值类型的数组,它的值与sliderValues与min max 是一一对应的,所以sliderValue变化的时候同时更新数值型的数组
useEffect(() => {
if (sliderValues && min !== undefined && max !== undefined) {
const newNumberValues: number[] = [];
for (let i: number = 0; i < groupCount + 1; i++) {
newNumberValues.push(min + (max - min) * sliderValues[i]);
}
setNumberValues(newNumberValues);
}
}, [sliderValues]);
// 还原数据
useEffect(() => {
if (currentColorBlockInfo) {
setSliderValues(currentColorBlockInfo.sliderValues);
setNumberValues(currentColorBlockInfo.numberValues);
}
}, []);
// 组件的关键值的变化,将关键值 交给上层进行存储或者变更处理
useEffect(() => {
const groupRangeArrs: number[][] = [];
const newSliderValues = [...sliderValues];
const newNumberValues = [...numberValues];
// 百分比显示模式
if (type !== 'percent' && newSliderValues.length > 2) {
for (let i: number = 0; i < newSliderValues.length - 1; i++)
groupRangeArrs.push([newSliderValues[i], newSliderValues[i + 1]]);
}
// 数值显示模式
else if (type === 'percent' && newNumberValues.length > 2) {
for (let i: number = 0; i < newNumberValues.length - 1; i++)
groupRangeArrs.push([newNumberValues[i], newNumberValues[i + 1]]);
}
// 下面注释掉的为我自己的更新currentColorBlockInfo 的信息的方法,是交给上层进行处理,可根据实际情况自己去实现属性的存储,只需存储这三个属性即可
// {
// sliderValues: sliderValues,
// numberValues: numberValues,
// groupRangeArrs: groupRangeArrs,
// },
// setAttr(
// {
// colorBlockSlider: {
// sliderValues: sliderValues,
// numberValues: numberValues,
// groupRangeArrs: groupRangeArrs,
// // timeFlag:moment().format('YYYYMMDDHHmmss')
// },
// },
// 'setOtherAttr',
// );
}, [sliderValues, numberValues]);
return (
<div className={styles.container}>
<div>
percent
<Slider
range
step={0.01}
min={0}
max={1}
count={groupCount}
value={sliderValues}
pushable={0.01}
trackStyle={colorGroupArrayHex.map((item: string) => {
return { width: 30, backgroundColor: item as string };
})}
vertical
reverse
allowCross={true}
onChange={handleSliderChange}
/>
<div>
{sliderValues.map((item: number, index: number) => {
return type !== 'percent' ? (
// 百分比显示模式
<div
key={item}
style={{
border: '0px solid #000',
marginLeft: 30,
marginTop: item === 0 ? -11 : 0,
height:
sliderValues[index] === 1 ? '1%' : `${(sliderValues[index + 1] - sliderValues[index]) * 100}%`,
}}
>
<CaretLeftOutlined />
<InputNumber
disabled={item === 0 || item === 1}
size="small"
value={item * 100}
className={zIndexArray[index] === 9999 ? 'onFocusInput' : ''}
precision={0}
step={1}
onChange={(value) => {
handlePercentInputChange(index, value);
}}
onFocus={() => {
handleInputOnFocus(index);
}}
></InputNumber>
<span>%</span>
</div>
) : (
// 数值显示模式
<div
key={item}
style={{
border: '0px solid #000',
marginLeft: 30,
marginTop: item === 0 ? -11 : 0,
height:
sliderValues[index] === 1 ? '1%' : `${(sliderValues[index + 1] - sliderValues[index]) * 100}%`,
}}
>
<CaretLeftOutlined />
<InputNumber
// addonAfter={<span>%</span>}
disabled={item === 0 || item === 1}
size="small"
style={{ width: 150 }}
value={numberValues[index]}
className={zIndexArray[index] === 9999 ? 'onFocusInput' : ''}
precision={
numberValues.length > 1
? bignumberFc
.divide(bignumberFc.subtract(numberValues[numberValues.length - 1], numberValues[0]), 100)
.toString()
.split('.')[1]
? bignumberFc
.divide(bignumberFc.subtract(numberValues[numberValues.length - 1], numberValues[0]), 100)
.toString()
.split('.')[1].length
: 0
: 0
}
step={
numberValues.length > 1
? bignumberFc.divide(
bignumberFc.subtract(numberValues[numberValues.length - 1], numberValues[0]),
100,
)
: 1
}
onChange={(value) => {
handleValueInputChange(index, value);
}}
onFocus={() => {
handleInputOnFocus(index);
}}
></InputNumber>
</div>
);
})}
</div>
</div>
</div>
);
};
export default ColorBlock;
ps:bigNumberFc是基于bignumber.js封装的更高精度的计算方法,也贴在下面:
import BigNumber from 'bignumber.js';
// 高精度计算的处理函数集合
const bignumberFc = {
add: (a: number, b: number) => {
return new BigNumber(a).plus(b).toNumber();
},
subtract: (a: number, b: number) => {
return new BigNumber(a).minus(b).toNumber();
},
multiply: (a: number, b: number) => {
return new BigNumber(a).multipliedBy(b).toNumber();
},
divide: (a: number, b: number) => {
return new BigNumber(a).dividedBy(b).toNumber();
},
// 均值
avg: (arr: number[]) => {
let total: number = 0;
for (let i: number = 0; i < arr.length; i++) {
total = bignumberFc.add(total, arr[i]);
}
return bignumberFc.divide(total, arr.length);
},
// 方差
variance: (arr: number[]) => {
let total: number = 0;
for (let i: number = 0; i < arr.length; i++) {
total = bignumberFc.add(total, arr[i]);
}
const mean = bignumberFc.divide(total, arr.length);
let totalS: number = 0;
for (let i: number = 0; i < arr.length; i++) {
const tempEverySubtract: number = bignumberFc.subtract(arr[i], mean);
totalS = bignumberFc.add(totalS, bignumberFc.multiply(tempEverySubtract, tempEverySubtract));
}
return bignumberFc.divide(totalS, arr.length);
},
// 线性回归
regression: (list: any[]) => {
if (!list) return '';
if (list.length === 0) return '';
// x,y总数
let totalX: number = 0,
totalY: number = 0;
// x,y平均数
let avgX: number = 0,
avgY: number = 0;
// 数组长度
let n: number = list?.length;
// xy阶乘
let xy2: number = 0;
// x平方阶乘
let x2: number = 0;
// 斜率b,常量a
let b: number = 0,
a: number = 0;
// 分子、分母
let fz: number = 0,
fm: number = 0;
// 回归线
let fn: string = '';
for (let i = 0; i < n; i++) {
totalX += list[i][0];
totalY += list[i][1];
xy2 += bignumberFc.multiply(list[i][0], list[i][1]);
x2 += bignumberFc.multiply(list[i][0], list[i][0]);
}
// 均值
avgX = bignumberFc.divide(totalX, n);
avgY = bignumberFc.divide(totalY, n);
// 分子分母
let nxy = bignumberFc.multiply(n, bignumberFc.multiply(avgX, avgY));
fz = bignumberFc.subtract(xy2, nxy);
let nx = bignumberFc.multiply(n, bignumberFc.multiply(avgX, avgX));
fm = bignumberFc.subtract(x2, nx);
b = bignumberFc.divide(fz, fm);
a = bignumberFc.subtract(list[0][1], bignumberFc.multiply(b, list[0][0]));
fn = a < 0 ? 'y=' + (b == 0 ? '' : b + 'x') + a : 'y=' + (b == 0 ? '' : b + 'x+') + a;
return fn;
// console.log(avgX,'平均数x');
// console.log(avgY,'平均数y');
// console.log(nxy,'nxy');
// console.log(nx,'nx');
// console.log(fz,'分子');
// console.log(x2,'x2');
// console.log(xy2,'xy2');
// console.log(fm,'分母');
// console.log(fn,'线性回归方程');
},
};
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。