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)
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)
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
}
}
}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
Editor Settings Reference
All settings can be changed via the Settings UI (command palette → "Open Settings").
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 |
Editing
| Setting | Description | Default |
|---|---|---|
| Tab size | Spaces per indent level | 4 |
| 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 |
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
}
}
}
}See docs/PROCESS_LIMITS.md.