mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-25 09:35:37 -05:00
* allow new workspace location to be editable * fix workspace inputbox not showing up after toggling open workspace radio buttons * add a few tests * cleanup * fix errors * addressing comments * fix filter for windows * add error message if existing workspace file is selected and change picker to be folder only * address comments * fix typos and update tests
189 lines
7.1 KiB
TypeScript
189 lines
7.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 * as azdata from 'azdata';
|
|
import * as vscode from 'vscode';
|
|
import * as path from 'path';
|
|
import * as constants from '../common/constants';
|
|
import { IconPathHelper } from '../common/iconHelper';
|
|
import { directoryExist, fileExist } from '../common/utils';
|
|
|
|
interface Deferred<T> {
|
|
resolve: (result: T | Promise<T>) => void;
|
|
reject: (reason: any) => void;
|
|
}
|
|
|
|
export abstract class DialogBase {
|
|
protected _toDispose: vscode.Disposable[] = [];
|
|
protected _dialogObject: azdata.window.Dialog;
|
|
protected initDialogComplete: Deferred<void> | undefined;
|
|
protected initDialogPromise: Promise<void> = new Promise<void>((resolve, reject) => this.initDialogComplete = { resolve, reject });
|
|
protected workspaceDescriptionFormComponent: azdata.FormComponent | undefined;
|
|
public workspaceInputBox: azdata.InputBoxComponent | undefined;
|
|
protected workspaceInputFormComponent: azdata.FormComponent | undefined;
|
|
|
|
constructor(dialogTitle: string, dialogName: string, dialogWidth: azdata.window.DialogWidth = 600) {
|
|
this._dialogObject = azdata.window.createModelViewDialog(dialogTitle, dialogName, dialogWidth);
|
|
this._dialogObject.okButton.label = constants.OkButtonText;
|
|
this.register(this._dialogObject.cancelButton.onClick(() => this.onCancelButtonClicked()));
|
|
this.register(this._dialogObject.okButton.onClick(() => this.onOkButtonClicked()));
|
|
this._dialogObject.registerCloseValidator(async () => {
|
|
return this.validate();
|
|
});
|
|
}
|
|
|
|
protected abstract initialize(view: azdata.ModelView): Promise<void>;
|
|
|
|
protected async validate(): Promise<boolean> {
|
|
return Promise.resolve(true);
|
|
}
|
|
|
|
public async open(): Promise<void> {
|
|
const tab = azdata.window.createTab('');
|
|
tab.registerContent(async (view: azdata.ModelView) => {
|
|
return this.initialize(view);
|
|
});
|
|
this._dialogObject.content = [tab];
|
|
azdata.window.openDialog(this._dialogObject);
|
|
await this.initDialogPromise;
|
|
}
|
|
|
|
private onCancelButtonClicked(): void {
|
|
this.dispose();
|
|
}
|
|
|
|
private async onOkButtonClicked(): Promise<void> {
|
|
await this.onComplete();
|
|
this.dispose();
|
|
}
|
|
|
|
protected async onComplete(): Promise<void> {
|
|
}
|
|
|
|
protected dispose(): void {
|
|
this._toDispose.forEach(disposable => disposable.dispose());
|
|
}
|
|
|
|
protected register(disposable: vscode.Disposable): void {
|
|
this._toDispose.push(disposable);
|
|
}
|
|
|
|
protected showErrorMessage(message: string): void {
|
|
this._dialogObject.message = {
|
|
text: message,
|
|
level: azdata.window.MessageLevel.Error
|
|
};
|
|
}
|
|
|
|
protected createHorizontalContainer(view: azdata.ModelView, items: azdata.Component[]): azdata.FlexContainer {
|
|
return view.modelBuilder.flexContainer().withItems(items, { CSSStyles: { 'margin-right': '5px', 'margin-bottom': '10px' } }).withLayout({ flexFlow: 'row', alignItems: 'center' }).component();
|
|
}
|
|
|
|
/**
|
|
* Creates container with information on which workspace the project will be added to and where the workspace will be
|
|
* created if no workspace is currently open
|
|
* @param view
|
|
*/
|
|
protected createWorkspaceContainer(view: azdata.ModelView): void {
|
|
const workspaceDescription = view.modelBuilder.text().withProperties<azdata.TextComponentProperties>({
|
|
value: vscode.workspace.workspaceFile ? constants.AddProjectToCurrentWorkspace : constants.NewWorkspaceWillBeCreated,
|
|
CSSStyles: { 'margin-top': '3px', 'margin-bottom': '0px' }
|
|
}).component();
|
|
|
|
this.workspaceInputBox = view.modelBuilder.inputBox().withProperties<azdata.InputBoxProperties>({
|
|
ariaLabel: constants.WorkspaceLocationTitle,
|
|
width: constants.DefaultInputWidth,
|
|
enabled: !vscode.workspace.workspaceFile, // want it editable if no workspace is open
|
|
value: vscode.workspace.workspaceFile?.fsPath ?? '',
|
|
title: vscode.workspace.workspaceFile?.fsPath ?? '' // hovertext for if file path is too long to be seen in textbox
|
|
}).component();
|
|
|
|
const browseFolderButton = view.modelBuilder.button().withProperties<azdata.ButtonProperties>({
|
|
ariaLabel: constants.BrowseButtonText,
|
|
iconPath: IconPathHelper.folder,
|
|
height: '16px',
|
|
width: '18px'
|
|
}).component();
|
|
|
|
this.register(browseFolderButton.onDidClick(async () => {
|
|
const currentFileName = path.parse(this.workspaceInputBox!.value!).base;
|
|
|
|
// let user select folder for workspace file to be created in
|
|
const folderUris = await vscode.window.showOpenDialog({
|
|
canSelectFiles: false,
|
|
canSelectFolders: true,
|
|
canSelectMany: false,
|
|
defaultUri: vscode.Uri.file(path.parse(this.workspaceInputBox!.value!).dir)
|
|
});
|
|
if (!folderUris || folderUris.length === 0) {
|
|
return;
|
|
}
|
|
const selectedFolder = folderUris[0].fsPath;
|
|
|
|
const selectedFile = path.join(selectedFolder, currentFileName);
|
|
this.workspaceInputBox!.value = selectedFile;
|
|
this.workspaceInputBox!.title = selectedFile;
|
|
}));
|
|
|
|
if (vscode.workspace.workspaceFile) {
|
|
this.workspaceInputFormComponent = {
|
|
component: this.workspaceInputBox
|
|
};
|
|
} else {
|
|
// have browse button to help select where the workspace file should be created
|
|
const horizontalContainer = this.createHorizontalContainer(view, [this.workspaceInputBox, browseFolderButton]);
|
|
this.workspaceInputFormComponent = {
|
|
component: horizontalContainer
|
|
};
|
|
}
|
|
|
|
this.workspaceDescriptionFormComponent = {
|
|
title: constants.Workspace,
|
|
component: workspaceDescription,
|
|
required: true
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Update the workspace inputbox based on the passed in location and name if there isn't a workspace currently open
|
|
* @param location
|
|
* @param name
|
|
*/
|
|
protected updateWorkspaceInputbox(location: string, name: string): void {
|
|
if (!vscode.workspace.workspaceFile) {
|
|
const fileLocation = location && name ? path.join(location, `${name}.code-workspace`) : '';
|
|
this.workspaceInputBox!.value = fileLocation;
|
|
this.workspaceInputBox!.title = fileLocation;
|
|
}
|
|
}
|
|
|
|
protected async validateNewWorkspace(sameFolderAsNewProject: boolean): Promise<boolean> {
|
|
// workspace file should end in .code-workspace
|
|
const workspaceValid = this.workspaceInputBox!.value!.endsWith(constants.WorkspaceFileExtension);
|
|
if (!workspaceValid) {
|
|
this.showErrorMessage(constants.WorkspaceFileInvalidError(this.workspaceInputBox!.value!));
|
|
return false;
|
|
}
|
|
|
|
// if the workspace file is not going to be in the same folder as the newly created project, then check that it's a valid folder
|
|
if (!sameFolderAsNewProject) {
|
|
const workspaceParentDirectoryExists = await directoryExist(path.dirname(this.workspaceInputBox!.value!));
|
|
if (!workspaceParentDirectoryExists) {
|
|
this.showErrorMessage(constants.WorkspaceParentDirectoryNotExistError(this.workspaceInputBox!.value!));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// workspace file should not be an existing workspace file
|
|
const workspaceFileExists = await fileExist(this.workspaceInputBox!.value!);
|
|
if (workspaceFileExists) {
|
|
this.showErrorMessage(constants.WorkspaceFileAlreadyExistsError(this.workspaceInputBox!.value!));
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|