Files
theia-code-os/packages/vsx-registry/src/browser/vsx-extensions-widget.tsx
mawkone 8bb5110148
Some checks failed
Playwright Tests / Playwright Tests (ubuntu-22.04, Node.js 22.x) (push) Has been cancelled
3PP License Check / 3PP License Check (11, 22.x, ubuntu-22.04) (push) Has been cancelled
Publish packages to NPM / Perform Publishing (push) Has been cancelled
deploy: current vibn theia state
Made-with: Cursor
2026-02-27 12:01:08 -08:00

166 lines
6.2 KiB
TypeScript

// *****************************************************************************
// Copyright (C) 2020 TypeFox and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// http://www.eclipse.org/legal/epl-2.0.
//
// This Source Code may also be made available under the following Secondary
// Licenses when the conditions for such availability set forth in the Eclipse
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
// with the GNU Classpath Exception which is available at
// https://www.gnu.org/software/classpath/license.html.
//
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
// *****************************************************************************
import { injectable, interfaces, postConstruct, inject } from '@theia/core/shared/inversify';
import { Message, TreeModel, TreeNode } from '@theia/core/lib/browser';
import { SourceTreeWidget } from '@theia/core/lib/browser/source-tree';
import { VSXExtensionsSource, VSXExtensionsSourceOptions } from './vsx-extensions-source';
import { nls } from '@theia/core/lib/common/nls';
import { BadgeWidget } from '@theia/core/lib/browser/view-container';
import { Emitter, Event } from '@theia/core/lib/common';
import { AlertMessage } from '@theia/core/lib/browser/widgets/alert-message';
import * as React from '@theia/core/shared/react';
@injectable()
export class VSXExtensionsWidgetOptions extends VSXExtensionsSourceOptions {
title?: string;
}
export const generateExtensionWidgetId = (widgetId: string): string => VSXExtensionsWidget.ID + ':' + widgetId;
@injectable()
export class VSXExtensionsWidget extends SourceTreeWidget implements BadgeWidget {
static ID = 'vsx-extensions';
static createWidget(parent: interfaces.Container, options: VSXExtensionsWidgetOptions): VSXExtensionsWidget {
const child = SourceTreeWidget.createContainer(parent, {
virtualized: false,
scrollIfActive: true
});
child.bind(VSXExtensionsSourceOptions).toConstantValue(options);
child.bind(VSXExtensionsSource).toSelf();
child.unbind(SourceTreeWidget);
child.bind(VSXExtensionsWidgetOptions).toConstantValue(options);
child.bind(VSXExtensionsWidget).toSelf();
return child.get(VSXExtensionsWidget);
}
protected _badge?: number;
protected onDidChangeBadgeEmitter = new Emitter<void>();
protected _badgeTooltip?: string;
protected onDidChangeBadgeTooltipEmitter = new Emitter<void>();
@inject(VSXExtensionsWidgetOptions)
protected readonly options: VSXExtensionsWidgetOptions;
@inject(VSXExtensionsSource)
protected readonly extensionsSource: VSXExtensionsSource;
@postConstruct()
protected override init(): void {
super.init();
this.addClass('theia-vsx-extensions');
this.id = generateExtensionWidgetId(this.options.id);
this.toDispose.push(this.extensionsSource);
this.source = this.extensionsSource;
const title = this.options.title ?? this.computeTitle();
this.title.label = title;
this.title.caption = title;
this.toDispose.push(this.source.onDidChange(async () => {
this.badge = await this.resolveCount();
}));
}
get onDidChangeBadge(): Event<void> {
return this.onDidChangeBadgeEmitter.event;
}
get badge(): number | undefined {
return this._badge;
}
set badge(count: number | undefined) {
this._badge = count;
this.onDidChangeBadgeEmitter.fire();
}
get onDidChangeBadgeTooltip(): Event<void> {
return this.onDidChangeBadgeTooltipEmitter.event;
}
get badgeTooltip(): string | undefined {
return this._badgeTooltip;
}
set badgeTooltip(tooltip: string | undefined) {
this._badgeTooltip = tooltip;
this.onDidChangeBadgeTooltipEmitter.fire();
}
protected computeTitle(): string {
switch (this.options.id) {
case VSXExtensionsSourceOptions.INSTALLED:
return nls.localizeByDefault('Installed');
case VSXExtensionsSourceOptions.BUILT_IN:
return nls.localizeByDefault('Built-in');
case VSXExtensionsSourceOptions.RECOMMENDED:
return nls.localizeByDefault('Recommended');
case VSXExtensionsSourceOptions.SEARCH_RESULT:
return 'Open VSX Registry';
default:
return '';
}
}
protected async resolveCount(): Promise<number | undefined> {
if (this.options.id !== VSXExtensionsSourceOptions.SEARCH_RESULT) {
const elements = await this.source?.getElements() || [];
return [...elements].length;
}
return undefined;
}
protected override tapNode(node?: TreeNode): void {
super.tapNode(node);
this.model.openNode(node);
}
protected override handleDblClickEvent(): void {
// Don't open the editor view on a double click.
}
protected override renderTree(model: TreeModel): React.ReactNode {
if (this.options.id === VSXExtensionsSourceOptions.SEARCH_RESULT) {
const searchError = this.extensionsSource.getModel().searchError;
if (!!searchError) {
const message = nls.localize('theia/vsx-registry/errorFetching', 'Error fetching extensions.');
const configurationHint = nls.localize('theia/vsx-registry/errorFetchingConfigurationHint', 'This could be caused by network configuration issues.');
const hint = searchError.includes('ENOTFOUND') ? configurationHint : '';
return <AlertMessage
type='ERROR'
header={`${message} ${searchError} ${hint}`}
/>;
}
}
return super.renderTree(model);
}
protected override onAfterShow(msg: Message): void {
super.onAfterShow(msg);
if (this.options.id === VSXExtensionsSourceOptions.INSTALLED) {
// This is needed when an Extension was installed outside of the extension view.
// E.g. using explorer context menu.
this.doUpdateRows();
}
}
}