blob: 8466e5bd461d6736ea0887d62132e97062a4fe30 [file] [log] [blame]
/** * Licensed to the Apache Software Foundation (ASF) under one or more *
contributor license agreements. See the NOTICE file distributed with * this work
for additional information regarding copyright ownership. * The ASF licenses
this file to You under the Apache License, Version 2.0 * (the "License"); you
may not use this file except in compliance with * the License. You may obtain a
copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless
required by applicable law or agreed to in writing, software * distributed under
the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR
CONDITIONS OF ANY KIND, either express or implied. * See the License for the
specific language governing permissions and * limitations under the License. */
<template>
<div
class="rk-bar-select cp flex-h"
v-clickout="
() => {
visible = false;
search = '';
}
"
:class="{ active: visible }"
>
<div class="rk-bar-i flex-h" @click="visible = !visible">
<div class="mr-15 rk-bar-i-text">
<div v-if="Array.isArray(current)">
<span class="selected" v-for="item in current" :key="item.key">
<span>{{ item.label }}</span>
<span
class="remove-icon"
v-if="current.length !== 1"
@click="removeSelected(item)"
>×</span
>
</span>
</div>
<div class="ell" v-else v-tooltip:right.ellipsis="current.label || ''">
{{ current.label }}
</div>
</div>
<svg
class="icon lg trans"
:style="`transform: rotate(${visible ? 180 : 0}deg)`"
>
<use xlink:href="#arrow-down"></use>
</svg>
</div>
<div class="rk-sel" v-if="visible">
<div>
<input type="text" class="rk-sel-search" v-model="search" />
<svg class="icon sm close" @click="search = ''" v-if="search">
<use xlink:href="#clear"></use>
</svg>
</div>
<div class="rk-opt-wrapper scroll_hide">
<div
class="rk-opt ell"
@click="handleSelect(i)"
:class="{ 'select-disabled': selectedOpt.includes(i.key) }"
v-for="i in filterData"
:key="i.key"
>
{{ i.label }}
</div>
</div>
</div>
</div>
</template>
<script lang="ts">
import { Vue, Component, Prop } from 'vue-property-decorator';
const Multiple = 'multiple';
@Component
export default class RkSelect extends Vue {
@Prop() public mode: any;
@Prop() public data!: any;
@Prop() public current!: any;
public search: string = '';
public visible: boolean = false;
get filterData() {
return this.data.filter(
(i: any) =>
i.label.toUpperCase().indexOf(this.search.toUpperCase()) !== -1,
);
}
get selectedOpt() {
return this.mode === Multiple
? this.current.map((item: any) => item.key)
: [this.current.key];
}
public handleOpen() {
this.visible = true;
}
public handleSelect(i: any) {
const selected =
this.mode === Multiple
? this.current.map((item: any) => item.key)
: [this.current.key];
if (selected.includes(i.key)) {
return;
}
this.$emit('onChoose', i);
this.visible = false;
}
private removeSelected(item: any) {
if (this.mode === Multiple) {
this.$emit('onChoose', item);
}
}
}
</script>
<style lang="scss" scoped>
.rk-bar-select {
position: relative;
min-height: 40px;
justify-content: space-between;
border: 1px solid #ddd;
background: #fff;
border-radius: 3px;
.sm {
line-height: 12px;
}
.icon {
flex-shrink: 0;
}
.selected {
display: inline-block;
padding: 5px;
border-radius: 3px;
margin: 3px;
overflow: hidden;
color: rgba(0, 0, 0, 0.65);
background-color: #fafafa;
border: 1px solid #e8e8e8;
}
.remove-icon {
display: inline-block;
margin-left: 5px;
cursor: pointer;
}
}
.rk-bar-i-text {
width: 100%;
}
.rk-bar-i {
height: 100%;
width: 100%;
padding: 5px 15px;
overflow: auto;
}
.rk-sel {
position: absolute;
top: 100%;
background: #fff;
box-shadow: 0 1px 6px rgba(99, 99, 99, 0.2);
border: 1px solid #ddd;
width: 100%;
border-radius: 0 0 3px 3px;
border-right-width: 1px !important;
z-index: 10;
.close {
position: absolute;
right: 10px;
top: 12px;
opacity: 0.6;
&:hover {
opacity: 1;
}
}
}
.rk-opt {
padding: 7px 15px;
&.select-disabled {
color: rgba(0, 0, 0, 0.25);
cursor: not-allowed;
}
&:hover {
background-color: #f5f5f5;
}
}
.rk-sel-search {
width: calc(100% - 4px);
border: 0;
border-bottom: 1px solid #ddd;
outline: 0;
padding: 7px 25px 7px 10px;
margin: 2px;
border-radius: 3px;
}
.rk-opt-wrapper {
overflow: auto;
max-height: 200px;
padding-bottom: 2px;
}
</style>