mirror of
https://github.com/VueFileManager/vuefilemanager.git
synced 2026-04-06 02:33:48 +00:00
291 lines
7.5 KiB
Vue
291 lines
7.5 KiB
Vue
<template>
|
|
<div class="select">
|
|
|
|
<!--Area-->
|
|
<div class="input-area" :class="{'is-active': isOpen, 'is-error': isError}" @click="openMenu">
|
|
|
|
<!--If is selected-->
|
|
<div class="selected" v-if="selected">
|
|
<div class="option-icon" v-if="selected.icon">
|
|
<user-icon v-if="selected.icon === 'user'" size="14" />
|
|
<edit2-icon v-if="selected.icon === 'user-edit'" size="14" />
|
|
</div>
|
|
<span class="option-value">{{ selected.label }}</span>
|
|
</div>
|
|
|
|
<!--If is empty-->
|
|
<div class="not-selected" v-if="! selected">
|
|
<span class="option-value placehoder">{{ placeholder }}</span>
|
|
</div>
|
|
|
|
<chevron-down-icon size="19" class="chevron" />
|
|
</div>
|
|
|
|
<!--Options-->
|
|
<transition name="slide-in">
|
|
<div class="input-options" v-if="isOpen">
|
|
<div v-if="options.length > 5" class="select-search">
|
|
<input v-model="query" ref="search" type="text" :placeholder="$t('select_search_placeholder')" class="search-input focus-border-theme">
|
|
</div>
|
|
<ul class="option-list">
|
|
<li class="option-item" @click="selectOption(option)" v-for="(option, i) in optionList" :key="i">
|
|
<div class="option-icon" v-if="option.icon">
|
|
<user-icon v-if="option.icon === 'user'" size="14" />
|
|
<edit2-icon v-if="option.icon === 'user-edit'" size="14" />
|
|
</div>
|
|
<span class="option-value">{{ option.label }}</span>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</transition>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import { ChevronDownIcon, Edit2Icon, UserIcon } from 'vue-feather-icons'
|
|
import {debounce, omitBy} from "lodash"
|
|
|
|
export default {
|
|
name:'SelectInput',
|
|
props: [
|
|
'placeholder',
|
|
'options',
|
|
'isError',
|
|
'default',
|
|
],
|
|
components: {
|
|
Edit2Icon,
|
|
UserIcon,
|
|
ChevronDownIcon
|
|
},
|
|
watch: {
|
|
query: debounce(function (val) {
|
|
this.searchedResults = omitBy(this.options, string => {
|
|
return !string.label.toLowerCase().includes(val.toLowerCase())
|
|
})
|
|
}, 200),
|
|
},
|
|
computed: {
|
|
isSearching() {
|
|
return this.searchedResults && this.query !== ''
|
|
},
|
|
optionList() {
|
|
return this.isSearching
|
|
? this.searchedResults
|
|
: this.options
|
|
}
|
|
},
|
|
data() {
|
|
return {
|
|
searchedResults: undefined,
|
|
selected: undefined,
|
|
isOpen: false,
|
|
query: '',
|
|
}
|
|
},
|
|
methods: {
|
|
selectOption(option) {
|
|
|
|
// Emit selected
|
|
this.$emit('input', option.value)
|
|
|
|
// Get selected
|
|
this.selected = option
|
|
|
|
// Close menu
|
|
this.isOpen = false
|
|
},
|
|
openMenu() {
|
|
this.isOpen = ! this.isOpen
|
|
|
|
if (this.isOpen) {
|
|
this.$nextTick(() => this.$refs.search.focus());
|
|
}
|
|
},
|
|
},
|
|
created() {
|
|
if (this.default)
|
|
this.selected = this.options.find(option => option.value === this.default)
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
@import '@assets/vuefilemanager/_variables';
|
|
@import '@assets/vuefilemanager/_mixins';
|
|
|
|
.select {
|
|
position: relative;
|
|
user-select: none;
|
|
width: 100%;
|
|
}
|
|
|
|
.select-search {
|
|
background: white;
|
|
position: sticky;
|
|
top: 0;
|
|
padding: 13px;
|
|
|
|
.search-input {
|
|
border: 1px solid transparent;
|
|
background: $light_background;
|
|
@include transition(150ms);
|
|
@include font-size(14);
|
|
border-radius: 8px;
|
|
padding: 13px 20px;
|
|
appearance: none;
|
|
font-weight: 700;
|
|
outline: 0;
|
|
width: 100%;
|
|
}
|
|
}
|
|
|
|
.input-options {
|
|
box-shadow: 0 5px 15px rgba(0, 0, 0, 0.12);
|
|
background: white;
|
|
border-radius: 8px;
|
|
position: absolute;
|
|
overflow: hidden;
|
|
top: 65px;
|
|
left: 0;
|
|
right: 0;
|
|
z-index: 9;
|
|
max-height: 295px;
|
|
overflow-y: auto;
|
|
|
|
.option-item {
|
|
padding: 13px 20px;
|
|
display: block;
|
|
cursor: pointer;
|
|
|
|
&:hover {
|
|
color: $theme;
|
|
background: $light_background;
|
|
}
|
|
|
|
&:last-child {
|
|
border-bottom: none;
|
|
}
|
|
}
|
|
}
|
|
|
|
.input-area {
|
|
border-width: 1px;
|
|
border-style: solid;
|
|
border-color: transparent;
|
|
justify-content: space-between;
|
|
box-shadow: 0 1px 5px rgba(0, 0, 0, 0.12);
|
|
//background: $light_mode_input_background;
|
|
@include transition(150ms);
|
|
align-items: center;
|
|
border-radius: 8px;
|
|
padding: 13px 20px;
|
|
display: flex;
|
|
outline: 0;
|
|
width: 100%;
|
|
cursor: pointer;
|
|
|
|
.chevron {
|
|
@include transition(150ms);
|
|
}
|
|
|
|
&.is-active {
|
|
//box-shadow: 0 0 7px rgba($theme, 0.3);
|
|
|
|
.chevron {
|
|
@include transform(rotate(180deg));
|
|
}
|
|
}
|
|
|
|
&.is-error {
|
|
border-color: $danger;
|
|
box-shadow: 0 0 7px rgba($danger, 0.3);
|
|
}
|
|
}
|
|
|
|
.option-icon {
|
|
width: 20px;
|
|
display: inline-block;
|
|
@include font-size(10);
|
|
|
|
svg {
|
|
margin-top: -4px;
|
|
vertical-align: middle;
|
|
}
|
|
}
|
|
|
|
.option-value {
|
|
@include font-size(14);
|
|
font-weight: 700;
|
|
width: 100%;
|
|
vertical-align: middle;
|
|
|
|
&.placehoder {
|
|
color: rgba($text, 0.5);
|
|
}
|
|
}
|
|
|
|
.slide-in-enter-active {
|
|
transition: all 150ms ease;
|
|
}
|
|
|
|
.slide-in-enter /* .list-leave-active below version 2.1.8 */
|
|
{
|
|
opacity: 0;
|
|
transform: translateY(-50px);
|
|
}
|
|
|
|
@media (prefers-color-scheme: dark) {
|
|
|
|
.select-search {
|
|
background: $dark_mode_foreground;
|
|
|
|
.search-input {
|
|
background: $dark_mode_background;
|
|
}
|
|
}
|
|
|
|
.input-area {
|
|
background: $dark_mode_foreground;
|
|
border-color: $dark_mode_foreground;
|
|
}
|
|
|
|
.popup-wrapper {
|
|
.input-area {
|
|
background: lighten($dark_mode_foreground, 3%);
|
|
}
|
|
}
|
|
|
|
.input-options {
|
|
background: $dark_mode_foreground;
|
|
|
|
.option-item {
|
|
border-bottom: none;
|
|
|
|
&:hover {
|
|
background: lighten($dark_mode_foreground, 5%);
|
|
|
|
.option-icon {
|
|
|
|
path, circle {
|
|
color: inherit;
|
|
}
|
|
}
|
|
}
|
|
|
|
&:last-child {
|
|
border-bottom: none;
|
|
}
|
|
}
|
|
}
|
|
|
|
.option-value {
|
|
|
|
&.placehoder {
|
|
color: $dark_mode_text_secondary;
|
|
}
|
|
}
|
|
}
|
|
|
|
</style>
|