# sveld **Repository Path**: mirrors_ibm/sveld ## Basic Information - **Project Name**: sveld - **Description**: Generate TypeScript definitions for your Svelte components - **Primary Language**: Unknown - **License**: Apache-2.0 - **Default Branch**: main - **Homepage**: None - **GVP Project**: No ## Statistics - **Stars**: 0 - **Forks**: 0 - **Created**: 2020-11-23 - **Last Updated**: 2025-08-03 ## Categories & Tags **Categories**: Uncategorized **Tags**: None ## README # sveld [![NPM][npm]][npm-url] ![GitHub](https://img.shields.io/github/license/carbon-design-system/sveld?color=262626&style=for-the-badge) ![npm downloads to date](https://img.shields.io/npm/dt/sveld?color=262626&style=for-the-badge) `sveld` is a TypeScript definition generator for Svelte components. It analyzes props, events, slots, and other component features through static analysis. Types and signatures can be defined using [JSDoc notation](https://jsdoc.app/). The tool can also generate component documentation in Markdown and JSON formats. The purpose of this project is to make third party Svelte component libraries compatible with the Svelte Language Server and TypeScript with minimal effort required by the author. For example, TypeScript definitions may be used during development via intelligent code completion in Integrated Development Environments (IDE) like VSCode. [Carbon Components Svelte](https://github.com/IBM/carbon-components-svelte) uses this library to auto-generate component types and API metadata: - [TypeScript definitions](https://github.com/IBM/carbon-components-svelte/blob/master/types): Component TypeScript definitions - [Component Index](https://github.com/IBM/carbon-components-svelte/blob/master/COMPONENT_INDEX.md): Markdown file documenting component props, slots, and events - [Component API](https://github.com/IBM/carbon-components-svelte/blob/master/docs/src/COMPONENT_API.json): Component API metadata in JSON format **Please note** that the generated TypeScript definitions require Svelte version 3.55 or greater. --- Given a Svelte component, `sveld` can infer basic prop types to generate TypeScript definitions compatible with the [Svelte Language Server](https://github.com/sveltejs/language-tools): **Button.svelte** ```svelte ``` The generated definition extends the official `SvelteComponentTyped` interface exported from Svelte. **Button.svelte.d.ts** ```ts import type { SvelteComponentTyped } from "svelte"; import type { SvelteHTMLElements } from "svelte/elements"; type $RestProps = SvelteHTMLElements["button"]; type $Props = { /** * @default "button" */ type?: string; /** * @default false */ primary?: boolean; [key: `data-${string}`]: any; }; export type ButtonProps = Omit<$RestProps, keyof $Props> & $Props; export default class Button extends SvelteComponentTyped< ButtonProps, { click: WindowEventMap["click"] }, { default: {} } > {} ``` Sometimes, inferring prop types is insufficient. Prop/event/slot types and signatures can be augmented using [JSDoc](https://jsdoc.app/) notations. ```js /** @type {"button" | "submit" | "reset"} */ export let type = "button"; /** * Set to `true` to use the primary variant */ export let primary = false; ``` The accompanying JSDoc annotations would generate the following: ```ts import type { SvelteHTMLElements } from "svelte/elements"; type $RestProps = SvelteHTMLElements["button"]; type $Props = { /** * @default "button" */ type?: "button" | "submit" | "reset"; /** * Set to `true` to use the primary variant * @default false */ primary?: boolean; }; export type ButtonProps = Omit<$RestProps, keyof $Props> & $Props; export default class Button extends SvelteComponentTyped< ButtonProps, { click: WindowEventMap["click"] }, { default: {} } > {} ``` --- ## Table of Contents - [Approach](#approach) - [Usage](#usage) - [Installation](#installation) - [Rollup](#rollup) - [Node.js](#nodejs) - [CLI](#cli) - [Publishing to NPM](#publishing-to-npm) - [Available Options](#available-options) - [API Reference](#api-reference) - [@type](#type) - [@typedef](#typedef) - [@slot](#slot) - [@event](#event) - [@restProps](#restprops) - [@extends](#extends) - [@generics](#generics) - [@component comments](#component-comments) - [Contributing](#contributing) - [License](#license) ## Approach `sveld` uses the Svelte compiler to statically analyze Svelte components exported from a library to generate documentation useful to the end user. Extracted metadata include: - props - slots - forwarded events - dispatched events - `$$restProps` This library adopts a progressively enhanced approach. Any property type that cannot be inferred (e.g., "hello" is a string) falls back to "any" to minimize incorrectly typed properties or signatures. To mitigate this, the library author can add JSDoc annotations to specify types that cannot be reliably inferred. This represents a progressively enhanced approach because JSDocs are comments that can be ignored by the compiler. ## Usage ### Installation Install `sveld` as a development dependency. ```sh # npm npm i -D sveld # pnpm pnpm i -D sveld # Bun bun i -D sveld # Yarn yarn add -D sveld ``` ### Rollup Import and add `sveld` as a plugin to your `rollup.config.js`. ```js // rollup.config.js import svelte from "rollup-plugin-svelte"; import resolve from "@rollup/plugin-node-resolve"; import sveld from "sveld"; export default { input: "src/index.js", output: { format: "es", file: "lib/index.mjs", }, plugins: [svelte(), resolve(), sveld()], }; ``` When building the library, TypeScript definitions are emitted to the `types` folder by default. Customize the output folder using the `typesOptions.outDir` option. The following example emits the output to the `dist` folder: ```diff sveld({ + typesOptions: { + outDir: 'dist' + } }) ``` The [tests/e2e](tests/e2e) folder contains example set-ups: - [single-export](tests/e2e/single-export): library that exports one component - [single-export-default-only](tests/e2e/single-export-default-only): library that exports one component using the concise `export { default } ...` syntax - [multi-export](tests/e2e/multi-export): multi-component library without JSDoc annotations (types are inferred) - [multi-export-typed](tests/e2e/multi-export-typed): multi-component library with JSDoc annotations - [multi-export-typed-ts-only](tests/e2e/multi-export-typed-ts-only): multi-component library that only generates TS definitions - [glob](tests/e2e/glob): library that uses the glob strategy to collect/analyze \*.svelte files - [carbon](tests/e2e/carbon): full `carbon-components-svelte` example ### CLI The CLI uses the `"svelte"` field from your `package.json` as the entry point: ```sh npx sveld ``` Generate documentation in JSON and/or Markdown formats using the following flags: ```sh npx sveld --json --markdown ``` ### Node.js You can also use `sveld` programmatically in Node.js. If no `input` is specified, `sveld` will infer the entry point based on the `package.json#svelte` field. ```js const { sveld } = require("sveld"); const pkg = require("./package.json"); sveld({ input: "./src/index.js", glob: true, markdown: true, markdownOptions: { onAppend: (type, document, components) => { if (type === "h1") document.append("quote", `${components.size} components exported from ${pkg.name}@${pkg.version}.`); }, }, json: true, jsonOptions: { outFile: "docs/src/COMPONENT_API.json", }, }); ``` #### `jsonOptions.outDir` If `json` is `true`, a `COMPONENT_API.json` file will be generated at the root of your project. This file contains documentation for all components. Use the `jsonOptions.outDir` option to specify the folder for individual JSON files to be emitted. ```js sveld({ json: true, jsonOptions: { // an individual JSON file will be generated for each component API // e.g. "docs/Button.api.json" outDir: "docs", }, }); ``` ### Publishing to NPM TypeScript definitions are outputted to the `types` folder by default. Don't forget to include the folder in your `package.json` when publishing the package to NPM. ```diff { "svelte": "./src/index.js", "main": "./lib/index.mjs", + "types": "./types/index.d.ts", "files": [ "src", "lib", + "types", ] } ``` ## Available Options By default, only TypeScript definitions are generated. To generate documentation in Markdown and JSON formats, set `markdown` and `json` to `true`. ```diff sveld({ + markdown: true, + json: true, }) ``` ## API Reference ### `@type` Without a `@type` annotation, `sveld` will infer the primitive type for a prop: ```js export let kind = "primary"; // inferred type: "string" ``` Use the `@type` tag to explicitly document the type. In the following example, the `kind` property has an enumerated (enum) type. Signature: ```js /** * Optional description * @type {Type} */ ``` Example: ```js /** * Specify the kind of button * @type {"primary" | "secondary" | "tertiary"} */ export let kind = "primary"; /** * Specify the Carbon icon to render * @type {typeof import("carbon-icons-svelte").CarbonIcon} */ export let renderIcon = Close20; ``` ### `@typedef` The `@typedef` tag can be used to define a common type that is used multiple times within a component. All typedefs defined in a component will be exported from the generated TypeScript definition file. Signature: ```js /** * @typedef {Type} TypeName */ ``` Example: ```js /** * @typedef {string} AuthorName * @typedef {{ name?: AuthorName; dob?: string; }} Author */ /** @type {Author} */ export let author = {}; /** @type {Author[]} */ export let authors = []; ``` ### `@slot` Use the `@slot` tag for typing component slots. Note that `@slot` is a non-standard JSDoc tag. Descriptions are optional for named slots. Currently, the default slot cannot have a description. Signature: ```js /** * @slot {Type} slot-name [slot description] */ Omit the `slot-name` to type the default slot. /** * @slot {Type} */ ``` Example: ```svelte

``` ### `@event` Use the `@event` tag to type dispatched events. An event name is required and a description optional. Use `null` as the value if no event detail is provided. Signature: ```js /** * @event {EventDetail} eventname [event description] */ ``` Example: ```js /** * @event {{ key: string }} button:key * @event {null} key – Fired when `key` changes. */ export let key = ""; import { createEventDispatcher } from "svelte"; const dispatch = createEventDispatcher(); $: dispatch("button:key", { key }); $: if (key) dispatch("key"); ``` Output: ```ts export default class Component extends SvelteComponentTyped< ComponentProps, { "button:key": CustomEvent<{ key: string }>; /** Fired when `key` changes. */ key: CustomEvent; }, {} > {} ``` ### `@restProps` `sveld` can pick up inline HTML elements that `$$restProps` is forwarded to. However, it cannot infer the underlying element for instantiated components. You can use the `@restProps` tag to specify the element tags that `$$restProps` is forwarded to. Signature: ```js /** * Single element * @restProps {tagname} * * Multiple elements * @restProps {tagname-1 | tagname-2 | tagname-3} */ ``` Example: ```svelte {#if edit} ``` Output: ```ts /** * @example * */ export default class Button extends SvelteComponentTyped {} ``` ## Contributing Refer to the [contributing guidelines](CONTRIBUTING.md). ## License [Apache-2.0](LICENSE) [npm]: https://img.shields.io/npm/v/sveld.svg?color=262626&style=for-the-badge [npm-url]: https://npmjs.com/package/sveld