extensionEditor.ts 25.0 KB
Newer Older
J
Joao Moreno 已提交
1 2 3 4 5 6 7
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

'use strict';

J
Joao Moreno 已提交
8
import 'vs/css!./media/extensionEditor';
J
Joao Moreno 已提交
9
import { localize } from 'vs/nls';
J
Joao Moreno 已提交
10 11
import { TPromise } from 'vs/base/common/winjs.base';
import { marked } from 'vs/base/common/marked/marked';
J
Joao Moreno 已提交
12
import { always } from 'vs/base/common/async';
J
Joao Moreno 已提交
13
import * as arrays from 'vs/base/common/arrays';
J
Joao Moreno 已提交
14
import Event, { Emitter, once, fromEventEmitter, chain } from 'vs/base/common/event';
J
Joao Moreno 已提交
15
import Cache from 'vs/base/common/cache';
J
Joao Moreno 已提交
16
import { Action } from 'vs/base/common/actions';
17
import { isPromiseCanceledError } from 'vs/base/common/errors';
18
import Severity from 'vs/base/common/severity';
J
Joao Moreno 已提交
19
import { IDisposable, empty, dispose, toDisposable } from 'vs/base/common/lifecycle';
J
Joao Moreno 已提交
20
import { Builder } from 'vs/base/browser/builder';
21
import { domEvent } from 'vs/base/browser/event';
22
import { append, $, addClass, removeClass, finalHandler, join } from 'vs/base/browser/dom';
J
Joao Moreno 已提交
23
import { BaseEditor } from 'vs/workbench/browser/parts/editor/baseEditor';
B
Benjamin Pasero 已提交
24
import { IViewletService } from 'vs/workbench/services/viewlet/browser/viewlet';
J
Joao Moreno 已提交
25 26
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation';
27
import { IExtensionGalleryService, IExtensionManifest, IKeyBinding } from 'vs/platform/extensionManagement/common/extensionManagement';
J
Joao Moreno 已提交
28
import { IThemeService } from 'vs/workbench/services/themes/common/themeService';
29
import { ExtensionsInput } from 'vs/workbench/parts/extensions/common/extensionsInput';
30
import { IExtensionsWorkbenchService, IExtensionsViewlet, VIEWLET_ID, IExtension, IExtensionDependencies } from 'vs/workbench/parts/extensions/common/extensions';
31
import { Renderer, DataSource, Controller } from 'vs/workbench/parts/extensions/browser/dependenciesViewer';
32
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
33 34
import { ITemplateData } from 'vs/workbench/parts/extensions/browser/extensionsList';
import { RatingsWidget, InstallWidget } from 'vs/workbench/parts/extensions/browser/extensionsWidgets';
J
Joao Moreno 已提交
35
import { EditorOptions } from 'vs/workbench/common/editor';
J
Joao Moreno 已提交
36
import { ActionBar } from 'vs/base/browser/ui/actionbar/actionbar';
37
import { CombinedInstallAction, UpdateAction, EnableAction, DisableAction, BuiltinStatusLabelAction, ReloadAction } from 'vs/workbench/parts/extensions/browser/extensionsActions';
J
Joao Moreno 已提交
38
import WebView from 'vs/workbench/parts/html/browser/webview';
39 40
import { Keybinding } from 'vs/base/common/keyCodes';
import { KeybindingLabels } from 'vs/base/common/keybinding';
J
Johannes Rieken 已提交
41
import { IKeybindingService } from 'vs/platform/keybinding/common/keybinding';
J
Joao Moreno 已提交
42
import { DomScrollableElement } from 'vs/base/browser/ui/scrollbar/scrollableElement';
43
import { IMessageService } from 'vs/platform/message/common/message';
44
import { IOpenerService } from 'vs/platform/opener/common/opener';
S
Sandeep Somavarapu 已提交
45
import { Tree } from 'vs/base/parts/tree/browser/treeImpl';
J
Joao Moreno 已提交
46
import { Position } from 'vs/platform/editor/common/editor';
47
import { IListService } from 'vs/platform/list/browser/listService';
J
Joao Moreno 已提交
48 49 50 51 52 53

function renderBody(body: string): string {
	return `<!DOCTYPE html>
		<html>
			<head>
				<meta http-equiv="Content-type" content="text/html;charset=UTF-8">
J
Johannes Rieken 已提交
54
				<link rel="stylesheet" type="text/css" href="${ require.toUrl('./media/markdown.css')}" >
J
Joao Moreno 已提交
55
			</head>
J
Johannes Rieken 已提交
56
			<body>${ body}</body>
J
Joao Moreno 已提交
57 58
		</html>`;
}
J
Joao Moreno 已提交
59

J
Joao Moreno 已提交
60 61 62 63 64
class NavBar {

	private _onChange = new Emitter<string>();
	get onChange(): Event<string> { return this._onChange.event; }

J
Joao Moreno 已提交
65
	private currentId: string = null;
J
Joao Moreno 已提交
66 67 68 69 70 71 72 73 74 75
	private actions: Action[];
	private actionbar: ActionBar;

	constructor(container: HTMLElement) {
		const element = append(container, $('.navbar'));
		this.actions = [];
		this.actionbar = new ActionBar(element, { animated: false });
	}

	push(id: string, label: string): void {
J
Joao Moreno 已提交
76
		const run = () => this._update(id);
J
Joao Moreno 已提交
77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
		const action = new Action(id, label, null, true, run);

		this.actions.push(action);
		this.actionbar.push(action);

		if (this.actions.length === 1) {
			run();
		}
	}

	clear(): void {
		this.actions = dispose(this.actions);
		this.actionbar.clear();
	}

J
Joao Moreno 已提交
92 93 94 95 96 97 98 99 100 101 102
	update(): void {
		this._update(this.currentId);
	}

	_update(id: string = this.currentId): TPromise<void> {
		this.currentId = id;
		this._onChange.fire(id);
		this.actions.forEach(a => a.enabled = a.id !== id);
		return TPromise.as(null);
	}

J
Joao Moreno 已提交
103 104 105 106 107 108 109
	dispose(): void {
		this.actionbar = dispose(this.actionbar);
	}
}

const NavbarSection = {
	Readme: 'readme',
X
XVincentX 已提交
110
	Contributions: 'contributions',
S
Sandeep Somavarapu 已提交
111 112
	Changelog: 'changelog',
	Dependencies: 'dependencies'
J
Joao Moreno 已提交
113 114
};

J
Joao Moreno 已提交
115 116 117 118
interface ILayoutParticipant {
	layout(): void;
}

J
Joao Moreno 已提交
119 120 121 122
export class ExtensionEditor extends BaseEditor {

	static ID: string = 'workbench.editor.extension';

123
	private icon: HTMLImageElement;
J
Joao Moreno 已提交
124
	private name: HTMLElement;
S
Sandeep Somavarapu 已提交
125
	private identifier: HTMLElement;
J
Joao Moreno 已提交
126 127
	private license: HTMLElement;
	private publisher: HTMLElement;
J
Joao Moreno 已提交
128
	private installCount: HTMLElement;
J
Joao Moreno 已提交
129
	private rating: HTMLElement;
J
Joao Moreno 已提交
130
	private description: HTMLElement;
J
Joao Moreno 已提交
131 132 133
	private extensionActionBar: ActionBar;
	private navbar: NavBar;
	private content: HTMLElement;
J
Joao Moreno 已提交
134 135 136 137

	private _highlight: ITemplateData;
	private highlightDisposable: IDisposable;

J
Joao Moreno 已提交
138
	private extensionReadme: Cache<string>;
X
XVincentX 已提交
139
	private extensionChangelog: Cache<string>;
J
Joao Moreno 已提交
140
	private extensionManifest: Cache<IExtensionManifest>;
S
Sandeep Somavarapu 已提交
141
	private extensionDependencies: Cache<IExtensionDependencies>;
J
Joao Moreno 已提交
142

J
Joao Moreno 已提交
143
	private layoutParticipants: ILayoutParticipant[] = [];
J
Joao Moreno 已提交
144 145
	private contentDisposables: IDisposable[] = [];
	private transientDisposables: IDisposable[] = [];
J
Joao Moreno 已提交
146
	private disposables: IDisposable[];
J
Joao Moreno 已提交
147 148 149

	constructor(
		@ITelemetryService telemetryService: ITelemetryService,
150 151
		@IExtensionGalleryService private galleryService: IExtensionGalleryService,
		@IConfigurationService private configurationService: IConfigurationService,
J
Joao Moreno 已提交
152
		@IInstantiationService private instantiationService: IInstantiationService,
153
		@IViewletService private viewletService: IViewletService,
J
Joao Moreno 已提交
154
		@IExtensionsWorkbenchService private extensionsWorkbenchService: IExtensionsWorkbenchService,
155
		@IThemeService private themeService: IThemeService,
156
		@IKeybindingService private keybindingService: IKeybindingService,
157
		@IMessageService private messageService: IMessageService,
158 159
		@IOpenerService private openerService: IOpenerService,
		@IListService private listService: IListService
J
Joao Moreno 已提交
160 161 162 163
	) {
		super(ExtensionEditor.ID, telemetryService);
		this._highlight = null;
		this.highlightDisposable = empty;
J
Joao Moreno 已提交
164
		this.disposables = [];
J
Joao Moreno 已提交
165
		this.extensionReadme = null;
X
XVincentX 已提交
166
		this.extensionChangelog = null;
J
Joao Moreno 已提交
167
		this.extensionManifest = null;
S
Sandeep Somavarapu 已提交
168
		this.extensionDependencies = null;
J
Joao Moreno 已提交
169 170 171 172
	}

	createEditor(parent: Builder): void {
		const container = parent.getHTMLElement();
J
Joao Moreno 已提交
173 174 175

		const root = append(container, $('.extension-editor'));
		const header = append(root, $('.header'));
J
Joao Moreno 已提交
176

J
Joao Moreno 已提交
177
		this.icon = append(header, $<HTMLImageElement>('img.icon', { draggable: false }));
J
Joao Moreno 已提交
178

J
Joao Moreno 已提交
179
		const details = append(header, $('.details'));
J
Joao Moreno 已提交
180
		const title = append(details, $('.title'));
J
Joao Moreno 已提交
181
		this.name = append(title, $('span.name.clickable', { title: localize('name', "Extension name") }));
J
Joao Moreno 已提交
182
		this.identifier = append(title, $('span.identifier', { title: localize('extension id', "Extension identifier") }));
J
Joao Moreno 已提交
183 184

		const subtitle = append(details, $('.subtitle'));
J
Joao Moreno 已提交
185
		this.publisher = append(subtitle, $('span.publisher.clickable', { title: localize('publisher', "Publisher name") }));
J
Joao Moreno 已提交
186

J
Joao Moreno 已提交
187
		this.installCount = append(subtitle, $('span.install', { title: localize('install count', "Install count") }));
J
Joao Moreno 已提交
188

J
Joao Moreno 已提交
189
		this.rating = append(subtitle, $('span.rating.clickable', { title: localize('rating', "Rating") }));
J
Joao Moreno 已提交
190

J
Joao Moreno 已提交
191
		this.license = append(subtitle, $('span.license.clickable'));
J
Joao Moreno 已提交
192
		this.license.textContent = localize('license', 'License');
193
		this.license.style.display = 'none';
J
Joao Moreno 已提交
194

J
Joao Moreno 已提交
195 196
		this.description = append(details, $('.description'));

J
Joao Moreno 已提交
197
		const extensionActions = append(details, $('.actions'));
198 199 200 201 202 203 204 205 206 207 208 209
		this.extensionActionBar = new ActionBar(extensionActions, {
			animated: false,
			actionItemProvider: (action: Action) => {
				if (action.id === EnableAction.ID) {
					return (<EnableAction>action).actionItem;
				}
				if (action.id === DisableAction.ID) {
					return (<DisableAction>action).actionItem;
				}
				return null;
			}
		});
J
Joao Moreno 已提交
210 211
		this.disposables.push(this.extensionActionBar);

J
Joao Moreno 已提交
212 213 214 215
		chain(fromEventEmitter<{ error?: any; }>(this.extensionActionBar, 'run'))
			.map(({ error }) => error)
			.filter(error => !!error)
			.on(this.onError, this, this.disposables);
216

J
Joao Moreno 已提交
217 218
		const body = append(root, $('.body'));
		this.navbar = new NavBar(body);
J
Joao Moreno 已提交
219

J
Joao Moreno 已提交
220
		this.content = append(body, $('.content'));
J
Joao Moreno 已提交
221 222 223
	}

	setInput(input: ExtensionsInput, options: EditorOptions): TPromise<void> {
J
Joao Moreno 已提交
224 225
		const extension = input.extension;

J
Joao Moreno 已提交
226 227
		this.transientDisposables = dispose(this.transientDisposables);

228
		this.telemetryService.publicLog('extensionGallery:openExtension', extension.telemetryData);
J
Joao Moreno 已提交
229

J
Joao Moreno 已提交
230
		this.extensionReadme = new Cache(() => extension.getReadme());
X
XVincentX 已提交
231
		this.extensionChangelog = new Cache(() => extension.getChangelog());
J
Joao Moreno 已提交
232
		this.extensionManifest = new Cache(() => extension.getManifest());
S
Sandeep Somavarapu 已提交
233
		this.extensionDependencies = new Cache(() => this.extensionsWorkbenchService.loadDependencies(extension));
J
Joao Moreno 已提交
234

235 236 237 238
		const onError = once(domEvent(this.icon, 'error'));
		onError(() => this.icon.src = extension.iconUrlFallback, null, this.transientDisposables);
		this.icon.src = extension.iconUrl;

J
Joao Moreno 已提交
239
		this.name.textContent = extension.displayName;
240
		this.identifier.textContent = extension.identifier;
241

J
Joao Moreno 已提交
242 243
		this.publisher.textContent = extension.publisherDisplayName;
		this.description.textContent = extension.description;
J
Joao Moreno 已提交
244

245 246 247
		if (extension.url) {
			this.name.onclick = finalHandler(() => window.open(extension.url));
			this.rating.onclick = finalHandler(() => window.open(`${extension.url}#review-details`));
J
Joao Moreno 已提交
248
			this.publisher.onclick = finalHandler(() => {
249
				this.viewletService.openViewlet(VIEWLET_ID, true)
J
Joao Moreno 已提交
250
					.then(viewlet => viewlet as IExtensionsViewlet)
J
Johannes Rieken 已提交
251
					.done(viewlet => viewlet.search(`publisher:"${extension.publisherDisplayName}"`));
J
Joao Moreno 已提交
252
			});
253 254

			if (extension.licenseUrl) {
B
Benjamin Pasero 已提交
255
				this.license.onclick = finalHandler(() => window.open(extension.licenseUrl));
256 257 258 259 260
				this.license.style.display = 'initial';
			} else {
				this.license.onclick = null;
				this.license.style.display = 'none';
			}
J
Joao Moreno 已提交
261 262
		}

J
Joao Moreno 已提交
263
		const install = this.instantiationService.createInstance(InstallWidget, this.installCount, { extension });
J
Joao Moreno 已提交
264 265
		this.transientDisposables.push(install);

J
Joao Moreno 已提交
266
		const ratings = this.instantiationService.createInstance(RatingsWidget, this.rating, { extension });
J
Joao Moreno 已提交
267
		this.transientDisposables.push(ratings);
J
Joao Moreno 已提交
268

J
Joao Moreno 已提交
269
		const builtinStatusAction = this.instantiationService.createInstance(BuiltinStatusLabelAction);
J
Joao Moreno 已提交
270 271 272
		const installAction = this.instantiationService.createInstance(CombinedInstallAction);
		const updateAction = this.instantiationService.createInstance(UpdateAction);
		const enableAction = this.instantiationService.createInstance(EnableAction);
273
		const disableAction = this.instantiationService.createInstance(DisableAction);
274
		const reloadAction = this.instantiationService.createInstance(ReloadAction);
J
Joao Moreno 已提交
275 276

		installAction.extension = extension;
J
Joao Moreno 已提交
277
		builtinStatusAction.extension = extension;
J
Joao Moreno 已提交
278 279
		updateAction.extension = extension;
		enableAction.extension = extension;
280
		disableAction.extension = extension;
281
		reloadAction.extension = extension;
J
Joao Moreno 已提交
282

J
Joao Moreno 已提交
283
		this.extensionActionBar.clear();
284
		this.extensionActionBar.push([reloadAction, updateAction, enableAction, disableAction, installAction, builtinStatusAction], { icon: true, label: true });
285
		this.transientDisposables.push(enableAction, updateAction, reloadAction, disableAction, installAction, builtinStatusAction);
J
Joao Moreno 已提交
286

J
Joao Moreno 已提交
287 288
		this.navbar.clear();
		this.navbar.onChange(this.onNavbarChange.bind(this, extension), this, this.transientDisposables);
289 290
		this.navbar.push(NavbarSection.Readme, localize('details', "Details"));
		this.navbar.push(NavbarSection.Contributions, localize('contributions', "Contributions"));
291 292
		this.navbar.push(NavbarSection.Changelog, localize('changelog', "Changelog"));
		this.navbar.push(NavbarSection.Dependencies, localize('dependencies', "Dependencies"));
J
Joao Moreno 已提交
293 294 295 296 297 298

		this.content.innerHTML = '';

		return super.setInput(input, options);
	}

J
Joao Moreno 已提交
299 300 301 302 303
	changePosition(position: Position): void {
		this.navbar.update();
		super.changePosition(position);
	}

J
Joao Moreno 已提交
304
	private onNavbarChange(extension: IExtension, id: string): void {
S
Sandeep Somavarapu 已提交
305 306
		this.contentDisposables = dispose(this.contentDisposables);
		this.content.innerHTML = '';
J
Joao Moreno 已提交
307
		switch (id) {
308 309 310
			case NavbarSection.Readme: return this.openReadme();
			case NavbarSection.Contributions: return this.openContributions();
			case NavbarSection.Changelog: return this.openChangelog();
311
			case NavbarSection.Dependencies: return this.openDependencies(extension);
J
Joao Moreno 已提交
312
		}
J
Joao Moreno 已提交
313
	}
J
Joao Moreno 已提交
314

315
	private openMarkdown(content: TPromise<string>, noContentCopy: string) {
316
		return this.loadContents(() => content
J
Joao Moreno 已提交
317
			.then(marked.parse)
J
Joao Moreno 已提交
318
			.then(renderBody)
J
Joao Moreno 已提交
319 320 321
			.then<void>(body => {
				const webview = new WebView(
					this.content,
M
Matt Bierner 已提交
322
					document.querySelector('.monaco-editor-background')
J
Joao Moreno 已提交
323 324 325
				);

				webview.style(this.themeService.getColorTheme());
J
Joao Moreno 已提交
326
				webview.contents = [body];
J
Joao Moreno 已提交
327

328
				webview.onDidClickLink(link => this.openerService.open(link), null, this.contentDisposables);
M
Martin Aeschlimann 已提交
329
				this.themeService.onDidColorThemeChange(theme => webview.style(theme), null, this.contentDisposables);
330
				this.contentDisposables.push(webview);
J
Joao Moreno 已提交
331
			})
J
Joao Moreno 已提交
332
			.then(null, () => {
333
				const p = append(this.content, $('p.nocontent'));
X
XVincentX 已提交
334
				p.textContent = noContentCopy;
J
Joao Moreno 已提交
335
			}));
J
Joao Moreno 已提交
336
	}
J
Joao Moreno 已提交
337

338 339
	private openReadme() {
		return this.openMarkdown(this.extensionReadme.get(), localize('noReadme', "No README available."));
340 341
	}

342
	private openChangelog() {
343
		return this.openMarkdown(this.extensionChangelog.get(), localize('noChangelog', "No Changelog available."));
344 345
	}

346
	private openContributions() {
J
Joao Moreno 已提交
347
		return this.loadContents(() => this.extensionManifest.get()
348
			.then(manifest => {
J
Joao Moreno 已提交
349 350 351 352 353 354 355
				const content = $('div', { class: 'subcontent' });
				const scrollableContent = new DomScrollableElement(content, { canUseTranslate3d: false });

				const layout = () => scrollableContent.scanDomNode();
				const removeLayoutParticipant = arrays.insert(this.layoutParticipants, { layout });
				this.contentDisposables.push(toDisposable(removeLayoutParticipant));

J
Joao Moreno 已提交
356
				const renders = [
B
Benjamin Pasero 已提交
357
					this.renderSettings(content, manifest, layout),
J
Joao Moreno 已提交
358
					this.renderCommands(content, manifest, layout),
B
Benjamin Pasero 已提交
359 360 361 362
					this.renderLanguages(content, manifest, layout),
					this.renderThemes(content, manifest, layout),
					this.renderJSONValidation(content, manifest, layout),
					this.renderDebuggers(content, manifest, layout)
J
Joao Moreno 已提交
363 364 365
				];

				const isEmpty = !renders.reduce((v, r) => r || v, false);
J
Joao Moreno 已提交
366
				scrollableContent.scanDomNode();
367 368 369 370 371 372 373 374

				if (isEmpty) {
					append(this.content, $('p.nocontent')).textContent = localize('noContributions', "No Contributions");
					return;
				} else {
					append(this.content, scrollableContent.getDomNode());
					this.contentDisposables.push(scrollableContent);
				}
375
			}));
J
Joao Moreno 已提交
376 377
	}

378 379 380 381 382
	private openDependencies(extension: IExtension) {
		if (extension.dependencies.length === 0) {
			append(this.content, $('p.nocontent')).textContent = localize('noDependencies', "No Dependencies");
			return;
		}
S
Sandeep Somavarapu 已提交
383 384 385 386 387 388 389 390 391
		addClass(this.content, 'loading');
		this.extensionDependencies.get().then(extensionDependencies => {
			removeClass(this.content, 'loading');

			const content = $('div', { class: 'subcontent' });
			const scrollableContent = new DomScrollableElement(content, { canUseTranslate3d: false });
			append(this.content, scrollableContent.getDomNode());
			this.contentDisposables.push(scrollableContent);

392
			const tree = this.renderDependencies(content, extensionDependencies);
393 394
			const layout = () => {
				scrollableContent.scanDomNode();
A
Alex Dima 已提交
395 396
				const scrollState = scrollableContent.getScrollState();
				tree.layout(scrollState.height);
397
			};
S
Sandeep Somavarapu 已提交
398 399 400
			const removeLayoutParticipant = arrays.insert(this.layoutParticipants, { layout });
			this.contentDisposables.push(toDisposable(removeLayoutParticipant));

401
			this.contentDisposables.push(tree);
S
Sandeep Somavarapu 已提交
402
			scrollableContent.scanDomNode();
S
Sandeep Somavarapu 已提交
403 404 405 406 407
		}, error => {
			removeClass(this.content, 'loading');
			append(this.content, $('p.nocontent')).textContent = error;
			this.messageService.show(Severity.Error, error);
			return;
S
Sandeep Somavarapu 已提交
408 409 410
		});
	}

411 412 413
	private renderDependencies(container: HTMLElement, extensionDependencies: IExtensionDependencies): Tree {
		const renderer = this.instantiationService.createInstance(Renderer);
		const controller = this.instantiationService.createInstance(Controller);
S
Sandeep Somavarapu 已提交
414 415
		const tree = new Tree(container, {
			dataSource: new DataSource(),
416 417
			renderer,
			controller
S
Sandeep Somavarapu 已提交
418 419
		}, {
				indentPixels: 40,
420 421
				twistiePixels: 20,
				keyboardSupport: false
S
Sandeep Somavarapu 已提交
422 423
			});
		tree.setInput(extensionDependencies);
424 425 426 427 428 429 430 431 432

		this.contentDisposables.push(tree.addListener2('selection', event => {
			if (event && event.payload && event.payload.origin === 'keyboard') {
				controller.openExtension(tree, false);
			}
		}));

		this.contentDisposables.push(this.listService.register(tree));

S
Sandeep Somavarapu 已提交
433 434 435
		return tree;
	}

B
Benjamin Pasero 已提交
436
	private renderSettings(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
437 438
		const contributes = manifest.contributes;
		const configuration = contributes && contributes.configuration;
J
Joao Moreno 已提交
439
		const properties = configuration && configuration.properties;
J
Joao Moreno 已提交
440
		const contrib = properties ? Object.keys(properties) : [];
J
Joao Moreno 已提交
441

J
Joao Moreno 已提交
442
		if (!contrib.length) {
443
			return false;
J
Joao Moreno 已提交
444 445
		}

J
Joao Moreno 已提交
446
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
447
			$('summary', null, localize('settings', "Settings ({0})", contrib.length)),
J
Joao Moreno 已提交
448
			$('table', null,
J
Joao Moreno 已提交
449 450 451
				$('tr', null,
					$('th', null, localize('setting name', "Name")),
					$('th', null, localize('description', "Description")),
J
nls  
Joao Moreno 已提交
452
					$('th', null, localize('default', "Default"))
J
Joao Moreno 已提交
453 454 455 456 457 458
				),
				...contrib.map(key => $('tr', null,
					$('td', null, $('code', null, key)),
					$('td', null, properties[key].description),
					$('td', null, $('code', null, properties[key].default))
				))
J
Joao Moreno 已提交
459
			)
J
Joao Moreno 已提交
460 461 462
		);

		append(container, details);
463
		return true;
J
Joao Moreno 已提交
464 465
	}

B
Benjamin Pasero 已提交
466
	private renderDebuggers(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
467 468
		const contributes = manifest.contributes;
		const contrib = contributes && contributes.debuggers || [];
J
Joao Moreno 已提交
469

J
Joao Moreno 已提交
470
		if (!contrib.length) {
471
			return false;
J
Joao Moreno 已提交
472 473
		}

J
Joao Moreno 已提交
474
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
475
			$('summary', null, localize('debuggers', "Debuggers ({0})", contrib.length)),
J
Joao Moreno 已提交
476
			$('table', null,
J
Joao Moreno 已提交
477 478
				$('tr', null, $('th', null, localize('debugger name', "Name"))),
				...contrib.map(d => $('tr', null, $('td', null, d.label || d.type)))
J
Joao Moreno 已提交
479
			)
J
Joao Moreno 已提交
480 481 482
		);

		append(container, details);
483
		return true;
J
Joao Moreno 已提交
484 485
	}

B
Benjamin Pasero 已提交
486
	private renderThemes(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
487 488
		const contributes = manifest.contributes;
		const contrib = contributes && contributes.themes || [];
J
Joao Moreno 已提交
489

J
Joao Moreno 已提交
490
		if (!contrib.length) {
491
			return false;
J
Joao Moreno 已提交
492 493
		}

J
Joao Moreno 已提交
494
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
495
			$('summary', null, localize('themes', "Themes ({0})", contrib.length)),
J
Joao Moreno 已提交
496 497 498 499
			$('ul', null, ...contrib.map(theme => $('li', null, theme.label)))
		);

		append(container, details);
500
		return true;
J
Joao Moreno 已提交
501 502
	}

B
Benjamin Pasero 已提交
503
	private renderJSONValidation(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
504 505
		const contributes = manifest.contributes;
		const contrib = contributes && contributes.jsonValidation || [];
J
Joao Moreno 已提交
506 507

		if (!contrib.length) {
508
			return false;
J
Joao Moreno 已提交
509 510
		}

J
Joao Moreno 已提交
511
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
512
			$('summary', null, localize('JSON Validation', "JSON Validation ({0})", contrib.length)),
J
Joao Moreno 已提交
513 514 515 516
			$('ul', null, ...contrib.map(v => $('li', null, v.fileMatch)))
		);

		append(container, details);
517
		return true;
J
Joao Moreno 已提交
518 519
	}

520
	private renderCommands(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
521 522
		const contributes = manifest.contributes;
		const rawCommands = contributes && contributes.commands || [];
J
Joao Moreno 已提交
523
		const commands = rawCommands.map(c => ({
524 525
			id: c.command,
			title: c.title,
J
Joao Moreno 已提交
526
			keybindings: [],
527 528 529
			menus: []
		}));

J
Joao Moreno 已提交
530
		const byId = arrays.index(commands, c => c.id);
531

J
Joao Moreno 已提交
532
		const menus = contributes && contributes.menus || {};
J
Joao Moreno 已提交
533

534 535
		Object.keys(menus).forEach(context => {
			menus[context].forEach(menu => {
J
Joao Moreno 已提交
536
				let command = byId[menu.command];
537 538

				if (!command) {
J
Joao Moreno 已提交
539
					command = { id: menu.command, title: '', keybindings: [], menus: [context] };
J
Joao Moreno 已提交
540
					byId[command.id] = command;
541 542 543 544 545 546 547
					commands.push(command);
				} else {
					command.menus.push(context);
				}
			});
		});

J
Joao Moreno 已提交
548
		const rawKeybindings = contributes && contributes.keybindings || [];
J
Joao Moreno 已提交
549

550 551
		rawKeybindings.forEach(rawKeybinding => {
			const keyLabel = this.keybindingToLabel(rawKeybinding);
J
Joao Moreno 已提交
552 553 554 555 556

			if (!keyLabel) {
				return;
			}

J
Joao Moreno 已提交
557
			let command = byId[rawKeybinding.command];
J
Joao Moreno 已提交
558 559

			if (!command) {
560
				command = { id: rawKeybinding.command, title: '', keybindings: [keyLabel], menus: [] };
J
Joao Moreno 已提交
561
				byId[command.id] = command;
J
Joao Moreno 已提交
562 563
				commands.push(command);
			} else {
564
				command.keybindings.push(keyLabel);
J
Joao Moreno 已提交
565 566 567
			}
		});

568
		if (!commands.length) {
569
			return false;
570 571
		}

J
Joao Moreno 已提交
572
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
573 574 575 576 577
			$('summary', null, localize('commands', "Commands ({0})", commands.length)),
			$('table', null,
				$('tr', null,
					$('th', null, localize('command name', "Name")),
					$('th', null, localize('description', "Description")),
J
Joao Moreno 已提交
578
					$('th', null, localize('keyboard shortcuts', "Keyboard Shortcuts")),
579 580 581
					$('th', null, localize('menuContexts', "Menu Contexts"))
				),
				...commands.map(c => $('tr', null,
582
					$('td', null, $('code', null, c.id)),
583
					$('td', null, c.title),
584
					$('td', null, ...join(c.keybindings.map(keybinding => $('code', null, keybinding)), ' ')),
585 586 587
					$('td', null, ...c.menus.map(context => $('code', null, context)))
				))
			)
J
Joao Moreno 已提交
588 589 590
		);

		append(container, details);
591
		return true;
592 593
	}

B
Benjamin Pasero 已提交
594
	private renderLanguages(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
595 596
		const contributes = manifest.contributes;
		const rawLanguages = contributes && contributes.languages || [];
J
Joao Moreno 已提交
597 598
		const languages = rawLanguages.map(l => ({
			id: l.id,
J
Joao Moreno 已提交
599
			name: (l.aliases || [])[0] || l.id,
J
Joao Moreno 已提交
600 601 602
			extensions: l.extensions || [],
			hasGrammar: false,
			hasSnippets: false
J
Joao Moreno 已提交
603 604
		}));

J
Joao Moreno 已提交
605 606
		const byId = arrays.index(languages, l => l.id);

J
Joao Moreno 已提交
607
		const grammars = contributes && contributes.grammars || [];
J
Joao Moreno 已提交
608 609 610 611 612

		grammars.forEach(grammar => {
			let language = byId[grammar.language];

			if (!language) {
J
Joao Moreno 已提交
613
				language = { id: grammar.language, name: grammar.language, extensions: [], hasGrammar: true, hasSnippets: false };
J
Joao Moreno 已提交
614 615 616 617 618 619 620
				byId[language.id] = language;
				languages.push(language);
			} else {
				language.hasGrammar = true;
			}
		});

J
Joao Moreno 已提交
621 622 623 624 625 626 627 628 629 630 631 632 633 634
		const snippets = contributes && contributes.snippets || [];

		snippets.forEach(snippet => {
			let language = byId[snippet.language];

			if (!language) {
				language = { id: snippet.language, name: snippet.language, extensions: [], hasGrammar: false, hasSnippets: true };
				byId[language.id] = language;
				languages.push(language);
			} else {
				language.hasSnippets = true;
			}
		});

J
Joao Moreno 已提交
635
		if (!languages.length) {
636
			return false;
J
Joao Moreno 已提交
637 638
		}

J
Joao Moreno 已提交
639
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
640 641 642
			$('summary', null, localize('languages', "Languages ({0})", languages.length)),
			$('table', null,
				$('tr', null,
J
Joao Moreno 已提交
643 644
					$('th', null, localize('language id', "ID")),
					$('th', null, localize('language name', "Name")),
J
Joao Moreno 已提交
645
					$('th', null, localize('file extensions', "File Extensions")),
J
Joao Moreno 已提交
646 647
					$('th', null, localize('grammar', "Grammar")),
					$('th', null, localize('snippets', "Snippets"))
J
Joao Moreno 已提交
648 649
				),
				...languages.map(l => $('tr', null,
J
Joao Moreno 已提交
650
					$('td', null, l.id),
J
Joao Moreno 已提交
651
					$('td', null, l.name),
J
Joao Moreno 已提交
652
					$('td', null, ...join(l.extensions.map(ext => $('code', null, ext)), ' ')),
J
Joao Moreno 已提交
653 654
					$('td', null, document.createTextNode(l.hasGrammar ? '✔︎' : '')),
					$('td', null, document.createTextNode(l.hasSnippets ? '✔︎' : ''))
J
Joao Moreno 已提交
655 656
				))
			)
J
Joao Moreno 已提交
657 658 659
		);

		append(container, details);
660
		return true;
J
Joao Moreno 已提交
661 662
	}

663 664 665
	private keybindingToLabel(rawKeyBinding: IKeyBinding): string {
		let key: string;

J
Johannes Rieken 已提交
666
		switch (process.platform) {
667 668 669 670 671
			case 'win32': key = rawKeyBinding.win; break;
			case 'linux': key = rawKeyBinding.linux; break;
			case 'darwin': key = rawKeyBinding.mac; break;
		}

672
		const keyBinding = new Keybinding(KeybindingLabels.fromUserSettingsLabel(key || rawKeyBinding.key));
J
Joao Moreno 已提交
673 674
		const result = this.keybindingService.getLabelFor(keyBinding);
		return result === 'unknown' ? null : result;
675 676
	}

J
Johannes Rieken 已提交
677
	private loadContents(loadingTask: () => TPromise<any>): void {
J
Joao Moreno 已提交
678
		addClass(this.content, 'loading');
J
Joao Moreno 已提交
679

J
Joao Moreno 已提交
680 681
		let promise = loadingTask();
		promise = always(promise, () => removeClass(this.content, 'loading'));
J
Joao Moreno 已提交
682

J
Joao Moreno 已提交
683
		this.contentDisposables.push(toDisposable(() => promise.cancel()));
J
Joao Moreno 已提交
684 685 686
	}

	layout(): void {
J
Joao Moreno 已提交
687
		this.layoutParticipants.forEach(p => p.layout());
J
Joao Moreno 已提交
688 689
	}

690 691 692 693 694 695 696 697
	private onError(err: any): void {
		if (isPromiseCanceledError(err)) {
			return;
		}

		this.messageService.show(Severity.Error, err);
	}

J
Joao Moreno 已提交
698 699
	dispose(): void {
		this._highlight = null;
J
Joao Moreno 已提交
700 701
		this.transientDisposables = dispose(this.transientDisposables);
		this.disposables = dispose(this.disposables);
J
Joao Moreno 已提交
702 703 704
		super.dispose();
	}
}