import * as React from "react"; import * as ReactDOM from "react-dom"; import { SquiggleChart } from "./SquiggleChart"; import { ReactCodeJar } from "react-codejar"; import type { exportEnv } from "@quri/squiggle-lang"; export interface SquiggleEditorProps { /** The input string for squiggle */ initialSquiggleString?: string; /** If the output requires monte carlo sampling, the amount of samples */ sampleCount?: number; /** The amount of points returned to draw the distribution */ outputXYPoints?: number; kernelWidth?: number; pointDistLength?: number; /** If the result is a function, where the function starts */ diagramStart?: number; /** If the result is a function, where the function ends */ diagramStop?: number; /** If the result is a function, how many points along the function it samples */ diagramCount?: number; /** The environment, other variables that were already declared */ environment?: exportEnv; /** when the environment changes. Used again for notebook magic*/ onEnvChange?(env: exportEnv): void; } const highlight = (editor: HTMLInputElement) => { let code = editor.textContent; code = code.replace(/\((\w+?)(\b)/g, '($1$2'); editor.innerHTML = code; }; interface SquiggleEditorState { expression: string; env: exportEnv; } export class SquiggleEditor extends React.Component< SquiggleEditorProps, SquiggleEditorState > { constructor(props: SquiggleEditorProps) { super(props); let code = props.initialSquiggleString ? props.initialSquiggleString : ""; this.state = { expression: code, env: props.environment }; } render() { let { expression, env } = this.state; let props = this.props; return (
{ this.setState({ expression: e }); }} style={{ borderRadius: "6px", width: "530px", border: "1px solid grey", fontFamily: "'Source Code Pro', monospace", fontSize: "14px", fontWeight: "400", letterSpacing: "normal", lineHeight: "20px", padding: "10px", tabSize: "4", }} highlight={highlight} lineNumbers={false} />
); } } export function renderSquiggleEditor(props: SquiggleEditorProps) { let parent = document.createElement("div"); ReactDOM.render( { // 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 // @ts-ignore parent.value = env; parent.dispatchEvent(new CustomEvent("input")); if (props.onEnvChange) props.onEnvChange(env); }} />, parent ); return parent; }