feat(Range): new component (#290)

Co-authored-by: Tom Smith <tom.smith2711@gmail.com>
Co-authored-by: Benjamin Canac <canacb1@gmail.com>
Co-authored-by: Tom Smith <tom.smith@qunifi.com>
This commit is contained in:
TomSmith27
2023-06-20 15:20:17 +01:00
committed by Benjamin Canac
parent c2ebb0416e
commit 97a1c86433
11 changed files with 329 additions and 13 deletions

View File

@@ -2,6 +2,8 @@ module.exports = {
root: true, root: true,
extends: ['@nuxt/eslint-config'], extends: ['@nuxt/eslint-config'],
rules: { rules: {
'comma-dangle': ['error', 'never'],
'space-before-function-paren': ['error', 'always'],
'vue/multi-word-component-names': 0, 'vue/multi-word-component-names': 0,
'vue/max-attributes-per-line': ['error', { 'vue/max-attributes-per-line': ['error', {
singleline: { singleline: {

1
.gitignore vendored
View File

@@ -7,3 +7,4 @@ dist
.DS_Store .DS_Store
.history .history
.vercel .vercel
.idea

6
.prettierrc.json Normal file
View File

@@ -0,0 +1,6 @@
{
"trailingComma": "none",
"tabWidth": 2,
"semi": false,
"singleQuote": true
}

View File

@@ -0,0 +1,7 @@
<script setup>
const value = ref(50)
</script>
<template>
<URange v-model="value" />
</template>

View File

@@ -33,7 +33,7 @@ Likewise, you can't define a `primary` color in your `tailwind.config.ts` as it
We'd advise you to use those colors in your components and pages, e.g. `text-primary-500 dark:text-primary-400`, `bg-gray-100 dark:bg-gray-900`, etc. so your app automatically adapts when changing your `app.config.ts`. We'd advise you to use those colors in your components and pages, e.g. `text-primary-500 dark:text-primary-400`, `bg-gray-100 dark:bg-gray-900`, etc. so your app automatically adapts when changing your `app.config.ts`.
:: ::
Components having a `color` prop like [Avatar](/elements/avatar#chip), [Badge](/elements/badge#style), [Button](/elements/button#style), [Input](/elements/input#style) (inherited in [Select](/forms/select) and [SelectMenu](/forms/select-menu)) and [Notification](/overlays/notification#timeout) will use the `primary` color by default but will handle all the colors defined in your `tailwind.config.ts` or the default Tailwind CSS colors. Components having a `color` prop like [Avatar](/elements/avatar#chip), [Badge](/elements/badge#style), [Button](/elements/button#style), [Input](/elements/input#style) (inherited in [Select](/forms/select) and [SelectMenu](/forms/select-menu)), [Range](/forms/range) and [Notification](/overlays/notification#timeout) will use the `primary` color by default but will handle all the colors defined in your `tailwind.config.ts` or the default Tailwind CSS colors.
Variant classes of those components are defined with a syntax like `bg-{color}-500 dark:bg-{color}-400` so they can be used with any color. However, this means that Tailwind will not find those classes and therefore will not generate the corresponding CSS. Variant classes of those components are defined with a syntax like `bg-{color}-500 dark:bg-{color}-400` so they can be used with any color. However, this means that Tailwind will not find those classes and therefore will not generate the corresponding CSS.

View File

@@ -0,0 +1,101 @@
---
github: true
description: Display a range field
navigation:
badge: "Edge"
---
## Usage
Use a `v-model` to make the Range reactive.
::component-example
#default
:range-example
#code
```vue
<script setup>
const value = ref(50)
</script>
<template>
<URange v-model="value" />
</template>
```
::
### Style
Use the `color` prop to change the visual style of the Range.
::component-card
---
baseProps:
name: range'
placeholder: 'Search...'
props:
color: 'primary'
---
::
### Size
Use the `size` prop to change the size of the Range.
::component-card
---
baseProps:
name: 'range'
props:
size: 'md'
---
::
### Disabled
Use the `disabled` prop to disable the Range.
::component-card
---
baseProps:
name: 'range'
props:
disabled: true
---
::
### Min and Max
Use the `min` and `max` prop to configure the Range.
::component-card
---
baseProps:
name: 'range'
props:
min: 0
max: 100
---
::
### Step
Use the `step` prop to change the step increment.
::component-card
---
baseProps:
name: 'range'
props:
step: 20
---
::
## Props
:component-props
## Preset
:component-preset

View File

@@ -27,26 +27,26 @@ const kebabCase = (str: string) => {
const safelistByComponent = { const safelistByComponent = {
avatar: (colorsAsRegex) => [{ avatar: (colorsAsRegex) => [{
pattern: new RegExp(`bg-(${colorsAsRegex})-500`)
}, {
pattern: new RegExp(`bg-(${colorsAsRegex})-400`), pattern: new RegExp(`bg-(${colorsAsRegex})-400`),
variants: ['dark'] variants: ['dark']
}, {
pattern: new RegExp(`bg-(${colorsAsRegex})-500`)
}], }],
badge: (colorsAsRegex) => [{ badge: (colorsAsRegex) => [{
pattern: new RegExp(`bg-(${colorsAsRegex})-50`) pattern: new RegExp(`bg-(${colorsAsRegex})-50`)
}, { }, {
pattern: new RegExp(`bg-(${colorsAsRegex})-400`), pattern: new RegExp(`bg-(${colorsAsRegex})-400`),
variants: ['dark'] variants: ['dark']
}, {
pattern: new RegExp(`text-(${colorsAsRegex})-500`)
}, { }, {
pattern: new RegExp(`text-(${colorsAsRegex})-400`), pattern: new RegExp(`text-(${colorsAsRegex})-400`),
variants: ['dark'] variants: ['dark']
}, { }, {
pattern: new RegExp(`ring-(${colorsAsRegex})-500`) pattern: new RegExp(`text-(${colorsAsRegex})-500`)
}, { }, {
pattern: new RegExp(`ring-(${colorsAsRegex})-400`), pattern: new RegExp(`ring-(${colorsAsRegex})-400`),
variants: ['dark'] variants: ['dark']
}, {
pattern: new RegExp(`ring-(${colorsAsRegex})-500`)
}], }],
button: (colorsAsRegex) => [{ button: (colorsAsRegex) => [{
pattern: new RegExp(`bg-(${colorsAsRegex})-50`), pattern: new RegExp(`bg-(${colorsAsRegex})-50`),
@@ -103,16 +103,33 @@ const safelistByComponent = {
pattern: new RegExp(`ring-(${colorsAsRegex})-500`), pattern: new RegExp(`ring-(${colorsAsRegex})-500`),
variants: ['focus'] variants: ['focus']
}], }],
notification: (colorsAsRegex) => [{ range: (colorsAsRegex) => [{
pattern: new RegExp(`bg-(${colorsAsRegex})-400`),
variants: ['dark']
}, {
pattern: new RegExp(`bg-(${colorsAsRegex})-500`) pattern: new RegExp(`bg-(${colorsAsRegex})-500`)
}, { }, {
pattern: new RegExp(`bg-(${colorsAsRegex})-400`), pattern: new RegExp(`text-(${colorsAsRegex})-400`),
variants: ['dark'] variants: ['dark']
}, { }, {
pattern: new RegExp(`text-(${colorsAsRegex})-500`) pattern: new RegExp(`text-(${colorsAsRegex})-500`)
}, {
pattern: new RegExp(`ring-(${colorsAsRegex})-400`),
variants: ['dark:focus-visible']
}, {
pattern: new RegExp(`ring-(${colorsAsRegex})-500`),
variants: ['focus-visible']
}],
notification: (colorsAsRegex) => [{
pattern: new RegExp(`bg-(${colorsAsRegex})-400`),
variants: ['dark']
}, {
pattern: new RegExp(`bg-(${colorsAsRegex})-500`)
}, { }, {
pattern: new RegExp(`text-(${colorsAsRegex})-400`), pattern: new RegExp(`text-(${colorsAsRegex})-400`),
variants: ['dark'] variants: ['dark']
}, {
pattern: new RegExp(`text-(${colorsAsRegex})-500`)
}] }]
} }
@@ -127,7 +144,7 @@ const colorsAsRegex = (colors: string[]): string => colors.join('|')
export const excludeColors = (colors: object) => Object.keys(omit(colors, colorsToExclude)).map(color => kebabCase(color)) as string[] export const excludeColors = (colors: object) => Object.keys(omit(colors, colorsToExclude)).map(color => kebabCase(color)) as string[]
export const generateSafelist = (colors: string[]) => { export const generateSafelist = (colors: string[]) => {
const safelist = ['avatar', 'badge', 'button', 'input', 'notification'].flatMap(component => safelistByComponent[component](colorsAsRegex(colors))) const safelist = ['avatar', 'badge', 'button', 'input', 'range', 'notification'].flatMap(component => safelistByComponent[component](colorsAsRegex(colors)))
return [ return [
...safelist, ...safelist,

View File

@@ -327,10 +327,10 @@ const input = {
}, },
color: { color: {
white: { white: {
outline: 'shadow-sm bg-white dark:bg-gray-900 text-gray-900 dark:text-white ring-1 ring-inset ring-gray-300 dark:ring-gray-700 focus:ring-2 focus:ring-primary-500 dark:focus:ring-primary-400', outline: 'shadow-sm bg-white dark:bg-gray-900 text-gray-900 dark:text-white ring-1 ring-inset ring-gray-300 dark:ring-gray-700 focus:ring-2 focus:ring-primary-500 dark:focus:ring-primary-400'
}, },
gray: { gray: {
outline: 'shadow-sm bg-gray-50 dark:bg-gray-800 text-gray-900 dark:text-white ring-1 ring-inset ring-gray-300 dark:ring-gray-700 focus:ring-2 focus:ring-primary-500 dark:focus:ring-primary-400', outline: 'shadow-sm bg-gray-50 dark:bg-gray-800 text-gray-900 dark:text-white ring-1 ring-inset ring-gray-300 dark:ring-gray-700 focus:ring-2 focus:ring-primary-500 dark:focus:ring-primary-400'
} }
}, },
variant: { variant: {
@@ -400,7 +400,7 @@ const textarea = {
default: { default: {
size: 'sm', size: 'sm',
color: 'white', color: 'white',
variant: 'outline', variant: 'outline'
} }
} }
@@ -510,6 +510,39 @@ const toggle = {
} }
} }
const range = {
wrapper: 'relative w-full',
base: 'w-full absolute appearance-none cursor-pointer disabled:cursor-not-allowed disabled:opacity-50 focus:outline-none [&::-webkit-slider-runnable-track]:h-full [&::-moz-slider-runnable-track]:h-full',
background: 'bg-gray-200 dark:bg-gray-700',
rounded: 'rounded-lg',
ring: 'focus-visible:ring-2 focus-visible:ring-{color}-500 dark:focus-visible:ring-{color}-400 focus-visible:ring-offset-2 focus-visible:ring-offset-white dark:focus-visible:ring-offset-gray-900',
progress: {
base: 'absolute inset-0 h-full pointer-events-none',
rounded: 'rounded-l-lg',
background: 'bg-{color}-500 dark:bg-{color}-400'
},
thumb: {
base: `[&::-webkit-slider-thumb]:relative [&::-moz-range-thumb]:relative [&::-webkit-slider-thumb]:z-[1] [&::-moz-range-thumb]:z-[1] [&::-webkit-slider-thumb]:appearance-none [&::-moz-range-thumb]:appearance-none [&::-webkit-slider-thumb]:rounded-full [&::-moz-range-thumb]:rounded-full [&::-moz-range-thumb]:border-0`,
color: 'text-{color}-500 dark:text-{color}-400',
background: '[&::-webkit-slider-thumb]:bg-white [&::-webkit-slider-thumb]:dark:bg-gray-900 [&::-moz-range-thumb]:bg-current',
ring: '[&::-webkit-slider-thumb]:ring-2 [&::-webkit-slider-thumb]:ring-current',
size: {
sm: '[&::-webkit-slider-thumb]:h-3 [&::-moz-range-thumb]:h-3 [&::-webkit-slider-thumb]:w-3 [&::-moz-range-thumb]:w-3 [&::-webkit-slider-thumb]:-mt-1 [&::-moz-range-thumb]:-mt-1',
md: '[&::-webkit-slider-thumb]:h-4 [&::-moz-range-thumb]:h-4 [&::-webkit-slider-thumb]:w-4 [&::-moz-range-thumb]:w-4 [&::-webkit-slider-thumb]:-mt-1 [&::-moz-range-thumb]:-mt-1',
lg: '[&::-webkit-slider-thumb]:h-5 [&::-moz-range-thumb]:h-5 [&::-webkit-slider-thumb]:w-5 [&::-moz-range-thumb]:w-5 [&::-webkit-slider-thumb]:-mt-1 [&::-moz-range-thumb]:-mt-1'
}
},
size: {
sm: 'h-1',
md: 'h-2',
lg: 'h-3'
},
default: {
size: 'md',
color: 'primary'
}
}
// Layout // Layout
const card = { const card = {
@@ -872,6 +905,7 @@ export default {
checkbox, checkbox,
radio, radio,
toggle, toggle,
range,
card, card,
container, container,
skeleton, skeleton,

View File

@@ -77,7 +77,7 @@ import appConfig from '#build/app.config'
// const appConfig = useAppConfig() // const appConfig = useAppConfig()
function defaultComparator<T>(a: T, z: T): boolean { function defaultComparator<T> (a: T, z: T): boolean {
return a === z return a === z
} }

View File

@@ -0,0 +1,148 @@
<template>
<div :class="wrapperClass">
<input
:id="name"
ref="input"
v-model.number="value"
:name="name"
:min="min"
:max="max"
:disabled="disabled"
:step="step"
type="range"
:class="[inputClass, thumbClass]"
v-bind="$attrs"
>
<span :class="progressClass" :style="progressStyle" />
</div>
</template>
<script lang="ts">
import { computed, defineComponent } from 'vue'
import type { PropType } from 'vue'
import { defu } from 'defu'
import { classNames } from '../../utils'
import { useAppConfig } from '#imports'
// TODO: Remove
// @ts-expect-error
import appConfig from '#build/app.config'
export default defineComponent({
inheritAttrs: false,
props: {
modelValue: {
type: Number,
default: 0
},
name: {
type: String,
default: null
},
disabled: {
type: Boolean,
default: false
},
min: {
type: Number,
default: 0
},
max: {
type: Number,
default: 100
},
step: {
type: Number,
default: 1
},
size: {
type: String,
default: () => appConfig.ui.range.default.size,
validator (value: string) {
return Object.keys(appConfig.ui.range.size).includes(value)
}
},
color: {
type: String,
default: () => appConfig.ui.range.default.color,
validator (value: string) {
return appConfig.ui.colors.includes(value)
}
},
ui: {
type: Object as PropType<Partial<typeof appConfig.ui.range>>,
default: () => appConfig.ui.range
}
},
emits: ['update:modelValue'],
setup (props, { emit }) {
// TODO: Remove
const appConfig = useAppConfig()
const ui = computed<Partial<typeof appConfig.ui.range>>(() => defu({}, props.ui, appConfig.ui.range))
const value = computed({
get () {
return props.modelValue
},
set (value) {
emit('update:modelValue', value)
}
})
const wrapperClass = computed(() => {
return classNames(
ui.value.wrapper,
ui.value.size[props.size]
)
})
const inputClass = computed(() => {
return classNames(
ui.value.base,
ui.value.background,
ui.value.rounded,
ui.value.ring.replaceAll('{color}', props.color),
ui.value.size[props.size]
)
})
const thumbClass = computed(() => {
return classNames(
ui.value.thumb.base,
// Intermediate class to allow thumb ring or background color (set to `current`) as it's impossible to safelist with arbitrary values
ui.value.thumb.color.replaceAll('{color}', props.color),
ui.value.thumb.ring,
ui.value.thumb.background,
ui.value.thumb.size[props.size]
)
})
const progressClass = computed(() => {
return classNames(
ui.value.progress.base,
ui.value.progress.rounded,
ui.value.progress.background.replaceAll('{color}', props.color),
ui.value.size[props.size]
)
})
const progressStyle = computed(() => {
return {
width: `${(props.modelValue / props.max) * 100}%`
}
})
return {
// eslint-disable-next-line vue/no-dupe-keys
ui,
value,
wrapperClass,
inputClass,
thumbClass,
progressClass,
progressStyle
}
}
})
</script>