extensionEditor.ts 24.9 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 product from 'vs/platform/product';
J
Joao Moreno 已提交
37
import { ActionBar } from 'vs/base/browser/ui/actionbar/actionbar';
38
import { CombinedInstallAction, UpdateAction, EnableAction, DisableAction, BuiltinStatusLabelAction, ReloadAction } from 'vs/workbench/parts/extensions/browser/extensionsActions';
J
Joao Moreno 已提交
39
import WebView from 'vs/workbench/parts/html/browser/webview';
A
Alexandru Dima 已提交
40
import { Keybinding } 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';
J
Joao Moreno 已提交
47 48 49 50 51 52

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

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

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

J
Joao Moreno 已提交
64
	private currentId: string = null;
J
Joao Moreno 已提交
65 66 67 68 69 70 71 72 73 74
	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 已提交
75
		const run = () => this._update(id);
J
Joao Moreno 已提交
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90
		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 已提交
91 92 93 94 95 96 97 98 99 100 101
	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 已提交
102 103 104 105 106 107 108
	dispose(): void {
		this.actionbar = dispose(this.actionbar);
	}
}

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

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

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

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

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

	private _highlight: ITemplateData;
	private highlightDisposable: IDisposable;

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

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

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

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

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

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

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

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

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

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

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

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

J
Joao Moreno 已提交
195
		const extensionActions = append(details, $('.actions'));
196 197 198 199 200 201 202 203 204 205 206 207
		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 已提交
208 209
		this.disposables.push(this.extensionActionBar);

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

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

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

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

J
Joao Moreno 已提交
224 225
		this.transientDisposables = dispose(this.transientDisposables);

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

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

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

J
Joao Moreno 已提交
237
		this.name.textContent = extension.displayName;
S
Sandeep Somavarapu 已提交
238
		this.identifier.textContent = `${extension.publisher}.${extension.name}`;
239

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

J
Joao Moreno 已提交
243
		if (product.extensionsGallery) {
J
Johannes Rieken 已提交
244
			const extensionUrl = `${product.extensionsGallery.itemUrl}?itemName=${extension.publisher}.${extension.name}`;
J
Joao Moreno 已提交
245

B
Benjamin Pasero 已提交
246 247
			this.name.onclick = finalHandler(() => window.open(extensionUrl));
			this.rating.onclick = finalHandler(() => window.open(`${extensionUrl}#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 323
					document.querySelector('.monaco-editor-background'),
					{ nodeintegration: false }
J
Joao Moreno 已提交
324 325 326
				);

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

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

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

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

347
	private openContributions() {
J
Joao Moreno 已提交
348
		return this.loadContents(() => this.extensionManifest.get()
349
			.then(manifest => {
J
Joao Moreno 已提交
350 351 352 353 354 355 356
				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 已提交
357 358 359 360 361 362 363 364 365 366
				const renders = [
					ExtensionEditor.renderSettings(content, manifest, layout),
					this.renderCommands(content, manifest, layout),
					ExtensionEditor.renderLanguages(content, manifest, layout),
					ExtensionEditor.renderThemes(content, manifest, layout),
					ExtensionEditor.renderJSONValidation(content, manifest, layout),
					ExtensionEditor.renderDebuggers(content, manifest, layout)
				];

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

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

379 380 381 382 383
	private openDependencies(extension: IExtension) {
		if (extension.dependencies.length === 0) {
			append(this.content, $('p.nocontent')).textContent = localize('noDependencies', "No Dependencies");
			return;
		}
S
Sandeep Somavarapu 已提交
384 385 386 387 388 389 390 391 392
		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);

393
			const tree = ExtensionEditor.renderDependencies(content, extensionDependencies, this.instantiationService);
394 395 396 397
			const layout = () => {
				scrollableContent.scanDomNode();
				tree.layout(scrollableContent.getHeight());
			};
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
	private static renderDependencies(container: HTMLElement, extensionDependencies: IExtensionDependencies, instantiationService: IInstantiationService): Tree {
412
		const renderer = instantiationService.createInstance(Renderer);
413
		const controller = 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 420 421 422 423 424 425
		}, {
				indentPixels: 40,
				twistiePixels: 20
			});
		tree.setInput(extensionDependencies);
		return tree;
	}

426
	private static renderSettings(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
427 428
		const contributes = manifest.contributes;
		const configuration = contributes && contributes.configuration;
J
Joao Moreno 已提交
429
		const properties = configuration && configuration.properties;
J
Joao Moreno 已提交
430
		const contrib = properties ? Object.keys(properties) : [];
J
Joao Moreno 已提交
431

J
Joao Moreno 已提交
432
		if (!contrib.length) {
433
			return false;
J
Joao Moreno 已提交
434 435
		}

J
Joao Moreno 已提交
436
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
437
			$('summary', null, localize('settings', "Settings ({0})", contrib.length)),
J
Joao Moreno 已提交
438
			$('table', null,
J
Joao Moreno 已提交
439 440 441
				$('tr', null,
					$('th', null, localize('setting name', "Name")),
					$('th', null, localize('description', "Description")),
J
nls  
Joao Moreno 已提交
442
					$('th', null, localize('default', "Default"))
J
Joao Moreno 已提交
443 444 445 446 447 448
				),
				...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 已提交
449
			)
J
Joao Moreno 已提交
450 451 452
		);

		append(container, details);
453
		return true;
J
Joao Moreno 已提交
454 455
	}

456
	private static renderDebuggers(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
457 458
		const contributes = manifest.contributes;
		const contrib = contributes && contributes.debuggers || [];
J
Joao Moreno 已提交
459

J
Joao Moreno 已提交
460
		if (!contrib.length) {
461
			return false;
J
Joao Moreno 已提交
462 463
		}

J
Joao Moreno 已提交
464
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
465
			$('summary', null, localize('debuggers', "Debuggers ({0})", contrib.length)),
J
Joao Moreno 已提交
466
			$('table', null,
J
Joao Moreno 已提交
467 468
				$('tr', null, $('th', null, localize('debugger name', "Name"))),
				...contrib.map(d => $('tr', null, $('td', null, d.label || d.type)))
J
Joao Moreno 已提交
469
			)
J
Joao Moreno 已提交
470 471 472
		);

		append(container, details);
473
		return true;
J
Joao Moreno 已提交
474 475
	}

476
	private static renderThemes(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
477 478
		const contributes = manifest.contributes;
		const contrib = contributes && contributes.themes || [];
J
Joao Moreno 已提交
479

J
Joao Moreno 已提交
480
		if (!contrib.length) {
481
			return false;
J
Joao Moreno 已提交
482 483
		}

J
Joao Moreno 已提交
484
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
485
			$('summary', null, localize('themes', "Themes ({0})", contrib.length)),
J
Joao Moreno 已提交
486 487 488 489
			$('ul', null, ...contrib.map(theme => $('li', null, theme.label)))
		);

		append(container, details);
490
		return true;
J
Joao Moreno 已提交
491 492
	}

493
	private static renderJSONValidation(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
494 495
		const contributes = manifest.contributes;
		const contrib = contributes && contributes.jsonValidation || [];
J
Joao Moreno 已提交
496 497

		if (!contrib.length) {
498
			return false;
J
Joao Moreno 已提交
499 500
		}

J
Joao Moreno 已提交
501
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
502
			$('summary', null, localize('JSON Validation', "JSON Validation ({0})", contrib.length)),
J
Joao Moreno 已提交
503 504 505 506
			$('ul', null, ...contrib.map(v => $('li', null, v.fileMatch)))
		);

		append(container, details);
507
		return true;
J
Joao Moreno 已提交
508 509
	}

510
	private renderCommands(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
511 512
		const contributes = manifest.contributes;
		const rawCommands = contributes && contributes.commands || [];
J
Joao Moreno 已提交
513
		const commands = rawCommands.map(c => ({
514 515
			id: c.command,
			title: c.title,
J
Joao Moreno 已提交
516
			keybindings: [],
517 518 519
			menus: []
		}));

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

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

524 525
		Object.keys(menus).forEach(context => {
			menus[context].forEach(menu => {
J
Joao Moreno 已提交
526
				let command = byId[menu.command];
527 528

				if (!command) {
J
Joao Moreno 已提交
529
					command = { id: menu.command, title: '', keybindings: [], menus: [context] };
J
Joao Moreno 已提交
530
					byId[command.id] = command;
531 532 533 534 535 536 537
					commands.push(command);
				} else {
					command.menus.push(context);
				}
			});
		});

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

540 541
		rawKeybindings.forEach(rawKeybinding => {
			const keyLabel = this.keybindingToLabel(rawKeybinding);
J
Joao Moreno 已提交
542 543 544 545 546

			if (!keyLabel) {
				return;
			}

J
Joao Moreno 已提交
547
			let command = byId[rawKeybinding.command];
J
Joao Moreno 已提交
548 549

			if (!command) {
550
				command = { id: rawKeybinding.command, title: '', keybindings: [keyLabel], menus: [] };
J
Joao Moreno 已提交
551
				byId[command.id] = command;
J
Joao Moreno 已提交
552 553
				commands.push(command);
			} else {
554
				command.keybindings.push(keyLabel);
J
Joao Moreno 已提交
555 556 557
			}
		});

558
		if (!commands.length) {
559
			return false;
560 561
		}

J
Joao Moreno 已提交
562
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
563 564 565 566 567
			$('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 已提交
568
					$('th', null, localize('keyboard shortcuts', "Keyboard Shortcuts")),
569 570 571
					$('th', null, localize('menuContexts', "Menu Contexts"))
				),
				...commands.map(c => $('tr', null,
572
					$('td', null, $('code', null, c.id)),
573
					$('td', null, c.title),
574
					$('td', null, ...join(c.keybindings.map(keybinding => $('code', null, keybinding)), ' ')),
575 576 577
					$('td', null, ...c.menus.map(context => $('code', null, context)))
				))
			)
J
Joao Moreno 已提交
578 579 580
		);

		append(container, details);
581
		return true;
582 583
	}

584
	private static renderLanguages(container: HTMLElement, manifest: IExtensionManifest, onDetailsToggle: Function): boolean {
J
Joao Moreno 已提交
585 586
		const contributes = manifest.contributes;
		const rawLanguages = contributes && contributes.languages || [];
J
Joao Moreno 已提交
587 588
		const languages = rawLanguages.map(l => ({
			id: l.id,
J
Joao Moreno 已提交
589
			name: (l.aliases || [])[0] || l.id,
J
Joao Moreno 已提交
590 591 592
			extensions: l.extensions || [],
			hasGrammar: false,
			hasSnippets: false
J
Joao Moreno 已提交
593 594
		}));

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

J
Joao Moreno 已提交
597
		const grammars = contributes && contributes.grammars || [];
J
Joao Moreno 已提交
598 599 600 601 602

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

			if (!language) {
J
Joao Moreno 已提交
603
				language = { id: grammar.language, name: grammar.language, extensions: [], hasGrammar: true, hasSnippets: false };
J
Joao Moreno 已提交
604 605 606 607 608 609 610
				byId[language.id] = language;
				languages.push(language);
			} else {
				language.hasGrammar = true;
			}
		});

J
Joao Moreno 已提交
611 612 613 614 615 616 617 618 619 620 621 622 623 624
		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 已提交
625
		if (!languages.length) {
626
			return false;
J
Joao Moreno 已提交
627 628
		}

J
Joao Moreno 已提交
629
		const details = $('details', { open: true, ontoggle: onDetailsToggle },
J
Joao Moreno 已提交
630 631 632
			$('summary', null, localize('languages', "Languages ({0})", languages.length)),
			$('table', null,
				$('tr', null,
J
Joao Moreno 已提交
633 634
					$('th', null, localize('language id', "ID")),
					$('th', null, localize('language name', "Name")),
J
Joao Moreno 已提交
635
					$('th', null, localize('file extensions', "File Extensions")),
J
Joao Moreno 已提交
636 637
					$('th', null, localize('grammar', "Grammar")),
					$('th', null, localize('snippets', "Snippets"))
J
Joao Moreno 已提交
638 639
				),
				...languages.map(l => $('tr', null,
J
Joao Moreno 已提交
640
					$('td', null, l.id),
J
Joao Moreno 已提交
641
					$('td', null, l.name),
J
Joao Moreno 已提交
642
					$('td', null, ...join(l.extensions.map(ext => $('code', null, ext)), ' ')),
J
Joao Moreno 已提交
643 644
					$('td', null, document.createTextNode(l.hasGrammar ? '✔︎' : '')),
					$('td', null, document.createTextNode(l.hasSnippets ? '✔︎' : ''))
J
Joao Moreno 已提交
645 646
				))
			)
J
Joao Moreno 已提交
647 648 649
		);

		append(container, details);
650
		return true;
J
Joao Moreno 已提交
651 652
	}

653 654 655
	private keybindingToLabel(rawKeyBinding: IKeyBinding): string {
		let key: string;

J
Johannes Rieken 已提交
656
		switch (process.platform) {
657 658 659 660 661 662
			case 'win32': key = rawKeyBinding.win; break;
			case 'linux': key = rawKeyBinding.linux; break;
			case 'darwin': key = rawKeyBinding.mac; break;
		}

		const keyBinding = new Keybinding(Keybinding.fromUserSettingsLabel(key || rawKeyBinding.key));
J
Joao Moreno 已提交
663 664
		const result = this.keybindingService.getLabelFor(keyBinding);
		return result === 'unknown' ? null : result;
665 666
	}

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

J
Joao Moreno 已提交
670 671
		let promise = loadingTask();
		promise = always(promise, () => removeClass(this.content, 'loading'));
J
Joao Moreno 已提交
672

J
Joao Moreno 已提交
673
		this.contentDisposables.push(toDisposable(() => promise.cancel()));
J
Joao Moreno 已提交
674 675 676
	}

	layout(): void {
J
Joao Moreno 已提交
677
		this.layoutParticipants.forEach(p => p.layout());
J
Joao Moreno 已提交
678 679
	}

680 681 682 683 684 685 686 687
	private onError(err: any): void {
		if (isPromiseCanceledError(err)) {
			return;
		}

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

J
Joao Moreno 已提交
688 689
	dispose(): void {
		this._highlight = null;
J
Joao Moreno 已提交
690 691
		this.transientDisposables = dispose(this.transientDisposables);
		this.disposables = dispose(this.disposables);
J
Joao Moreno 已提交
692 693 694
		super.dispose();
	}
}