diff --git a/schema/main.json b/schema/main.json index e21a3de7..e6c3de7f 100644 --- a/schema/main.json +++ b/schema/main.json @@ -36,6 +36,7 @@ "$ref": "#/definitions/variableFilters", "default": { "xpython": [ + "debugpy", "display", "get_ipython", "ptvsd", diff --git a/src/panels/variables/grid.ts b/src/panels/variables/grid.ts index f32c9c67..f158f7a5 100644 --- a/src/panels/variables/grid.ts +++ b/src/panels/variables/grid.ts @@ -35,12 +35,12 @@ export class VariablesBodyGrid extends Panel { */ constructor(options: VariablesBodyGrid.IOptions) { super(); - const { model, commands, scopes, themeManager } = options; + const { model, commands, themeManager, scopes } = options; this._grid = new Grid({ commands, themeManager }); this._grid.addClass('jp-DebuggerVariables-grid'); this._model = model; this._model.changed.connect((model: VariablesModel): void => { - this._grid.dataModel.setData(model.scopes); + this._update(); }, this); this._grid.dataModel.setData(scopes ?? []); this.addWidget(this._grid); @@ -54,7 +54,24 @@ export class VariablesBodyGrid extends Panel { */ set filter(filter: Set) { (this._grid.dataModel as GridModel).filter = filter; - this._grid.dataModel.setData(this._model.scopes); + this._update(); + } + + /** + * Set the current scope. + * + * @param scope The current scope for the variables. + */ + set scope(scope: string) { + (this._grid.dataModel as GridModel).scope = scope; + this._update(); + } + + /** + * Update the underlying data model + */ + private _update(): void { + this._grid.dataModel.setData(this._model.scopes ?? []); } private _grid: Grid; @@ -139,6 +156,16 @@ class Grid extends Panel { this.update(); } + /** + * Set the scope for the variables data model. + * + * @param scope The scopes for the variables + */ + set scope(scope: string) { + (this._grid.dataModel as GridModel).scope = scope; + this.update(); + } + /** * Get the data model for the data grid. */ @@ -199,6 +226,20 @@ class GridModel extends DataModel { this._filter = filter; } + /** + * Get the current scope for the variables. + */ + get scope(): string { + return this._scope; + } + + /** + * Set the variable scope + */ + set scope(scope: string) { + this._scope = scope; + } + /** * Get the row count for a particular region in the data grid. * @@ -268,22 +309,22 @@ class GridModel extends DataModel { type: 'model-reset', region: 'body' }); - scopes.forEach(scope => { - const filtered = scope.variables.filter( - variable => !this._filter.has(variable.evaluateName) - ); - filtered.forEach((variable, index) => { - this._data.name[index] = variable.evaluateName; - this._data.type[index] = variable.type; - this._data.value[index] = variable.value; - this._data.variablesReference[index] = variable.variablesReference; - }); - this.emitChanged({ - type: 'rows-inserted', - region: 'body', - index: 1, - span: filtered.length - }); + const scope = scopes.find(scope => scope.name === this._scope) ?? scopes[0]; + const variables = scope?.variables ?? []; + const filtered = variables.filter( + variable => variable.name && !this._filter.has(variable.name) + ); + filtered.forEach((variable, index) => { + this._data.name[index] = variable.name; + this._data.type[index] = variable.type ?? ''; + this._data.value[index] = variable.value; + this._data.variablesReference[index] = variable.variablesReference; + }); + this.emitChanged({ + type: 'rows-inserted', + region: 'body', + index: 1, + span: filtered.length }); } @@ -300,6 +341,7 @@ class GridModel extends DataModel { } private _filter = new Set(); + private _scope = ''; private _data: { name: string[]; type: string[]; diff --git a/src/panels/variables/index.ts b/src/panels/variables/index.ts index ca63628b..b4a90691 100644 --- a/src/panels/variables/index.ts +++ b/src/panels/variables/index.ts @@ -13,6 +13,8 @@ import { VariablesBodyGrid } from './grid'; import { VariablesHeader } from './header'; +import { ScopeSwitcher } from './scope'; + import { VariablesBodyTree } from './tree'; /** @@ -34,7 +36,7 @@ export class Variables extends Panel { this._table = new VariablesBodyGrid({ model, commands, themeManager }); this._table.hide(); - const onClick = (): void => { + const onViewChange = (): void => { if (this._table.isHidden) { this._tree.hide(); this._table.show(); @@ -47,11 +49,20 @@ export class Variables extends Panel { this.update(); }; + this._header.toolbar.addItem( + 'scope-switcher', + new ScopeSwitcher({ + model, + tree: this._tree, + grid: this._table + }) + ); + this._header.toolbar.addItem( 'view-VariableSwitch', new ToolbarButton({ iconClass: 'jp-ToggleSwitch', - onClick, + onClick: onViewChange, tooltip: 'Table / Tree View' }) ); @@ -112,7 +123,7 @@ export const convertType = (variable: IDebugger.IVariable): string | number => { case 'str': return value.slice(1, value.length - 1); default: - return type; + return type ?? value; } }; diff --git a/src/panels/variables/scope.tsx b/src/panels/variables/scope.tsx new file mode 100644 index 00000000..bb1933dc --- /dev/null +++ b/src/panels/variables/scope.tsx @@ -0,0 +1,118 @@ +import { ReactWidget, UseSignal } from '@jupyterlab/apputils'; + +import { HTMLSelect } from '@jupyterlab/ui-components'; + +import React, { useState } from 'react'; + +import { IDebugger } from '../../tokens'; + +import { VariablesBodyGrid } from './grid'; + +import { VariablesBodyTree } from './tree'; + +/** + * A React component to handle scope changes. + * + * @param {object} props The component props. + * @param props.model The variables model. + * @param props.tree The variables tree widget. + * @param props.grid The variables grid widget. + */ +const ScopeSwitcherComponent = ({ + model, + tree, + grid +}: { + model: IDebugger.Model.IVariables; + tree: VariablesBodyTree; + grid: VariablesBodyGrid; +}): JSX.Element => { + const [value, setValue] = useState('-'); + const scopes = model.scopes; + + const onChange = (event: React.ChangeEvent): void => { + const value = event.target.value; + setValue(value); + tree.scope = value; + grid.scope = value; + }; + + return ( + + {scopes.map(scope => ( + + ))} + + ); +}; + +/** + * A widget to switch between scopes. + */ +export class ScopeSwitcher extends ReactWidget { + /** + * Instantiate a new scope switcher. + * + * @param options The instantiation options for a ScopeSwitcher + */ + constructor(options: ScopeSwitcher.IOptions) { + super(); + const { model, tree, grid } = options; + this._model = model; + this._tree = tree; + this._grid = grid; + } + + /** + * Render the scope switcher. + */ + render(): JSX.Element { + return ( + + {(): JSX.Element => ( + + )} + + ); + } + + private _model: IDebugger.Model.IVariables; + private _tree: VariablesBodyTree; + private _grid: VariablesBodyGrid; +} + +/** + * A namespace for ScopeSwitcher statics + */ +export namespace ScopeSwitcher { + /** + * The ScopeSwitcher instantiation options. + */ + export interface IOptions { + /** + * The variables model. + */ + model: IDebugger.Model.IVariables; + + /** + * The variables tree viewer. + */ + tree: VariablesBodyTree; + + /** + * The variables table viewer. + */ + grid: VariablesBodyGrid; + } +} diff --git a/src/panels/variables/tree.tsx b/src/panels/variables/tree.tsx index 327d856a..d4546b39 100644 --- a/src/panels/variables/tree.tsx +++ b/src/panels/variables/tree.tsx @@ -11,10 +11,10 @@ import React, { useEffect, useState } from 'react'; import { IDebugger } from '../../tokens'; -import { convertType } from '.'; - import { VariablesModel } from './model'; +import { convertType } from '.'; + /** * The body for tree of variables. */ @@ -38,17 +38,18 @@ export class VariablesBodyTree extends ReactWidget { * Render the VariablesBodyTree. */ render(): JSX.Element { - return ( - <> - {this._scopes.map(scope => ( - - ))} - + const scope = + this._scopes.find(scope => scope.name === this._scope) ?? this._scopes[0]; + + return scope ? ( + + ) : ( +
); } @@ -60,6 +61,14 @@ export class VariablesBodyTree extends ReactWidget { this.update(); } + /** + * Set the current scope + */ + set scope(scope: string) { + this._scope = scope; + this.update(); + } + /** * Update the scopes and the tree of variables. * @@ -73,6 +82,7 @@ export class VariablesBodyTree extends ReactWidget { this.update(); } + private _scope = ''; private _scopes: IDebugger.IScope[] = []; private _filter = new Set(); private _service: IDebugger; @@ -106,7 +116,7 @@ const VariablesComponent = ({ {variables ?.filter(variable => !filter.has(variable.evaluateName)) .map(variable => { - const key = `${variable.evaluateName}-${variable.type}-${variable.value}`; + const key = `${variable.name}-${variable.evaluateName}-${variable.type}-${variable.value}`; return ( { + return scopes.map((scope, i) => { return { name: scope.name, - variables: variables.map(variable => { + variables: variables[i].map(variable => { return { ...variable }; }) }; @@ -601,7 +601,9 @@ export class DebuggerService implements IDebugger, IDisposable { return; } const scopes = await this._getScopes(frame); - const variables = await this._getVariables(scopes[0]); + const variables = await Promise.all( + scopes.map(scope => this._getVariables(scope)) + ); const variableScopes = this._convertScopes(scopes, variables); this._model.variables.scopes = variableScopes; }