mirror of
				https://github.com/scinote-eln/scinote-web.git
				synced 2025-10-25 05:27:33 +08:00 
			
		
		
		
	Add a listener for when TinyMCE is opened and adjust headers [SCI-9449]
This commit is contained in:
		
							parent
							
								
									4e25b3921a
								
							
						
					
					
						commit
						a93c35a3be
					
				
					 2 changed files with 95 additions and 13 deletions
				
			
		
							
								
								
									
										19
									
								
								app/javascript/packs/tiny_mce.js
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								app/javascript/packs/tiny_mce.js
									
										
									
									
										vendored
									
									
								
							|  | @ -178,10 +178,12 @@ window.TinyMCE = (() => { | |||
|       if (typeof tinyMCE !== 'undefined') { | ||||
|         // Hide element containing HTML view of RTE field
 | ||||
|         const tinyMceContainer = $(selector).closest('form').find('.tinymce-view'); | ||||
|         const editorForm = $(selector).closest('form'); | ||||
|         const tinyMceInitSize = tinyMceContainer.height(); | ||||
|         $(selector).closest('.form-group, .tinymce-editor-container') | ||||
|           .before(`<div class="tinymce-placeholder" style="height:${tinyMceInitSize}px"></div>`); | ||||
| 
 | ||||
|         editorForm.parent().height(tinyMceInitSize); | ||||
|         tinyMceContainer.addClass('hidden'); | ||||
| 
 | ||||
|         const plugins = ` | ||||
|           image table autosave autoresize link advlist codesample code autolink lists | ||||
|           charmap anchor searchreplace wordcount visualblocks visualchars | ||||
|  | @ -301,11 +303,14 @@ window.TinyMCE = (() => { | |||
|             const editorForm = editorContainer.closest('form'); | ||||
|             const menuBar = editorForm.find('.tox-menubar'); | ||||
| 
 | ||||
|             $('.tinymce-placeholder').css('height', `${$(editor.editorContainer).height()}px`); | ||||
|             setTimeout(() => { | ||||
|               editorContainer.addClass('tox-tinymce--loaded'); | ||||
|               $('.tinymce-placeholder').remove(); | ||||
|             }, 400); | ||||
|             editorContainer.addClass('tox-tinymce--loaded'); | ||||
|             const event = new CustomEvent('tinyMCEOpened', { | ||||
|               detail: { | ||||
|                 target: editorForm.parent(), | ||||
|               } | ||||
|             }); | ||||
|             window.dispatchEvent(event); | ||||
|             editorForm.parent().css('height', ''); | ||||
| 
 | ||||
|             // Init saved status label
 | ||||
|             if (editor.getContent() !== '') { | ||||
|  |  | |||
|  | @ -22,20 +22,97 @@ export default { | |||
|   mounted() { | ||||
|     this.secondaryNavigation = document.querySelector('#taskSecondaryMenu'); | ||||
| 
 | ||||
|     this.resizeObserver = new ResizeObserver((entries) => { | ||||
|       entries.forEach((entry) => { | ||||
|         this.taskSecondaryMenuHeight = entry.target.offsetHeight; | ||||
|     if (this.secondaryNavigation) { | ||||
|       this.resizeObserver = new ResizeObserver((entries) => { | ||||
|         entries.forEach((entry) => { | ||||
|           this.taskSecondaryMenuHeight = entry.target.offsetHeight; | ||||
|         }); | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     this.resizeObserver.observe(this.secondaryNavigation); | ||||
|       this.resizeObserver.observe(this.secondaryNavigation); | ||||
|     } | ||||
|     window.addEventListener('tinyMCEOpened', (e) => { | ||||
|       this.handleTinyMCEOpened(e.detail.target); | ||||
|     }); | ||||
|   }, | ||||
|   beforeDestroy() { | ||||
|     if (this.resizeObserver) { | ||||
|       this.resizeObserver.disconnect(); | ||||
|     } | ||||
|     window.removeEventListener('tinyMCEOpened', this.handleTinyMCEOpened); | ||||
|   }, | ||||
|   methods: { | ||||
|     handleTinyMCEOpened(target) { | ||||
|       const getVisibleHeight = (elemTop, elemHeight) => { | ||||
|         let visibleHeight = 0; | ||||
|         if (elemTop >= 0) { | ||||
|           visibleHeight = Math.min(elemHeight, window.innerHeight - elemTop); | ||||
|         } else if (elemTop + elemHeight > 0) { | ||||
|           visibleHeight = elemTop + elemHeight; | ||||
|         } | ||||
|         return visibleHeight; | ||||
|       }; | ||||
| 
 | ||||
|       let headerHeight = 0; | ||||
|       let headerTop = 0; | ||||
|       let secondaryNavigationHeight = 0; | ||||
|       let secondaryNavigationTop = 0; | ||||
| 
 | ||||
|       if (this.headerRef) { | ||||
|         headerHeight = this.headerRef.offsetHeight; | ||||
|         headerTop = this.headerRef.getBoundingClientRect().top; | ||||
|       } | ||||
| 
 | ||||
|       if (this.secondaryNavigation) { | ||||
|         secondaryNavigationHeight = this.secondaryNavigation.offsetHeight; | ||||
|         secondaryNavigationTop = this.secondaryNavigation.getBoundingClientRect().top; | ||||
|       } | ||||
| 
 | ||||
|       const editorHeaderTop = target.offset().top; | ||||
|       let totalHeight = 0; | ||||
| 
 | ||||
|       const visibleHeaderHeight = getVisibleHeight(headerTop, headerHeight); | ||||
|       if (headerTop + visibleHeaderHeight < editorHeaderTop) { | ||||
|         totalHeight += visibleHeaderHeight; | ||||
|       } | ||||
| 
 | ||||
|       const visibleSecondaryNavHeight = getVisibleHeight(secondaryNavigationTop, secondaryNavigationHeight); | ||||
|       if (secondaryNavigationTop + visibleSecondaryNavHeight < editorHeaderTop) { | ||||
|         totalHeight += visibleSecondaryNavHeight; | ||||
|       } | ||||
| 
 | ||||
|       const editorHeader = $('.tox-editor-header'); | ||||
| 
 | ||||
|       // For Protocol Templates only reset the left value
 | ||||
|       if (!this.headerRef && !this.secondaryNavigation) { | ||||
|         editorHeader.css('left', ''); | ||||
|         return; | ||||
|       } | ||||
| 
 | ||||
|       // Handle opening TinyMCE toolbars when only a small bottom area of editor is visible
 | ||||
|       const targetBottom = target[0].getBoundingClientRect().bottom; | ||||
|       if (targetBottom < 3 * headerHeight) { | ||||
|         $('html, body').animate({ | ||||
|           scrollTop: target.offset().top, | ||||
|         }, 800); | ||||
|         return; | ||||
|       } | ||||
| 
 | ||||
|       // Handle showing TinyMCE toolbar for fixed/static position of toolbar
 | ||||
|       if (editorHeader.css('position') === 'fixed') { | ||||
|         editorHeader.css({ | ||||
|           top: totalHeight - 1, | ||||
|           left: '', | ||||
|         }); | ||||
|       } else if (headerTop < (visibleHeaderHeight + visibleSecondaryNavHeight) | ||||
|         && target[0].getBoundingClientRect().top <= headerTop) { | ||||
|         $('html, body').animate({ | ||||
|               scrollTop: editorHeader.offset().top | ||||
|           }, 800); | ||||
|       } | ||||
| 
 | ||||
|       target.focus(); | ||||
|     }, | ||||
|     initStackableHeaders() { | ||||
|       const header = this.headerRef; | ||||
|       const headerHeight = header.offsetHeight; | ||||
|  | @ -93,7 +170,7 @@ export default { | |||
|       // Apply TinyMCE offset
 | ||||
|       $('.tox-editor-header').css( | ||||
|         'top', | ||||
|         stickyNavigationHeight + parseInt($(this.secondaryNavigation).css('top'), 10) | ||||
|         stickyNavigationHeight + parseInt($(this.secondaryNavigation).css('top'), 10) - 1, | ||||
|       ); | ||||
|       this.lastScrollTop = window.scrollY; // Save last scroll position to when user scroll up/down
 | ||||
|     }, | ||||
|  |  | |||
		Loading…
	
	Add table
		
		Reference in a new issue