mr_item_model.ts 3.1 KB
Newer Older
1 2 3 4 5
import * as vscode from 'vscode';
import { PROGRAMMATIC_COMMANDS } from '../../command_names';
import { createGitLabNewService } from '../../service_factory';
import { ChangedFileItem } from './changed_file_item';
import { ItemModel } from './item_model';
6
import { GqlDiscussion, GqlTextDiffDiscussion } from '../../gitlab/gitlab_new_service';
7
import { handleError } from '../../log';
8
import { UserFriendlyError } from '../../errors/user_friendly_error';
9
import { GitLabCommentThread } from '../../review/gitlab_comment_thread';
10
import { CommentingRangeProvider } from '../../review/commenting_range_provider';
11

12
const isTextDiffDiscussion = (discussion: GqlDiscussion): discussion is GqlTextDiffDiscussion => {
13 14 15 16
  const firstNote = discussion.notes.nodes[0];
  return firstNote?.position?.positionType === 'text';
};

17
export class MrItemModel extends ItemModel {
18
  constructor(readonly mr: RestIssuable, readonly workspace: GitLabWorkspace) {
19 20 21 22 23 24 25 26 27
    super();
  }

  getTreeItem(): vscode.TreeItem {
    const { iid, title, author } = this.mr;
    const item = new vscode.TreeItem(
      `!${iid} · ${title}`,
      vscode.TreeItemCollapsibleState.Collapsed,
    );
28 29 30
    if (author.avatar_url) {
      item.iconPath = vscode.Uri.parse(author.avatar_url);
    }
31 32 33 34
    return item;
  }

  async getChildren(): Promise<vscode.TreeItem[]> {
35 36 37
    const overview = new vscode.TreeItem('Overview');
    overview.iconPath = new vscode.ThemeIcon('note');
    overview.command = {
38
      command: PROGRAMMATIC_COMMANDS.SHOW_RICH_CONTENT,
39
      arguments: [this.mr, this.workspace.uri],
40
      title: 'Show MR Overview',
41
    };
42 43
    const gitlabService = await createGitLabNewService(this.workspace.uri);
    const mrVersion = await gitlabService.getMrDiff(this.mr);
44
    try {
45
      await this.getMrDiscussions(mrVersion);
46
    } catch (e) {
47 48
      handleError(
        new UserFriendlyError(
49 50 51
          `The extension failed to preload discussions on the MR diff.
            It's possible that you've encountered
            https://gitlab.com/gitlab-org/gitlab/-/issues/298827.`,
52 53 54
          e,
        ),
      );
55
    }
56 57 58 59

    const changedFiles = mrVersion.diffs.map(
      d => new ChangedFileItem(this.mr, mrVersion, d, this.workspace),
    );
60
    return [overview, ...changedFiles];
61 62
  }

63
  private async getMrDiscussions(mrVersion: RestMrVersion): Promise<void> {
64 65 66 67 68
    const commentController = vscode.comments.createCommentController(
      this.mr.references.full,
      this.mr.title,
    );

69 70
    commentController.commentingRangeProvider = new CommentingRangeProvider(this.mr, mrVersion);

71
    const gitlabService = await createGitLabNewService(this.workspace.uri);
72

73 74 75
    const discussions = await gitlabService.getDiscussions({
      issuable: this.mr,
    });
76
    const discussionsOnDiff = discussions.filter(isTextDiffDiscussion);
77 78
    const threads = discussionsOnDiff.map(discussion => {
      return GitLabCommentThread.createThread({
79
        commentController,
80
        workspaceFolder: this.workspace.uri,
81
        mr: this.mr,
82
        discussion,
83
        gitlabService,
84
      });
85 86 87
    });
    this.setDisposableChildren([...threads, commentController]);
  }
88
}