squiggle/packages/components/src/SquiggleEditor.tsx

121 lines
3.8 KiB
TypeScript
Raw Normal View History

2022-03-23 00:38:01 +00:00
import * as React from "react";
import * as ReactDOM from "react-dom";
import { SquiggleChart } from "./SquiggleChart";
2022-03-22 02:33:28 +00:00
import { ReactCodeJar } from "react-codejar";
2022-03-23 00:38:01 +00:00
import type { exportEnv } from "@quri/squiggle-lang";
2022-03-22 02:33:28 +00:00
export interface SquiggleEditorProps {
/** The input string for squiggle */
2022-03-23 00:38:01 +00:00
initialSquiggleString?: string;
2022-03-22 02:33:28 +00:00
/** If the output requires monte carlo sampling, the amount of samples */
2022-03-23 00:38:01 +00:00
sampleCount?: number;
2022-03-22 02:33:28 +00:00
/** The amount of points returned to draw the distribution */
2022-03-23 00:38:01 +00:00
outputXYPoints?: number;
kernelWidth?: number;
pointDistLength?: number;
2022-03-22 02:33:28 +00:00
/** If the result is a function, where the function starts */
2022-03-23 00:38:01 +00:00
diagramStart?: number;
2022-03-22 02:33:28 +00:00
/** If the result is a function, where the function ends */
2022-03-23 00:38:01 +00:00
diagramStop?: number;
2022-03-22 02:33:28 +00:00
/** If the result is a function, how many points along the function it samples */
2022-03-23 00:38:01 +00:00
diagramCount?: number;
2022-03-22 02:33:28 +00:00
/** The environment, other variables that were already declared */
2022-03-23 00:38:01 +00:00
environment?: exportEnv;
2022-03-22 02:33:28 +00:00
/** when the environment changes. Used again for notebook magic*/
2022-03-23 00:38:01 +00:00
onEnvChange?(env: exportEnv): void;
2022-03-22 02:33:28 +00:00
}
const highlight = (editor: HTMLInputElement) => {
let code = editor.textContent;
code = code.replace(/\((\w+?)(\b)/g, '(<font color="#8a2be2">$1</font>$2');
editor.innerHTML = code;
};
interface SquiggleEditorState {
2022-03-23 00:38:01 +00:00
expression: string;
env: exportEnv;
2022-03-22 02:33:28 +00:00
}
2022-03-23 00:38:01 +00:00
export class SquiggleEditor extends React.Component<
SquiggleEditorProps,
SquiggleEditorState
> {
2022-03-22 02:33:28 +00:00
constructor(props: SquiggleEditorProps) {
2022-03-23 00:38:01 +00:00
super(props);
let code = props.initialSquiggleString ? props.initialSquiggleString : "";
this.state = { expression: code, env: props.environment };
2022-03-22 02:33:28 +00:00
}
render() {
2022-03-23 00:38:01 +00:00
let { expression, env } = this.state;
let props = this.props;
2022-03-22 02:33:28 +00:00
return (
2022-03-23 00:38:01 +00:00
<div>
<ReactCodeJar
code={expression}
onUpdate={(e) => {
this.setState({ expression: e });
2022-03-22 02:33:28 +00:00
}}
2022-03-23 00:38:01 +00:00
style={{
2022-03-22 02:33:28 +00:00
borderRadius: "6px",
width: "530px",
border: "1px solid grey",
fontFamily: "'Source Code Pro', monospace",
fontSize: "14px",
fontWeight: "400",
letterSpacing: "normal",
lineHeight: "20px",
padding: "10px",
2022-03-23 00:38:01 +00:00
tabSize: "4",
}}
highlight={highlight}
lineNumbers={false}
/>
<SquiggleChart
squiggleString={expression}
sampleCount={props.sampleCount}
outputXYPoints={props.outputXYPoints}
kernelWidth={props.kernelWidth}
pointDistLength={props.pointDistLength}
diagramStart={props.diagramStart}
diagramStop={props.diagramStop}
diagramCount={props.diagramCount}
environment={env}
onEnvChange={props.onEnvChange}
2022-03-22 02:33:28 +00:00
/>
2022-03-23 00:38:01 +00:00
</div>
);
2022-03-22 02:33:28 +00:00
}
}
2022-03-23 00:38:01 +00:00
export function renderSquiggleEditor(props: SquiggleEditorProps) {
let parent = document.createElement("div");
ReactDOM.render(
<SquiggleEditor
{...props}
onEnvChange={(env) => {
// Typescript complains on two levels here.
// - Div elements don't have a value property
// - Even if it did (like it was an input element), it would have to
// be a string
//
// Which are reasonable in most web contexts.
//
// However we're using observable, neither of those things have to be
// true there. div elements can contain the value property, and can have
// the value be any datatype they wish.
//
// This is here to get the 'viewof' part of:
// viewof env = cell('normal(0,1)')
// to work
2022-03-23 00:38:01 +00:00
// @ts-ignore
parent.value = env;
2022-03-22 02:33:28 +00:00
2022-03-23 00:38:01 +00:00
parent.dispatchEvent(new CustomEvent("input"));
if (props.onEnvChange) props.onEnvChange(env);
}}
/>,
parent
);
return parent;
2022-03-22 02:33:28 +00:00
}