generated from nhcarrigan/template
feat: multiple UI improvements, font settings, and memory file display names (#175)
## Summary - **fix**: `show_thinking_blocks` setting now persists across sessions — it was defined on the TypeScript side but missing from the Rust `HikariConfig` struct, so serde silently dropped it on every save/load - **feat**: Tool calls are now rendered as collapsible blocks matching the Extended Thinking block aesthetic, replacing the old inline dropdown approach - **feat**: Add configurable max output tokens setting - **feat**: Use random creative names for conversation tabs - **test**: Significantly expanded frontend unit test coverage - **docs**: Require tests for all changes in CLAUDE.md - **feat**: Allow users to specify a custom terminal font (Closes #176) - **feat**: Display friendly names for memory files derived from the first heading (Closes #177) - **feat**: Add custom UI font support for the app chrome (buttons, labels, tabs) - **fix**: Apply custom UI font to the full app interface — `.app-container` was hardcoded, blocking inheritance from `body`; also renamed "Custom Font" to "Custom Terminal Font" for clarity ✨ This PR was created with help from Hikari~ 🌸 Reviewed-on: #175 Co-authored-by: Hikari <hikari@nhcarrigan.com> Co-committed-by: Hikari <hikari@nhcarrigan.com>
This commit was merged in pull request #175.
This commit is contained in:
@@ -5,6 +5,8 @@
|
||||
type Theme,
|
||||
type CustomThemeColors,
|
||||
applyFontSize,
|
||||
applyCustomFont,
|
||||
applyCustomUiFont,
|
||||
applyCustomThemeColors,
|
||||
MIN_FONT_SIZE,
|
||||
MAX_FONT_SIZE,
|
||||
@@ -56,12 +58,23 @@
|
||||
show_thinking_blocks: true,
|
||||
use_worktree: false,
|
||||
disable_1m_context: false,
|
||||
max_output_tokens: null,
|
||||
trusted_workspaces: [],
|
||||
background_image_path: null,
|
||||
background_image_opacity: 0.3,
|
||||
custom_font_path: null,
|
||||
custom_font_family: null,
|
||||
custom_ui_font_path: null,
|
||||
custom_ui_font_family: null,
|
||||
});
|
||||
|
||||
let showCustomThemeEditor = $state(false);
|
||||
let customFontPathInput = $state("");
|
||||
let customFontFamilyInput = $state("");
|
||||
let customFontStatus: string | null = $state(null);
|
||||
let customUiFontPathInput = $state("");
|
||||
let customUiFontFamilyInput = $state("");
|
||||
let customUiFontStatus: string | null = $state(null);
|
||||
|
||||
interface AuthStatus {
|
||||
is_logged_in: boolean;
|
||||
@@ -87,6 +100,10 @@
|
||||
|
||||
configStore.config.subscribe((c) => {
|
||||
config = { ...c };
|
||||
customFontPathInput = c.custom_font_path ?? "";
|
||||
customFontFamilyInput = c.custom_font_family ?? "";
|
||||
customUiFontPathInput = c.custom_ui_font_path ?? "";
|
||||
customUiFontFamilyInput = c.custom_ui_font_family ?? "";
|
||||
});
|
||||
|
||||
configStore.isSidebarOpen.subscribe((open) => {
|
||||
@@ -533,6 +550,25 @@
|
||||
context window
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<!-- Max Output Tokens -->
|
||||
<div class="mb-4">
|
||||
<label class="block text-sm text-[var(--text-primary)] mb-1" for="max-output-tokens">
|
||||
Max output tokens
|
||||
</label>
|
||||
<input
|
||||
id="max-output-tokens"
|
||||
type="number"
|
||||
min="1"
|
||||
placeholder="Default (32000)"
|
||||
bind:value={config.max_output_tokens}
|
||||
class="w-full px-3 py-2 text-sm bg-[var(--bg-primary)] border border-[var(--border-color)] rounded text-[var(--text-primary)] placeholder-[var(--text-tertiary)] focus:outline-none focus:border-[var(--accent-primary)]"
|
||||
/>
|
||||
<p class="text-xs text-[var(--text-tertiary)] mt-1">
|
||||
Sets <code class="font-mono">CLAUDE_CODE_MAX_OUTPUT_TOKENS</code> — increase if responses are
|
||||
being cut off mid-reply
|
||||
</p>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<!-- Greeting Section -->
|
||||
@@ -917,6 +953,130 @@
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<!-- Custom Terminal Font -->
|
||||
<div class="mb-4">
|
||||
<span class="block text-sm text-[var(--text-secondary)] mb-2">Custom Terminal Font</span>
|
||||
<div class="flex flex-col gap-2">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={customFontPathInput}
|
||||
placeholder="URL or local file path (e.g. /path/to/font.ttf)"
|
||||
class="w-full px-3 py-2 text-sm rounded-lg border border-[var(--border-color)] bg-[var(--bg-primary)] text-[var(--text-primary)] placeholder-gray-500 focus:outline-none focus:border-[var(--accent-primary)]"
|
||||
/>
|
||||
<input
|
||||
type="text"
|
||||
bind:value={customFontFamilyInput}
|
||||
placeholder="Font family name (e.g. FiraCode)"
|
||||
class="w-full px-3 py-2 text-sm rounded-lg border border-[var(--border-color)] bg-[var(--bg-primary)] text-[var(--text-primary)] placeholder-gray-500 focus:outline-none focus:border-[var(--accent-primary)]"
|
||||
/>
|
||||
<div class="flex gap-2">
|
||||
<button
|
||||
onclick={async () => {
|
||||
customFontStatus = null;
|
||||
try {
|
||||
await configStore.setCustomFont(
|
||||
customFontPathInput || null,
|
||||
customFontFamilyInput || null
|
||||
);
|
||||
customFontStatus = "Font applied!";
|
||||
} catch (e) {
|
||||
customFontStatus = `Error: ${e instanceof Error ? e.message : String(e)}`;
|
||||
}
|
||||
}}
|
||||
class="flex-1 px-3 py-2 text-sm rounded-lg border border-[var(--border-color)] bg-[var(--bg-primary)] text-[var(--text-secondary)] hover:border-[var(--accent-primary)] transition-colors"
|
||||
>
|
||||
Apply
|
||||
</button>
|
||||
<button
|
||||
onclick={async () => {
|
||||
customFontStatus = null;
|
||||
customFontPathInput = "";
|
||||
customFontFamilyInput = "";
|
||||
try {
|
||||
await configStore.setCustomFont(null, null);
|
||||
await applyCustomFont(null, null);
|
||||
customFontStatus = "Font reset to default.";
|
||||
} catch (e) {
|
||||
customFontStatus = `Error: ${e instanceof Error ? e.message : String(e)}`;
|
||||
}
|
||||
}}
|
||||
class="px-3 py-2 text-sm rounded-lg border border-[var(--border-color)] bg-[var(--bg-primary)] text-[var(--text-secondary)] hover:border-red-400 hover:text-red-400 transition-colors"
|
||||
>
|
||||
Reset
|
||||
</button>
|
||||
</div>
|
||||
{#if customFontStatus}
|
||||
<p class="text-xs text-[var(--text-tertiary)]">{customFontStatus}</p>
|
||||
{/if}
|
||||
</div>
|
||||
<p class="text-xs text-[var(--text-tertiary)] mt-1">
|
||||
Supports Google Fonts URLs, direct font file URLs, or local file paths. Family name is
|
||||
required to apply the font.
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<!-- Custom UI Font -->
|
||||
<div class="mb-4">
|
||||
<span class="block text-sm text-[var(--text-secondary)] mb-2">Custom UI Font</span>
|
||||
<div class="flex flex-col gap-2">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={customUiFontPathInput}
|
||||
placeholder="URL or local file path (e.g. /path/to/font.ttf)"
|
||||
class="w-full px-3 py-2 text-sm rounded-lg border border-[var(--border-color)] bg-[var(--bg-primary)] text-[var(--text-primary)] placeholder-gray-500 focus:outline-none focus:border-[var(--accent-primary)]"
|
||||
/>
|
||||
<input
|
||||
type="text"
|
||||
bind:value={customUiFontFamilyInput}
|
||||
placeholder="Font family name (e.g. Inter)"
|
||||
class="w-full px-3 py-2 text-sm rounded-lg border border-[var(--border-color)] bg-[var(--bg-primary)] text-[var(--text-primary)] placeholder-gray-500 focus:outline-none focus:border-[var(--accent-primary)]"
|
||||
/>
|
||||
<div class="flex gap-2">
|
||||
<button
|
||||
onclick={async () => {
|
||||
customUiFontStatus = null;
|
||||
try {
|
||||
await configStore.setCustomUiFont(
|
||||
customUiFontPathInput || null,
|
||||
customUiFontFamilyInput || null
|
||||
);
|
||||
customUiFontStatus = "Font applied!";
|
||||
} catch (e) {
|
||||
customUiFontStatus = `Error: ${e instanceof Error ? e.message : String(e)}`;
|
||||
}
|
||||
}}
|
||||
class="flex-1 px-3 py-2 text-sm rounded-lg border border-[var(--border-color)] bg-[var(--bg-primary)] text-[var(--text-secondary)] hover:border-[var(--accent-primary)] transition-colors"
|
||||
>
|
||||
Apply UI Font
|
||||
</button>
|
||||
<button
|
||||
onclick={async () => {
|
||||
customUiFontStatus = null;
|
||||
customUiFontPathInput = "";
|
||||
customUiFontFamilyInput = "";
|
||||
try {
|
||||
await configStore.setCustomUiFont(null, null);
|
||||
await applyCustomUiFont(null, null);
|
||||
customUiFontStatus = "Font reset to default.";
|
||||
} catch (e) {
|
||||
customUiFontStatus = `Error: ${e instanceof Error ? e.message : String(e)}`;
|
||||
}
|
||||
}}
|
||||
class="px-3 py-2 text-sm rounded-lg border border-[var(--border-color)] bg-[var(--bg-primary)] text-[var(--text-secondary)] hover:border-red-400 hover:text-red-400 transition-colors"
|
||||
>
|
||||
Reset
|
||||
</button>
|
||||
</div>
|
||||
{#if customUiFontStatus}
|
||||
<p class="text-xs text-[var(--text-tertiary)]">{customUiFontStatus}</p>
|
||||
{/if}
|
||||
</div>
|
||||
<p class="text-xs text-[var(--text-tertiary)] mt-1">
|
||||
Applies to the entire app interface (menus, labels, buttons). Supports Google Fonts URLs,
|
||||
direct font file URLs, or local file paths.
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<!-- Show Thinking Blocks Toggle -->
|
||||
<div class="mb-4">
|
||||
<label class="flex items-center gap-3 cursor-pointer">
|
||||
|
||||
Reference in New Issue
Block a user