0
0
mirror of https://github.com/naturalcrit/homebrewery.git synced 2025-12-25 14:02:40 +00:00
This commit is contained in:
Trevor Buckner
2025-11-17 16:15:44 +00:00
parent 677e8eaf6c
commit e497e42913
8 changed files with 290 additions and 50 deletions

View File

@@ -8,36 +8,36 @@ const autoCompleteEmoji = require('./autocompleteEmoji');
let CodeMirror;
if(typeof window !== 'undefined'){
CodeMirror = require('codemirror');
CodeMirror = require('codemirror5');
//Language Modes
require('codemirror/mode/gfm/gfm.js'); //Github flavoured markdown
require('codemirror/mode/css/css.js');
require('codemirror/mode/javascript/javascript.js');
require('codemirror5/mode/gfm/gfm.js'); //Github flavoured markdown
require('codemirror5/mode/css/css.js');
require('codemirror5/mode/javascript/javascript.js');
//Addons
//Code folding
require('codemirror/addon/fold/foldcode.js');
require('codemirror/addon/fold/foldgutter.js');
require('codemirror5/addon/fold/foldcode.js');
require('codemirror5/addon/fold/foldgutter.js');
//Search and replace
require('codemirror/addon/search/search.js');
require('codemirror/addon/search/searchcursor.js');
require('codemirror/addon/search/jump-to-line.js');
require('codemirror/addon/search/match-highlighter.js');
require('codemirror/addon/search/matchesonscrollbar.js');
require('codemirror/addon/dialog/dialog.js');
require('codemirror5/addon/search/search.js');
require('codemirror5/addon/search/searchcursor.js');
require('codemirror5/addon/search/jump-to-line.js');
require('codemirror5/addon/search/match-highlighter.js');
require('codemirror5/addon/search/matchesonscrollbar.js');
require('codemirror5/addon/dialog/dialog.js');
//Trailing space highlighting
// require('codemirror/addon/edit/trailingspace.js');
//Active line highlighting
// require('codemirror/addon/selection/active-line.js');
//Scroll past last line
require('codemirror/addon/scroll/scrollpastend.js');
require('codemirror5/addon/scroll/scrollpastend.js');
//Auto-closing
//XML code folding is a requirement of the auto-closing tag feature and is not enabled
require('codemirror/addon/fold/xml-fold.js');
require('codemirror/addon/edit/closetag.js');
require('codemirror5/addon/fold/xml-fold.js');
require('codemirror5/addon/edit/closetag.js');
//Autocompletion
require('codemirror/addon/hint/show-hint.js');
require('codemirror5/addon/hint/show-hint.js');
const foldPagesCode = require('./fold-pages');
foldPagesCode.registerHomebreweryHelper(CodeMirror);
@@ -462,4 +462,3 @@ const CodeEditor = createClass({
});
module.exports = CodeEditor;

View File

@@ -1,8 +1,8 @@
@import (less) 'codemirror/lib/codemirror.css';
@import (less) 'codemirror/addon/fold/foldgutter.css';
@import (less) 'codemirror/addon/search/matchesonscrollbar.css';
@import (less) 'codemirror/addon/dialog/dialog.css';
@import (less) 'codemirror/addon/hint/show-hint.css';
@import (less) 'codemirror5/lib/codemirror.css';
@import (less) 'codemirror5/addon/fold/foldgutter.css';
@import (less) 'codemirror5/addon/search/matchesonscrollbar.css';
@import (less) 'codemirror5/addon/dialog/dialog.css';
@import (less) 'codemirror5/addon/hint/show-hint.css';
//Icon fonts included so they can appear in emoji autosuggest dropdown
@import (less) './themes/fonts/iconFonts/diceFont.less';
@@ -57,4 +57,4 @@
.emojiPreview {
font-size : 1.5em;
line-height : 1.2em;
}
}

View File

@@ -0,0 +1,191 @@
import './codeEditor.less';
import React, { useEffect, useImperativeHandle, useMemo, useRef } from 'react';
import { Compartment, EditorSelection, EditorState } from '@codemirror/state';
import { EditorView, keymap, highlightActiveLine, lineNumbers, highlightActiveLineGutter } from '@codemirror/view';
import { history, historyKeymap, undo as historyUndo, redo as historyRedo, undoDepth, redoDepth } from '@codemirror/history';
import { defaultKeymap, indentWithTab } from '@codemirror/commands';
import { searchKeymap, highlightSelectionMatches } from '@codemirror/search';
import { closeBrackets, closeBracketsKeymap } from '@codemirror/autocomplete';
import { defaultHighlightStyle } from '@codemirror/language';
import { syntaxHighlighting } from '@codemirror/language';
import { markdown, markdownLanguage } from '@codemirror/lang-markdown';
import { languages } from '@codemirror/language-data';
import { javascript } from '@codemirror/lang-javascript';
import { css } from '@codemirror/lang-css';
const baseExtensions = [
lineNumbers(),
highlightActiveLineGutter(),
EditorView.lineWrapping,
history(),
keymap.of([
indentWithTab,
...closeBracketsKeymap,
...searchKeymap,
...historyKeymap,
...defaultKeymap
]),
closeBrackets(),
highlightSelectionMatches(),
highlightActiveLine(),
syntaxHighlighting(defaultHighlightStyle, { fallback: true })
];
const languageExtension = (language)=>{
switch(language){
case 'css':
return css();
case 'javascript':
return javascript({ jsx: true, typescript: true });
case 'gfm':
default:
return markdown({ base: markdownLanguage, codeLanguages: languages });
}
};
const CodeEditorV6 = React.forwardRef(({
value = '',
onChange = ()=>{},
language = 'gfm',
readOnly = false,
style = {},
editorTheme,
onSelectionChange = ()=>{},
onScroll = ()=>{}
}, ref)=>{
const containerRef = useRef(null);
const viewRef = useRef(null);
const languageCompartment = useMemo(()=>new Compartment(), []);
const readOnlyCompartment = useMemo(()=>new Compartment(), []);
useImperativeHandle(ref, ()=>({
focus : ()=>viewRef.current?.focus(),
getView : ()=>viewRef.current,
getCM6View : ()=>viewRef.current,
getCursorPosition : ()=>{
const view = viewRef.current;
if(!view) return { line: 0, ch: 0 };
const { head } = view.state.selection.main;
const lineInfo = view.state.doc.lineAt(head);
return { line: lineInfo.number - 1, ch: head - lineInfo.from };
},
setCursorPosition : ({ line = 0, ch = 0 })=>{
const view = viewRef.current;
if(!view) return;
const docLine = view.state.doc.line(Math.max(1, line + 1));
const pos = Math.min(docLine.from + ch, docLine.to);
view.dispatch({ selection: EditorSelection.cursor(pos), scrollIntoView: true });
},
getTopVisibleLine : ()=>{
const view = viewRef.current;
if(!view) return 0;
const top = view.scrollDOM.scrollTop;
const block = view.lineBlockAtHeight(top);
const lineInfo = view.state.doc.lineAt(block.from);
return lineInfo.number - 1;
},
updateSize : ()=>viewRef.current?.requestMeasure(),
injectText : (text, overwrite=true)=>{
const view = viewRef.current;
if(!view) return;
const { from, to } = view.state.selection.main;
const insertFrom = overwrite ? from : from;
const insertTo = overwrite ? to : from;
view.dispatch({
changes : { from: insertFrom, to: insertTo, insert: text },
selection : EditorSelection.cursor(insertFrom + text.length),
scrollIntoView : true
});
view.focus();
},
undo : ()=>{
const view = viewRef.current;
if(!view) return;
historyUndo(view);
},
redo : ()=>{
const view = viewRef.current;
if(!view) return;
historyRedo(view);
},
historySize : ()=>{
const view = viewRef.current;
if(!view) return { undo: 0, redo: 0 };
return {
undo : undoDepth(view.state),
redo : redoDepth(view.state)
};
},
foldAllCode : ()=>{},
unfoldAllCode : ()=>{}
}), []);
useEffect(()=>{
if(!containerRef.current) return;
const initialExtensions = [
...baseExtensions,
languageCompartment.of(languageExtension(language)),
readOnlyCompartment.of(EditorState.readOnly.of(readOnly)),
EditorView.updateListener.of((update)=>{
if(update.docChanged) onChange(update.state.doc.toString());
if(update.selectionSet) onSelectionChange(update.view);
})
];
const state = EditorState.create({
doc : value,
extensions : initialExtensions
});
const view = new EditorView({
state,
parent : containerRef.current
});
viewRef.current = view;
const handleScroll = ()=>onScroll(view);
view.scrollDOM.addEventListener('scroll', handleScroll);
return ()=>{
view.scrollDOM.removeEventListener('scroll', handleScroll);
view.destroy();
viewRef.current = null;
};
}, []);
useEffect(()=>{
const view = viewRef.current;
if(!view) return;
view.dispatch({
effects : languageCompartment.reconfigure(languageExtension(language))
});
}, [language, languageCompartment]);
useEffect(()=>{
const view = viewRef.current;
if(!view) return;
view.dispatch({
effects : readOnlyCompartment.reconfigure(EditorState.readOnly.of(readOnly))
});
}, [readOnly, readOnlyCompartment]);
useEffect(()=>{
const view = viewRef.current;
if(!view) return;
const currentValue = view.state.doc.toString();
if(value === currentValue) return;
const transaction = view.state.update({
changes : { from: 0, to: currentValue.length, insert: value }
});
view.dispatch(transaction);
}, [value]);
return (
<div className='codeEditor' ref={containerRef} style={style} data-editor='cm6'></div>
);
});
CodeEditorV6.displayName = 'CodeEditorV6';
export default CodeEditorV6;