rust/editors/code/src/highlighting.ts

150 lines
4.7 KiB
TypeScript
Raw Normal View History

import seedrandom = require('seedrandom');
2018-10-07 20:44:25 +00:00
import * as vscode from 'vscode';
2018-10-07 20:59:02 +00:00
import * as lc from 'vscode-languageclient';
2018-10-07 20:44:25 +00:00
import { Server } from './server';
export interface Decoration {
2018-10-07 20:59:02 +00:00
range: lc.Range;
tag: string;
2019-05-27 09:26:15 +00:00
bindingHash?: string;
}
// Based on this HSL-based color generator: https://gist.github.com/bendc/76c48ce53299e6078a76
function fancify(seed: string, shade: 'light' | 'dark') {
const random = seedrandom(seed);
const randomInt = (min: number, max: number) => {
return Math.floor(random() * (max - min + 1)) + min;
};
const h = randomInt(0, 360);
const s = randomInt(42, 98);
const l = shade === 'light' ? randomInt(15, 40) : randomInt(40, 90);
return `hsl(${h},${s}%,${l}%)`;
2018-10-07 20:44:25 +00:00
}
export class Highlighter {
2018-10-08 21:38:33 +00:00
private static initDecorations(): Map<
string,
vscode.TextEditorDecorationType
> {
2019-07-19 11:43:36 +00:00
const decoration = (
tag: string,
textDecoration?: string
): [string, vscode.TextEditorDecorationType] => {
const color = new vscode.ThemeColor('ralsp.' + tag);
const decor = vscode.window.createTextEditorDecorationType({
2019-07-19 11:43:36 +00:00
color,
textDecoration
});
return [tag, decor];
};
2018-10-08 18:18:55 +00:00
2019-12-08 18:27:50 +00:00
const decorations: Iterable<[
string,
vscode.TextEditorDecorationType
]> = [
2019-07-19 11:43:36 +00:00
decoration('comment'),
decoration('string'),
decoration('keyword'),
decoration('keyword.control'),
decoration('keyword.unsafe'),
decoration('function'),
decoration('parameter'),
decoration('constant'),
decoration('type'),
decoration('builtin'),
decoration('text'),
decoration('attribute'),
decoration('literal'),
decoration('macro'),
decoration('variable'),
decoration('variable.mut', 'underline'),
decoration('field'),
decoration('module')
2018-10-08 18:18:55 +00:00
];
return new Map<string, vscode.TextEditorDecorationType>(decorations);
2018-10-07 20:44:25 +00:00
}
2018-10-08 21:38:33 +00:00
private decorations: Map<
string,
vscode.TextEditorDecorationType
> | null = null;
2018-10-08 18:18:55 +00:00
2018-10-07 20:59:02 +00:00
public removeHighlights() {
2018-10-08 18:18:55 +00:00
if (this.decorations == null) {
return;
2018-10-07 20:44:25 +00:00
}
2018-10-08 18:18:55 +00:00
// Decorations are removed when the object is disposed
for (const decoration of this.decorations.values()) {
decoration.dispose();
}
this.decorations = null;
2018-10-07 20:44:25 +00:00
}
2018-10-08 21:38:33 +00:00
public setHighlights(editor: vscode.TextEditor, highlights: Decoration[]) {
2018-10-07 20:44:25 +00:00
// Initialize decorations if necessary
//
// Note: decoration objects need to be kept around so we can dispose them
// if the user disables syntax highlighting
2018-10-08 18:18:55 +00:00
if (this.decorations == null) {
this.decorations = Highlighter.initDecorations();
2018-10-07 20:44:25 +00:00
}
2018-10-07 20:59:02 +00:00
const byTag: Map<string, vscode.Range[]> = new Map();
2019-07-19 11:43:36 +00:00
const colorfulIdents: Map<
string,
[vscode.Range[], boolean]
> = new Map();
2019-05-27 09:26:15 +00:00
const rainbowTime = Server.config.rainbowHighlightingOn;
2018-10-08 18:18:55 +00:00
for (const tag of this.decorations.keys()) {
2018-10-07 20:59:02 +00:00
byTag.set(tag, []);
2018-10-07 20:44:25 +00:00
}
2018-10-07 20:59:02 +00:00
for (const d of highlights) {
2018-10-07 20:44:25 +00:00
if (!byTag.get(d.tag)) {
2018-10-07 20:59:02 +00:00
continue;
2018-10-07 20:44:25 +00:00
}
2019-05-27 09:26:15 +00:00
if (rainbowTime && d.bindingHash) {
if (!colorfulIdents.has(d.bindingHash)) {
2019-07-19 11:43:36 +00:00
const mut = d.tag.endsWith('.mut');
colorfulIdents.set(d.bindingHash, [[], mut]);
}
colorfulIdents
2019-07-19 11:43:36 +00:00
.get(d.bindingHash)![0]
.push(
Server.client.protocol2CodeConverter.asRange(d.range)
);
} else {
byTag
.get(d.tag)!
.push(
Server.client.protocol2CodeConverter.asRange(d.range)
);
}
2018-10-07 20:44:25 +00:00
}
2018-10-07 20:59:02 +00:00
for (const tag of byTag.keys()) {
2018-10-08 21:38:33 +00:00
const dec = this.decorations.get(
tag
) as vscode.TextEditorDecorationType;
2018-10-07 20:59:02 +00:00
const ranges = byTag.get(tag)!;
editor.setDecorations(dec, ranges);
2018-10-07 20:44:25 +00:00
}
2019-07-19 11:43:36 +00:00
for (const [hash, [ranges, mut]] of colorfulIdents.entries()) {
const textDecoration = mut ? 'underline' : undefined;
const dec = vscode.window.createTextEditorDecorationType({
2019-07-19 11:43:36 +00:00
light: { color: fancify(hash, 'light'), textDecoration },
dark: { color: fancify(hash, 'dark'), textDecoration }
});
editor.setDecorations(dec, ranges);
}
2018-10-07 20:44:25 +00:00
}
}