This commit is contained in:
Yanzhen Yu 2021-12-07 22:32:19 +08:00
parent 7d34972861
commit 52c4961f9a
2 changed files with 103 additions and 8 deletions

View File

@ -10,20 +10,28 @@ import 'codemirror/lib/codemirror.css';
import 'codemirror/theme/ayu-mirage.css';
// tern
import 'codemirror/addon/tern/tern';
import 'codemirror/addon/selection/active-line';
import 'codemirror/addon/comment/comment';
import 'codemirror/addon/hint/show-hint';
import 'codemirror/addon/dialog/dialog.css';
import 'codemirror/addon/hint/show-hint.css';
import 'codemirror/addon/tern/tern.css';
import 'tern/plugin/doc_comment';
import 'tern/plugin/complete_strings';
import ecma from 'tern/defs/ecmascript.json';
import { Def } from 'tern';
import tern, { Def } from 'tern';
(window as unknown as { tern: typeof tern }).tern = tern;
function installTern(cm: CodeMirror.Editor) {
const tern = new CodeMirror.TernServer({ defs: [ecma as unknown as Def] });
cm.on('cursorActivity', cm => tern.updateArgHints(cm));
const t = new CodeMirror.TernServer({ defs: [ecma as unknown as Def] });
cm.on('cursorActivity', cm => t.updateArgHints(cm));
cm.on('change', (_instance, change) => {
if (change.text.length === 1 && change.text[0] === '.') {
tern.complete(cm);
t.complete(cm);
}
});
return tern;
return t;
}
export const TernEditor: React.FC<{
@ -31,7 +39,8 @@ export const TernEditor: React.FC<{
onChange?: (v: string) => void;
onBlur?: (v: string) => void;
lineNumbers?: boolean;
}> = ({ defaultCode, onChange, onBlur, lineNumbers = true }) => {
defs?: tern.Def[];
}> = ({ defaultCode, onChange, onBlur, lineNumbers = true, defs }) => {
const style = css`
.CodeMirror {
width: 100%;
@ -41,6 +50,7 @@ export const TernEditor: React.FC<{
const wrapperEl = useRef<HTMLDivElement>(null);
const cm = useRef<CodeMirror.Editor | null>(null);
const tServer = useRef<tern.Server | null>(null);
useEffect(() => {
if (!wrapperEl.current) {
return;
@ -64,7 +74,8 @@ export const TernEditor: React.FC<{
theme: 'ayu-mirage',
viewportMargin: Infinity,
});
installTern(cm.current);
const t = installTern(cm.current);
tServer.current = t.server;
}
const changeHandler = (instance: CodeMirror.Editor) => {
onChange?.(instance.getValue());
@ -79,6 +90,13 @@ export const TernEditor: React.FC<{
cm.current?.off('blur', blurHandler);
};
}, [defaultCode]);
useEffect(() => {
if (defs) {
console.log(tServer.current, 'add', defs);
tServer.current?.deleteDefs('customDataTree');
tServer.current?.addDefs(defs[0] as any, true);
}
}, [defs]);
return <Box css={style} ref={wrapperEl} height="100%" width="100%"></Box>;
};

View File

@ -1,11 +1,87 @@
import React from 'react';
import React, { useEffect, useState } from 'react';
import { FieldProps } from '../fields';
import { TernEditor } from 'components/CodeEditor';
import { stateStore } from 'setup';
import _ from 'lodash-es';
type Props = FieldProps;
export enum Types {
URL = 'URL',
STRING = 'STRING',
NUMBER = 'NUMBER',
BOOLEAN = 'BOOLEAN',
OBJECT = 'OBJECT',
ARRAY = 'ARRAY',
FUNCTION = 'FUNCTION',
UNDEFINED = 'UNDEFINED',
NULL = 'NULL',
UNKNOWN = 'UNKNOWN',
}
export const getType = (value: unknown) => {
if (_.isString(value)) return Types.STRING;
if (_.isNumber(value)) return Types.NUMBER;
if (_.isBoolean(value)) return Types.BOOLEAN;
if (Array.isArray(value)) return Types.ARRAY;
if (_.isFunction(value)) return Types.FUNCTION;
if (_.isObject(value)) return Types.OBJECT;
if (_.isUndefined(value)) return Types.UNDEFINED;
if (_.isNull(value)) return Types.NULL;
return Types.UNKNOWN;
};
function generateTypeDef(
obj: any
): string | Record<string, string | Record<string, unknown>> {
const type = getType(obj);
switch (type) {
case Types.ARRAY: {
const arrayType = getType(obj[0]);
return `[${arrayType}]`;
}
case Types.OBJECT: {
const objType: Record<string, string | Record<string, unknown>> = {};
Object.keys(obj).forEach(k => {
objType[k] = generateTypeDef(obj[k]);
});
return objType;
}
case Types.STRING:
return 'string';
case Types.NUMBER:
return 'number';
case Types.BOOLEAN:
return 'bool';
case Types.NULL:
case Types.UNDEFINED:
return '?';
default:
return '?';
}
}
let extraDefs: any = {};
const customTreeTypeDefCreator = (dataTree: Record<string, Record<string, unknown>>) => {
const def: any = {
'!name': 'customDataTree',
};
Object.keys(dataTree).forEach(entityName => {
const entity = dataTree[entityName];
def[entityName] = generateTypeDef(entity);
});
def['!define'] = { ...extraDefs };
extraDefs = {};
return { ...def };
};
const GeneralWidget: React.FC<Props> = props => {
const { formData, onChange } = props;
const [defs, setDefs] = useState<any>();
useEffect(() => {
setDefs([customTreeTypeDefCreator(stateStore)]);
}, []);
return (
<TernEditor
@ -14,6 +90,7 @@ const GeneralWidget: React.FC<Props> = props => {
onBlur={v => {
onChange(v);
}}
defs={defs}
/>
);
};