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>
48 lines
3.2 KiB
TypeScript
48 lines
3.2 KiB
TypeScript
import {findUrlAtPosition, trimUrlPunctuation, urlRawRegex} from './utils.ts';
|
|
|
|
function matchUrls(text: string): string[] {
|
|
return Array.from(text.matchAll(urlRawRegex), (m) => trimUrlPunctuation(m[0]));
|
|
}
|
|
|
|
test('matchUrls', () => {
|
|
expect(matchUrls('visit https://example.com for info')).toEqual(['https://example.com']);
|
|
expect(matchUrls('see https://example.com.')).toEqual(['https://example.com']);
|
|
expect(matchUrls('see https://example.com, and')).toEqual(['https://example.com']);
|
|
expect(matchUrls('see https://example.com; and')).toEqual(['https://example.com']);
|
|
expect(matchUrls('(https://example.com)')).toEqual(['https://example.com']);
|
|
expect(matchUrls('"https://example.com"')).toEqual(['https://example.com']);
|
|
expect(matchUrls('https://example.com/path?q=1&b=2#hash')).toEqual(['https://example.com/path?q=1&b=2#hash']);
|
|
expect(matchUrls('https://example.com/path?q=1&b=2#hash.')).toEqual(['https://example.com/path?q=1&b=2#hash']);
|
|
expect(matchUrls('https://x.co')).toEqual(['https://x.co']);
|
|
expect(matchUrls('https://example.com/path_(wiki)')).toEqual(['https://example.com/path_(wiki)']);
|
|
expect(matchUrls('https://en.wikipedia.org/wiki/Rust_(programming_language)')).toEqual(['https://en.wikipedia.org/wiki/Rust_(programming_language)']);
|
|
expect(matchUrls('(https://en.wikipedia.org/wiki/Rust_(programming_language))')).toEqual(['https://en.wikipedia.org/wiki/Rust_(programming_language)']);
|
|
expect(matchUrls('http://example.com')).toEqual(['http://example.com']);
|
|
expect(matchUrls('no url here')).toEqual([]);
|
|
expect(matchUrls('https://a.com and https://b.com')).toEqual(['https://a.com', 'https://b.com']);
|
|
expect(matchUrls('[](https://www.npmjs.org/package/pkg)')).toEqual(['https://img.shields.io/npm/v/pkg.svg?style=flat', 'https://www.npmjs.org/package/pkg']);
|
|
});
|
|
|
|
test('trimUrlPunctuation', () => {
|
|
expect(trimUrlPunctuation('https://example.com.')).toEqual('https://example.com');
|
|
expect(trimUrlPunctuation('https://example.com,')).toEqual('https://example.com');
|
|
expect(trimUrlPunctuation('https://example.com;')).toEqual('https://example.com');
|
|
expect(trimUrlPunctuation('https://example.com:')).toEqual('https://example.com');
|
|
expect(trimUrlPunctuation("https://example.com'")).toEqual('https://example.com');
|
|
expect(trimUrlPunctuation('https://example.com"')).toEqual('https://example.com');
|
|
expect(trimUrlPunctuation('https://example.com.,;')).toEqual('https://example.com');
|
|
expect(trimUrlPunctuation('https://example.com/path')).toEqual('https://example.com/path');
|
|
expect(trimUrlPunctuation('https://example.com/path_(wiki)')).toEqual('https://example.com/path_(wiki)');
|
|
expect(trimUrlPunctuation('https://example.com)')).toEqual('https://example.com');
|
|
expect(trimUrlPunctuation('https://en.wikipedia.org/wiki/Rust_(lang))')).toEqual('https://en.wikipedia.org/wiki/Rust_(lang)');
|
|
});
|
|
|
|
test('findUrlAtPosition', () => {
|
|
const doc = 'visit https://example.com for info';
|
|
expect(findUrlAtPosition(doc, 0)).toBeNull();
|
|
expect(findUrlAtPosition(doc, 6)).toEqual('https://example.com');
|
|
expect(findUrlAtPosition(doc, 15)).toEqual('https://example.com');
|
|
expect(findUrlAtPosition(doc, 24)).toEqual('https://example.com');
|
|
expect(findUrlAtPosition(doc, 25)).toBeNull();
|
|
});
|