OPEN TO WORK

Available for new opportunities! Let's build something amazing together.

Cancel progress programmatically in VS Code extensions

VSCode Extensions
post

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

With the Visual Studio Code progress notification (vscode.window.withProgress), you can make it cancellable. This cancellable option allows the user to cancel it by clicking on a cancel button.

Show image Progress notification in Visual Studio Code
Progress notification in Visual Studio Code

What if you want to cancel it programmatically? That was the case for one of my extensions. When a user performs another action triggered outside the progress, it should stop the current progress and remove the notification.

For example, in your progress, you are processing all files because the user requested to analyze these. Once the user opens another view or requests to analyze a single file, the other progress can be stopped.

The cancellation token

With the progress its callback, you get a CancellationToken argument, although this token can only be used to monitor if the user has canceled the operation.

vscode.window.withProgress({
title: 'Please wait...',
location: vscode.ProgressLocation.Notification,
cancellable: true
},
async (progress, token) => {
// You code to process the progress
const seconds = 10;
for (let i = 0; i < seconds; i++) {
// Increment is summed up with the previous value
progress.report({ increment: seconds })
await sleep(1000);
}
});

If we want to do this programmatically, you must create your own CancellationToken logic. The nice thing is that Visual Studio Code’s extension API already provides this.

To create your cancellation logic, you can use the vscode.CancellationTokenSource class. When creating a new instance, you will get a cancellation token that you can cancel and dismiss. All you have to do is hook it up in your progress and listen to the cancellation trigger.

The code for this looks as follows:

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) {
vscode.window.withProgress({
title: 'Please wait...',
location: vscode.ProgressLocation.Notification,
cancellable: true
},
async (progress, token) => {
return new Promise((async (resolve) => {
// You code to process the progress
customCancellationToken = new vscode.CancellationTokenSource();
customCancellationToken.token.onCancellationRequested(() => {
customCancellationToken?.dispose();
customCancellationToken = null;
vscode.window.showInformationMessage("Cancelled the progress");
resolve(null);
return;
});
const seconds = 30;
for (let i = 0; i < seconds; i++) {
await sleep(1000);
}
resolve(null);
}));
});
// Cancel the progress after 10 seconds
setTimeout(() => {
if (customCancellationToken) {
customCancellationToken.cancel();
}
}, 10000);
}
Show image Cancelled message from the code snippet
Cancelled message from the code snippet

Related articles

Report issues or make changes on GitHub

Found a typo or issue in this article? Visit the GitHub repository to make changes or submit a bug report.

Comments

Elio Struyf

Solutions Architect & Developer Expert

Loading...

Let's build together

Manage content in VS Code

Present from VS Code