Configuration
- Overview - Layered configuration system
- Startup Script (init.ts) - Run TypeScript at startup for environment-dependent setup
- Keyboard - Keyboard shortcuts and keybinding configuration
Fresh uses layered configuration.
Configuration Layers
Settings are loaded from multiple layers, with higher layers overriding lower ones:
| Layer | Location | Scope | Use Case |
|---|---|---|---|
| System | Built-in defaults | Global | Factory defaults (read-only) |
| User | ~/.config/fresh/config.json | All projects | Personal preferences |
| Project | .fresh/config.json in project root | Single project | Project-specific settings |
| Session | .fresh/session.json (temporary) | Current session | Temporary overrides |
Path Notes:
- On Windows, User config is at
%APPDATA%\fresh\config.json - Project config is found by searching up from the current directory for
.fresh/config.json
How Layers Are Merged
Fresh merges all layers. Merge behavior depends on the setting type:
Simple Values (strings, numbers, booleans)
Higher layers override lower layers. If a setting is not specified in a higher layer, it falls through to the next lower layer.
System: theme = "default" ← Base default
User: theme = "dark" ← Overrides system
Project: (not set) ← Falls through
Session: theme = "light" ← Final value: "light"Nested Objects (editor, terminal, file_explorer)
Nested objects are deep-merged field by field. Each field follows the same "higher wins" rule independently.
Example: If User sets editor.tab_size = 4 and Project sets editor.line_wrap = true:
// User config
{ "editor": { "tab_size": 4, "line_numbers": true } }
// Project config
{ "editor": { "line_wrap": true } }
// Result: All fields merged
{ "editor": { "tab_size": 4, "line_numbers": true, "line_wrap": true } }Languages Map (deep merge)
The languages map uses deep merging with field-level override:
- Entries from all layers are combined (you can add new languages at any layer)
- For the same language key, individual fields are merged (not replaced entirely)
- Editor settings including
line_wrap,wrap_column,page_view, andpage_widthcan be set per-language — e.g. wrap Markdown at 80 columns while leaving code unwrapped
Example: Extending built-in Rust settings in your project:
// System (built-in): rust has extensions, grammar, etc.
// Project config - only need to specify what you're changing:
{
"languages": {
"rust": {
"tab_size": 2,
"format_on_save": true
}
}
}
// Result: Rust keeps all system defaults, with tab_size and format_on_save overriddenLSP Map (deep merge)
For the LSP feature itself (multi-server config, root markers, formatters,
only_features/except_features, etc.), see LSP Integration. This section only covers how thelspmap is merged across config layers.
The lsp map uses deep merging with field-level override:
- Entries from all layers are combined
- For the same language key, individual fields are merged (not replaced entirely)
- Unspecified fields inherit from lower layers (you only need to specify what you're changing)
Example: To disable an LSP while preserving its default command:
{
"lsp": {
"rust": {
"enabled": false
}
}
}
// Result: rust-analyzer command preserved from defaults, just disabledExample: To add initialization options without repeating the command:
{
"lsp": {
"rust": {
"initialization_options": { "checkOnSave": { "command": "clippy" } }
}
}
}
// Result: command="rust-analyzer" (from defaults) + your initialization_optionsLists (keybindings, on_save actions)
Lists are replaced entirely by higher layers - they are not merged or appended.
Example: If you define keybindings in your Project config, it completely replaces User keybindings (not extends them).
Removing/Unsetting Values
You cannot remove or unset a value from a lower layer — only override it. For boolean settings, you can set them to false to disable a feature enabled in a lower layer.
Using the Settings UI
To configure Fresh through the Settings UI:
- Open Settings: Use Edit → Settings... or Command Palette (
Ctrl+P) → "Open Settings" - Browse Categories: Use arrow keys or click to navigate
- Change Values: Toggle booleans, adjust numbers, select from dropdowns
- Choose Target Layer: Click the layer button (e.g.,
[ User ]) to switch between User/Project/Session - Save: Press Enter on the Save button or use
Ctrl+S
Advanced: Edit Config File Directly
For complex configurations (like LSP args or custom keybindings), click the [ Edit ] button in the Settings footer to open the raw JSON config file for the selected layer.
Example Configurations
User config (~/.config/fresh/config.json) - your personal defaults:
{
"version": 1,
"theme": "dark",
"editor": {
"tab_size": 4,
"line_numbers": true
}
}Project config (.fresh/config.json) - project-specific overrides:
{
"version": 1,
"editor": {
"tab_size": 2
},
"languages": {
"javascript": {
"formatter": "prettier --write"
}
}
}Common Configuration Tasks
Add a Custom Language
To add syntax highlighting and LSP support for a new language:
{
"languages": {
"mylang": {
"extensions": ["ml", "myl"],
"grammar": "mylang",
"comment_prefix": "#",
"auto_indent": true
}
},
"lsp": {
"mylang": {
"command": "mylang-lsp",
"args": ["--stdio"],
"enabled": true
}
}
}The grammar field accepts a short name like "bash" or "rust" as well as the full display name. To see every grammar available in your environment — including built-in grammars, user-installed grammars, language packs, bundles, and plugin-registered grammars — run:
fresh --cmd grammar listSet a Default Language for Unrecognized Files
When Fresh opens a file whose type it cannot detect (no matching extension, filename, or glob pattern), it shows it as "Plain Text" with no syntax highlighting. Set default_language to the name of any entry in the languages map and unrecognized files will use that language's full configuration — useful for .conf, .rc, .rules, and other config files that Fresh doesn't recognize.
{ "default_language": "bash" }This tells Fresh: "When you don't know what language a file is, treat it as bash." The file picks up bash syntax highlighting, # comments, indent rules, and anything else defined for bash in languages.
Any language name works — try yaml, json, toml, or a custom entry of your own. To disable (the default), leave default_language unset.
Customize LSP Settings
Configure initialization options for a language server:
{
"lsp": {
"rust": {
"command": "rust-analyzer",
"enabled": true,
"initialization_options": {
"checkOnSave": { "command": "clippy" }
}
}
}
}Project-Specific Tab Size
Create .fresh/config.json in your project:
{
"version": 1,
"editor": {
"tab_size": 2
}
}Layer Source Indicators
In the Settings UI, each setting shows where its current value comes from:
- (user) - Set in your User config
- (project) - Set in the Project config
- (session) - Temporary session override
- (no indicator) - Using system default
Status Bar
The left and right sides of the status bar are configurable through the Settings UI. Each side uses a DualList picker: items live in an Available column or an Included column, and you move them back and forth to show or hide them. Use the arrow buttons next to the Included list to reorder. Elements include the filename, cursor position, encoding, LSP indicator, git branch, warning counts, palette hint, a {clock} element that shows HH:MM with a blinking colon, and a {remote} indicator that lights up when you're attached to an SSH remote or a devcontainer.
The {remote} indicator is clickable — activate it to open a context-aware menu for the current authority (detach, show container logs, retry attach, etc.). It also reflects connection state: Connecting, Connected, or FailedAttach.
Save Behavior
If the target directory doesn't exist when you save a file, Fresh prompts to create it for you instead of failing. This applies to both brand-new files and to saving an existing buffer under a new path.
Editor Settings Reference
All settings can be changed via the Settings UI (run Open Settings from the palette).
Display
| Setting | Description | Default |
|---|---|---|
| Line numbers | Show line numbers in gutter | on |
| Line wrap | Soft-wrap long lines | off |
| Rulers | Column positions for vertical ruler lines | none |
| Vertical scrollbar | Show vertical scrollbar | on |
| Horizontal scrollbar | Show horizontal scrollbar | off |
| Terminal background | Let terminal background show through | off |
| Bracket matching | Highlight matching bracket pairs | on |
| Status bar | Show/hide the status bar | on |
| Whitespace indicators | Show space/tab characters (leading, inner, trailing) | off |
| Diagnostics inline text | Show diagnostics at end of line | off |
| Show tilde | Show ~ markers after end of file | on |
| Menu bar mnemonics | Enable Alt+key shortcuts for menu bar | on |
Editing
| Setting | Description | Default |
|---|---|---|
| Auto-close | Auto-close brackets and quotes | on |
| Auto-surround | Wrap selection when typing a delimiter | on |
| Trim trailing whitespace on save | Remove trailing whitespace when saving | off |
| Ensure final newline on save | Add trailing newline when saving | off |
Auto-Save
| Setting | Description | Default |
|---|---|---|
| Auto-save | Save modified buffers to disk automatically | off |
| Auto-save interval | Seconds between auto-saves (when enabled) | 30 |
| Recovery save interval | Seconds between crash-recovery saves | 2 |
| Hot exit | Persist all buffers (including scratch) across sessions | on |
Indentation
| Setting | Description | Default |
|---|---|---|
| Tab size | Spaces per indent level | 4 |
| Use tabs | Indent with tabs instead of spaces | off |
UI
| Setting | Description | Default |
|---|---|---|
| Show prompt line | Show the prompt line at the bottom | on |
Clipboard
| Setting | Description | Default |
|---|---|---|
| OSC 52 | Use OSC 52 escape sequence for clipboard | on |
| System clipboard | Use system clipboard | on |
If copy/paste hangs (common with PuTTY), try disabling one or both of these.
Process Resource Limits
To prevent LSP servers from consuming too many resources, Fresh can limit their memory and CPU usage.
{
"lsp": {
"rust": {
"command": "rust-analyzer",
"enabled": true,
"process_limits": {
"max_memory_mb": 4096,
"max_cpu_percent": 200
}
}
}
}The max_memory_mb limit is enforced via platform-specific mechanisms. max_cpu_percent is relative to one core (e.g. 200 = two full cores).