From b810251a0c4c2ba72a4d4e56a43ae2baed92a638 Mon Sep 17 00:00:00 2001 From: RTGK Date: Thu, 20 Jun 2024 11:26:44 +0800 Subject: [PATCH] first commit --- .eslintrc.json | 19 + .gitignore | 35 + README.md | 34 + components/client/Button.js | 124 + components/client/FormItem.js | 148 + .../client/PackagingOperation/ModalClient.js | 555 ++ components/client/PackagingOperation/hooks.js | 199 + components/client/PackagingOperation/index.js | 619 ++ components/client/PackagingOperation/logo.js | 2 + components/client/utils.js | 17 + components/screen/Card/index.js | 97 + components/screen/Chart/echarts-liquidfill.js | 1688 ++++ components/screen/Chart/index.js | 38 + components/screen/Chart/theme.json | 440 + components/screen/CircleProcess/index.js | 162 + components/screen/Footer/DateItem.js | 60 + components/screen/Footer/index.js | 105 + components/screen/Header/index.js | 63 + components/screen/InfoCardList/InfoCard.js | 138 + components/screen/InfoCardList/index.js | 361 + components/screen/Row/index.js | 207 + components/screen/Table/index.js | 181 + components/screen/TableProgress/index.js | 82 + components/screen/TimerShaft/index.js | 141 + components/screen/WithConfig/index.js | 12 + next.config.js | 11 + package-lock.json | 8361 +++++++++++++++++ package.json | 39 + pages/_app.js | 34 + pages/api/hello.js | 5 + pages/client/PackagingOperation/index.js | 2 + pages/components/Alert/index.js | 107 + pages/components/Card/index.js | 367 + pages/components/Combobox/index.js | 84 + pages/components/Header/Header.js | 163 + pages/components/Header/LabelCheck.js | 55 + pages/components/Header/StateCheck.js | 10 + pages/components/Header/VLabel.js | 8 + pages/components/Modal/index.js | 119 + pages/components/ProcessCard/index.js | 107 + pages/components/Switch/index.js | 24 + pages/components/Tab/index.js | 75 + pages/components/processTaskCard/index.js | 167 + pages/index.js | 984 ++ pages/reports/curveParam/index.js | 216 + pages/reports/equipmentCurve/index.js | 592 ++ pages/reports/homePage/index.js | 981 ++ pages/reports/picView/data.json | 14 + pages/reports/picView/index.js | 108 + pages/reports/xmr/DemoDualAxes.js | 335 + pages/reports/xmr/DemoLine.js | 216 + pages/reports/xmr/index.js | 1359 +++ pages/screen/comprehensive/index.js | 438 + pages/screen/equipment/index.js | 978 ++ pages/screen/planBoard/index.js | 668 ++ pages/screen/planDailyBoard/index.js | 1740 ++++ pages/screen/processBoard/index.js | 416 + pages/screen/processItem/index.js | 809 ++ pages/screen/qualityBoard/index.js | 1087 +++ postcss.config.js | 6 + public/config.js | 179 + public/favicon.ico | Bin 0 -> 4286 bytes public/font/PingFang SC Regular.ttf | Bin 0 -> 10244 bytes public/font/PingFang SC.ttf | Bin 0 -> 13609012 bytes public/font/UnidreamLED.ttf | Bin 0 -> 31600 bytes public/font/rubik/Rubik-Black.ttf | Bin 0 -> 211696 bytes public/font/rubik/Rubik-BlackItalic.ttf | Bin 0 -> 192004 bytes public/font/rubik/Rubik-Bold.ttf | Bin 0 -> 212840 bytes public/font/rubik/Rubik-BoldItalic.ttf | Bin 0 -> 191132 bytes public/font/rubik/Rubik-Italic.ttf | Bin 0 -> 211768 bytes public/font/rubik/Rubik-Light.ttf | Bin 0 -> 216812 bytes public/font/rubik/Rubik-LightItalic.ttf | Bin 0 -> 208068 bytes public/font/rubik/Rubik-Medium.ttf | Bin 0 -> 216244 bytes public/font/rubik/Rubik-MediumItalic.ttf | Bin 0 -> 193052 bytes public/font/rubik/Rubik-Regular.ttf | Bin 0 -> 220600 bytes public/font/rubik/SIL Open Font License.txt | 48 + public/img/arrow.png | Bin 0 -> 3340 bytes public/img/bg.png | Bin 0 -> 5270 bytes public/img/dc.png | Bin 0 -> 1767 bytes public/img/eqimg.jpeg | Bin 0 -> 18715 bytes public/img/footerBg.png | Bin 0 -> 61179 bytes public/img/headerBg.png | Bin 0 -> 107527 bytes public/img/hpPercent.png | Bin 0 -> 10639 bytes public/img/img1.png | Bin 0 -> 2468 bytes public/img/jt.png | Bin 0 -> 707 bytes public/img/logo.png | Bin 0 -> 30896 bytes public/img/noImg.png | Bin 0 -> 96888 bytes public/img/process/L7-BZ-SB.png | Bin 0 -> 261798 bytes public/img/process/L7-BZ-ZZP.png | Bin 0 -> 167043 bytes public/img/process/L7-BZ.png | Bin 0 -> 85476 bytes public/img/process/L7-CC-SB.png | Bin 0 -> 307558 bytes public/img/process/L7-CC-ZZP.png | Bin 0 -> 165390 bytes public/img/process/L7-CC.png | Bin 0 -> 92987 bytes public/img/process/L7-CM-SB.png | Bin 0 -> 311417 bytes public/img/process/L7-CM-ZZP.png | Bin 0 -> 141454 bytes public/img/process/L7-CM.png | Bin 0 -> 93750 bytes public/img/process/L7-FS-SB.png | Bin 0 -> 256019 bytes public/img/process/L7-FS-ZZP.png | Bin 0 -> 180572 bytes public/img/process/L7-FS.png | Bin 0 -> 83332 bytes public/img/process/L7-HL-SB.png | Bin 0 -> 188303 bytes public/img/process/L7-HL-ZZP.png | Bin 0 -> 144946 bytes public/img/process/L7-HL.png | Bin 0 -> 65371 bytes public/img/process/L7-PWGZ-SB.png | Bin 0 -> 292737 bytes public/img/process/L7-PWGZ-ZZP.png | Bin 0 -> 176130 bytes public/img/process/L7-PWGZ.png | Bin 0 -> 88749 bytes public/img/process/L7-SCCC.png | Bin 0 -> 72453 bytes public/img/process/L7-SJ-SB.png | Bin 0 -> 232682 bytes public/img/process/L7-SJ-ZZP.png | Bin 0 -> 174570 bytes public/img/process/L7-SJ.png | Bin 0 -> 74824 bytes public/img/process/L7-TL-SB.png | Bin 0 -> 316342 bytes public/img/process/L7-TL-ZZP.png | Bin 0 -> 150834 bytes public/img/process/L7-TL.png | Bin 0 -> 89639 bytes public/img/process/L7-XM-SB.png | Bin 0 -> 311417 bytes public/img/process/L7-XM-ZZP.png | Bin 0 -> 141454 bytes public/img/process/L7-XM.png | Bin 0 -> 88909 bytes public/img/process/L7-YH-SB.png | Bin 0 -> 304742 bytes public/img/process/L7-YH-ZZP.png | Bin 0 -> 96025 bytes public/img/process/L7-YH.png | Bin 0 -> 95264 bytes public/img/processBg/baozhuang.png | Bin 0 -> 326779 bytes public/img/processBg/chuci.png | Bin 0 -> 265986 bytes public/img/processBg/fensui.png | Bin 0 -> 362099 bytes public/img/processBg/penwu.png | Bin 0 -> 370232 bytes public/img/processBg/shaojie.png | Bin 0 -> 349401 bytes public/img/processBg/total.png | Bin 0 -> 471811 bytes public/img/processBg/touliao.png | Bin 0 -> 364010 bytes public/img/processBg/yanmo.png | Bin 0 -> 274484 bytes public/img/sz.png | Bin 0 -> 1787 bytes public/img/tc.png | Bin 0 -> 1774 bytes reportUtils/getHomePageData.js | 105 + reportUtils/getQueryData.js | 343 + reportUtils/request.js | 74 + reportUtils/theme.json | 396 + styles/globals.css | 158 + styles/homecss.module.css | 322 + tailwind.config.js | 45 + utils/checks.js | 8 + utils/eqs.js | 456 + utils/hooks.js | 68 + utils/requests.js | 113 + utils/socket.io.js | 6619 +++++++++++++ 140 files changed, 35146 insertions(+) create mode 100644 .eslintrc.json create mode 100644 .gitignore create mode 100644 README.md create mode 100644 components/client/Button.js create mode 100644 components/client/FormItem.js create mode 100644 components/client/PackagingOperation/ModalClient.js create mode 100644 components/client/PackagingOperation/hooks.js create mode 100644 components/client/PackagingOperation/index.js create mode 100644 components/client/PackagingOperation/logo.js create mode 100644 components/client/utils.js create mode 100644 components/screen/Card/index.js create mode 100644 components/screen/Chart/echarts-liquidfill.js create mode 100644 components/screen/Chart/index.js create mode 100644 components/screen/Chart/theme.json create mode 100644 components/screen/CircleProcess/index.js create mode 100644 components/screen/Footer/DateItem.js create mode 100644 components/screen/Footer/index.js create mode 100644 components/screen/Header/index.js create mode 100644 components/screen/InfoCardList/InfoCard.js create mode 100644 components/screen/InfoCardList/index.js create mode 100644 components/screen/Row/index.js create mode 100644 components/screen/Table/index.js create mode 100644 components/screen/TableProgress/index.js create mode 100644 components/screen/TimerShaft/index.js create mode 100644 components/screen/WithConfig/index.js create mode 100644 next.config.js create mode 100644 package-lock.json create mode 100644 package.json create mode 100644 pages/_app.js create mode 100644 pages/api/hello.js create mode 100644 pages/client/PackagingOperation/index.js create mode 100644 pages/components/Alert/index.js create mode 100644 pages/components/Card/index.js create mode 100644 pages/components/Combobox/index.js create mode 100644 pages/components/Header/Header.js create mode 100644 pages/components/Header/LabelCheck.js create mode 100644 pages/components/Header/StateCheck.js create mode 100644 pages/components/Header/VLabel.js create mode 100644 pages/components/Modal/index.js create mode 100644 pages/components/ProcessCard/index.js create mode 100644 pages/components/Switch/index.js create mode 100644 pages/components/Tab/index.js create mode 100644 pages/components/processTaskCard/index.js create mode 100644 pages/index.js create mode 100644 pages/reports/curveParam/index.js create mode 100644 pages/reports/equipmentCurve/index.js create mode 100644 pages/reports/homePage/index.js create mode 100644 pages/reports/picView/data.json create mode 100644 pages/reports/picView/index.js create mode 100644 pages/reports/xmr/DemoDualAxes.js create mode 100644 pages/reports/xmr/DemoLine.js create mode 100644 pages/reports/xmr/index.js create mode 100644 pages/screen/comprehensive/index.js create mode 100644 pages/screen/equipment/index.js create mode 100644 pages/screen/planBoard/index.js create mode 100644 pages/screen/planDailyBoard/index.js create mode 100644 pages/screen/processBoard/index.js create mode 100644 pages/screen/processItem/index.js create mode 100644 pages/screen/qualityBoard/index.js create mode 100644 postcss.config.js create mode 100644 public/config.js create mode 100644 public/favicon.ico create mode 100644 public/font/PingFang SC Regular.ttf create mode 100644 public/font/PingFang SC.ttf create mode 100644 public/font/UnidreamLED.ttf create mode 100644 public/font/rubik/Rubik-Black.ttf create mode 100644 public/font/rubik/Rubik-BlackItalic.ttf create mode 100644 public/font/rubik/Rubik-Bold.ttf create mode 100644 public/font/rubik/Rubik-BoldItalic.ttf create mode 100644 public/font/rubik/Rubik-Italic.ttf create mode 100644 public/font/rubik/Rubik-Light.ttf create mode 100644 public/font/rubik/Rubik-LightItalic.ttf create mode 100644 public/font/rubik/Rubik-Medium.ttf create mode 100644 public/font/rubik/Rubik-MediumItalic.ttf create mode 100644 public/font/rubik/Rubik-Regular.ttf create mode 100644 public/font/rubik/SIL Open Font License.txt create mode 100644 public/img/arrow.png create mode 100644 public/img/bg.png create mode 100644 public/img/dc.png create mode 100644 public/img/eqimg.jpeg create mode 100644 public/img/footerBg.png create mode 100644 public/img/headerBg.png create mode 100644 public/img/hpPercent.png create mode 100644 public/img/img1.png create mode 100644 public/img/jt.png create mode 100644 public/img/logo.png create mode 100644 public/img/noImg.png create mode 100644 public/img/process/L7-BZ-SB.png create mode 100644 public/img/process/L7-BZ-ZZP.png create mode 100644 public/img/process/L7-BZ.png create mode 100644 public/img/process/L7-CC-SB.png create mode 100644 public/img/process/L7-CC-ZZP.png create mode 100644 public/img/process/L7-CC.png create mode 100644 public/img/process/L7-CM-SB.png create mode 100644 public/img/process/L7-CM-ZZP.png create mode 100644 public/img/process/L7-CM.png create mode 100644 public/img/process/L7-FS-SB.png create mode 100644 public/img/process/L7-FS-ZZP.png create mode 100644 public/img/process/L7-FS.png create mode 100644 public/img/process/L7-HL-SB.png create mode 100644 public/img/process/L7-HL-ZZP.png create mode 100644 public/img/process/L7-HL.png create mode 100644 public/img/process/L7-PWGZ-SB.png create mode 100644 public/img/process/L7-PWGZ-ZZP.png create mode 100644 public/img/process/L7-PWGZ.png create mode 100644 public/img/process/L7-SCCC.png create mode 100644 public/img/process/L7-SJ-SB.png create mode 100644 public/img/process/L7-SJ-ZZP.png create mode 100644 public/img/process/L7-SJ.png create mode 100644 public/img/process/L7-TL-SB.png create mode 100644 public/img/process/L7-TL-ZZP.png create mode 100644 public/img/process/L7-TL.png create mode 100644 public/img/process/L7-XM-SB.png create mode 100644 public/img/process/L7-XM-ZZP.png create mode 100644 public/img/process/L7-XM.png create mode 100644 public/img/process/L7-YH-SB.png create mode 100644 public/img/process/L7-YH-ZZP.png create mode 100644 public/img/process/L7-YH.png create mode 100644 public/img/processBg/baozhuang.png create mode 100644 public/img/processBg/chuci.png create mode 100644 public/img/processBg/fensui.png create mode 100644 public/img/processBg/penwu.png create mode 100644 public/img/processBg/shaojie.png create mode 100644 public/img/processBg/total.png create mode 100644 public/img/processBg/touliao.png create mode 100644 public/img/processBg/yanmo.png create mode 100644 public/img/sz.png create mode 100644 public/img/tc.png create mode 100644 reportUtils/getHomePageData.js create mode 100644 reportUtils/getQueryData.js create mode 100644 reportUtils/request.js create mode 100644 reportUtils/theme.json create mode 100644 styles/globals.css create mode 100644 styles/homecss.module.css create mode 100644 tailwind.config.js create mode 100644 utils/checks.js create mode 100644 utils/eqs.js create mode 100644 utils/hooks.js create mode 100644 utils/requests.js create mode 100644 utils/socket.io.js diff --git a/.eslintrc.json b/.eslintrc.json new file mode 100644 index 0000000..9c600d2 --- /dev/null +++ b/.eslintrc.json @@ -0,0 +1,19 @@ +{ + "extends": "next/core-web-vitals", + "rules": { + "react-hooks/exhaustive-deps": "off", + "react/display-name": "off", + "react/jsx-key": "off", + "@next/next/no-sync-scripts": "off", + "react/no-unknown-property": [ + 2, + { + "ignore": [ + "jsx", + "global" + ] + } + ] + } +} + diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..601f8c8 --- /dev/null +++ b/.gitignore @@ -0,0 +1,35 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# dependencies +/node_modules +/.pnp +.pnp.js + +# testing +/coverage + +# next.js +/.next/ +/out/ + +# production +/build + +# misc +.DS_Store +*.pem + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* +.pnpm-debug.log* + +# local env files +.env.local +.env.development.local +.env.test.local +.env.production.local + +# vercel +.vercel diff --git a/README.md b/README.md new file mode 100644 index 0000000..b12f3e3 --- /dev/null +++ b/README.md @@ -0,0 +1,34 @@ +This is a [Next.js](https://nextjs.org/) project bootstrapped with [`create-next-app`](https://github.com/vercel/next.js/tree/canary/packages/create-next-app). + +## Getting Started + +First, run the development server: + +```bash +npm run dev +# or +yarn dev +``` + +Open [http://localhost:3000](http://localhost:3000) with your browser to see the result. + +You can start editing the page by modifying `pages/index.js`. The page auto-updates as you edit the file. + +[API routes](https://nextjs.org/docs/api-routes/introduction) can be accessed on [http://localhost:3000/api/hello](http://localhost:3000/api/hello). This endpoint can be edited in `pages/api/hello.js`. + +The `pages/api` directory is mapped to `/api/*`. Files in this directory are treated as [API routes](https://nextjs.org/docs/api-routes/introduction) instead of React pages. + +## Learn More + +To learn more about Next.js, take a look at the following resources: + +- [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API. +- [Learn Next.js](https://nextjs.org/learn) - an interactive Next.js tutorial. + +You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js/) - your feedback and contributions are welcome! + +## Deploy on Vercel + +The easiest way to deploy your Next.js app is to use the [Vercel Platform](https://vercel.com/new?utm_medium=default-template&filter=next.js&utm_source=create-next-app&utm_campaign=create-next-app-readme) from the creators of Next.js. + +Check out our [Next.js deployment documentation](https://nextjs.org/docs/deployment) for more details. diff --git a/components/client/Button.js b/components/client/Button.js new file mode 100644 index 0000000..e1cb38f --- /dev/null +++ b/components/client/Button.js @@ -0,0 +1,124 @@ +import React, { useCallback, useEffect, useMemo, useRef, useState } from "react"; +import { Spin } from "antd"; +import { LoadingOutlined } from "@ant-design/icons"; +import { useRule } from "./PackagingOperation/hooks" + +const btnTyps = [ + { type: "primary", color: "#0185FF" }, + { type: "danger", color: "#F5222D" }, + { type: "warning", color: "#FAAB0C" }, + { type: "success", color: "#85C700" }, + { type: "info", color: "#26c1e1" }, + { type: "default", color: "rgba(0,0,0,.85)" }, +]; +export const Button = React.memo((props) => { + const { + visible = true, + enable = true, + title, + type = "primary", + ghost = false, + size, + onClick = () => { }, + ruleCode, + } = props; + + const rule = useRule(ruleCode); + // useEffect(() => { + // console.log("rule", rule) + // }, [rule]) + + const types = useRef(btnTyps.map(({ type }) => type)); + + const sizes = useRef(["large"]); + const className = useMemo(() => { + let res = "client-btn"; + if (type && types.current.includes(type)) res += ` btn-${type}`; + if (ghost === true) res += ` btn-ghost`; + if (size && sizes.current.includes(size)) res += ` btn-${size}`; + if (enable === false) res += ` btn-disabled`; + return res; + }, [type, ghost, size, enable]); + + const [loading, setLoading] = useState(false); + const handleClick = useCallback(() => { + if (enable && !loading) { + let res = onClick(); + if (res && typeof res.then == "function" && typeof res.finally == "function") { + setLoading(true); + res.finally(() => setLoading(false)) + } + } + }, [enable, onClick, loading]); + + if (!rule) return null; + if (!visible) return null; + return ( +
enable && handleClick()}> + {title} + } + /> + + +
+ ); +}); diff --git a/components/client/FormItem.js b/components/client/FormItem.js new file mode 100644 index 0000000..551fb56 --- /dev/null +++ b/components/client/FormItem.js @@ -0,0 +1,148 @@ +import React, { + useMemo, +} from "react"; + +export const FormItem = React.memo((props) => { + const { + title, + value = "", + readonly = true, + horizontal = true, + type, + ...otherProps + } = props; + + const valueMemo = useMemo(() => { + if (readonly) { + if (["", undefined, null].includes(value)) return "-"; + return value; + } + if (type === "number") { + return ; + } + return value; + }, [readonly, value, type, otherProps]); + + return ( +
+
{title}
+
{valueMemo}
+ +
+ ); +}); + +const FormItemNumber = React.memo((props) => { + const { value, onChange = () => {}, min } = props; + const onInputChange = (e) => { + onChange(e.target.value); + }; + const onInputBlur = (e) => { + let res = parseFloat(e.target.value); + if (isNaN(res)) res = "-"; + onChange(res + ""); + }; + const onBtnClick = (add) => { + let res = parseFloat(value); + if (isNaN(res)) res = add; + res = res + add; + if (min !== undefined && res < min) res = min; + onChange(res + ""); + }; + const btnDisabled = useMemo(() => [value <= min], [value, min]); + return ( +
+
value > min && onBtnClick(-1)} + > + - +
+ +
onBtnClick(1)}> + + +
+ +
+ ); +}); + diff --git a/components/client/PackagingOperation/ModalClient.js b/components/client/PackagingOperation/ModalClient.js new file mode 100644 index 0000000..28dbadd --- /dev/null +++ b/components/client/PackagingOperation/ModalClient.js @@ -0,0 +1,555 @@ +import React, { + useCallback, + useEffect, + useImperativeHandle, + useRef, + useState, +} from "react"; +import { Modal, Row, Col, Form, Input, Select, message } from "antd"; +import { Button } from "../Button"; +import { request } from "../utils"; + +const ModalClient = React.memo((props) => { + const { + visible = false, + title, + onOk = () => {}, + onCancel = () => {}, + children, + } = props; + return ( + +
{title}
+
{children}
+
+
+ + +
+ ); +}); + +// 设置 +export const ModelSetting = React.forwardRef((props, ref) => { + const {} = props; + const callbackRef = useRef(() => {}); + const [form] = Form.useForm(); + const [visible, setVisible] = useState(false); + + const handleCancel = useCallback(() => setVisible(false), []); + + const [equipList, setEquipList] = useState([]); + const [operationList, setOperationList] = useState([]); + const [workShopList, setWorkShopList] = useState([]); + + // 查询包装机 + const getEquip = useCallback((gid) => { + if (gid) { + request({ + method: "POST", + url: `${window.config.mesUrl}/mdgeneric/md/bmWorkCenter/getEquipByOperationGid?operationGid=${gid}`, + }).then(({ data, success } = {}) => { + if (!success) data = []; + setEquipList(data); + }); + } + }, []); + + // 工序变化 + const onOperationChange = useCallback( + (code, a, data) => { + if (!data) data = operationList; + // console.log("val", key, value, data); + let {gid: operationGid, name} = data.find((item) => item.code === code) || {}; + getEquip(operationGid); + form.setFieldsValue({ + operationName: name, + operationCode: code, + equipName: undefined, + equipCode: undefined, + }); + setEquipList([]); + }, + [operationList, form, getEquip] + ); + + // 查询工序 + const getOperation = useCallback( + (gid, callback) => { + if (gid) { + request({ + method: "POST", + url: `${window.config.mesUrl}/mdgeneric/md/bmWorkCenter/getOpByWorkCenter?workCenterGid=${gid}`, + }).then(({ data, success } = {}) => { + if (!success) data = []; + setOperationList(data); + if (callback) { + callback(data); + } else if (data.length) { + let { name, code } = data[0]; + form.setFieldsValue({ + operationName: name, + equipName: code, + }); + onOperationChange(code,undefined, data) + } + }); + } else setOperationList([]); + }, + [onOperationChange] + ); + + // 车间变化 + const onWorkShopChange = useCallback( + (key = {}) => { + let { gid: workShopGid, name: value } = workShopList.find(({ code }) => code === key) || {}; + getOperation(workShopGid); + form.setFieldsValue({ + workCenterCode: key, + workCenterName: value, + operationName: undefined, + operationCode: undefined, + equipName: undefined, + equipCode: undefined, + }); + setOperationList([]); + setEquipList([]); + }, + [workShopList, form, getOperation] + ); + + // 查询车间 + const getWorkShop = useCallback((callback) => { + request({ + method: "POST", + url: `${window.config.mesUrl}/mdgeneric/md/bmWorkCenter/findAllWithPage`, + bodys: { + // removePermission: true, + page: -1, + pageSize: -1, + sorted: "createTime desc", + filter: "type eq 'workshop'", + }, + }).then(({ data, success } = {}) => { + if (!success) data = []; + setWorkShopList(data); + callback && callback(data); + }); + }, []); + + // 设备变化 + const onEquipChange = useCallback( + (key) => { + let { name: value} = equipList.find(({ code }) => code === key) || {} + form.setFieldsValue({ + equipName: value, + equipCode: key, + }); + }, + [form, equipList] + ); + + // 初始化 + const init = useCallback( + (data, callback) => { + setVisible(true); + form.resetFields(); + form.setFieldsValue(data); + callbackRef.current = callback; + let { workCenterCode, operationCode } = data; + getWorkShop((res) => { + if (workCenterCode) { + let gid = res.find(({ code }) => code === workCenterCode)?.gid; + getOperation(gid, (res) => { + if (operationCode) { + let gid = res.find(({ code }) => code === operationCode)?.gid; + getEquip(gid); + } + }); + } + }); + }, + [form, getWorkShop, getOperation, getEquip] + ); + + const handleOk = useCallback(() => { + let formData = form.getFieldValue(); + if (["", null, undefined].includes(formData.workCenterCode)) + return message.error("请完善车间!"); + if (["", null, undefined].includes(formData.equipCode)) + return message.error("请完善包装机!"); + if (["", null, undefined].includes(formData.bagWeight)) + return message.error("请完善包装袋重量!"); + if (["", null, undefined].includes(formData.packSpecs)) + return message.error("请完包装规格!"); + if (["", null, undefined].includes(formData.upperLimit)) + return message.error("请完善上限值!"); + if (["", null, undefined].includes(formData.lowerLimit)) + return message.error("请完善下限值!"); + // console.log("form", formData); + return new Promise((resolve, reject) => { + request({ + url: `${window.config.mesUrl}/prodexec/prodexec/packageEquipment/setPackInfo`, + bodys: formData, + }).then((res) => { + handleCancel(); + callbackRef.current(); + }) + .finally(() => { + resolve(); + }); + }); + }, [handleCancel, form, workShopList, equipList]); + + useImperativeHandle( + ref, + () => ({ + show: init, + }), + [init] + ); + return ( + +
+ + + + + + + + + + + + + + + + + + + + + + + + +
-
+ + + + + + +
+
+
+
+
+ ); +}); + +// 中断 +export const ModelBreak = React.forwardRef((props, ref) => { + const {} = props; + const callbackRef = useRef(() => {}); + const [form] = Form.useForm(); + const [visible, setVisible] = useState(false); + + const handleCancel = useCallback(() => setVisible(false), []); + + const handleOk = useCallback(() => { + let formData = form.getFieldValue(); + if (["", null, undefined].includes(formData.reason)) + return message.error("请完善中断原因!"); + return new Promise((resolve, reject) => { + request({ + url: `${window.config.mesUrl}/prodexec/prodexec/packageEquipment/interruptPack`, + bodys: { + gid: formData.gid, + }, + }) + .then((res) => { + if (res.success) { + handleCancel(); + callbackRef.current(); + } else message.error(res.message || "操作失败!"); + }) + .finally(() => { + resolve(); + }); + }); + }, [handleCancel]); + useImperativeHandle( + ref, + () => ({ + show() { + setVisible(true); + }, + show: (data, callback) => { + setVisible(true); + form.resetFields(); + form.setFieldsValue(data); + callbackRef.current = callback; + }, + }), + [form] + ); + return ( + +
+ + + +
+
+ ); +}); + +// 结束 +export const ModelFinish = React.forwardRef((props, ref) => { + const {} = props; + const [form, setForm] = useState([]); + const callbackRef = useRef(() => {}); + const [visible, setVisible] = useState(false); + const handleCancel = useCallback(() => setVisible(false), []); + const handleOk = useCallback(() => { + return new Promise((resolve, reject) => { + request({ + url: `${window.config.mesUrl}/prodexec/prodexec/packageEquipment/endPack`, + bodys: form, + }) + .then((res) => { + if (res.success) { + handleCancel(); + callbackRef.current(); + message.success("包装完成!"); + } else message.error(res.message || "操作失败!"); + }) + .finally(() => { + resolve(); + }); + }); + }, [handleCancel, form]); + useImperativeHandle( + ref, + () => ({ + show: (data, callback) => { + setVisible(true); + setForm(data); + callbackRef.current = callback; + }, + }), + [] + ); + return ( + +

是否确认结束?

+
+ ); +}); + +// 超产 +export const ModelOverproductionBatch = React.forwardRef((props, ref) => { + const { } = props; + const [form] = Form.useForm(); + const codeRef = useRef(); + const callbackRef = useRef(() => {}); + const [visible, setVisible] = useState(false); + const handleCancel = useCallback(() => setVisible(false), []); + const handleOk = useCallback(() => { + let formData = form.getFieldValue(); + if (codeRef.current && codeRef.current === formData.outBatch) { + message.error("超产批号不能与原单号一致!"); + } else { + callbackRef.current && callbackRef.current(formData.outBatch); + handleCancel(); + }; + }, [handleCancel, form]); + useImperativeHandle( + ref, + () => ({ + show(code, callback) { + codeRef.current = code; + callbackRef.current = callback; + setVisible(true); + }, + }), + [] + ); + return ( + +
+ + + +
+
+ ); +}); + +// 超量打印 +export const ModelOverPrint = React.forwardRef((props, ref) => { + const {} = props; + const [form, setForm] = useState(""); + const callbackRef = useRef(() => {}); + const [visible, setVisible] = useState(false); + const handleCancel = useCallback(() => setVisible(false), []); + const handleOk = useCallback(() => { + return callbackRef.current(handleCancel); + }, [handleCancel, form]); + useImperativeHandle( + ref, + () => ({ + show: (data, callback) => { + setVisible(true); + setForm(data); + callbackRef.current = callback; + }, + }), + [] + ); + return ( + +

{ form }

+
+ ); +}); \ No newline at end of file diff --git a/components/client/PackagingOperation/hooks.js b/components/client/PackagingOperation/hooks.js new file mode 100644 index 0000000..4a3acfd --- /dev/null +++ b/components/client/PackagingOperation/hooks.js @@ -0,0 +1,199 @@ +import React, { useCallback, useEffect, useState, useRef } from "react"; +import { message } from "antd"; +import { request, getQueryVariable } from "../utils"; + +export const isDev = false; + +let rules = (function () { + let data = []; + let callback = () => {}; + let setData = (res) => { + setData = () => {}; + data = res; + callback({ success: true, data: res }); + }; + return { + setData, + getData: () => data, + loaded: new Promise((resolve, reject) => { + callback = resolve; + }), + }; +})(); + +let getRule = () => { + getRule = () => rules.loaded; + return request({ + method: "get", + url: `${window.config.mesUrl}/console/getpermissions`, + }); +}; + +export const useRule = (code) => { + const [ruleList, setRuleList] = useState(rules.getData()); + useEffect(() => { + getRule().then(({ success, data } = {}) => { + if (success) { + setRuleList(data); + rules.setData(data); + } + }); + }, []); + + if (code) return ruleList.includes(code); + return true; +}; + +export const useUser = () => { + const [data, setData] = useState(); + const getUser = useCallback(() => { + // let user = localStorage.getItem("userInfo"); + // try { + // user = JSON.parse(user) + // } catch (e) { + // console.log(e); + // user = {}; + // } + // console.log("user", user, localStorage.getItem("userInfo")) + // setData(user); + request({ + method: "get", + url: `${window.config.mesUrl}/console/currentinfo`, + }).then(({ success, data } = {}) => { + if (success) { + setData(data); + } else setData(); + }); + }, []) + // useEffect(() => { + // getUser(); + // }, [getUser]); + return [data, getUser]; +}; + +export const useEquipKey = () => { + const [equipKey, setEquipKey] = useState(); + const getEquipKey = useCallback(() => { + let ip = getQueryVariable("ip"); + console.log("ip", ip); + if (!ip || ip === "null") { + message.error("获取 ip 失败!"); + } else { + setEquipKey(ip); + } + }, []); + useEffect(() => { + getEquipKey(); + }, [getEquipKey]); + return { equipKey, getEquipKey }; +}; + +export const useSetting = (equipKey) => { + const [settingData, setSettingData] = useState({}); + + const getSettingData = useCallback(() => { + if (equipKey) { + request({ + method: "post", + url: `${window.config.mesUrl}/prodexec/prodexec/packageEquipment/getPackInfoByKey?equipKey=${equipKey}`, + }).then(({ success, data } = {}) => { + if (success) setSettingData(data || {}); + }); + } else setSettingData({}); + }, [equipKey]); + + useEffect(() => { + getSettingData(); + }, [getSettingData]); + return { settingData, getSettingData }; +}; + +export const useMetaInfo = (equipCode) => { + const [data, setData] = useState({}); // bagNo outBagNo + const timmerRef = useRef(); + const getData = useCallback(() => { + if (timmerRef.current) { + clearTimeout(timmerRef.current); + timmerRef.current = undefined; + } + if (equipCode) { + request({ + method: "post", + url: `${window.config.mesUrl}/prodexec/prodexec/packageEquipment/getEquipRecord?equipCode=${equipCode}`, + }).then(({ success, data, message: msg } = {}) => { + if (success) { + let { status } = data || {}; + // 没有数据,或者数据为中断状态,尝试定时重复请求 + if (!data || status === "INTERRUPT") { + if (!data) { + data = { status: "no_job" }; + message.error("请先分配任务!"); + } + let { reloadTime } = window.config.packagingOperaConfig || {}; + if (reloadTime !== undefined) { + timmerRef.current = setTimeout(getData, reloadTime); + } + } + setData(data); + } else { + message.error(msg); + } + }); + } else setData({}); + }, [equipCode]); + + useEffect(() => { + getData(); + }, [getData]); + return [data, getData]; +}; + +export const useWsInfo = (equipCode, isEdit) => { + const [data, setData] = useState({}); + useEffect(() => { + if (equipCode) { + let url = window.config.edgewsUrl; + if (isDev) { + url = "ws://120.202.38.15:8081/rongtong-edge-application-ws/"; + } + if (!isEdit) { + console.log("link: ", url); + let ws = new WebSocket(url); + ws.onmessage = (evt) => { + let res = evt.data; + // if (isDev) { + // res = { + // czzl: 1.11, + // fczl: 2.22, + // } + // } + if (typeof res === "string") { + try { + res = JSON.parse(res); + } catch (e) { + console.log("ws", res); + res = {}; + } + } + console.log("ws res:", res); + setData(res); + }; + ws.onopen = () => { + if (isDev) { + // 01 02 03 + equipCode = "L5-BZ-02"; + } + console.log("Connection open ... send:", equipCode); + ws.send(equipCode); + }; + } + return () => { + if (!isEdit) { + console.log("ws close:", equipCode); + ws.close(); + } + }; + } + }, [equipCode, isEdit]); + return [data]; +}; diff --git a/components/client/PackagingOperation/index.js b/components/client/PackagingOperation/index.js new file mode 100644 index 0000000..31bcaec --- /dev/null +++ b/components/client/PackagingOperation/index.js @@ -0,0 +1,619 @@ +import React, { + useCallback, + useRef, + useReducer, + useEffect, + useState, + useMemo, +} from "react"; +import { message } from "antd"; +import moment from "moment"; +import Decimal from "decimal.js" +import Row from "../../../components/screen/Row"; +import { FormItem } from "../FormItem"; +import { Button } from "../Button"; +import { + ModelSetting, + ModelBreak, + ModelOverproductionBatch, + ModelFinish, + ModelOverPrint, +} from "./ModalClient"; +import { logo } from "./logo"; +import { request } from "../utils"; +import { + useEquipKey, + useSetting, + useMetaInfo, + useWsInfo, + useUser, + isDev, +} from "./hooks"; + +const headerHeight = 56; + +/** + * url获取token + * 获取用户信息 + * 获取权限信息 + * + * url获取ip + * ip - 设置信息 + * 设置信息 - 产品信息 + * 设置信息 - socket 重量信息 + * + * 任务状态: + * 无任务 待包装 包装中 中断 包装完成 + */ + +function formReducer(state, action) { + let { type, value } = action; + // console.log("action", action); + if (type === "FROM_INIT") return value; + if (type === "FROM_CHANGE") return Object.assign({}, state, value); + return state; +} + +function PackagingOperation(props) { + const ModelSettingRef = useRef(); + const ModelBreakRef = useRef(); + const ModelOverproductionBatchRef = useRef(); + const ModelFinishRef = useRef(); + + const [isEdit, setIsEdit] = useState(false); + + const { equipKey, getEquipKey } = useEquipKey(); + const { settingData, getSettingData } = useSetting(equipKey); + const [metaInfo, getMetaInfo] = useMetaInfo(settingData?.equipCode); + const [wsData] = useWsInfo(settingData?.equipCode, isEdit); + const [user, getUser] = useUser(); + + // 页面表单变化 + const [form, dispatch] = useReducer(formReducer, {}); + const onFormChange = useCallback((type, value) => { + dispatch({ type: "FROM_CHANGE", value: { [type]: value } }); + }, []); + + // 更新当前登录用户 + useEffect(() => { + dispatch({ type: "FROM_CHANGE", value: { userName: user?.userName } }); + }, [user]); + + // 初始化表单 + useEffect(() => { + dispatch({ type: "FROM_INIT", value: metaInfo }); + getUser(); + }, [metaInfo, getUser]); + + // ws更新表单 + useEffect(() => { + let { czzl, fczl } = wsData || {}; + dispatch({ type: "FROM_CHANGE", value: { czzl, fczl } }); + }, [wsData]); + + // 更新净重 + const netWeight = useMemo(() => { + let { bagWeight } = settingData; + let { fczl } = form; + if (isNaN(parseFloat(fczl))) return ""; + if (isNaN(parseFloat(bagWeight))) return ""; + let res = Decimal(fczl).sub(Decimal(bagWeight)).toNumber(); + if (isNaN(res)) return ""; + return res; + }, [form, settingData]); + + // 更新持续时长 + const [durationTime, setDurationTime] = useState(0); + useEffect(() => { + let getTime = () => { + let { + startTime, + status, + recoverDate, + interruptDate, + intrptDuration = 0, + } = metaInfo; + // console.log("metaInfo", metaInfo, {startTime, status, recoverDate, interruptDate, intrptDuration }) + /** + * 无任务 “” + * 待包装 “” + * 包装中 currentDate - startTime - intrptDuration + * 中断 interruptDate - startTime - intrptDuration + * 包装完成 endTime - startTime - intrptDuration + */ + if (["no_job", "TO_BE_PACKAGED"].includes(status)) return ""; + let dif = new Date(startTime).getTime() + intrptDuration * 1000; + if (status === "IN_PACKAGING") return new Date().getTime() - dif; + if (status === "INTERRUPT") + return new Date(interruptDate).getTime() - dif; + if (status === "PACKAGING_COMPLETED") + return new Date(endTime).getTime() - dif; + }; + let timmer; + let run = () => { + setDurationTime(getTime()); + timmer = setTimeout(() => { + run(); + }, 1000); + }; + run(); + return () => { + clearTimeout(timmer); + }; + }, [metaInfo]); + const durationTimeStr = useMemo(() => { + let time = moment.duration(durationTime); + let res = ""; + let sec = time.seconds(); + let min = time.minutes(); + let hor = time.hours(); + let day = time.days(); + let m = time.months(); + let y = time.years(); + let flag = false; + let arr = [y, m, day, hor, min, sec]; + let unt = ["年", "月", "日", "时", "分", "秒"]; + arr.forEach((val, index) => { + if (val) { + flag = true; + res += val + unt[index]; + } else if (flag) { + res += 0 + unt[index]; + } + }); + return res; + }, [durationTime]); + + // 包装袋数 + const [realBagNo, setRealBagNo] = useState(1); + useEffect(() => { + let { bagNo, outBagNo, outBatch } = form; + let res = 0; + if (outBatch) { + res = parseFloat(outBagNo) + 1; + } else { + res = parseFloat(bagNo) + 1; + } + setRealBagNo(isNaN(res) ? 0 : res); + }, [form.bagNo, form.outBagNo, form.outBatch]); + + // 设备状态 + const equipStatus = useMemo(() => { + if (form.status === "no_job") return "无任务"; + if (form.status === "INTERRUPT") return "中断"; + if (form.status === "TO_BE_PACKAGED") return "待包装"; + if (form.status === "IN_PACKAGING") return "包装中"; + if (form.status === "PACKAGING_COMPLETED") return "包装完成"; + return "-"; + }, [form.status]); + + // 按钮启用禁用 + const btnEnable = useMemo(() => { + /** + * 超产批号 ? + * 设置 - 权限 + * 历史记录 - 非无任务 + * 开始 - 待包装 + * 打印 - 包装中 + * 结束 - 包装中 中断 + * 中断 - 包装中 + * 编辑 - 权限 包装中 + * 编辑完成 - 权限 包装中 + */ + return { + overPro: true, + setting: true, + history: ["中断", "待包装", "包装中", "包装完成"].includes(equipStatus), + start: ["待包装"].includes(equipStatus), + print: ["包装中"].includes(equipStatus), + finish: ["包装中", "中断"].includes(equipStatus), + break: ["包装中"].includes(equipStatus), + edit: ["包装中"].includes(equipStatus), + }; + }, [equipStatus]); + + // 退出 + const handleCancel = useCallback(() => { + window.parent.postMessage({ type: "logOut" }, window.config.mesHost); + }, []); + + // 设置 + const handleSetting = useCallback(() => { + if (equipKey) { + let settingRes = Object.assign({}, settingData || {}, { equipKey }); + ModelSettingRef.current.show(settingRes, getSettingData); + } + }, [equipKey, getSettingData, settingData]); + + // 超产批号 + const handleOverproduct = useCallback(() => { + ModelOverproductionBatchRef.current.show( + metaInfo.workOrderCode, + (outBatch) => { + onFormChange("outBatch", outBatch); + } + ); + }, [metaInfo.workOrderCode, onFormChange]); + + // 历史记录 + const handleHistory = useCallback(() => { + window.parent.postMessage( + { type: "history", data: metaInfo }, + window.config.mesHost + ); + }, [metaInfo]); + + // 开始 + const handleStart = useCallback(() => { + if (metaInfo.gid) + return new Promise((resolve, reject) => { + request({ + url: `${window.config.mesUrl}/prodexec/prodexec/packageEquipment/startPack`, + bodys: { gid: metaInfo.gid }, + }) + .then((res) => { + if (res.success) { + message.success("开始成功!"); + getMetaInfo(); + } + }) + .finally(() => { + resolve(); + }); + }); + }, [metaInfo.gid, getMetaInfo]); + + // 打印 + const ModelOverPrintRef = useRef(); + const [isGT, setIsGT] = useState(false) + const handlePrint = useCallback((GT, callback) => { + if (GT === undefined) GT = isGT; + let { gid: recordGid } = metaInfo; + let { gid: setInfoGid, bagWeight } = settingData; + let { + fczl, + czzl, + outBatch, + materialCode, + productLevel, + startTime, + materialName, + workOrderCode, + } = form; + // collectValue 页面可见数据 除自定义字段外的 对象json 字符串 + let collectValue = { + materialCode, + productLevel, + startTime, + materialName, + workOrderCode, + czzl, + fczl, + bagWeight, + yetWeight: netWeight, + bagNo: realBagNo, + }; + let data = { + recordGid, + mark: GT? "GT": undefined, + fczl, + bagNo: realBagNo, + setInfoGid, + yetWeight: netWeight, + outBatch, + collectValue: JSON.stringify(collectValue), + }; + if (recordGid) + return new Promise((resolve, reject) => { + request({ + url: `${window.config.mesUrl}/prodexec/prodexec/packageEquipment/print`, + bodys: data, + }) + .then((res) => { + // if (isDev) { + // if (res.message.indexOf('打印命令发送失败') > -1) { + // res = { + // success: false, + // data: "GT", + // message: "message text" + // } + // } + // } + if (res && res.success) { + message.success("打印成功!"); + setRealBagNo(parseFloat(realBagNo) + 1); + } else { + if (res?.data === "GT") { + ModelOverPrintRef.current.show(res.message, (close) => { + setIsGT(true); + return handlePrint(true, close); + }) + }else message.error(res?.message); + } + }) + .finally(() => { + resolve(); + callback && callback(); + }); + }); + }, [metaInfo, settingData, form, netWeight, realBagNo, isGT]); + + // 结束 + const handleFinish = useCallback(() => { + ModelFinishRef.current.show({ gid: metaInfo.gid }, getMetaInfo); + }, [metaInfo.gid, getMetaInfo]); + + // 中断 + const handleBreak = useCallback(() => { + ModelBreakRef.current.show({ gid: metaInfo.gid }, getMetaInfo); + }, [metaInfo.gid, getMetaInfo]); + + // 编辑 + const handleEdit = useCallback(() => { + // console.log("handleEdit"); + if (isEdit) { + setIsEdit(false); + } else { + setIsEdit(true); + } + }, [isEdit]); + + return ( +
+
+
+ logo +
+
+
+ 退出 +
+
+
+
+
+
+ 产品信息 +
+
+
+
+
{equipStatus}
+
+ + + + + + + + + + + + + + + + + + + + +
+
+
+
+
+ 生产信息 +
+
+
+
+ + + onFormChange("czzl", val)} + /> + + + onFormChange("fczl", val)} + /> + + + + + + + + + + + + setRealBagNo(val)} + /> + + +
+
+
+
+
+ +
+ ); +} + +export default PackagingOperation; diff --git a/components/client/PackagingOperation/logo.js b/components/client/PackagingOperation/logo.js new file mode 100644 index 0000000..e251606 --- /dev/null +++ b/components/client/PackagingOperation/logo.js @@ -0,0 +1,2 @@ +export const logo = +""; diff --git a/components/client/utils.js b/components/client/utils.js new file mode 100644 index 0000000..31a4dc9 --- /dev/null +++ b/components/client/utils.js @@ -0,0 +1,17 @@ +import {default as reportRequest} from "../../reportUtils/request" + + +export const getQueryVariable = (variable) => { + let query = window.location.search.substring(1); + let vars = query.split("&"); + for (let i = 0; i < vars.length; i++) { + let pair = vars[i].split("="); + if (pair[0] == variable) { return pair[1]; } + } + return (false); +} + +export const request = (params) => { + let token = getQueryVariable("token"); + return reportRequest(params, token) +} \ No newline at end of file diff --git a/components/screen/Card/index.js b/components/screen/Card/index.js new file mode 100644 index 0000000..5a16961 --- /dev/null +++ b/components/screen/Card/index.js @@ -0,0 +1,97 @@ +import React from 'react'; + +const titleSize = 28; + +function Card(props) { + const { + full = true, + title, + titleSpace, + overVisible, + padding, + withBorder = true, + withBg = "rgba(6,36,109,0.80)" + } = props; + return ( +
+ {title &&
+ {title} +
} +
+
+ {React.Children.map(props.children, item => item)} +
+
+ +
+ ); +} + +export default Card; \ No newline at end of file diff --git a/components/screen/Chart/echarts-liquidfill.js b/components/screen/Chart/echarts-liquidfill.js new file mode 100644 index 0000000..f7f377d --- /dev/null +++ b/components/screen/Chart/echarts-liquidfill.js @@ -0,0 +1,1688 @@ +(function webpackUniversalModuleDefinition(root, factory) { + if(typeof exports === 'object' && typeof module === 'object') + module.exports = factory(require("echarts")); + else if(typeof define === 'function' && define.amd) + define(["echarts"], factory); + else if(typeof exports === 'object') + exports["echarts-liquidfill"] = factory(require("echarts")); + else + root["echarts-liquidfill"] = factory(root["echarts"]); +})(this, function(__WEBPACK_EXTERNAL_MODULE_echarts_lib_echarts__) { +return /******/ (() => { // webpackBootstrap +/******/ "use strict"; +/******/ var __webpack_modules__ = ({ + +/***/ "./index.js": +/*!******************************!*\ + !*** ./index.js + 6 modules ***! + \******************************/ +/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { + +// ESM COMPAT FLAG +__webpack_require__.r(__webpack_exports__); + +// EXTERNAL MODULE: external "echarts" +var external_echarts_ = __webpack_require__("echarts/lib/echarts"); +;// CONCATENATED MODULE: ./src/liquidFillSeries.js + + +external_echarts_.extendSeriesModel({ + + type: 'series.liquidFill', + + optionUpdated: function () { + var option = this.option; + option.gridSize = Math.max(Math.floor(option.gridSize), 4); + }, + + getInitialData: function (option, ecModel) { + var dimensions = external_echarts_.helper.createDimensions(option.data, { + coordDimensions: ['value'] + }); + var list = new external_echarts_.List(dimensions, this); + list.initData(option.data); + return list; + }, + + defaultOption: { + color: ['#294D99', '#156ACF', '#1598ED', '#45BDFF'], + center: ['50%', '50%'], + radius: '50%', + amplitude: '8%', + waveLength: '80%', + phase: 'auto', + period: 'auto', + direction: 'right', + shape: 'circle', + + waveAnimation: true, + animationEasing: 'linear', + animationEasingUpdate: 'linear', + animationDuration: 2000, + animationDurationUpdate: 1000, + + outline: { + show: true, + borderDistance: 8, + itemStyle: { + color: 'none', + borderColor: '#294D99', + borderWidth: 8, + shadowBlur: 20, + shadowColor: 'rgba(0, 0, 0, 0.25)' + } + }, + + backgroundStyle: { + color: '#E3F7FF' + }, + + itemStyle: { + opacity: 0.95, + shadowBlur: 50, + shadowColor: 'rgba(0, 0, 0, 0.4)' + }, + + label: { + show: true, + color: '#294D99', + insideColor: '#fff', + fontSize: 50, + fontWeight: 'bold', + + align: 'center', + baseline: 'middle', + position: 'inside' + }, + + emphasis: { + itemStyle: { + opacity: 0.8 + } + } + } +}); + +;// CONCATENATED MODULE: ./node_modules/zrender/lib/core/util.js +var BUILTIN_OBJECT = { + '[object Function]': true, + '[object RegExp]': true, + '[object Date]': true, + '[object Error]': true, + '[object CanvasGradient]': true, + '[object CanvasPattern]': true, + '[object Image]': true, + '[object Canvas]': true +}; +var TYPED_ARRAY = { + '[object Int8Array]': true, + '[object Uint8Array]': true, + '[object Uint8ClampedArray]': true, + '[object Int16Array]': true, + '[object Uint16Array]': true, + '[object Int32Array]': true, + '[object Uint32Array]': true, + '[object Float32Array]': true, + '[object Float64Array]': true +}; +var objToString = Object.prototype.toString; +var arrayProto = Array.prototype; +var nativeForEach = arrayProto.forEach; +var nativeFilter = arrayProto.filter; +var nativeSlice = arrayProto.slice; +var nativeMap = arrayProto.map; +var ctorFunction = function () { }.constructor; +var protoFunction = ctorFunction ? ctorFunction.prototype : null; +var methods = {}; +function $override(name, fn) { + methods[name] = fn; +} +var idStart = 0x0907; +function guid() { + return idStart++; +} +function logError() { + var args = []; + for (var _i = 0; _i < arguments.length; _i++) { + args[_i] = arguments[_i]; + } + if (typeof console !== 'undefined') { + console.error.apply(console, args); + } +} +function clone(source) { + if (source == null || typeof source !== 'object') { + return source; + } + var result = source; + var typeStr = objToString.call(source); + if (typeStr === '[object Array]') { + if (!isPrimitive(source)) { + result = []; + for (var i = 0, len = source.length; i < len; i++) { + result[i] = clone(source[i]); + } + } + } + else if (TYPED_ARRAY[typeStr]) { + if (!isPrimitive(source)) { + var Ctor = source.constructor; + if (Ctor.from) { + result = Ctor.from(source); + } + else { + result = new Ctor(source.length); + for (var i = 0, len = source.length; i < len; i++) { + result[i] = clone(source[i]); + } + } + } + } + else if (!BUILTIN_OBJECT[typeStr] && !isPrimitive(source) && !isDom(source)) { + result = {}; + for (var key in source) { + if (source.hasOwnProperty(key)) { + result[key] = clone(source[key]); + } + } + } + return result; +} +function merge(target, source, overwrite) { + if (!isObject(source) || !isObject(target)) { + return overwrite ? clone(source) : target; + } + for (var key in source) { + if (source.hasOwnProperty(key)) { + var targetProp = target[key]; + var sourceProp = source[key]; + if (isObject(sourceProp) + && isObject(targetProp) + && !isArray(sourceProp) + && !isArray(targetProp) + && !isDom(sourceProp) + && !isDom(targetProp) + && !isBuiltInObject(sourceProp) + && !isBuiltInObject(targetProp) + && !isPrimitive(sourceProp) + && !isPrimitive(targetProp)) { + merge(targetProp, sourceProp, overwrite); + } + else if (overwrite || !(key in target)) { + target[key] = clone(source[key]); + } + } + } + return target; +} +function mergeAll(targetAndSources, overwrite) { + var result = targetAndSources[0]; + for (var i = 1, len = targetAndSources.length; i < len; i++) { + result = merge(result, targetAndSources[i], overwrite); + } + return result; +} +function extend(target, source) { + if (Object.assign) { + Object.assign(target, source); + } + else { + for (var key in source) { + if (source.hasOwnProperty(key)) { + target[key] = source[key]; + } + } + } + return target; +} +function defaults(target, source, overlay) { + var keysArr = keys(source); + for (var i = 0; i < keysArr.length; i++) { + var key = keysArr[i]; + if ((overlay ? source[key] != null : target[key] == null)) { + target[key] = source[key]; + } + } + return target; +} +var createCanvas = function () { + return methods.createCanvas(); +}; +methods.createCanvas = function () { + return document.createElement('canvas'); +}; +function indexOf(array, value) { + if (array) { + if (array.indexOf) { + return array.indexOf(value); + } + for (var i = 0, len = array.length; i < len; i++) { + if (array[i] === value) { + return i; + } + } + } + return -1; +} +function inherits(clazz, baseClazz) { + var clazzPrototype = clazz.prototype; + function F() { } + F.prototype = baseClazz.prototype; + clazz.prototype = new F(); + for (var prop in clazzPrototype) { + if (clazzPrototype.hasOwnProperty(prop)) { + clazz.prototype[prop] = clazzPrototype[prop]; + } + } + clazz.prototype.constructor = clazz; + clazz.superClass = baseClazz; +} +function mixin(target, source, override) { + target = 'prototype' in target ? target.prototype : target; + source = 'prototype' in source ? source.prototype : source; + if (Object.getOwnPropertyNames) { + var keyList = Object.getOwnPropertyNames(source); + for (var i = 0; i < keyList.length; i++) { + var key = keyList[i]; + if (key !== 'constructor') { + if ((override ? source[key] != null : target[key] == null)) { + target[key] = source[key]; + } + } + } + } + else { + defaults(target, source, override); + } +} +function isArrayLike(data) { + if (!data) { + return false; + } + if (typeof data === 'string') { + return false; + } + return typeof data.length === 'number'; +} +function each(arr, cb, context) { + if (!(arr && cb)) { + return; + } + if (arr.forEach && arr.forEach === nativeForEach) { + arr.forEach(cb, context); + } + else if (arr.length === +arr.length) { + for (var i = 0, len = arr.length; i < len; i++) { + cb.call(context, arr[i], i, arr); + } + } + else { + for (var key in arr) { + if (arr.hasOwnProperty(key)) { + cb.call(context, arr[key], key, arr); + } + } + } +} +function map(arr, cb, context) { + if (!arr) { + return []; + } + if (!cb) { + return slice(arr); + } + if (arr.map && arr.map === nativeMap) { + return arr.map(cb, context); + } + else { + var result = []; + for (var i = 0, len = arr.length; i < len; i++) { + result.push(cb.call(context, arr[i], i, arr)); + } + return result; + } +} +function reduce(arr, cb, memo, context) { + if (!(arr && cb)) { + return; + } + for (var i = 0, len = arr.length; i < len; i++) { + memo = cb.call(context, memo, arr[i], i, arr); + } + return memo; +} +function filter(arr, cb, context) { + if (!arr) { + return []; + } + if (!cb) { + return slice(arr); + } + if (arr.filter && arr.filter === nativeFilter) { + return arr.filter(cb, context); + } + else { + var result = []; + for (var i = 0, len = arr.length; i < len; i++) { + if (cb.call(context, arr[i], i, arr)) { + result.push(arr[i]); + } + } + return result; + } +} +function find(arr, cb, context) { + if (!(arr && cb)) { + return; + } + for (var i = 0, len = arr.length; i < len; i++) { + if (cb.call(context, arr[i], i, arr)) { + return arr[i]; + } + } +} +function keys(obj) { + if (!obj) { + return []; + } + if (Object.keys) { + return Object.keys(obj); + } + var keyList = []; + for (var key in obj) { + if (obj.hasOwnProperty(key)) { + keyList.push(key); + } + } + return keyList; +} +function bindPolyfill(func, context) { + var args = []; + for (var _i = 2; _i < arguments.length; _i++) { + args[_i - 2] = arguments[_i]; + } + return function () { + return func.apply(context, args.concat(nativeSlice.call(arguments))); + }; +} +var bind = (protoFunction && isFunction(protoFunction.bind)) + ? protoFunction.call.bind(protoFunction.bind) + : bindPolyfill; +function curry(func) { + var args = []; + for (var _i = 1; _i < arguments.length; _i++) { + args[_i - 1] = arguments[_i]; + } + return function () { + return func.apply(this, args.concat(nativeSlice.call(arguments))); + }; +} + +function isArray(value) { + if (Array.isArray) { + return Array.isArray(value); + } + return objToString.call(value) === '[object Array]'; +} +function isFunction(value) { + return typeof value === 'function'; +} +function isString(value) { + return typeof value === 'string'; +} +function isStringSafe(value) { + return objToString.call(value) === '[object String]'; +} +function isNumber(value) { + return typeof value === 'number'; +} +function isObject(value) { + var type = typeof value; + return type === 'function' || (!!value && type === 'object'); +} +function isBuiltInObject(value) { + return !!BUILTIN_OBJECT[objToString.call(value)]; +} +function isTypedArray(value) { + return !!TYPED_ARRAY[objToString.call(value)]; +} +function isDom(value) { + return typeof value === 'object' + && typeof value.nodeType === 'number' + && typeof value.ownerDocument === 'object'; +} +function isGradientObject(value) { + return value.colorStops != null; +} +function isPatternObject(value) { + return value.image != null; +} +function isRegExp(value) { + return objToString.call(value) === '[object RegExp]'; +} +function eqNaN(value) { + return value !== value; +} +function retrieve() { + var args = []; + for (var _i = 0; _i < arguments.length; _i++) { + args[_i] = arguments[_i]; + } + for (var i = 0, len = args.length; i < len; i++) { + if (args[i] != null) { + return args[i]; + } + } +} +function retrieve2(value0, value1) { + return value0 != null + ? value0 + : value1; +} +function retrieve3(value0, value1, value2) { + return value0 != null + ? value0 + : value1 != null + ? value1 + : value2; +} +function slice(arr) { + var args = []; + for (var _i = 1; _i < arguments.length; _i++) { + args[_i - 1] = arguments[_i]; + } + return nativeSlice.apply(arr, args); +} +function normalizeCssArray(val) { + if (typeof (val) === 'number') { + return [val, val, val, val]; + } + var len = val.length; + if (len === 2) { + return [val[0], val[1], val[0], val[1]]; + } + else if (len === 3) { + return [val[0], val[1], val[2], val[1]]; + } + return val; +} +function assert(condition, message) { + if (!condition) { + throw new Error(message); + } +} +function trim(str) { + if (str == null) { + return null; + } + else if (typeof str.trim === 'function') { + return str.trim(); + } + else { + return str.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, ''); + } +} +var primitiveKey = '__ec_primitive__'; +function setAsPrimitive(obj) { + obj[primitiveKey] = true; +} +function isPrimitive(obj) { + return obj[primitiveKey]; +} +var HashMap = (function () { + function HashMap(obj) { + this.data = {}; + var isArr = isArray(obj); + this.data = {}; + var thisMap = this; + (obj instanceof HashMap) + ? obj.each(visit) + : (obj && each(obj, visit)); + function visit(value, key) { + isArr ? thisMap.set(value, key) : thisMap.set(key, value); + } + } + HashMap.prototype.get = function (key) { + return this.data.hasOwnProperty(key) ? this.data[key] : null; + }; + HashMap.prototype.set = function (key, value) { + return (this.data[key] = value); + }; + HashMap.prototype.each = function (cb, context) { + for (var key in this.data) { + if (this.data.hasOwnProperty(key)) { + cb.call(context, this.data[key], key); + } + } + }; + HashMap.prototype.keys = function () { + return keys(this.data); + }; + HashMap.prototype.removeKey = function (key) { + delete this.data[key]; + }; + return HashMap; +}()); + +function createHashMap(obj) { + return new HashMap(obj); +} +function concatArray(a, b) { + var newArray = new a.constructor(a.length + b.length); + for (var i = 0; i < a.length; i++) { + newArray[i] = a[i]; + } + var offset = a.length; + for (var i = 0; i < b.length; i++) { + newArray[i + offset] = b[i]; + } + return newArray; +} +function createObject(proto, properties) { + var obj; + if (Object.create) { + obj = Object.create(proto); + } + else { + var StyleCtor = function () { }; + StyleCtor.prototype = proto; + obj = new StyleCtor(); + } + if (properties) { + extend(obj, properties); + } + return obj; +} +function hasOwn(own, prop) { + return own.hasOwnProperty(prop); +} +function noop() { } + +;// CONCATENATED MODULE: ./node_modules/echarts/lib/util/number.js + +/* +* Licensed to the Apache Software Foundation (ASF) under one +* or more contributor license agreements. See the NOTICE file +* distributed with this work for additional information +* regarding copyright ownership. The ASF licenses this file +* to you under the Apache License, Version 2.0 (the +* "License"); you may not use this file except in compliance +* with the License. You may obtain a copy of the License at +* +* http://www.apache.org/licenses/LICENSE-2.0 +* +* Unless required by applicable law or agreed to in writing, +* software distributed under the License is distributed on an +* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +* KIND, either express or implied. See the License for the +* specific language governing permissions and limitations +* under the License. +*/ + + +/** + * AUTO-GENERATED FILE. DO NOT MODIFY. + */ + + +var RADIAN_EPSILON = 1e-4; + +function _trim(str) { + return str.replace(/^\s+|\s+$/g, ''); +} + +function linearMap(val, domain, range, clamp) { + var subDomain = domain[1] - domain[0]; + var subRange = range[1] - range[0]; + + if (subDomain === 0) { + return subRange === 0 ? range[0] : (range[0] + range[1]) / 2; + } + + if (clamp) { + if (subDomain > 0) { + if (val <= domain[0]) { + return range[0]; + } else if (val >= domain[1]) { + return range[1]; + } + } else { + if (val >= domain[0]) { + return range[0]; + } else if (val <= domain[1]) { + return range[1]; + } + } + } else { + if (val === domain[0]) { + return range[0]; + } + + if (val === domain[1]) { + return range[1]; + } + } + + return (val - domain[0]) / subDomain * subRange + range[0]; +} +function parsePercent(percent, all) { + switch (percent) { + case 'center': + case 'middle': + percent = '50%'; + break; + + case 'left': + case 'top': + percent = '0%'; + break; + + case 'right': + case 'bottom': + percent = '100%'; + break; + } + + if (typeof percent === 'string') { + if (_trim(percent).match(/%$/)) { + return parseFloat(percent) / 100 * all; + } + + return parseFloat(percent); + } + + return percent == null ? NaN : +percent; +} +function round(x, precision, returnStr) { + if (precision == null) { + precision = 10; + } + + precision = Math.min(Math.max(0, precision), 20); + x = (+x).toFixed(precision); + return returnStr ? x : +x; +} +function asc(arr) { + arr.sort(function (a, b) { + return a - b; + }); + return arr; +} +function getPrecision(val) { + val = +val; + + if (isNaN(val)) { + return 0; + } + + var e = 1; + var count = 0; + + while (Math.round(val * e) / e !== val) { + e *= 10; + count++; + } + + return count; +} +function getPrecisionSafe(val) { + var str = val.toString(); + var eIndex = str.indexOf('e'); + + if (eIndex > 0) { + var precision = +str.slice(eIndex + 1); + return precision < 0 ? -precision : 0; + } else { + var dotIndex = str.indexOf('.'); + return dotIndex < 0 ? 0 : str.length - 1 - dotIndex; + } +} +function getPixelPrecision(dataExtent, pixelExtent) { + var log = Math.log; + var LN10 = Math.LN10; + var dataQuantity = Math.floor(log(dataExtent[1] - dataExtent[0]) / LN10); + var sizeQuantity = Math.round(log(Math.abs(pixelExtent[1] - pixelExtent[0])) / LN10); + var precision = Math.min(Math.max(-dataQuantity + sizeQuantity, 0), 20); + return !isFinite(precision) ? 20 : precision; +} +function getPercentWithPrecision(valueList, idx, precision) { + if (!valueList[idx]) { + return 0; + } + + var sum = reduce(valueList, function (acc, val) { + return acc + (isNaN(val) ? 0 : val); + }, 0); + + if (sum === 0) { + return 0; + } + + var digits = Math.pow(10, precision); + var votesPerQuota = map(valueList, function (val) { + return (isNaN(val) ? 0 : val) / sum * digits * 100; + }); + var targetSeats = digits * 100; + var seats = map(votesPerQuota, function (votes) { + return Math.floor(votes); + }); + var currentSum = reduce(seats, function (acc, val) { + return acc + val; + }, 0); + var remainder = map(votesPerQuota, function (votes, idx) { + return votes - seats[idx]; + }); + + while (currentSum < targetSeats) { + var max = Number.NEGATIVE_INFINITY; + var maxId = null; + + for (var i = 0, len = remainder.length; i < len; ++i) { + if (remainder[i] > max) { + max = remainder[i]; + maxId = i; + } + } + + ++seats[maxId]; + remainder[maxId] = 0; + ++currentSum; + } + + return seats[idx] / digits; +} +var MAX_SAFE_INTEGER = 9007199254740991; +function remRadian(radian) { + var pi2 = Math.PI * 2; + return (radian % pi2 + pi2) % pi2; +} +function isRadianAroundZero(val) { + return val > -RADIAN_EPSILON && val < RADIAN_EPSILON; +} +var TIME_REG = /^(?:(\d{4})(?:[-\/](\d{1,2})(?:[-\/](\d{1,2})(?:[T ](\d{1,2})(?::(\d{1,2})(?::(\d{1,2})(?:[.,](\d+))?)?)?(Z|[\+\-]\d\d:?\d\d)?)?)?)?)?$/; +function parseDate(value) { + if (value instanceof Date) { + return value; + } else if (typeof value === 'string') { + var match = TIME_REG.exec(value); + + if (!match) { + return new Date(NaN); + } + + if (!match[8]) { + return new Date(+match[1], +(match[2] || 1) - 1, +match[3] || 1, +match[4] || 0, +(match[5] || 0), +match[6] || 0, +match[7] || 0); + } else { + var hour = +match[4] || 0; + + if (match[8].toUpperCase() !== 'Z') { + hour -= +match[8].slice(0, 3); + } + + return new Date(Date.UTC(+match[1], +(match[2] || 1) - 1, +match[3] || 1, hour, +(match[5] || 0), +match[6] || 0, +match[7] || 0)); + } + } else if (value == null) { + return new Date(NaN); + } + + return new Date(Math.round(value)); +} +function quantity(val) { + return Math.pow(10, quantityExponent(val)); +} +function quantityExponent(val) { + if (val === 0) { + return 0; + } + + var exp = Math.floor(Math.log(val) / Math.LN10); + + if (val / Math.pow(10, exp) >= 10) { + exp++; + } + + return exp; +} +function nice(val, round) { + var exponent = quantityExponent(val); + var exp10 = Math.pow(10, exponent); + var f = val / exp10; + var nf; + + if (round) { + if (f < 1.5) { + nf = 1; + } else if (f < 2.5) { + nf = 2; + } else if (f < 4) { + nf = 3; + } else if (f < 7) { + nf = 5; + } else { + nf = 10; + } + } else { + if (f < 1) { + nf = 1; + } else if (f < 2) { + nf = 2; + } else if (f < 3) { + nf = 3; + } else if (f < 5) { + nf = 5; + } else { + nf = 10; + } + } + + val = nf * exp10; + return exponent >= -20 ? +val.toFixed(exponent < 0 ? -exponent : 0) : val; +} +function quantile(ascArr, p) { + var H = (ascArr.length - 1) * p + 1; + var h = Math.floor(H); + var v = +ascArr[h - 1]; + var e = H - h; + return e ? v + e * (ascArr[h] - v) : v; +} +function reformIntervals(list) { + list.sort(function (a, b) { + return littleThan(a, b, 0) ? -1 : 1; + }); + var curr = -Infinity; + var currClose = 1; + + for (var i = 0; i < list.length;) { + var interval = list[i].interval; + var close_1 = list[i].close; + + for (var lg = 0; lg < 2; lg++) { + if (interval[lg] <= curr) { + interval[lg] = curr; + close_1[lg] = !lg ? 1 - currClose : 1; + } + + curr = interval[lg]; + currClose = close_1[lg]; + } + + if (interval[0] === interval[1] && close_1[0] * close_1[1] !== 1) { + list.splice(i, 1); + } else { + i++; + } + } + + return list; + + function littleThan(a, b, lg) { + return a.interval[lg] < b.interval[lg] || a.interval[lg] === b.interval[lg] && (a.close[lg] - b.close[lg] === (!lg ? 1 : -1) || !lg && littleThan(a, b, 1)); + } +} +function numericToNumber(val) { + var valFloat = parseFloat(val); + return valFloat == val && (valFloat !== 0 || typeof val !== 'string' || val.indexOf('x') <= 0) ? valFloat : NaN; +} +function isNumeric(val) { + return !isNaN(numericToNumber(val)); +} +function getRandomIdBase() { + return Math.round(Math.random() * 9); +} +function getGreatestCommonDividor(a, b) { + if (b === 0) { + return a; + } + + return getGreatestCommonDividor(b, a % b); +} +function getLeastCommonMultiple(a, b) { + if (a == null) { + return b; + } + + if (b == null) { + return a; + } + + return a * b / getGreatestCommonDividor(a, b); +} +;// CONCATENATED MODULE: ./src/liquidFillShape.js + + +/* harmony default export */ const liquidFillShape = (external_echarts_.graphic.extendShape({ + type: 'ec-liquid-fill', + + shape: { + waveLength: 0, + radius: 0, + radiusY: 0, + cx: 0, + cy: 0, + waterLevel: 0, + amplitude: 0, + phase: 0, + inverse: false + }, + + buildPath: function (ctx, shape) { + if (shape.radiusY == null) { + shape.radiusY = shape.radius; + } + + /** + * We define a sine wave having 4 waves, and make sure at least 8 curves + * is drawn. Otherwise, it may cause blank area for some waves when + * wave length is large enough. + */ + var curves = Math.max( + Math.ceil(2 * shape.radius / shape.waveLength * 4) * 2, + 8 + ); + + // map phase to [-Math.PI * 2, 0] + while (shape.phase < -Math.PI * 2) { + shape.phase += Math.PI * 2; + } + while (shape.phase > 0) { + shape.phase -= Math.PI * 2; + } + var phase = shape.phase / Math.PI / 2 * shape.waveLength; + + var left = shape.cx - shape.radius + phase - shape.radius * 2; + + /** + * top-left corner as start point + * + * draws this point + * | + * \|/ + * ~~~~~~~~ + * | | + * +------+ + */ + ctx.moveTo(left, shape.waterLevel); + + /** + * top wave + * + * ~~~~~~~~ <- draws this sine wave + * | | + * +------+ + */ + var waveRight = 0; + for (var c = 0; c < curves; ++c) { + var stage = c % 4; + var pos = getWaterPositions(c * shape.waveLength / 4, stage, + shape.waveLength, shape.amplitude); + ctx.bezierCurveTo(pos[0][0] + left, -pos[0][1] + shape.waterLevel, + pos[1][0] + left, -pos[1][1] + shape.waterLevel, + pos[2][0] + left, -pos[2][1] + shape.waterLevel); + + if (c === curves - 1) { + waveRight = pos[2][0]; + } + } + + if (shape.inverse) { + /** + * top-right corner + * 2. draws this line + * | + * +------+ + * 3. draws this line -> | | <- 1. draws this line + * ~~~~~~~~ + */ + ctx.lineTo(waveRight + left, shape.cy - shape.radiusY); + ctx.lineTo(left, shape.cy - shape.radiusY); + ctx.lineTo(left, shape.waterLevel); + } + else { + /** + * top-right corner + * + * ~~~~~~~~ + * 3. draws this line -> | | <- 1. draws this line + * +------+ + * ^ + * | + * 2. draws this line + */ + ctx.lineTo(waveRight + left, shape.cy + shape.radiusY); + ctx.lineTo(left, shape.cy + shape.radiusY); + ctx.lineTo(left, shape.waterLevel); + } + + ctx.closePath(); + } +})); + + + +/** + * Using Bezier curves to fit sine wave. + * There is 4 control points for each curve of wave, + * which is at 1/4 wave length of the sine wave. + * + * The control points for a wave from (a) to (d) are a-b-c-d: + * c *----* d + * b * + * | + * ... a * .................. + * + * whose positions are a: (0, 0), b: (0.5, 0.5), c: (1, 1), d: (PI / 2, 1) + * + * @param {number} x x position of the left-most point (a) + * @param {number} stage 0-3, stating which part of the wave it is + * @param {number} waveLength wave length of the sine wave + * @param {number} amplitude wave amplitude + */ +function getWaterPositions(x, stage, waveLength, amplitude) { + if (stage === 0) { + return [ + [x + 1 / 2 * waveLength / Math.PI / 2, amplitude / 2], + [x + 1 / 2 * waveLength / Math.PI, amplitude], + [x + waveLength / 4, amplitude] + ]; + } + else if (stage === 1) { + return [ + [x + 1 / 2 * waveLength / Math.PI / 2 * (Math.PI - 2), + amplitude], + [x + 1 / 2 * waveLength / Math.PI / 2 * (Math.PI - 1), + amplitude / 2], + [x + waveLength / 4, 0] + ] + } + else if (stage === 2) { + return [ + [x + 1 / 2 * waveLength / Math.PI / 2, -amplitude / 2], + [x + 1 / 2 * waveLength / Math.PI, -amplitude], + [x + waveLength / 4, -amplitude] + ] + } + else { + return [ + [x + 1 / 2 * waveLength / Math.PI / 2 * (Math.PI - 2), + -amplitude], + [x + 1 / 2 * waveLength / Math.PI / 2 * (Math.PI - 1), + -amplitude / 2], + [x + waveLength / 4, 0] + ] + } +} + +;// CONCATENATED MODULE: ./src/liquidFillView.js + + + + +var liquidFillView_parsePercent = parsePercent; + +function isPathSymbol(symbol) { + return symbol && symbol.indexOf('path://') === 0 +} + +external_echarts_.extendChartView({ + + type: 'liquidFill', + + render: function (seriesModel, ecModel, api) { + var self = this; + var group = this.group; + group.removeAll(); + + var data = seriesModel.getData(); + + var itemModel = data.getItemModel(0); + + var center = itemModel.get('center'); + var radius = itemModel.get('radius'); + + var width = api.getWidth(); + var height = api.getHeight(); + var size = Math.min(width, height); + // itemStyle + var outlineDistance = 0; + var outlineBorderWidth = 0; + var showOutline = seriesModel.get('outline.show'); + + if (showOutline) { + outlineDistance = seriesModel.get('outline.borderDistance'); + outlineBorderWidth = liquidFillView_parsePercent( + seriesModel.get('outline.itemStyle.borderWidth'), size + ); + } + + var cx = liquidFillView_parsePercent(center[0], width); + var cy = liquidFillView_parsePercent(center[1], height); + + var outterRadius; + var innerRadius; + var paddingRadius; + + var isFillContainer = false; + + var symbol = seriesModel.get('shape'); + if (symbol === 'container') { + // a shape that fully fills the container + isFillContainer = true; + + outterRadius = [ + width / 2, + height / 2 + ]; + innerRadius = [ + outterRadius[0] - outlineBorderWidth / 2, + outterRadius[1] - outlineBorderWidth / 2 + ]; + paddingRadius = [ + liquidFillView_parsePercent(outlineDistance, width), + liquidFillView_parsePercent(outlineDistance, height) + ]; + + radius = [ + Math.max(innerRadius[0] - paddingRadius[0], 0), + Math.max(innerRadius[1] - paddingRadius[1], 0) + ]; + } + else { + outterRadius = liquidFillView_parsePercent(radius, size) / 2; + innerRadius = outterRadius - outlineBorderWidth / 2; + paddingRadius = liquidFillView_parsePercent(outlineDistance, size); + + radius = Math.max(innerRadius - paddingRadius, 0); + } + + if (showOutline) { + var outline = getOutline(); + outline.style.lineWidth = outlineBorderWidth; + group.add(getOutline()); + } + + var left = isFillContainer ? 0 : cx - radius; + var top = isFillContainer ? 0 : cy - radius; + + var wavePath = null; + + group.add(getBackground()); + + // each data item for a wave + var oldData = this._data; + var waves = []; + data.diff(oldData) + .add(function (idx) { + var wave = getWave(idx, false); + + var waterLevel = wave.shape.waterLevel; + wave.shape.waterLevel = isFillContainer ? height / 2 : radius; + external_echarts_.graphic.initProps(wave, { + shape: { + waterLevel: waterLevel + } + }, seriesModel); + + wave.z2 = 2; + setWaveAnimation(idx, wave, null); + + group.add(wave); + data.setItemGraphicEl(idx, wave); + waves.push(wave); + }) + .update(function (newIdx, oldIdx) { + var waveElement = oldData.getItemGraphicEl(oldIdx); + + // new wave is used to calculate position, but not added + var newWave = getWave(newIdx, false, waveElement); + + // changes with animation + var shape = {}; + var shapeAttrs = ['amplitude', 'cx', 'cy', 'phase', 'radius', 'radiusY', 'waterLevel', 'waveLength']; + for (var i = 0; i < shapeAttrs.length; ++i) { + var attr = shapeAttrs[i]; + if (newWave.shape.hasOwnProperty(attr)) { + shape[attr] = newWave.shape[attr]; + } + } + + var style = {}; + var styleAttrs = ['fill', 'opacity', 'shadowBlur', 'shadowColor']; + for (var i = 0; i < styleAttrs.length; ++i) { + var attr = styleAttrs[i]; + if (newWave.style.hasOwnProperty(attr)) { + style[attr] = newWave.style[attr]; + } + } + + if (isFillContainer) { + shape.radiusY = height / 2; + } + + // changes with animation + external_echarts_.graphic.updateProps(waveElement, { + shape: shape, + x: newWave.x, + y: newWave.y + }, seriesModel); + + if (seriesModel.isUniversalTransitionEnabled && seriesModel.isUniversalTransitionEnabled()) { + external_echarts_.graphic.updateProps(waveElement, { + style: style + }, seriesModel); + } + else { + waveElement.useStyle(style); + } + + // instant changes + var oldWaveClipPath = waveElement.getClipPath(); + var newWaveClipPath = newWave.getClipPath(); + + waveElement.setClipPath(newWave.getClipPath()); + waveElement.shape.inverse = newWave.inverse; + + if (oldWaveClipPath && newWaveClipPath + && self._shape === symbol + // TODO use zrender morphing to apply complex symbol animation. + && !isPathSymbol(symbol) + ) { + // Can be animated. + external_echarts_.graphic.updateProps(newWaveClipPath, { + shape: oldWaveClipPath.shape + }, seriesModel, { isFrom: true }); + } + + setWaveAnimation(newIdx, waveElement, waveElement); + group.add(waveElement); + data.setItemGraphicEl(newIdx, waveElement); + waves.push(waveElement); + }) + .remove(function (idx) { + var wave = oldData.getItemGraphicEl(idx); + group.remove(wave); + }) + .execute(); + + if (itemModel.get('label.show')) { + group.add(getText(waves)); + } + + this._shape = symbol; + this._data = data; + + /** + * Get path for outline, background and clipping + * + * @param {number} r outter radius of shape + * @param {boolean|undefined} isForClipping if the shape is used + * for clipping + */ + function getPath(r, isForClipping) { + if (symbol) { + // customed symbol path + if (isPathSymbol(symbol)) { + var path = external_echarts_.graphic.makePath(symbol.slice(7), {}); + var bouding = path.getBoundingRect(); + var w = bouding.width; + var h = bouding.height; + if (w > h) { + h = r * 2 / w * h; + w = r * 2; + } + else { + w = r * 2 / h * w; + h = r * 2; + } + + var left = isForClipping ? 0 : cx - w / 2; + var top = isForClipping ? 0 : cy - h / 2; + path = external_echarts_.graphic.makePath( + symbol.slice(7), + {}, + new external_echarts_.graphic.BoundingRect(left, top, w, h) + ); + if (isForClipping) { + path.x = -w / 2; + path.y = -h / 2; + } + return path; + } + else if (isFillContainer) { + // fully fill the container + var x = isForClipping ? -r[0] : cx - r[0]; + var y = isForClipping ? -r[1] : cy - r[1]; + return external_echarts_.helper.createSymbol( + 'rect', x, y, r[0] * 2, r[1] * 2 + ); + } + else { + var x = isForClipping ? -r : cx - r; + var y = isForClipping ? -r : cy - r; + if (symbol === 'pin') { + y += r; + } + else if (symbol === 'arrow') { + y -= r; + } + return external_echarts_.helper.createSymbol(symbol, x, y, r * 2, r * 2); + } + } + + return new external_echarts_.graphic.Circle({ + shape: { + cx: isForClipping ? 0 : cx, + cy: isForClipping ? 0 : cy, + r: r + } + }); + } + /** + * Create outline + */ + function getOutline() { + var outlinePath = getPath(outterRadius); + outlinePath.style.fill = null; + + outlinePath.setStyle(seriesModel.getModel('outline.itemStyle') + .getItemStyle()); + + return outlinePath; + } + + /** + * Create background + */ + function getBackground() { + // Seperate stroke and fill, so we can use stroke to cover the alias of clipping. + var strokePath = getPath(radius); + strokePath.setStyle(seriesModel.getModel('backgroundStyle') + .getItemStyle()); + strokePath.style.fill = null; + + // Stroke is front of wave + strokePath.z2 = 5; + + var fillPath = getPath(radius); + fillPath.setStyle(seriesModel.getModel('backgroundStyle') + .getItemStyle()); + fillPath.style.stroke = null; + + var group = new external_echarts_.graphic.Group(); + group.add(strokePath); + group.add(fillPath); + + return group; + } + + /** + * wave shape + */ + function getWave(idx, isInverse, oldWave) { + var radiusX = isFillContainer ? radius[0] : radius; + var radiusY = isFillContainer ? height / 2 : radius; + + var itemModel = data.getItemModel(idx); + var itemStyleModel = itemModel.getModel('itemStyle'); + var phase = itemModel.get('phase'); + var amplitude = liquidFillView_parsePercent(itemModel.get('amplitude'), + radiusY * 2); + var waveLength = liquidFillView_parsePercent(itemModel.get('waveLength'), + radiusX * 2); + + var value = data.get('value', idx); + var waterLevel = radiusY - value * radiusY * 2; + phase = oldWave ? oldWave.shape.phase + : (phase === 'auto' ? idx * Math.PI / 4 : phase); + var normalStyle = itemStyleModel.getItemStyle(); + if (!normalStyle.fill) { + var seriesColor = seriesModel.get('color'); + var id = idx % seriesColor.length; + normalStyle.fill = seriesColor[id]; + } + + var x = radiusX * 2; + var wave = new liquidFillShape({ + shape: { + waveLength: waveLength, + radius: radiusX, + radiusY: radiusY, + cx: x, + cy: 0, + waterLevel: waterLevel, + amplitude: amplitude, + phase: phase, + inverse: isInverse + }, + style: normalStyle, + x: cx, + y: cy, + }); + wave.shape._waterLevel = waterLevel; + + var hoverStyle = itemModel.getModel('emphasis.itemStyle') + .getItemStyle(); + hoverStyle.lineWidth = 0; + + wave.ensureState('emphasis').style = hoverStyle; + external_echarts_.helper.enableHoverEmphasis(wave); + + // clip out the part outside the circle + var clip = getPath(radius, true); + // set fill for clipPath, otherwise it will not trigger hover event + clip.setStyle({ + fill: 'white' + }); + wave.setClipPath(clip); + + return wave; + } + + function setWaveAnimation(idx, wave, oldWave) { + var itemModel = data.getItemModel(idx); + + var maxSpeed = itemModel.get('period'); + var direction = itemModel.get('direction'); + + var value = data.get('value', idx); + + var phase = itemModel.get('phase'); + phase = oldWave ? oldWave.shape.phase + : (phase === 'auto' ? idx * Math.PI / 4 : phase); + + var defaultSpeed = function (maxSpeed) { + var cnt = data.count(); + return cnt === 0 ? maxSpeed : maxSpeed * + (0.2 + (cnt - idx) / cnt * 0.8); + }; + var speed = 0; + if (maxSpeed === 'auto') { + speed = defaultSpeed(5000); + } + else { + speed = typeof maxSpeed === 'function' + ? maxSpeed(value, idx) : maxSpeed; + } + + // phase for moving left/right + var phaseOffset = 0; + if (direction === 'right' || direction == null) { + phaseOffset = Math.PI; + } + else if (direction === 'left') { + phaseOffset = -Math.PI; + } + else if (direction === 'none') { + phaseOffset = 0; + } + else { + console.error('Illegal direction value for liquid fill.'); + } + + // wave animation of moving left/right + if (direction !== 'none' && itemModel.get('waveAnimation')) { + wave + .animate('shape', true) + .when(0, { + phase: phase + }) + .when(speed / 2, { + phase: phaseOffset + phase + }) + .when(speed, { + phase: phaseOffset * 2 + phase + }) + .during(function () { + if (wavePath) { + wavePath.dirty(true); + } + }) + .start(); + } + } + + /** + * text on wave + */ + function getText(waves) { + var labelModel = itemModel.getModel('label'); + + function formatLabel() { + var formatted = seriesModel.getFormattedLabel(0, 'normal'); + var defaultVal = (data.get('value', 0) * 100); + var defaultLabel = data.getName(0) || seriesModel.name; + if (!isNaN(defaultVal)) { + defaultLabel = defaultVal.toFixed(0) + '%'; + } + return formatted == null ? defaultLabel : formatted; + } + + var textRectOption = { + z2: 10, + shape: { + x: left, + y: top, + width: (isFillContainer ? radius[0] : radius) * 2, + height: (isFillContainer ? radius[1] : radius) * 2 + }, + style: { + fill: 'transparent' + }, + textConfig: { + position: labelModel.get('position') || 'inside' + }, + silent: true + }; + var textOption = { + style: { + text: formatLabel(), + textAlign: labelModel.get('align'), + textVerticalAlign: labelModel.get('baseline') + } + }; + Object.assign(textOption.style, external_echarts_.helper.createTextStyle(labelModel)); + + var outsideTextRect = new external_echarts_.graphic.Rect(textRectOption); + var insideTextRect = new external_echarts_.graphic.Rect(textRectOption); + insideTextRect.disableLabelAnimation = true; + outsideTextRect.disableLabelAnimation = true; + + var outsideText = new external_echarts_.graphic.Text(textOption); + var insideText = new external_echarts_.graphic.Text(textOption); + outsideTextRect.setTextContent(outsideText); + + insideTextRect.setTextContent(insideText); + var insColor = labelModel.get('insideColor'); + insideText.style.fill = insColor; + + var group = new external_echarts_.graphic.Group(); + group.add(outsideTextRect); + group.add(insideTextRect); + + // clip out waves for insideText + var boundingCircle = getPath(radius, true); + + wavePath = new external_echarts_.graphic.CompoundPath({ + shape: { + paths: waves + }, + x: cx, + y: cy + }); + + wavePath.setClipPath(boundingCircle); + insideTextRect.setClipPath(wavePath); + + return group; + } + }, + + dispose: function () { + // dispose nothing here + } +}); + +;// CONCATENATED MODULE: ./src/liquidFill.js + + +;// CONCATENATED MODULE: ./index.js + + + +/***/ }), + +/***/ "echarts/lib/echarts": +/*!**************************!*\ + !*** external "echarts" ***! + \**************************/ +/***/ ((module) => { + +module.exports = __WEBPACK_EXTERNAL_MODULE_echarts_lib_echarts__; + +/***/ }) + +/******/ }); +/************************************************************************/ +/******/ // The module cache +/******/ var __webpack_module_cache__ = {}; +/******/ +/******/ // The require function +/******/ function __webpack_require__(moduleId) { +/******/ // Check if module is in cache +/******/ if(__webpack_module_cache__[moduleId]) { +/******/ return __webpack_module_cache__[moduleId].exports; +/******/ } +/******/ // Create a new module (and put it into the cache) +/******/ var module = __webpack_module_cache__[moduleId] = { +/******/ // no module.id needed +/******/ // no module.loaded needed +/******/ exports: {} +/******/ }; +/******/ +/******/ // Execute the module function +/******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__); +/******/ +/******/ // Return the exports of the module +/******/ return module.exports; +/******/ } +/******/ +/************************************************************************/ +/******/ /* webpack/runtime/make namespace object */ +/******/ (() => { +/******/ // define __esModule on exports +/******/ __webpack_require__.r = (exports) => { +/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) { +/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); +/******/ } +/******/ Object.defineProperty(exports, '__esModule', { value: true }); +/******/ }; +/******/ })(); +/******/ +/************************************************************************/ +/******/ // module exports must be returned from runtime so entry inlining is disabled +/******/ // startup +/******/ // Load entry module and return exports +/******/ return __webpack_require__("./index.js"); +/******/ })() +; +}); +//# sourceMappingURL=echarts-liquidfill.js.map \ No newline at end of file diff --git a/components/screen/Chart/index.js b/components/screen/Chart/index.js new file mode 100644 index 0000000..064b0dc --- /dev/null +++ b/components/screen/Chart/index.js @@ -0,0 +1,38 @@ +import React, {useRef, useEffect, useImperativeHandle} from "react"; +import * as echarts from "echarts"; +import theme from "./theme.json"; +import './echarts-liquidfill' +// import('echarts-liquidfill'); + +echarts.registerTheme("darkBlue", theme); + +export default React.forwardRef(({option}, ref) => { + const chartRef = useRef(null); + const chart = useRef(); + + useEffect(() => { + chart.current = echarts.init(chartRef.current, "darkBlue"); + let resize = () => { + chart.current.resize(); + }; + window.addEventListener("resize", () => resize()); + return window.removeEventListener("resize", () => resize()); + }, []); + + useEffect(() => { + if (chart && option) { + // console.log("chart",chart) + chart.current.setOption(option); + } + }); + + useImperativeHandle(ref, () => ({ + setOption: (opt) => { + // chart.current.setOption(Object.assign(opt, option)); + chart.current.setOption(opt); + }, + getChart: () => chart.current, + })); + + return
; +}); diff --git a/components/screen/Chart/theme.json b/components/screen/Chart/theme.json new file mode 100644 index 0000000..9d8da9a --- /dev/null +++ b/components/screen/Chart/theme.json @@ -0,0 +1,440 @@ +{ + "color": [ + "#28a3ff", + "#ffc760", + "#6fe621", + "#f95757", + "#54c4f1", + "#fc8452", + "#9a60b4", + "#ea7ccc", + "#7289ab", + "#91ca8c", + "#f49f42" + ], + "backgroundColor": "rgba(4,20,87,0)", + "textStyle": {}, + "xAxis": { + "nameGap": 5 + }, + "yAxis": { + "nameGap": 5 + }, + "title": { + "textStyle": { + "color": "#eeeeee" + }, + "subtextStyle": { + "color": "#aaaaaa" + } + }, + "grid": { + "top": 30, + "left": 30, + "bottom": 30, + "right": 30 + }, + "line": { + "itemStyle": { + "borderWidth": 1 + }, + "lineStyle": { + "width": 2 + }, + "symbolSize": 10, + "symbol": "circle", + "smooth": false + }, + "radar": { + "itemStyle": { + "borderWidth": 1 + }, + "lineStyle": { + "width": 2 + }, + "symbolSize": 4, + "symbol": "circle", + "smooth": false + }, + "bar": { + "itemStyle": { + "barBorderWidth": 0, + "barBorderColor": "#ccc" + } + }, + "pie": { + "itemStyle": { + "borderWidth": 0, + "borderColor": "#ccc" + } + }, + "scatter": { + "itemStyle": { + "borderWidth": 0, + "borderColor": "#ccc" + } + }, + "boxplot": { + "itemStyle": { + "borderWidth": 0, + "borderColor": "#ccc" + } + }, + "parallel": { + "itemStyle": { + "borderWidth": 0, + "borderColor": "#ccc" + } + }, + "sankey": { + "itemStyle": { + "borderWidth": 0, + "borderColor": "#ccc" + } + }, + "funnel": { + "itemStyle": { + "borderWidth": 0, + "borderColor": "#ccc" + } + }, + "gauge": { + "itemStyle": { + "borderWidth": 0, + "borderColor": "#ccc" + } + }, + "candlestick": { + "itemStyle": { + "color": "#fd1050", + "color0": "#0cf49b", + "borderColor": "#fd1050", + "borderColor0": "#0cf49b", + "borderWidth": 1 + } + }, + "graph": { + "itemStyle": { + "borderWidth": 0, + "borderColor": "#ccc" + }, + "lineStyle": { + "width": 1, + "color": "#aaaaaa" + }, + "symbolSize": 4, + "symbol": "circle", + "smooth": false, + "color": [ + "#dd6b66", + "#759aa0", + "#e69d87", + "#8dc1a9", + "#ea7e53", + "#eedd78", + "#73a373", + "#73b9bc", + "#7289ab", + "#91ca8c", + "#f49f42" + ], + "label": { + "color": "#eeeeee" + } + }, + "map": { + "itemStyle": { + "normal": { + "areaColor": "#eee", + "borderColor": "#444", + "borderWidth": 0.5 + }, + "emphasis": { + "areaColor": "rgba(255,215,0,0.8)", + "borderColor": "#444", + "borderWidth": 1 + } + }, + "label": { + "normal": { + "textStyle": { + "color": "#000" + } + }, + "emphasis": { + "textStyle": { + "color": "rgb(100,0,0)" + } + } + } + }, + "geo": { + "itemStyle": { + "normal": { + "areaColor": "#eee", + "borderColor": "#444", + "borderWidth": 0.5 + }, + "emphasis": { + "areaColor": "rgba(255,215,0,0.8)", + "borderColor": "#444", + "borderWidth": 1 + } + }, + "label": { + "normal": { + "textStyle": { + "color": "#000" + } + }, + "emphasis": { + "textStyle": { + "color": "rgb(100,0,0)" + } + } + } + }, + "categoryAxis": { + "axisLine": { + "show": true, + "lineStyle": { + "color": "#eeeeee" + } + }, + "axisTick": { + "show": true, + "lineStyle": { + "color": "#eeeeee" + } + }, + "axisLabel": { + "show": true, + "textStyle": { + "color": "#eeeeee" + } + }, + "splitLine": { + "show": false, + "lineStyle": { + "color": [ + "#aaaaaa" + ] + } + }, + "splitArea": { + "show": false, + "areaStyle": { + "color": [ + "#eeeeee" + ] + } + } + }, + "valueAxis": { + "axisLine": { + "show": true, + "lineStyle": { + "color": "#eeeeee" + } + }, + "axisTick": { + "show": true, + "lineStyle": { + "color": "#eeeeee" + } + }, + "axisLabel": { + "show": true, + "textStyle": { + "color": "#eeeeee" + } + }, + "splitLine": { + "show": false, + "lineStyle": { + "color": [ + "#aaaaaa" + ] + } + }, + "splitArea": { + "show": false, + "areaStyle": { + "color": [ + "#eeeeee" + ] + } + } + }, + "logAxis": { + "axisLine": { + "show": true, + "lineStyle": { + "color": "#eeeeee" + } + }, + "axisTick": { + "show": true, + "lineStyle": { + "color": "#eeeeee" + } + }, + "axisLabel": { + "show": true, + "textStyle": { + "color": "#eeeeee" + } + }, + "splitLine": { + "show": false, + "lineStyle": { + "color": [ + "#aaaaaa" + ] + } + }, + "splitArea": { + "show": false, + "areaStyle": { + "color": [ + "#eeeeee" + ] + } + } + }, + "timeAxis": { + "axisLine": { + "show": true, + "lineStyle": { + "color": "#eeeeee" + } + }, + "axisTick": { + "show": true, + "lineStyle": { + "color": "#eeeeee" + } + }, + "axisLabel": { + "show": true, + "textStyle": { + "color": "#eeeeee" + } + }, + "splitLine": { + "show": false, + "lineStyle": { + "color": [ + "#aaaaaa" + ] + } + }, + "splitArea": { + "show": false, + "areaStyle": { + "color": [ + "#eeeeee" + ] + } + } + }, + "toolbox": { + "iconStyle": { + "normal": { + "borderColor": "rgba(153,153,153,1)" + }, + "emphasis": { + "borderColor": "rgba(102,102,102,1)" + } + } + }, + "legend": { + "show": true, + "left": "center", + "textStyle": { + "color": "#eeeeee" + } + }, + "tooltip": { + "axisPointer": { + "lineStyle": { + "color": "rgba(238,238,238,1)", + "width": "1" + }, + "crossStyle": { + "color": "rgba(238,238,238,1)", + "width": "1" + } + } + }, + "timeline": { + "lineStyle": { + "color": "#eeeeee", + "width": 1 + }, + "itemStyle": { + "normal": { + "color": "#dd6b66", + "borderWidth": 1 + }, + "emphasis": { + "color": "#a9334c" + } + }, + "controlStyle": { + "normal": { + "color": "#eeeeee", + "borderColor": "#eeeeee", + "borderWidth": 0.5 + }, + "emphasis": { + "color": "#eeeeee", + "borderColor": "#eeeeee", + "borderWidth": 0.5 + } + }, + "checkpointStyle": { + "color": "#e43c59", + "borderColor": "#c23531" + }, + "label": { + "normal": { + "textStyle": { + "color": "#eeeeee" + } + }, + "emphasis": { + "textStyle": { + "color": "#eeeeee" + } + } + } + }, + "visualMap": { + "color": [ + "#bf444c", + "#d88273", + "#f6efa6" + ] + }, + "dataZoom": { + "backgroundColor": "rgba(47,69,84,0)", + "dataBackgroundColor": "rgba(255,255,255,0.3)", + "fillerColor": "rgba(167,183,204,0.4)", + "handleColor": "#a7b7cc", + "handleSize": "100%", + "textStyle": { + "color": "#eeeeee" + } + }, + "markPoint": { + "label": { + "color": "#eeeeee" + }, + "emphasis": { + "label": { + "color": "#eeeeee" + } + } + } +} \ No newline at end of file diff --git a/components/screen/CircleProcess/index.js b/components/screen/CircleProcess/index.js new file mode 100644 index 0000000..4e0fb33 --- /dev/null +++ b/components/screen/CircleProcess/index.js @@ -0,0 +1,162 @@ +import React, { useRef, useEffect, useState, useMemo } from 'react' + +const imgHeight = 951; //图片原始高度 +const imgWidth = 184; //图片原始高度 +const circleBorder = 2; //圆环的宽度 +const baseSize = 170;//满状态宽度 +const itemHeight = 370; //单个高度(包含空白) +const waveHeight = 29; //波浪高度 +const colors = ["#8ee34d", "#f6c971", "#ea335d"]; //圆边框颜色 +const colorIndexTrans = [2, 1, 0]; //调换第一个和最后一个位置 + +function CircleProcess(props) { + const sizeRef = useRef(); + const outerRef = useRef(); + const innerRef = useRef(); + const [scale, setScale] = useState(1) + const [innerWidth, setInnerWidth] = useState(0) + + //接受外部参数 + let { percent:realPercent = 0.5, percentRule = [0, 0.8, 0.95], size = 0.9 } = props; + + const percent = useMemo(() => { + return realPercent > 1 ? 1 : realPercent + }, [realPercent]) + + //颜色的下标 + const colorIndex = useMemo(() => { + let res = 0; + percentRule.forEach((val, index) => { + if (percent > val) res = index; + }) + return colorIndexTrans[res]; + }, [percentRule, percent]) + + //颜色的位置 + const positionY = useMemo(() => { + return -(-innerWidth + percent * innerWidth + waveHeight * scale + colorIndex * scale * itemHeight) + }, [innerWidth, percent, scale, colorIndex]) + + //计算组件宽度以及比例 + const tableSetting = () => { + //组件实际宽度 + let comWidth = sizeRef.current.clientWidth; + outerRef.current.style.height = comWidth * size + 'px'; + outerRef.current.style.width = comWidth * size + 'px'; + //波浪实际宽度 + let innerWidth = innerRef.current.clientWidth; + setInnerWidth(innerWidth); + //计算比例 + setScale(innerWidth / baseSize); + } + + useEffect(() => { + //监听父级宽度变化 + let innerWindow = sizeRef.current.contentDocument.defaultView; + innerWindow.addEventListener("resize", tableSetting); + tableSetting(); + return () => { + innerWindow.removeEventListener("resize", tableSetting); + } + }, []) + + return ( +
+