mirror of
https://github.com/lucide-icons/lucide.git
synced 2025-12-16 06:07:40 +01:00
* Small adjustments * Format code * format code * Remove default value * format code * update yml file * Format code * Update docs/.vitepress/theme/components/base/Input.vue Co-authored-by: Karsa <contact@karsa.org> * Add extra check if null or undefined --------- Co-authored-by: Karsa <contact@karsa.org>
167 lines
3.1 KiB
Vue
167 lines
3.1 KiB
Vue
<script lang="ts">
|
|
export default {
|
|
inheritAttrs: false,
|
|
};
|
|
|
|
export interface InputProps {
|
|
type: string;
|
|
modelValue: string;
|
|
shortcut?: string;
|
|
}
|
|
</script>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, onMounted, nextTick, watch } from 'vue';
|
|
import Icon from 'lucide-vue-next/src/Icon';
|
|
import { x } from '../../../data/iconNodes';
|
|
import IconButton from './IconButton.vue';
|
|
|
|
const props = withDefaults(defineProps<InputProps>(), {
|
|
type: 'text',
|
|
});
|
|
|
|
const input = ref();
|
|
const wrapperEl = ref();
|
|
const shortcutEl = ref();
|
|
|
|
const emit = defineEmits(['change', 'input', 'update:modelValue']);
|
|
|
|
const updateShortcutSpacing = () => {
|
|
nextTick(() => {
|
|
if (shortcutEl.value && wrapperEl.value) {
|
|
const shortcutWidth = shortcutEl.value.offsetWidth;
|
|
wrapperEl.value.style.setProperty('--shortcut-width', `${shortcutWidth}px`);
|
|
}
|
|
});
|
|
};
|
|
|
|
onMounted(updateShortcutSpacing);
|
|
watch(() => props.shortcut, updateShortcutSpacing);
|
|
|
|
function onClear() {
|
|
emit('update:modelValue', '');
|
|
input.value.focus();
|
|
}
|
|
|
|
defineExpose({
|
|
focus: () => {
|
|
input.value.focus();
|
|
},
|
|
});
|
|
</script>
|
|
|
|
<template>
|
|
<div
|
|
class="input-wrapper"
|
|
ref="wrapperEl"
|
|
>
|
|
<slot
|
|
name="icon"
|
|
class="icon"
|
|
/>
|
|
<input
|
|
:type="type"
|
|
class="input"
|
|
:class="{ 'has-icon': $slots.icon, 'has-shortcut': shortcut }"
|
|
ref="input"
|
|
:value="modelValue"
|
|
v-bind="$attrs"
|
|
@input="$emit('update:modelValue', $event.target.value)"
|
|
/>
|
|
<IconButton
|
|
@click="onClear"
|
|
v-if="type === 'search' && modelValue"
|
|
class="clear-button"
|
|
aria-label="Clear input"
|
|
>
|
|
<Icon
|
|
:iconNode="x"
|
|
:size="20"
|
|
/>
|
|
</IconButton>
|
|
<kbd
|
|
v-if="shortcut"
|
|
class="shortcut"
|
|
ref="shortcutEl"
|
|
>{{ shortcut }}</kbd
|
|
>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped>
|
|
.input-wrapper {
|
|
position: relative;
|
|
}
|
|
|
|
.input {
|
|
justify-content: flex-start;
|
|
border: 1px solid transparent;
|
|
border-radius: 8px;
|
|
padding: 0 10px 0 12px;
|
|
width: 100%;
|
|
height: 40px;
|
|
background-color: var(--vp-c-bg-alt);
|
|
font-size: 14px;
|
|
transition:
|
|
color 0.25s,
|
|
border-color 0.25s,
|
|
background-color 0.25s;
|
|
}
|
|
|
|
.input.has-shortcut {
|
|
padding-right: calc(var(--shortcut-width, 40px) + 22px);
|
|
}
|
|
|
|
.input:hover,
|
|
.input:focus {
|
|
border-color: var(--vp-c-brand);
|
|
background: var(--vp-c-bg-alt);
|
|
}
|
|
|
|
.input.has-icon {
|
|
padding-left: 52px;
|
|
}
|
|
|
|
.clear-button {
|
|
position: absolute;
|
|
right: 56px;
|
|
top: 9px;
|
|
padding: 4px;
|
|
transition: background-color .25s;
|
|
}
|
|
|
|
.shortcut {
|
|
position: absolute;
|
|
right: 12px;
|
|
top: 50%;
|
|
transform: translateY(-50%);
|
|
padding: 2px 6px;
|
|
font-size: 12px;
|
|
font-family: inherit;
|
|
font-weight: 500;
|
|
line-height: 1.5;
|
|
color: var(--vp-c-text-3);
|
|
background: var(--vp-c-default-soft);
|
|
border: 1px solid var(--vp-c-divider);
|
|
border-radius: 4px;
|
|
pointer-events: none;
|
|
}
|
|
|
|
@media (hover: none) {
|
|
.shortcut {
|
|
display: none;
|
|
}
|
|
}
|
|
</style>
|
|
|
|
<style>
|
|
.input-wrapper > svg {
|
|
position: absolute;
|
|
left: 16px;
|
|
top: 12px;
|
|
z-index: 1;
|
|
color: var(--vp-c-text-2);
|
|
pointer-events: none;
|
|
}
|
|
</style>
|