use project DropdownMenu component for style selector

- show full style names in trigger (Paragraph, Heading 1, etc)
- portal dropdown to avoid clipping
This commit is contained in:
Justin Edmund 2025-12-21 15:33:07 -08:00
parent 8329ec9de3
commit 0339a3f832

View file

@ -1,6 +1,8 @@
<script lang="ts"> <script lang="ts">
import { onMount } from 'svelte' import { onMount } from 'svelte'
import type { Editor, Content } from '@tiptap/core' import type { Editor, Content } from '@tiptap/core'
import { DropdownMenu as DropdownMenuBase } from 'bits-ui'
import DropdownMenu from '$lib/components/ui/DropdownMenu.svelte'
import EdraEditor from '$lib/components/edra/headless/editor.svelte' import EdraEditor from '$lib/components/edra/headless/editor.svelte'
import { sidebar } from '$lib/stores/sidebar.svelte' import { sidebar } from '$lib/stores/sidebar.svelte'
@ -29,9 +31,6 @@
let editor = $state<Editor>() let editor = $state<Editor>()
let initialContent = $state<Content | undefined>() let initialContent = $state<Content | undefined>()
// Style dropdown state
let styleDropdownOpen = $state(false)
// Parse description JSON on mount // Parse description JSON on mount
onMount(() => { onMount(() => {
if (description) { if (description) {
@ -57,20 +56,18 @@
} }
function getStyleLabel(): string { function getStyleLabel(): string {
if (editor?.isActive('heading', { level: 1 })) return 'H1' if (editor?.isActive('heading', { level: 1 })) return 'Heading 1'
if (editor?.isActive('heading', { level: 2 })) return 'H2' if (editor?.isActive('heading', { level: 2 })) return 'Heading 2'
if (editor?.isActive('heading', { level: 3 })) return 'H3' if (editor?.isActive('heading', { level: 3 })) return 'Heading 3'
return 'P' return 'Paragraph'
} }
function setHeading(level: 1 | 2 | 3) { function setHeading(level: 1 | 2 | 3) {
editor?.chain().focus().toggleHeading({ level }).run() editor?.chain().focus().toggleHeading({ level }).run()
styleDropdownOpen = false
} }
function setParagraph() { function setParagraph() {
editor?.chain().focus().setParagraph().run() editor?.chain().focus().setParagraph().run()
styleDropdownOpen = false
} }
function toggleLink() { function toggleLink() {
@ -90,12 +87,9 @@
<div class="toolbar-container"> <div class="toolbar-container">
<div class="description-toolbar"> <div class="description-toolbar">
<!-- Text Style Dropdown --> <!-- Text Style Dropdown -->
<div class="style-dropdown-wrapper"> <DropdownMenu>
<button {#snippet trigger({ props })}
class="toolbar-button style-trigger" <button class="toolbar-button style-trigger" disabled={!editor} {...props}>
onclick={() => (styleDropdownOpen = !styleDropdownOpen)}
disabled={!editor}
>
{#if editor?.isActive('heading', { level: 1 })} {#if editor?.isActive('heading', { level: 1 })}
<Heading1 size={16} /> <Heading1 size={16} />
{:else if editor?.isActive('heading', { level: 2 })} {:else if editor?.isActive('heading', { level: 2 })}
@ -108,46 +102,38 @@
<span>{getStyleLabel()}</span> <span>{getStyleLabel()}</span>
<ChevronDown size={12} /> <ChevronDown size={12} />
</button> </button>
{/snippet}
{#if styleDropdownOpen} {#snippet menu()}
<!-- svelte-ignore a11y_no_static_element_interactions --> <DropdownMenuBase.Item
<!-- svelte-ignore a11y_click_events_have_key_events --> class="dropdown-menu-item {editor?.isActive('heading', { level: 1 }) ? 'active' : ''}"
<div class="style-dropdown" onclick={(e) => e.stopPropagation()}> onSelect={() => setHeading(1)}
<button
class="style-item"
class:active={editor?.isActive('heading', { level: 1 })}
onclick={() => setHeading(1)}
> >
<Heading1 size={16} /> <Heading1 size={16} />
<span>Heading 1</span> <span>Heading 1</span>
</button> </DropdownMenuBase.Item>
<button <DropdownMenuBase.Item
class="style-item" class="dropdown-menu-item {editor?.isActive('heading', { level: 2 }) ? 'active' : ''}"
class:active={editor?.isActive('heading', { level: 2 })} onSelect={() => setHeading(2)}
onclick={() => setHeading(2)}
> >
<Heading2 size={16} /> <Heading2 size={16} />
<span>Heading 2</span> <span>Heading 2</span>
</button> </DropdownMenuBase.Item>
<button <DropdownMenuBase.Item
class="style-item" class="dropdown-menu-item {editor?.isActive('heading', { level: 3 }) ? 'active' : ''}"
class:active={editor?.isActive('heading', { level: 3 })} onSelect={() => setHeading(3)}
onclick={() => setHeading(3)}
> >
<Heading3 size={16} /> <Heading3 size={16} />
<span>Heading 3</span> <span>Heading 3</span>
</button> </DropdownMenuBase.Item>
<button <DropdownMenuBase.Item
class="style-item" class="dropdown-menu-item {editor?.isActive('paragraph') ? 'active' : ''}"
class:active={editor?.isActive('paragraph')} onSelect={setParagraph}
onclick={setParagraph}
> >
<Pilcrow size={16} /> <Pilcrow size={16} />
<span>Paragraph</span> <span>Paragraph</span>
</button> </DropdownMenuBase.Item>
</div> {/snippet}
{/if} </DropdownMenu>
</div>
<div class="separator"></div> <div class="separator"></div>
@ -240,9 +226,6 @@
</div> </div>
</div> </div>
<!-- Close dropdown when clicking outside -->
<svelte:window onclick={() => (styleDropdownOpen = false)} />
<style lang="scss"> <style lang="scss">
@use '$src/themes/spacing' as *; @use '$src/themes/spacing' as *;
@use '$src/themes/layout' as *; @use '$src/themes/layout' as *;
@ -308,57 +291,19 @@
} }
} }
.style-dropdown-wrapper {
position: relative;
}
.style-trigger { .style-trigger {
display: flex; display: flex;
align-items: center; align-items: center;
gap: $unit-half; gap: $unit-half;
width: auto; width: auto;
min-width: 56px; min-width: 100px;
padding: 0 $unit; padding: 0 $unit;
font-size: $font-small; font-size: $font-small;
font-weight: $medium; font-weight: $medium;
} }
.style-dropdown { :global(.dropdown-menu-item.active) {
position: absolute; background: var(--button-bg-active);
top: 100%;
left: 0;
margin-top: 4px;
background: var(--menu-bg);
border: 1px solid var(--menu-border);
border-radius: $card-corner;
padding: $unit-half;
min-width: 140px;
box-shadow: var(--shadow-floating);
z-index: 100;
}
.style-item {
display: flex;
align-items: center;
gap: $unit;
width: 100%;
padding: $unit ($unit * 1.5);
border: none;
background: transparent;
border-radius: 6px;
font-size: $font-small;
font-weight: $medium;
color: var(--text-primary);
cursor: pointer;
text-align: left;
&:hover {
background: var(--menu-bg-item-hover);
}
&.active {
background: var(--menu-bg-item-active);
}
} }
.editor-container { .editor-container {