* feat: rough draft of tool * fix: example doc * feat: multiselect chips * feat: conditional actions+messaages + utils for handling conditions * feat: migrate checklist v1 to new format. * fix: lint issues * fix: severity util * feat: README.md * feat: start implementing new moderation checklist * feat: message assembly + fix imports * fix: lint issues * feat: add input suggestions * feat: utility cleanup * fix: icon * chore: remove debug logging * chore: remove debug button * feat: modpack permissions flow into it's own component * feat: icons + use id in stage selection button * Support md/plain text in stages. * fix: checklist not persisting/showing on subpages * feat: message gen + appr/with/deny buttons * feat: better notification placement + queue navigation * fix: default props for futureProjects * fix: modpack perms message * fix: issue with future projects props * fix: tab index + z index fixes * feat: keybinds * fix: file approval types * fix: generate message for non-modpack projects * feat: add generate message to stages dropdown * fix: variables not expanding * feat: requests * fix: empty message approval * fix: issues from sync * chore: add comment for old moderation checklist impl * fix: git artifacts * fix: update visibility logic for stages and actions * fix: cleanup logic for should show * fix: markdown editor accidental edit
214 lines
5.4 KiB
Vue
214 lines
5.4 KiB
Vue
<template>
|
|
<div
|
|
class="vue-notification-group experimental-styles-within"
|
|
:class="{
|
|
'intercom-present': isIntercomPresent,
|
|
rightwards: moveNotificationsRight,
|
|
}"
|
|
>
|
|
<transition-group name="notifs">
|
|
<div
|
|
v-for="(item, index) in notifications"
|
|
:key="item.id"
|
|
class="vue-notification-wrapper"
|
|
@mouseenter="stopTimer(item)"
|
|
@mouseleave="setNotificationTimer(item)"
|
|
>
|
|
<div class="flex w-full gap-2 overflow-hidden rounded-lg bg-bg-raised shadow-xl">
|
|
<div
|
|
class="w-2"
|
|
:class="{
|
|
'bg-red': item.type === 'error',
|
|
'bg-orange': item.type === 'warning',
|
|
'bg-green': item.type === 'success',
|
|
'bg-blue': !item.type || !['error', 'warning', 'success'].includes(item.type),
|
|
}"
|
|
></div>
|
|
<div
|
|
class="grid w-full grid-cols-[auto_1fr_auto] items-center gap-x-2 gap-y-1 py-2 pl-1 pr-3"
|
|
>
|
|
<div
|
|
class="flex items-center"
|
|
:class="{
|
|
'text-red': item.type === 'error',
|
|
'text-orange': item.type === 'warning',
|
|
'text-green': item.type === 'success',
|
|
'text-blue': !item.type || !['error', 'warning', 'success'].includes(item.type),
|
|
}"
|
|
>
|
|
<IssuesIcon v-if="item.type === 'warning'" class="h-6 w-6" />
|
|
<CheckCircleIcon v-else-if="item.type === 'success'" class="h-6 w-6" />
|
|
<XCircleIcon v-else-if="item.type === 'error'" class="h-6 w-6" />
|
|
<InfoIcon v-else class="h-6 w-6" />
|
|
</div>
|
|
<div class="m-0 text-wrap font-bold text-contrast" v-html="item.title"></div>
|
|
<div class="flex items-center gap-1">
|
|
<div v-if="item.count && item.count > 1" class="text-xs font-bold text-contrast">
|
|
x{{ item.count }}
|
|
</div>
|
|
<ButtonStyled circular size="small">
|
|
<button v-tooltip="'Copy to clipboard'" @click="copyToClipboard(item)">
|
|
<CheckIcon v-if="copied[createNotifText(item)]" />
|
|
<CopyIcon v-else />
|
|
</button>
|
|
</ButtonStyled>
|
|
<ButtonStyled circular size="small">
|
|
<button v-tooltip="`Dismiss`" @click="notifications.splice(index, 1)">
|
|
<XIcon />
|
|
</button>
|
|
</ButtonStyled>
|
|
</div>
|
|
<div></div>
|
|
<div class="col-span-2 text-sm text-primary" v-html="item.text"></div>
|
|
<template v-if="item.errorCode">
|
|
<div></div>
|
|
<div
|
|
class="m-0 text-wrap text-xs font-medium text-secondary"
|
|
v-html="item.errorCode"
|
|
></div>
|
|
</template>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</transition-group>
|
|
</div>
|
|
</template>
|
|
<script setup>
|
|
import { ButtonStyled } from "@modrinth/ui";
|
|
import {
|
|
XCircleIcon,
|
|
CheckCircleIcon,
|
|
CheckIcon,
|
|
InfoIcon,
|
|
IssuesIcon,
|
|
XIcon,
|
|
CopyIcon,
|
|
} from "@modrinth/assets";
|
|
const notifications = useNotifications();
|
|
const { isVisible: moveNotificationsRight } = useNotificationRightwards();
|
|
|
|
const isIntercomPresent = ref(false);
|
|
|
|
function stopTimer(notif) {
|
|
clearTimeout(notif.timer);
|
|
}
|
|
|
|
const copied = ref({});
|
|
|
|
const createNotifText = (notif) => {
|
|
let text = "";
|
|
if (notif.title) {
|
|
text += notif.title;
|
|
}
|
|
if (notif.text) {
|
|
if (text.length > 0) {
|
|
text += "\n";
|
|
}
|
|
text += notif.text;
|
|
}
|
|
if (notif.errorCode) {
|
|
if (text.length > 0) {
|
|
text += "\n";
|
|
}
|
|
text += notif.errorCode;
|
|
}
|
|
return text;
|
|
};
|
|
|
|
function checkIntercomPresence() {
|
|
isIntercomPresent.value = !!document.querySelector(".intercom-lightweight-app");
|
|
}
|
|
|
|
onMounted(() => {
|
|
checkIntercomPresence();
|
|
|
|
const observer = new MutationObserver(() => {
|
|
checkIntercomPresence();
|
|
});
|
|
|
|
observer.observe(document.body, {
|
|
childList: true,
|
|
subtree: true,
|
|
});
|
|
|
|
onBeforeUnmount(() => {
|
|
observer.disconnect();
|
|
});
|
|
});
|
|
|
|
function copyToClipboard(notif) {
|
|
const text = createNotifText(notif);
|
|
|
|
copied.value[text] = true;
|
|
navigator.clipboard.writeText(text);
|
|
setTimeout(() => {
|
|
delete copied.value[text];
|
|
}, 2000);
|
|
}
|
|
</script>
|
|
<style lang="scss" scoped>
|
|
.vue-notification-group {
|
|
position: fixed;
|
|
right: 1.5rem;
|
|
bottom: 1.5rem;
|
|
z-index: 200;
|
|
width: 450px;
|
|
|
|
@media screen and (max-width: 500px) {
|
|
width: calc(100% - 0.75rem * 2);
|
|
right: 0.75rem;
|
|
bottom: 0.75rem;
|
|
}
|
|
|
|
&.intercom-present {
|
|
bottom: 5rem;
|
|
}
|
|
|
|
&.rightwards {
|
|
right: unset !important;
|
|
left: 1.5rem;
|
|
|
|
@media screen and (max-width: 500px) {
|
|
left: 0.75rem;
|
|
}
|
|
}
|
|
|
|
.vue-notification-wrapper {
|
|
width: 100%;
|
|
overflow: hidden;
|
|
margin-bottom: 10px;
|
|
|
|
&:last-child {
|
|
margin: 0;
|
|
}
|
|
}
|
|
|
|
@media screen and (max-width: 750px) {
|
|
transition: bottom 0.25s ease-in-out;
|
|
bottom: calc(var(--size-mobile-navbar-height) + 10px) !important;
|
|
|
|
&.browse-menu-open {
|
|
bottom: calc(var(--size-mobile-navbar-height-expanded) + 10px) !important;
|
|
}
|
|
}
|
|
}
|
|
|
|
.notifs-enter-active,
|
|
.notifs-leave-active,
|
|
.notifs-move {
|
|
transition: all 0.25s ease-in-out;
|
|
}
|
|
.notifs-enter-from,
|
|
.notifs-leave-to {
|
|
opacity: 0;
|
|
}
|
|
|
|
.notifs-enter-from {
|
|
transform: translateY(100%) scale(0.8);
|
|
}
|
|
|
|
.notifs-leave-to {
|
|
transform: translateX(100%) scale(0.8);
|
|
}
|
|
</style>
|