Fix:Dynamically importing jsonEditor reduces packaging time.

This commit is contained in:
chanx 2025-12-03 15:51:36 +08:00
parent a6681d6366
commit 24bdf1731a
2 changed files with 92 additions and 47 deletions

View file

@ -1,8 +1,9 @@
import React, { useEffect, useRef } from 'react'; import React, { useEffect, useRef, useState } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import './css/cloud9_night.less'; import './css/cloud9_night.less';
import './css/index.less'; import './css/index.less';
import { JsonEditorOptions, JsonEditorProps } from './interface'; import { JsonEditorOptions, JsonEditorProps } from './interface';
const defaultConfig: JsonEditorOptions = { const defaultConfig: JsonEditorOptions = {
mode: 'code', mode: 'code',
modes: ['tree', 'code'], modes: ['tree', 'code'],
@ -14,6 +15,7 @@ const defaultConfig: JsonEditorOptions = {
enableTransform: false, enableTransform: false,
indentation: 2, indentation: 2,
}; };
const JsonEditor: React.FC<JsonEditorProps> = ({ const JsonEditor: React.FC<JsonEditorProps> = ({
value, value,
onChange, onChange,
@ -25,43 +27,62 @@ const JsonEditor: React.FC<JsonEditorProps> = ({
const editorRef = useRef<any>(null); const editorRef = useRef<any>(null);
const { i18n } = useTranslation(); const { i18n } = useTranslation();
const currentLanguageRef = useRef<string>(i18n.language); const currentLanguageRef = useRef<string>(i18n.language);
const [isLoading, setIsLoading] = useState(true);
useEffect(() => { useEffect(() => {
if (typeof window !== 'undefined') { let isMounted = true;
const JSONEditor = require('jsoneditor');
import('jsoneditor/dist/jsoneditor.min.css');
if (containerRef.current) { const initEditor = async () => {
// Default configuration options if (typeof window !== 'undefined') {
const defaultOptions: JsonEditorOptions = { try {
...defaultConfig, const JSONEditorModule = await import('jsoneditor');
language: i18n.language === 'zh' ? 'zh-CN' : 'en', const JSONEditor = JSONEditorModule.default || JSONEditorModule;
onChange: () => {
if (editorRef.current && onChange) { await import('jsoneditor/dist/jsoneditor.min.css');
try {
const updatedJson = editorRef.current.get(); if (isMounted && containerRef.current) {
onChange(updatedJson); // Default configuration options
} catch (err) { const defaultOptions: JsonEditorOptions = {
// Do not trigger onChange when parsing error occurs ...defaultConfig,
console.error(err); language: i18n.language === 'zh' ? 'zh-CN' : 'en',
} onChange: () => {
if (editorRef.current && onChange) {
try {
const updatedJson = editorRef.current.get();
onChange(updatedJson);
} catch (err) {
// Do not trigger onChange when parsing error occurs
console.error(err);
}
}
},
...options, // Merge user provided options with defaults
};
editorRef.current = new JSONEditor(
containerRef.current,
defaultOptions,
);
if (value) {
editorRef.current.set(value);
} }
},
...options, // Merge user provided options with defaults
};
editorRef.current = new JSONEditor( setIsLoading(false);
containerRef.current, }
defaultOptions, } catch (error) {
); console.error('Failed to load jsoneditor:', error);
if (isMounted) {
if (value) { setIsLoading(false);
editorRef.current.set(value); }
} }
} }
} };
initEditor();
return () => { return () => {
isMounted = false;
if (editorRef.current) { if (editorRef.current) {
if (typeof editorRef.current.destroy === 'function') { if (typeof editorRef.current.destroy === 'function') {
editorRef.current.destroy(); editorRef.current.destroy();
@ -92,26 +113,38 @@ const JsonEditor: React.FC<JsonEditorProps> = ({
} }
// Recreate the editor with new language // Recreate the editor with new language
const JSONEditor = require('jsoneditor'); const initEditorWithNewLanguage = async () => {
try {
const JSONEditorModule = await import('jsoneditor');
const JSONEditor = JSONEditorModule.default || JSONEditorModule;
const newOptions: JsonEditorOptions = { const newOptions: JsonEditorOptions = {
...defaultConfig, ...defaultConfig,
language: i18n.language === 'zh' ? 'zh-CN' : 'en', language: i18n.language === 'zh' ? 'zh-CN' : 'en',
onChange: () => { onChange: () => {
if (editorRef.current && onChange) { if (editorRef.current && onChange) {
try { try {
const updatedJson = editorRef.current.get(); const updatedJson = editorRef.current.get();
onChange(updatedJson); onChange(updatedJson);
} catch (err) { } catch (err) {
// Do not trigger onChange when parsing error occurs // Do not trigger onChange when parsing error occurs
} }
} }
}, },
...options, // Merge user provided options with defaults ...options, // Merge user provided options with defaults
};
editorRef.current = new JSONEditor(containerRef.current, newOptions);
editorRef.current.set(currentData);
} catch (error) {
console.error(
'Failed to reload jsoneditor with new language:',
error,
);
}
}; };
editorRef.current = new JSONEditor(containerRef.current, newOptions); initEditorWithNewLanguage();
editorRef.current.set(currentData);
} }
}, [i18n.language, value, onChange, options]); }, [i18n.language, value, onChange, options]);
@ -135,7 +168,13 @@ const JsonEditor: React.FC<JsonEditorProps> = ({
ref={containerRef} ref={containerRef}
style={{ height }} style={{ height }}
className={`ace-tomorrow-night w-full border border-border-button rounded-lg overflow-hidden bg-bg-input ${className} `} className={`ace-tomorrow-night w-full border border-border-button rounded-lg overflow-hidden bg-bg-input ${className} `}
/> >
{isLoading && (
<div className="flex items-center justify-center h-full">
<div className="text-text-secondary">Loading editor...</div>
</div>
)}
</div>
); );
}; };

6
web/src/custom.d.ts vendored
View file

@ -2,3 +2,9 @@ declare module '*.md' {
const content: string; const content: string;
export default content; export default content;
} }
declare module 'jsoneditor' {
const JSONEditor: any;
export default JSONEditor;
export = JSONEditor;
}