mirror of https://github.com/go-gitea/gitea
Add toasts to UI (#25449)
Fixes https://github.com/go-gitea/gitea/issues/24353 In some case like async success/error, it is useful to show toasts in UI.pull/25334/head
parent
72c60f94c1
commit
c71e8abbc3
@ -0,0 +1,78 @@ |
||||
.toastify { |
||||
color: var(--color-white); |
||||
position: fixed; |
||||
opacity: 0; |
||||
transition: all .2s ease; |
||||
z-index: 500; |
||||
border-radius: 4px; |
||||
box-shadow: 0 8px 24px var(--color-shadow); |
||||
display: flex; |
||||
max-width: 50vw; |
||||
min-width: 300px; |
||||
padding: 4px; |
||||
} |
||||
|
||||
.toastify.on { |
||||
opacity: 1; |
||||
} |
||||
|
||||
.toast-body { |
||||
flex: 1; |
||||
padding: 5px 0; |
||||
overflow-wrap: anywhere; |
||||
} |
||||
|
||||
.toast-close, |
||||
.toast-icon { |
||||
color: currentcolor; |
||||
border-radius: 3px; |
||||
background: transparent; |
||||
border: none; |
||||
display: inline-block; |
||||
display: flex; |
||||
width: 30px; |
||||
height: 30px; |
||||
justify-content: center; |
||||
align-items: center; |
||||
} |
||||
|
||||
.toast-close:hover { |
||||
background: var(--color-hover); |
||||
} |
||||
|
||||
.toast-close:active { |
||||
background: var(--color-active); |
||||
} |
||||
|
||||
.toastify-right { |
||||
right: 15px; |
||||
} |
||||
|
||||
.toastify-left { |
||||
left: 15px; |
||||
} |
||||
|
||||
.toastify-top { |
||||
top: -150px; |
||||
} |
||||
|
||||
.toastify-bottom { |
||||
bottom: -150px; |
||||
} |
||||
|
||||
.toastify-center { |
||||
margin-left: auto; |
||||
margin-right: auto; |
||||
left: 0; |
||||
right: 0; |
||||
} |
||||
|
||||
@media (max-width: 360px) { |
||||
.toastify-right, .toastify-left { |
||||
margin-left: auto; |
||||
margin-right: auto; |
||||
left: 0; |
||||
right: 0; |
||||
max-width: fit-content; |
||||
} |
||||
} |
@ -0,0 +1,16 @@ |
||||
.button-sample-groups { |
||||
margin: 0; padding: 0; |
||||
} |
||||
|
||||
.button-sample-groups .sample-group { |
||||
list-style: none; margin: 0; padding: 0; |
||||
} |
||||
|
||||
.button-sample-groups .sample-group .ui.button { |
||||
margin-bottom: 5px; |
||||
} |
||||
|
||||
h1, h2 { |
||||
margin: 0; |
||||
padding: 10px 0; |
||||
} |
@ -0,0 +1,60 @@ |
||||
import {htmlEscape} from 'escape-goat'; |
||||
import {svg} from '../svg.js'; |
||||
|
||||
const levels = { |
||||
info: { |
||||
icon: 'octicon-check', |
||||
background: 'var(--color-green)', |
||||
duration: 2500, |
||||
}, |
||||
warning: { |
||||
icon: 'gitea-exclamation', |
||||
background: 'var(--color-orange)', |
||||
duration: -1, // requires dismissal to hide
|
||||
}, |
||||
error: { |
||||
icon: 'gitea-exclamation', |
||||
background: 'var(--color-red)', |
||||
duration: -1, // requires dismissal to hide
|
||||
}, |
||||
}; |
||||
|
||||
// See https://github.com/apvarun/toastify-js#api for options
|
||||
async function showToast(message, level, {gravity, position, duration, ...other} = {}) { |
||||
if (!message) return; |
||||
|
||||
const {default: Toastify} = await import(/* webpackChunkName: 'toastify' */'toastify-js'); |
||||
const {icon, background, duration: levelDuration} = levels[level ?? 'info']; |
||||
|
||||
const toast = Toastify({ |
||||
text: ` |
||||
<div class='toast-icon'>${svg(icon)}</div> |
||||
<div class='toast-body'>${htmlEscape(message)}</div> |
||||
<button class='toast-close'>${svg('octicon-x')}</button> |
||||
`,
|
||||
escapeMarkup: false, |
||||
gravity: gravity ?? 'top', |
||||
position: position ?? 'center', |
||||
duration: duration ?? levelDuration, |
||||
style: {background}, |
||||
...other, |
||||
}); |
||||
|
||||
toast.showToast(); |
||||
|
||||
toast.toastElement.querySelector('.toast-close').addEventListener('click', () => { |
||||
toast.removeElement(toast.toastElement); |
||||
}); |
||||
} |
||||
|
||||
export async function showInfoToast(message, opts) { |
||||
return await showToast(message, 'info', opts); |
||||
} |
||||
|
||||
export async function showWarningToast(message, opts) { |
||||
return await showToast(message, 'warning', opts); |
||||
} |
||||
|
||||
export async function showErrorToast(message, opts) { |
||||
return await showToast(message, 'error', opts); |
||||
} |
@ -0,0 +1,17 @@ |
||||
import {test, expect} from 'vitest'; |
||||
import {showInfoToast, showErrorToast, showWarningToast} from './toast.js'; |
||||
|
||||
test('showInfoToast', async () => { |
||||
await showInfoToast('success 😀', {duration: -1}); |
||||
expect(document.querySelector('.toastify')).toBeTruthy(); |
||||
}); |
||||
|
||||
test('showWarningToast', async () => { |
||||
await showWarningToast('warning 😐', {duration: -1}); |
||||
expect(document.querySelector('.toastify')).toBeTruthy(); |
||||
}); |
||||
|
||||
test('showErrorToast', async () => { |
||||
await showErrorToast('error 🙁', {duration: -1}); |
||||
expect(document.querySelector('.toastify')).toBeTruthy(); |
||||
}); |
@ -0,0 +1,11 @@ |
||||
import {showInfoToast, showWarningToast, showErrorToast} from '../modules/toast.js'; |
||||
|
||||
document.getElementById('info-toast').addEventListener('click', () => { |
||||
showInfoToast('success 😀'); |
||||
}); |
||||
document.getElementById('warning-toast').addEventListener('click', () => { |
||||
showWarningToast('warning 😐'); |
||||
}); |
||||
document.getElementById('error-toast').addEventListener('click', () => { |
||||
showErrorToast('error 🙁'); |
||||
}); |
Loading…
Reference in new issue