You've already forked AstralRinth
forked from didirus/AstralRinth
381ea51cce
* fix: files.vue bugs before styling changes * feat: move files tab to shared layout structure * fix: qa * fix: qa * fix: bugs * fix: lint * fix: admonition cleanup with progress + actions * fix: cleanup * fix: modals * fix: admon title * fix: i18n standard * fix: lint + i18n pass * fix: remove transition * fix: type errors * feat: files tab in app * fix: qa * fix: backup item minmax * fix: use ContentPageHeader for server panel * fix: lint * fix: lint * fix: lint * feat: page leave safety * fix: lint * fix: cargo fmt fix * fix: blank in prod * fix: content card table stuff * Revert "fix: blank in prod" This reverts commit 74758fe185cf85a4a20355857f889cb091b97ace. * fix: import * feat: browse worlds/servers flow * fix: worlds tab parity with content tab * fix: perf bug + shader filter pill copy * feat: singleplayer filter * fix: ordering * fix: breadcrumbs * fix: lint * fix: qa * feat: store server proj id when adding to a non-linked instance * fix: lint * fix: i18n + qa * fix: conflict * qa: already installed modal + placeholders not server-specific * fix: qa * fix: add + edit server modals * fix: qa * fix: security * fix: devin flags * fix: lint * chore: change file to break build cache * fix: admon * fix: import path stuff * feat: qa * fix: fmt fmt idiot --------- Signed-off-by: Calum H. <calum@modrinth.com>
94 lines
1.5 KiB
Vue
94 lines
1.5 KiB
Vue
<template>
|
|
<Dropdown
|
|
ref="dropdown"
|
|
no-auto-focus
|
|
:aria-id="dropdownId || null"
|
|
:placement="placement"
|
|
:class="dropdownClass"
|
|
@apply-hide="focusTrigger"
|
|
>
|
|
<button ref="trigger" v-bind="$attrs" v-tooltip="tooltip">
|
|
<slot></slot>
|
|
</button>
|
|
<template #popper="{ hide: hideFunction }">
|
|
<button class="dummy-button" @focusin="hideAndFocusTrigger(hideFunction)"></button>
|
|
<div ref="menu" class="contents">
|
|
<slot name="menu"> </slot>
|
|
</div>
|
|
<button class="dummy-button" @focusin="hideAndFocusTrigger(hideFunction)"></button>
|
|
</template>
|
|
</Dropdown>
|
|
</template>
|
|
|
|
<script setup>
|
|
import { Dropdown } from 'floating-vue'
|
|
import { ref } from 'vue'
|
|
|
|
const trigger = ref()
|
|
const menu = ref()
|
|
const dropdown = ref()
|
|
|
|
defineProps({
|
|
dropdownId: {
|
|
type: String,
|
|
default: null,
|
|
required: false,
|
|
},
|
|
dropdownClass: {
|
|
type: String,
|
|
default: null,
|
|
required: false,
|
|
},
|
|
tooltip: {
|
|
type: String,
|
|
default: null,
|
|
required: false,
|
|
},
|
|
placement: {
|
|
type: String,
|
|
default: 'bottom-end',
|
|
required: false,
|
|
},
|
|
})
|
|
|
|
function hideAndFocusTrigger(hide) {
|
|
hide()
|
|
focusTrigger()
|
|
}
|
|
|
|
function focusTrigger() {
|
|
trigger.value.focus()
|
|
}
|
|
|
|
defineOptions({
|
|
inheritAttrs: false,
|
|
})
|
|
|
|
function hide() {
|
|
dropdown.value.hide()
|
|
}
|
|
|
|
function show() {
|
|
dropdown.value.show()
|
|
}
|
|
|
|
defineExpose({
|
|
show,
|
|
hide,
|
|
})
|
|
</script>
|
|
<style scoped>
|
|
.dummy-button {
|
|
position: absolute;
|
|
width: 0;
|
|
height: 0;
|
|
margin: 0;
|
|
padding: 0;
|
|
border: none;
|
|
overflow: hidden;
|
|
clip: rect(0 0 0 0);
|
|
white-space: nowrap;
|
|
outline: none;
|
|
}
|
|
</style>
|