You've already forked AstralRinth
c1780eef7d
* 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>
213 lines
5.0 KiB
Vue
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>
|