extensionEditor.ts 24.8 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';
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

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

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

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

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

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

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

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

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

		this.content.innerHTML = '';

		return super.setInput(input, options);
	}

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

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

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

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

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

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

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

345
	private openContributions() {
J
Joao Moreno 已提交
346
		return this.loadContents(() => this.extensionManifest.get()
347
			.then(manifest => {
J
Joao Moreno 已提交
348 349 350 351 352 353 354
				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 已提交
355 356 357 358 359 360 361 362 363 364
				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 已提交
365
				scrollableContent.scanDomNode();
366 367 368 369 370 371 372 373

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

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

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

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

410
	private static renderDependencies(container: HTMLElement, extensionDependencies: IExtensionDependencies, instantiationService: IInstantiationService): Tree {
411
		const renderer = instantiationService.createInstance(Renderer);
412
		const controller = instantiationService.createInstance(Controller);
S
Sandeep Somavarapu 已提交
413 414
		const tree = new Tree(container, {
			dataSource: new DataSource(),
415 416
			renderer,
			controller
S
Sandeep Somavarapu 已提交
417 418 419 420 421 422 423 424
		}, {
				indentPixels: 40,
				twistiePixels: 20
			});
		tree.setInput(extensionDependencies);
		return tree;
	}

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

			if (!keyLabel) {
				return;
			}

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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