A template for building vscode extensions using svelte.
Most of what you need for development can be done from the root directory.
Install deps
npm install
Start a debug session with F5
or Debug > Start Debugging
in the command palette. This will open a new instance of vscode with the extension loaded.
In the new extension host window, open the webview in one of 2 ways
VSvelteCode: Hello World
from the command palette.There are two subdirectories to this project: The /extension
itself and the /svelte
webview.
This is where the business logic of interacting with vscode should live. Check out extension.ts
for the main entry point. It registers a command that launches a webview using svelteWebview.ts
, which does all the heavy lifting of pulling in the built svelte code and rendering it, passing a few necessary variables, etc.
This is just a regular vite/svelte app + tailwind with a few notable exceptions. The vscode
api is available as a global to send messages (see lib/Messages.ts
).
The default way to send state back and forth is via messages, which is covered in the official docs. On the svelte side, that means handling the messages like this:
<script lang="ts">
function handleMessage({ message }: MessageEvent) {
switch (message.command) {
case 'foo':
doBar(message.content)
break
}
}
function sendMessage() {
vscode.postMessage({ command: 'baz', content: 'boz' })
}
</script>
<svelte:window onmessage={handleMessage} />
<button onclick={sendMessage}>Send</button>
But that's a little clunky and I wanted to abstract things a little, so I'm working on a pattern so I can think less. The two main points of interest here are the State
class in the extension and the useVscodeState
rune in svelte. These are intended as a bonded pair so the two can seamlessly pass state back and forth. Here's a convoluted example:
// In your extension.ts
function handleSvelteMessage(message: Message) {
switch (message.command) {
case 'loadSomething':
loadSomething()
break
// ...
}
}
async function loadSomething() {
state.update({ loading: true })
const result = await fetchSomething()
state.update({ myData: result, loading: false })
}
<!-- Then in svelte, the state will responsively update -->
<script lang="ts">
import * as Messages from '../lib/Messages'
import { useVscodeState } from '../lib/useVscodeState'
const state = useVscodeState()
function loadSomething() {
Messages.post('loadSomething')
}
</script>
{#if state.loading}
Loading...
{:else if state.myData != null}
{state.myData}
{:else}
<button onclick={loadSomething}>
{/if}
The actual app state is just a POJO, vaguely inspired by streamlit's session state. There's an interface in types.ts
for the actual data that you can add to. The State
class lets you update those values and does 2 things:
I'm still working on updates going back the other way. For now, your svelte should use lib/Messages.post()
, and the messages should be handled in the handler you pass to svelteWebview()
. I did my best to demonstrate a strong typesafe pattern, but it's a little verbose.
The api needs work still, but it's a decent start.
TODO(Ian): Explain this better lol
Extensions are packaged using vsce
. The resulting .vsix
file will appear in /extension
and can be installed in any vscode instance/codespace/whatever.
npm run package
Publishing is done using vsce
as well. You will need to have a publisher account on the Visual Studio Marketplace and have your publisher name set in extension/package.json
file.
npm run publish