diff --git a/src/kernels/execution/cellExecutionMessageHandler.ts b/src/kernels/execution/cellExecutionMessageHandler.ts index 785bf3d9e1a..88f00d61d21 100644 --- a/src/kernels/execution/cellExecutionMessageHandler.ts +++ b/src/kernels/execution/cellExecutionMessageHandler.ts @@ -21,7 +21,8 @@ import { NotebookEdit, NotebookCellOutputItem, Disposable, - window + window, + extensions } from 'vscode'; import type { Kernel } from '@jupyterlab/services'; @@ -41,9 +42,10 @@ import { swallowExceptions } from '../../platform/common/utils/decorators'; import { noop } from '../../platform/common/utils/misc'; import { IKernelController, ITracebackFormatter } from '../../kernels/types'; import { handleTensorBoardDisplayDataOutput } from './executionHelpers'; -import { Identifiers, WIDGET_MIMETYPE } from '../../platform/common/constants'; +import { Identifiers, RendererExtension, WIDGET_MIMETYPE } from '../../platform/common/constants'; import { CellOutputDisplayIdTracker } from './cellDisplayIdTracker'; import { createDeferred } from '../../platform/common/utils/async'; +import { coerce, SemVer } from 'semver'; // Helper interface for the set_next_input execute reply payload interface ISetNextInputPayload { @@ -714,13 +716,15 @@ export class CellExecutionMessageHandler implements IDisposable { // Jupyter Output widgets cannot be rendered properly by the widget manager, // We need to render that. if (typeof data.model_id === 'string' && this.commIdsMappedToWidgetOutputModels.has(data.model_id)) { - return false; + // New version of renderer supports this. + return doesNotebookRendererSupportRenderingNestedOutputsInWidgets(); } return true; } if (mime.startsWith('application/vnd')) { - // Custom vendored mimetypes cannot be rendered by the widget manager, it relies on the output renderers. - return false; + // Custom vendored mimetypes cannot be rendered by the widget manager in older versions, it relies on the output renderers. + // New version of renderer supports this. + return doesNotebookRendererSupportRenderingNestedOutputsInWidgets(); } // Everything else can be rendered by the Jupyter Lab widget manager. return true; @@ -1209,3 +1213,16 @@ export class CellExecutionMessageHandler implements IDisposable { this.endTemporaryTask(); } } + +function doesNotebookRendererSupportRenderingNestedOutputsInWidgets() { + const rendererExtension = extensions.getExtension(RendererExtension); + if (!rendererExtension) { + return false; + } + + const version = coerce(rendererExtension.packageJSON.version); + if (!version) { + return false; + } + return version.compare(new SemVer('1.0.23')) >= 0; +} diff --git a/src/notebooks/controllers/ipywidgets/message/ipyWidgetMessageDispatcher.ts b/src/notebooks/controllers/ipywidgets/message/ipyWidgetMessageDispatcher.ts index c8b4673dfe7..2ebbb7d56b6 100644 --- a/src/notebooks/controllers/ipywidgets/message/ipyWidgetMessageDispatcher.ts +++ b/src/notebooks/controllers/ipywidgets/message/ipyWidgetMessageDispatcher.ts @@ -16,6 +16,7 @@ import { IKernel, IKernelProvider, type IKernelSocket } from '../../../../kernel import { IIPyWidgetMessageDispatcher, IPyWidgetMessage } from '../types'; import { shouldMessageBeMirroredWithRenderer } from '../../../../kernels/kernel'; import { KernelSocketMap } from '../../../../kernels/kernelSocket'; +import type { IDisplayDataMsg } from '@jupyterlab/services/lib/kernel/messages'; type PendingMessage = { resultPromise: Deferred; @@ -52,6 +53,8 @@ export class IPyWidgetMessageDispatcher implements IIPyWidgetMessageDispatcher { private pendingTargetNames = new Set(); private kernel?: IKernel; private _postMessageEmitter = new EventEmitter(); + private _onDisplayMessage = new EventEmitter(); + public readonly onDisplayMessage = this._onDisplayMessage.event; private messageHooks = new Map boolean | PromiseLike>(); private pendingHookRemovals = new Map(); private messageHookRequests = new Map>(); @@ -367,10 +370,17 @@ export class IPyWidgetMessageDispatcher implements IIPyWidgetMessageDispatcher { const msgUuid = uuid(); const promise = createDeferred(); this.waitingMessageIds.set(msgUuid, { startTime: Date.now(), resultPromise: promise }); - + let deserializedMessage: KernelMessage.IMessage | undefined = undefined; if (typeof data === 'string') { if (shouldMessageBeMirroredWithRenderer(data)) { this.raisePostMessage(IPyWidgetMessages.IPyWidgets_msg, { id: msgUuid, data }); + if (data.includes('display_data')) { + deserializedMessage = this.deserialize(data as any, protocol); + const jupyterLab = require('@jupyterlab/services') as typeof import('@jupyterlab/services'); + if (jupyterLab.KernelMessage.isDisplayDataMsg(deserializedMessage)) { + this._onDisplayMessage.fire(deserializedMessage); + } + } } } else { const dataToSend = serializeDataViews([data as any]); @@ -392,7 +402,7 @@ export class IPyWidgetMessageDispatcher implements IIPyWidgetMessageDispatcher { data.includes('comm_close') || data.includes('comm_msg'); if (mustDeserialize) { - const message = this.deserialize(data as any, protocol) as any; + const message = deserializedMessage || this.deserialize(data as any, protocol) as any; if (!shouldMessageBeMirroredWithRenderer(message)) { return; } diff --git a/src/notebooks/controllers/ipywidgets/message/ipyWidgetMessageDispatcherFactory.ts b/src/notebooks/controllers/ipywidgets/message/ipyWidgetMessageDispatcherFactory.ts index adcc311b002..b5e82488618 100644 --- a/src/notebooks/controllers/ipywidgets/message/ipyWidgetMessageDispatcherFactory.ts +++ b/src/notebooks/controllers/ipywidgets/message/ipyWidgetMessageDispatcherFactory.ts @@ -8,6 +8,7 @@ import { IPyWidgetMessages } from '../../../../messageTypes'; import { IKernel, IKernelProvider } from '../../../../kernels/types'; import { IPyWidgetMessageDispatcher } from './ipyWidgetMessageDispatcher'; import { IIPyWidgetMessageDispatcher, IPyWidgetMessage } from '../types'; +import type { IDisplayDataMsg } from '@jupyterlab/services/lib/kernel/messages'; /** * This just wraps the iPyWidgetMessageDispatcher class. @@ -19,12 +20,21 @@ class IPyWidgetMessageDispatcherWithOldMessages implements IIPyWidgetMessageDisp return this._postMessageEmitter.event; } private _postMessageEmitter = new EventEmitter(); + private _onDisplayMessage = new EventEmitter(); + public readonly onDisplayMessage = this._onDisplayMessage.event; private readonly disposables: IDisposable[] = []; constructor( private readonly baseMulticaster: IPyWidgetMessageDispatcher, private oldMessages: ReadonlyArray ) { baseMulticaster.postMessage(this.raisePostMessage, this, this.disposables); + baseMulticaster.onDisplayMessage( + (e) => { + this._onDisplayMessage.fire(e); + }, + this, + this.disposables + ); } public dispose() { diff --git a/src/notebooks/controllers/ipywidgets/types.ts b/src/notebooks/controllers/ipywidgets/types.ts index 6c9725b3a67..6843464a790 100644 --- a/src/notebooks/controllers/ipywidgets/types.ts +++ b/src/notebooks/controllers/ipywidgets/types.ts @@ -5,6 +5,7 @@ import { Event, Uri } from 'vscode'; import { IDisposable } from '../../../platform/common/types'; import { IPyWidgetMessages } from '../../../messageTypes'; import { IKernel } from '../../../kernels/types'; +import type { IDisplayDataMsg } from '@jupyterlab/services/lib/kernel/messages'; export interface IPyWidgetMessage { message: IPyWidgetMessages; @@ -16,6 +17,7 @@ export interface IPyWidgetMessage { * Used to send/receive messages related to IPyWidgets */ export interface IIPyWidgetMessageDispatcher extends IDisposable { + onDisplayMessage: Event; // eslint-disable-next-line @typescript-eslint/no-explicit-any postMessage: Event; // eslint-disable-next-line @typescript-eslint/no-explicit-any diff --git a/src/webviews/extension-side/ipywidgets/rendererComms.ts b/src/webviews/extension-side/ipywidgets/rendererComms.ts index 3aad55c9a24..c400753be71 100644 --- a/src/webviews/extension-side/ipywidgets/rendererComms.ts +++ b/src/webviews/extension-side/ipywidgets/rendererComms.ts @@ -14,6 +14,7 @@ import { dispose } from '../../../platform/common/utils/lifecycle'; import { IDisposable } from '../../../platform/common/types'; import { noop } from '../../../platform/common/utils/misc'; import { logger } from '../../../platform/logging'; +import { IPyWidgetMessageDispatcherFactory } from '../../../notebooks/controllers/ipywidgets/message/ipyWidgetMessageDispatcherFactory'; type WidgetData = { model_id: string; @@ -27,7 +28,9 @@ export class IPyWidgetRendererComms implements IExtensionSyncActivationService { private readonly disposables: IDisposable[] = []; constructor( @inject(IKernelProvider) private readonly kernelProvider: IKernelProvider, - @inject(IControllerRegistration) private readonly controllers: IControllerRegistration + @inject(IControllerRegistration) private readonly controllers: IControllerRegistration, + @inject(IPyWidgetMessageDispatcherFactory) + private readonly ipywidgetMessageDispatcher: IPyWidgetMessageDispatcherFactory ) {} private readonly widgetOutputsPerNotebook = new WeakMap>(); public dispose() { @@ -51,6 +54,21 @@ export class IPyWidgetRendererComms implements IExtensionSyncActivationService { return; } + // If we have an output widget nested within another output widget. + // Then the output output widget will be displayed by us. + // However nested outputs (any) widgets will be displayed by widget manager. + // And in this case, its possible the display_data message is sent to the webview, + // Sooner than we get the messages from the IKernel above. + // Hence we need to hook into the lower level kernel socket messages to see if that happens. + // Else what happens is the display_data is sent to the webview, but the widget manager doesn't know about it. + // Thats because we have not tracked this model and we don't know about it. + const ipyWidgetMessageDispatcher = this.ipywidgetMessageDispatcher.create(kernel.notebook); + this.disposables.push( + ipyWidgetMessageDispatcher.onDisplayMessage((msg) => { + this.trackModelId(kernel.notebook, msg); + }) + ); + // eslint-disable-next-line @typescript-eslint/no-require-imports const jupyterLab = require('@jupyterlab/services') as typeof import('@jupyterlab/services'); const handler = (kernelConnection: IKernelConnection, msg: IIOPubMessage) => { diff --git a/src/webviews/webview-side/ipywidgets/kernel/kernel.ts b/src/webviews/webview-side/ipywidgets/kernel/kernel.ts index 9d8dc407f53..e4b7b6128ad 100644 --- a/src/webviews/webview-side/ipywidgets/kernel/kernel.ts +++ b/src/webviews/webview-side/ipywidgets/kernel/kernel.ts @@ -104,8 +104,14 @@ class ProxyKernel implements IMessageHandler, Kernel.IKernelConnection { private hookResults = new Map>(); private websocket: WebSocketWS & { sendEnabled: boolean }; private messageHook: (msg: KernelMessage.IIOPubMessage) => boolean | PromiseLike; - private messageHooks: Map boolean | PromiseLike>; - private lastHookedMessageId: string | undefined; + private readonly messageHooks = new Map< + string, + { + current: (msg: KernelMessage.IIOPubMessage) => boolean | PromiseLike; + previous: ((msg: KernelMessage.IIOPubMessage) => boolean | PromiseLike)[]; + } + >(); + private readonly lastHookedMessageId: string[] = []; private _options: KernelSocketOptions; // Messages that are awaiting extension messages to be fully handled private awaitingExtensionMessage: Map>; @@ -169,7 +175,6 @@ class ProxyKernel implements IMessageHandler, Kernel.IKernelConnection { postOffice.addHandler(this); this.websocket = proxySocketInstance; this.messageHook = this.messageHookInterceptor.bind(this); - this.messageHooks = new Map boolean | PromiseLike>(); this.fakeOpenSocket(); } @@ -343,7 +348,14 @@ class ProxyKernel implements IMessageHandler, Kernel.IKernelConnection { this.postOffice.sendMessage(IPyWidgetMessages.IPyWidgets_RegisterMessageHook, msgId); // Save the real hook so we can call it - this.messageHooks.set(msgId, hook); + const item = this.messageHooks.get(msgId); + if (item) { + // Preserve the previous hook and setup a new hook for the same comm msg. + item.previous.push(item.current); + item.current = hook; + } else { + this.messageHooks.set(msgId, { current: hook, previous: [] }); + } // Wrap the hook and send it to the real kernel this.realKernel.registerMessageHook(msgId, this.messageHook); @@ -363,15 +375,20 @@ class ProxyKernel implements IMessageHandler, Kernel.IKernelConnection { this.postOffice.sendMessage(IPyWidgetMessages.IPyWidgets_RemoveMessageHook, { hookMsgId: msgId, - lastHookedMsgId: this.lastHookedMessageId + lastHookedMsgId: this.lastHookedMessageId.length ? this.lastHookedMessageId.pop() : undefined }); // Remove our mapping - this.messageHooks.delete(msgId); - this.lastHookedMessageId = undefined; - - // Remove from the real kernel - this.realKernel.removeMessageHook(msgId, this.messageHook); + const item = this.messageHooks.get(msgId); + if (item) { + if (item.previous.length > 0) { + item.current = item.previous.pop()!; + } else { + this.messageHooks.delete(msgId); + // Remove from the real kernel + this.realKernel.removeMessageHook(msgId, this.messageHook); + } + } } // Called when the extension has finished an operation that we are waiting for in message processing @@ -415,12 +432,12 @@ class ProxyKernel implements IMessageHandler, Kernel.IKernelConnection { try { // Save the active message that is currently being hooked. The Extension // side needs this information during removeMessageHook so it can delay removal until after a message is called - this.lastHookedMessageId = msg.header.msg_id; + this.lastHookedMessageId.push(msg.header.msg_id); const hook = this.messageHooks.get((msg.parent_header as any).msg_id); if (hook) { // When the kernel calls the hook, save the result for this message. The other side will ask for it - const result = hook(msg); + const result = hook.current(msg); this.hookResults.set(msg.header.msg_id, result); if ((result as any).then) { return (result as any).then((r: boolean) => { diff --git a/src/webviews/webview-side/ipywidgets/kernel/manager.ts b/src/webviews/webview-side/ipywidgets/kernel/manager.ts index d9e6e84e12e..470438c1481 100644 --- a/src/webviews/webview-side/ipywidgets/kernel/manager.ts +++ b/src/webviews/webview-side/ipywidgets/kernel/manager.ts @@ -2,7 +2,7 @@ // Licensed under the MIT License. import '@jupyter-widgets/controls/css/labvariables.css'; - +import { IRenderMime, RenderedCommon, RenderMimeRegistry, standardRendererFactories } from '@jupyterlab/rendermime'; import type { Kernel, KernelMessage } from '@jupyterlab/services'; import type * as nbformat from '@jupyterlab/nbformat'; import { Widget } from '@lumino/widgets'; @@ -24,8 +24,9 @@ import { IInteractiveWindowMapping, IPyWidgetMessages, InteractiveWindowMessages import { WIDGET_MIMETYPE, WIDGET_STATE_MIMETYPE } from '../../../../platform/common/constants'; import { NotebookMetadata } from '../../../../platform/common/utils'; import { noop } from '../../../../platform/common/utils/misc'; - -/* eslint-disable @typescript-eslint/no-explicit-any */ +import type { RendererContext } from 'vscode-notebook-renderer'; +import { renderersAndMimetypes } from './mimeTypes'; +import { base64ToUint8Array } from '../../../../platform/common/utils/string'; export class WidgetManager implements IIPyWidgetManager, IMessageHandler { public static get onDidChangeInstance(): Event { @@ -36,6 +37,7 @@ export class WidgetManager implements IIPyWidgetManager, IMessageHandler { private manager?: IJupyterLabWidgetManager; private proxyKernel?: Kernel.IKernelConnection; private options?: KernelSocketOptions; + // eslint-disable-next-line @typescript-eslint/no-explicit-any private pendingMessages: { message: string; payload: any }[] = []; /** * Contains promises related to model_ids that need to be displayed. @@ -224,13 +226,108 @@ export class WidgetManager implements IIPyWidgetManager, IMessageHandler { this.manager?.dispose(); // NOSONAR try { // Create the real manager and point it at our proxy kernel. - this.manager = new this.JupyterLabWidgetManager( + const manager = (this.manager = new this.JupyterLabWidgetManager( this.proxyKernel, this.widgetContainer, this.scriptLoader, logMessage, widgetState - ); + )); + + const registeredMimeTypes = new Set(); + renderersAndMimetypes.forEach((mimeTypes, rendererId) => { + mimeTypes.forEach((mime) => { + if (registeredMimeTypes.has(mime) || !manager.registerMimeRenderer) { + return; + } + registeredMimeTypes.add(mime); + // // eslint-disable-next-line @typescript-eslint/no-explicit-any + // (manager as any).rendermime.addFactory({ + // safe: true, + // mimeTypes: [mime], + // createRenderer: (options: IRenderMime.IRendererOptions) => { + // debugger; + // return new (class MimeRenderer extends RenderedCommon { + // override async render(model: IRenderMime.IMimeModel): Promise { + // debugger; + // const data = model.data; + // const metadata = model.metadata; + // const context = // eslint-disable-next-line @typescript-eslint/no-explicit-any + // (globalThis as any).jupyter_vscode_rendererContext as RendererContext; + // const renderer = await context.getRenderer(rendererId); + // const isImage = + // mime.toLowerCase().startsWith('image/') && !mime.toLowerCase().includes('svg'); + // renderer?.renderOutputItem( + // { + // id: new Date().getTime().toString(), // Not used except when saving plots, but with nested outputs, thats not possible. + // metadata, + // text: () => { + // return JSON.stringify(data[mime]); + // }, + // json: () => { + // return data[mime]; + // }, + // blob() { + // if (isImage) { + // const bytes = base64ToUint8Array(data[mime] as string); + // return new Blob([bytes], { type: mime }); + // } else { + // throw new Error(`Not able to get blob for ${mime}.`); + // } + // }, + // data() { + // if (isImage) { + // return base64ToUint8Array(data[mime] as string); + // } else { + // throw new Error(`Not able to get blob for ${mime}.`); + // } + // }, + // mime + // }, + // this.node + // ); + // } + // })(options); + // } + // }); + + manager.registerMimeRenderer(mime, async ({ data, metadata }, node) => { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const context = (globalThis as any).jupyter_vscode_rendererContext as RendererContext; + const renderer = await context.getRenderer(rendererId); + const isImage = mime.toLowerCase().startsWith('image/') && !mime.toLowerCase().includes('svg'); + renderer?.renderOutputItem( + { + id: new Date().getTime().toString(), // Not used except when saving plots, but with nested outputs, thats not possible. + metadata, + text: () => { + return JSON.stringify(data[mime]); + }, + json: () => { + return data[mime]; + }, + blob() { + if (isImage) { + const bytes = base64ToUint8Array(data[mime] as string); + return new Blob([bytes], { type: mime }); + } else { + throw new Error(`Not able to get blob for ${mime}.`); + } + }, + data() { + if (isImage) { + return base64ToUint8Array(data[mime] as string); + } else { + throw new Error(`Not able to get blob for ${mime}.`); + } + }, + mime + }, + node + ); + }); + }); + }); // Listen for display data messages so we can prime the model for a display data this.proxyKernel.iopubMessage.connect(this.handleDisplayDataMessage.bind(this)); @@ -249,7 +346,8 @@ export class WidgetManager implements IIPyWidgetManager, IMessageHandler { /** * Ensure we create the model for the display data. */ - private handleDisplayDataMessage(_sender: any, payload: KernelMessage.IIOPubMessage) { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + private async handleDisplayDataMessage(_sender: any, payload: KernelMessage.IIOPubMessage) { // eslint-disable-next-line @typescript-eslint/no-require-imports const jupyterLab = require('@jupyterlab/services') as typeof import('@jupyterlab/services'); // NOSONAR @@ -276,7 +374,13 @@ export class WidgetManager implements IIPyWidgetManager, IMessageHandler { } const modelPromise = this.manager.get_model(data.model_id); if (modelPromise) { - modelPromise.then((_m) => deferred?.resolve()).catch((e) => deferred?.reject(e)); + try { + const m = await modelPromise; + console.log(m); + deferred.resolve(); + } catch (ex) { + deferred.reject(ex); + } } else { deferred.resolve(); } diff --git a/src/webviews/webview-side/ipywidgets/kernel/mimeTypes.ts b/src/webviews/webview-side/ipywidgets/kernel/mimeTypes.ts new file mode 100644 index 00000000000..328ea52e274 --- /dev/null +++ b/src/webviews/webview-side/ipywidgets/kernel/mimeTypes.ts @@ -0,0 +1,48 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +export const renderersAndMimetypes = new Map([ + // Ability to render nested outputs widgets. + // I.e. Jupyter Labl widget manager must be able to render a widget as well, not just regular mimetypes. + ['jupyter-ipywidget-renderer', ['application/vnd.jupyter.widget-view+json']], + // https://github.com/microsoft/vscode-notebook-renderers/blob/homely-louse/package.json#L80 + [ + 'jupyter-notebook-renderer', + [ + 'image/gif', + 'image/png', + 'image/jpeg', + 'image/webp', + 'image/svg+xml', + 'application/geo+json', + 'application/vdom.v1+json', + 'application/vnd.dataresource+json', + 'application/vnd.plotly.v1+json', + 'application/vnd.vega.v2+json', + 'application/vnd.vega.v3+json', + 'application/vnd.vega.v4+json', + 'application/vnd.vegalite.v1+json', + 'application/vnd.vegalite.v2+json', + 'application/x-nteract-model-debug+json', + 'text/vnd.plotly.v1+html' + ] + ], + // Built in extensions in core. + ['vscode.markdown-it-renderer', ['text/markdown', 'text/latex', 'application/json']], + // Built in extensions in core. + [ + 'vscode.builtin-renderer', + [ + 'image/git', + 'text/html', + 'application/javascript', + 'application/vnd.code.notebook.error', + 'application/vnd.code.notebook.stdout', + 'application/vnd.code.notebook.stderr', + 'application/x.notebook.stdout', + 'application/x.notebook.stream', + 'application/x.notebook.stderr', + 'text/plain' + ] + ] +]); diff --git a/src/webviews/webview-side/ipywidgets/kernel/types.ts b/src/webviews/webview-side/ipywidgets/kernel/types.ts index 55b007cfa69..0e8ab690e3e 100644 --- a/src/webviews/webview-side/ipywidgets/kernel/types.ts +++ b/src/webviews/webview-side/ipywidgets/kernel/types.ts @@ -7,6 +7,7 @@ import type * as nbformat from '@jupyterlab/nbformat'; import { ISignal } from '@lumino/signaling'; import { Widget } from '@lumino/widgets'; import { NotebookMetadata } from '../../../../platform/common/utils'; +import type { ReadonlyPartialJSONObject } from '@lumino/coreutils'; export type ScriptLoader = { readonly widgetsRegisteredInRequireJs: Readonly>; @@ -74,6 +75,35 @@ export interface IJupyterLabWidgetManager { loadNotebook: boolean; } ): Promise; + + /** + * Registers a mime type to be rendered by Widget Manager. + * This is useful when widgets have nested outputs. + * E.g. output widget has a plotly output as a child. + * In this case we'd like the widget manager to render the plotly output using our renderers. + * + * This is optional, as only the later versions of the vscode-jupyter-ipywidgets support this. + * I.e. only later versions of notebook renderers. + */ + registerMimeRenderer?( + mimeType: string, + render: ( + model: { + /** + * The data associated with the model. + */ + readonly data: ReadonlyPartialJSONObject; + /** + * The metadata associated with the model. + * + * Among others, it can include an attribute named `fragment` + * that stores a URI fragment identifier for the MIME resource. + */ + readonly metadata: ReadonlyPartialJSONObject; + }, + node: HTMLElement + ) => Promise + ): void; } // export interface IIPyWidgetManager extends IMessageHandler { diff --git a/src/webviews/webview-side/ipywidgets/renderer/index.ts b/src/webviews/webview-side/ipywidgets/renderer/index.ts index 26f0fcef577..6e48d7cf357 100644 --- a/src/webviews/webview-side/ipywidgets/renderer/index.ts +++ b/src/webviews/webview-side/ipywidgets/renderer/index.ts @@ -66,7 +66,8 @@ export const activate: ActivationFunction = (context) => { console.error(message); } }; - + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (globalThis as any).jupyter_vscode_rendererContext = context; logger('Jupyter IPyWidget Renderer Activated'); hookupTestScripts(context); const modelAvailabilityResponse = new Map>();