Alert This post is over a year old, some of this information may be out of date.

Creating timer dismissable notifications in a Visual Studio Code extension

post

Notifications in Visual Studio Code are trivial for your extensions to notify the user when something happens or a process completes. A downside of these notifications is that they cannot be dismissed programmatically; they require manual user interaction.

As in my current extension, I wanted to be able to have a notification that can do the following:

  1. Show notification with an undo button;
  2. Hide the notification after 10 seconds;
  3. If you click on the undo action, hide the notification needs to be removed.

You might think it should be easy, but unfortunately, it is not the case. In the above requirements, there are two issues.

  1. Cannot dismiss the notification programmatically;
  2. Cannot dismiss it after a given time.

To better understand the problem(s), I will go into more detail and provide you with a solution.

The problems

There is no dismiss method for notifications

First, the window.showInformationMessage, window.showWarningMessage, and window.showErrorMessage methods do not provide a way to dismiss these programmatically.

Luckily, I found a solution to use a progress notification (window.withProgress) to overcome this. It is a pretty neat solution, but it has another issue. Progress notifications do not allow you to provide additional actions. For this problem, I decided to use a link with a command URI which will trigger the undo functionality and dismisses the progress notification.

Dismiss the progress notification after a given time

The last issue is when a user clicks on the hyperlink/command action in the progress notification, it will not automatically dismiss.

Additional logic is required to solve the dismiss issue in progress notifications.

info

I wrote about this logic in the Cancel progress programmatically in Visual Studio Code extensions article.

The solution

The whole solution looks as follows:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
import * as vscode from 'vscode';

const sleep = (time: number) => {
  return new Promise((resolve) => {
    setTimeout(() => {
      resolve(true);
    }, time);
  });
}

let customCancellationToken: vscode.CancellationTokenSource | null = null;

export function activate(context: vscode.ExtensionContext) {

  const message = "Please wait...";
  const commandId = "vscode-examples.undo";
  vscode.window.withProgress({
      location: vscode.ProgressLocation.Notification,
      cancellable: false
    },
    async (progress, token) => {
      return new Promise((async (resolve) => {
        customCancellationToken = new vscode.CancellationTokenSource();

        customCancellationToken.token.onCancellationRequested(() => {
          customCancellationToken?.dispose();
          customCancellationToken = null;

          vscode.window.showInformationMessage("Cancelled the progress");
          resolve(null);
          return;
        });

        const seconds = 10;
        for (let i = 0; i < seconds; i++) {
          // Increment is summed up with the previous value
          progress.report({ increment: seconds, message: `${message} [Undo](command:${commandId})` })
          await sleep(1000);
        }
        
        resolve(null);
      }));
    }
  );

  vscode.commands.registerCommand(commandId, () => {
    if (customCancellationToken) {
      customCancellationToken.cancel();
    }

    // Add your undo logic here
  })
}

The code from the above snippet shows a progress notification for 10 seconds with a undo link that uses a command URI.

When the user clicks on the link, it triggers the undo command.

Click event on a progress notification

Comments

Back to top