mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-19 21:11:32 +01:00
137 lines
3.9 KiB
TypeScript
137 lines
3.9 KiB
TypeScript
// Copyright 2024, Command Line Inc.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
import { Button } from "@/element/button";
|
|
import { WaveModal } from "@/element/modal";
|
|
import * as Plot from "@observablehq/plot";
|
|
import * as d3 from "d3";
|
|
import * as React from "react";
|
|
|
|
import "./plotview.less";
|
|
|
|
function PlotWindow() {
|
|
return <div className="plot-window"></div>;
|
|
}
|
|
|
|
function PlotConfig() {
|
|
return <input type="text" className="plot-config" />;
|
|
}
|
|
|
|
const AsyncFunction = Object.getPrototypeOf(async function () {}).constructor;
|
|
|
|
function evalAsync(Plot: any, d3: any, funcText: string): Promise<unknown> {
|
|
return new Promise((resolve, reject) => {
|
|
new AsyncFunction(
|
|
"resolve",
|
|
"reject",
|
|
"Plot",
|
|
"d3",
|
|
`try { await ${funcText}; resolve(); } catch(e) { reject(e); } }`
|
|
)(resolve, reject, Plot, d3);
|
|
});
|
|
}
|
|
|
|
function PlotView() {
|
|
const containerRef = React.useRef<HTMLInputElement>();
|
|
const [plotDef, setPlotDef] = React.useState<string>();
|
|
const [savedDef, setSavedDef] = React.useState<string>();
|
|
const [modalUp, setModalUp] = React.useState(false);
|
|
/*
|
|
const [data, setData] = React.useState();
|
|
|
|
React.useEffect(() => {
|
|
d3.csv("/plotdata/congress.csv", d3.autoType).then(setData);
|
|
}, []);
|
|
*/
|
|
|
|
React.useEffect(() => {
|
|
// replace start
|
|
/*
|
|
d3.csv("/plotdata/congress.csv", d3.autoType).then((out) => data = out);
|
|
return Plot.plot({
|
|
aspectRatio: 1,
|
|
x: { label: "Age (years)" },
|
|
y: {
|
|
grid: true,
|
|
label: "← Women · Men →",
|
|
labelAnchor: "center",
|
|
tickFormat: Math.abs,
|
|
},
|
|
marks: [
|
|
Plot.dot(
|
|
data,
|
|
Plot.stackY2({
|
|
x: (d) => 2023 - d.birthday.getUTCFullYear(),
|
|
y: (d) => (d.gender === "M" ? 1 : -1),
|
|
fill: "gender",
|
|
title: "full_name",
|
|
})
|
|
),
|
|
Plot.ruleY([0]),
|
|
],
|
|
});
|
|
*/
|
|
// replace end
|
|
let plot;
|
|
let plotErr;
|
|
try {
|
|
console.log(plotDef);
|
|
plot = new Function("Plot", "d3", plotDef)(Plot, d3);
|
|
//plot = new Function("Plot", "data", "d3", plotDef)(Plot, data, d3);
|
|
//evalAsync(Plot, d3, plotDef).then((out) => (plot = out));
|
|
} catch (e) {
|
|
plotErr = e;
|
|
console.log("error: ", e);
|
|
return;
|
|
}
|
|
console.log(plot);
|
|
|
|
if (plot !== undefined) {
|
|
containerRef.current.append(plot);
|
|
} else {
|
|
// todo
|
|
}
|
|
|
|
return () => {
|
|
if (plot !== undefined) {
|
|
plot.remove();
|
|
}
|
|
};
|
|
}, [plotDef]);
|
|
|
|
const handleOpen = React.useCallback(() => {
|
|
setSavedDef(plotDef);
|
|
setModalUp(true);
|
|
}, []);
|
|
|
|
const handleCancel = React.useCallback(() => {
|
|
setPlotDef(savedDef);
|
|
setModalUp(false);
|
|
}, []);
|
|
|
|
const handleSave = React.useCallback(() => {
|
|
setModalUp(false);
|
|
}, []);
|
|
|
|
return (
|
|
<div className="plot-view">
|
|
<Button onClick={handleOpen}>Edit</Button>
|
|
<div className="plot-window" ref={containerRef} />
|
|
{modalUp && (
|
|
<WaveModal title="Plot Definition" onCancel={handleCancel} onSubmit={handleSave} buttonLabel={"Save"}>
|
|
<textarea
|
|
className="plot-config"
|
|
rows={5}
|
|
onChange={(e) => setPlotDef(e.target.value)}
|
|
spellCheck={false}
|
|
defaultValue={plotDef}
|
|
wrap={"off"}
|
|
/>
|
|
</WaveModal>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export { PlotView };
|