streamline scm extension api

This commit is contained in:
Joao Moreno
2016-11-24 16:24:32 +01:00
parent 0edd066b17
commit ceda84ef28
5 changed files with 102 additions and 72 deletions

View File

@@ -7,6 +7,7 @@
"engines": {
"vscode": "^1.5.0"
},
"enableProposedApi": true,
"categories": [
"Other"
],

View File

@@ -5,7 +5,7 @@
'use strict';
import { scm, ExtensionContext, workspace } from 'vscode';
import { scm, ExtensionContext, workspace, Uri } from 'vscode';
import * as path from 'path';
import { findGit, Git } from './git';
@@ -13,6 +13,19 @@ export function log(...args: any[]): void {
console.log.apply(console, ['git:', ...args]);
}
class GitSCMProvider {
resourceGroups = [];
onDidChangeResourceGroup = null;
getOriginalResource(uri: Uri): Uri {
if (uri.scheme !== 'file') {
return null;
}
return uri.with({ scheme: 'git-index' });
}
}
export function activate(context: ExtensionContext): any {
if (!workspace) {
return;
@@ -24,16 +37,8 @@ export function activate(context: ExtensionContext): any {
log(`Using git ${info.version} from ${info.path}`);
const git = new Git({ gitPath: info.path, version: info.version });
const scmProvider = scm.createSCMProvider('git', {
getOriginalResource: uri => {
if (uri.scheme !== 'file') {
return null;
}
return uri.with({ scheme: 'git-index' });
}
});
const provider = new GitSCMProvider();
const providerDisposable = scm.registerSCMProvider('git', provider);
const contentProvider = workspace.registerTextDocumentContentProvider('git-index', {
provideTextDocumentContent: uri => {
@@ -49,6 +54,6 @@ export function activate(context: ExtensionContext): any {
}
});
context.subscriptions.push(scmProvider, contentProvider);
context.subscriptions.push(providerDisposable, contentProvider);
});
}

View File

@@ -86,28 +86,26 @@ declare module 'vscode' {
getClickCommand?(node: T): string;
}
export namespace scm {
export function createSCMProvider(id: string, delegate: SCMDelegate): SCMProvider;
export interface SCMResource {
uri: Uri;
}
export interface SCMDelegate {
export interface SCMResourceGroup {
resources: SCMResource[];
}
export interface SCMProvider {
commitCommand?: string;
clickCommand?: string;
dragCommand?: string;
resourceGroups: SCMResourceGroup[];
onDidChangeResourceGroup: Event<SCMResourceGroup>;
getOriginalResource?(uri: Uri, token: CancellationToken): Uri | Thenable<Uri>;
}
export interface SCMProvider extends Disposable {
createResourceGroup(id: string, label: string): SCMResourceGroup;
}
export interface SCMResourceGroup extends Disposable {
set(...resources: SCMResource[]): void;
get(): SCMResource[];
}
export interface SCMResource {
uri: Uri;
// TODO: status type, icon decoration, etc
export namespace scm {
export const onDidChangeActiveProvider: Event<SCMProvider>;
export let activeProvider: SCMProvider | undefined;
export function registerSCMProvider(id: string, provider: SCMProvider): Disposable;
}
}

View File

@@ -60,6 +60,33 @@ function proposedApiFunction<T>(extension: IExtensionDescription, fn: T): T {
}
}
function proposedAPI(extension: IExtensionDescription): Function {
return (target: any, key: string, descriptor: any) => {
let fnKey: string = null;
let fn: Function = null;
if (typeof descriptor.value === 'function') {
fnKey = 'value';
fn = descriptor.value;
} else if (typeof descriptor.get === 'function') {
fnKey = 'get';
fn = descriptor.get;
}
if (!fn) {
throw new Error('not supported');
}
if (extension.enableProposedApi) {
return;
}
descriptor[fnKey] = () => {
throw new Error(`${extension.id} cannot access proposed api`);
};
};
}
/**
* This method instantiates and returns the extension API surface
*/
@@ -360,12 +387,26 @@ export function createApiFactory(initData: IInitData, threadService: IThreadServ
}
};
// namespace: scm
const scm: typeof vscode.scm = {
createSCMProvider: (id, delegate): vscode.SCMProvider => {
return extHostSCM.createSCMProvider(id, delegate);
class SCM {
@proposedAPI(extension)
get activeProvider() {
return extHostSCM.activeProvider;
}
};
@proposedAPI(extension)
get onDidChangeActiveProvider() {
return extHostSCM.onDidChangeActiveProvider;
}
@proposedAPI(extension)
registerSCMProvider(id, provider) {
return extHostSCM.registerSCMProvider(id, provider);
}
}
// namespace: scm
const scm: typeof vscode.scm = new SCM();
return {
version: pkg.version,

View File

@@ -6,65 +6,50 @@
import URI from 'vs/base/common/uri';
import { TPromise } from 'vs/base/common/winjs.base';
import Event, { Emitter } from 'vs/base/common/event';
import { asWinJsPromise } from 'vs/base/common/async';
import { IThreadService } from 'vs/workbench/services/thread/common/threadService';
import { SCMProvider, SCMDelegate, SCMResourceGroup } from 'vscode';
import { SCMProvider } from 'vscode';
import { Disposable } from 'vs/workbench/api/node/extHostTypes';
import { MainContext, MainThreadSCMShape } from './extHost.protocol';
class ExtHostSCMProvider implements SCMProvider {
static Providers: { [id: string]: ExtHostSCMProvider; } = Object.create(null);
constructor(
private _proxy: MainThreadSCMShape,
private _id: string,
private _delegate: SCMDelegate
) {
if (ExtHostSCMProvider.Providers[_id]) {
throw new Error('provider already exists');
}
ExtHostSCMProvider.Providers[_id] = this;
_proxy.$register(this._id, !!this._delegate.getOriginalResource);
}
get id(): string {
return this._id;
}
createResourceGroup(id: string, label: string): SCMResourceGroup {
throw new Error('JOAO not implemented');
}
getBaselineResource(uri: URI): TPromise<URI> {
return asWinJsPromise(token => this._delegate.getOriginalResource(uri, token));
}
dispose(): void {
this._proxy.$unregister(this._id);
delete ExtHostSCMProvider.Providers[this.id];
}
}
export class ExtHostSCM {
private _proxy: MainThreadSCMShape;
private _providers: { [id: string]: SCMProvider; } = Object.create(null);
private _onDidChangeActiveProvider = new Emitter<SCMProvider>();
get onDidChangeActiveProvider(): Event<SCMProvider> { return this._onDidChangeActiveProvider.event; }
private _activeProvider: SCMProvider;
get activeProvider(): SCMProvider | undefined { return this._activeProvider; }
constructor(threadService: IThreadService) {
this._proxy = threadService.get(MainContext.MainThreadSCM);
}
createSCMProvider(id: string, delegate: SCMDelegate): SCMProvider {
return new ExtHostSCMProvider(this._proxy, id, delegate);
registerSCMProvider(id: string, provider: SCMProvider): Disposable {
if (this._providers[id]) {
throw new Error(`Provider ${id} already registered`);
}
// TODO@joao: should pluck all the things out of the provider
this._providers[id] = provider;
this._proxy.$register(id, !!provider.getOriginalResource);
return new Disposable(() => {
delete this._providers[id];
this._proxy.$unregister(id);
});
}
$getBaselineResource(id: string, uri: URI): TPromise<URI> {
const provider = ExtHostSCMProvider.Providers[id];
const provider = this._providers[id];
if (!provider) {
return TPromise.as(null);
}
return provider.getBaselineResource(uri);
return asWinJsPromise(token => provider.getOriginalResource(uri, token));
}
}