| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  | import { | 
					
						
							|  |  |  |   customElement, | 
					
						
							|  |  |  |   html, | 
					
						
							|  |  |  |   DeesElement, | 
					
						
							|  |  |  |   type TemplateResult, | 
					
						
							|  |  |  |   cssManager, | 
					
						
							|  |  |  |   css, | 
					
						
							|  |  |  |   state, | 
					
						
							|  |  |  | } from '@design.estate/dees-element'; | 
					
						
							| 
									
										
										
										
											2025-06-26 20:20:34 +00:00
										 |  |  | import { zIndexRegistry } from '../00zindex.js'; | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  | import '../dees-icon.js'; | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import { type ISlashMenuItem } from './wysiwyg.types.js'; | 
					
						
							|  |  |  | import { WysiwygShortcuts } from './wysiwyg.shortcuts.js'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | declare global { | 
					
						
							|  |  |  |   interface HTMLElementTagNameMap { | 
					
						
							|  |  |  |     'dees-slash-menu': DeesSlashMenu; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @customElement('dees-slash-menu') | 
					
						
							|  |  |  | export class DeesSlashMenu extends DeesElement { | 
					
						
							|  |  |  |   private static instance: DeesSlashMenu; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   public static getInstance(): DeesSlashMenu { | 
					
						
							|  |  |  |     if (!DeesSlashMenu.instance) { | 
					
						
							|  |  |  |       DeesSlashMenu.instance = new DeesSlashMenu(); | 
					
						
							|  |  |  |       document.body.appendChild(DeesSlashMenu.instance); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return DeesSlashMenu.instance; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @state() | 
					
						
							|  |  |  |   public visible: boolean = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @state() | 
					
						
							|  |  |  |   private position: { x: number; y: number } = { x: 0, y: 0 }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @state() | 
					
						
							|  |  |  |   private filter: string = ''; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @state() | 
					
						
							|  |  |  |   private selectedIndex: number = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-26 20:20:34 +00:00
										 |  |  |   @state() | 
					
						
							|  |  |  |   private menuZIndex: number = 1000; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |   private callback: ((type: string) => void) | null = null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   public static styles = [ | 
					
						
							|  |  |  |     cssManager.defaultStyles, | 
					
						
							|  |  |  |     css`
 | 
					
						
							|  |  |  |       :host { | 
					
						
							|  |  |  |         position: fixed; | 
					
						
							|  |  |  |         pointer-events: none; | 
					
						
							| 
									
										
										
										
											2025-06-27 00:35:06 +00:00
										 |  |  |         top: 0; | 
					
						
							|  |  |  |         left: 0; | 
					
						
							|  |  |  |         width: 0; | 
					
						
							|  |  |  |         height: 0; | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       .slash-menu { | 
					
						
							| 
									
										
										
										
											2025-06-27 00:35:06 +00:00
										 |  |  |         position: fixed; | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  |         background: ${cssManager.bdTheme('#ffffff', '#09090b')}; | 
					
						
							|  |  |  |         border: 1px solid ${cssManager.bdTheme('#e5e7eb', '#27272a')}; | 
					
						
							|  |  |  |         border-radius: 4px; | 
					
						
							|  |  |  |         box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1), 0 1px 2px rgba(0, 0, 0, 0.06); | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |         padding: 4px; | 
					
						
							|  |  |  |         min-width: 220px; | 
					
						
							|  |  |  |         max-height: 300px; | 
					
						
							|  |  |  |         overflow-y: auto; | 
					
						
							|  |  |  |         pointer-events: auto; | 
					
						
							|  |  |  |         user-select: none; | 
					
						
							|  |  |  |         animation: fadeInScale 0.15s ease-out; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       @keyframes fadeInScale { | 
					
						
							|  |  |  |         from { | 
					
						
							|  |  |  |           opacity: 0; | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  |           transform: scale(0.98) translateY(-2px); | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         to { | 
					
						
							|  |  |  |           opacity: 1; | 
					
						
							|  |  |  |           transform: scale(1) translateY(0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       .slash-menu-item { | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  |         padding: 8px 10px; | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |         cursor: pointer; | 
					
						
							|  |  |  |         transition: all 0.15s ease; | 
					
						
							|  |  |  |         display: flex; | 
					
						
							|  |  |  |         align-items: center; | 
					
						
							|  |  |  |         gap: 12px; | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  |         border-radius: 3px; | 
					
						
							|  |  |  |         color: ${cssManager.bdTheme('#09090b', '#fafafa')}; | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |         font-size: 14px; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       .slash-menu-item:hover, | 
					
						
							|  |  |  |       .slash-menu-item.selected { | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  |         background: ${cssManager.bdTheme('#f4f4f5', '#27272a')}; | 
					
						
							|  |  |  |         color: ${cssManager.bdTheme('#09090b', '#fafafa')}; | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       .slash-menu-item .icon { | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  |         width: 20px; | 
					
						
							|  |  |  |         height: 20px; | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |         display: flex; | 
					
						
							|  |  |  |         align-items: center; | 
					
						
							|  |  |  |         justify-content: center; | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  |         color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       .slash-menu-item:hover .icon, | 
					
						
							|  |  |  |       .slash-menu-item.selected .icon { | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  |         color: ${cssManager.bdTheme('#3b82f6', '#3b82f6')}; | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     `,
 | 
					
						
							|  |  |  |   ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   render(): TemplateResult { | 
					
						
							|  |  |  |     if (!this.visible) return html``; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-27 00:35:06 +00:00
										 |  |  |     // Ensure z-index is applied to host element
 | 
					
						
							| 
									
										
										
										
											2025-06-26 20:20:34 +00:00
										 |  |  |     this.style.zIndex = this.menuZIndex.toString(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |     const menuItems = this.getFilteredMenuItems(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return html`
 | 
					
						
							|  |  |  |       <div  | 
					
						
							|  |  |  |         class="slash-menu"  | 
					
						
							|  |  |  |         style="left: ${this.position.x}px; top: ${this.position.y}px;" | 
					
						
							|  |  |  |         tabindex="-1" | 
					
						
							| 
									
										
										
										
											2025-06-24 16:49:40 +00:00
										 |  |  |         data-menu-type="slash" | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |       > | 
					
						
							|  |  |  |         ${menuItems.map((item, index) => html`
 | 
					
						
							|  |  |  |           <div  | 
					
						
							|  |  |  |             class="slash-menu-item ${index === this.selectedIndex ? 'selected' : ''}" | 
					
						
							| 
									
										
										
										
											2025-06-24 16:49:40 +00:00
										 |  |  |             data-item-type="${item.type}" | 
					
						
							|  |  |  |             data-item-index="${index}" | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |           > | 
					
						
							| 
									
										
										
										
											2025-06-27 18:38:39 +00:00
										 |  |  |             <dees-icon class="icon" .icon="${item.icon}" iconSize="16"></dees-icon> | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |             <span>${item.label}</span> | 
					
						
							|  |  |  |           </div> | 
					
						
							|  |  |  |         `)}
 | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     `;
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   private getFilteredMenuItems(): ISlashMenuItem[] { | 
					
						
							|  |  |  |     const allItems = WysiwygShortcuts.getSlashMenuItems(); | 
					
						
							|  |  |  |     return allItems.filter(item =>  | 
					
						
							|  |  |  |       this.filter === '' ||  | 
					
						
							|  |  |  |       item.label.toLowerCase().includes(this.filter.toLowerCase()) | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   private selectItem(type: string): void { | 
					
						
							|  |  |  |     if (this.callback) { | 
					
						
							|  |  |  |       this.callback(type); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     this.hide(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   public show(position: { x: number; y: number }, callback: (type: string) => void): void { | 
					
						
							|  |  |  |     this.position = position; | 
					
						
							|  |  |  |     this.callback = callback; | 
					
						
							|  |  |  |     this.filter = ''; | 
					
						
							|  |  |  |     this.selectedIndex = 0; | 
					
						
							| 
									
										
										
										
											2025-06-26 20:20:34 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2025-06-27 00:35:06 +00:00
										 |  |  |     // Get z-index from registry and apply immediately
 | 
					
						
							| 
									
										
										
										
											2025-06-26 20:20:34 +00:00
										 |  |  |     this.menuZIndex = zIndexRegistry.getNextZIndex(); | 
					
						
							|  |  |  |     zIndexRegistry.register(this, this.menuZIndex); | 
					
						
							| 
									
										
										
										
											2025-06-27 00:35:06 +00:00
										 |  |  |     this.style.zIndex = this.menuZIndex.toString(); | 
					
						
							| 
									
										
										
										
											2025-06-26 20:20:34 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |     this.visible = true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   public hide(): void { | 
					
						
							|  |  |  |     this.visible = false; | 
					
						
							|  |  |  |     this.callback = null; | 
					
						
							|  |  |  |     this.filter = ''; | 
					
						
							|  |  |  |     this.selectedIndex = 0; | 
					
						
							| 
									
										
										
										
											2025-06-26 20:20:34 +00:00
										 |  |  |      | 
					
						
							|  |  |  |     // Unregister from z-index registry
 | 
					
						
							|  |  |  |     zIndexRegistry.unregister(this); | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   public updateFilter(filter: string): void { | 
					
						
							|  |  |  |     this.filter = filter; | 
					
						
							|  |  |  |     this.selectedIndex = 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   public navigate(direction: 'up' | 'down'): void { | 
					
						
							|  |  |  |     const items = this.getFilteredMenuItems(); | 
					
						
							|  |  |  |     if (direction === 'down') { | 
					
						
							|  |  |  |       this.selectedIndex = (this.selectedIndex + 1) % items.length; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       this.selectedIndex = this.selectedIndex === 0  | 
					
						
							|  |  |  |         ? items.length - 1  | 
					
						
							|  |  |  |         : this.selectedIndex - 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   public selectCurrent(): void { | 
					
						
							|  |  |  |     const items = this.getFilteredMenuItems(); | 
					
						
							|  |  |  |     if (items[this.selectedIndex]) { | 
					
						
							|  |  |  |       this.selectItem(items[this.selectedIndex].type); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2025-06-24 16:49:40 +00:00
										 |  |  |    | 
					
						
							|  |  |  |   public firstUpdated(): void { | 
					
						
							|  |  |  |     // Set up event delegation
 | 
					
						
							|  |  |  |     this.shadowRoot?.addEventListener('mousedown', (e: MouseEvent) => { | 
					
						
							|  |  |  |       const menu = this.shadowRoot?.querySelector('.slash-menu'); | 
					
						
							|  |  |  |       if (menu && menu.contains(e.target as Node)) { | 
					
						
							|  |  |  |         // Prevent focus loss
 | 
					
						
							|  |  |  |         e.preventDefault(); | 
					
						
							|  |  |  |         e.stopPropagation(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     this.shadowRoot?.addEventListener('click', (e: MouseEvent) => { | 
					
						
							|  |  |  |       const target = e.target as HTMLElement; | 
					
						
							|  |  |  |       const menuItem = target.closest('.slash-menu-item') as HTMLElement; | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       if (menuItem) { | 
					
						
							|  |  |  |         e.preventDefault(); | 
					
						
							|  |  |  |         e.stopPropagation(); | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         const itemType = menuItem.getAttribute('data-item-type'); | 
					
						
							|  |  |  |         if (itemType) { | 
					
						
							|  |  |  |           this.selectItem(itemType); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     this.shadowRoot?.addEventListener('mouseenter', (e: MouseEvent) => { | 
					
						
							|  |  |  |       const target = e.target as HTMLElement; | 
					
						
							|  |  |  |       const menuItem = target.closest('.slash-menu-item') as HTMLElement; | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       if (menuItem) { | 
					
						
							|  |  |  |         const index = parseInt(menuItem.getAttribute('data-item-index') || '0', 10); | 
					
						
							|  |  |  |         this.selectedIndex = index; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }, true); // Use capture phase
 | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     this.shadowRoot?.addEventListener('focus', (e: FocusEvent) => { | 
					
						
							|  |  |  |       const menu = this.shadowRoot?.querySelector('.slash-menu'); | 
					
						
							|  |  |  |       if (menu && menu.contains(e.target as Node)) { | 
					
						
							|  |  |  |         // Prevent menu from taking focus
 | 
					
						
							|  |  |  |         e.preventDefault(); | 
					
						
							|  |  |  |         e.stopPropagation(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }, true); // Use capture phase
 | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2025-06-24 10:45:06 +00:00
										 |  |  | } |