label_token.vue 3.5 KB
Newer Older
1 2 3 4 5
<script>
import {
  GlToken,
  GlFilteredSearchToken,
  GlFilteredSearchSuggestion,
6
  GlNewDropdownDivider as GlDropdownDivider,
7 8 9 10
  GlLoadingIcon,
} from '@gitlab/ui';
import { debounce } from 'lodash';

11
import { deprecatedCreateFlash as createFlash } from '~/flash';
12 13 14 15
import { __ } from '~/locale';

import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';

16
import { stripQuotes } from '../filtered_search_utils';
17
import { DEFAULT_LABELS, DEBOUNCE_DELAY } from '../constants';
18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39

export default {
  components: {
    GlToken,
    GlFilteredSearchToken,
    GlFilteredSearchSuggestion,
    GlDropdownDivider,
    GlLoadingIcon,
  },
  props: {
    config: {
      type: Object,
      required: true,
    },
    value: {
      type: Object,
      required: true,
    },
  },
  data() {
    return {
      labels: this.config.initialLabels || [],
40
      defaultLabels: this.config.defaultLabels || DEFAULT_LABELS,
41 42 43 44 45 46 47 48
      loading: true,
    };
  },
  computed: {
    currentValue() {
      return this.value.data.toLowerCase();
    },
    activeLabel() {
49 50 51
      return this.labels.find(
        label => label.title.toLowerCase() === stripQuotes(this.currentValue),
      );
52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
    },
    containerStyle() {
      if (this.activeLabel) {
        const { color, textColor } = convertObjectPropsToCamelCase(this.activeLabel);

        return { backgroundColor: color, color: textColor };
      }
      return {};
    },
  },
  watch: {
    active: {
      immediate: true,
      handler(newValue) {
        if (!newValue && !this.labels.length) {
          this.fetchLabelBySearchTerm(this.value.data);
        }
      },
    },
  },
  methods: {
    fetchLabelBySearchTerm(searchTerm) {
      this.loading = true;
      this.config
        .fetchLabels(searchTerm)
        .then(res => {
          // We'd want to avoid doing this check but
          // labels.json and /groups/:id/labels & /projects/:id/labels
          // return response differently.
          this.labels = Array.isArray(res) ? res : res.data;
        })
        .catch(() => createFlash(__('There was a problem fetching labels.')))
        .finally(() => {
          this.loading = false;
        });
    },
    searchLabels: debounce(function debouncedSearch({ data }) {
      this.fetchLabelBySearchTerm(data);
    }, DEBOUNCE_DELAY),
  },
};
</script>

<template>
  <gl-filtered-search-token
    :config="config"
    v-bind="{ ...$props, ...$attrs }"
    v-on="$listeners"
    @input="searchLabels"
  >
    <template #view-token="{ inputValue, cssClasses, listeners }">
103 104 105
      <gl-token variant="search-value" :class="cssClasses" :style="containerStyle" v-on="listeners"
        >~{{ activeLabel ? activeLabel.title : inputValue }}</gl-token
      >
106 107
    </template>
    <template #suggestions>
108 109 110 111 112 113 114
      <gl-filtered-search-suggestion
        v-for="label in defaultLabels"
        :key="label.value"
        :value="label.value"
      >
        {{ label.text }}
      </gl-filtered-search-suggestion>
115
      <gl-dropdown-divider v-if="defaultLabels.length" />
116 117 118 119 120 121 122 123 124 125 126 127 128 129 130
      <gl-loading-icon v-if="loading" />
      <template v-else>
        <gl-filtered-search-suggestion v-for="label in labels" :key="label.id" :value="label.title">
          <div class="gl-display-flex">
            <span
              :style="{ backgroundColor: label.color }"
              class="gl-display-inline-block mr-2 p-2"
            ></span>
            <div>{{ label.title }}</div>
          </div>
        </gl-filtered-search-suggestion>
      </template>
    </template>
  </gl-filtered-search-token>
</template>