mirror of
https://github.com/go-gitea/gitea.git
synced 2026-04-02 00:18:35 +01:00
- Replace monaco-editor with CodeMirror 6 - Add `--color-syntax-*` CSS variables for all syntax token types, shared by CodeMirror, Chroma and EasyMDE - Consolidate chroma CSS into a single theme-independent file (`modules/chroma.css`) - Syntax colors in the code editor now match the code view and light/dark themes - Code editor is now 12px instead of 14px font size to match code view and GitHub - Use a global style for kbd elements - When editing existing files, focus will be on codemirror instead of filename input. - Keyboard shortcuts are roughtly the same as VSCode - Add a "Find" button, useful for mobile - Add context menu similar to Monaco - Add a command palette (Ctrl/Cmd+Shift+P or F1) or via button - Add clickable URLs via Ctrl/Cmd+click - Add e2e test for the code editor - Remove `window.codeEditors` global - The main missing Monaco features are hover types and semantic rename but these were not fully working because monaco operated only on single files and only for JS/TS/HTML/CSS/JSON. | | Monaco (main) | CodeMirror (cm) | Delta | |---|---|---|---| | **Build time** | 7.8s | 5.3s | **-32%** | | **JS output** | 25 MB | 14 MB | **-44%** | | **CSS output** | 1.2 MB | 1012 KB | **-17%** | | **Total (no maps)** | 23.3 MB | 12.1 MB | **-48%** | Fixes: #36311 Fixes: #14776 Fixes: #12171 <img width="1333" height="555" alt="image" src="https://github.com/user-attachments/assets/f0fe3a28-1ed9-4f22-bf25-2b161501d7ce" /> --------- Signed-off-by: silverwind <me@silverwind.io> Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com> Co-authored-by: wxiaoguang <wxiaoguang@gmail.com> Co-authored-by: Giteabot <teabot@gitea.io> Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com>
570 lines
14 KiB
CSS
570 lines
14 KiB
CSS
.code-editor-options {
|
|
flex-wrap: wrap;
|
|
}
|
|
|
|
.editor-loading {
|
|
padding: 1rem;
|
|
text-align: center;
|
|
}
|
|
|
|
.editor-loading.is-loading {
|
|
width: 100%;
|
|
min-height: 200px;
|
|
height: 90vh;
|
|
}
|
|
|
|
.edit.githook .code-editor-container {
|
|
border: 1px solid var(--color-secondary);
|
|
}
|
|
|
|
/* editor layout */
|
|
.code-editor-container .cm-editor {
|
|
color: var(--color-text);
|
|
background-color: var(--color-code-bg);
|
|
font-family: var(--fonts-monospace);
|
|
font-size: 12px;
|
|
max-height: 90vh;
|
|
}
|
|
|
|
.code-editor-container .cm-editor,
|
|
.code-editor-container .cm-editor .cm-scroller {
|
|
border-radius: 0 0 var(--border-radius) var(--border-radius);
|
|
}
|
|
|
|
.code-editor-container .cm-content,
|
|
.code-editor-container .cm-gutter {
|
|
min-height: 200px;
|
|
}
|
|
|
|
.code-editor-container .cm-scroller {
|
|
overflow: auto;
|
|
line-height: var(--line-height-code);
|
|
}
|
|
|
|
.code-editor-container .cm-content * {
|
|
caret-color: inherit;
|
|
}
|
|
|
|
.code-editor-container .cm-content {
|
|
padding: 0;
|
|
}
|
|
|
|
.code-editor-container .cm-cursor,
|
|
.code-editor-container .cm-dropCursor {
|
|
border-left-color: var(--color-caret);
|
|
}
|
|
|
|
.code-editor-container .cm-editor.cm-focused {
|
|
outline: none;
|
|
}
|
|
|
|
.code-editor-container .cm-editor.cm-focused > .cm-scroller > .cm-selectionLayer .cm-selectionBackground,
|
|
.code-editor-container .cm-selectionBackground {
|
|
background-color: var(--color-primary-alpha-30);
|
|
}
|
|
|
|
.code-editor-container .cm-panels {
|
|
background-color: var(--color-body);
|
|
color: var(--color-text);
|
|
border-color: var(--color-secondary);
|
|
font-family: var(--fonts-regular);
|
|
font-size: 13px;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search {
|
|
display: flex;
|
|
flex-wrap: wrap;
|
|
align-items: center;
|
|
gap: 4px;
|
|
position: relative;
|
|
padding-right: 24px;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search br {
|
|
display: none;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search::after {
|
|
content: "";
|
|
flex-basis: 100%;
|
|
order: 1;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search .cm-textfield {
|
|
width: 200px;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search br + input.cm-textfield,
|
|
.code-editor-container .cm-panel.cm-search br + input.cm-textfield ~ * {
|
|
order: 2;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search input,
|
|
.code-editor-container .cm-panel.cm-search button,
|
|
.code-editor-container .cm-panel.cm-search label {
|
|
margin: 0;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search .cm-button + label,
|
|
.code-editor-container .cm-panel.cm-search label + label {
|
|
margin-left: 4px;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search label {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
gap: 4px;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search input[type="checkbox"] {
|
|
cursor: pointer;
|
|
}
|
|
|
|
.code-editor-container .cm-editor .cm-panel button[name="close"] {
|
|
font-size: 0;
|
|
width: 16px;
|
|
height: 16px;
|
|
background-color: currentcolor;
|
|
mask-image: var(--octicon-x);
|
|
mask-size: cover;
|
|
-webkit-mask-image: var(--octicon-x);
|
|
-webkit-mask-size: cover;
|
|
border-radius: var(--border-radius);
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-search button[name="close"] {
|
|
position: absolute;
|
|
right: 8px;
|
|
top: 8px;
|
|
}
|
|
|
|
.code-editor-container .cm-editor .cm-panel button[name="close"]:focus-visible {
|
|
outline: 2px solid var(--color-primary);
|
|
outline-offset: 2px;
|
|
}
|
|
|
|
.code-editor-container .cm-activeLine,
|
|
.code-editor-container .cm-activeLineGutter {
|
|
background-color: var(--color-editor-line-highlight);
|
|
}
|
|
|
|
.code-editor-container .cm-gutters {
|
|
background-color: var(--color-code-bg);
|
|
color: var(--color-secondary-dark-6);
|
|
border-right: none;
|
|
}
|
|
|
|
.code-editor-container .cm-gutters .cm-lineNumbers .cm-gutterElement {
|
|
min-width: 30px;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: flex-end;
|
|
}
|
|
|
|
.code-editor-container .cm-foldGutter .cm-gutterElement {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
padding: 0 2px;
|
|
overflow: hidden;
|
|
}
|
|
|
|
.code-editor-container .cm-foldGutter .cm-gutterElement svg {
|
|
color: var(--color-text-light-2);
|
|
cursor: pointer;
|
|
visibility: hidden;
|
|
}
|
|
|
|
.code-editor-container .cm-gutters:hover .cm-foldGutter .cm-gutterElement svg {
|
|
visibility: visible;
|
|
}
|
|
|
|
.code-editor-container .cm-foldGutter .cm-gutterElement:hover svg {
|
|
color: var(--color-text);
|
|
}
|
|
|
|
.code-editor-container .cm-gutters .cm-lineNumbers .cm-activeLineGutter {
|
|
color: var(--color-text-light);
|
|
}
|
|
|
|
.code-editor-container .cm-editor .cm-line ::selection,
|
|
.code-editor-container .cm-editor .cm-line::selection {
|
|
color: currentcolor;
|
|
background-color: var(--color-editor-selection);
|
|
}
|
|
|
|
.code-editor-container .cm-foldPlaceholder {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
background: none;
|
|
border: none;
|
|
color: var(--color-text-light-2);
|
|
cursor: pointer;
|
|
vertical-align: middle;
|
|
padding: 0 1px;
|
|
margin: 0 2px;
|
|
}
|
|
|
|
.code-editor-container .cm-foldPlaceholder:hover {
|
|
color: var(--color-text);
|
|
}
|
|
|
|
.code-editor-container .cm-searchMatch {
|
|
background-color: var(--color-highlight-bg);
|
|
outline: 1px solid var(--color-highlight-fg);
|
|
}
|
|
|
|
.code-editor-container .cm-searchMatch-selected {
|
|
background-color: var(--color-primary-alpha-30);
|
|
}
|
|
|
|
.code-editor-container .cm-selectionMatch {
|
|
background-color: var(--color-highlight-bg);
|
|
}
|
|
|
|
.code-editor-container .cm-tooltip {
|
|
background-color: var(--color-body);
|
|
color: var(--color-text);
|
|
border-color: var(--color-secondary);
|
|
}
|
|
|
|
.code-editor-container .cm-tooltip.cm-tooltip-autocomplete > ul > li {
|
|
display: flex;
|
|
align-items: center;
|
|
padding: 4px;
|
|
}
|
|
|
|
.code-editor-container .cm-completionIcon {
|
|
display: flex;
|
|
align-items: center;
|
|
font-size: 16px;
|
|
padding-right: 4px;
|
|
color: var(--color-text-light);
|
|
opacity: 1;
|
|
}
|
|
|
|
.code-editor-container .cm-tooltip-autocomplete ul li[aria-selected] {
|
|
background-color: var(--color-primary-alpha-30);
|
|
color: var(--color-text);
|
|
}
|
|
|
|
.code-editor-container .cm-completionMatchedText {
|
|
text-decoration: none;
|
|
color: var(--color-primary-dark-1);
|
|
font-weight: var(--font-weight-semibold);
|
|
}
|
|
|
|
.code-editor-container .cm-placeholder {
|
|
color: var(--color-placeholder-text);
|
|
}
|
|
|
|
.code-editor-container .cm-button {
|
|
background-image: none;
|
|
background-color: var(--color-button);
|
|
color: var(--color-text);
|
|
border: 1px solid var(--color-secondary);
|
|
border-radius: var(--border-radius);
|
|
height: 28px;
|
|
padding: 0 10px;
|
|
cursor: pointer;
|
|
font-size: 12px;
|
|
}
|
|
|
|
.code-editor-container .cm-button:hover {
|
|
background-color: var(--color-hover);
|
|
}
|
|
|
|
.code-editor-container .cm-textfield {
|
|
background-color: var(--color-input-background);
|
|
color: var(--color-input-text);
|
|
border: 1px solid var(--color-input-border);
|
|
border-radius: var(--border-radius);
|
|
height: 28px;
|
|
padding: 0 6px;
|
|
}
|
|
|
|
.code-editor-container .cm-textfield:focus {
|
|
border-color: var(--color-primary);
|
|
}
|
|
|
|
.code-editor-container .cm-specialChar {
|
|
color: var(--color-syntax-invalid);
|
|
}
|
|
|
|
.code-editor-container .cm-trailingSpace {
|
|
background-color: var(--color-error-bg);
|
|
}
|
|
|
|
.code-editor-container .cm-activeLine .cm-trailingSpace {
|
|
background-color: transparent;
|
|
}
|
|
|
|
.code-editor-container.cm-mod-held .cm-url {
|
|
text-decoration: underline dotted var(--color-syntax-link);
|
|
cursor: pointer;
|
|
}
|
|
|
|
.code-editor-container .cm-editor.cm-focused .cm-matchingBracket {
|
|
background-color: var(--color-syntax-matching-bracket-bg);
|
|
}
|
|
|
|
.code-editor-container .cm-editor.cm-focused .cm-nonmatchingBracket {
|
|
background-color: var(--color-syntax-nonmatching-bracket-bg);
|
|
}
|
|
|
|
.code-editor-container .cm-panels-top {
|
|
border-bottom-color: var(--color-secondary);
|
|
}
|
|
|
|
.code-editor-container .cm-panels-bottom {
|
|
border-top-color: var(--color-secondary);
|
|
border-radius: 0 0 var(--border-radius) var(--border-radius);
|
|
}
|
|
|
|
.code-editor-container .cm-snippetField {
|
|
background-color: var(--color-primary-alpha-10);
|
|
}
|
|
|
|
.code-editor-container .cm-snippetFieldPosition {
|
|
border-left-color: var(--color-text-light-3);
|
|
}
|
|
|
|
.code-editor-container .cm-tooltip.cm-tooltip-autocomplete > ul > completion-section {
|
|
border-bottom-color: var(--color-secondary);
|
|
}
|
|
|
|
.code-editor-container .cm-tooltip-autocomplete-disabled ul li[aria-selected] {
|
|
background-color: var(--color-secondary);
|
|
}
|
|
|
|
/* command palette */
|
|
.code-editor-container {
|
|
position: relative;
|
|
min-height: 90vh;
|
|
}
|
|
|
|
.cm-command-palette {
|
|
position: absolute;
|
|
top: 0;
|
|
left: 50%;
|
|
transform: translateX(-50%);
|
|
z-index: 301;
|
|
width: 400px;
|
|
max-width: min(calc(100% - 16px), 90vw);
|
|
background-color: var(--color-body);
|
|
border: 1px solid var(--color-secondary);
|
|
border-radius: 0 0 var(--border-radius) var(--border-radius);
|
|
box-shadow: 0 4px 12px var(--color-shadow);
|
|
font-family: var(--fonts-regular);
|
|
font-size: 13px;
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 4px;
|
|
padding: 4px;
|
|
}
|
|
|
|
.cm-command-palette-input {
|
|
display: block;
|
|
width: 100%;
|
|
box-sizing: border-box;
|
|
padding: 4px 6px !important;
|
|
border: 1px solid var(--color-secondary);
|
|
border-radius: var(--border-radius);
|
|
background-color: var(--color-input-background);
|
|
color: var(--color-input-text);
|
|
font-size: 13px;
|
|
outline: none;
|
|
}
|
|
|
|
.cm-command-palette-input:focus {
|
|
border-color: var(--color-primary);
|
|
}
|
|
|
|
.cm-command-palette-list {
|
|
position: relative;
|
|
max-height: calc(8 * 24px);
|
|
overflow-y: auto;
|
|
}
|
|
|
|
.cm-command-palette-item {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
height: 24px;
|
|
padding: 0 6px;
|
|
border-radius: 2px;
|
|
cursor: pointer;
|
|
}
|
|
|
|
.cm-command-palette-item[aria-selected="true"] {
|
|
background-color: var(--color-primary-alpha-30);
|
|
}
|
|
|
|
.cm-command-palette-empty {
|
|
color: var(--color-text-light);
|
|
text-align: center;
|
|
}
|
|
|
|
.cm-command-palette-label {
|
|
flex: 1;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
white-space: nowrap;
|
|
}
|
|
|
|
.cm-command-palette-label mark {
|
|
background: none;
|
|
color: var(--color-primary-dark-1);
|
|
font-weight: var(--font-weight-semibold);
|
|
}
|
|
|
|
.cm-command-palette-keys {
|
|
flex-shrink: 0;
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 2px;
|
|
margin-left: 8px;
|
|
}
|
|
|
|
.code-editor-container .cm-gutter-lint {
|
|
width: 14px;
|
|
}
|
|
|
|
.code-editor-container .cm-gutter-lint .cm-gutterElement {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
padding: 0;
|
|
}
|
|
|
|
.code-editor-container .cm-lint-marker-error {
|
|
content: "";
|
|
width: 8px;
|
|
height: 8px;
|
|
border-radius: 50%;
|
|
background-color: var(--color-red);
|
|
}
|
|
|
|
.code-editor-container .cm-lint-marker-warning {
|
|
content: "";
|
|
width: 8px;
|
|
height: 8px;
|
|
border-radius: 50%;
|
|
background-color: var(--color-yellow);
|
|
}
|
|
|
|
.code-editor-container .cm-lintRange-error {
|
|
background-image: none;
|
|
text-decoration: wavy underline var(--color-red);
|
|
text-underline-offset: 3px;
|
|
}
|
|
|
|
.code-editor-container .cm-lintRange-warning {
|
|
background-image: none;
|
|
text-decoration: wavy underline var(--color-yellow);
|
|
text-underline-offset: 3px;
|
|
}
|
|
|
|
.code-editor-container .cm-lintPoint-error::after {
|
|
border-bottom-color: var(--color-red);
|
|
}
|
|
|
|
.code-editor-container .cm-lintPoint-warning::after {
|
|
border-bottom-color: var(--color-yellow);
|
|
}
|
|
|
|
.code-editor-container .cm-tooltip-lint {
|
|
background-color: var(--color-body);
|
|
color: var(--color-text);
|
|
border-color: var(--color-secondary);
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-panel-lint button[name="close"] {
|
|
top: 4px;
|
|
right: 8px;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-panel-lint ul {
|
|
max-height: 120px;
|
|
outline: none;
|
|
}
|
|
|
|
.code-editor-container .cm-panel.cm-panel-lint [aria-selected] {
|
|
background-color: var(--color-primary-alpha-30) !important;
|
|
}
|
|
|
|
.code-editor-container .cm-diagnostic.cm-diagnostic-error {
|
|
border-left-color: var(--color-red);
|
|
}
|
|
|
|
.code-editor-container .cm-diagnostic.cm-diagnostic-warning {
|
|
border-left-color: var(--color-yellow);
|
|
}
|
|
|
|
/* syntax highlighting classes from @lezer/highlight classHighlighter */
|
|
.code-editor-container .tok-keyword,
|
|
.code-editor-container .tok-atom { color: var(--color-syntax-keyword); }
|
|
.code-editor-container .tok-bool { color: var(--color-syntax-bool); }
|
|
.code-editor-container .tok-variableName { color: var(--color-syntax-variable); }
|
|
.code-editor-container .tok-variableName2 { color: var(--color-syntax-keyword); }
|
|
.code-editor-container .tok-propertyName { color: var(--color-syntax-property); }
|
|
.code-editor-container .tok-typeName,
|
|
.code-editor-container .tok-className { color: var(--color-syntax-type); }
|
|
.code-editor-container .tok-namespace { color: var(--color-syntax-namespace); }
|
|
.code-editor-container .tok-macroName { color: var(--color-syntax-name); }
|
|
.code-editor-container .tok-labelName { color: var(--color-syntax-name); }
|
|
.code-editor-container .tok-number { color: var(--color-syntax-number); }
|
|
.code-editor-container .tok-string { color: var(--color-syntax-string); }
|
|
.code-editor-container .tok-string2 { color: var(--color-syntax-regexp); }
|
|
.code-editor-container .tok-operator { color: var(--color-syntax-operator); }
|
|
.code-editor-container .tok-punctuation { color: var(--color-syntax-punctuation); }
|
|
.code-editor-container .tok-comment { color: var(--color-syntax-comment); }
|
|
.code-editor-container .tok-meta { color: var(--color-syntax-preproc); }
|
|
.code-editor-container .tok-invalid { color: var(--color-syntax-invalid); }
|
|
.code-editor-container .tok-link { color: var(--color-syntax-link); }
|
|
.code-editor-container .tok-heading { color: var(--color-syntax-heading); }
|
|
.code-editor-container .tok-emphasis { color: var(--color-syntax-emph); font-style: italic; }
|
|
.code-editor-container .tok-strong { font-weight: var(--font-weight-bold); }
|
|
.code-editor-container .tok-inserted { color: var(--color-syntax-string); }
|
|
.code-editor-container .tok-deleted { color: var(--color-syntax-invalid); }
|
|
|
|
/* language-specific overrides */
|
|
.code-editor-container[data-language="json"] .tok-propertyName,
|
|
.code-editor-container[data-language="json5"] .tok-propertyName,
|
|
.code-editor-container[data-language="yaml"] .tok-propertyName { color: var(--color-syntax-tag); }
|
|
.code-editor-container[data-language="css"] .tok-propertyName { color: var(--color-syntax-name); }
|
|
.code-editor-container[data-language="html"] .tok-propertyName,
|
|
.code-editor-container[data-language="xml"] .tok-propertyName { color: var(--color-syntax-attribute); }
|
|
|
|
/* context menu — uses tippy "menu" theme for base styling */
|
|
.cm-context-menu {
|
|
min-width: 200px;
|
|
font-size: 13px;
|
|
user-select: none;
|
|
}
|
|
|
|
.cm-context-menu .item {
|
|
height: 24px !important;
|
|
padding: 0 12px !important;
|
|
cursor: pointer;
|
|
}
|
|
|
|
.cm-context-menu .item.disabled {
|
|
color: var(--color-text-light-3);
|
|
cursor: default;
|
|
pointer-events: none;
|
|
}
|
|
|
|
.cm-context-menu-keys {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 2px;
|
|
margin-left: auto;
|
|
padding-left: 30px;
|
|
}
|
|
|
|
.cm-context-menu-separator {
|
|
border-top: 1px solid var(--color-secondary);
|
|
margin: 4px 0;
|
|
}
|