| 
									
										
										
										
											2024-01-22 17:12:58 +01:00
										 |  |  | import { | 
					
						
							|  |  |  |   DeesElement, | 
					
						
							|  |  |  |   property, | 
					
						
							|  |  |  |   html, | 
					
						
							|  |  |  |   customElement, | 
					
						
							|  |  |  |   type TemplateResult, | 
					
						
							|  |  |  |   css, | 
					
						
							|  |  |  |   cssManager, | 
					
						
							|  |  |  | } from '@design.estate/dees-element'; | 
					
						
							|  |  |  | import * as domtools from '@design.estate/dees-domtools'; | 
					
						
							| 
									
										
										
										
											2025-09-18 13:39:59 +00:00
										 |  |  | import { MONACO_VERSION } from './version.js'; | 
					
						
							| 
									
										
										
										
											2024-01-22 17:12:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | import type * as monaco from 'monaco-editor'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | declare global { | 
					
						
							|  |  |  |   interface HTMLElementTagNameMap { | 
					
						
							|  |  |  |     'dees-editor': DeesEditor; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @customElement('dees-editor') | 
					
						
							|  |  |  | export class DeesEditor extends DeesElement { | 
					
						
							|  |  |  |   // DEMO
 | 
					
						
							|  |  |  |   public static demo = () => html` <dees-editor></dees-editor> `; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // STATIC
 | 
					
						
							|  |  |  |   public static monacoDeferred: ReturnType<typeof domtools.plugins.smartpromise.defer>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // INSTANCE
 | 
					
						
							|  |  |  |   public editorDeferred = domtools.plugins.smartpromise.defer<monaco.editor.IStandaloneCodeEditor>(); | 
					
						
							|  |  |  |   public language = 'typescript'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @property({ | 
					
						
							|  |  |  |     type: String | 
					
						
							|  |  |  |   }) | 
					
						
							|  |  |  |   public content = "function hello() {\n\talert('Hello world!');\n}"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @property({ | 
					
						
							|  |  |  |     type: Object | 
					
						
							|  |  |  |   }) | 
					
						
							|  |  |  |   public contentSubject = new domtools.plugins.smartrx.rxjs.Subject<string>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @property({ | 
					
						
							|  |  |  |     type: Boolean | 
					
						
							|  |  |  |   }) | 
					
						
							|  |  |  |   public wordWrap: monaco.editor.IStandaloneEditorConstructionOptions['wordWrap'] = 'off'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   constructor() { | 
					
						
							|  |  |  |     super(); | 
					
						
							|  |  |  |     domtools.DomTools.setupDomTools(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   public static styles = [ | 
					
						
							| 
									
										
										
										
											2024-02-05 13:12:26 +01:00
										 |  |  |     cssManager.defaultStyles, | 
					
						
							| 
									
										
										
										
											2024-01-22 17:12:58 +01:00
										 |  |  |     css`
 | 
					
						
							|  |  |  |       :host { | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       * { | 
					
						
							|  |  |  |         box-sizing: border-box; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       #container { | 
					
						
							|  |  |  |         position: absolute; | 
					
						
							|  |  |  |         height: 100%; | 
					
						
							|  |  |  |         width: 100%; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     `,
 | 
					
						
							|  |  |  |   ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   public render(): TemplateResult { | 
					
						
							|  |  |  |     return html`
 | 
					
						
							|  |  |  |       <div class="mainbox"> | 
					
						
							|  |  |  |         <div id="container"></div> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     `;
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   public async firstUpdated( | 
					
						
							|  |  |  |     _changedProperties: Map<string | number | symbol, unknown> | 
					
						
							|  |  |  |   ): Promise<void> { | 
					
						
							|  |  |  |     super.firstUpdated(_changedProperties); | 
					
						
							|  |  |  |     const container = this.shadowRoot.getElementById('container'); | 
					
						
							| 
									
										
										
										
											2025-09-18 13:39:59 +00:00
										 |  |  |     const monacoCdnBase = `https://cdn.jsdelivr.net/npm/monaco-editor@${MONACO_VERSION}`; | 
					
						
							| 
									
										
										
										
											2024-01-22 17:12:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!DeesEditor.monacoDeferred) { | 
					
						
							|  |  |  |       DeesEditor.monacoDeferred = domtools.plugins.smartpromise.defer(); | 
					
						
							| 
									
										
										
										
											2025-09-18 13:39:59 +00:00
										 |  |  |       const scriptUrl = `${monacoCdnBase}/min/vs/loader.js`; | 
					
						
							| 
									
										
										
										
											2024-01-22 17:12:58 +01:00
										 |  |  |       const script = document.createElement('script'); | 
					
						
							|  |  |  |       script.src = scriptUrl; | 
					
						
							|  |  |  |       script.onload = () => { | 
					
						
							|  |  |  |         DeesEditor.monacoDeferred.resolve(); | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |       document.head.appendChild(script); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     await DeesEditor.monacoDeferred.promise; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     (window as any).require.config({ | 
					
						
							| 
									
										
										
										
											2025-09-18 13:39:59 +00:00
										 |  |  |       paths: { vs: `${monacoCdnBase}/min/vs` }, | 
					
						
							| 
									
										
										
										
											2024-01-22 17:12:58 +01:00
										 |  |  |     }); | 
					
						
							|  |  |  |     (window as any).require(['vs/editor/editor.main'], async () => { | 
					
						
							|  |  |  |       const editor = ((window as any).monaco.editor as typeof monaco.editor).create(container, { | 
					
						
							|  |  |  |         value: this.content, | 
					
						
							|  |  |  |         language: this.language, | 
					
						
							|  |  |  |         theme: 'vs-dark', | 
					
						
							|  |  |  |         useShadowDOM: true, | 
					
						
							|  |  |  |         fontSize: 16, | 
					
						
							|  |  |  |         automaticLayout: true, | 
					
						
							|  |  |  |         wordWrap: this.wordWrap | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |       this.editorDeferred.resolve(editor); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     const css = await ( | 
					
						
							| 
									
										
										
										
											2025-09-18 13:39:59 +00:00
										 |  |  |       await fetch(`${monacoCdnBase}/min/vs/editor/editor.main.css`) | 
					
						
							| 
									
										
										
										
											2024-01-22 17:12:58 +01:00
										 |  |  |     ).text(); | 
					
						
							|  |  |  |     const styleElement = document.createElement('style'); | 
					
						
							|  |  |  |     styleElement.textContent = css; | 
					
						
							|  |  |  |     this.shadowRoot.append(styleElement); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // editor is setup let do the rest
 | 
					
						
							|  |  |  |     const editor = await this.editorDeferred.promise; | 
					
						
							|  |  |  |     editor.onDidChangeModelContent(async eventArg => { | 
					
						
							|  |  |  |       this.contentSubject.next(editor.getValue()); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     this.contentSubject.next(editor.getValue()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |