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) => {
|
2022-03-23 00:51:04 +00:00
|
|
|
// 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
|
|
|
}
|