mirror of
https://github.com/microsoft/vscode.git
synced 2026-02-24 11:47:43 +00:00
382 lines
13 KiB
TypeScript
382 lines
13 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import { CancellationToken, ConfigurationChangeEvent, Disposable, env, Event, EventEmitter, MarkdownString, ThemeIcon, Timeline, TimelineChangeEvent, TimelineItem, TimelineOptions, TimelineProvider, Uri, workspace, l10n, Command } from 'vscode';
|
|
import { Model } from './model';
|
|
import { Repository, Resource } from './repository';
|
|
import { debounce } from './decorators';
|
|
import { emojify, ensureEmojis } from './emoji';
|
|
import { CommandCenter } from './commands';
|
|
import { OperationKind, OperationResult } from './operation';
|
|
import { getCommitShortHash } from './util';
|
|
import { CommitShortStat } from './git';
|
|
import { provideSourceControlHistoryItemAvatar, provideSourceControlHistoryItemHoverCommands, provideSourceControlHistoryItemMessageLinks } from './historyItemDetailsProvider';
|
|
import { AvatarQuery, AvatarQueryCommit } from './api/git';
|
|
|
|
const AVATAR_SIZE = 20;
|
|
|
|
export class GitTimelineItem extends TimelineItem {
|
|
static is(item: TimelineItem): item is GitTimelineItem {
|
|
return item instanceof GitTimelineItem;
|
|
}
|
|
|
|
readonly ref: string;
|
|
readonly previousRef: string;
|
|
readonly message: string;
|
|
|
|
constructor(
|
|
ref: string,
|
|
previousRef: string,
|
|
message: string,
|
|
timestamp: number,
|
|
id: string,
|
|
contextValue: string
|
|
) {
|
|
const index = message.indexOf('\n');
|
|
const label = index !== -1 ? `${message.substring(0, index)} \u2026` : message;
|
|
|
|
super(label, timestamp);
|
|
|
|
this.ref = ref;
|
|
this.previousRef = previousRef;
|
|
this.message = message;
|
|
this.id = id;
|
|
this.contextValue = contextValue;
|
|
}
|
|
|
|
get shortRef() {
|
|
return this.shortenRef(this.ref);
|
|
}
|
|
|
|
get shortPreviousRef() {
|
|
return this.shortenRef(this.previousRef);
|
|
}
|
|
|
|
setItemDetails(uri: Uri, hash: string | undefined, avatar: string | undefined, author: string, email: string | undefined, date: string, message: string, shortStat?: CommitShortStat, remoteSourceCommands: Command[] = []): void {
|
|
this.tooltip = new MarkdownString('', true);
|
|
this.tooltip.isTrusted = true;
|
|
this.tooltip.supportHtml = true;
|
|
|
|
const avatarMarkdown = avatar
|
|
? ``
|
|
: '$(account)';
|
|
|
|
if (email) {
|
|
const emailTitle = l10n.t('Email');
|
|
this.tooltip.appendMarkdown(`${avatarMarkdown} [**${author}**](mailto:${email} "${emailTitle} ${author}")`);
|
|
} else {
|
|
this.tooltip.appendMarkdown(`${avatarMarkdown} **${author}**`);
|
|
}
|
|
|
|
this.tooltip.appendMarkdown(`, $(history) ${date}\n\n`);
|
|
this.tooltip.appendMarkdown(`${message}\n\n`);
|
|
|
|
if (shortStat) {
|
|
this.tooltip.appendMarkdown(`---\n\n`);
|
|
|
|
const labels: string[] = [];
|
|
if (shortStat.insertions) {
|
|
labels.push(`<span style="color:var(--vscode-scmGraph-historyItemHoverAdditionsForeground);">${shortStat.insertions === 1 ?
|
|
l10n.t('{0} insertion{1}', shortStat.insertions, '(+)') :
|
|
l10n.t('{0} insertions{1}', shortStat.insertions, '(+)')}</span>`);
|
|
}
|
|
|
|
if (shortStat.deletions) {
|
|
labels.push(`<span style="color:var(--vscode-scmGraph-historyItemHoverDeletionsForeground);">${shortStat.deletions === 1 ?
|
|
l10n.t('{0} deletion{1}', shortStat.deletions, '(-)') :
|
|
l10n.t('{0} deletions{1}', shortStat.deletions, '(-)')}</span>`);
|
|
}
|
|
|
|
this.tooltip.appendMarkdown(`${labels.join(', ')}\n\n`);
|
|
}
|
|
|
|
if (hash) {
|
|
this.tooltip.appendMarkdown(`---\n\n`);
|
|
|
|
this.tooltip.appendMarkdown(`[\`$(git-commit) ${getCommitShortHash(uri, hash)} \`](command:git.viewCommit?${encodeURIComponent(JSON.stringify([uri, hash]))} "${l10n.t('Open Commit')}")`);
|
|
this.tooltip.appendMarkdown(' ');
|
|
this.tooltip.appendMarkdown(`[$(copy)](command:git.copyContentToClipboard?${encodeURIComponent(JSON.stringify(hash))} "${l10n.t('Copy Commit Hash')}")`);
|
|
|
|
// Remote commands
|
|
if (remoteSourceCommands.length > 0) {
|
|
this.tooltip.appendMarkdown(' | ');
|
|
|
|
const remoteCommandsMarkdown = remoteSourceCommands
|
|
.map(command => `[${command.title}](command:${command.command}?${encodeURIComponent(JSON.stringify([...command.arguments ?? [], hash]))} "${command.tooltip}")`);
|
|
this.tooltip.appendMarkdown(remoteCommandsMarkdown.join(' '));
|
|
}
|
|
}
|
|
}
|
|
|
|
private shortenRef(ref: string): string {
|
|
if (ref === '' || ref === '~' || ref === 'HEAD') {
|
|
return ref;
|
|
}
|
|
return ref.endsWith('^') ? `${ref.substr(0, 8)}^` : ref.substr(0, 8);
|
|
}
|
|
}
|
|
|
|
export class GitTimelineProvider implements TimelineProvider {
|
|
private _onDidChange = new EventEmitter<TimelineChangeEvent | undefined>();
|
|
get onDidChange(): Event<TimelineChangeEvent | undefined> {
|
|
return this._onDidChange.event;
|
|
}
|
|
|
|
readonly id = 'git-history';
|
|
readonly label = l10n.t('Git History');
|
|
|
|
private readonly disposable: Disposable;
|
|
private providerDisposable: Disposable | undefined;
|
|
|
|
private repo: Repository | undefined;
|
|
private repoDisposable: Disposable | undefined;
|
|
private repoOperationDate: Date | undefined;
|
|
|
|
constructor(private readonly model: Model, private commands: CommandCenter) {
|
|
this.disposable = Disposable.from(
|
|
model.onDidOpenRepository(this.onRepositoriesChanged, this),
|
|
workspace.onDidChangeConfiguration(this.onConfigurationChanged, this)
|
|
);
|
|
|
|
if (model.repositories.length) {
|
|
this.ensureProviderRegistration();
|
|
}
|
|
}
|
|
|
|
dispose() {
|
|
this.providerDisposable?.dispose();
|
|
this.disposable.dispose();
|
|
}
|
|
|
|
async provideTimeline(uri: Uri, options: TimelineOptions, _token: CancellationToken): Promise<Timeline> {
|
|
// console.log(`GitTimelineProvider.provideTimeline: uri=${uri}`);
|
|
|
|
const repo = this.model.getRepository(uri);
|
|
if (!repo) {
|
|
this.repoDisposable?.dispose();
|
|
this.repoOperationDate = undefined;
|
|
this.repo = undefined;
|
|
|
|
return { items: [] };
|
|
}
|
|
|
|
if (this.repo?.root !== repo.root) {
|
|
this.repoDisposable?.dispose();
|
|
|
|
this.repo = repo;
|
|
this.repoOperationDate = new Date();
|
|
this.repoDisposable = Disposable.from(
|
|
repo.onDidChangeRepository(uri => this.onRepositoryChanged(repo, uri)),
|
|
repo.onDidRunGitStatus(() => this.onRepositoryStatusChanged(repo)),
|
|
repo.onDidRunOperation(result => this.onRepositoryOperationRun(repo, result))
|
|
);
|
|
}
|
|
|
|
const config = workspace.getConfiguration('git.timeline');
|
|
|
|
// TODO@eamodio: Ensure that the uri is a file -- if not we could get the history of the repo?
|
|
|
|
let limit: number | undefined;
|
|
if (options.limit !== undefined && typeof options.limit !== 'number') {
|
|
try {
|
|
const result = await this.model.git.exec(repo.root, ['rev-list', '--count', `${options.limit.id}..`, '--', uri.fsPath]);
|
|
if (!result.exitCode) {
|
|
// Ask for 2 more (1 for the limit commit and 1 for the next commit) than so we can determine if there are more commits
|
|
limit = Number(result.stdout) + 2;
|
|
}
|
|
}
|
|
catch {
|
|
limit = undefined;
|
|
}
|
|
} else {
|
|
// If we are not getting everything, ask for 1 more than so we can determine if there are more commits
|
|
limit = options.limit === undefined ? undefined : options.limit + 1;
|
|
}
|
|
|
|
await ensureEmojis();
|
|
|
|
const commits = await repo.logFile(uri, {
|
|
maxEntries: limit,
|
|
hash: options.cursor,
|
|
follow: true,
|
|
shortStats: true,
|
|
// sortByAuthorDate: true
|
|
});
|
|
|
|
const paging = commits.length ? {
|
|
cursor: limit === undefined ? undefined : (commits.length >= limit ? commits[commits.length - 1]?.hash : undefined)
|
|
} : undefined;
|
|
|
|
// If we asked for an extra commit, strip it off
|
|
if (limit !== undefined && commits.length >= limit) {
|
|
commits.splice(commits.length - 1, 1);
|
|
}
|
|
|
|
const dateFormatter = new Intl.DateTimeFormat(env.language, { year: 'numeric', month: 'long', day: 'numeric', hour: 'numeric', minute: 'numeric' });
|
|
|
|
const dateType = config.get<'committed' | 'authored'>('date');
|
|
const showAuthor = config.get<boolean>('showAuthor');
|
|
const showUncommitted = config.get<boolean>('showUncommitted');
|
|
|
|
const openComparison = l10n.t('Open Comparison');
|
|
|
|
const emptyTree = await repo.getEmptyTree();
|
|
const unpublishedCommits = await repo.getUnpublishedCommits();
|
|
const remoteHoverCommands = await provideSourceControlHistoryItemHoverCommands(this.model, repo);
|
|
|
|
const avatarQuery = {
|
|
commits: commits.map(c => ({
|
|
hash: c.hash,
|
|
authorName: c.authorName,
|
|
authorEmail: c.authorEmail
|
|
}) satisfies AvatarQueryCommit),
|
|
size: 20
|
|
} satisfies AvatarQuery;
|
|
const avatars = await provideSourceControlHistoryItemAvatar(this.model, repo, avatarQuery);
|
|
|
|
const items: GitTimelineItem[] = [];
|
|
for (let index = 0; index < commits.length; index++) {
|
|
const c = commits[index];
|
|
|
|
const date = dateType === 'authored' ? c.authorDate : c.commitDate;
|
|
|
|
const message = emojify(c.message);
|
|
|
|
const previousRef = commits[index + 1]?.hash ?? emptyTree;
|
|
const item = new GitTimelineItem(c.hash, previousRef, message, date?.getTime() ?? 0, c.hash, 'git:file:commit');
|
|
item.iconPath = new ThemeIcon('git-commit');
|
|
if (showAuthor) {
|
|
item.description = c.authorName;
|
|
}
|
|
|
|
const commitRemoteSourceCommands = !unpublishedCommits.has(c.hash) ? remoteHoverCommands : [];
|
|
const messageWithLinks = await provideSourceControlHistoryItemMessageLinks(this.model, repo, message) ?? message;
|
|
|
|
item.setItemDetails(uri, c.hash, avatars?.get(c.hash), c.authorName!, c.authorEmail, dateFormatter.format(date), messageWithLinks, c.shortStat, commitRemoteSourceCommands);
|
|
|
|
const cmd = this.commands.resolveTimelineOpenDiffCommand(item, uri);
|
|
if (cmd) {
|
|
item.command = {
|
|
title: openComparison,
|
|
command: cmd.command,
|
|
arguments: cmd.arguments,
|
|
};
|
|
}
|
|
|
|
items.push(item);
|
|
}
|
|
|
|
if (options.cursor === undefined) {
|
|
const you = l10n.t('You');
|
|
|
|
const index = repo.indexGroup.resourceStates.find(r => r.resourceUri.fsPath === uri.fsPath);
|
|
if (index) {
|
|
const date = this.repoOperationDate ?? new Date();
|
|
|
|
const item = new GitTimelineItem('~', 'HEAD', l10n.t('Staged Changes'), date.getTime(), 'index', 'git:file:index');
|
|
// TODO@eamodio: Replace with a better icon -- reflecting its status maybe?
|
|
item.iconPath = new ThemeIcon('git-commit');
|
|
item.description = '';
|
|
item.setItemDetails(uri, undefined, undefined, you, undefined, dateFormatter.format(date), Resource.getStatusText(index.type));
|
|
|
|
const cmd = this.commands.resolveTimelineOpenDiffCommand(item, uri);
|
|
if (cmd) {
|
|
item.command = {
|
|
title: openComparison,
|
|
command: cmd.command,
|
|
arguments: cmd.arguments,
|
|
};
|
|
}
|
|
|
|
items.splice(0, 0, item);
|
|
}
|
|
|
|
if (showUncommitted) {
|
|
const working = repo.workingTreeGroup.resourceStates.find(r => r.resourceUri.fsPath === uri.fsPath);
|
|
if (working) {
|
|
const date = new Date();
|
|
|
|
const item = new GitTimelineItem('', index ? '~' : 'HEAD', l10n.t('Uncommitted Changes'), date.getTime(), 'working', 'git:file:working');
|
|
item.iconPath = new ThemeIcon('circle-outline');
|
|
item.description = '';
|
|
item.setItemDetails(uri, undefined, undefined, you, undefined, dateFormatter.format(date), Resource.getStatusText(working.type));
|
|
|
|
const cmd = this.commands.resolveTimelineOpenDiffCommand(item, uri);
|
|
if (cmd) {
|
|
item.command = {
|
|
title: openComparison,
|
|
command: cmd.command,
|
|
arguments: cmd.arguments,
|
|
};
|
|
}
|
|
|
|
items.splice(0, 0, item);
|
|
}
|
|
}
|
|
}
|
|
|
|
return {
|
|
items: items,
|
|
paging: paging
|
|
};
|
|
}
|
|
|
|
private ensureProviderRegistration() {
|
|
if (this.providerDisposable === undefined) {
|
|
this.providerDisposable = workspace.registerTimelineProvider(['file', 'git', 'vscode-remote', 'vscode-local-history'], this);
|
|
}
|
|
}
|
|
|
|
private onConfigurationChanged(e: ConfigurationChangeEvent) {
|
|
if (e.affectsConfiguration('git.timeline.date') || e.affectsConfiguration('git.timeline.showAuthor') || e.affectsConfiguration('git.timeline.showUncommitted')) {
|
|
this.fireChanged();
|
|
}
|
|
}
|
|
|
|
private onRepositoriesChanged(_repo: Repository) {
|
|
// console.log(`GitTimelineProvider.onRepositoriesChanged`);
|
|
|
|
this.ensureProviderRegistration();
|
|
|
|
// TODO@eamodio: Being naive for now and just always refreshing each time there is a new repository
|
|
this.fireChanged();
|
|
}
|
|
|
|
private onRepositoryChanged(_repo: Repository, _uri: Uri) {
|
|
// console.log(`GitTimelineProvider.onRepositoryChanged: uri=${uri.toString(true)}`);
|
|
|
|
this.fireChanged();
|
|
}
|
|
|
|
private onRepositoryStatusChanged(_repo: Repository) {
|
|
// console.log(`GitTimelineProvider.onRepositoryStatusChanged`);
|
|
|
|
const config = workspace.getConfiguration('git.timeline');
|
|
const showUncommitted = config.get<boolean>('showUncommitted') === true;
|
|
|
|
if (showUncommitted) {
|
|
this.fireChanged();
|
|
}
|
|
}
|
|
|
|
private onRepositoryOperationRun(_repo: Repository, _result: OperationResult) {
|
|
// console.log(`GitTimelineProvider.onRepositoryOperationRun`);
|
|
|
|
// Successful operations that are not read-only and not status operations
|
|
if (!_result.error && !_result.operation.readOnly && _result.operation.kind !== OperationKind.Status) {
|
|
// This is less than ideal, but for now just save the last time an
|
|
// operation was run and use that as the timestamp for staged items
|
|
this.repoOperationDate = new Date();
|
|
|
|
this.fireChanged();
|
|
}
|
|
}
|
|
|
|
@debounce(500)
|
|
private fireChanged() {
|
|
this._onDidChange.fire(undefined);
|
|
}
|
|
}
|