-
Notifications
You must be signed in to change notification settings - Fork 81
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(ui): implement new Slider. WF-37
- Loading branch information
Showing
5 changed files
with
285 additions
and
49 deletions.
There are no files selected for viewing
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,253 @@ | ||
<template> | ||
<div | ||
role="slider" | ||
:aria-valuemin="min" | ||
:aria-valuemax="max" | ||
:aria-valuenow="value" | ||
:aria-valuetext="`${value} (${Math.round(progress)}%)`" | ||
class="BaseInputRange" | ||
:class="{ | ||
'BaseInputRange--popover-always-visible': | ||
popoverDisplayMode === 'always', | ||
}" | ||
> | ||
<transition> | ||
<div | ||
v-show="isPopoverDisplayed" | ||
class="BaseInputRange__popover" | ||
:style="{ left: popoverLeft }" | ||
> | ||
<span class="BaseInputRange__popover__content"> | ||
{{ value }} | ||
</span> | ||
</div> | ||
</transition> | ||
<div | ||
ref="slider" | ||
class="BaseInputRange__slider" | ||
@mousedown="handleMouseDown" | ||
> | ||
<div | ||
class="BaseInputRange__slider__progress" | ||
:style="{ width: `calc(${progress}% + ${thumbRadius}px)` }" | ||
></div> | ||
</div> | ||
<button | ||
ref="thumb" | ||
type="button" | ||
class="BaseInputRange__thumb" | ||
:style="{ left: thumbLeft }" | ||
aria-label="Use the arrow keys to increase or decrease the value." | ||
@keydown.left="updateValue(+model - step)" | ||
@keydown.right="updateValue(+model + step)" | ||
@mousedown="handleMouseDown" | ||
></button> | ||
</div> | ||
</template> | ||
|
||
<script setup lang="ts"> | ||
import { computed, PropType, ref, toRef, watch } from "vue"; | ||
const props = defineProps({ | ||
min: { type: Number, default: 0 }, | ||
max: { type: Number, default: 100 }, | ||
step: { type: Number, default: 1 }, | ||
popoverDisplayMode: { | ||
type: String as PropType<"always" | "onChange">, | ||
default: "onChange", | ||
}, | ||
}); | ||
const model = defineModel("value", { type: Number, default: 50 }); | ||
const thumbRadius = 9; | ||
const thumb = ref<HTMLElement>(); | ||
const slider = ref<HTMLElement>(); | ||
const progress = computed(() => { | ||
if (typeof model.value !== "number") return 50; | ||
return ((model.value - props.min) / (props.max - props.min)) * 100; | ||
}); | ||
const isPopoverDisplayed = ref(props.popoverDisplayMode === "always"); | ||
let popoverTimeout = null; | ||
watch(toRef(props.popoverDisplayMode), () => { | ||
if (props.popoverDisplayMode === "always") isPopoverDisplayed.value = true; | ||
}); | ||
function displayPopover() { | ||
if (props.popoverDisplayMode === "always") return; | ||
isPopoverDisplayed.value = true; | ||
if (popoverTimeout) { | ||
clearTimeout(popoverTimeout); | ||
popoverTimeout = null; | ||
} | ||
popoverTimeout = setTimeout(() => { | ||
isPopoverDisplayed.value = false; | ||
popoverTimeout = null; | ||
}, 1_000); | ||
} | ||
// clamp(0px, calc(62% - 9px), calc(100% - 18px)) | ||
const thumbLeft = computed( | ||
() => `clamp( | ||
0px, | ||
calc(${progress.value}% - ${thumbRadius}px), | ||
calc(100% - ${thumbRadius * 2}px) | ||
)`, | ||
); | ||
const popoverLeft = computed(() => `calc(${thumbLeft.value} - 3px)`); | ||
function updateValue(value: number) { | ||
displayPopover(); | ||
if (props.min !== undefined && value < props.min) return; | ||
if (props.max !== undefined && value > props.max) return; | ||
// round the value to the closest step | ||
const relativeValue = value - props.min; | ||
const stepIndex = Math.round(relativeValue / props.step); | ||
const roundedValue = props.min + stepIndex * props.step; | ||
if (model.value !== roundedValue) model.value = roundedValue; | ||
} | ||
function handleMouseDown(initialEvent: MouseEvent) { | ||
document.addEventListener("mousemove", onMouseMove); | ||
document.addEventListener("mouseup", onMouseUp); | ||
const sliderBoundingRect = slider.value.getBoundingClientRect(); | ||
// trigger immediate value update to handle user click | ||
onMouseMove(initialEvent); | ||
function onMouseMove(event: MouseEvent) { | ||
const progress = | ||
(event.x - sliderBoundingRect.left) / | ||
(sliderBoundingRect.right - sliderBoundingRect.left); | ||
if (progress > 1 || progress < 0) return; | ||
const value = Math.round( | ||
(props.max - props.min) * progress + props.min, | ||
); | ||
updateValue(value); | ||
} | ||
function onMouseUp() { | ||
document.removeEventListener("mouseup", onMouseUp); | ||
document.removeEventListener("mousemove", onMouseMove); | ||
} | ||
} | ||
</script> | ||
|
||
<style scoped> | ||
.BaseInputRange { | ||
--thumb-color: var(--accentColor); | ||
--thumb-shadow-color: rgba(228, 231, 237, 0.4); | ||
--slider-color: var(--softenedAccentColor); | ||
--slider-bg-color: var(--separatorColor); | ||
--popover-bg-color: var(--popoverBackgroundColor, rgba(0, 0, 0, 1)); | ||
width: 100%; | ||
position: relative; | ||
padding-top: 5px; | ||
padding-bottom: 5px; | ||
} | ||
.BaseInputRange--popover-always-visible { | ||
/* add extra margin to make sure the popover does not overflow on something */ | ||
margin-top: 24px; | ||
} | ||
.BaseInputRange__slider { | ||
height: 8px; | ||
width: 100%; | ||
border-radius: 4px; | ||
background-color: var(--slider-bg-color); | ||
} | ||
.BaseInputRange__slider__progress { | ||
height: 100%; | ||
max-width: 100%; | ||
border-radius: 4px; | ||
background-color: var(--slider-color); | ||
} | ||
.BaseInputRange__thumb { | ||
border: none; | ||
height: 18px; | ||
width: 18px; | ||
border-radius: 50%; | ||
background-color: var(--thumb-color); | ||
position: absolute; | ||
top: 0; | ||
transition: box-shadow ease-in-out 0.5s; | ||
} | ||
.BaseInputRange__thumb:active { | ||
box-shadow: 0 0 0 6px var(--thumb-shadow-color); | ||
} | ||
.BaseInputRange__thumb:focus-visible { | ||
box-shadow: 0 0 0 6px var(--thumb-shadow-color); | ||
outline: 1px solid blue; | ||
outline-offset: 2px; | ||
} | ||
.BaseInputRange__popover { | ||
position: absolute; | ||
top: -24px; | ||
font-size: 10px; | ||
z-index: 2; | ||
color: var(--popoverColor, white); | ||
background: var(--popover-bg-color); | ||
height: 18px; | ||
width: 24px; | ||
max-width: 24px; | ||
border-radius: 4px; | ||
display: flex; | ||
align-items: center; | ||
justify-content: center; | ||
} | ||
.BaseInputRange__popover::after { | ||
content: ""; | ||
width: 0; | ||
height: 0; | ||
border-left: 4px solid transparent; | ||
border-right: 4px solid transparent; | ||
border-top: 5px solid var(--popover-bg-color); | ||
position: absolute; | ||
bottom: -5px; | ||
left: calc(50% - 4px); | ||
} | ||
.BaseInputRange__popover__content { | ||
overflow: hidden; | ||
text-overflow: ellipsis; | ||
text-align: center; | ||
} | ||
/* start invisible */ | ||
.v-enter-active, | ||
.v-leave-active { | ||
transition: opacity 0.2s ease-in-out; | ||
} | ||
.v-enter-from, | ||
.v-leave-to { | ||
opacity: 0; | ||
} | ||
</style> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters