|
|
- 'use strict';
- import {
- CancellationToken,
- DecorationOptions,
- Disposable,
- Hover,
- languages,
- Position,
- Range,
- Selection,
- TextDocument,
- TextEditor,
- TextEditorDecorationType,
- TextEditorRevealType,
- } from 'vscode';
- import { AnnotationProviderBase } from './annotationProvider';
- import { FileAnnotationType } from '../configuration';
- import { Container } from '../container';
- import { Decorations } from './fileAnnotationController';
- import { GitDiff, GitLogCommit } from '../git/git';
- import { Hovers } from '../hovers/hovers';
- import { Logger } from '../logger';
- import { log, Strings } from '../system';
- import { GitDocumentState, TrackedDocument } from '../trackers/gitDocumentTracker';
-
- export class GutterChangesAnnotationProvider extends AnnotationProviderBase {
- private state: { commit: GitLogCommit | undefined; diffs: GitDiff[] } | undefined;
- private hoverProviderDisposable: Disposable | undefined;
-
- constructor(editor: TextEditor, trackedDocument: TrackedDocument<GitDocumentState>) {
- super(editor, trackedDocument);
- }
-
- clear() {
- this.state = undefined;
- if (this.hoverProviderDisposable != null) {
- this.hoverProviderDisposable.dispose();
- this.hoverProviderDisposable = undefined;
- }
- super.clear();
- }
-
- selection(_shaOrLine?: string | number): Promise<void> {
- return Promise.resolve();
- }
-
- validate(): Promise<boolean> {
- return Promise.resolve(true);
- }
-
- @log()
- async onProvideAnnotation(shaOrLine?: string | number): Promise<boolean> {
- const cc = Logger.getCorrelationContext();
-
- this.annotationType = FileAnnotationType.Changes;
-
- let ref1 = this.trackedDocument.uri.sha;
- let ref2;
- if (typeof shaOrLine === 'string') {
- if (shaOrLine !== this.trackedDocument.uri.sha) {
- ref2 = `${shaOrLine}^`;
- }
- }
-
- let commit: GitLogCommit | undefined;
-
- let localChanges = ref1 == null && ref2 == null;
- if (localChanges) {
- let ref = await Container.git.getOldestUnpushedRefForFile(
- this.trackedDocument.uri.repoPath!,
- this.trackedDocument.uri.fsPath,
- );
- if (ref != null) {
- ref = `${ref}^`;
- commit = await Container.git.getCommitForFile(
- this.trackedDocument.uri.repoPath,
- this.trackedDocument.uri.fsPath,
- { ref: ref },
- );
- if (commit != null) {
- if (ref2 != null) {
- ref2 = ref;
- } else {
- ref1 = ref;
- ref2 = '';
- }
- } else {
- localChanges = false;
- }
- } else {
- const status = await Container.git.getStatusForFile(
- this.trackedDocument.uri.repoPath!,
- this.trackedDocument.uri.fsPath,
- );
- const commits = await status?.toPsuedoCommits();
- if (commits?.length) {
- commit = await Container.git.getCommitForFile(
- this.trackedDocument.uri.repoPath,
- this.trackedDocument.uri.fsPath,
- );
- ref1 = 'HEAD';
- } else if (this.trackedDocument.dirty) {
- ref1 = 'HEAD';
- } else {
- localChanges = false;
- }
- }
- }
-
- if (!localChanges) {
- commit = await Container.git.getCommitForFile(
- this.trackedDocument.uri.repoPath,
- this.trackedDocument.uri.fsPath,
- {
- ref: ref2 ?? ref1,
- },
- );
- if (commit == null) return false;
-
- if (ref2 != null) {
- ref2 = commit.ref;
- } else {
- ref1 = `${commit.ref}^`;
- ref2 = commit.ref;
- }
- }
-
- const diffs = (
- await Promise.all(
- ref2 == null && this.editor.document.isDirty
- ? [
- Container.git.getDiffForFileContents(
- this.trackedDocument.uri,
- ref1!,
- this.editor.document.getText(),
- ),
- Container.git.getDiffForFile(this.trackedDocument.uri, ref1, ref2),
- ]
- : [Container.git.getDiffForFile(this.trackedDocument.uri, ref1, ref2)],
- )
- ).filter(<T>(d?: T): d is T => Boolean(d));
- if (!diffs?.length) return false;
-
- let start = process.hrtime();
-
- const decorationsMap = new Map<
- string,
- { decorationType: TextEditorDecorationType; rangesOrOptions: DecorationOptions[] }
- >();
-
- let selection: Selection | undefined;
-
- for (const diff of diffs) {
- for (const hunk of diff.hunks) {
- // Subtract 2 because editor lines are 0-based and we will be adding 1 in the first iteration of the loop
- let count = Math.max(hunk.current.position.start - 2, -1);
- let index = -1;
- for (const hunkLine of hunk.lines) {
- index++;
- count++;
-
- if (hunkLine.current?.state === 'unchanged') continue;
-
- const range = this.editor.document.validateRange(
- new Range(new Position(count, 0), new Position(count, Number.MAX_SAFE_INTEGER)),
- );
- if (selection == null) {
- selection = new Selection(range.start, range.end);
- }
-
- let state;
- if (hunkLine.current == null) {
- const previous = hunk.lines[index - 1];
- if (hunkLine.previous != null && (previous == null || previous.current != null)) {
- // Check if there are more deleted lines than added lines show a deleted indicator
- if (hunk.previous.count > hunk.current.count) {
- state = 'removed';
- } else {
- continue;
- }
- } else {
- continue;
- }
- } else if (hunkLine.current?.state === 'added') {
- if (hunkLine.previous?.state === 'removed') {
- state = 'changed';
- } else {
- state = 'added';
- }
- } else if (hunkLine?.current.state === 'removed') {
- // Check if there are more deleted lines than added lines show a deleted indicator
- if (hunk.previous.count > hunk.current.count) {
- state = 'removed';
- } else {
- continue;
- }
- } else {
- state = 'changed';
- }
-
- let decoration = decorationsMap.get(state);
- if (decoration == null) {
- decoration = {
- decorationType: (state === 'added'
- ? Decorations.changesLineAddedAnnotation
- : state === 'removed'
- ? Decorations.changesLineDeletedAnnotation
- : Decorations.changesLineChangedAnnotation)!,
- rangesOrOptions: [{ range: range }],
- };
- decorationsMap.set(state, decoration);
- } else {
- decoration.rangesOrOptions.push({ range: range });
- }
- }
- }
- }
-
- Logger.log(cc, `${Strings.getDurationMilliseconds(start)} ms to compute recent changes annotations`);
-
- if (decorationsMap.size) {
- start = process.hrtime();
-
- this.setDecorations([...decorationsMap.values()]);
-
- Logger.log(cc, `${Strings.getDurationMilliseconds(start)} ms to apply recent changes annotations`);
-
- if (selection != null) {
- this.editor.selection = selection;
- this.editor.revealRange(selection, TextEditorRevealType.InCenterIfOutsideViewport);
- }
- }
-
- this.state = { commit: commit, diffs: diffs };
- this.registerHoverProvider();
- return true;
- }
-
- registerHoverProvider() {
- if (!Container.config.hovers.enabled || !Container.config.hovers.annotations.enabled) {
- return;
- }
-
- this.hoverProviderDisposable = languages.registerHoverProvider(
- { pattern: this.document.uri.fsPath },
- {
- provideHover: (document, position, token) => this.provideHover(document, position, token),
- },
- );
- }
-
- provideHover(document: TextDocument, position: Position, _token: CancellationToken): Hover | undefined {
- if (this.state == null) return undefined;
- if (Container.config.hovers.annotations.over !== 'line' && position.character !== 0) return undefined;
-
- const { commit, diffs } = this.state;
-
- for (const diff of diffs) {
- for (const hunk of diff.hunks) {
- // If we have a "mixed" diff hunk, check if we have more deleted lines than added, to include a trailing line for the deleted indicator
- const hasMoreDeletedLines = hunk.state === 'changed' && hunk.previous.count > hunk.current.count;
- if (
- position.line >= hunk.current.position.start - 1 &&
- position.line <= hunk.current.position.end - (hasMoreDeletedLines ? 0 : 1)
- ) {
- return new Hover(
- Hovers.localChangesMessage(commit, this.trackedDocument.uri, position.line, hunk),
- document.validateRange(
- new Range(
- hunk.current.position.start - 1,
- 0,
- hunk.current.position.end - (hasMoreDeletedLines ? 0 : 1),
- Number.MAX_SAFE_INTEGER,
- ),
- ),
- );
- }
- }
- }
-
- return undefined;
- }
- }
|