首页 > 其他分享 >react echarts tooltip 区域新加一个输入框,可以编辑保存数据

react echarts tooltip 区域新加一个输入框,可以编辑保存数据

时间:2023-05-11 11:12:35浏览次数:40  
标签:const tooltip annotation react 输入框 id tooltipId echarts


 

// demo页面
// 需求:产品要求在折线图的tooltip上新加一个输入框,可以编辑这个输入框保存备注信息,需要两种交互方式: 1.鼠标滑过展示备注信息。2.鼠标点击某一个日期时,鼠标可以滑到tooltip上做保存/编辑操作。
// 思路:1.保留初始鼠标滑过echarts图效果。
// 2.主要难点是点击时tooltip固定可编辑,有尝试通过动态改变echarts自带tooltip的显示隐藏,但是效果并不好,并且因为react是虚拟dom, 所以在react中没办法使用on监听事件,使用ReactEcharts也只是必须要点击某一个symbol圆点才可以出发点击事件,最后放弃使用自有api的想法
// 3.最终方案:用一个div包裹echarts, 然后在echarts的同级新建一个div用来用来模拟真实tooltip,通过鼠标移入移出事件控制真实tooltip的显示与隐藏。
// 思路大概就是这样,下面就是代码部分的实现。因为此页面只是用来当做demo,所以只提供伪代码
import * as echarts from "echarts";

 

const [echartsParams, setEchartsParams] = useState([]); // tooltip formatter的params数据 const [echartsData, setEchartsData] = useState({}); // 编辑或保存时输入框的数据 const [chart, setChart] = useState(); // 记录当前echarts, 鼠标移入移出时控制tooltip显示隐藏 const [chartAllData, setChartAllData] = useState(); // echaers数据源   const save = (date, id) => {   const value = echartsData.annotation;     const url = id === null     ? axios('/save', {       method: 'post',       data: {         date,         annotation: value       }     })     : axios('/update', {       method: 'post',       data: {         date,         annotation: value,         id,       }     }) }   window.save = save; // 必须挂载在window上 // echarts。 data: 数据源 const initCharts = (data) => {   setChartAllData(data)    let option = {     tooltip: {         className: 'chartsTooltip',         trigger: "axis",         enterable: true,         appendToBody: true,         axisPointer: {           type: "line",           label: {             backgroundColor: "#6a7985",           },         },         formatter: (params) => {           setEchartsParams(params);           let str = '';           const annotation = data.filter((v) => v.time === params[0].name)[0].annotation || { annotation: '', id: null };           params.forEach((item) => {             str += '<div class="box">' + `<div>${item.marker + item.seriesName}</div>` + `<div>${item.value}</div>` + '</div>';           })
          return `<div class="test" id="tooltipDom">             ${params[0].name}             <br />             ${str}             <textarea class="ipt" id="inputId">${annotation.annotation}</textarea>             <p onclick='save("${params[0].name}", "${annotation.id}")' class="save-btn">保存</p>           </div>`         },       },   }     let Chart = echarts.getInstanceByDom(       document.getElementById("enterprise-ratio")     ); //有的话就获取已有echarts实例的DOM节点。     if (Chart) {       // 如果不存在,就进行初始化。       Chart.clear();     } else {       Chart = echarts.init(document.getElementById("enterprise-ratio"));     }     Chart.setOption(option);     setChart(Chart); }   return(   <div // 最外层的盒子     onMouseLeave={() => {         chart.dispatchAction({           type: 'showTip',         })         chart.setOption({          tooltip: {            show: true,          },         })         const chartsTooltip = document.getElementsByClassName('chartsTooltip')[0];         if (chartsTooltip) {           chartsTooltip.style.display = 'none';         }         const tooltipId = document.getElementById('tooltipId')         if (tooltipId) {           tooltipId.style.display = 'none';         }   >     <div>这是这个区域的标题</div>    <div     className="echarts-content"        onClick={() => {          chart.dispatchAction({            type: 'hideTip',          })          chart.setOption({            tooltip: {              show: false,            },          })      const annotationData = chartAllData.filter((v) => v.time === echartsParams[0].name)[0].annotation || { annotation: '', id: null }; // annotation: '后端接口返回的数据', id: '编辑时需要的id, 根据id是否未null判断是要保存还是编辑'          setEchartsData(annotationData)          const tooltipId = document.getElementById('tooltipId')            if (tooltipId) {              tooltipId.style.display = 'block';            }        }}        >              {               echartsParams.length ? (                 <div className="tooltip-box" id="tooltipId">                   {echartsParams[0].name}                   {                     echartsParams.map((item, i) => {                       return (                         <div className="box" key={i}>                           <>                             <div>                               <span dangerouslySetInnerHTML={{ __html: item.marker }}></span>                               <span>{item.seriesName}</span>                             </div>                             <div>{item.value}</div>                           </>                         </div>                       )                     })                   }                   <Input.TextArea                     value={echartsData.annotation}                     className="ipt"                     onChange={(e) => {                       const v = e.target.value;                       echartsData.annotation = v;                       setEchartsData({ ...echartsData })                     }}                   ></Input.TextArea>                   <p onClick={() => save(echartsParams[0].name, echartsData.id)} className="save-btn">保存</p>                 </div>               ) : ''             }     <div className="flex-1" id="enterprise-ratio" style={{ height: '200px', width: '100%' }}></div> // echarts折线图    </div>   </div> )
 

 

标签:const,tooltip,annotation,react,输入框,id,tooltipId,echarts
From: https://www.cnblogs.com/lou-0820/p/17390455.html

相关文章

  • uniapp自定义开发一个文本输入框
    开发component中的一个input标签一、在原来的模块上面创建一个新的类TestComponent1.新建TestComponent2.配置json文件二、uniapp准备工作1.在uniapp中写一下刚刚创建的输入框2.打包导出资源3.资源替换复制刚刚生成的本地资源文件夹到AndroidStudio项目中......
  • react-html2canvas-jspdf 自动分页导出pdf
    //新建exportPDF.js文件importhtml2canvasfrom'html2canvas';importjsPDFfrom'jspdf';functiongeneratePDF(id,title){ //下载pdf方法 letdemo=document.getElementById(id); demo.style.overflow='visible'; html2canvas(......
  • ReactRedux工具包reduxjs/toolkit的使用
    首先可以先看一下Redux如何工作store负责存储数据,相当于仓库,action负责dispatch派发数据,reducer负责接收处理数据然后交给store(个人理解可能有些偏差欢迎交流斧正)传统redux写法(旧)//reducerconstcounterReducer=(state={counter:0},action)=>{if(action.type......
  • 直播网站源码,安卓防止输入框自动弹出
    直播网站源码,安卓防止输入框自动弹出可以在edittext的父布局结构中(例如LinearLayout,RelativeLayout等)添加  android:focusable="true" android:focusableInTouchMode="true"​以上就是直播网站源码,安卓防止输入框自动弹出,更多内容欢迎关注之后的文章 ......
  • React
    ReactReactsetState异步同步在setTimeout、Promise等原生事件API调用中setState和useState是同步执行的,立即执行renderClassComponent能获取到最新值=>this.state=> 引用类型FunctionComponent不能获取到最新值=>只能得到之前的值=> 闭包多次执......
  • React笔记-组件(一)
    React学习笔记-组件(一未完成)特点声明式组件化跨平台React脚手架a.全局安装react脚手架create-react-appnpminstallcreate-react-app-g&npxcreate-react-appmy-appb.使用create-react-app创建react应用,如果直接使用npx则无需执行这一步,直接执行第3步c......
  • React笔记-事件(三)
    React学习笔记-事件(三)定义事件React元素的事件处理和DOM元素的很相似但是有一点语法上的不同React事件的命名采用小驼峰式(camelCase)而不是纯小写如点击事件onClickimportReactfrom'react'exportdefaultclasslearnEventextendsReact.Component{///......
  • React笔记-样式(二)
    React学习笔记-样式(二)内联样式importReactfrom"react";exportdefaultclassLearnStyleextendsReact.Component{render(){return(<div>{/*以下两种方法都可以一种不用引号将css以小驼峰方式写另一种加上引号写......
  • React笔记-渲染列表Key(五)
    React学习笔记-渲染列表Key(五)渲染列表需要添加key属性importReactfrom"react"exportdefaultclassLearnKeyextendsReact.Component{state={infos:[{name:'Bob',age:18},{name:'kitty',age:20},{name:......
  • React笔记-state(四)
    React学习笔记-state(四)概念state的主要作用是用于组件保存控制以及修改自己的状态它算是组件的私有属性不可通过外部访问和修改只能通过组件内部的this.setState来修改修改state属性会导致组件的重新渲染注意:如果直接通过this.state.xxx的方式修改,组件不会重新渲染,但......