rust/editors/code/src/highlighting.ts

142 lines
4.4 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
> {
const colorContrib = (
tag: string
): [string, vscode.TextEditorDecorationType] => {
const color = new vscode.ThemeColor('ralsp.' + tag);
const decor = vscode.window.createTextEditorDecorationType({
color
});
return [tag, decor];
};
2018-10-08 18:18:55 +00:00
2018-10-08 21:38:33 +00:00
const decorations: Iterable<
[string, vscode.TextEditorDecorationType]
> = [
colorContrib('comment'),
colorContrib('string'),
colorContrib('keyword'),
2019-05-21 13:28:10 +00:00
colorContrib('keyword.control'),
colorContrib('keyword.unsafe'),
colorContrib('function'),
colorContrib('parameter'),
2019-05-23 10:26:38 +00:00
colorContrib('constant'),
colorContrib('type'),
colorContrib('builtin'),
colorContrib('text'),
colorContrib('attribute'),
colorContrib('literal'),
2019-05-23 10:26:38 +00:00
colorContrib('macro'),
colorContrib('variable'),
colorContrib('variable.mut'),
2019-05-23 10:26:38 +00:00
colorContrib('field'),
colorContrib('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();
const colorfulIdents: Map<string, vscode.Range[]> = 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)) {
colorfulIdents.set(d.bindingHash, []);
}
colorfulIdents
2019-05-27 09:26:15 +00:00
.get(d.bindingHash)!
.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
}
for (const [hash, ranges] of colorfulIdents.entries()) {
const dec = vscode.window.createTextEditorDecorationType({
light: { color: fancify(hash, 'light') },
dark: { color: fancify(hash, 'dark') }
});
editor.setDecorations(dec, ranges);
}
2018-10-07 20:44:25 +00:00
}
}