summaryrefslogtreecommitdiff
path: root/app/assets/javascripts/registry/explorer/components/image_list.vue
blob: bc209b12738490bb187c74c50e3b5dbfd685e960 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
<script>
import { GlPagination, GlTooltipDirective, GlDeprecatedButton, GlIcon } from '@gitlab/ui';
import ClipboardButton from '~/vue_shared/components/clipboard_button.vue';

import {
  ASYNC_DELETE_IMAGE_ERROR_MESSAGE,
  LIST_DELETE_BUTTON_DISABLED,
  REMOVE_REPOSITORY_LABEL,
  ROW_SCHEDULED_FOR_DELETION,
} from '../constants';

export default {
  name: 'ImageList',
  components: {
    GlPagination,
    ClipboardButton,
    GlDeprecatedButton,
    GlIcon,
  },
  directives: {
    GlTooltip: GlTooltipDirective,
  },
  props: {
    images: {
      type: Array,
      required: true,
    },
    pagination: {
      type: Object,
      required: true,
    },
  },
  i18n: {
    LIST_DELETE_BUTTON_DISABLED,
    REMOVE_REPOSITORY_LABEL,
    ROW_SCHEDULED_FOR_DELETION,
    ASYNC_DELETE_IMAGE_ERROR_MESSAGE,
  },
  computed: {
    currentPage: {
      get() {
        return this.pagination.page;
      },
      set(page) {
        this.$emit('pageChange', page);
      },
    },
  },
  methods: {
    encodeListItem(item) {
      const params = JSON.stringify({ name: item.path, tags_path: item.tags_path, id: item.id });
      return window.btoa(params);
    },
  },
};
</script>

<template>
  <div class="gl-display-flex gl-flex-direction-column">
    <div
      v-for="(listItem, index) in images"
      :key="index"
      v-gl-tooltip="{
        placement: 'left',
        disabled: !listItem.deleting,
        title: $options.i18n.ROW_SCHEDULED_FOR_DELETION,
      }"
      data-testid="rowItem"
    >
      <div
        class="gl-display-flex gl-justify-content-space-between gl-align-items-center gl-py-2 gl-px-1 border-bottom"
        :class="{ 'border-top': index === 0, 'disabled-content': listItem.deleting }"
      >
        <div class="gl-display-flex gl-align-items-center">
          <router-link
            data-testid="detailsLink"
            :to="{ name: 'details', params: { id: encodeListItem(listItem) } }"
          >
            {{ listItem.path }}
          </router-link>
          <clipboard-button
            v-if="listItem.location"
            :disabled="listItem.deleting"
            :text="listItem.location"
            :title="listItem.location"
            css-class="btn-default btn-transparent btn-clipboard"
          />
          <gl-icon
            v-if="listItem.failedDelete"
            v-gl-tooltip
            :title="$options.i18n.ASYNC_DELETE_IMAGE_ERROR_MESSAGE"
            name="warning"
            class="text-warning align-middle"
          />
        </div>
        <div
          v-gl-tooltip="{ disabled: listItem.destroy_path }"
          class="d-none d-sm-block"
          :title="$options.i18n.LIST_DELETE_BUTTON_DISABLED"
        >
          <gl-deprecated-button
            v-gl-tooltip
            data-testid="deleteImageButton"
            :disabled="!listItem.destroy_path || listItem.deleting"
            :title="$options.i18n.REMOVE_REPOSITORY_LABEL"
            :aria-label="$options.i18n.REMOVE_REPOSITORY_LABEL"
            class="btn-inverted"
            variant="danger"
            @click="$emit('delete', listItem)"
          >
            <gl-icon name="remove" />
          </gl-deprecated-button>
        </div>
      </div>
    </div>
    <gl-pagination
      v-model="currentPage"
      :per-page="pagination.perPage"
      :total-items="pagination.total"
      align="center"
      class="w-100 gl-mt-2"
    />
  </div>
</template>