Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 5ee2f8f

Browse files
committedAug 16, 2021
add file url handler
1 parent e30969a commit 5ee2f8f

File tree

4 files changed

+72
-21
lines changed

4 files changed

+72
-21
lines changed
 

‎lib/components/term.tsx

+21-13
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
import React from 'react';
22
import {Terminal, ITerminalOptions, IDisposable} from 'xterm';
33
import {FitAddon} from 'xterm-addon-fit';
4-
import {WebLinksAddon} from 'xterm-addon-web-links';
54
import {SearchAddon} from 'xterm-addon-search';
65
import {WebglAddon} from 'xterm-addon-webgl';
76
import {LigaturesAddon} from 'xterm-addon-ligatures';
@@ -13,6 +12,8 @@ import processClipboard from '../utils/paste';
1312
import SearchBox from './searchBox';
1413
import {TermProps} from '../hyper';
1514
import {ObjectTypedKeys} from '../utils/object';
15+
import {LinkProvider} from 'xterm-link-provider';
16+
import {FileURLMatcher, URLMatcher} from '../utils/link-matchers';
1617

1718
const isWindows = ['Windows', 'Win16', 'Win32', 'WinCE'].includes(navigator.platform);
1819

@@ -157,19 +158,26 @@ export default class Term extends React.PureComponent<TermProps> {
157158
this.term.attachCustomKeyEventHandler(this.keyboardHandler);
158159
this.term.loadAddon(this.fitAddon);
159160
this.term.loadAddon(this.searchAddon);
160-
this.term.loadAddon(
161-
new WebLinksAddon(
162-
(event: MouseEvent | undefined, uri: string) => {
163-
if (shallActivateWebLink(event)) void shell.openExternal(uri);
161+
this.term.registerLinkProvider(
162+
new LinkProvider(
163+
this.term,
164+
URLMatcher.regex,
165+
(event, text) => {
166+
if (shallActivateWebLink(event)) void shell.openExternal(text);
164167
},
165-
{
166-
// prevent default electron link handling to allow selection, e.g. via double-click
167-
willLinkActivate: (event: MouseEvent | undefined) => {
168-
event?.preventDefault();
169-
return shallActivateWebLink(event);
170-
},
171-
priority: Date.now()
172-
}
168+
{},
169+
URLMatcher.matchIndex
170+
)
171+
);
172+
this.term.registerLinkProvider(
173+
new LinkProvider(
174+
this.term,
175+
FileURLMatcher.regex,
176+
(event, text) => {
177+
if (shallActivateWebLink(event)) void shell.openExternal(text);
178+
},
179+
{},
180+
FileURLMatcher.matchIndex
173181
)
174182
);
175183
this.term.open(this.termRef);

‎lib/utils/link-matchers.ts

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
interface linkMatcher {
2+
regex: RegExp;
3+
matchIndex: number;
4+
}
5+
6+
// Based on the regex used in xterm-addon-web-links
7+
export const URLMatcher: linkMatcher = (() => {
8+
const protocolClause = '(https?:\\/\\/)';
9+
const domainCharacterSet = '[\\da-z\\.-]+';
10+
const negatedDomainCharacterSet = '[^\\da-z\\.-]+';
11+
const domainBodyClause = '(' + domainCharacterSet + ')';
12+
const tldClause = '([a-z\\.]{2,6})';
13+
const ipClause = '((\\d{1,3}\\.){3}\\d{1,3})';
14+
const localHostClause = '(localhost)';
15+
const portClause = '(:\\d{1,5})';
16+
const hostClause =
17+
'((' + domainBodyClause + '\\.' + tldClause + ')|' + ipClause + '|' + localHostClause + ')' + portClause + '?';
18+
const pathCharacterSet = '(\\/[\\/\\w\\.\\-%~:+@]*)*([^:"\'\\s])';
19+
const pathClause = '(' + pathCharacterSet + ')?';
20+
const queryStringHashFragmentCharacterSet = "[0-9\\w\\[\\]\\(\\)\\/\\?\\!#@$%&'*+,:;~\\=\\.\\-]*";
21+
const queryStringClause = '(\\?' + queryStringHashFragmentCharacterSet + ')?';
22+
const hashFragmentClause = '(#' + queryStringHashFragmentCharacterSet + ')?';
23+
const negatedPathCharacterSet = '[^\\/\\w\\.\\-%]+';
24+
const bodyClause = hostClause + pathClause + queryStringClause + hashFragmentClause;
25+
const start = '(?:^|' + negatedDomainCharacterSet + ')(';
26+
const end = ')($|' + negatedPathCharacterSet + ')';
27+
return {regex: new RegExp(start + protocolClause + bodyClause + end), matchIndex: 1};
28+
})();
29+
30+
// Simple file url matcher
31+
export const FileURLMatcher: linkMatcher = (() => {
32+
const protocolClause = '(file:\\/\\/)';
33+
const negatedDomainCharacterSet = '[^\\da-z\\.-]+';
34+
const pathCharacterSet = '(\\/[\\/\\w\\.\\-%~:+@]*)*([^:"\'\\s])';
35+
const pathClause = '(' + pathCharacterSet + ')';
36+
const negatedPathCharacterSet = '[^\\/\\w\\.\\-%]+';
37+
const bodyClause = pathClause;
38+
const start = '(?:^|' + negatedDomainCharacterSet + ')(';
39+
const end = ')($|' + negatedPathCharacterSet + ')';
40+
return {regex: new RegExp(start + protocolClause + bodyClause + end), matchIndex: 1};
41+
})();

‎package.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -56,8 +56,8 @@
5656
"xterm-addon-ligatures": "^0.5.1",
5757
"xterm-addon-search": "^0.8.0",
5858
"xterm-addon-unicode11": "^0.2.0",
59-
"xterm-addon-web-links": "^0.4.0",
60-
"xterm-addon-webgl": "^0.11.1"
59+
"xterm-addon-webgl": "^0.11.1",
60+
"xterm-link-provider": "1.3.1"
6161
},
6262
"devDependencies": {
6363
"@ava/babel": "2.0.0",

‎yarn.lock

+8-6
Original file line numberDiff line numberDiff line change
@@ -8432,17 +8432,19 @@ xterm-addon-unicode11@^0.2.0:
84328432
resolved "https://registry.npmjs.org/xterm-addon-unicode11/-/xterm-addon-unicode11-0.2.0.tgz#9ed0c482b353908bba27778893ca80823382737c"
84338433
integrity sha512-rjFDItPc/IDoSiEnoDFwKroNwLD/7t9vYKENjrcKVZg5tgJuuUj8D4rZtP6iVCjSB1LTLYmUs4L/EmCqIyLR/Q==
84348434

8435-
xterm-addon-web-links@^0.4.0:
8436-
version "0.4.0"
8437-
resolved "https://registry.npmjs.org/xterm-addon-web-links/-/xterm-addon-web-links-0.4.0.tgz#265cbf8221b9b315d0a748e1323bee331cd5da03"
8438-
integrity sha512-xv8GeiINmx0zENO9hf5k+5bnkaE8mRzF+OBAr9WeFq2eLaQSudioQSiT34M1ofKbzcdjSsKiZm19Rw3i4eXamg==
8439-
84408435
xterm-addon-webgl@^0.11.1:
84418436
version "0.11.1"
84428437
resolved "https://registry.npmjs.org/xterm-addon-webgl/-/xterm-addon-webgl-0.11.1.tgz#33dd250ab52e9f51d2ff52396447962e6f53e24c"
84438438
integrity sha512-xF6DnEoV+rPtzetMBXBZVe1kLKtus7AKdEcyfq2eMHQzhaRvC+pfnU+XiCXC85kueguqu2UkBHXZs5mihK9jOQ==
84448439

8445-
xterm@^4.13.0:
8440+
xterm-link-provider@1.3.1:
8441+
version "1.3.1"
8442+
resolved "https://registry.npmjs.org/xterm-link-provider/-/xterm-link-provider-1.3.1.tgz#69727223220dfa8758056ad6b2b5394a8454b9cb"
8443+
integrity sha512-uOlaIeUED6kJeL2nIIf5YwreO0obMhsC0RWypEUmWkz7SAQewzgwdWFjQ2He7NGcT93c4KUf8bRgAu8cV9bAYA==
8444+
dependencies:
8445+
xterm "^4.6.0"
8446+
8447+
xterm@^4.13.0, xterm@^4.6.0:
84468448
version "4.13.0"
84478449
resolved "https://registry.npmjs.org/xterm/-/xterm-4.13.0.tgz#7998de1e2ad92c4796fe45807be4f31061f3d9d1"
84488450
integrity sha512-HVW1gdoLOTnkMaqQCr2r3mQy4fX9iSa5gWxKZ2UTYdLa4iqavv7QxJ8n1Ypse32shPVkhTYPLS6vHEFZp5ghzw==

0 commit comments

Comments
 (0)
Please sign in to comment.