Code Blocks
Beautiful syntax highlighting and interactive code blocks powered by Shiki.
Streamdown provides beautiful, interactive code blocks with syntax highlighting powered by Shiki. Every code block includes a copy button and supports a wide range of programming languages.
Basic Usage
Create code blocks using triple backticks with an optional language identifier:
```javascript
function greet(name) {
return `Hello, ${name}!`;
}
```Streamdown will automatically apply syntax highlighting based on the specified language.
Supported Languages
Shiki supports 200+ programming languages out of the box, including:
- Web: JavaScript, TypeScript, HTML, CSS, JSX, TSX, Vue, Svelte
- Backend: Python, Java, Go, Rust, C, C++, C#, PHP, Ruby
- Data: SQL, JSON, YAML, TOML, XML, GraphQL
- Shell: Bash, PowerShell, Zsh
- Markup: Markdown, MDX, LaTeX
- And many more: Kotlin, Swift, Scala, Haskell, Elixir, Clojure...
Language Examples
TypeScript
```typescript
interface User {
id: number;
name: string;
email: string;
}
async function fetchUser(id: number): Promise<User> {
const response = await fetch(`/api/users/${id}`);
return response.json();
}
```Python
```python
def fibonacci(n: int) -> list[int]:
"""Generate Fibonacci sequence up to n terms."""
fib = [0, 1]
for i in range(2, n):
fib.append(fib[i-1] + fib[i-2])
return fib
print(fibonacci(10))
```Rust
```rust
fn main() {
let numbers = vec![1, 2, 3, 4, 5];
let sum: i32 = numbers.iter().sum();
println!("Sum: {}", sum);
}
```Theme Configuration
Streamdown uses dual themes for light and dark modes. You can customize the themes using the shikiTheme prop:
import { Streamdown } from 'streamdown';
import type { BundledTheme } from 'shiki';
const themes = ['github-light', 'github-dark'] as [BundledTheme, BundledTheme];
export default function Page() {
return (
<Streamdown shikiTheme={themes}>
{markdown}
</Streamdown>
);
}Refer to the Shiki Themes page for a full list of available themes.
Interactive Features
Copy Button
Every code block includes a copy button that appears on hover. Users can click to copy the entire code block content to their clipboard.
The copy button:
- Appears on hover (desktop) or is always visible (mobile)
- Provides visual feedback on successful copy
- Is automatically disabled during streaming (when
isAnimating={true})
Disable Copy Button
You can disable the copy button using the controls prop:
<Streamdown controls={{ code: false }}>
{markdown}
</Streamdown>Or disable all controls:
<Streamdown controls={false}>
{markdown}
</Streamdown>Inline Code
Inline code uses backticks and receives subtle styling:
Use the `useState` hook to manage state in React.Inline code is styled with:
- Monospace font family
- Subtle background color
- Rounded corners
- Appropriate padding
Code Block Styling
Code blocks include:
- Line Numbers - Optional line numbers for reference
- Rounded Corners - Modern, polished appearance
- Proper Padding - Comfortable spacing
- Scrolling - Horizontal scroll for long lines
- Responsive Design - Adapts to container width
Streaming Considerations
Code blocks work seamlessly with streaming content:
Incomplete Code Blocks
When a code block is streaming in, Streamdown handles the incomplete state gracefully:
```javascript
function example() {
// Streaming in progress...The unterminated block parser ensures the code block renders properly even without the closing backticks.
Disabling Interactions During Streaming
Use the isAnimating prop to disable copy buttons while streaming:
<Streamdown isAnimating={isStreaming}>
{markdown}
</Streamdown>This prevents users from copying incomplete code.