mirror of
https://github.com/ckaczor/vscode-gitlens.git
synced 2026-01-14 01:25:43 -05:00
206 lines
9.9 KiB
TypeScript
206 lines
9.9 KiB
TypeScript
'use strict';
|
|
import { Arrays, Iterables } from '../system';
|
|
import { QuickPickItem, QuickPickOptions, Uri, window } from 'vscode';
|
|
import { Commands, Keyboard, KeyNoopCommand } from '../commands';
|
|
import { CommandQuickPickItem, getQuickPickIgnoreFocusOut, KeyCommandQuickPickItem, OpenFileCommandQuickPickItem, OpenFilesCommandQuickPickItem } from './common';
|
|
import { getGitStatusIcon, Git, GitCommit, GitLogCommit, GitService, GitStashCommit, GitStatusFileStatus, GitUri, IGitLog, IGitStatusFile } from '../gitService';
|
|
import { OpenRemotesCommandQuickPickItem } from './remotes';
|
|
import * as moment from 'moment';
|
|
import * as path from 'path';
|
|
|
|
export class CommitWithFileStatusQuickPickItem extends OpenFileCommandQuickPickItem {
|
|
|
|
fileName: string;
|
|
gitUri: GitUri;
|
|
sha: string;
|
|
shortSha: string;
|
|
status: GitStatusFileStatus;
|
|
|
|
constructor(commit: GitCommit, status: IGitStatusFile) {
|
|
const icon = getGitStatusIcon(status.status);
|
|
|
|
let directory = Git.normalizePath(path.dirname(status.fileName));
|
|
if (!directory || directory === '.') {
|
|
directory = undefined;
|
|
}
|
|
|
|
let description = (status.status === 'R' && status.originalFileName)
|
|
? `${directory || ''} \u00a0\u2190\u00a0 ${status.originalFileName}`
|
|
: directory;
|
|
|
|
super(GitService.toGitContentUri(commit.sha, commit.shortSha, status.fileName, commit.repoPath, commit.originalFileName), {
|
|
label: `\u00a0\u00a0\u00a0\u00a0${icon}\u00a0\u00a0 ${path.basename(status.fileName)}`,
|
|
description: description
|
|
});
|
|
|
|
this.fileName = status.fileName;
|
|
this.gitUri = GitUri.fromFileStatus(status, commit.repoPath);
|
|
this.sha = commit.sha;
|
|
this.shortSha = commit.shortSha;
|
|
this.status = status.status;
|
|
}
|
|
}
|
|
|
|
export class OpenCommitFilesCommandQuickPickItem extends OpenFilesCommandQuickPickItem {
|
|
|
|
constructor(commit: GitLogCommit, item?: QuickPickItem) {
|
|
const repoPath = commit.repoPath;
|
|
const uris = commit.fileStatuses.map(_ => GitService.toGitContentUri(commit.sha, commit.shortSha, _.fileName, repoPath, commit.originalFileName));
|
|
super(uris, item || {
|
|
label: `$(file-symlink-file) Open Changed Files`,
|
|
description: `\u00a0 \u2014 \u00a0\u00a0 in \u00a0$(git-commit) ${commit.shortSha}`
|
|
//detail: `Opens all of the changed files in $(git-commit) ${commit.shortSha}`
|
|
});
|
|
}
|
|
}
|
|
|
|
export class OpenCommitWorkingTreeFilesCommandQuickPickItem extends OpenFilesCommandQuickPickItem {
|
|
|
|
constructor(commit: GitLogCommit, versioned: boolean = false, item?: QuickPickItem) {
|
|
const repoPath = commit.repoPath;
|
|
const uris = commit.fileStatuses.map(_ => GitUri.fromFileStatus(_, repoPath));
|
|
super(uris, item || {
|
|
label: `$(file-symlink-file) Open Changed Working Files`,
|
|
description: undefined
|
|
//detail: `Opens all of the changed file in the working tree`
|
|
});
|
|
}
|
|
}
|
|
|
|
export class CommitDetailsQuickPick {
|
|
|
|
static async show(git: GitService, commit: GitLogCommit, uri: Uri, goBackCommand?: CommandQuickPickItem, currentCommand?: CommandQuickPickItem, repoLog?: IGitLog): Promise<CommitWithFileStatusQuickPickItem | CommandQuickPickItem | undefined> {
|
|
const items: (CommitWithFileStatusQuickPickItem | CommandQuickPickItem)[] = commit.fileStatuses.map(fs => new CommitWithFileStatusQuickPickItem(commit, fs));
|
|
|
|
const stash = commit.type === 'stash';
|
|
|
|
let index = 0;
|
|
|
|
if (stash && git.config.insiders) {
|
|
items.splice(index++, 0, new CommandQuickPickItem({
|
|
label: `$(git-pull-request) Apply Stashed Changes`,
|
|
description: `\u00a0 \u2014 \u00a0\u00a0 ${commit.message}`
|
|
}, Commands.StashApply, [commit as GitStashCommit, true, false]));
|
|
|
|
items.splice(index++, 0, new CommandQuickPickItem({
|
|
label: `$(x) Delete Stashed Changes`,
|
|
description: `\u00a0 \u2014 \u00a0\u00a0 ${commit.message}`
|
|
}, Commands.StashDelete, [commit as GitStashCommit, true]));
|
|
}
|
|
|
|
if (!stash) {
|
|
items.splice(index++, 0, new CommandQuickPickItem({
|
|
label: `$(clippy) Copy Commit Sha to Clipboard`,
|
|
description: `\u00a0 \u2014 \u00a0\u00a0 ${commit.shortSha}`
|
|
}, Commands.CopyShaToClipboard, [uri, commit.sha]));
|
|
}
|
|
|
|
items.splice(index++, 0, new CommandQuickPickItem({
|
|
label: `$(clippy) Copy Message to Clipboard`,
|
|
description: `\u00a0 \u2014 \u00a0\u00a0 ${commit.message}`
|
|
}, Commands.CopyMessageToClipboard, [uri, commit.sha, commit.message]));
|
|
|
|
if (!stash) {
|
|
const remotes = Arrays.uniqueBy(await git.getRemotes(commit.repoPath), _ => _.url, _ => !!_.provider);
|
|
if (remotes.length) {
|
|
items.splice(index++, 0, new OpenRemotesCommandQuickPickItem(remotes, 'commit', commit.sha, currentCommand));
|
|
}
|
|
|
|
items.splice(index++, 0, new CommandQuickPickItem({
|
|
label: `$(git-compare) Directory Compare with Previous Commit`,
|
|
description: `\u00a0 \u2014 \u00a0\u00a0 $(git-commit) ${commit.previousShortSha || `${commit.shortSha}^`} \u00a0 $(git-compare) \u00a0 $(git-commit) ${commit.shortSha}`
|
|
}, Commands.DiffDirectory, [commit.uri, commit.previousSha || `${commit.sha}^`, commit.sha]));
|
|
}
|
|
|
|
items.splice(index++, 0, new CommandQuickPickItem({
|
|
label: `$(git-compare) Directory Compare with Working Tree`,
|
|
description: `\u00a0 \u2014 \u00a0\u00a0 $(git-commit) ${commit.shortSha} \u00a0 $(git-compare) \u00a0 $(file-directory) Working Tree`
|
|
}, Commands.DiffDirectory, [uri, commit.sha]));
|
|
|
|
const added = commit.fileStatuses.filter(_ => _.status === 'A' || _.status === '?').length;
|
|
const deleted = commit.fileStatuses.filter(_ => _.status === 'D').length;
|
|
const changed = commit.fileStatuses.filter(_ => _.status !== 'A' && _.status !== '?' && _.status !== 'D').length;
|
|
|
|
items.splice(index++, 0, new CommandQuickPickItem({
|
|
label: `Changed Files`,
|
|
description: `+${added} ~${changed} -${deleted}`
|
|
}, Commands.ShowQuickCommitDetails, [uri, commit.sha, commit, goBackCommand, repoLog]));
|
|
|
|
items.push(new OpenCommitFilesCommandQuickPickItem(commit));
|
|
items.push(new OpenCommitWorkingTreeFilesCommandQuickPickItem(commit));
|
|
|
|
if (goBackCommand) {
|
|
items.splice(0, 0, goBackCommand);
|
|
}
|
|
|
|
let previousCommand: CommandQuickPickItem | (() => Promise<CommandQuickPickItem>);
|
|
let nextCommand: CommandQuickPickItem | (() => Promise<CommandQuickPickItem>);
|
|
if (!stash) {
|
|
// If we have the full history, we are good
|
|
if (repoLog && !repoLog.truncated && !repoLog.sha) {
|
|
previousCommand = commit.previousSha && new KeyCommandQuickPickItem(Commands.ShowQuickCommitDetails, [commit.previousUri, commit.previousSha, undefined, goBackCommand, repoLog]);
|
|
nextCommand = commit.nextSha && new KeyCommandQuickPickItem(Commands.ShowQuickCommitDetails, [commit.nextUri, commit.nextSha, undefined, goBackCommand, repoLog]);
|
|
}
|
|
else {
|
|
previousCommand = async () => {
|
|
let log = repoLog;
|
|
let c = log && log.commits.get(commit.sha);
|
|
|
|
// If we can't find the commit or the previous commit isn't available (since it isn't trustworthy)
|
|
if (!c || !c.previousSha) {
|
|
log = await git.getLogForRepo(commit.repoPath, commit.sha, git.config.advanced.maxQuickHistory);
|
|
c = log && log.commits.get(commit.sha);
|
|
|
|
if (c) {
|
|
// Copy over next info, since it is trustworthy at this point
|
|
c.nextSha = commit.nextSha;
|
|
}
|
|
}
|
|
if (!c || !c.previousSha) return KeyNoopCommand;
|
|
return new KeyCommandQuickPickItem(Commands.ShowQuickCommitDetails, [c.previousUri, c.previousSha, undefined, goBackCommand, log]);
|
|
};
|
|
|
|
nextCommand = async () => {
|
|
let log = repoLog;
|
|
let c = log && log.commits.get(commit.sha);
|
|
|
|
// If we can't find the commit or the next commit isn't available (since it isn't trustworthy)
|
|
if (!c || !c.nextSha) {
|
|
log = undefined;
|
|
c = undefined;
|
|
|
|
// Try to find the next commit
|
|
const nextLog = await git.getLogForRepo(commit.repoPath, commit.sha, 1, true);
|
|
const next = nextLog && Iterables.first(nextLog.commits.values());
|
|
if (next && next.sha !== commit.sha) {
|
|
c = commit;
|
|
c.nextSha = next.sha;
|
|
}
|
|
}
|
|
if (!c || !c.nextSha) return KeyNoopCommand;
|
|
return new KeyCommandQuickPickItem(Commands.ShowQuickCommitDetails, [c.nextUri, c.nextSha, undefined, goBackCommand, log]);
|
|
};
|
|
}
|
|
}
|
|
|
|
const scope = await Keyboard.instance.beginScope({
|
|
left: goBackCommand,
|
|
',': previousCommand,
|
|
'.': nextCommand
|
|
});
|
|
|
|
const pick = await window.showQuickPick(items, {
|
|
matchOnDescription: true,
|
|
matchOnDetail: true,
|
|
placeHolder: `${commit.shortSha} \u00a0\u2022\u00a0 ${commit.author ? `${commit.author}, ` : ''}${moment(commit.date).fromNow()} \u00a0\u2022\u00a0 ${commit.message}`,
|
|
ignoreFocusOut: getQuickPickIgnoreFocusOut(),
|
|
onDidSelectItem: (item: QuickPickItem) => {
|
|
scope.setKeyCommand('right', item);
|
|
}
|
|
} as QuickPickOptions);
|
|
|
|
await scope.dispose();
|
|
|
|
return pick;
|
|
}
|
|
} |