quave:alert-react-tailwind
quave:alert-react-tailwind
is a Meteor package that provides a plug-and-play alert system for React with TailwindCSS.
Why
It is designed to simplify the process of showing alerts to users inside the app.
We are using tailwindcss
styles.
We believe we are not reinventing the wheel in this package but what we are doing is like putting together the wheels in the vehicle :).
Installation
meteor add quave:alert-react-tailwind
You should have the following npm dependencies in your project:
- react
- react-dom
- react-router-dom
Usage
Configuration
You need to declare the context provider for the alerts, we recommend right after the Router from React Router, see AlertProvider
in the example below:
1import React from 'react'; 2import { BrowserRouter as Router } from 'react-router-dom'; 3import { Routes } from './Routes'; 4import { AlertProvider, Alert } from 'meteor/quave:alert-react-tailwind'; 5 6// classnames tailwind alert: 7// fixed inset-0 flex items-end px-4 py-6 pointer-events-none z-10 sm:p-6 sm:items-start w-full flex-col items-center space-y-4 sm:items-end transform ease-out duration-300 transition translate-y-2 opacity-0 sm:translate-y-0 sm:translate-x-2 translate-y-0 opacity-100 sm:translate-x-0 ease-in duration-100 max-w-sm bg-white shadow-lg rounded-lg pointer-events-auto ring-1 ring-black ring-opacity-5 overflow-hidden p-4 items-start flex-shrink-0 h-6 w-6 text-gray-400 text-red-500 text-green-500 ml-3 w-0 flex-1 pt-0.5 text-sm font-medium text-gray-900 mt-1 text-gray-500 mt-3 space-x-7 rounded-md text-indigo-600 hover:text-indigo-500 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 text-gray-700 hover:text-gray-500 ml-4 inline-flex sr-only h-5 w-5 8 9export const App = () => ( 10 <Router> 11 <AlertProvider> 12 <div className="bg-indigo-50 h-full"> 13 <Alert /> 14 <div className="max-w-7xl mx-auto py-12 px-4 sm:px-6 lg:py-24 lg:px-8 lg:flex lg:items-center lg:justify-between"> 15 <Routes /> 16 </div> 17 </div> 18 </AlertProvider> 19 </Router> 20);
This alert is a small pop-over that will appear on top of the page. You need to render it somewhere in your app as well. See Alert
in the example above.
You can also provide your custom component to render the alert, see Component
prop in the Alert
component.
1const MyAlert = ({ message, isOpen, clear }) => { 2 if (!message || !isOpen) return null; 3 4 return ( 5 <div className="fixed right-4 top-4 z-50"> 6 <div className="max-w-sm rounded-lg bg-white p-4 shadow-lg"> 7 <div className="flex items-center justify-between"> 8 <p className="text-gray-800">{message}</p> 9 <button 10 onClick={clear} 11 aria-label="Clear message" 12 className="ml-4 text-gray-500 hover:text-gray-700 focus:outline-none" 13 > 14 <svg className="h-5 w-5" fill="currentColor" viewBox="0 0 20 20"> 15 <path 16 fillRule="evenodd" 17 d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" 18 clipRule="evenodd" 19 /> 20 </svg> 21 </button> 22 </div> 23 </div> 24 </div> 25 ); 26}; 27 28 29export function App() { 30 return ( 31 <BrowserRouter> 32 <Suspense 33 fallback={ 34 <PageWithHeader> 35 <Loading name="suspense" /> 36 </PageWithHeader> 37 } 38 > 39 <AlertProvider> 40 <div className="h-full bg-indigo-50 p-4 lg:p-24"> 41 <Alert Component={MyAlert} /> 42 <Router /> 43 </div> 44 </AlertProvider> 45 </Suspense> 46 </BrowserRouter> 47 ); 48}
You can provide 3 icons as props to the Alert
component if you only want to choose icons for the alert:
ExclamationIcon
MessageIcon
XIcon
Tailwind
By default, we provide tailwindcss classes, you should include safelist
property in your tailwind.config.js
1 safelist: [ 2 'bg-white', 3 'duration-100', 4 'duration-300', 5 'ease-in', 6 'ease-out', 7 'fixed', 8 'flex', 9 'flex-1', 10 'flex-col', 11 'flex-shrink-0', 12 'focus:outline-none', 13 'focus:ring-2', 14 'focus:ring-indigo-500', 15 'focus:ring-offset-2', 16 'font-medium', 17 'h-5', 18 'h-6', 19 'hover:text-gray-500', 20 'hover:text-indigo-500', 21 'inline-flex', 22 'inset-0', 23 'items-center', 24 'items-end', 25 'items-start', 26 'max-w-sm', 27 'ml-3', 28 'ml-4', 29 'mt-1', 30 'mt-3', 31 'opacity-0', 32 'opacity-100', 33 'overflow-hidden', 34 'p-4', 35 'pointer-events-auto', 36 'pointer-events-none', 37 'pt-0.5', 38 'px-4', 39 'py-6', 40 'ring-1', 41 'ring-black', 42 'ring-opacity-5', 43 'rounded-lg', 44 'rounded-md', 45 'shadow-lg', 46 'sm:items-end', 47 'sm:items-start', 48 'sm:p-6', 49 'sm:translate-x-0', 50 'sm:translate-x-2', 51 'sm:translate-y-0', 52 'space-x-7', 53 'space-y-4', 54 'sr-only', 55 'text-gray-400', 56 'text-gray-500', 57 'text-gray-700', 58 'text-gray-900', 59 'text-green-500', 60 'text-indigo-600', 61 'text-red-500', 62 'text-sm', 63 'transform', 64 'transition', 65 'translate-y-0', 66 'translate-y-2', 67 'w-0', 68 'w-5', 69 'w-6', 70 'w-full', 71 'z-10', 72 ]
or this comment somewhere in your code to avoid purging these classes:
1// classnames tailwind alert: 2// fixed inset-0 flex items-end px-4 py-6 pointer-events-none z-10 sm:p-6 sm:items-start w-full flex-col items-center space-y-4 sm:items-end transform ease-out duration-300 transition translate-y-2 opacity-0 sm:translate-y-0 sm:translate-x-2 translate-y-0 opacity-100 sm:translate-x-0 ease-in duration-100 max-w-sm bg-white shadow-lg rounded-lg pointer-events-auto ring-1 ring-black ring-opacity-5 overflow-hidden p-4 items-start flex-shrink-0 h-6 w-6 text-gray-400 text-red-500 text-green-500 ml-3 w-0 flex-1 pt-0.5 text-sm font-medium text-gray-900 mt-1 text-gray-500 mt-3 space-x-7 rounded-md text-indigo-600 hover:text-indigo-500 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 text-gray-700 hover:text-gray-500 ml-4 inline-flex sr-only h-5 w-5
Code usage
If you want to show an alert, you need to call openAlert
, this function is produced by useAlert
hook, see the example below:
1import React from 'react'; 2import { useNavigate } from 'react-router-dom'; 3import { Passwordless } from 'meteor/quave:accounts-passwordless-react'; 4import { useLoggedUser } from 'meteor/quave:logged-user-react'; 5import { useAlert } from 'meteor/quave:alert-react-tailwind'; 6 7import { RoutePaths } from '../general/RoutePaths'; 8 9export const Access = () => { 10 const { openAlert } = useAlert(); 11 const { loggedUser } = useLoggedUser(); 12 const navigate = useNavigate(); 13 14 const onEnterToken = () => { 15 navigate(RoutePaths.HOME); 16 openAlert('Welcome!'); 17 }; 18 19 if (loggedUser) { 20 return ( 21 <div className="flex flex-col items-center"> 22 <h3 className="text-lg px-3 py-2 text-base font-medium"> 23 You are already authenticated. 24 </h3> 25 <button onClick={() => navigate(RoutePaths.HOME)} type="button"> 26 Go Home 27 </button> 28 </div> 29 ); 30 } 31 return ( 32 <div className="flex flex-col items-center flex-grow"> 33 <Passwordless onEnterToken={onEnterToken} /> 34 <a 35 onClick={() => navigate(RoutePaths.HOME)} 36 className="mt-5 text-base font-medium text-indigo-700 hover:text-indigo-600 cursor-pointer" 37 > 38 <span aria-hidden="true"> →</span> Back to Home 39 </a> 40 </div> 41 ); 42};
We produce other utility functions as well, see AlertContext.Provider
value in the code if you have more complex use cases. You can also submit PRs adding more examples to this readme.
Changes
If you want to keep your project up-to-date with the changes made here, read our CHANGELOG.
Limitations
N/A
License
MIT