feat: add snackbar alert with refresh button (#1193)
* feat: add snackbar alert with refresh button fixes #77 * fixup * change refresh to reload
This commit is contained in:
		
							parent
							
								
									c56d561e9d
								
							
						
					
					
						commit
						0887196db4
					
				
					 7 changed files with 173 additions and 2 deletions
				
			
		|  | @ -45,6 +45,9 @@ | |||
|   <!-- Toast.html gets rendered here --> | ||||
|   <div id="theToast"></div> | ||||
| 
 | ||||
|   <!-- Snackbar.html gets rendered here --> | ||||
|   <div id="theSnackbar"></div> | ||||
| 
 | ||||
|   <!-- LoadingMask.html gets rendered here --> | ||||
|   <div id="loading-mask" aria-hidden="true"></div> | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										137
									
								
								src/routes/_components/snackbar/Snackbar.html
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										137
									
								
								src/routes/_components/snackbar/Snackbar.html
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,137 @@ | |||
| <section class="snackbar-modal {shown ? 'shown' : ''}" | ||||
|      aria-live="assertive" | ||||
|      aria-atomic="true" | ||||
|      aria-hidden={!shown} | ||||
|      aria-label="Alert" | ||||
| > | ||||
|   <div class="snackbar-container"> | ||||
|     <span class="text"> | ||||
|       {text} | ||||
|     </span> | ||||
|     <div class="button-wrapper"> | ||||
|       <button class="button" on:click="onClick(event)"> | ||||
|         {buttonText} | ||||
|       </button> | ||||
|       <button class="button" aria-label="Close" on:click="close(event)"> | ||||
|         <SvgIcon className="close-snackbar-button" href="#fa-times" /> | ||||
|       </button> | ||||
|     </div> | ||||
|   </div> | ||||
| </section> | ||||
| <style> | ||||
|   .snackbar-modal { | ||||
|     position: fixed; | ||||
|     bottom: 0; | ||||
|     left: 0; | ||||
|     right: 0; | ||||
|     transition: transform 333ms ease-in-out; | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     align-items: center; | ||||
|     z-index: 99000; | ||||
|     transform: translateY(100%); | ||||
|   } | ||||
| 
 | ||||
|   .snackbar-container { | ||||
|     width: 562px; /* same as .main, minus 20px padding */ | ||||
|     overflow: hidden; | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     background: var(--toast-bg); | ||||
|     padding: 10px 20px; | ||||
|     font-size: 1.3em; | ||||
|     color: var(--toast-text); | ||||
|     border-radius: 4px 4px 0 0; | ||||
|     border: 1px solid var(--toast-border); | ||||
|     border-bottom: none; | ||||
|   } | ||||
| 
 | ||||
|   .button-wrapper { | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|   } | ||||
| 
 | ||||
|   .text { | ||||
|     flex: 1; | ||||
|   } | ||||
| 
 | ||||
|   button { | ||||
|     font-size: 1em; | ||||
|     border: none; | ||||
|     color: var(--toast-anchor-color); | ||||
|     text-transform: uppercase; | ||||
|     font-weight: 500; | ||||
|     background: var(--toast-bg); | ||||
|     margin-left: 5px; | ||||
|   } | ||||
| 
 | ||||
|   button:active { | ||||
|     background: var(--toast-button-active); | ||||
|   } | ||||
| 
 | ||||
|   button:hover { | ||||
|     background: var(--toast-button-hover); | ||||
|   } | ||||
| 
 | ||||
|   .snackbar-modal.shown { | ||||
|     transform: translateY(0); | ||||
|   } | ||||
| 
 | ||||
|   :global(.close-snackbar-button) { | ||||
|     margin-top: 3px; | ||||
|     width: 18px; | ||||
|     height: 18px; | ||||
|     fill: var(--toast-text); | ||||
|   } | ||||
| 
 | ||||
|   @media (max-width: 767px) { | ||||
|     .snackbar-container { | ||||
|       width: 100%; | ||||
|     } | ||||
|   } | ||||
| </style> | ||||
| <script> | ||||
|   import { doubleRAF } from '../../_utils/doubleRAF' | ||||
|   import SvgIcon from '../SvgIcon.html' | ||||
| 
 | ||||
|   export default { | ||||
|     data: () => ({ | ||||
|       shown: false, | ||||
|       text: '', | ||||
|       buttonText: '', | ||||
|       buttonAction: null | ||||
|     }), | ||||
|     methods: { | ||||
|       announce (text, buttonText, buttonAction) { | ||||
|         this.set({ text, buttonText, buttonAction }) | ||||
|         doubleRAF(() => { | ||||
|           this.set({ shown: true }) | ||||
|         }) | ||||
|       }, | ||||
|       onClick (e) { | ||||
|         e.preventDefault() | ||||
|         e.stopPropagation() | ||||
|         let { buttonAction } = this.get() | ||||
|         if (buttonAction) { | ||||
|           buttonAction() | ||||
|         } | ||||
|         this.close() | ||||
|       }, | ||||
|       close (e) { | ||||
|         if (e) { | ||||
|           e.preventDefault() | ||||
|           e.stopPropagation() | ||||
|         } | ||||
|         requestAnimationFrame(() => { | ||||
|           this.set({ | ||||
|             buttonAction: null, // avoid memory leaks from the closure | ||||
|             shown: false | ||||
|           }) | ||||
|         }) | ||||
|       } | ||||
|     }, | ||||
|     components: { | ||||
|       SvgIcon | ||||
|     } | ||||
|   } | ||||
| </script> | ||||
							
								
								
									
										22
									
								
								src/routes/_components/snackbar/snackbar.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/routes/_components/snackbar/snackbar.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | |||
| import { importSnackbar } from '../../_utils/asyncModules' | ||||
| 
 | ||||
| let snackbar | ||||
| 
 | ||||
| const lazySnackbar = { | ||||
|   async announce (text, buttonText, buttonAction) { | ||||
|     if (!snackbar) { | ||||
|       let Snackbar = await importSnackbar() | ||||
|       if (!snackbar) { | ||||
|         snackbar = new Snackbar({ | ||||
|           target: document.querySelector('#theSnackbar') | ||||
|         }) | ||||
|         if (process.env.NODE_ENV !== 'production') { | ||||
|           window.snackbar = snackbar // for debugging
 | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     snackbar.announce(text, buttonText, buttonAction) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export { lazySnackbar as snackbar } | ||||
|  | @ -47,3 +47,7 @@ export const importEmojiMart = () => import( | |||
| export const importToast = () => import( | ||||
|   /* webpackChunkName: 'Toast.html' */ '../_components/toast/Toast.html' | ||||
|   ).then(getDefault) | ||||
| 
 | ||||
| export const importSnackbar = () => import( | ||||
|   /* webpackChunkName: 'Snackbar.html' */ '../_components/snackbar/Snackbar.html' | ||||
|   ).then(getDefault) | ||||
|  |  | |||
|  | @ -1,11 +1,11 @@ | |||
| import { toast } from '../_components/toast/toast' | ||||
| import { snackbar } from '../_components/snackbar/snackbar' | ||||
| 
 | ||||
| function onUpdateFound (registration) { | ||||
|   const newWorker = registration.installing | ||||
| 
 | ||||
|   newWorker.addEventListener('statechange', async () => { | ||||
|     if (newWorker.state === 'installed' && navigator.serviceWorker.controller) { | ||||
|       toast.say('App update available. Reload to update.') | ||||
|       snackbar.announce('App update available.', 'Reload', () => document.location.reload(true)) | ||||
|     } | ||||
|   }) | ||||
| } | ||||
|  |  | |||
|  | @ -66,6 +66,9 @@ | |||
|   --toast-bg: #{$toast-bg}; | ||||
|   --toast-border: #{$toast-border}; | ||||
|   --toast-text: #{$secondary-text-color}; | ||||
|   --toast-button-hover: #{lighten($toast-bg, 5%)}; | ||||
|   --toast-button-active: #{lighten($toast-bg, 10%)}; | ||||
|   --toast-anchor-color: #{lighten($anchor-color, 20%)}; | ||||
| 
 | ||||
|   --mask-bg: #{$toast-bg}; | ||||
|   --mask-svg-fill: #{$secondary-text-color}; | ||||
|  |  | |||
|  | @ -44,4 +44,6 @@ | |||
|   --tab-bg-active: #{lighten($main-bg-color, 15%)}; | ||||
|   --tab-bg-hover: #{lighten($main-bg-color, 1%)}; | ||||
|   --tab-bg-hover-non-selected: #{darken($main-bg-color, 1%)}; | ||||
| 
 | ||||
|   --toast-anchor-color: #{$anchor-color}; | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		
		Reference in a new issue