echarts渐变色与css渐变色互转(两个坐标点转角度)

前言

用于 echarts 的小伙伴都知道,他使用的渐变色写法和 css 的写法不一样。css 中直接使用角度定义渐变的方向,而 echarts 使用的是两个坐标点来进行标识方向(线性渐变)。

本文主要针对线性渐变的转换

那怎么在 css 中使用 echarts 的渐变色呢🤔? 也就是需要相互转换,下文将对此进行实现。

最终实现目标如下图演示:
左边是 echarts 柱状图,右边是 div
坐标转换.gif

两种渐变色格式

css 线性渐变使用方法 linear-gradient

<div
     style={{
        backgroundImage: `linear-gradient(90deg, red, blue)`
    }}
></div>

echarts 线性渐变使用方法


// echarts 颜色
const color = {
    type: "linear",  
    x: 0,
    y: 0.5,
    x2: 1,
    y2: 0.5, 
    colorStops: [
        {
                offset: 0,
                color: "red", // 0% 处的颜色
        },
        {
                offset: 1,
                color: "blue", // 100% 处的颜色
        },
    ]
};
        
const options = { 
     // ...
    series: [
        {
            // ...
            itemStyle: {
                 color,
            },
        },
    ],
};

代码目标

从上面可以看出来,其实就是 echarts 用了坐标的形式,而 css 使用的角度。所以针对坐标和角度进行转换即可

实现代码

具体实现见下面代码即可,懒得写了~

import React, { useEffect, useState, useRef } from "react";
// import Test from './views/Test';

/**
 * 线性渐变,前四个参数分别是 x0, y0, x2, y2
 * x y 语义
 *
 * x=0.5 y=0,     x2=0.5, y2=1   从上到下
 * x=1   y=0.5,   x2=0.5,   y2=0   从下到上
 * x=0   y=0.5,   x2=1,   y2=0.5 从左到右
 * x=1   y=0.5,   x2=0,   y2=0.5 从右到左
 * 
 * css 中 0deg 是从上到下,顺时针方向是从左到右渐变(90deg => 左到右)
 *
 * */ 

function App() {
	const pointArr = useRef();
	const myChart = useRef();
	const [deg, setDeg] = useState();

	// echarts 颜色
	const color = {
		type: "linear", 
		// 左到右  90 deg
		x: 0,
		y: 0.5,
		x2: 1,
		y2: 0.5,

		colorStops: [
			{
				offset: 0,
				color: "red", // 0% 处的颜色
			},
			{
				offset: 1,
				color: "blue", // 100% 处的颜色
			},
		],
		global: false, // 缺省为 false
	};

	useEffect(() => {
		const format = ({ x: x1, y: y1, x2, y2 }) => {
			var getYAngle = function (cx, cy, x2, y2) {
				var x = Math.abs(cx - x2);
				var y = Math.abs(cy - y2);
				var tan = x / y;
				var radina = Math.atan(tan); //用反三角函数求弧度
				var angle = Math.floor(180 / (Math.PI / radina)) || 0; //将弧度转换成角度
				console.log("angle", angle);
				/**
				 * 根据目标点判断象限(注意是笛卡尔坐标)
				 * 一: +,+
				 * 二: -,+
				 * 三: -,+
				 * 一: +,-
				 */

				//  * 二、三象限要加 180°
				if (x2 < 0 && y2 >= 0) {
					angle = 180 + angle;
				}
				if (x2 < 0 && y2 < 0) {
					angle = 180 + angle;
				}

				// 一、二象限 === 0 就是 180°
				if (angle === 0) {
					if ((x2 >= 0 && y2 > 0) || (x2 <= 0 && y2 > 0)) {
						angle = 180 + angle;
					}
				}

				return angle;
			};

			/**
			 * 1、将 二维 坐标看成一个正方形([0, 0],[1, 0],[1, 1],[0, 1]), 坐落于一象限
			 * 2、根据二维坐标转一个新的坐标(相对于正方形中心点的,所以线段会贯穿正方形),
			 *    把相对于笛卡尔坐标系中心点的坐标,转为相对于正方形中心点的坐标
			 *    eg: x 0.5 => 0,
			 *        y 0   => -0.5
			 *
			 *        其实就是  x - 0.5,  y - 0.5
			 */
			const deg = getYAngle(color.x - 0.5, color.y - 0.5, color.x2 - 0.5, color.y2 - 0.5); 
		};
		const deg = format(color);

		// 计算成 css 渐变
		setDeg(deg);

		const options = {
			grid: {
				left: -100,
				top: 0,
				right: 0,
				bottom: 0,
				width: "180%",
			},
			xAxis: { show: false, data: ["数据"], boundaryGap: false },
			yAxis: { show: false },
			series: [
				{
					data: [100],
					type: "bar",
					itemStyle: {
						color,
					},
				},
			],
		};
		myChart.current = echarts.init(document.getElementById("main"));
		myChart.current.setOption(options);
	}, []);

	// 是否按下
	const isDwon = useRef();
	const onMouseDown = () => {
		isDwon.current = true; 
	};
	const onMouseUp = () => {
		isDwon.current = false; 
	};
	const onMouseMove = (event) => {
		if (!isDwon.current) return;
		const rece = pointArr.current.getBoundingClientRect();
		// 中心点
		const center = {
			x: rece.left + (rece.right - rece.left) / 2,
			y: rece.top + (rece.bottom - rece.top) / 2,
		};
		const rad = Math.atan2(center.y - event.pageY, center.x - event.pageX);
		let deg = rad / (Math.PI / 180) - 90; 
		if (deg < 0) {
			deg += 360;
		}
		setDeg(deg); 

		// 角度转坐标 
		const format2 = (deg) => { 

			// 假定旋转半径 0.5 
			const start = { x: 0, y: -0.5 };
			const end = {};
			end.x2 = start.x * Math.cos((deg * Math.PI) / 180) - start.y * Math.sin((deg * Math.PI) / 180);
			end.y2 = start.x * Math.sin((deg * Math.PI) / 180) + start.y * Math.cos((deg * Math.PI) / 180);
			 
      // 算出对应其他象限中对应的点
      end.x = 0 - end.x2;
      end.y = 0 - end.y2;

      // console.log(deg, end);
      return end; 
		};
		const { x, y, x2, y2 } = format2(deg);

		myChart.current.setOption({
			series: [
				{
					data: [100],
					type: "bar",
					itemStyle: {
						color: {
							...color,
							x: x + 0.5,
							y: y + 0.5,
							x2: x2 + 0.5,
							y2: y2 + 0.5,
						},
					},
				},
			],
		}); 
	};

	return (
		<>
			<div id="main" style={{ width: 300, height: 300, display: "inline-block", border: "3px solid green" }}></div>
			<div
				style={{
					backgroundImage: `linear-gradient(${deg}deg, red, blue)`,
					width: 300,
					height: 300,
					display: "inline-block",
					border: "3px solid green",
					marginLeft: 12,
				}}
			></div>

			<div onMouseDown={onMouseDown} onMouseUp={onMouseUp} onMouseMove={onMouseMove} style={{ width: 300, height: 300, position: "relative", border: "3px solid green" }}>
				<div
					ref={pointArr}
					style={{
						position: "absolute",
						fontSize: 50,
						textAlign: "center",
						display: "flex",
						justifyContent: "center",
						alignContent: "center",
						left: "0px",
						right: "0px",
						top: "0px",
						bottom: "0px",
						margin: "auto",
						height: 60,
						width: 60,
						transform: `rotate(${deg - 90}deg)`,
					}}
				>
					<span>→</span>
				</div>
			</div> 
		</>
	);
}

export default App;


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/668642.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

BrainGPT1,一个帮你b站点歌放视频的多模态多轮对话模型

BrainGPT1&#xff0c;一个帮你b站点歌放视频的多模态多轮对话模型 返回论文目录 项目地址 模型地址 作者&#xff1a;华东师范大学&#xff0c;计算机科学与技术学院&#xff0c;智能教育研究院的小怪兽会微笑。 介绍 BrainGPT1是一个工具调用多轮对话模型&#xff0c;与G…

[机器学习]GPT LoRA 大模型微调,生成猫耳娘

往期热门专栏回顾 专栏描述Java项目实战介绍Java组件安装、使用&#xff1b;手写框架等Aws服务器实战Aws Linux服务器上操作nginx、git、JDK、VueJava微服务实战Java 微服务实战&#xff0c;Spring Cloud Netflix套件、Spring Cloud Alibaba套件、Seata、gateway、shadingjdbc…

BU01板卡引脚

概述 BU01 是一款高速采集卡&#xff0c;主要用于高带宽数据采集及传输&#xff0c;应用领域多为数据中 心及数据采集领域。 端口提供60Gbps 传输带宽&#xff0c;可兼容2 个SFP万兆网口&#xff0c;和1 个40GE QSFP 光 口。和主机通信采用的是PCIE 2.0 x8 模式&#xff0c;最…

C++哈希的应用:位图 布隆过滤器 哈希切割

目录 位图 bitset 构造空间 将某个位变为0 将某个位变为1 检查是否存在 完整代码 拓展问题一 ​编辑 拓展问题二 布隆过滤器 判断是否存在 使用场景 哈希切割 拓展问题一 拓展问题二 位图 问题&#xff1a;有四十个亿未排序的不重复的无符号整数&#xff0c;此…

算法导论 总结索引 | 第三部分 第十四章:数据结构的扩张

1、通过存储 额外信息的方法来扩张一 种标准的数据结构&#xff0c;然后对这种数据结构&#xff0c;编写新的操作来支持所需的应用。因为添加的信息 必须要能被该数据结构上的常规操作更新和维护 2、通过扩张红黑树构造出的两种数据结构&#xff1a;14.1介绍 一种支持一般动态…

对boot项目拆分成cloud项目的笔记

引言&#xff1a;这里我用的是新版本的技术栈 spring-boot-starter-parent >3.2.5 mybatis-spring-boot-starter >3.0.3 mybatis-plus-boot-starter >3.5.5 spring-cloud-dependencies …

给Docker一个辈分(备份),免得无后...

定期备份所有 Docker 镜像 Linux 脚本 创建一个名为 backup_all_docker_images.sh 的脚本文件&#xff0c;内容如下&#xff1a; #!/bin/bash# 定义变量 BACKUP_DIR"/backup/docker" TIMESTAMP$(date "%Y%m%d%H%M") BACKUP_FILE"${BACKUP_DIR}/doc…

vx小程序初学

小程序初学 在我还没接触到微信小程序之前&#xff0c;通常使用轮播要么手写或使用swiper插件去实现&#xff0c;当我接触到微信小程序之后&#xff0c;我看到了微信小程序的强大之处&#xff0c;让我为大家介绍一下吧&#xff01; swiper与swiper-item一起使用可以做轮播图 …

Facebook开户 | Facebook二不限户

Facebook二不限户的正确使用方法 Facebook 二不限是指 Facebook 国内二不限户&#xff0c;是通过代理开出来的一种特殊账户&#xff0c;️需要广告主准备主页。 其特点是&#xff1a;限主页、不限域名、额度没解限&#xff0c;解限后则不限额度。 相比于三不限户&#xff0c;…

Keras 3.0强势回归,助力深度学习

大家好&#xff0c;Keras的简洁代码风格一直受到开发者的青睐&#xff0c;自从Keras宣布支持Pytorch和Jax后&#xff0c;开发者们迎来了新的选择。 本文将介绍Keras 3.0的实用技巧&#xff0c;以一个典型的编码器-解码器循环神经网络为例&#xff0c;展示如何利用子类化API构建…

【Ubuntu】100 系统字体安装和更改

系统&#xff1a;Ubuntu18.04LTS 1 Why we need&#xff1f; 写这篇经验贴的原因&#xff1a; ①我需要装一下中文字体&#xff08;Qt要用&#xff09;&#xff1b; ②想调一下字体大小和默认中文字体的样式 2 装第三方字体 Step1&#xff1a;安装软件Font Manager sudo ap…

AI数据分析:用kimi生成一个正弦波数学动画

正弦波公式: ƒ(x) a * sin(x x0) b 公式中&#xff1a; a: 决定正弦函数振动幅度的大小&#xff1b; x0:表示x开始比0拖后的弧度值&#xff1b; b&#xff1a;表示函数偏离X轴的距离&#xff1b; 对于难以理解的学生来说&#xff0c;可以用动画把这个公式直观的展现出…

数据结构05:树与二叉树 习题02[C++]

考研笔记整理&#xff0c;本篇作为二叉树的入门习题&#xff0c;供小伙伴们参考~&#x1f95d;&#x1f95d; 之前的博文链接在此&#xff1a;数据结构05&#xff1a;树与二叉树[C]-CSDN博客~&#x1f95d;&#x1f95d; 第1版&#xff1a;王道书的课后习题~&#x1f9e9;&am…

曲面细分技术在AI去衣中的创新应用

引言&#xff1a; 随着人工智能技术的飞速发展&#xff0c;其在图像处理领域的应用日益广泛。其中&#xff0c;AI去衣技术因其独特的应用场景而备受瞩目。在这一技术的发展过程中&#xff0c;曲面细分技术发挥了至关重要的作用。本文将深入探讨曲面细分技术在AI去衣中的作用及其…

[AI OpenAI] 推出ChatGPT Edu

一种负担得起的解决方案&#xff0c;帮助大学将AI负责任地引入校园。 我们宣布推出ChatGPT Edu&#xff0c;这是一个专为大学设计的ChatGPT版本&#xff0c;旨在负责任地向学生、教职员工、研究人员和校园运营部署AI。ChatGPT Edu由GPT-4o提供支持&#xff0c;能够跨文本和视觉…

【计算机毕设】设计与实现基于SpringBoot的在线文档管理系统 - 源码免费(私信领取)

免费领取源码 &#xff5c; 项目完整可运行 &#xff5c; v&#xff1a;chengn7890 诚招源码校园代理&#xff01; 1. 研究目的 在当今信息爆炸的时代&#xff0c;文档管理对于任何组织都至关重要。基于SpringBoot的在线文档管理系统的设计旨在为用户提供一个便捷、高效、安全的…

unityBIM

Revit模型到Unity勉强能用 1、Revit直接导出FBX&#xff0c;然后拖到unity里面 2、通过Navisworks导出FBX&#xff0c;拖到unity里面。 我什么都还没做&#xff0c;只建立了一个空的URP效果&#xff0c;把FBX拖进去&#xff0c;挂了一个相机控制器&#xff0c;效果勉强看得过…

IP路由策略1

控制层面:路由协议传递路由信息的流量--对应的方向 数据层面:设备间具体访问时请求的流量--对应方向 控制层面方向与数据层面方向一定相反 在控制层面流量进或出的接口上&#xff0c;抓取流量后&#xff0c;修改其中参数或删除该信息&#xff0c;最终起到影响路由器路由表的生…

React UseMemo源码分析

useMemo useMemo 是 React 提供的内置 Hooks&#xff0c;主要作用就是缓存&#xff0c;如果依赖项没有变化&#xff0c;Memo 方法不会再次执行&#xff0c;计算量比较高的方法可以使用&#xff0c;从而提高用户体验。本文将通过一个例子跟踪 Memo 的创建、更新流程。 App.js …

【机器学习】让大模型变得更聪明

文章目录 前言1. 理解大模型的局限性1.1 理解力的挑战1.2 泛化能力的挑战1.3 适应性的挑战 2. 算法创新&#xff1a;提高模型学习和推理能力2.1 自监督学习2.2 强化学习2.3 联邦学习 3. 数据质量与多样性&#xff1a;增强模型的泛化能力3.1 高质量数据的获取3.2 数据多样性的重…