You've already forked AstralRinth
forked from didirus/AstralRinth
* Start of app redesign * format * continue progress * Content page nearly done * Fix recursion issues with content page * Fix update all alignment * Discover page progress * Settings progress * Removed unlocked-size hack that breaks web * Revamp project page, refactor web project page to share code with app, fixed loading bar, misc UI/UX enhancements, update ko-fi logo, update arrow icons, fix web issues caused by floating-vue migration, fix tooltip issues, update web tooltips, clean up web hydration issues * Ads + run prettier * Begin auth refactor, move common messages to ui lib, add i18n extraction to all apps, begin Library refactor * fix ads not hiding when plus log in * rev lockfile changes/conflicts * Fix sign in page * Add generated * (mostly) Data driven search * Fix search mobile issue * profile fixes * Project versions page, fix typescript on UI lib and misc fixes * Remove unused gallery component * Fix linkfunction err * Search filter controls at top, localization for locked filters * Fix provided filter names * Fix navigating from instance browse to main browse * Friends frontend (#2995) * Friends system frontend * (almost) finish frontend * finish friends, fix lint * Fix lint --------- Signed-off-by: Geometrically <18202329+Geometrically@users.noreply.github.com> * Refresh macOS app icon * Update web search UI more * Fix link opens * Fix frontend build --------- Signed-off-by: Geometrically <18202329+Geometrically@users.noreply.github.com> Co-authored-by: Jai A <jaiagr+gpg@pm.me> Co-authored-by: Geometrically <18202329+Geometrically@users.noreply.github.com>
132 lines
2.8 KiB
Vue
132 lines
2.8 KiB
Vue
<template>
|
|
<PopoutMenu
|
|
ref="dropdown"
|
|
v-bind="$attrs"
|
|
:disabled="disabled"
|
|
:dropdown-id="dropdownId"
|
|
:tooltip="tooltip"
|
|
>
|
|
<slot></slot>
|
|
<template #menu>
|
|
<template v-for="(option, index) in options.filter((x) => x.shown === undefined || x.shown)">
|
|
<div
|
|
v-if="option.divider"
|
|
:key="`divider-${index}`"
|
|
class="h-px mx-3 my-2 bg-button-bg"
|
|
></div>
|
|
<Button
|
|
v-else
|
|
:key="`option-${option.id}`"
|
|
v-tooltip="option.tooltip"
|
|
:color="option.color ? option.color : 'default'"
|
|
:hover-filled="option.hoverFilled"
|
|
:hover-filled-only="option.hoverFilledOnly"
|
|
transparent
|
|
:v-close-popper="!option.remainOnClick"
|
|
:action="
|
|
option.action
|
|
? (event) => {
|
|
option.action(event)
|
|
if (!option.remainOnClick) {
|
|
close()
|
|
}
|
|
}
|
|
: null
|
|
"
|
|
:link="option.link ? option.link : null"
|
|
:external="option.external ? option.external : false"
|
|
:disabled="option.disabled"
|
|
@click="
|
|
() => {
|
|
if (option.link && !option.remainOnClick) {
|
|
close()
|
|
}
|
|
}
|
|
"
|
|
>
|
|
<template v-if="!$slots[option.id]">{{ option.id }}</template>
|
|
<slot :name="option.id"></slot>
|
|
</Button>
|
|
</template>
|
|
</template>
|
|
</PopoutMenu>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref } from 'vue'
|
|
import Button from './Button.vue'
|
|
import PopoutMenu from './PopoutMenu.vue'
|
|
|
|
interface BaseOption {
|
|
shown?: boolean
|
|
}
|
|
|
|
interface Divider extends BaseOption {
|
|
divider?: boolean
|
|
}
|
|
|
|
interface Item extends BaseOption {
|
|
id: string
|
|
action?: () => void
|
|
link?: string
|
|
external?: boolean
|
|
color?:
|
|
| 'primary'
|
|
| 'danger'
|
|
| 'secondary'
|
|
| 'highlight'
|
|
| 'red'
|
|
| 'orange'
|
|
| 'green'
|
|
| 'blue'
|
|
| 'purple'
|
|
hoverFilled?: boolean
|
|
hoverFilledOnly?: boolean
|
|
remainOnClick?: boolean
|
|
disabled?: boolean
|
|
tooltip?: string
|
|
}
|
|
|
|
type Option = Divider | Item
|
|
|
|
const props = withDefaults(
|
|
defineProps<{
|
|
options: Option[]
|
|
disabled?: boolean
|
|
dropdownId?: string
|
|
tooltip?: string
|
|
}>(),
|
|
{
|
|
options: () => [],
|
|
disabled: false,
|
|
dropdownId: null,
|
|
tooltip: null,
|
|
},
|
|
)
|
|
|
|
defineOptions({
|
|
inheritAttrs: false,
|
|
})
|
|
|
|
const dropdown = ref(null)
|
|
|
|
const close = () => {
|
|
dropdown.value.hide()
|
|
}
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.btn {
|
|
white-space: nowrap;
|
|
width: 100%;
|
|
box-shadow: none;
|
|
--text-color: var(--color-base);
|
|
--background-color: transparent;
|
|
justify-content: flex-start;
|
|
|
|
&:not(:last-child) {
|
|
margin-bottom: var(--gap-xs);
|
|
}
|
|
}
|
|
</style>
|