aboutsummaryrefslogtreecommitdiff
path: root/src/client/views/nodes/ScriptingBox.tsx
diff options
context:
space:
mode:
Diffstat (limited to 'src/client/views/nodes/ScriptingBox.tsx')
-rw-r--r--src/client/views/nodes/ScriptingBox.tsx137
1 files changed, 81 insertions, 56 deletions
diff --git a/src/client/views/nodes/ScriptingBox.tsx b/src/client/views/nodes/ScriptingBox.tsx
index a2dd134ed..74d9c2e94 100644
--- a/src/client/views/nodes/ScriptingBox.tsx
+++ b/src/client/views/nodes/ScriptingBox.tsx
@@ -1,71 +1,96 @@
-import { library } from '@fortawesome/fontawesome-svg-core';
-import { faEdit } from '@fortawesome/free-regular-svg-icons';
-import { computed } from 'mobx';
-import { observer } from 'mobx-react';
-import * as React from 'react';
-import { Doc } from '../../../new_fields/Doc';
-import { documentSchema } from '../../../new_fields/documentSchemas';
+import { action, observable, computed } from "mobx";
+import { observer } from "mobx-react";
+import * as React from "react";
+import { Opt } from "../../../new_fields/Doc";
+import { documentSchema } from "../../../new_fields/documentSchemas";
+import { createSchema, makeInterface } from "../../../new_fields/Schema";
+import { ScriptField } from "../../../new_fields/ScriptField";
+import { StrCast, ScriptCast } from "../../../new_fields/Types";
+import { returnTrue } from "../../../Utils";
+import { InteractionUtils } from "../../util/InteractionUtils";
import { CompileScript } from "../../util/Scripting";
-import { ScriptBox } from '../ScriptBox';
-import { createSchema, listSpec, makeInterface } from '../../../new_fields/Schema';
-import { ScriptField } from '../../../new_fields/ScriptField';
-import { BoolCast, ScriptCast } from '../../../new_fields/Types';
-import { DocComponent } from '../DocComponent';
-import { FieldView, FieldViewProps } from './FieldView';
-import './ScriptingBox.scss';
-import { DocumentIconContainer } from './DocumentIcon';
-
-
-library.add(faEdit as any);
+import { DocAnnotatableComponent } from "../DocComponent";
+import { EditableView } from "../EditableView";
+import { FieldView, FieldViewProps } from "../nodes/FieldView";
+import "./ScriptingBox.scss";
const ScriptingSchema = createSchema({
- onClick: ScriptField,
- buttonParams: listSpec("string"),
- text: "string"
});
type ScriptingDocument = makeInterface<[typeof ScriptingSchema, typeof documentSchema]>;
const ScriptingDocument = makeInterface(ScriptingSchema, documentSchema);
@observer
-export class ScriptingBox extends DocComponent<FieldViewProps, ScriptingDocument>(ScriptingDocument) {
- public static LayoutString(fieldKey: string) { return FieldView.LayoutString(ScriptingBox, fieldKey); }
+export class ScriptingBox extends DocAnnotatableComponent<FieldViewProps, ScriptingDocument>(ScriptingDocument) {
+ protected multiTouchDisposer?: InteractionUtils.MultiTouchEventDisposer | undefined;
+ public static LayoutString(fieldStr: string) { return FieldView.LayoutString(ScriptingBox, fieldStr); }
- @computed get dataDoc() {
- return this.props.DataDoc &&
- (this.Document.isTemplateForField || BoolCast(this.props.DataDoc.isTemplateForField) ||
- this.props.DataDoc.layout === this.props.Document) ? this.props.DataDoc : Doc.GetProto(this.props.Document);
+ @observable private _errorMessage: string = "";
+ @computed get rawScript() { return StrCast(this.dataDoc[this.props.fieldKey + "-raw"]); }
+ set rawScript(value) { this.dataDoc[this.props.fieldKey + "-raw"] = value; }
+
+ @action
+ componentDidMount() {
+ this.rawScript = ScriptCast(this.dataDoc[this.props.fieldKey])?.script?.originalScript || this.rawScript;
}
- specificContextMenu = (e: React.MouseEvent): void => { }
+ @action
+ onChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => {
+ this.rawScript = e.target.value;
+ }
- render() {
- const script = ScriptCast(this.props.Document[this.props.fieldKey]);
- let originalText: string | undefined = undefined;
- if (script) {
- originalText = script.script.originalScript;
+ @action
+ onError = (error: any) => {
+ this._errorMessage = (error as any).map((e: any) => e.messageText).join(" ");
+ }
+
+ @action
+ onCompile = () => {
+ const result = CompileScript(this.rawScript, {});
+ this._errorMessage = "";
+ if (result.compiled) {
+ this._errorMessage = "";
+ this.dataDoc[this.props.fieldKey] = new ScriptField(result);
}
- return !(this.props.Document instanceof Doc) ? (null) :
- <ScriptBox initialText={originalText}
- setParams={() => { }}
- onCancel={() => { }}
- onSave={(text, onError) => {
- if (!text) {
- this.dataDoc[this.props.fieldKey] = undefined;
- } else {
- const script = CompileScript(text, {
- params: { this: Doc.name },
- typecheck: false,
- editable: true,
- transformer: DocumentIconContainer.getTransformer()
- });
- if (!script.compiled) {
- onError(script.errors.map(error => error.messageText).join("\n"));
- }
- else {
- this.dataDoc[this.props.fieldKey] = new ScriptField(script);
- }
- }
- }} showDocumentIcons />;
+ else {
+ this.dataDoc[this.props.fieldKey] = undefined;
+ this.onError(result.errors);
+ }
+ }
+
+ @action
+ onRun = () => {
+ this.onCompile();
+ ScriptCast(this.dataDoc[this.props.fieldKey])?.script.run({},
+ (err: any) => {
+ this._errorMessage = "";
+ this.onError(err);
+ });
+ }
+
+ render() {
+ let onFocus: Opt<() => void> = undefined, onBlur: Opt<() => void> = undefined;
+ const params = <EditableView
+ contents={""}
+ display={"block"}
+ maxHeight={72}
+ height={35}
+ fontSize={28}
+ GetValue={() => ""}
+ SetValue={returnTrue}
+ />;
+ return (
+ <div className="scriptingBox-outerDiv" onPointerDown={(e) => this.props.isSelected() && e.stopPropagation()} onWheel={(e) => this.props.isSelected() && e.stopPropagation()}>
+ <div className="scriptingBox-inputDiv" >
+ <textarea className="scriptingBox-textarea" placeholder="write your script here" onChange={this.onChange} value={this.rawScript} onFocus={onFocus} onBlur={onBlur} />
+ <div className="scriptingBox-errorMessage">{this._errorMessage}</div>
+ <div style={{ background: "beige" }} >{params}</div>
+ </div>
+ <div className="scriptingBox-toolbar">
+ <button className="scriptingBox-button" onPointerDown={e => { this.onCompile(); e.stopPropagation(); }}>Compile</button>
+ <button className="scriptingBox-button" onPointerDown={e => { this.onRun(); e.stopPropagation(); }}>Run</button>
+ </div>
+ </div>
+ );
}
-} \ No newline at end of file
+}