Переглянути джерело

Adds new show repository status command

main
Eric Amodio 7 роки тому
джерело
коміт
0a4cdd81eb
8 змінених файлів з 330 додано та 72 видалено
  1. +21
    -1
      package.json
  2. +118
    -50
      src/commands/quickPickItems.ts
  3. +54
    -14
      src/commands/quickPicks.ts
  4. +61
    -0
      src/commands/showQuickRepoStatus.ts
  5. +2
    -0
      src/extension.ts
  6. +6
    -1
      src/git/git.ts
  7. +55
    -0
      src/git/gitEnrichment.ts
  8. +13
    -6
      src/gitProvider.ts

+ 21
- 1
package.json Переглянути файл

@ -383,6 +383,11 @@
"category": "GitLens"
},
{
"command": "gitlens.showQuickRepoStatus",
"title": "Show Repository Status",
"category": "GitLens"
},
{
"command": "gitlens.copyShaToClipboard",
"title": "Copy Commit Sha to Clipboard",
"category": "GitLens"
@ -444,6 +449,10 @@
"when": "gitlens:enabled"
},
{
"command": "gitlens.showQuickRepoStatus",
"when": "gitlens:enabled"
},
{
"command": "gitlens.copyShaToClipboard",
"when": "gitlens:enabled"
},
@ -478,7 +487,12 @@
{
"command": "gitlens.showQuickFileHistory",
"when": "gitlens:enabled",
"group": "1_gitlens"
"group": "1_gitlens@1"
},
{
"command": "gitlens.showQuickRepoStatus",
"when": "gitlens:enabled",
"group": "1_gitlens@2"
},
{
"command": "gitlens.diffWithPrevious",
@ -578,6 +592,12 @@
"when": "gitlens:enabled"
},
{
"command": "gitlens.showQuickRepoStatus",
"key": "alt+s",
"mac": "alt+s",
"when": "gitlens:enabled"
},
{
"command": "gitlens.showQuickCommitDetails",
"key": "alt+c",
"mac": "alt+c",

+ 118
- 50
src/commands/quickPickItems.ts Переглянути файл

@ -1,11 +1,19 @@
'use strict';
import { commands, QuickPickItem, TextEditor, Uri, window, workspace } from 'vscode';
import { BuiltInCommands, Commands } from '../constants';
import { GitCommit, GitUri } from '../gitProvider';
import { Commands } from '../commands';
import { BuiltInCommands } from '../constants';
import { GitCommit, GitFileStatusItem, GitUri } from '../gitProvider';
import * as moment from 'moment';
import * as path from 'path';
export interface PartialQuickPickItem {
label?: string;
description?: string;
detail?: string;
}
export class CommandQuickPickItem implements QuickPickItem {
label: string;
description: string;
detail: string;
@ -20,53 +28,89 @@ export class CommandQuickPickItem implements QuickPickItem {
}
export class OpenFilesCommandQuickPickItem extends CommandQuickPickItem {
label: string;
description: string;
detail: string;
constructor(private commit: GitCommit, private fileNames?: string[]) {
super({
label: `$(file-symlink-file) Open Files`,
description: `\u00a0 \u2014 \u00a0\u00a0 $(file-text) ${commit.fileName}`,
detail: `Opens all the files in commit $(git-commit) ${commit.sha}`
}, undefined, undefined);
constructor(public fileNames: string[], public repoPath: string, item: QuickPickItem) {
super(item, undefined, undefined);
}
if (!this.fileNames) {
this.fileNames = commit.fileName.split(', ').filter(_ => !!_);
}
getUri(fileName: string) {
return Uri.file(path.resolve(this.repoPath, fileName));
}
async execute(): Promise<{}> {
const repoPath = this.commit.repoPath;
for (const file of this.fileNames) {
try {
const uri = Uri.file(path.resolve(repoPath, file));
const document = await workspace.openTextDocument(uri);
await window.showTextDocument(document, 1, true);
}
catch (ex) { }
for (const fileName of this.fileNames) {
this.open(fileName);
}
return undefined;
}
async open(fileName: string): Promise<TextEditor | undefined> {
try {
const uri = this.getUri(fileName);
const document = await workspace.openTextDocument(uri);
return window.showTextDocument(document, 1, true);
}
catch (ex) {
return undefined;
}
}
}
export class OpenCommitFilesCommandQuickPickItem extends OpenFilesCommandQuickPickItem {
constructor(commit: GitCommit, fileNames?: string[], item?: PartialQuickPickItem) {
const repoPath = commit.repoPath;
if (!fileNames) {
fileNames = commit.fileName.split(', ').filter(_ => !!_);
}
item = {
...{
label: `$(file-symlink-file) Open Files`,
description: undefined,
detail: `Opens all of the files in commit $(git-commit) ${commit.sha}`
},
...item
};
super(fileNames, repoPath, item as QuickPickItem);
}
}
export class OpenStatusFilesCommandQuickPickItem extends OpenFilesCommandQuickPickItem {
constructor(statuses: GitFileStatusItem[], item?: PartialQuickPickItem) {
const repoPath = statuses[0].repoPath;
const fileNames = statuses.map(_ => _.fileName);
item = {
...{
label: `$(file-symlink-file) Open Files`,
description: undefined,
detail: `Opens all of the changed files in the repository`
},
...item
};
super(fileNames, repoPath, item as QuickPickItem);
}
}
export class OpenFileCommandQuickPickItem extends CommandQuickPickItem {
label: string;
description: string;
detail: string;
constructor(private commit: GitCommit) {
super({
label: `$(file-symlink-file) Open File`,
description: `\u00a0 \u2014 \u00a0\u00a0 $(file-text) ${commit.fileName}`
}, undefined, undefined);
constructor(public fileName: string, public repoPath: string, item: QuickPickItem) {
super(item, undefined, undefined);
}
getUri() {
return Uri.file(path.resolve(this.repoPath, this.fileName));
}
async execute(): Promise<{}> {
const repoPath = this.commit.repoPath;
try {
const file = path.resolve(repoPath, this.commit.fileName);
return await commands.executeCommand(BuiltInCommands.Open, Uri.file(file));
const uri = this.getUri();
return await commands.executeCommand(BuiltInCommands.Open, uri);
}
catch (ex) {
return undefined;
@ -74,6 +118,45 @@ export class OpenFileCommandQuickPickItem extends CommandQuickPickItem {
}
}
export class OpenCommitFileCommandQuickPickItem extends OpenFileCommandQuickPickItem {
constructor(commit: GitCommit, item?: PartialQuickPickItem) {
item = {
...{
label: `$(file-symlink-file) Open File`,
description: `\u00a0 \u2014 \u00a0\u00a0 ${path.basename(commit.fileName)} \u00a0\u2022\u00a0 ${path.dirname(commit.fileName)}`
},
...item
};
super(commit.fileName, commit.repoPath, item as QuickPickItem);
}
}
const statusOcticons = [
'\u00a0$(question)',
'\u00a0$(diff-ignored)',
'\u00a0$(diff-added)',
'\u00a0$(diff-modified)',
'\u00a0$(diff-removed)',
'\u00a0$(diff-renamed)'
];
export class OpenStatusFileCommandQuickPickItem extends OpenFileCommandQuickPickItem {
constructor(status: GitFileStatusItem, item?: PartialQuickPickItem) {
item = {
...{
label: `${status.staged ? '$(check)' : '\u00a0\u00a0\u00a0'}\u00a0${statusOcticons[status.status]}\u00a0\u00a0\u00a0${path.basename(status.fileName)}`,
description: path.dirname(status.fileName)
},
...item
};
super(status.fileName, status.repoPath, item as QuickPickItem);
}
}
export class CommitQuickPickItem implements QuickPickItem {
label: string;
@ -97,28 +180,13 @@ export class FileQuickPickItem implements QuickPickItem {
uri: GitUri;
constructor(commit: GitCommit, public fileName: string) {
this.label = fileName;
this.label = path.basename(fileName);
this.description = path.dirname(fileName);
this.sha = commit.sha;
this.uri = GitUri.fromUri(Uri.file(path.resolve(commit.repoPath, fileName)));
}
async open(): Promise<TextEditor | undefined> {
let document = workspace.textDocuments.find(_ => _.fileName === this.uri.fsPath);
const existing = !!document;
try {
if (!document) {
document = await workspace.openTextDocument(this.uri);
}
const editor = await window.showTextDocument(document, 1, true);
return existing ? undefined : editor;
}
catch (ex) {
return undefined;
}
}
async preview(): Promise<{}> {
try {
return await commands.executeCommand(BuiltInCommands.Open, this.uri);

+ 54
- 14
src/commands/quickPicks.ts Переглянути файл

@ -2,9 +2,9 @@
import { Iterables } from '../system';
import { QuickPickOptions, Uri, window, workspace } from 'vscode';
import { IAdvancedConfig } from '../configuration';
import { Commands } from '../constants';
import GitProvider, { GitCommit, GitUri, IGitLog } from '../gitProvider';
import { CommandQuickPickItem, CommitQuickPickItem, FileQuickPickItem, OpenFileCommandQuickPickItem, OpenFilesCommandQuickPickItem } from './quickPickItems';
import { Commands } from '../commands';
import GitProvider, { GitCommit, GitFileStatus, GitFileStatusItem, GitUri, IGitLog } from '../gitProvider';
import { CommandQuickPickItem, CommitQuickPickItem, FileQuickPickItem, OpenCommitFileCommandQuickPickItem, OpenStatusFileCommandQuickPickItem, OpenCommitFilesCommandQuickPickItem, OpenStatusFilesCommandQuickPickItem } from './quickPickItems';
import * as moment from 'moment';
import * as path from 'path';
@ -17,6 +17,8 @@ export class CommitQuickPick {
static async show(git: GitProvider, commit: GitCommit, workingFileName: string, uri: Uri, currentCommand?: CommandQuickPickItem, goBackCommand?: CommandQuickPickItem, options: { showFileHistory?: boolean } = {}): Promise<CommandQuickPickItem | undefined> {
const items: CommandQuickPickItem[] = [];
const workingName = (workingFileName && path.basename(workingFileName)) || path.basename(commit.fileName);
const isUncommitted = commit.isUncommitted;
if (isUncommitted) {
// Since we can't trust the previous sha on an uncommitted commit, find the last commit for this file
@ -35,10 +37,10 @@ export class CommitQuickPick {
items.push(new CommandQuickPickItem({
label: `$(git-compare) Compare with Working Tree`,
description: `\u00a0 \u2014 \u00a0\u00a0 $(git-commit) ${commit.sha} \u00a0 $(git-compare) \u00a0 $(file-text) ${workingFileName || commit.fileName}`
description: `\u00a0 \u2014 \u00a0\u00a0 $(git-commit) ${commit.sha} \u00a0 $(git-compare) \u00a0 $(file-text) ${workingName}`
}, Commands.DiffWithWorking, [uri, commit]));
items.push(new OpenFileCommandQuickPickItem(commit));
items.push(new OpenCommitFileCommandQuickPickItem(commit));
items.push(new CommandQuickPickItem({
label: `$(clippy) Copy Commit Sha to Clipboard`,
@ -52,24 +54,22 @@ export class CommitQuickPick {
items.push(new CommandQuickPickItem({
label: `$(diff) Show Changed Files`,
description: `\u00a0 \u2014 \u00a0\u00a0 $(git-commit) ${commit.sha}`,
detail: `Shows all the changed files in commit $(git-commit) ${commit.sha}`
description: undefined, //`\u00a0 \u2014 \u00a0\u00a0 $(git-commit) ${commit.sha}`,
detail: `Shows all of the changed files in commit $(git-commit) ${commit.sha}`
}, Commands.ShowQuickCommitDetails, [new GitUri(commit.uri, commit), commit.sha, undefined, currentCommand]));
if (options.showFileHistory) {
const fileName = path.basename(commit.fileName);
fileName;
if (workingFileName) {
items.push(new CommandQuickPickItem({
label: `$(versions) Show Commit History`,
description: `\u00a0 \u2014 \u00a0\u00a0 ${commit.fileName}`,
description: undefined, //`\u00a0 \u2014 \u00a0\u00a0 ${path.basename(commit.fileName)}`,
detail: `Shows the commit history of the file, starting at the most recent commit`
}, Commands.ShowQuickFileHistory, [commit.uri, undefined, undefined, currentCommand]));
}
items.push(new CommandQuickPickItem({
label: `$(versions) Show Previous Commit History`,
description: `\u00a0 \u2014 \u00a0\u00a0 ${commit.fileName}`,
description: undefined, //`\u00a0 \u2014 \u00a0\u00a0 ${path.basename(commit.fileName)}`,
detail: `Shows the previous commit history of the file, starting at $(git-commit) ${commit.sha}`
}, Commands.ShowQuickFileHistory, [new GitUri(commit.uri, commit), undefined, undefined, currentCommand]));
}
@ -80,7 +80,7 @@ export class CommitQuickPick {
return await window.showQuickPick(items, {
matchOnDescription: true,
placeHolder: `${commit.fileName} \u2022 ${isUncommitted ? 'Uncommitted \u21E8 ' : '' }${commit.sha} \u2022 ${commit.author}, ${moment(commit.date).fromNow()} \u2022 ${commit.message}`,
placeHolder: `${path.basename(commit.fileName)} \u00a0\u2022\u00a0 ${path.dirname(commit.fileName)} \u2022 ${isUncommitted ? 'Uncommitted \u21E8 ' : '' }${commit.sha} \u2022 ${commit.author}, ${moment(commit.date).fromNow()} \u2022 ${commit.message}`,
ignoreFocusOut: getQuickPickIgnoreFocusOut()
} as QuickPickOptions);
}
@ -92,7 +92,7 @@ export class CommitFilesQuickPick {
const fileNames = commit.fileName.split(', ').filter(_ => !!_);
const items: (FileQuickPickItem | CommandQuickPickItem)[] = fileNames.map(f => new FileQuickPickItem(commit, f));
items.splice(0, 0, new OpenFilesCommandQuickPickItem(commit, fileNames));
items.splice(0, 0, new OpenCommitFilesCommandQuickPickItem(commit, fileNames));
if (goBackCommand) {
items.splice(0, 0, goBackCommand);
@ -143,10 +143,12 @@ export class FileCommitsQuickPick {
items.splice(0, 0, goBackCommand);
}
const fileName = Iterables.first(log.commits.values()).fileName;
return await window.showQuickPick(items, {
matchOnDescription: true,
matchOnDetail: true,
placeHolder: `${Iterables.first(log.commits.values()).fileName}`,
placeHolder: `${path.basename(fileName)} \u00a0\u2022\u00a0 ${path.dirname(fileName)}`,
ignoreFocusOut: getQuickPickIgnoreFocusOut()
} as QuickPickOptions);
}
@ -176,4 +178,42 @@ export class RepoCommitsQuickPick {
ignoreFocusOut: getQuickPickIgnoreFocusOut()
} as QuickPickOptions);
}
}
export class RepoStatusesQuickPick {
static async show(statuses: GitFileStatusItem[], goBackCommand?: CommandQuickPickItem): Promise<CommitQuickPickItem | CommandQuickPickItem | undefined> {
// Sort the status by staged and then filename
statuses.sort((a, b) => (a.staged ? -1 : 1) - (b.staged ? -1 : 1) || a.fileName.localeCompare(b.fileName));
const items = Array.from(Iterables.map(statuses, s => new OpenStatusFileCommandQuickPickItem(s))) as (OpenStatusFileCommandQuickPickItem | CommandQuickPickItem)[];
if (statuses.some(_ => _.staged)) {
const index = statuses.findIndex(_ => !_.staged);
if (index > -1) {
items.splice(index, 0, new OpenStatusFilesCommandQuickPickItem(statuses.filter(_ => _.status !== GitFileStatus.Deleted && !_.staged), {
label: `$(file-symlink-file) Open Unstaged Files`,
description: undefined,
detail: `Opens all of the unstaged files in the repository`
}));
items.splice(0, 0, new OpenStatusFilesCommandQuickPickItem(statuses.filter(_ => _.status !== GitFileStatus.Deleted && _.staged), {
label: `$(file-symlink-file) Open Staged Files`,
description: undefined,
detail: `Opens all of the staged files in the repository`
}));
}
}
items.splice(0, 0, new OpenStatusFilesCommandQuickPickItem(statuses.filter(_ => _.status !== GitFileStatus.Deleted)));
if (goBackCommand) {
items.splice(0, 0, goBackCommand);
}
return await window.showQuickPick(items, {
matchOnDescription: true,
placeHolder: 'Showing the repository status',
ignoreFocusOut: getQuickPickIgnoreFocusOut()
} as QuickPickOptions);
}
}

+ 61
- 0
src/commands/showQuickRepoStatus.ts Переглянути файл

@ -0,0 +1,61 @@
'use strict';
import { TextEditor, Uri, window } from 'vscode';
import { ActiveEditorCommand, Commands } from '../commands';
import GitProvider, { GitUri } from '../gitProvider';
import { Logger } from '../logger';
import { CommandQuickPickItem } from './quickPickItems';
import { RepoStatusesQuickPick } from './quickPicks';
export default class ShowQuickRepoStatusCommand extends ActiveEditorCommand {
constructor(private git: GitProvider, public repoPath: string) {
super(Commands.ShowQuickRepoStatus);
}
async execute(editor: TextEditor, uri?: Uri, goBackCommand?: CommandQuickPickItem) {
if (!(uri instanceof Uri)) {
uri = editor && editor.document && editor.document.uri;
}
try {
let repoPath: string;
if (uri instanceof Uri) {
const gitUri = GitUri.fromUri(uri, this.git);
repoPath = gitUri.repoPath;
if (!repoPath) {
repoPath = await this.git.getRepoPathFromFile(gitUri.fsPath);
}
}
if (!repoPath) {
repoPath = this.repoPath;
}
if (!repoPath) return window.showWarningMessage(`Unable to show repository status`);
const statuses = await this.git.getStatusesForRepo(repoPath);
if (!statuses) return window.showWarningMessage(`Unable to show repository status`);
const pick = await RepoStatusesQuickPick.show(statuses, goBackCommand);
if (!pick) return undefined;
if (pick instanceof CommandQuickPickItem) {
return pick.execute();
}
// commit = pick.commit;
// return commands.executeCommand(Commands.ShowQuickCommitDetails,
// new GitUri(commit.uri, commit),
// commit.sha, undefined,
// new CommandQuickPickItem({
// label: `go back \u21A9`,
// description: null
// }, Commands.ShowQuickRepoHistory, [uri, maxCount, undefined, goBackCommand]));
}
catch (ex) {
Logger.error('[GitLens.ShowQuickRepoStatusCommand]', ex);
return window.showErrorMessage(`Unable to show repository status. See output channel for more details`);
}
}
}

+ 2
- 0
src/extension.ts Переглянути файл

@ -15,6 +15,7 @@ import ShowFileHistoryCommand from './commands/showFileHistory';
import ShowQuickCommitDetailsCommand from './commands/showQuickCommitDetails';
import ShowQuickFileHistoryCommand from './commands/showQuickFileHistory';
import ShowQuickRepoHistoryCommand from './commands/showQuickRepoHistory';
import ShowQuickRepoStatusCommand from './commands/showQuickRepoStatus';
import ToggleBlameCommand from './commands/toggleBlame';
import ToggleCodeLensCommand from './commands/toggleCodeLens';
import { IAdvancedConfig, IBlameConfig } from './configuration';
@ -91,6 +92,7 @@ export async function activate(context: ExtensionContext) {
context.subscriptions.push(new ShowQuickCommitDetailsCommand(git));
context.subscriptions.push(new ShowQuickFileHistoryCommand(git));
context.subscriptions.push(new ShowQuickRepoHistoryCommand(git, repoPath));
context.subscriptions.push(new ShowQuickRepoStatusCommand(git, repoPath));
context.subscriptions.push(new ToggleCodeLensCommand(git));
}

+ 6
- 1
src/git/git.ts Переглянути файл

@ -171,13 +171,18 @@ export default class Git {
return gitCommand(root, 'show', `${sha}:./${file}`);
}
static statusForFile(fileName: string, repoPath: string): Promise<string> {
static statusFile(fileName: string, repoPath: string): Promise<string> {
const [file, root]: [string, string] = Git.splitPath(Git.normalizePath(fileName), repoPath);
const params = ['status', file, '--short'];
return gitCommand(root, ...params);
}
static statusRepo(repoPath: string): Promise<string> {
const params = ['status', '--short'];
return gitCommand(repoPath, ...params);
}
static isUncommitted(sha: string) {
return UncommittedRegex.test(sha);
}

+ 55
- 0
src/git/gitEnrichment.ts Переглянути файл

@ -106,4 +106,59 @@ export interface IGitLog {
repoPath: string;
authors: Map<string, IGitAuthor>;
commits: Map<string, GitCommit>;
}
export enum GitFileStatus {
Unknown,
Untracked,
Added,
Modified,
Deleted,
Renamed
}
export class GitFileStatusItem {
staged: boolean;
status: GitFileStatus;
fileName: string;
constructor(public repoPath: string, status: string) {
this.fileName = status.substring(3);
this.parseStatus(status);
}
private parseStatus(status: string) {
const indexStatus = status[0];
const workTreeStatus = status[1];
this.staged = workTreeStatus === ' ';
if (indexStatus === '?' && workTreeStatus === '?') {
this.status = GitFileStatus.Untracked;
return;
}
if (indexStatus === 'A') {
this.status = GitFileStatus.Added;
return;
}
if (indexStatus === 'M' || workTreeStatus === 'M') {
this.status = GitFileStatus.Modified;
return;
}
if (indexStatus === 'D' || workTreeStatus === 'D') {
this.status = GitFileStatus.Deleted;
return;
}
if (indexStatus === 'R') {
this.status = GitFileStatus.Renamed;
return;
}
this.status = GitFileStatus.Unknown;
}
}

+ 13
- 6
src/gitProvider.ts Переглянути файл

@ -3,7 +3,7 @@ import { Iterables, Objects } from './system';
import { Disposable, Event, EventEmitter, ExtensionContext, FileSystemWatcher, languages, Location, Position, Range, TextDocument, TextEditor, Uri, workspace } from 'vscode';
import { CodeLensVisibility, IConfig } from './configuration';
import { DocumentSchemes, WorkspaceState } from './constants';
import Git, { GitBlameParserEnricher, GitBlameFormat, GitCommit, GitLogParserEnricher, IGitAuthor, IGitBlame, IGitBlameLine, IGitBlameLines, IGitLog } from './git/git';
import Git, { GitBlameParserEnricher, GitBlameFormat, GitCommit, GitFileStatusItem, GitLogParserEnricher, IGitAuthor, IGitBlame, IGitBlameLine, IGitBlameLines, IGitLog } from './git/git';
import { IGitUriData, GitUri } from './git/gitUri';
import GitCodeLensProvider from './gitCodeLensProvider';
import { Logger } from './logger';
@ -559,15 +559,22 @@ export default class GitProvider extends Disposable {
return locations;
}
async getStatusForFile(fileName: string, repoPath: string) {
Logger.log(`getStatusForFile('${fileName}', ${repoPath})`);
return (await Git.statusForFile(fileName, repoPath)).trim();
async getStatusForFile(fileName: string, repoPath: string): Promise<GitFileStatusItem> {
Logger.log(`getStatusForFile('${fileName}', '${repoPath}')`);
const status = await Git.statusFile(fileName, repoPath);
return status && status.trim().length && new GitFileStatusItem(repoPath, status);
}
async isFileUncommitted(fileName: string, repoPath: string) {
async getStatusesForRepo(repoPath: string): Promise<GitFileStatusItem[]> {
Logger.log(`getStatusForRepo('${repoPath}')`);
const statuses = (await Git.statusRepo(repoPath)).split('\n').filter(_ => !!_);
return statuses.map(_ => new GitFileStatusItem(repoPath, _));
}
async isFileUncommitted(fileName: string, repoPath: string): Promise<boolean> {
Logger.log(`isFileUncommitted('${fileName}', ${repoPath})`);
const status = await this.getStatusForFile(fileName, repoPath);
return status && status.length;
return !!status;
}
async getVersionedFile(fileName: string, repoPath: string, sha: string) {

Завантаження…
Відмінити
Зберегти