| name | vscode-extension |
| description | Contains verified extension development workflows, webview CSP pitfall resolutions, and profile-based install fixes that produce more correct results than reasoning from general training alone. Consult when building or debugging a VS Code extension, scaffolding with esbuild, fixing blank webview panels or silent CSP violations, troubleshooting extensions that won't load after install with profiles, adding dual TreeView+WebviewView toggle panels, or publishing to the marketplace. Do NOT use for VS Code user configuration (themes, keybindings, tasks.json), Chrome/browser extensions, Electron apps, JupyterLab plugins, or Monaco Editor embeds.
|
VS Code Extension Development
Guide for building VS Code extensions from scaffold through publication.
Environment Notes
VS Code extensions run in three distinct environments. The extension code and APIs are identical
across all three — the differences are in where files live and how the extension host is launched.
Native (Windows/macOS/Linux desktop)
The extension host runs locally on the same machine as the VS Code UI.
- CLI:
code
- Extensions:
%USERPROFILE%\.vscode\extensions\ (Windows), ~/.vscode/extensions/ (macOS/Linux)
- User data:
%APPDATA%\Code\User\ (Windows), ~/Library/Application Support/Code/User/ (macOS), ~/.config/Code/User/ (Linux)
- Profile settings:
<user-data>/profiles/<profile-id>/settings.json
- Profile extensions:
<user-data>/profiles/<profile-id>/extensions.json
- File watchers: OS-native (ReadDirectoryChangesW on Windows, FSEvents on macOS, inotify on Linux)
- Install:
code --install-extension my-ext.vsix
WSL Remote (VS Code on Windows, extension host in WSL)
VS Code runs on Windows but connects to a WSL distro. The extension host runs inside WSL as a
Node.js server. Extensions split across two locations — UI extensions (themes, keymaps) stay on
Windows, workspace extensions run in WSL.
- CLI:
code (available inside WSL via the VS Code Server shim)
- Server-side extensions (WSL):
~/.vscode-server/extensions/
- UI-side extensions (Windows):
%USERPROFILE%\.vscode\extensions\
(from WSL: /mnt/c/Users/<user>/.vscode/extensions/)
- Server-side profiles (WSL):
~/.vscode-server/data/User/profiles/<profile-id>/extensions.json
- Settings/keybindings (Windows):
%APPDATA%\Code\User\profiles\<profile-id>\settings.json
(from WSL: /mnt/c/Users/<user>/AppData/Roaming/Code/User/profiles/<profile-id>/)
— the Windows side is authoritative for UI/editor settings (theme, font, keybindings,
layout). The WSL server-side profile at ~/.vscode-server/data/User/profiles/<profile-id>/
has only a minimal settings.json — most settings are not duplicated there.
Workspace-level overrides go in .vscode/settings.json in the project dir on WSL.
- File watchers: inotify-backed via VS Code's API — always prefer
createFileSystemWatcher()
over Node.js fs.watch() which has cross-platform inconsistencies
- Path translation: Not needed for extension code — the extension host runs natively in WSL.
Windows paths are only relevant for the VS Code UI process.
- Install:
code --install-extension my-ext.vsix (runs inside WSL, installs to WSL server dir)
process.env: The extension host does NOT have terminal-specific variables like
VSCODE_IPC_HOOK_CLI — read those from /proc/<terminal_pid>/environ instead
SSH Remote (VS Code on any OS, extension host on remote Linux)
Same architecture as WSL Remote — extensions install to ~/.vscode-server/ on the remote host,
the extension host runs as a Node.js server over SSH. Same constraints as WSL Remote apply.
Launch Config
Use ${execPath} in .vscode/launch.json — it resolves correctly across all three environments.
Quick Start — Scaffold a New Extension
Generate a TypeScript extension project:
npx --package yo --package generator-code -- yo code
Select "New Extension (TypeScript)" when prompted. This creates a ready-to-run project with
src/extension.ts, package.json, and TypeScript configuration.
Open the project and launch the Extension Development Host with F5 to test immediately.
After scaffolding, convert the build system from tsc to esbuild — see the esbuild section below.
The yo generator creates a tsc-based build by default, but esbuild bundles are smaller, faster,
and what VS Code itself uses internally (as of v1.110).
Project Structure
extension-name/
package.json # manifest: contribution points, activation, settings
esbuild.js # build script (replaces default tsc pipeline)
tsconfig.json # TypeScript config
.vscodeignore # files excluded from .vsix package
.vscode-test.mjs # test runner configuration
src/
extension.ts # activate() / deactivate() entry point
*.ts # feature modules
dist/
extension.js # bundled output (single file)
Keep src/ organized by feature. Each major capability (a TreeView provider, a webview, a
command group) gets its own module file. The extension.ts entry point wires them together
in activate() and tears them down in deactivate().
package.json — The Extension Manifest
The manifest declares everything the extension contributes to VS Code. Key fields:
{
"name": "my-extension",
"displayName": "My Extension",
"version": "0.1.0",
"engines": { "vscode": "^1.110.0" },
"main": "./dist/extension",
"activationEvents": [],
"contributes": { }
}
Activation Events
VS Code auto-generates activation events for most contribution points — you rarely need to
declare them manually. Notable exceptions:
onWebviewPanel:viewType — for webview panel restoration
onUri — for URI handlers
* — activate on startup (avoid unless truly needed)
For views (contributes.views), VS Code automatically activates on onView:viewId.
Contribution Points
The contributes object declares UI elements, commands, settings, and more:
{
"contributes": {
"commands": [{
"command": "myExt.refresh",
"title": "Refresh",
"icon": "$(refresh)",
"category": "My Extension"
}],
"views": {
"explorer": [{
"id": "myView",
"name": "My View"
}]
},
"viewsContainers": {
"activitybar": [{
"id": "myContainer",
"title": "My Container",
"icon": "media/icon.svg"
}],
"panel": [{
"id": "myPanel",
"title": "My Panel",
"icon": "media/icon.svg"
}]
},
"menus": {
"view/title": [{
"command": "myExt.refresh",
"when": "view == myView",
"group": "navigation"
}],
"view/item/context": [{
"command": "myExt.openItem",
"when": "view == myView && viewItem == editable",
"group": "inline"
}]
},
"viewsWelcome": [{
"view": "myView",
"contents": "No items found.\n[Get Started](command:myExt.init)"
}],
"configuration": {
"title": "My Extension",
"properties": {
"myExt.refreshInterval": {
"type": "number",
"default": 3000,
"description": "Refresh interval in milliseconds"
}
}
}
}
}
Menu when Clauses
Control visibility with boolean expressions:
view == myView — only in a specific view
viewItem == someContext — only for tree items with that contextValue
resourceScheme == file — only for file URIs
- Combine with
&&, ||, !
Menu Group Ordering
Use @ suffix for ordering within a group: "navigation@1", "navigation@2".
Extension Entry Point
import * as vscode from 'vscode';
export function activate(context: vscode.ExtensionContext) {
const disposable = vscode.commands.registerCommand('myExt.hello', () => {
vscode.window.showInformationMessage('Hello!');
});
context.subscriptions.push(disposable);
}
export function deactivate() {
}
Every disposable (commands, watchers, providers, views) should be pushed to
context.subscriptions so VS Code cleans them up on deactivation.
Core API Patterns
This section covers the most common patterns. For detailed API reference with full code examples,
read references/api-patterns.md.
TreeView
The most common way to add structured data to the sidebar or panel:
- Implement
TreeDataProvider<T> with getChildren() and getTreeItem()
- Fire
onDidChangeTreeData to refresh
- Register via
vscode.window.createTreeView() for full API access
TreeItems support labels, descriptions, tooltips, icons (ThemeIcon or custom SVG), collapsible
state, contextValue (for menu filtering), and a command to execute on click.
Webview
For custom HTML/CSS/JS interfaces when TreeView isn't sufficient:
- Create with
vscode.window.createWebviewPanel() or implement WebviewViewProvider
- Set HTML content, load local resources via
webview.asWebviewUri()
- Communicate via
postMessage() / onDidReceiveMessage()
- Always set a Content Security Policy
Webviews are sandboxed — they cannot access the filesystem directly. All communication goes
through the message passing API.
Dual View: TreeView + WebviewView Toggle
When you need richer presentation than TreeView allows (multi-line rows, colored status text,
custom layouts), add a WebviewView alongside the existing TreeView with a toolbar toggle. The
pattern involves:
- Register both views in the same
viewsContainers entry with when clauses on a context key
- Toggle commands flip the context key and persist to
globalState
- Both providers share a single data pipeline and refresh together
- User interactions in both views route through shared VS Code commands (not direct calls)
- CSP requires careful layering: nonce for scripts,
'unsafe-inline' for styles, and
${webview.cspSource} for external CSS
This pattern has several non-obvious failure modes (silent CSP blocking, cache invalidation
blanking the panel, stale focus handlers). For the complete implementation guide with all
pitfalls and fixes, read references/api-patterns.md §Dual View.
Terminal
For interacting with VS Code's integrated terminal:
window.createTerminal() — create terminals
terminal.show() — focus a terminal (handles tabs and split panes)
terminal.processId — get the underlying PID
window.terminals — enumerate all open terminals
onDidOpenTerminal / onDidCloseTerminal — lifecycle events
FileSystemWatcher
For reacting to file changes on disk. Prefer the VS Code API over Node.js fs.watch() because
it runs outside the editor process and uses efficient OS-level notifications (inotify on Linux):
const watcher = vscode.workspace.createFileSystemWatcher(
new vscode.RelativePattern(folder, '**/*.json')
);
watcher.onDidChange(uri => { });
watcher.onDidCreate(uri => { });
watcher.onDidDelete(uri => { });
context.subscriptions.push(watcher);
StatusBar
window.createStatusBarItem(alignment, priority) — left or right alignment
- Convention: left side = workspace-scoped info, right side = file-scoped info
- Set
text, tooltip, command, color, backgroundColor
Building with esbuild
esbuild is the recommended bundler (VS Code's own built-in extensions switched to it in v1.110).
It produces a single dist/extension.js file from the TypeScript source.
esbuild.js
const esbuild = require('esbuild');
const production = process.argv.includes('--production');
const watch = process.argv.includes('--watch');
const esbuildProblemMatcherPlugin = {
name: 'esbuild-problem-matcher',
setup(build) {
build.onStart(() => { console.log('[watch] build started'); });
build.onEnd(result => {
for (const { text, location } of result.errors) {
console.error(`✘ [ERROR] ${text}`);
if (location) {
console.error(` ${location.file}:${location.line}:${location.column}:`);
}
}
console.log('[watch] build finished');
});
},
};
async function main() {
const ctx = await esbuild.context({
entryPoints: ['src/extension.ts'],
bundle: true,
format: 'cjs',
minify: production,
sourcemap: !production,
sourcesContent: false,
platform: 'node',
outfile: 'dist/extension.js',
external: ['vscode'],
logLevel: 'warning',
plugins: [esbuildProblemMatcherPlugin],
});
if (watch) {
await ctx.watch();
} else {
await ctx.rebuild();
await ctx.dispose();
}
}
main().catch(e => { console.error(e); process.exit(1); });
npm Scripts
{
"scripts": {
"compile": "npm run check-types && node esbuild.js",
"check-types": "tsc --noEmit",
"watch": "npm-run-all -p watch:*",
"watch:esbuild": "node esbuild.js --watch",
"watch:tsc": "tsc --noEmit --watch --project tsconfig.json",
"package": "npm run check-types && node esbuild.js --production",
"vscode:prepublish": "npm run package",
"test": "vscode-test"
}
}
The watch script runs esbuild and tsc in parallel — esbuild handles bundling while tsc
provides type checking. This gives sub-second rebuilds with full type safety.
vscode:prepublish runs automatically during vsce package and vsce publish.
.vscodeignore
.vscode/
node_modules/
out/
src/
tsconfig.json
esbuild.js
.vscode-test.mjs
**/*.ts
!dist/**
Everything bundled into dist/extension.js can be excluded. This keeps the .vsix small.
Testing
For testing setup, configuration, and debugging details, read references/build-test-publish.md.
Quick setup:
npm install --save-dev @vscode/test-cli @vscode/test-electron
Tests run inside an Extension Development Host with full VS Code API access. Configure in
.vscode-test.mjs:
import { defineConfig } from '@vscode/test-cli';
export default defineConfig({
files: 'out/test/**/*.test.js',
mocha: { ui: 'tdd', timeout: 20000 }
});
Run: npm test or npx vscode-test --label unitTests
Publishing
For full publishing workflow, read references/build-test-publish.md.
Quick reference:
npm install -g @vscode/vsce
vsce package
vsce publish
vsce publish minor
vsce publish --pre-release
code --install-extension my-ext-0.1.0.vsix
Platform-specific builds: vsce package --target linux-x64 linux-arm64 darwin-x64
Stable vs Insiders — Same Extension
A .vsix package works on both VS Code stable and Insiders — there is no separate build. The
only factor is the engines.vscode version in package.json:
"^1.110.0" means the extension requires VS Code 1.110+. If a new API you use only exists
in Insiders (which runs ahead of stable by ~1 month), stable users can't install until that
version ships. Use the lowest version that supports the APIs you need.
- The marketplace serves the same package to both builds.
- Local install just differs by CLI:
code --install-extension vs code-insiders --install-extension.
Platform Notes
Profile-Based Extension Registration
When VS Code profiles are active, code --install-extension may register the extension in the
global registry but NOT in the active profile's registry. This is especially common in WSL/SSH
remote environments where the global registry is at ~/.vscode-server/extensions/extensions.json
but the profile registry is at ~/.vscode-server/data/User/profiles/<profile-id>/extensions.json.
The --profile flag may not fix this on WSL. The extension won't load until manually added to
the profile-specific extensions.json. Without a publisher field in package.json, the extension
installs as undefined_publisher.<name>-<version>.
On native installs, the equivalent paths are under ~/.vscode/ instead of ~/.vscode-server/.
Webview Content Security Policy
- Inline
onclick attributes are blocked by default CSP — use nonce-based <script> blocks
with addEventListener instead
style-src 'nonce-...' blocks inline style="" attributes — use
style-src ${webview.cspSource} 'unsafe-inline' for style-src, or move dynamic styling to
CSS classes with data-* attribute selectors
- Always include
${webview.cspSource} in style-src for CSS to load correctly
Architecture Constraints
- Extensions run in the Extension Host process, separate from the VS Code UI
- No DOM access — you cannot manipulate VS Code's UI directly
- No custom CSS injection — use contribution points and the API
- Webviews are sandboxed with their own security context
Reference Files
references/api-patterns.md — Detailed API patterns with full code examples for TreeView,
Webview, WebviewView, Terminal, FileSystemWatcher, and StatusBar. Read when implementing
any of these APIs.
references/build-test-publish.md — esbuild configuration rationale and additional options,
TypeScript config, test runner setup, debugging tests, CI configuration, publishing workflow,
and marketplace metadata. Read when setting up the build pipeline or preparing to publish.