Files
AstralRinth/packages/ui/src/components/skin/SkinButton.vue
T
Truman Gao c1780eef7d feat: drag and drop skins to reorder (#6357)
* feat: drag and drop skins to reorder

* feat: implement drag to reorder skins

* fix: ci

* remove: backend implementation

* regenerate sqlx

* fix: remove v-if selectable

* feat: remove drag handle

* refactor: pnpm prepr

* cargo fmt

* fix: dragging disable hover, wrong evt for edit skin + remove back of skin hover

---------

Co-authored-by: Calum H. (IMB11) <contact@cal.engineer>
2026-06-11 12:22:38 +00:00

213 lines
5.0 KiB
Vue

<script setup lang="ts">
import { ref, watch } from 'vue'
const emit = defineEmits<{
(e: 'select'): void
(e: 'edit', event: MouseEvent): void
}>()
const props = withDefaults(
defineProps<{
forwardImageSrc?: string
selected: boolean
active?: boolean
tooltip?: string
disabled?: boolean
isDragging?: boolean
}>(),
{
forwardImageSrc: undefined,
active: false,
tooltip: undefined,
disabled: false,
isDragging: false,
},
)
const imagesLoaded = ref({
forward: false,
})
function onImageLoad() {
imagesLoaded.value.forward = true
}
watch(
() => props.forwardImageSrc,
() => {
imagesLoaded.value.forward = false
},
)
</script>
<template>
<div
v-tooltip="tooltip ?? undefined"
class="skin-button group relative flex items-end justify-center overflow-hidden border border-solid transition-[border-color,box-shadow] duration-200"
:class="[
selected ? 'skin-button--selected' : '',
active ? 'skin-button--active' : '',
{
'skin-button--with-actions': $slots['overlay-buttons'] && !disabled,
'skin-button--disabled': disabled,
'skin-button--dragging': isDragging,
},
]"
>
<span
v-if="$slots['top-buttons']"
class="pointer-events-none absolute right-3 top-3 z-30 flex items-center gap-1"
>
<slot name="top-buttons" />
</span>
<button
class="absolute inset-0 z-10 cursor-pointer border-none bg-transparent p-0 focus-visible:outline-none"
:aria-label="tooltip ? `Select ${tooltip}` : 'Select skin'"
:aria-pressed="selected"
:disabled="disabled"
@click="emit('select')"
></button>
<span
v-if="active && !selected && !$slots['top-buttons']"
class="pointer-events-none absolute right-3 top-3 z-20 size-3 rounded-full border-2 border-solid border-surface-3 bg-green"
></span>
<div v-if="!imagesLoaded.forward" class="skeleton-loader h-full w-full">
<div class="skeleton absolute inset-0 aspect-[5/7]"></div>
</div>
<span
v-show="imagesLoaded.forward"
:key="`${selected}-${active}`"
:class="[
'skin-button__image-parent pointer-events-none relative z-0 mb-[1.5px] grid place-items-stretch with-shadow',
]"
>
<img
alt=""
:src="forwardImageSrc"
class="skin-button__image-facing col-start-1 row-start-1 h-full w-full object-contain"
height="504"
@load="onImageLoad"
/>
</span>
<span
v-if="$slots['overlay-buttons'] && !disabled"
class="pointer-events-none absolute inset-x-0 bottom-3 z-30 flex translate-y-2 items-center justify-start gap-1.5 px-3 opacity-0 transition-all duration-200 group-focus-within:translate-y-0 group-focus-within:opacity-100 group-hover:translate-y-0 group-hover:opacity-100"
>
<slot name="overlay-buttons" />
</span>
</div>
</template>
<style scoped lang="scss">
.skeleton-loader {
aspect-ratio: 31 / 40;
}
.skeleton {
background: linear-gradient(
90deg,
var(--color-bg) 25%,
var(--color-raised-bg) 50%,
var(--color-bg) 75%
);
background-size: 200% 100%;
animation: wave 1500ms infinite linear;
}
@keyframes wave {
0% {
background-position: -200% 0;
}
100% {
background-position: 200% 0;
}
}
.skin-button {
aspect-ratio: 31 / 40;
border-color: var(--surface-4);
border-radius: 20px;
background: var(--surface-3);
isolation: isolate;
box-shadow:
0 1px 1px rgba(0, 0, 0, 0.25),
0 1px 2px rgba(0, 0, 0, 0.15);
}
.skin-button::after {
position: absolute;
inset: 0;
z-index: 5;
pointer-events: none;
content: '';
background: linear-gradient(180deg, rgba(0, 0, 0, 0) 0%, rgba(37, 39, 45, 0.2) 100%);
}
.skin-button:has(:focus-visible) {
outline: 2px solid var(--color-brand);
outline-offset: 2px;
}
.skin-button:not(.skin-button--disabled):hover,
.skin-button:not(.skin-button--disabled):focus-within,
.skin-button--with-actions:hover,
.skin-button--with-actions:focus-within {
border-color: var(--surface-5);
background: var(--surface-4);
box-shadow:
0 1px 2px rgba(0, 0, 0, 0.25),
0 1px 4px rgba(0, 0, 0, 0.15);
}
.skin-button.skin-button--selected,
.skin-button.skin-button--selected:hover,
.skin-button.skin-button--selected:focus-within,
.skin-button.skin-button--selected.skin-button--with-actions:hover,
.skin-button.skin-button--selected.skin-button--with-actions:focus-within,
.skin-button.skin-button--active:hover,
.skin-button.skin-button--active:focus-within,
.skin-button.skin-button--active.skin-button--with-actions:hover,
.skin-button.skin-button--active.skin-button--with-actions:focus-within {
border-color: var(--color-brand);
background: var(--color-brand-highlight);
}
.skin-button--disabled {
opacity: 0.65;
}
.skin-button--disabled button {
cursor: not-allowed;
}
.skin-button--dragging {
pointer-events: none;
}
.skin-button__image-parent {
width: 100%;
height: 95%;
}
.skin-button__image-facing {
transform: translateZ(0);
}
.with-shadow img {
filter: drop-shadow(0 4px 8px rgba(0, 0, 0, 0.4));
}
.skin-button__image-parent img {
transition: filter 200ms ease-in-out;
}
.group:not(.skin-button--disabled):hover .skin-button__image-parent img {
filter: drop-shadow(0 2px 4px rgba(0, 0, 0, 0.2));
}
</style>