mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-16 17:22:29 -05:00
* add go to next output query tab shotcut * add a new keyboard shortcut for focus on current query * minor change
273 lines
8.1 KiB
TypeScript
273 lines
8.1 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import { Component, ContentChildren, QueryList, AfterContentInit, Inject, forwardRef, NgZone, OnInit, Input, EventEmitter, Output, ViewChild, ElementRef, OnChanges, OnDestroy, ViewChildren, AfterViewInit } from '@angular/core';
|
|
|
|
import { TabComponent } from './tab.component';
|
|
import { TabHeaderComponent } from './tabHeader.component';
|
|
import './panelStyles';
|
|
|
|
import { ActionBar } from 'vs/base/browser/ui/actionbar/actionbar';
|
|
import { Action } from 'vs/base/common/actions';
|
|
import * as types from 'vs/base/common/types';
|
|
import { mixin } from 'vs/base/common/objects';
|
|
import { ScrollableElement } from 'vs/base/browser/ui/scrollbar/scrollableElement';
|
|
import { ScrollbarVisibility } from 'vs/base/common/scrollable';
|
|
import { addDisposableListener, EventType } from 'vs/base/browser/dom';
|
|
import { Disposable } from 'vs/base/common/lifecycle';
|
|
|
|
export interface IPanelOptions {
|
|
/**
|
|
* Whether or not to show the tabs if there is only one tab present
|
|
*/
|
|
showTabsWhenOne?: boolean;
|
|
layout?: NavigationBarLayout;
|
|
showIcon?: boolean;
|
|
}
|
|
|
|
export enum NavigationBarLayout {
|
|
horizontal = 0,
|
|
vertical = 1
|
|
}
|
|
|
|
const defaultOptions: IPanelOptions = {
|
|
showTabsWhenOne: true,
|
|
layout: NavigationBarLayout.horizontal,
|
|
showIcon: false
|
|
};
|
|
|
|
const verticalLayout = 'vertical';
|
|
const horizontalLayout = 'horizontal';
|
|
|
|
let idPool = 0;
|
|
|
|
@Component({
|
|
selector: 'panel',
|
|
template: `
|
|
|
|
<div class="tabbedPanel fullsize" #tabbedPanel>
|
|
<div *ngIf="!options.showTabsWhenOne ? _tabs.length !== 1 : true" class="composite title" #titleContainer>
|
|
<div class="tabList" #tabList>
|
|
<div *ngFor="let tab of _tabs">
|
|
<tab-header [tab]="tab" [showIcon]="options.showIcon" (onSelectTab)='selectTab($event)' (onCloseTab)='closeTab($event)'> </tab-header>
|
|
</div>
|
|
</div>
|
|
<div class="title-actions">
|
|
<div #panelActionbar class="panel-actions" style="flex: 0 0 auto; align-self: end; margin-top: auto; margin-bottom: auto;" >
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="tab-content">
|
|
<div class="fullsize" style="position: absolute">
|
|
<ng-content></ng-content>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
`
|
|
})
|
|
export class PanelComponent extends Disposable implements AfterContentInit, OnInit, OnChanges, OnDestroy, AfterViewInit {
|
|
@Input() public options: IPanelOptions;
|
|
@Input() public actions: Array<Action>;
|
|
@ContentChildren(TabComponent) private _tabs: QueryList<TabComponent>;
|
|
@ViewChildren(TabHeaderComponent) private _headerTabs: QueryList<TabHeaderComponent>;
|
|
|
|
@Output() public onTabChange = new EventEmitter<TabComponent>();
|
|
@Output() public onTabClose = new EventEmitter<TabComponent>();
|
|
|
|
private _activeTab: TabComponent;
|
|
private _actionbar: ActionBar;
|
|
private _mru: TabComponent[];
|
|
private _scrollableElement: ScrollableElement;
|
|
|
|
@ViewChild('panelActionbar', { read: ElementRef }) private _actionbarRef: ElementRef;
|
|
@ViewChild('tabbedPanel', { read: ElementRef }) private _tabbedPanelRef: ElementRef;
|
|
@ViewChild('titleContainer', { read: ElementRef }) private _titleContainer: ElementRef;
|
|
@ViewChild('tabList', { read: ElementRef }) private _tabList: ElementRef;
|
|
constructor( @Inject(forwardRef(() => NgZone)) private _zone: NgZone) {
|
|
super();
|
|
}
|
|
|
|
ngOnInit(): void {
|
|
this.options = mixin(this.options || {}, defaultOptions, false);
|
|
this._mru = [];
|
|
}
|
|
|
|
ngAfterContentInit(): void {
|
|
if (this._tabs && this._tabs.length > 0) {
|
|
this._activeTab = this._tabs.first;
|
|
this._activeTab.active = true;
|
|
}
|
|
}
|
|
|
|
ngAfterViewInit(): void {
|
|
let container = this._titleContainer.nativeElement as HTMLElement;
|
|
let tabList = this._tabList.nativeElement as HTMLElement;
|
|
container.removeChild(tabList);
|
|
|
|
this._scrollableElement = new ScrollableElement(tabList, {
|
|
horizontal: ScrollbarVisibility.Auto,
|
|
vertical: ScrollbarVisibility.Hidden,
|
|
scrollYToX: true,
|
|
useShadows: false,
|
|
horizontalScrollbarSize: 3
|
|
});
|
|
|
|
this._scrollableElement.onScroll(e => {
|
|
tabList.scrollLeft = e.scrollLeft;
|
|
});
|
|
|
|
container.insertBefore(this._scrollableElement.getDomNode(), container.firstChild);
|
|
|
|
this._scrollableElement.setScrollDimensions({
|
|
width: tabList.offsetWidth,
|
|
scrollWidth: tabList.scrollWidth
|
|
});
|
|
|
|
this._register(addDisposableListener(window, EventType.RESIZE, () => {
|
|
// Todo: Need to set timeout because we have to make sure that the grids have already rearraged before the getContentHeight gets called.
|
|
setTimeout(() => {
|
|
this._scrollableElement.setScrollDimensions({
|
|
width: tabList.offsetWidth,
|
|
scrollWidth: tabList.scrollWidth
|
|
});
|
|
}, 100);
|
|
}));
|
|
|
|
if (this.options.layout === NavigationBarLayout.horizontal) {
|
|
(<HTMLElement>this._tabbedPanelRef.nativeElement).classList.add(horizontalLayout);
|
|
} else {
|
|
(<HTMLElement>this._tabbedPanelRef.nativeElement).classList.add(verticalLayout);
|
|
}
|
|
}
|
|
|
|
ngOnChanges(): void {
|
|
if (this._actionbarRef && !this._actionbar) {
|
|
this._actionbar = new ActionBar(this._actionbarRef.nativeElement);
|
|
}
|
|
if (this.actions && this._actionbar) {
|
|
this._actionbar.clear();
|
|
this._actionbar.push(this.actions, { icon: true, label: false });
|
|
}
|
|
}
|
|
|
|
ngOnDestroy() {
|
|
if (this._actionbar) {
|
|
this._actionbar.dispose();
|
|
}
|
|
if (this.actions && this.actions.length > 0) {
|
|
this.actions.forEach((action) => action.dispose());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Select a tab based on index (unrecommended)
|
|
* @param index index of tab in the html
|
|
*/
|
|
selectTab(index: number);
|
|
/**
|
|
* Select a tab based on the identifier that was passed into the tab
|
|
* @param identifier specified identifer of the tab
|
|
*/
|
|
selectTab(identifier: string);
|
|
/**
|
|
* Select a tab directly if you have access to the object
|
|
* @param tab tab to navigate to
|
|
*/
|
|
selectTab(tab: TabComponent);
|
|
selectTab(input: TabComponent | number | string) {
|
|
if (this._tabs && this._tabs.length > 0) {
|
|
let tab: TabComponent;
|
|
if (input instanceof TabComponent) {
|
|
tab = input;
|
|
} else if (types.isNumber(input)) {
|
|
tab = this._tabs.toArray()[input];
|
|
} else if (types.isString(input)) {
|
|
tab = this._tabs.find(i => i.identifier === input);
|
|
}
|
|
|
|
// since we need to compare identifiers in this next step we are going to go through and make sure all tabs have one
|
|
this._tabs.forEach(i => {
|
|
if (!i.identifier) {
|
|
i.identifier = 'tabIndex_' + idPool++;
|
|
}
|
|
});
|
|
|
|
if (this._activeTab && tab === this._activeTab) {
|
|
this.onTabChange.emit(tab);
|
|
return;
|
|
}
|
|
|
|
this._zone.run(() => {
|
|
if (this._activeTab) {
|
|
this._activeTab.active = false;
|
|
}
|
|
|
|
this._activeTab = tab;
|
|
this.setMostRecentlyUsed(tab);
|
|
this._activeTab.active = true;
|
|
|
|
// Make the tab header focus on the new selected tab
|
|
let activeTabHeader = this._headerTabs.find(i => i.tab === this._activeTab);
|
|
if (activeTabHeader) {
|
|
activeTabHeader.focusOnTabHeader();
|
|
}
|
|
|
|
this.onTabChange.emit(tab);
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get the id of the active tab
|
|
*/
|
|
public get getActiveTab(): string {
|
|
return this._activeTab.identifier;
|
|
}
|
|
|
|
/**
|
|
* Select on the next tab
|
|
*/
|
|
public selectOnNextTab(): void {
|
|
let activeIndex = this._tabs.toArray().findIndex(i => i === this._activeTab);
|
|
let nextTabIndex = activeIndex + 1;
|
|
if (nextTabIndex === this._tabs.length) {
|
|
nextTabIndex = 0;
|
|
}
|
|
this.selectTab(nextTabIndex);
|
|
}
|
|
|
|
private findAndRemoveTabFromMRU(tab: TabComponent): void {
|
|
let mruIndex = this._mru.findIndex(i => i === tab);
|
|
|
|
if (mruIndex !== -1) {
|
|
// Remove old index
|
|
this._mru.splice(mruIndex, 1);
|
|
}
|
|
}
|
|
|
|
private setMostRecentlyUsed(tab: TabComponent): void {
|
|
this.findAndRemoveTabFromMRU(tab);
|
|
|
|
// Set tab to front
|
|
this._mru.unshift(tab);
|
|
}
|
|
|
|
/**
|
|
* Close a tab
|
|
* @param tab tab to close
|
|
*/
|
|
closeTab(tab: TabComponent) {
|
|
this.onTabClose.emit(tab);
|
|
|
|
// remove the closed tab from mru
|
|
this.findAndRemoveTabFromMRU(tab);
|
|
|
|
// Open the most recent tab
|
|
if (this._mru.length > 0) {
|
|
this.selectTab(this._mru[0]);
|
|
}
|
|
}
|
|
}
|