layout.ts 19.2 KB
Newer Older
E
Erich Gamma 已提交
1 2 3 4 5 6
/*---------------------------------------------------------------------------------------------
 *  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
Johannes Rieken 已提交
7 8 9 10 11 12 13 14 15 16 17 18 19
import { Dimension, Builder, Box } from 'vs/base/browser/builder';
import { Part } from 'vs/workbench/browser/part';
import { QuickOpenController } from 'vs/workbench/browser/parts/quickopen/quickOpenController';
import { Sash, ISashEvent, IVerticalSashLayoutProvider, IHorizontalSashLayoutProvider, Orientation } from 'vs/base/browser/ui/sash/sash';
import { IWorkbenchEditorService } from 'vs/workbench/services/editor/common/editorService';
import { IPartService, Position } from 'vs/workbench/services/part/common/partService';
import { IViewletService } from 'vs/workbench/services/viewlet/common/viewletService';
import { IStorageService, StorageScope } from 'vs/platform/storage/common/storage';
import { IContextViewService } from 'vs/platform/contextview/browser/contextView';
import { IEventService } from 'vs/platform/event/common/event';
import { IThemeService } from 'vs/workbench/services/themes/common/themeService';
import { IDisposable, dispose } from 'vs/base/common/lifecycle';
import { IEditorGroupService } from 'vs/workbench/services/group/common/groupService';
E
Erich Gamma 已提交
20 21

const DEFAULT_MIN_PART_WIDTH = 170;
I
isidor 已提交
22
const DEFAULT_MIN_PANEL_PART_HEIGHT = 77;
B
Benjamin Pasero 已提交
23
const DEFAULT_MIN_EDITOR_PART_HEIGHT = 210; /* 3 x 70px min height of editors when stacked vertically */
E
Erich Gamma 已提交
24
const HIDE_SIDEBAR_WIDTH_THRESHOLD = 50;
I
isidor 已提交
25
const HIDE_PANEL_HEIGHT_THRESHOLD = 50;
E
Erich Gamma 已提交
26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43

export class LayoutOptions {
	public margin: Box;

	constructor() {
		this.margin = new Box(0, 0, 0, 0);
	}

	public setMargin(margin: Box): LayoutOptions {
		this.margin = margin;

		return this;
	}
}

interface ComputedStyles {
	activitybar: { minWidth: number; };
	sidebar: { minWidth: number; };
B
Benjamin Pasero 已提交
44
	panel: { minHeight: number; };
E
Erich Gamma 已提交
45 46 47 48
	editor: { minWidth: number; };
	statusbar: { height: number; };
}

B
Benjamin Pasero 已提交
49 50 51 52 53
export interface ILayoutOptions {
	forceStyleReCompute?: boolean;
	toggleMaximizedPanel?: boolean;
}

E
Erich Gamma 已提交
54
/**
B
Benjamin Pasero 已提交
55
 * The workbench layout is responsible to lay out all parts that make the Workbench.
E
Erich Gamma 已提交
56
 */
I
isidor 已提交
57
export class WorkbenchLayout implements IVerticalSashLayoutProvider, IHorizontalSashLayoutProvider {
E
Erich Gamma 已提交
58

I
isidor 已提交
59
	private static sashXWidthSettingsKey = 'workbench.sidebar.width';
I
isidor 已提交
60
	private static sashYHeightSettingsKey = 'workbench.panel.height';
E
Erich Gamma 已提交
61 62 63 64 65 66

	private parent: Builder;
	private workbenchContainer: Builder;
	private activitybar: Part;
	private editor: Part;
	private sidebar: Part;
B
Benjamin Pasero 已提交
67
	private panel: Part;
E
Erich Gamma 已提交
68 69 70
	private statusbar: Part;
	private quickopen: QuickOpenController;
	private options: LayoutOptions;
M
Martin Aeschlimann 已提交
71
	private toUnbind: IDisposable[];
E
Erich Gamma 已提交
72 73
	private computedStyles: ComputedStyles;
	private workbenchSize: Dimension;
I
isidor 已提交
74 75
	private sashX: Sash;
	private sashY: Sash;
E
Erich Gamma 已提交
76 77
	private startSidebarWidth: number;
	private sidebarWidth: number;
I
isidor 已提交
78
	private sidebarHeight: number;
I
isidor 已提交
79 80 81
	private startPanelHeight: number;
	private panelHeight: number;
	private panelWidth: number;
E
Erich Gamma 已提交
82

I
isidor 已提交
83
	// Take parts as an object bag since instatation service does not have typings for constructors with 9+ arguments
E
Erich Gamma 已提交
84 85 86
	constructor(
		parent: Builder,
		workbenchContainer: Builder,
I
isidor 已提交
87 88 89 90 91 92 93
		parts: {
			activitybar: Part,
			editor: Part,
			sidebar: Part,
			panel: Part,
			statusbar: Part
		},
E
Erich Gamma 已提交
94 95 96
		quickopen: QuickOpenController,
		options: LayoutOptions,
		@IStorageService private storageService: IStorageService,
M
Martin Aeschlimann 已提交
97
		@IEventService eventService: IEventService,
E
Erich Gamma 已提交
98 99
		@IContextViewService private contextViewService: IContextViewService,
		@IWorkbenchEditorService private editorService: IWorkbenchEditorService,
100
		@IEditorGroupService editorGroupService: IEditorGroupService,
M
Martin Aeschlimann 已提交
101
		@IPartService private partService: IPartService,
M
Maxime Quandalle 已提交
102
		@IViewletService private viewletService: IViewletService,
M
Martin Aeschlimann 已提交
103
		@IThemeService themeService: IThemeService
E
Erich Gamma 已提交
104 105 106
	) {
		this.parent = parent;
		this.workbenchContainer = workbenchContainer;
I
isidor 已提交
107 108 109
		this.activitybar = parts.activitybar;
		this.editor = parts.editor;
		this.sidebar = parts.sidebar;
B
Benjamin Pasero 已提交
110
		this.panel = parts.panel;
I
isidor 已提交
111
		this.statusbar = parts.statusbar;
E
Erich Gamma 已提交
112 113 114 115
		this.quickopen = quickopen;
		this.options = options || new LayoutOptions();
		this.toUnbind = [];
		this.computedStyles = null;
I
isidor 已提交
116
		this.sashX = new Sash(this.workbenchContainer.getHTMLElement(), this, {
E
Erich Gamma 已提交
117 118
			baseSize: 5
		});
I
isidor 已提交
119
		this.sashY = new Sash(this.workbenchContainer.getHTMLElement(), this, {
I
isidor 已提交
120
			baseSize: 4,
I
isidor 已提交
121 122
			orientation: Orientation.HORIZONTAL
		});
E
Erich Gamma 已提交
123

I
isidor 已提交
124
		this.sidebarWidth = this.storageService.getInteger(WorkbenchLayout.sashXWidthSettingsKey, StorageScope.GLOBAL, -1);
I
isidor 已提交
125
		this.panelHeight = this.storageService.getInteger(WorkbenchLayout.sashYHeightSettingsKey, StorageScope.GLOBAL, 0);
E
Erich Gamma 已提交
126

M
Martin Aeschlimann 已提交
127
		this.toUnbind.push(themeService.onDidColorThemeChange(_ => this.relayout()));
128
		this.toUnbind.push(editorGroupService.onEditorsChanged(() => this.onEditorsChanged()));
M
Martin Aeschlimann 已提交
129

E
Erich Gamma 已提交
130 131 132 133 134 135
		this.registerSashListeners();
	}


	private registerSashListeners(): void {
		let startX: number = 0;
I
isidor 已提交
136
		let startY: number = 0;
E
Erich Gamma 已提交
137

A
Alex Dima 已提交
138
		this.sashX.addListener2('start', (e: ISashEvent) => {
E
Erich Gamma 已提交
139 140 141 142
			this.startSidebarWidth = this.sidebarWidth;
			startX = e.startX;
		});

A
Alex Dima 已提交
143
		this.sashY.addListener2('start', (e: ISashEvent) => {
I
isidor 已提交
144
			this.startPanelHeight = this.panelHeight;
I
isidor 已提交
145 146 147
			startY = e.startY;
		});

A
Alex Dima 已提交
148
		this.sashX.addListener2('change', (e: ISashEvent) => {
E
Erich Gamma 已提交
149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186
			let doLayout = false;
			let sidebarPosition = this.partService.getSideBarPosition();
			let isSidebarHidden = this.partService.isSideBarHidden();
			let newSashWidth = (sidebarPosition === Position.LEFT) ? this.startSidebarWidth + e.currentX - startX : this.startSidebarWidth - e.currentX + startX;

			// Sidebar visible
			if (!isSidebarHidden) {

				// Automatically hide side bar when a certain threshold is met
				if (newSashWidth + HIDE_SIDEBAR_WIDTH_THRESHOLD < this.computedStyles.sidebar.minWidth) {
					let dragCompensation = DEFAULT_MIN_PART_WIDTH - HIDE_SIDEBAR_WIDTH_THRESHOLD;
					this.partService.setSideBarHidden(true);
					startX = (sidebarPosition === Position.LEFT) ? Math.max(this.computedStyles.activitybar.minWidth, e.currentX - dragCompensation) : Math.min(e.currentX + dragCompensation, this.workbenchSize.width - this.computedStyles.activitybar.minWidth);
					this.sidebarWidth = this.startSidebarWidth; // when restoring sidebar, restore to the sidebar width we started from
				}

				// Otherwise size the sidebar accordingly
				else {
					this.sidebarWidth = Math.max(this.computedStyles.sidebar.minWidth, newSashWidth); // Sidebar can not become smaller than MIN_PART_WIDTH
					doLayout = newSashWidth >= this.computedStyles.sidebar.minWidth;
				}
			}

			// Sidebar hidden
			else {
				if ((sidebarPosition === Position.LEFT && e.currentX - startX >= this.computedStyles.sidebar.minWidth) ||
					(sidebarPosition === Position.RIGHT && startX - e.currentX >= this.computedStyles.sidebar.minWidth)) {
					this.startSidebarWidth = this.computedStyles.sidebar.minWidth - (sidebarPosition === Position.LEFT ? e.currentX - startX : startX - e.currentX);
					this.sidebarWidth = this.computedStyles.sidebar.minWidth;
					this.partService.setSideBarHidden(false);
				}
			}

			if (doLayout) {
				this.layout();
			}
		});

A
Alex Dima 已提交
187
		this.sashY.addListener2('change', (e: ISashEvent) => {
I
isidor 已提交
188
			let doLayout = false;
I
isidor 已提交
189
			let isPanelHidden = this.partService.isPanelHidden();
190
			let isStatusbarHidden = this.partService.isStatusBarHidden();
I
isidor 已提交
191
			let newSashHeight = this.startPanelHeight - (e.currentY - startY);
I
isidor 已提交
192 193

			// Panel visible
I
isidor 已提交
194
			if (!isPanelHidden) {
195 196 197 198
				// Automatically hide panel when a certain threshold is met
				if (newSashHeight + HIDE_PANEL_HEIGHT_THRESHOLD < this.computedStyles.panel.minHeight) {
					let dragCompensation = DEFAULT_MIN_PANEL_PART_HEIGHT - HIDE_PANEL_HEIGHT_THRESHOLD;
					this.partService.setPanelHidden(true);
199 200
					let statusbarHeight = isStatusbarHidden ? 0 : this.computedStyles.statusbar.height;
					startY = Math.min(this.sidebarHeight - statusbarHeight, e.currentY + dragCompensation);
M
Maxime Quandalle 已提交
201
					this.panelHeight = this.startPanelHeight; // when restoring panel, restore to the panel height we started from
202 203 204 205 206 207 208 209
				}

				// Otherwise size the panel accordingly
				else {
					this.panelHeight = Math.max(this.computedStyles.panel.minHeight, newSashHeight); // Panel can not become smaller than MIN_PART_HEIGHT
					doLayout = newSashHeight >= this.computedStyles.panel.minHeight;
				}
			}
I
isidor 已提交
210

211 212 213 214 215 216 217
			// Panel hidden
			else {
				if (startY - e.currentY >= this.computedStyles.panel.minHeight) {
					this.startPanelHeight = 0;
					this.panelHeight = this.computedStyles.panel.minHeight;
					this.partService.setPanelHidden(false);
				}
I
isidor 已提交
218
			}
219

I
isidor 已提交
220 221 222 223 224
			if (doLayout) {
				this.layout();
			}
		});

A
Alex Dima 已提交
225
		this.sashX.addListener2('end', () => {
I
isidor 已提交
226 227
			this.storageService.store(WorkbenchLayout.sashXWidthSettingsKey, this.sidebarWidth, StorageScope.GLOBAL);
		});
M
Maxime Quandalle 已提交
228

A
Alex Dima 已提交
229
		this.sashY.addListener2('end', () => {
I
isidor 已提交
230
			this.storageService.store(WorkbenchLayout.sashYHeightSettingsKey, this.panelHeight, StorageScope.GLOBAL);
E
Erich Gamma 已提交
231
		});
M
Maxime Quandalle 已提交
232

A
Alex Dima 已提交
233
		this.sashY.addListener2('reset', () => {
M
Maxime Quandalle 已提交
234 235 236 237 238 239
			this.panelHeight = DEFAULT_MIN_PANEL_PART_HEIGHT;
			this.storageService.store(WorkbenchLayout.sashYHeightSettingsKey, this.panelHeight, StorageScope.GLOBAL);
			this.partService.setPanelHidden(false);
			this.layout();
		});

A
Alex Dima 已提交
240
		this.sashX.addListener2('reset', () => {
M
Maxime Quandalle 已提交
241 242 243 244 245 246 247
			let activeViewlet = this.viewletService.getActiveViewlet();
			let optimalWidth = activeViewlet && activeViewlet.getOptimalWidth();
			this.sidebarWidth = Math.max(DEFAULT_MIN_PART_WIDTH, optimalWidth || 0);
			this.storageService.store(WorkbenchLayout.sashXWidthSettingsKey, this.sidebarWidth, StorageScope.GLOBAL);
			this.partService.setSideBarHidden(false);
			this.layout();
		});
E
Erich Gamma 已提交
248 249
	}

250
	private onEditorsChanged(): void {
E
Erich Gamma 已提交
251 252 253 254 255 256 257 258 259 260 261 262

		// Make sure that we layout properly in case we detect that the sidebar is large enought to cause
		// multiple opened editors to go below minimal size. The fix is to trigger a layout for any editor
		// input change that falls into this category.
		if (this.workbenchSize && this.sidebarWidth) {
			let visibleEditors = this.editorService.getVisibleEditors().length;
			if (visibleEditors > 1 && this.workbenchSize.width - this.sidebarWidth < visibleEditors * DEFAULT_MIN_PART_WIDTH) {
				this.layout();
			}
		}
	}

M
Martin Aeschlimann 已提交
263
	private relayout(): void {
E
Erich Gamma 已提交
264

M
Martin Aeschlimann 已提交
265 266 267 268 269
		// Recompute Styles
		this.computeStyle();
		this.editor.getLayout().computeStyle();
		this.sidebar.getLayout().computeStyle();
		this.panel.getLayout().computeStyle();
E
Erich Gamma 已提交
270

M
Martin Aeschlimann 已提交
271 272
		// Trigger Layout
		this.layout();
E
Erich Gamma 已提交
273 274 275
	}

	private computeStyle(): void {
I
isidor 已提交
276
		const sidebarStyle = this.sidebar.getContainer().getComputedStyle();
B
Benjamin Pasero 已提交
277
		const panelStyle = this.panel.getContainer().getComputedStyle();
I
isidor 已提交
278 279
		const editorStyle = this.editor.getContainer().getComputedStyle();
		const activitybarStyle = this.activitybar.getContainer().getComputedStyle();
B
Benjamin Pasero 已提交
280
		const statusbarStyle = this.statusbar.getContainer().getComputedStyle();
E
Erich Gamma 已提交
281 282 283 284 285 286 287 288 289 290

		this.computedStyles = {
			activitybar: {
				minWidth: parseInt(activitybarStyle.getPropertyValue('min-width'), 10) || 0
			},

			sidebar: {
				minWidth: parseInt(sidebarStyle.getPropertyValue('min-width'), 10) || DEFAULT_MIN_PART_WIDTH
			},

I
isidor 已提交
291
			panel: {
B
Benjamin Pasero 已提交
292
				minHeight: parseInt(panelStyle.getPropertyValue('min-height'), 10) || DEFAULT_MIN_PANEL_PART_HEIGHT
I
isidor 已提交
293 294
			},

E
Erich Gamma 已提交
295 296 297 298 299
			editor: {
				minWidth: parseInt(editorStyle.getPropertyValue('min-width'), 10) || DEFAULT_MIN_PART_WIDTH
			},

			statusbar: {
B
Benjamin Pasero 已提交
300
				height: parseInt(statusbarStyle.getPropertyValue('height'), 10) || 18
E
Erich Gamma 已提交
301 302 303 304
			}
		};
	}

B
Benjamin Pasero 已提交
305 306
	public layout(options?: ILayoutOptions): void {
		if (options && options.forceStyleReCompute) {
E
Erich Gamma 已提交
307 308 309
			this.computeStyle();
			this.editor.getLayout().computeStyle();
			this.sidebar.getLayout().computeStyle();
B
Benjamin Pasero 已提交
310
			this.panel.getLayout().computeStyle();
E
Erich Gamma 已提交
311 312 313 314 315 316 317 318
		}

		if (!this.computedStyles) {
			this.computeStyle();
		}

		this.workbenchSize = this.getWorkbenchArea();

I
isidor 已提交
319
		const isSidebarHidden = this.partService.isSideBarHidden();
I
isidor 已提交
320
		const isPanelHidden = this.partService.isPanelHidden();
I
isidor 已提交
321
		const sidebarPosition = this.partService.getSideBarPosition();
322
		const isStatusbarHidden = this.partService.isStatusBarHidden();
E
Erich Gamma 已提交
323 324 325 326 327 328 329 330 331 332 333

		// Sidebar
		let sidebarWidth: number;
		if (isSidebarHidden) {
			sidebarWidth = 0;
		} else if (this.sidebarWidth !== -1) {
			sidebarWidth = Math.max(this.computedStyles.sidebar.minWidth, this.sidebarWidth);
		} else {
			sidebarWidth = this.workbenchSize.width / 5;
			this.sidebarWidth = sidebarWidth;
		}
B
Benjamin Pasero 已提交
334

335
		let statusbarHeight = isStatusbarHidden ? 0 : this.computedStyles.statusbar.height;
E
Erich Gamma 已提交
336

337
		this.sidebarHeight = this.workbenchSize.height - statusbarHeight;
I
isidor 已提交
338
		let sidebarSize = new Dimension(sidebarWidth, this.sidebarHeight);
E
Erich Gamma 已提交
339 340 341 342 343

		// Activity Bar
		let activityBarMinWidth = this.computedStyles.activitybar.minWidth;
		let activityBarSize = new Dimension(activityBarMinWidth, sidebarSize.height);

I
isidor 已提交
344 345
		// Panel part
		let panelHeight: number;
I
isidor 已提交
346
		const maxPanelHeight = sidebarSize.height - DEFAULT_MIN_EDITOR_PART_HEIGHT;
I
isidor 已提交
347
		if (isPanelHidden) {
I
isidor 已提交
348
			panelHeight = 0;
I
isidor 已提交
349
		} else if (this.panelHeight > 0) {
I
isidor 已提交
350
			panelHeight = Math.min(maxPanelHeight, Math.max(this.computedStyles.panel.minHeight, this.panelHeight));
I
isidor 已提交
351 352 353
		} else {
			panelHeight = sidebarSize.height * 0.4;
		}
B
Benjamin Pasero 已提交
354
		if (options && options.toggleMaximizedPanel) {
I
isidor 已提交
355 356
			panelHeight = panelHeight === maxPanelHeight ? sidebarSize.height * 0.4 : maxPanelHeight;
		}
I
isidor 已提交
357
		const panelDimension = new Dimension(this.workbenchSize.width - sidebarSize.width - activityBarSize.width, panelHeight);
I
isidor 已提交
358
		this.panelWidth = panelDimension.width;
I
isidor 已提交
359

E
Erich Gamma 已提交
360 361 362 363 364 365 366 367
		// Editor
		let editorSize = {
			width: 0,
			height: 0,
			remainderLeft: 0,
			remainderRight: 0
		};

I
isidor 已提交
368
		let editorDimension = new Dimension(panelDimension.width, sidebarSize.height - panelDimension.height);
E
Erich Gamma 已提交
369
		editorSize.width = editorDimension.width;
I
isidor 已提交
370
		editorSize.height = editorDimension.height;
E
Erich Gamma 已提交
371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390

		// Sidebar hidden
		if (isSidebarHidden) {
			editorSize.width = Math.min(this.workbenchSize.width - activityBarSize.width, this.workbenchSize.width - activityBarMinWidth);

			if (sidebarPosition === Position.LEFT) {
				editorSize.remainderLeft = Math.round((this.workbenchSize.width - editorSize.width + activityBarSize.width) / 2);
				editorSize.remainderRight = this.workbenchSize.width - editorSize.width - editorSize.remainderLeft;
			} else {
				editorSize.remainderRight = Math.round((this.workbenchSize.width - editorSize.width + activityBarSize.width) / 2);
				editorSize.remainderLeft = this.workbenchSize.width - editorSize.width - editorSize.remainderRight;
			}
		}

		// Assert Sidebar and Editor Size to not overflow
		let editorMinWidth = this.computedStyles.editor.minWidth;
		let visibleEditorCount = this.editorService.getVisibleEditors().length;
		if (visibleEditorCount > 1) {
			editorMinWidth *= visibleEditorCount;
		}
B
Benjamin Pasero 已提交
391

E
Erich Gamma 已提交
392 393 394
		if (editorSize.width < editorMinWidth) {
			let diff = editorMinWidth - editorSize.width;
			editorSize.width = editorMinWidth;
395
			panelDimension.width = editorMinWidth;
E
Erich Gamma 已提交
396 397 398 399 400 401
			sidebarSize.width -= diff;
			sidebarSize.width = Math.max(DEFAULT_MIN_PART_WIDTH, sidebarSize.width);
		}

		if (!isSidebarHidden) {
			this.sidebarWidth = sidebarSize.width;
I
isidor 已提交
402
			this.storageService.store(WorkbenchLayout.sashXWidthSettingsKey, this.sidebarWidth, StorageScope.GLOBAL);
E
Erich Gamma 已提交
403 404
		}

I
isidor 已提交
405 406 407
		if (!isPanelHidden) {
			this.panelHeight = panelDimension.height;
			this.storageService.store(WorkbenchLayout.sashYHeightSettingsKey, this.panelHeight, StorageScope.GLOBAL);
I
isidor 已提交
408 409
		}

E
Erich Gamma 已提交
410 411 412 413 414 415 416 417 418 419
		// Workbench
		this.workbenchContainer
			.position(this.options.margin.top, this.options.margin.right, this.options.margin.bottom, this.options.margin.left, 'relative')
			.size(this.workbenchSize.width, this.workbenchSize.height);

		// Bug on Chrome: Sometimes Chrome wants to scroll the workbench container on layout changes. The fix is to reset scrollTop in this case.
		if (this.workbenchContainer.getHTMLElement().scrollTop > 0) {
			this.workbenchContainer.getHTMLElement().scrollTop = 0;
		}

I
isidor 已提交
420
		// Editor Part and Panel part
E
Erich Gamma 已提交
421
		this.editor.getContainer().size(editorSize.width, editorSize.height);
B
Benjamin Pasero 已提交
422
		this.panel.getContainer().size(panelDimension.width, panelDimension.height);
E
Erich Gamma 已提交
423

424
		const editorBottom = statusbarHeight + panelDimension.height;
E
Erich Gamma 已提交
425
		if (isSidebarHidden) {
426
			this.editor.getContainer().position(0, editorSize.remainderRight, editorBottom, editorSize.remainderLeft);
427
			this.panel.getContainer().position(editorDimension.height, editorSize.remainderRight, statusbarHeight, editorSize.remainderLeft);
E
Erich Gamma 已提交
428
		} else if (sidebarPosition === Position.LEFT) {
429
			this.editor.getContainer().position(0, 0, editorBottom, sidebarSize.width + activityBarSize.width);
430
			this.panel.getContainer().position(editorDimension.height, 0, statusbarHeight, sidebarSize.width + activityBarSize.width);
E
Erich Gamma 已提交
431
		} else {
432
			this.editor.getContainer().position(0, sidebarSize.width, editorBottom, 0);
433
			this.panel.getContainer().position(editorDimension.height, sidebarSize.width, statusbarHeight, 0);
E
Erich Gamma 已提交
434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455
		}

		// Activity Bar Part
		this.activitybar.getContainer().size(null, activityBarSize.height);
		if (sidebarPosition === Position.LEFT) {
			this.activitybar.getContainer().getHTMLElement().style.right = '';
			this.activitybar.getContainer().position(0, null, 0, 0);
		} else {
			this.activitybar.getContainer().getHTMLElement().style.left = '';
			this.activitybar.getContainer().position(0, 0, 0, null);
		}

		// Sidebar Part
		this.sidebar.getContainer().size(sidebarSize.width, sidebarSize.height);

		if (sidebarPosition === Position.LEFT) {
			this.sidebar.getContainer().position(0, editorSize.width, 0, activityBarSize.width);
		} else {
			this.sidebar.getContainer().position(0, null, 0, editorSize.width);
		}

		// Statusbar Part
B
Benjamin Pasero 已提交
456
		this.statusbar.getContainer().position(this.workbenchSize.height - statusbarHeight);
E
Erich Gamma 已提交
457 458 459 460

		// Quick open
		this.quickopen.layout(this.workbenchSize);

I
isidor 已提交
461 462 463
		// Sashes
		this.sashX.layout();
		this.sashY.layout();
E
Erich Gamma 已提交
464 465 466 467

		// Propagate to Part Layouts
		this.editor.layout(new Dimension(editorSize.width, editorSize.height));
		this.sidebar.layout(sidebarSize);
B
Benjamin Pasero 已提交
468
		this.panel.layout(panelDimension);
E
Erich Gamma 已提交
469 470

		// Propagate to Context View
B
Benjamin Pasero 已提交
471
		this.contextViewService.layout();
E
Erich Gamma 已提交
472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499
	}

	private getWorkbenchArea(): Dimension {

		// Client Area: Parent
		let clientArea = this.parent.getClientArea();

		// Workbench: Client Area - Margins
		return clientArea.substract(this.options.margin);
	}

	public getVerticalSashTop(sash: Sash): number {
		return 0;
	}

	public getVerticalSashLeft(sash: Sash): number {
		let isSidebarHidden = this.partService.isSideBarHidden();
		let sidebarPosition = this.partService.getSideBarPosition();
		let activitybarWidth = this.computedStyles.activitybar.minWidth;

		if (sidebarPosition === Position.LEFT) {
			return !isSidebarHidden ? this.sidebarWidth + activitybarWidth : activitybarWidth;
		}

		return !isSidebarHidden ? this.workbenchSize.width - this.sidebarWidth - activitybarWidth : this.workbenchSize.width - activitybarWidth;
	}

	public getVerticalSashHeight(sash: Sash): number {
I
isidor 已提交
500
		return this.sidebarHeight;
E
Erich Gamma 已提交
501 502
	}

I
isidor 已提交
503
	public getHorizontalSashTop(sash: Sash): number {
B
Benjamin Pasero 已提交
504
		return 2 + (this.partService.isPanelHidden() ? this.sidebarHeight : this.sidebarHeight - this.panelHeight); // Horizontal sash should be a bit lower than the editor area, thus add 2px #5524
I
isidor 已提交
505 506 507
	}

	public getHorizontalSashLeft(sash: Sash): number {
508
		return this.partService.getSideBarPosition() === Position.LEFT ? this.getVerticalSashLeft(sash) : 0;
I
isidor 已提交
509 510 511
	}

	public getHorizontalSashWidth(sash: Sash): number {
I
isidor 已提交
512
		return this.panelWidth;
I
isidor 已提交
513 514
	}

E
Erich Gamma 已提交
515
	public dispose(): void {
M
Martin Aeschlimann 已提交
516 517 518
		if (this.toUnbind) {
			dispose(this.toUnbind);
			this.toUnbind = null;
E
Erich Gamma 已提交
519 520
		}
	}
B
Benjamin Pasero 已提交
521
}