| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
 | import { IReactionDisposer } from 'mobx';
import { observer } from 'mobx-react';
import { Node } from 'prosemirror-model';
import { TextSelection } from 'prosemirror-state';
import { EditorView } from 'prosemirror-view';
import * as React from 'react';
import * as ReactDOM from 'react-dom/client';
import { Doc } from '../../../../fields/Doc';
import { DocData } from '../../../../fields/DocSymbols';
import { StrCast } from '../../../../fields/Types';
import './DashFieldView.scss';
import EquationEditor from './EquationEditor';
import { FormattedTextBox } from './FormattedTextBox';
interface IEquationViewInternal {
    fieldKey: string;
    tbox: FormattedTextBox;
    width: number;
    height: number;
    getPos: () => number | undefined;
    setEditor: (editor: EquationEditor | undefined) => void;
}
@observer
export class EquationViewInternal extends React.Component<IEquationViewInternal> {
    _reactionDisposer: IReactionDisposer | undefined;
    _textBoxDoc: Doc;
    _fieldKey: string;
    _ref: React.RefObject<EquationEditor> = React.createRef();
    constructor(props: IEquationViewInternal) {
        super(props);
        this._fieldKey = props.fieldKey;
        this._textBoxDoc = props.tbox.Document;
    }
    componentDidMount() {
        this.props.setEditor(this._ref.current ?? undefined);
    }
    componentWillUnmount() {
        this._reactionDisposer?.();
    }
    render() {
        return (
            <div
                className="equationView"
                onKeyDown={e => {
                    if (e.key === 'Enter') {
                        this.props.tbox.EditorView!.dispatch(this.props.tbox.EditorView!.state.tr.setSelection(new TextSelection(this.props.tbox.EditorView!.state.doc.resolve((this.props.getPos() ?? 0) + 1))));
                        this.props.tbox.EditorView!.focus();
                        e.preventDefault();
                    }
                    e.stopPropagation();
                }}
                style={{
                    position: 'relative',
                    display: 'inline-block',
                    width: this.props.width,
                    height: this.props.height,
                    background: 'white',
                    borderRadius: '10%',
                }}>
                <EquationEditor
                    ref={this._ref}
                    value={StrCast(this._textBoxDoc[DocData][this._fieldKey])}
                    onChange={str => {
                        this._textBoxDoc[DocData][this._fieldKey] = str;
                    }}
                    autoCommands="pi theta sqrt sum prod alpha beta gamma rho"
                    autoOperatorNames="sin cos tan"
                    spaceBehavesLikeTab
                />
            </div>
        );
    }
}
export class EquationView {
    dom: HTMLDivElement; // container for label and value
    root: ReactDOM.Root;
    tbox: FormattedTextBox;
    view: EditorView;
    _editor: EquationEditor | undefined;
    getPos: () => number | undefined;
    constructor(node: Node, view: EditorView, getPos: () => number | undefined, tbox: FormattedTextBox) {
        this.tbox = tbox;
        this.view = view;
        this.getPos = getPos;
        this.dom = document.createElement('div');
        this.dom.style.width = node.attrs.width;
        this.dom.style.height = node.attrs.height;
        this.dom.style.position = 'relative';
        this.dom.style.display = 'inline-block';
        this.dom.onmousedown = (e: MouseEvent) => {
            e.stopPropagation();
        };
        this.root = ReactDOM.createRoot(this.dom);
        this.root.render(<EquationViewInternal fieldKey={node.attrs.fieldKey} width={node.attrs.width} height={node.attrs.height} getPos={getPos} setEditor={this.setEditor} tbox={tbox} />);
    }
    setEditor = (editor?: EquationEditor) => {
        this._editor = editor;
    };
    destroy() {
        this.root.unmount();
    }
    setSelection() {
        this._editor?.mathField.focus();
    }
    selectNode() {
        this.view.dispatch(this.view.state.tr.setSelection(new TextSelection(this.view.state.doc.resolve(this.getPos() ?? 0))));
        this.tbox._applyingChange = this.tbox.fieldKey; // setting focus will make prosemirror lose focus, which will cause it to change its selection to a text selection, which causes this view to get rebuilt but it's no longer node selected, so the equationview won't have focus
        setTimeout(() => {
            this._editor?.mathField.focus();
            setTimeout(() => {
                this.tbox._applyingChange = '';
            });
        });
    }
    deselectNode() {}
}
 |