'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 { 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); let nextCommand: CommandQuickPickItem | (() => Promise); 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; } }