mirror of
https://github.com/Azure/cosmos-explorer.git
synced 2025-03-28 12:39:57 +00:00
710 lines
25 KiB
TypeScript
710 lines
25 KiB
TypeScript
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
/* eslint-disable no-console */
|
|
import { FeedOptions, QueryOperationOptions } from "@azure/cosmos";
|
|
import { Platform, configContext } from "ConfigContext";
|
|
import { useDialog } from "Explorer/Controls/Dialog";
|
|
import { QueryCopilotFeedbackModal } from "Explorer/QueryCopilot/Modal/QueryCopilotFeedbackModal";
|
|
import { useCopilotStore } from "Explorer/QueryCopilot/QueryCopilotContext";
|
|
import { QueryCopilotPromptbar } from "Explorer/QueryCopilot/QueryCopilotPromptbar";
|
|
import { OnExecuteQueryClick, QueryDocumentsPerPage } from "Explorer/QueryCopilot/Shared/QueryCopilotClient";
|
|
import { QueryCopilotSidebar } from "Explorer/QueryCopilot/V2/Sidebar/QueryCopilotSidebar";
|
|
import { QueryResultSection } from "Explorer/Tabs/QueryTab/QueryResultSection";
|
|
import { useSelectedNode } from "Explorer/useSelectedNode";
|
|
import { KeyboardAction } from "KeyboardShortcuts";
|
|
import { QueryConstants } from "Shared/Constants";
|
|
import { LocalStorageUtility, StorageKey, getRUThreshold, ruThresholdEnabled } from "Shared/StorageUtility";
|
|
import { Action } from "Shared/Telemetry/TelemetryConstants";
|
|
import { QueryCopilotState, useQueryCopilot } from "hooks/useQueryCopilot";
|
|
import { TabsState, useTabs } from "hooks/useTabs";
|
|
import React, { Fragment } from "react";
|
|
import SplitterLayout from "react-splitter-layout";
|
|
import "react-splitter-layout/lib/index.css";
|
|
import { format } from "react-string-format";
|
|
import QueryCommandIcon from "../../../../images/CopilotCommand.svg";
|
|
import LaunchCopilot from "../../../../images/CopilotTabIcon.svg";
|
|
import DownloadQueryIcon from "../../../../images/DownloadQuery.svg";
|
|
import CancelQueryIcon from "../../../../images/Entity_cancel.svg";
|
|
import ExecuteQueryIcon from "../../../../images/ExecuteQuery.svg";
|
|
import SaveQueryIcon from "../../../../images/save-cosmos.svg";
|
|
import { NormalizedEventKey } from "../../../Common/Constants";
|
|
import { getErrorMessage } from "../../../Common/ErrorHandlingUtils";
|
|
import * as HeadersUtility from "../../../Common/HeadersUtility";
|
|
import { MinimalQueryIterator } from "../../../Common/IteratorUtilities";
|
|
import { queryIterator } from "../../../Common/MongoProxyClient";
|
|
import { queryDocuments } from "../../../Common/dataAccess/queryDocuments";
|
|
import { queryDocumentsPage } from "../../../Common/dataAccess/queryDocumentsPage";
|
|
import * as DataModels from "../../../Contracts/DataModels";
|
|
import * as ViewModels from "../../../Contracts/ViewModels";
|
|
import * as StringUtility from "../../../Shared/StringUtility";
|
|
import * as TelemetryProcessor from "../../../Shared/Telemetry/TelemetryProcessor";
|
|
import { userContext } from "../../../UserContext";
|
|
import * as QueryUtils from "../../../Utils/QueryUtils";
|
|
import { useSidePanel } from "../../../hooks/useSidePanel";
|
|
import { CommandButtonComponentProps } from "../../Controls/CommandButton/CommandButtonComponent";
|
|
import { EditorReact } from "../../Controls/Editor/EditorReact";
|
|
import Explorer from "../../Explorer";
|
|
import { useCommandBar } from "../../Menus/CommandBar/CommandBarComponentAdapter";
|
|
import { BrowseQueriesPane } from "../../Panes/BrowseQueriesPane/BrowseQueriesPane";
|
|
import { SaveQueryPane } from "../../Panes/SaveQueryPane/SaveQueryPane";
|
|
import TabsBase from "../TabsBase";
|
|
import "./QueryTabComponent.less";
|
|
|
|
enum ToggleState {
|
|
Result,
|
|
QueryMetrics,
|
|
}
|
|
|
|
export interface IDocument {
|
|
metric: string;
|
|
value: string;
|
|
toolTip: string;
|
|
}
|
|
|
|
export interface ITabAccessor {
|
|
onTabClickEvent: () => void;
|
|
onSaveClickEvent: () => string;
|
|
onCloseClickEvent: (isClicked: boolean) => void;
|
|
}
|
|
|
|
export interface Button {
|
|
visible: boolean;
|
|
enabled: boolean;
|
|
isSelected?: boolean;
|
|
}
|
|
|
|
export interface IQueryTabComponentProps {
|
|
collection: ViewModels.CollectionBase;
|
|
isExecutionError: boolean;
|
|
tabId: string;
|
|
tabsBaseInstance: TabsBase;
|
|
queryText: string;
|
|
partitionKey: DataModels.PartitionKey;
|
|
container: Explorer;
|
|
activeTab?: TabsBase;
|
|
onTabAccessor: (instance: ITabAccessor) => void;
|
|
isPreferredApiMongoDB?: boolean;
|
|
monacoEditorSetting?: string;
|
|
viewModelcollection?: ViewModels.Collection;
|
|
copilotEnabled?: boolean;
|
|
isSampleCopilotActive?: boolean;
|
|
copilotStore?: Partial<QueryCopilotState>;
|
|
}
|
|
|
|
interface IQueryTabStates {
|
|
toggleState: ToggleState;
|
|
sqlQueryEditorContent: string;
|
|
selectedContent: string;
|
|
queryResults: ViewModels.QueryResults;
|
|
error: string;
|
|
isExecutionError: boolean;
|
|
isExecuting: boolean;
|
|
showCopilotSidebar: boolean;
|
|
queryCopilotGeneratedQuery: string;
|
|
cancelQueryTimeoutID: NodeJS.Timeout;
|
|
copilotActive: boolean;
|
|
currentTabActive: boolean;
|
|
}
|
|
|
|
export const QueryTabFunctionComponent = (props: IQueryTabComponentProps): any => {
|
|
const copilotStore = useCopilotStore();
|
|
const isSampleCopilotActive = useSelectedNode.getState().isQueryCopilotCollectionSelected();
|
|
const queryTabProps = {
|
|
...props,
|
|
copilotEnabled:
|
|
useQueryCopilot().copilotEnabled &&
|
|
(useQueryCopilot().copilotUserDBEnabled || (isSampleCopilotActive && !!userContext.sampleDataConnectionInfo)),
|
|
isSampleCopilotActive: isSampleCopilotActive,
|
|
copilotStore: copilotStore,
|
|
};
|
|
return <QueryTabComponent {...queryTabProps}></QueryTabComponent>;
|
|
};
|
|
|
|
export default class QueryTabComponent extends React.Component<IQueryTabComponentProps, IQueryTabStates> {
|
|
public queryEditorId: string;
|
|
public executeQueryButton: Button;
|
|
public saveQueryButton: Button;
|
|
public launchCopilotButton: Button;
|
|
public splitterId: string;
|
|
public isPreferredApiMongoDB: boolean;
|
|
public isCloseClicked: boolean;
|
|
public isCopilotTabActive: boolean;
|
|
private _iterator: MinimalQueryIterator;
|
|
private queryAbortController: AbortController;
|
|
|
|
constructor(props: IQueryTabComponentProps) {
|
|
super(props);
|
|
|
|
this.state = {
|
|
toggleState: ToggleState.Result,
|
|
sqlQueryEditorContent: props.isPreferredApiMongoDB ? "{}" : props.queryText || "SELECT * FROM c",
|
|
selectedContent: "",
|
|
queryResults: undefined,
|
|
error: "",
|
|
isExecutionError: this.props.isExecutionError,
|
|
isExecuting: false,
|
|
showCopilotSidebar: useQueryCopilot.getState().showCopilotSidebar,
|
|
queryCopilotGeneratedQuery: useQueryCopilot.getState().query,
|
|
cancelQueryTimeoutID: undefined,
|
|
copilotActive: this._queryCopilotActive(),
|
|
currentTabActive: true,
|
|
};
|
|
this.isCloseClicked = false;
|
|
this.splitterId = this.props.tabId + "_splitter";
|
|
this.queryEditorId = `queryeditor${this.props.tabId}`;
|
|
this.isPreferredApiMongoDB = this.props.isPreferredApiMongoDB;
|
|
this.isCopilotTabActive = userContext.features.copilotVersion === "v3.0";
|
|
this.executeQueryButton = {
|
|
enabled: !!this.state.sqlQueryEditorContent && this.state.sqlQueryEditorContent.length > 0,
|
|
visible: true,
|
|
};
|
|
|
|
const isSaveQueryBtnEnabled = userContext.apiType === "SQL" || userContext.apiType === "Gremlin";
|
|
this.saveQueryButton = {
|
|
enabled: isSaveQueryBtnEnabled,
|
|
visible: isSaveQueryBtnEnabled,
|
|
};
|
|
|
|
this.launchCopilotButton = {
|
|
enabled: userContext.apiType === "SQL" && true,
|
|
visible: userContext.apiType === "SQL" && true,
|
|
};
|
|
|
|
this.props.tabsBaseInstance.updateNavbarWithTabsButtons();
|
|
props.onTabAccessor({
|
|
onTabClickEvent: this.onTabClick.bind(this),
|
|
onSaveClickEvent: this.getCurrentEditorQuery.bind(this),
|
|
onCloseClickEvent: this.onCloseClick.bind(this),
|
|
});
|
|
}
|
|
|
|
private _queryCopilotActive(): boolean {
|
|
if (this.props.copilotEnabled) {
|
|
const cachedCopilotToggleStatus: string = localStorage.getItem(
|
|
`${userContext.databaseAccount?.id}-queryCopilotToggleStatus`,
|
|
);
|
|
const copilotInitialActive: boolean = cachedCopilotToggleStatus
|
|
? StringUtility.toBoolean(cachedCopilotToggleStatus)
|
|
: true;
|
|
return copilotInitialActive;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public onCloseClick(isClicked: boolean): void {
|
|
this.isCloseClicked = isClicked;
|
|
if (useQueryCopilot.getState().wasCopilotUsed && this.isCopilotTabActive) {
|
|
useQueryCopilot.getState().resetQueryCopilotStates();
|
|
}
|
|
}
|
|
|
|
public getCurrentEditorQuery(): string {
|
|
return this.state.sqlQueryEditorContent;
|
|
}
|
|
|
|
public onTabClick(): void {
|
|
if (!this.isCloseClicked) {
|
|
useCommandBar.getState().setContextButtons(this.getTabsButtons());
|
|
} else {
|
|
this.isCloseClicked = false;
|
|
}
|
|
}
|
|
|
|
public onExecuteQueryClick = async (): Promise<void> => {
|
|
this._iterator = undefined;
|
|
setTimeout(async () => {
|
|
await this._executeQueryDocumentsPage(0);
|
|
}, 100);
|
|
if (this.state.copilotActive) {
|
|
const query = this.state.sqlQueryEditorContent.split("\r\n")?.pop();
|
|
const isqueryEdited = this.props.copilotStore.generatedQuery && this.props.copilotStore.generatedQuery !== query;
|
|
if (isqueryEdited) {
|
|
TelemetryProcessor.traceMark(Action.QueryEdited, {
|
|
databaseName: this.props.collection.databaseId,
|
|
collectionId: this.props.collection.id(),
|
|
});
|
|
}
|
|
}
|
|
};
|
|
|
|
public onDownloadQueryClick = (): void => {
|
|
const text = this.getCurrentEditorQuery();
|
|
const queryFile = new File([text], `SavedQuery.txt`, { type: "text/plain" });
|
|
|
|
// It appears the most consistent to download a file from a blob is to create an anchor element and simulate clicking it
|
|
const blobUrl = URL.createObjectURL(queryFile);
|
|
const anchor = document.createElement("a");
|
|
anchor.href = blobUrl;
|
|
anchor.download = queryFile.name;
|
|
document.body.appendChild(anchor); // Must put the anchor in the document.
|
|
anchor.click();
|
|
document.body.removeChild(anchor); // Clean up the anchor.
|
|
};
|
|
|
|
public onSaveQueryClick = (): void => {
|
|
useSidePanel.getState().openSidePanel("Save Query", <SaveQueryPane explorer={this.props.collection.container} />);
|
|
};
|
|
|
|
public launchQueryCopilotChat = (): void => {
|
|
useQueryCopilot.getState().setShowCopilotSidebar(!useQueryCopilot.getState().showCopilotSidebar);
|
|
};
|
|
|
|
public onSavedQueriesClick = (): void => {
|
|
useSidePanel
|
|
.getState()
|
|
.openSidePanel("Open Saved Queries", <BrowseQueriesPane explorer={this.props.collection.container} />);
|
|
};
|
|
|
|
public toggleResult(): void {
|
|
this.setState({
|
|
toggleState: ToggleState.Result,
|
|
});
|
|
}
|
|
|
|
public toggleMetrics(): void {
|
|
this.setState({
|
|
toggleState: ToggleState.QueryMetrics,
|
|
});
|
|
}
|
|
|
|
public handleCopilotKeyDown = (event: KeyboardEvent): void => {
|
|
if (this.isCopilotTabActive && event.altKey && event.key === "c") {
|
|
this.launchQueryCopilotChat();
|
|
}
|
|
};
|
|
|
|
public onToggleKeyDown = (event: React.KeyboardEvent<HTMLDivElement>): boolean => {
|
|
if (event.key === NormalizedEventKey.LeftArrow) {
|
|
this.toggleResult();
|
|
event.stopPropagation();
|
|
return false;
|
|
} else if (event.key === NormalizedEventKey.RightArrow) {
|
|
this.toggleMetrics();
|
|
event.stopPropagation();
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
public togglesOnFocus(): void {
|
|
const focusElement = document.getElementById("execute-query-toggles");
|
|
focusElement && focusElement.focus();
|
|
}
|
|
|
|
private async _executeQueryDocumentsPage(firstItemIndex: number): Promise<void> {
|
|
this.queryAbortController = new AbortController();
|
|
if (this._iterator === undefined) {
|
|
this._iterator = this.props.isPreferredApiMongoDB
|
|
? queryIterator(
|
|
this.props.collection.databaseId,
|
|
this.props.viewModelcollection,
|
|
this.state.selectedContent || this.state.sqlQueryEditorContent,
|
|
)
|
|
: queryDocuments(
|
|
this.props.collection.databaseId,
|
|
this.props.collection.id(),
|
|
this.state.selectedContent || this.state.sqlQueryEditorContent,
|
|
{
|
|
enableCrossPartitionQuery: HeadersUtility.shouldEnableCrossPartitionKey(),
|
|
abortSignal: this.queryAbortController.signal,
|
|
} as unknown as FeedOptions,
|
|
);
|
|
}
|
|
|
|
await this._queryDocumentsPage(firstItemIndex);
|
|
}
|
|
|
|
private async _queryDocumentsPage(firstItemIndex: number): Promise<void> {
|
|
this.props.tabsBaseInstance.isExecutionError(false);
|
|
this.setState({
|
|
isExecutionError: false,
|
|
});
|
|
|
|
let queryOperationOptions: QueryOperationOptions;
|
|
if (userContext.apiType === "SQL" && ruThresholdEnabled()) {
|
|
const ruThreshold: number = getRUThreshold();
|
|
queryOperationOptions = {
|
|
ruCapPerOperation: ruThreshold,
|
|
} as QueryOperationOptions;
|
|
}
|
|
const queryDocuments = async (firstItemIndex: number) =>
|
|
await queryDocumentsPage(
|
|
this.props.collection && this.props.collection.id(),
|
|
this._iterator,
|
|
firstItemIndex,
|
|
queryOperationOptions,
|
|
);
|
|
this.props.tabsBaseInstance.isExecuting(true);
|
|
this.setState({
|
|
isExecuting: true,
|
|
});
|
|
let automaticallyCancelQueryAfterTimeout: boolean;
|
|
if (this.queryTimeoutEnabled()) {
|
|
const queryTimeout: number = LocalStorageUtility.getEntryNumber(StorageKey.QueryTimeout);
|
|
automaticallyCancelQueryAfterTimeout = LocalStorageUtility.getEntryBoolean(
|
|
StorageKey.AutomaticallyCancelQueryAfterTimeout,
|
|
);
|
|
const cancelQueryTimeoutID: NodeJS.Timeout = setTimeout(() => {
|
|
if (this.state.isExecuting) {
|
|
if (automaticallyCancelQueryAfterTimeout) {
|
|
this.queryAbortController.abort();
|
|
} else {
|
|
useDialog
|
|
.getState()
|
|
.showOkCancelModalDialog(
|
|
QueryConstants.CancelQueryTitle,
|
|
format(QueryConstants.CancelQuerySubTextTemplate, QueryConstants.CancelQueryTimeoutThresholdReached),
|
|
"Yes",
|
|
() => this.queryAbortController.abort(),
|
|
"No",
|
|
undefined,
|
|
);
|
|
}
|
|
}
|
|
}, queryTimeout);
|
|
this.setState({
|
|
cancelQueryTimeoutID,
|
|
});
|
|
}
|
|
useCommandBar.getState().setContextButtons(this.getTabsButtons());
|
|
|
|
try {
|
|
const queryResults: ViewModels.QueryResults = await QueryUtils.queryPagesUntilContentPresent(
|
|
firstItemIndex,
|
|
queryDocuments,
|
|
);
|
|
this.setState({ queryResults, error: "" });
|
|
} catch (error) {
|
|
this.props.tabsBaseInstance.isExecutionError(true);
|
|
this.setState({
|
|
isExecutionError: true,
|
|
});
|
|
const errorMessage = getErrorMessage(error);
|
|
this.setState({
|
|
error: errorMessage,
|
|
});
|
|
|
|
document.getElementById("error-display").focus();
|
|
} finally {
|
|
this.props.tabsBaseInstance.isExecuting(false);
|
|
this.setState({
|
|
isExecuting: false,
|
|
cancelQueryTimeoutID: undefined,
|
|
});
|
|
if (this.queryTimeoutEnabled()) {
|
|
clearTimeout(this.state.cancelQueryTimeoutID);
|
|
if (!automaticallyCancelQueryAfterTimeout) {
|
|
useDialog.getState().closeDialog();
|
|
}
|
|
}
|
|
this.togglesOnFocus();
|
|
useCommandBar.getState().setContextButtons(this.getTabsButtons());
|
|
}
|
|
}
|
|
|
|
protected getTabsButtons(): CommandButtonComponentProps[] {
|
|
const buttons: CommandButtonComponentProps[] = [];
|
|
if (this.executeQueryButton.visible) {
|
|
const label = this.state.selectedContent?.length > 0 ? "Execute Selection" : "Execute Query";
|
|
buttons.push({
|
|
iconSrc: ExecuteQueryIcon,
|
|
iconAlt: label,
|
|
keyboardAction: KeyboardAction.EXECUTE_ITEM,
|
|
onCommandClick: this.props.isSampleCopilotActive
|
|
? () => OnExecuteQueryClick(this.props.copilotStore)
|
|
: this.onExecuteQueryClick,
|
|
commandButtonLabel: label,
|
|
ariaLabel: label,
|
|
hasPopup: false,
|
|
disabled: !this.executeQueryButton.enabled,
|
|
});
|
|
}
|
|
|
|
if (this.saveQueryButton.visible) {
|
|
if (configContext.platform !== Platform.Fabric) {
|
|
const label = "Save Query";
|
|
buttons.push({
|
|
iconSrc: SaveQueryIcon,
|
|
iconAlt: label,
|
|
keyboardAction: KeyboardAction.SAVE_ITEM,
|
|
onCommandClick: this.onSaveQueryClick,
|
|
commandButtonLabel: label,
|
|
ariaLabel: label,
|
|
hasPopup: false,
|
|
disabled: !this.saveQueryButton.enabled,
|
|
});
|
|
}
|
|
|
|
buttons.push({
|
|
iconSrc: DownloadQueryIcon,
|
|
iconAlt: "Download Query",
|
|
keyboardAction: KeyboardAction.DOWNLOAD_ITEM,
|
|
onCommandClick: this.onDownloadQueryClick,
|
|
commandButtonLabel: "Download Query",
|
|
ariaLabel: "Download Query",
|
|
hasPopup: false,
|
|
disabled: !this.saveQueryButton.enabled,
|
|
});
|
|
}
|
|
|
|
if (this.launchCopilotButton.visible && this.isCopilotTabActive) {
|
|
const mainButtonLabel = "Launch Copilot";
|
|
const chatPaneLabel = "Open Copilot in chat pane (ALT+C)";
|
|
const copilotSettingLabel = "Copilot settings";
|
|
|
|
const openCopilotChatButton: CommandButtonComponentProps = {
|
|
iconAlt: chatPaneLabel,
|
|
onCommandClick: this.launchQueryCopilotChat,
|
|
commandButtonLabel: chatPaneLabel,
|
|
ariaLabel: chatPaneLabel,
|
|
hasPopup: false,
|
|
};
|
|
|
|
const copilotSettingsButton: CommandButtonComponentProps = {
|
|
iconAlt: copilotSettingLabel,
|
|
onCommandClick: () => undefined,
|
|
commandButtonLabel: copilotSettingLabel,
|
|
ariaLabel: copilotSettingLabel,
|
|
hasPopup: false,
|
|
};
|
|
|
|
const launchCopilotButton: CommandButtonComponentProps = {
|
|
iconSrc: LaunchCopilot,
|
|
iconAlt: mainButtonLabel,
|
|
onCommandClick: this.launchQueryCopilotChat,
|
|
commandButtonLabel: mainButtonLabel,
|
|
ariaLabel: mainButtonLabel,
|
|
hasPopup: false,
|
|
children: [openCopilotChatButton, copilotSettingsButton],
|
|
};
|
|
buttons.push(launchCopilotButton);
|
|
}
|
|
|
|
if (this.props.copilotEnabled) {
|
|
const toggleCopilotButton: CommandButtonComponentProps = {
|
|
iconSrc: QueryCommandIcon,
|
|
iconAlt: "Copilot",
|
|
keyboardAction: KeyboardAction.TOGGLE_COPILOT,
|
|
onCommandClick: () => {
|
|
this._toggleCopilot(!this.state.copilotActive);
|
|
},
|
|
commandButtonLabel: this.state.copilotActive ? "Disable Copilot" : "Enable Copilot",
|
|
ariaLabel: this.state.copilotActive ? "Disable Copilot" : "Enable Copilot",
|
|
hasPopup: false,
|
|
};
|
|
buttons.push(toggleCopilotButton);
|
|
}
|
|
|
|
if (!this.props.isPreferredApiMongoDB && this.state.isExecuting) {
|
|
const label = "Cancel query";
|
|
buttons.push({
|
|
iconSrc: CancelQueryIcon,
|
|
iconAlt: label,
|
|
keyboardAction: KeyboardAction.CANCEL_OR_DISCARD,
|
|
onCommandClick: () => this.queryAbortController.abort(),
|
|
commandButtonLabel: label,
|
|
ariaLabel: label,
|
|
hasPopup: false,
|
|
});
|
|
}
|
|
|
|
return buttons;
|
|
}
|
|
|
|
private _toggleCopilot = (active: boolean) => {
|
|
this.setState({ copilotActive: active });
|
|
useQueryCopilot.getState().setCopilotEnabledforExecution(active);
|
|
localStorage.setItem(`${userContext.databaseAccount?.id}-queryCopilotToggleStatus`, active.toString());
|
|
|
|
TelemetryProcessor.traceSuccess(active ? Action.ActivateQueryCopilot : Action.DeactivateQueryCopilot, {
|
|
databaseName: this.props.collection.databaseId,
|
|
collectionId: this.props.collection.id(),
|
|
});
|
|
};
|
|
|
|
componentDidUpdate = (_prevProps: IQueryTabComponentProps, prevState: IQueryTabStates): void => {
|
|
if (prevState.copilotActive !== this.state.copilotActive) {
|
|
useCommandBar.getState().setContextButtons(this.getTabsButtons());
|
|
}
|
|
};
|
|
|
|
public onChangeContent(newContent: string): void {
|
|
// The copilot store's active query takes precedence over the local state,
|
|
// and we can't update both states in a single operation.
|
|
// So, we update the copilot store's state first, then update the local state.
|
|
if (this.state.copilotActive) {
|
|
this.props.copilotStore?.setQuery(newContent);
|
|
}
|
|
this.setState({
|
|
sqlQueryEditorContent: newContent,
|
|
queryCopilotGeneratedQuery: "",
|
|
});
|
|
if (this.isPreferredApiMongoDB) {
|
|
if (newContent.length > 0) {
|
|
this.executeQueryButton = {
|
|
enabled: true,
|
|
visible: true,
|
|
};
|
|
} else {
|
|
this.executeQueryButton = {
|
|
enabled: false,
|
|
visible: true,
|
|
};
|
|
}
|
|
}
|
|
|
|
this.saveQueryButton.enabled = newContent.length > 0;
|
|
|
|
useCommandBar.getState().setContextButtons(this.getTabsButtons());
|
|
}
|
|
|
|
public onSelectedContent(selectedContent: string): void {
|
|
if (selectedContent.trim().length > 0) {
|
|
this.setState({
|
|
selectedContent,
|
|
});
|
|
} else {
|
|
this.setState({
|
|
selectedContent: "",
|
|
});
|
|
}
|
|
|
|
if (this.isCopilotTabActive) {
|
|
selectedContent.trim().length > 0
|
|
? useQueryCopilot.getState().setSelectedQuery(selectedContent)
|
|
: useQueryCopilot.getState().setSelectedQuery("");
|
|
}
|
|
|
|
if (this.state.copilotActive) {
|
|
this.props.copilotStore?.setSelectedQuery(selectedContent);
|
|
}
|
|
|
|
useCommandBar.getState().setContextButtons(this.getTabsButtons());
|
|
}
|
|
|
|
public getEditorContent(): string {
|
|
if (this.isCopilotTabActive && this.state.queryCopilotGeneratedQuery) {
|
|
return this.state.queryCopilotGeneratedQuery;
|
|
}
|
|
|
|
if (this.state.copilotActive) {
|
|
return this.props.copilotStore?.query;
|
|
}
|
|
|
|
return this.state.sqlQueryEditorContent;
|
|
}
|
|
|
|
private queryTimeoutEnabled(): boolean {
|
|
return !this.isPreferredApiMongoDB && LocalStorageUtility.getEntryBoolean(StorageKey.QueryTimeoutEnabled);
|
|
}
|
|
|
|
private unsubscribeCopilotSidebar: () => void;
|
|
|
|
componentDidMount(): void {
|
|
useTabs.subscribe((state: TabsState) => {
|
|
if (this.state.currentTabActive && state.activeTab?.tabId !== this.props.tabId) {
|
|
this.setState({
|
|
currentTabActive: false,
|
|
});
|
|
} else if (!this.state.currentTabActive && state.activeTab?.tabId === this.props.tabId) {
|
|
this.setState({
|
|
currentTabActive: true,
|
|
});
|
|
}
|
|
});
|
|
|
|
useCommandBar.getState().setContextButtons(this.getTabsButtons());
|
|
document.addEventListener("keydown", this.handleCopilotKeyDown);
|
|
}
|
|
|
|
componentWillUnmount(): void {
|
|
document.removeEventListener("keydown", this.handleCopilotKeyDown);
|
|
}
|
|
|
|
private getEditorAndQueryResult(): JSX.Element {
|
|
return (
|
|
<Fragment>
|
|
<div className="tab-pane" id={this.props.tabId} role="tabpanel">
|
|
{this.props.copilotEnabled && this.state.currentTabActive && this.state.copilotActive && (
|
|
<QueryCopilotPromptbar
|
|
explorer={this.props.collection.container}
|
|
toggleCopilot={this._toggleCopilot}
|
|
databaseId={this.props.collection.databaseId}
|
|
containerId={this.props.collection.id()}
|
|
></QueryCopilotPromptbar>
|
|
)}
|
|
<div className="tabPaneContentContainer">
|
|
<SplitterLayout vertical={true} primaryIndex={0} primaryMinSize={100} secondaryMinSize={200}>
|
|
<Fragment>
|
|
<div className="queryEditor" style={{ height: "100%" }}>
|
|
<EditorReact
|
|
language={"sql"}
|
|
content={this.getEditorContent()}
|
|
isReadOnly={false}
|
|
wordWrap={"on"}
|
|
ariaLabel={"Editing Query"}
|
|
lineNumbers={"on"}
|
|
onContentChanged={(newContent: string) => this.onChangeContent(newContent)}
|
|
onContentSelected={(selectedContent: string) => this.onSelectedContent(selectedContent)}
|
|
/>
|
|
</div>
|
|
</Fragment>
|
|
{this.props.isSampleCopilotActive ? (
|
|
<QueryResultSection
|
|
isMongoDB={this.props.isPreferredApiMongoDB}
|
|
queryEditorContent={this.state.sqlQueryEditorContent}
|
|
error={this.props.copilotStore?.errorMessage}
|
|
queryResults={this.props.copilotStore?.queryResults}
|
|
isExecuting={this.props.copilotStore?.isExecuting}
|
|
executeQueryDocumentsPage={(firstItemIndex: number) =>
|
|
QueryDocumentsPerPage(
|
|
firstItemIndex,
|
|
this.props.copilotStore.queryIterator,
|
|
this.props.copilotStore,
|
|
)
|
|
}
|
|
/>
|
|
) : (
|
|
<QueryResultSection
|
|
isMongoDB={this.props.isPreferredApiMongoDB}
|
|
queryEditorContent={this.state.sqlQueryEditorContent}
|
|
error={this.state.error}
|
|
queryResults={this.state.queryResults}
|
|
isExecuting={this.state.isExecuting}
|
|
executeQueryDocumentsPage={(firstItemIndex: number) =>
|
|
this._executeQueryDocumentsPage(firstItemIndex)
|
|
}
|
|
/>
|
|
)}
|
|
</SplitterLayout>
|
|
</div>
|
|
</div>
|
|
{this.props.copilotEnabled && this.props.copilotStore?.showFeedbackModal && (
|
|
<QueryCopilotFeedbackModal
|
|
explorer={this.props.collection.container}
|
|
databaseId={this.props.collection.databaseId}
|
|
containerId={this.props.collection.id()}
|
|
mode={this.props.isSampleCopilotActive ? "Sample" : "User"}
|
|
/>
|
|
)}
|
|
</Fragment>
|
|
);
|
|
}
|
|
|
|
render(): JSX.Element {
|
|
const shouldScaleElements = this.state.showCopilotSidebar && this.isCopilotTabActive;
|
|
return (
|
|
<div style={{ display: "flex", flexDirection: "row", height: "100%" }}>
|
|
<div style={{ width: shouldScaleElements ? "70%" : "100%", height: "100%" }}>
|
|
{this.getEditorAndQueryResult()}
|
|
</div>
|
|
{shouldScaleElements && (
|
|
<div style={{ width: "30%", height: "100%" }}>
|
|
<QueryCopilotSidebar explorer={this.props.collection.container} />
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
}
|