debian-mirror-gitlab/app/assets/javascripts/security_configuration/components/feature_card.vue
2022-06-21 17:19:12 +05:30

189 lines
5.4 KiB
Vue

<script>
import { GlButton, GlCard, GlIcon, GlLink } from '@gitlab/ui';
import { __, s__, sprintf } from '~/locale';
import { REPORT_TYPE_SAST_IAC } from '~/vue_shared/security_reports/constants';
import ManageViaMr from '~/vue_shared/security_configuration/components/manage_via_mr.vue';
import FeatureCardBadge from './feature_card_badge.vue';
export default {
components: {
GlButton,
GlCard,
GlIcon,
GlLink,
FeatureCardBadge,
ManageViaMr,
},
props: {
feature: {
type: Object,
required: true,
},
},
computed: {
available() {
return this.feature.available;
},
enabled() {
return this.available && this.feature.configured;
},
shortName() {
return this.feature.shortName ?? this.feature.name;
},
configurationButton() {
const button = this.enabled
? {
text: this.$options.i18n.configureFeature,
}
: {
text: this.$options.i18n.enableFeature,
};
button.category = this.feature.category || 'secondary';
button.text = sprintf(button.text, { feature: this.shortName });
return button;
},
manageViaMrButtonCategory() {
return this.feature.category || 'secondary';
},
showManageViaMr() {
return ManageViaMr.canRender(this.feature);
},
cardClasses() {
return { 'gl-bg-gray-10': !this.available };
},
statusClasses() {
const { enabled, hasBadge } = this;
return {
'gl-ml-auto': true,
'gl-flex-shrink-0': true,
'gl-text-gray-500': !enabled,
'gl-text-green-500': enabled,
'gl-w-full': hasBadge,
'gl-justify-content-space-between': hasBadge,
'gl-display-flex': hasBadge,
'gl-mb-4': hasBadge,
};
},
hasSecondary() {
const { name, description, configurationText } = this.feature.secondary ?? {};
return Boolean(name && description && configurationText);
},
// This condition is a temporary hack to not display any wrong information
// until this BE Bug is fixed: https://gitlab.com/gitlab-org/gitlab/-/issues/350307.
// More Information: https://gitlab.com/gitlab-org/gitlab/-/issues/350307#note_825447417
isNotSastIACTemporaryHack() {
return this.feature.type !== REPORT_TYPE_SAST_IAC;
},
hasBadge() {
return Boolean(this.available && this.feature.badge?.text);
},
},
methods: {
onError(message) {
this.$emit('error', message);
},
},
i18n: {
enabled: s__('SecurityConfiguration|Enabled'),
notEnabled: s__('SecurityConfiguration|Not enabled'),
availableWith: s__('SecurityConfiguration|Available with Ultimate'),
configurationGuide: s__('SecurityConfiguration|Configuration guide'),
configureFeature: s__('SecurityConfiguration|Configure %{feature}'),
enableFeature: s__('SecurityConfiguration|Enable %{feature}'),
learnMore: __('Learn more'),
},
};
</script>
<template>
<gl-card :class="cardClasses">
<div
class="gl-display-flex gl-align-items-baseline"
:class="{ 'gl-flex-direction-column-reverse': hasBadge }"
>
<h3 class="gl-font-lg gl-m-0 gl-mr-3">{{ feature.name }}</h3>
<div
v-if="isNotSastIACTemporaryHack"
:class="statusClasses"
data-testid="feature-status"
:data-qa-selector="`${feature.type}_status`"
>
<feature-card-badge
v-if="hasBadge"
:badge="feature.badge"
:badge-href="feature.badge.badgeHref"
/>
<template v-if="enabled">
<gl-icon name="check-circle-filled" />
<span class="gl-text-green-700">{{ $options.i18n.enabled }}</span>
</template>
<template v-else-if="available">
<span>{{ $options.i18n.notEnabled }}</span>
</template>
<template v-else>
{{ $options.i18n.availableWith }}
</template>
</div>
</div>
<p class="gl-mb-0 gl-mt-5">
{{ feature.description }}
<gl-link :href="feature.helpPath">{{ $options.i18n.learnMore }}</gl-link>
</p>
<template v-if="available && isNotSastIACTemporaryHack">
<gl-button
v-if="feature.configurationPath"
:href="feature.configurationPath"
variant="confirm"
:category="configurationButton.category"
:data-qa-selector="`${feature.type}_enable_button`"
class="gl-mt-5"
>
{{ configurationButton.text }}
</gl-button>
<manage-via-mr
v-else-if="showManageViaMr"
:feature="feature"
variant="confirm"
:category="manageViaMrButtonCategory"
class="gl-mt-5"
:data-qa-selector="`${feature.type}_mr_button`"
@error="onError"
/>
<gl-button
v-else-if="feature.configurationHelpPath"
icon="external-link"
:href="feature.configurationHelpPath"
class="gl-mt-5"
>
{{ $options.i18n.configurationGuide }}
</gl-button>
</template>
<div v-if="hasSecondary" data-testid="secondary-feature">
<h4 class="gl-font-base gl-m-0 gl-mt-6">{{ feature.secondary.name }}</h4>
<p class="gl-mb-0 gl-mt-5">{{ feature.secondary.description }}</p>
<gl-button
v-if="available && feature.secondary.configurationPath"
:href="feature.secondary.configurationPath"
variant="confirm"
category="secondary"
class="gl-mt-5"
>
{{ feature.secondary.configurationText }}
</gl-button>
</div>
</gl-card>
</template>