TanStack Hotkeys supports multi-key sequences in Svelte, where keys are pressed one after another rather than simultaneously.
<script lang="ts">
import { createHotkeySequence } from '@tanstack/svelte-hotkeys'
createHotkeySequence(['G', 'G'], () => {
window.scrollTo({ top: 0, behavior: 'smooth' })
})
</script>
Use createHotkeySequences to register several global sequences in one place (including from a reactive getter). For multiple sequences on a focused element, use createHotkeySequencesAttachment the same way you would use createHotkeySequenceAttachment.
<script lang="ts">
import { createHotkeySequences } from '@tanstack/svelte-hotkeys'
createHotkeySequences([
{ sequence: ['G', 'G'], callback: () => scrollToTop() },
{ sequence: ['D', 'D'], callback: () => deleteLine(), options: { timeout: 500 } },
])
</script>
Use createHotkeySequenceAttachment when a sequence should only be active while a specific element owns focus.
<script lang="ts">
import { createHotkeySequenceAttachment } from '@tanstack/svelte-hotkeys'
const editorSequences = createHotkeySequenceAttachment(['G', 'G'], () => {
scrollToTop()
})
</script>
<div tabindex="0" {@attach editorSequences}>
Focus here, then press g then g
</div>
createHotkeySequence(['G', 'G'], callback, {
timeout: 1000,
enabled: true,
})
When disabled, the sequence stays registered (visible in devtools); only execution is suppressed.
<script lang="ts">
import { createHotkeySequence } from '@tanstack/svelte-hotkeys'
let isVimMode = $state(true)
createHotkeySequence(
['G', 'G'],
() => scrollToTop(),
() => ({ enabled: isVimMode }),
)
</script>
<script lang="ts">
import { setHotkeysContext } from '@tanstack/svelte-hotkeys'
setHotkeysContext({
hotkeySequence: { timeout: 1500 },
})
</script>
Sequences support the same meta option as hotkeys, allowing you to attach a name and description for use in shortcut palettes and devtools.
createHotkeySequence(['G', 'G'], () => scrollToTop(), {
meta: { name: 'Go to Top', description: 'Scroll to the top of the page' },
})
See the Hotkeys Guide for details on declaration merging and introspecting registrations.
You can use the same modifier on consecutive steps (for example Shift+R then Shift+T):
createHotkeySequence(['Shift+R', 'Shift+T'], () => doNextAction())
While a sequence is in progress, modifier-only keydown events (Shift, Control, Alt, or Meta pressed alone) are ignored: they do not advance the sequence and do not reset progress.
createHotkeySequence(['G', 'G'], () => scrollToTop())
createHotkeySequence(['G', 'Shift+G'], () => scrollToBottom())
createHotkeySequence(['D', 'D'], () => deleteLine())
createHotkeySequence(['D', 'W'], () => deleteWord())
createHotkeySequence(['C', 'I', 'W'], () => changeInnerWord())
createHotkeySequence(
['ArrowUp', 'ArrowUp', 'ArrowDown', 'ArrowDown', 'B', 'A'],
() => enableEasterEgg(),
{ timeout: 2000 },
)
createHotkeySequence uses the singleton SequenceManager. You can also access it directly:
import {
createSequenceMatcher,
getSequenceManager,
} from '@tanstack/svelte-hotkeys'
const manager = getSequenceManager()
const matcher = createSequenceMatcher(['G', 'G'], { timeout: 1000 })