mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-14 01:25:37 -05:00
* Merge from vscode fcf3346a8e9f5ee1e00674461d9e2c2292a14ee3 * Fix test build break * Update distro * Fix build errors * Update distro * Update REH build file * Update build task names for REL * Fix product build yaml * Fix product REH task name * Fix type in task name * Update linux build step * Update windows build tasks * Turn off server publish * Disable REH * Fix typo * Bump distro * Update vscode tests * Bump distro * Fix type in disto * Bump distro * Turn off docker build * Remove docker step from release Co-authored-by: ADS Merger <andresse@microsoft.com> Co-authored-by: Karl Burtram <karlb@microsoft.com>
391 lines
12 KiB
TypeScript
391 lines
12 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 'vs/css!./media/flexContainer';
|
|
|
|
import {
|
|
ChangeDetectorRef, ViewChildren, ElementRef, OnDestroy, OnInit, QueryList
|
|
} from '@angular/core';
|
|
|
|
import * as types from 'vs/base/common/types';
|
|
|
|
import * as azdata from 'azdata';
|
|
import { Emitter } from 'vs/base/common/event';
|
|
import { IDisposable, Disposable } from 'vs/base/common/lifecycle';
|
|
import { ModelComponentWrapper } from 'sql/workbench/browser/modelComponents/modelComponentWrapper.component';
|
|
import { URI } from 'vs/base/common/uri';
|
|
import * as nls from 'vs/nls';
|
|
import { EventType, addDisposableListener } from 'vs/base/browser/dom';
|
|
import { StandardKeyboardEvent } from 'vs/base/browser/keyboardEvent';
|
|
import { IComponentDescriptor, IComponent, IModelStore, IComponentEventArgs, ComponentEventType } from 'sql/platform/dashboard/browser/interfaces';
|
|
import { convertSize } from 'sql/base/browser/dom';
|
|
|
|
export type IUserFriendlyIcon = string | URI | { light: string | URI; dark: string | URI };
|
|
|
|
export class ItemDescriptor<T> {
|
|
constructor(public descriptor: IComponentDescriptor, public config: T) { }
|
|
}
|
|
|
|
export abstract class ComponentBase<TPropertyBag extends azdata.ComponentProperties> extends Disposable implements IComponent, OnDestroy, OnInit {
|
|
protected properties: { [key: string]: any; } = {};
|
|
private _valid: boolean = true;
|
|
protected _validations: (() => boolean | Thenable<boolean>)[] = [];
|
|
private _eventQueue: IComponentEventArgs[] = [];
|
|
|
|
constructor(
|
|
protected _changeRef: ChangeDetectorRef,
|
|
protected _el: ElementRef) {
|
|
super();
|
|
}
|
|
|
|
/// IComponent implementation
|
|
|
|
abstract descriptor: IComponentDescriptor;
|
|
abstract modelStore: IModelStore;
|
|
protected _onEventEmitter = new Emitter<IComponentEventArgs>();
|
|
|
|
public layout(): void {
|
|
if (!this._changeRef['destroyed']) {
|
|
this._changeRef.detectChanges();
|
|
}
|
|
}
|
|
|
|
protected baseInit(): void {
|
|
if (this.modelStore) {
|
|
this.modelStore.registerComponent(this);
|
|
this._validations.push(() => this.modelStore.validate(this));
|
|
}
|
|
}
|
|
|
|
abstract ngOnInit(): void;
|
|
|
|
protected baseDestroy(): void {
|
|
if (this.modelStore) {
|
|
this.modelStore.unregisterComponent(this);
|
|
}
|
|
this.dispose();
|
|
}
|
|
|
|
ngOnDestroy(): void {
|
|
this.dispose();
|
|
}
|
|
|
|
abstract setLayout(layout: any): void;
|
|
|
|
getHtml(): any {
|
|
return this._el.nativeElement;
|
|
}
|
|
|
|
public setDataProvider(handle: number, componentId: string, context: any): void {
|
|
}
|
|
|
|
public refreshDataProvider(item: any): void {
|
|
}
|
|
|
|
public updateStyles(): void {
|
|
const element = (<HTMLElement>this._el.nativeElement);
|
|
for (const style in this.CSSStyles) {
|
|
element.style[style] = this.CSSStyles[style];
|
|
}
|
|
}
|
|
|
|
public setProperties(properties: { [key: string]: any; }): void {
|
|
properties = properties || {};
|
|
this.properties = properties;
|
|
this.updateStyles();
|
|
this.layout();
|
|
this.validate();
|
|
}
|
|
|
|
// Helper Function to update single property
|
|
public updateProperty(key: string, value: any): void {
|
|
if (key) {
|
|
this.properties[key] = value;
|
|
this.updateStyles();
|
|
this.layout();
|
|
this.validate();
|
|
}
|
|
}
|
|
|
|
protected getProperties(): TPropertyBag {
|
|
return this.properties as TPropertyBag;
|
|
}
|
|
|
|
protected getPropertyOrDefault<TValue>(propertyGetter: (property: TPropertyBag) => TValue, defaultVal: TValue) {
|
|
let property = propertyGetter(this.getProperties());
|
|
return types.isUndefinedOrNull(property) ? defaultVal : property;
|
|
}
|
|
|
|
protected setPropertyFromUI<TValue>(propertySetter: (TPropertyBag, TValue) => void, value: TValue) {
|
|
propertySetter(this.getProperties(), value);
|
|
this.fireEvent({
|
|
eventType: ComponentEventType.PropertiesChanged,
|
|
args: this.getProperties()
|
|
});
|
|
this.validate();
|
|
}
|
|
|
|
public get enabled(): boolean {
|
|
let properties = this.getProperties();
|
|
let enabled = properties['enabled'];
|
|
if (enabled === undefined) {
|
|
enabled = true;
|
|
properties['enabled'] = enabled;
|
|
}
|
|
return <boolean>enabled;
|
|
}
|
|
|
|
public set enabled(value: boolean) {
|
|
let properties = this.getProperties();
|
|
properties['enabled'] = value;
|
|
this.setProperties(properties);
|
|
}
|
|
|
|
public get height(): number | string {
|
|
return this.getPropertyOrDefault<number | string>((props) => props.height, undefined);
|
|
}
|
|
|
|
public set height(newValue: number | string) {
|
|
this.setPropertyFromUI<number | string>((props, value) => props.height = value, newValue);
|
|
}
|
|
|
|
public get width(): number | string {
|
|
return this.getPropertyOrDefault<number | string>((props) => props.width, undefined);
|
|
}
|
|
|
|
public set width(newValue: number | string) {
|
|
this.setPropertyFromUI<number | string>((props, value) => props.width = value, newValue);
|
|
}
|
|
|
|
public get position(): string {
|
|
return this.getPropertyOrDefault<string>((props) => props.position, '');
|
|
}
|
|
|
|
public set position(newValue: string) {
|
|
this.setPropertyFromUI<string>((properties, position) => { properties.position = position; }, newValue);
|
|
}
|
|
|
|
public get display(): azdata.DisplayType {
|
|
return this.getPropertyOrDefault<azdata.DisplayType>((props) => props.display, undefined);
|
|
}
|
|
|
|
public set display(newValue: azdata.DisplayType) {
|
|
this.setPropertyFromUI<string>((properties, display) => { properties.display = display; }, newValue);
|
|
}
|
|
|
|
public get ariaLabel(): string {
|
|
return this.getPropertyOrDefault<string>((props) => props.ariaLabel, '');
|
|
}
|
|
|
|
public set ariaLabel(newValue: string) {
|
|
this.setPropertyFromUI<string>((props, value) => props.ariaLabel = value, newValue);
|
|
}
|
|
|
|
public get ariaRole(): string {
|
|
return this.getPropertyOrDefault<string>((props) => props.ariaRole, '');
|
|
}
|
|
|
|
public set ariaRole(newValue: string) {
|
|
this.setPropertyFromUI<string>((props, value) => props.ariaRole = value, newValue);
|
|
}
|
|
|
|
public get ariaSelected(): boolean {
|
|
return this.getPropertyOrDefault<boolean>((props) => props.ariaSelected, false);
|
|
}
|
|
|
|
public set ariaSelected(newValue: boolean) {
|
|
this.setPropertyFromUI<boolean>((props, value) => props.ariaSelected = value, newValue);
|
|
}
|
|
|
|
public get ariaHidden(): boolean {
|
|
return this.getPropertyOrDefault<boolean>((props) => props.ariaHidden, false);
|
|
}
|
|
|
|
public set ariaHidden(newValue: boolean) {
|
|
this.setPropertyFromUI<boolean>((props, value) => props.ariaHidden = value, newValue);
|
|
}
|
|
|
|
public get CSSStyles(): { [key: string]: string } {
|
|
return this.getPropertyOrDefault<{ [key: string]: string }>((props) => props.CSSStyles, {});
|
|
}
|
|
|
|
public set CSSStyles(newValue: { [key: string]: string }) {
|
|
this.setPropertyFromUI<{ [key: string]: string }>((properties, CSSStyles) => { properties.CSSStyles = CSSStyles; }, newValue);
|
|
}
|
|
|
|
protected getWidth(): string {
|
|
return this.width ? convertSize(this.width) : '';
|
|
}
|
|
|
|
protected getHeight(): string {
|
|
return this.height ? convertSize(this.height) : '';
|
|
}
|
|
|
|
public get valid(): boolean {
|
|
return this._valid;
|
|
}
|
|
|
|
public registerEventHandler(handler: (event: IComponentEventArgs) => void): IDisposable {
|
|
if (this._eventQueue) {
|
|
while (this._eventQueue.length > 0) {
|
|
let event = this._eventQueue.pop();
|
|
handler(event);
|
|
}
|
|
this._eventQueue = undefined;
|
|
}
|
|
return this._onEventEmitter.event(handler);
|
|
}
|
|
|
|
protected fireEvent(event: IComponentEventArgs) {
|
|
this._onEventEmitter.fire(event);
|
|
if (this._eventQueue) {
|
|
this._eventQueue.push(event);
|
|
}
|
|
}
|
|
|
|
public validate(): Thenable<boolean> {
|
|
let validations = this._validations.map(validation => Promise.resolve(validation()));
|
|
return Promise.all(validations).then(values => {
|
|
let isValid = values.every(value => value === true);
|
|
if (this._valid !== isValid) {
|
|
this._valid = isValid;
|
|
this.fireEvent({
|
|
eventType: ComponentEventType.validityChanged,
|
|
args: this._valid
|
|
});
|
|
}
|
|
return isValid;
|
|
});
|
|
}
|
|
|
|
public focus(): void {
|
|
// Default is to just focus on the native element, components should override this if they
|
|
// want their own behavior (such as focusing a particular child element)
|
|
(<HTMLElement>this._el.nativeElement).focus();
|
|
}
|
|
|
|
public doAction(action: string, ...args: any[]): void {
|
|
// no-op, components should override this if they want to handle actions
|
|
}
|
|
|
|
protected onkeydown(domNode: HTMLElement, listener: (e: StandardKeyboardEvent) => void): void {
|
|
this._register(addDisposableListener(domNode, EventType.KEY_DOWN, (e: KeyboardEvent) => listener(new StandardKeyboardEvent(e))));
|
|
}
|
|
}
|
|
|
|
export abstract class ContainerBase<T, TPropertyBag extends azdata.ComponentProperties = azdata.ComponentProperties> extends ComponentBase<TPropertyBag> {
|
|
protected items: ItemDescriptor<T>[];
|
|
|
|
@ViewChildren(ModelComponentWrapper) protected _componentWrappers: QueryList<ModelComponentWrapper>;
|
|
constructor(
|
|
_changeRef: ChangeDetectorRef,
|
|
_el: ElementRef
|
|
) {
|
|
super(_changeRef, _el);
|
|
this.items = [];
|
|
this._validations.push(() => this.items.every(item => {
|
|
return this.modelStore.getComponent(item.descriptor.id)?.valid || false;
|
|
}));
|
|
}
|
|
|
|
/// IComponent container-related implementation
|
|
public addToContainer(componentDescriptor: IComponentDescriptor, config: any, index?: number): void {
|
|
if (!componentDescriptor) {
|
|
return;
|
|
}
|
|
if (this.items.some(item => item.descriptor.id === componentDescriptor.id && item.descriptor.type === componentDescriptor.type)) {
|
|
return;
|
|
}
|
|
if (index !== undefined && index !== null && index >= 0 && index <= this.items.length) {
|
|
this.items.splice(index, 0, new ItemDescriptor(componentDescriptor, config));
|
|
} else if (!index) {
|
|
this.items.push(new ItemDescriptor(componentDescriptor, config));
|
|
} else {
|
|
throw new Error(nls.localize('invalidIndex', "The index {0} is invalid.", index));
|
|
}
|
|
this.modelStore.eventuallyRunOnComponent(componentDescriptor.id, component => component.registerEventHandler(event => {
|
|
if (event.eventType === ComponentEventType.validityChanged) {
|
|
this.validate();
|
|
}
|
|
}));
|
|
this._changeRef.detectChanges();
|
|
this.onItemsUpdated();
|
|
return;
|
|
}
|
|
|
|
public removeFromContainer(componentDescriptor: IComponentDescriptor): boolean {
|
|
if (!componentDescriptor) {
|
|
return false;
|
|
}
|
|
let index = this.items.findIndex(item => item.descriptor.id === componentDescriptor.id && item.descriptor.type === componentDescriptor.type);
|
|
if (index >= 0) {
|
|
this.items.splice(index, 1);
|
|
this._changeRef.detectChanges();
|
|
this.onItemsUpdated();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public clearContainer(): void {
|
|
this.items = [];
|
|
this.onItemsUpdated();
|
|
this._changeRef.detectChanges();
|
|
this.validate();
|
|
}
|
|
|
|
public setProperties(properties: { [key: string]: any; }): void {
|
|
super.setProperties(properties);
|
|
this.items.forEach(item => {
|
|
let component = this.modelStore.getComponent(item.descriptor.id);
|
|
if (component) {
|
|
component.enabled = this.enabled;
|
|
}
|
|
});
|
|
}
|
|
|
|
public layout(): void {
|
|
if (this._componentWrappers) {
|
|
this._componentWrappers.forEach(wrapper => {
|
|
wrapper.layout();
|
|
});
|
|
}
|
|
super.layout();
|
|
}
|
|
|
|
abstract setLayout(layout: any): void;
|
|
|
|
public setItemLayout(componentDescriptor: IComponentDescriptor, config: any): void {
|
|
if (!componentDescriptor) {
|
|
return;
|
|
}
|
|
const item = this.items.find(item => item.descriptor.id === componentDescriptor.id && item.descriptor.type === componentDescriptor.type);
|
|
if (item) {
|
|
item.config = config;
|
|
this.onItemLayoutUpdated(item);
|
|
this._changeRef.detectChanges();
|
|
} else {
|
|
throw new Error(`Unable to set item layout - unknown item ${componentDescriptor.id}`);
|
|
}
|
|
return;
|
|
}
|
|
|
|
public mergeCss(...styles: azdata.CssStyles[]): azdata.CssStyles {
|
|
const x = styles.reduce((previous, current) => {
|
|
if (current) {
|
|
return Object.assign(previous, current);
|
|
}
|
|
return previous;
|
|
}, {});
|
|
|
|
return x;
|
|
}
|
|
|
|
protected onItemsUpdated(): void {
|
|
}
|
|
|
|
protected onItemLayoutUpdated(item: ItemDescriptor<T>): void {
|
|
}
|
|
}
|