'use strict'; import { Arrays, Iterables, Strings } from '../system'; import { CancellationTokenSource, QuickPickOptions, Uri, window } from 'vscode'; import { Commands, ShowQuickCurrentBranchHistoryCommandArgs, ShowQuickFileHistoryCommandArgs } from '../commands'; import { CommandQuickPickItem, CommitQuickPickItem, getQuickPickIgnoreFocusOut, showQuickPickProgress } from './common'; import { GlyphChars } from '../constants'; import { GitLog, GitService, GitUri, RemoteResource } from '../gitService'; import { Keyboard, KeyNoopCommand } from '../keyboard'; import { OpenRemotesCommandQuickPickItem } from './remotes'; import * as path from 'path'; export class FileHistoryQuickPick { static showProgress(uri: GitUri) { return showQuickPickProgress(`${uri.getFormattedPath()}${uri.sha ? ` ${Strings.pad(GlyphChars.Dot, 1, 1)} ${uri.shortSha}` : ''}`, { left: KeyNoopCommand, ',': KeyNoopCommand, '.': KeyNoopCommand }); } static async show(git: GitService, log: GitLog, uri: GitUri, progressCancellation: CancellationTokenSource, goBackCommand?: CommandQuickPickItem, nextPageCommand?: CommandQuickPickItem): Promise { const items = Array.from(Iterables.map(log.commits.values(), c => new CommitQuickPickItem(c))) as (CommitQuickPickItem | CommandQuickPickItem)[]; let previousPageCommand: CommandQuickPickItem | undefined = undefined; let index = 0; if (log.truncated || log.sha) { if (log.truncated) { index++; items.splice(0, 0, new CommandQuickPickItem({ label: `$(sync) Show All Commits`, description: `${Strings.pad(GlyphChars.Dash, 2, 3)} this may take a while` }, Commands.ShowQuickFileHistory, [ Uri.file(uri.fsPath), { maxCount: 0, goBackCommand } as ShowQuickFileHistoryCommandArgs ])); } else { const workingFileName = await git.findWorkingFileName(log.repoPath, path.relative(log.repoPath, uri.fsPath)); if (workingFileName) { index++; items.splice(0, 0, new CommandQuickPickItem({ label: `$(history) Show File History`, description: `${Strings.pad(GlyphChars.Dash, 2, 3)} of ${path.basename(workingFileName)}` }, Commands.ShowQuickFileHistory, [ Uri.file(path.resolve(log.repoPath, workingFileName)), { goBackCommand: new CommandQuickPickItem({ label: `go back ${GlyphChars.ArrowBack}`, description: `${Strings.pad(GlyphChars.Dash, 2, 3)} to history of ${GlyphChars.Space}$(file-text) ${path.basename(uri.fsPath)}${uri.sha ? ` from ${GlyphChars.Space}$(git-commit) ${uri.shortSha}` : ''}` }, Commands.ShowQuickFileHistory, [ uri, { log: log, maxCount: log.maxCount, range: log.range, goBackCommand } as ShowQuickFileHistoryCommandArgs ]) } as ShowQuickFileHistoryCommandArgs ])); } } if (nextPageCommand) { index++; items.splice(0, 0, nextPageCommand); } if (log.truncated) { const npc = new CommandQuickPickItem({ label: `$(arrow-right) Show Next Commits`, description: `${Strings.pad(GlyphChars.Dash, 2, 3)} shows ${log.maxCount} newer commits` }, Commands.ShowQuickFileHistory, [ uri, { maxCount: log.maxCount, goBackCommand, nextPageCommand } as ShowQuickFileHistoryCommandArgs ]); const last = Iterables.last(log.commits.values()); if (last != null) { previousPageCommand = new CommandQuickPickItem({ label: `$(arrow-left) Show Previous Commits`, description: `${Strings.pad(GlyphChars.Dash, 2, 3)} shows ${log.maxCount} older commits` }, Commands.ShowQuickFileHistory, [ new GitUri(uri, last), { maxCount: log.maxCount, goBackCommand, nextPageCommand: npc } as ShowQuickFileHistoryCommandArgs ]); index++; items.splice(0, 0, previousPageCommand); } } } const branch = await git.getBranch(uri.repoPath!); const currentCommand = new CommandQuickPickItem({ label: `go back ${GlyphChars.ArrowBack}`, description: `${Strings.pad(GlyphChars.Dash, 2, 3)} to history of ${GlyphChars.Space}$(file-text) ${path.basename(uri.fsPath)}${uri.sha ? ` from ${GlyphChars.Space}$(git-commit) ${uri.shortSha}` : ''}` }, Commands.ShowQuickFileHistory, [ uri, { log, maxCount: log.maxCount, range: log.range } as ShowQuickFileHistoryCommandArgs ]); // Only show the full repo option if we are the root if (goBackCommand === undefined) { items.splice(index++, 0, new CommandQuickPickItem({ label: `$(history) Show Branch History`, description: `${Strings.pad(GlyphChars.Dash, 2, 3)} shows ${GlyphChars.Space}$(git-branch) ${branch!.name} history` }, Commands.ShowQuickCurrentBranchHistory, [ undefined, { goBackCommand: currentCommand } as ShowQuickCurrentBranchHistoryCommandArgs ])); } const remotes = Arrays.uniqueBy(await git.getRemotes(uri.repoPath!), _ => _.url, _ => !!_.provider); if (remotes.length) { items.splice(index++, 0, new OpenRemotesCommandQuickPickItem(remotes, { type: 'file', branch: branch!.name, fileName: uri.getRelativePath(), sha: uri.sha } as RemoteResource, currentCommand)); } if (goBackCommand) { items.splice(0, 0, goBackCommand); } if (progressCancellation.token.isCancellationRequested) return undefined; const scope = await Keyboard.instance.beginScope({ left: goBackCommand, ',': previousPageCommand, '.': nextPageCommand }); const commit = Iterables.first(log.commits.values()); progressCancellation.cancel(); const pick = await window.showQuickPick(items, { matchOnDescription: true, matchOnDetail: true, placeHolder: `${commit.getFormattedPath()}${uri.sha ? ` ${Strings.pad(GlyphChars.Dot, 1, 1)} ${uri.shortSha}` : ''}`, ignoreFocusOut: getQuickPickIgnoreFocusOut() // onDidSelectItem: (item: QuickPickItem) => { // scope.setKeyCommand('right', item); // } } as QuickPickOptions); await scope.dispose(); return pick; } }