Commit e3616635 authored by Nicolò Maria Mezzopera's avatar Nicolò Maria Mezzopera Committed by Paul Slaughter

Add info_line to registry/title_area component

https://gitlab.com/gitlab-org/gitlab/-/merge_requests/42443
parent fa2ee656
<script> <script>
import { GlSprintf, GlLink } from '@gitlab/ui';
import TitleArea from '~/vue_shared/components/registry/title_area.vue'; import TitleArea from '~/vue_shared/components/registry/title_area.vue';
import MetadataItem from '~/vue_shared/components/registry/metadata_item.vue'; import MetadataItem from '~/vue_shared/components/registry/metadata_item.vue';
import { n__, sprintf } from '~/locale'; import { n__, sprintf } from '~/locale';
...@@ -15,8 +14,6 @@ import { ...@@ -15,8 +14,6 @@ import {
export default { export default {
components: { components: {
GlSprintf,
GlLink,
TitleArea, TitleArea,
MetadataItem, MetadataItem,
}, },
...@@ -54,8 +51,6 @@ export default { ...@@ -54,8 +51,6 @@ export default {
}, },
i18n: { i18n: {
CONTAINER_REGISTRY_TITLE, CONTAINER_REGISTRY_TITLE,
LIST_INTRO_TEXT,
EXPIRATION_POLICY_DISABLED_MESSAGE,
}, },
computed: { computed: {
imagesCountText() { imagesCountText() {
...@@ -83,13 +78,21 @@ export default { ...@@ -83,13 +78,21 @@ export default {
!this.expirationPolicyEnabled && this.imagesCount > 0 && !this.hideExpirationPolicyData !this.expirationPolicyEnabled && this.imagesCount > 0 && !this.hideExpirationPolicyData
); );
}, },
infoMessages() {
const base = [{ text: LIST_INTRO_TEXT, link: this.helpPagePath }];
return this.showExpirationPolicyTip
? [
...base,
{ text: EXPIRATION_POLICY_DISABLED_MESSAGE, link: this.expirationPolicyHelpPagePath },
]
: base;
},
}, },
}; };
</script> </script>
<template> <template>
<div> <title-area :title="$options.i18n.CONTAINER_REGISTRY_TITLE" :info-messages="infoMessages">
<title-area :title="$options.i18n.CONTAINER_REGISTRY_TITLE">
<template #right-actions> <template #right-actions>
<slot name="commands"></slot> <slot name="commands"></slot>
</template> </template>
...@@ -111,24 +114,4 @@ export default { ...@@ -111,24 +114,4 @@ export default {
/> />
</template> </template>
</title-area> </title-area>
<div data-testid="info-area">
<p>
<span data-testid="default-intro">
<gl-sprintf :message="$options.i18n.LIST_INTRO_TEXT">
<template #docLink="{content}">
<gl-link :href="helpPagePath" target="_blank">{{ content }}</gl-link>
</template>
</gl-sprintf>
</span>
<span v-if="showExpirationPolicyTip" data-testid="expiration-disabled-message">
<gl-sprintf :message="$options.i18n.EXPIRATION_POLICY_DISABLED_MESSAGE">
<template #docLink="{content}">
<gl-link :href="expirationPolicyHelpPagePath" target="_blank">{{ content }}</gl-link>
</template>
</gl-sprintf>
</span>
</p>
</div>
</div>
</template> </template>
<script> <script>
import { GlAvatar } from '@gitlab/ui'; import { GlAvatar, GlSprintf, GlLink } from '@gitlab/ui';
export default { export default {
name: 'TitleArea', name: 'TitleArea',
components: { components: {
GlAvatar, GlAvatar,
GlSprintf,
GlLink,
}, },
props: { props: {
avatar: { avatar: {
...@@ -17,6 +19,11 @@ export default { ...@@ -17,6 +19,11 @@ export default {
default: null, default: null,
required: false, required: false,
}, },
infoMessages: {
type: Array,
default: () => [],
required: false,
},
}, },
data() { data() {
return { return {
...@@ -30,10 +37,16 @@ export default { ...@@ -30,10 +37,16 @@ export default {
</script> </script>
<template> <template>
<div class="gl-display-flex gl-flex-direction-column">
<div class="gl-display-flex gl-justify-content-space-between gl-py-3"> <div class="gl-display-flex gl-justify-content-space-between gl-py-3">
<div class="gl-flex-direction-column"> <div class="gl-flex-direction-column">
<div class="gl-display-flex"> <div class="gl-display-flex">
<gl-avatar v-if="avatar" :src="avatar" shape="rect" class="gl-align-self-center gl-mr-4" /> <gl-avatar
v-if="avatar"
:src="avatar"
shape="rect"
class="gl-align-self-center gl-mr-4"
/>
<div class="gl-display-flex gl-flex-direction-column"> <div class="gl-display-flex gl-flex-direction-column">
<h1 class="gl-font-size-h1 gl-mt-3 gl-mb-2" data-testid="title"> <h1 class="gl-font-size-h1 gl-mt-3 gl-mb-2" data-testid="title">
...@@ -63,4 +76,19 @@ export default { ...@@ -63,4 +76,19 @@ export default {
<slot name="right-actions"></slot> <slot name="right-actions"></slot>
</div> </div>
</div> </div>
<p>
<span
v-for="(message, index) in infoMessages"
:key="index"
class="gl-mr-2"
data-testid="info-message"
>
<gl-sprintf :message="message.text">
<template #docLink="{content}">
<gl-link :href="message.link" target="_blank">{{ content }}</gl-link>
</template>
</gl-sprintf>
</span>
</p>
</div>
</template> </template>
...@@ -2,9 +2,12 @@ ...@@ -2,9 +2,12 @@
exports[`PackageTitle renders with tags 1`] = ` exports[`PackageTitle renders with tags 1`] = `
<div <div
class="gl-display-flex gl-justify-content-space-between gl-py-3" class="gl-display-flex gl-flex-direction-column"
data-qa-selector="package_title" data-qa-selector="package_title"
> >
<div
class="gl-display-flex gl-justify-content-space-between gl-py-3"
>
<div <div
class="gl-flex-direction-column" class="gl-flex-direction-column"
> >
...@@ -77,14 +80,20 @@ exports[`PackageTitle renders with tags 1`] = ` ...@@ -77,14 +80,20 @@ exports[`PackageTitle renders with tags 1`] = `
</div> </div>
<!----> <!---->
</div>
<p />
</div> </div>
`; `;
exports[`PackageTitle renders without tags 1`] = ` exports[`PackageTitle renders without tags 1`] = `
<div <div
class="gl-display-flex gl-justify-content-space-between gl-py-3" class="gl-display-flex gl-flex-direction-column"
data-qa-selector="package_title" data-qa-selector="package_title"
> >
<div
class="gl-display-flex gl-justify-content-space-between gl-py-3"
>
<div <div
class="gl-flex-direction-column" class="gl-flex-direction-column"
> >
...@@ -148,5 +157,8 @@ exports[`PackageTitle renders without tags 1`] = ` ...@@ -148,5 +157,8 @@ exports[`PackageTitle renders without tags 1`] = `
</div> </div>
<!----> <!---->
</div>
<p />
</div> </div>
`; `;
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import { GlSprintf, GlLink } from '@gitlab/ui'; import { GlSprintf } from '@gitlab/ui';
import Component from '~/registry/explorer/components/list_page/registry_header.vue'; import Component from '~/registry/explorer/components/list_page/registry_header.vue';
import TitleArea from '~/vue_shared/components/registry/title_area.vue'; import TitleArea from '~/vue_shared/components/registry/title_area.vue';
import { import {
...@@ -19,12 +19,8 @@ describe('registry_header', () => { ...@@ -19,12 +19,8 @@ describe('registry_header', () => {
const findTitleArea = () => wrapper.find(TitleArea); const findTitleArea = () => wrapper.find(TitleArea);
const findCommandsSlot = () => wrapper.find('[data-testid="commands-slot"]'); const findCommandsSlot = () => wrapper.find('[data-testid="commands-slot"]');
const findInfoArea = () => wrapper.find('[data-testid="info-area"]');
const findIntroText = () => wrapper.find('[data-testid="default-intro"]');
const findImagesCountSubHeader = () => wrapper.find('[data-testid="images-count"]'); const findImagesCountSubHeader = () => wrapper.find('[data-testid="images-count"]');
const findExpirationPolicySubHeader = () => wrapper.find('[data-testid="expiration-policy"]'); const findExpirationPolicySubHeader = () => wrapper.find('[data-testid="expiration-policy"]');
const findDisabledExpirationPolicyMessage = () =>
wrapper.find('[data-testid="expiration-disabled-message"]');
const mountComponent = (propsData, slots) => { const mountComponent = (propsData, slots) => {
wrapper = shallowMount(Component, { wrapper = shallowMount(Component, {
...@@ -123,44 +119,18 @@ describe('registry_header', () => { ...@@ -123,44 +119,18 @@ describe('registry_header', () => {
}); });
}); });
describe('info area', () => { describe('info messages', () => {
it('exists', () => {
mountComponent();
expect(findInfoArea().exists()).toBe(true);
});
describe('default message', () => { describe('default message', () => {
beforeEach(() => { it('is correctly bound to title_area props', () => {
return mountComponent({ helpPagePath: 'bar' }); mountComponent({ helpPagePath: 'foo' });
});
it('exists', () => { expect(findTitleArea().props('infoMessages')).toEqual([
expect(findIntroText().exists()).toBe(true); { text: LIST_INTRO_TEXT, link: 'foo' },
}); ]);
it('has the correct copy', () => {
expect(findIntroText().text()).toMatchInterpolatedText(LIST_INTRO_TEXT);
});
it('has the correct link', () => {
expect(
findIntroText()
.find(GlLink)
.attributes('href'),
).toBe('bar');
}); });
}); });
describe('expiration policy info message', () => { describe('expiration policy info message', () => {
describe('when there are no images', () => {
it('is hidden', () => {
mountComponent();
expect(findDisabledExpirationPolicyMessage().exists()).toBe(false);
});
});
describe('when there are images', () => { describe('when there are images', () => {
describe('when expiration policy is disabled', () => { describe('when expiration policy is disabled', () => {
beforeEach(() => { beforeEach(() => {
...@@ -170,43 +140,27 @@ describe('registry_header', () => { ...@@ -170,43 +140,27 @@ describe('registry_header', () => {
imagesCount: 1, imagesCount: 1,
}); });
}); });
it('message exist', () => {
expect(findDisabledExpirationPolicyMessage().exists()).toBe(true);
});
it('has the correct copy', () => {
expect(findDisabledExpirationPolicyMessage().text()).toMatchInterpolatedText(
EXPIRATION_POLICY_DISABLED_MESSAGE,
);
});
it('has the correct link', () => { it('the prop is correctly bound', () => {
expect( expect(findTitleArea().props('infoMessages')).toEqual([
findDisabledExpirationPolicyMessage() { text: LIST_INTRO_TEXT, link: '' },
.find(GlLink) { text: EXPIRATION_POLICY_DISABLED_MESSAGE, link: 'foo' },
.attributes('href'), ]);
).toBe('foo');
});
}); });
describe('when expiration policy is enabled', () => {
it('message does not exist', () => {
mountComponent({
expirationPolicy: { enabled: true },
imagesCount: 1,
}); });
expect(findDisabledExpirationPolicyMessage().exists()).toBe(false); describe.each`
}); desc | props
}); ${'when there are no images'} | ${{ expirationPolicy: { enabled: false }, imagesCount: 0 }}
describe('when the expiration policy is completely disabled', () => { ${'when expiration policy is enabled'} | ${{ expirationPolicy: { enabled: true }, imagesCount: 1 }}
${'when the expiration policy is completely disabled'} | ${{ expirationPolicy: { enabled: false }, imagesCount: 1, hideExpirationPolicyData: true }}
`('$desc', ({ props }) => {
it('message does not exist', () => { it('message does not exist', () => {
mountComponent({ mountComponent(props);
expirationPolicy: { enabled: true },
imagesCount: 1,
hideExpirationPolicyData: true,
});
expect(findDisabledExpirationPolicyMessage().exists()).toBe(false); expect(findTitleArea().props('infoMessages')).toEqual([
{ text: LIST_INTRO_TEXT, link: '' },
]);
}); });
}); });
}); });
......
import { GlAvatar } from '@gitlab/ui'; import { GlAvatar, GlSprintf, GlLink } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import component from '~/vue_shared/components/registry/title_area.vue'; import component from '~/vue_shared/components/registry/title_area.vue';
...@@ -10,10 +10,12 @@ describe('title area', () => { ...@@ -10,10 +10,12 @@ describe('title area', () => {
const findMetadataSlot = name => wrapper.find(`[data-testid="${name}"]`); const findMetadataSlot = name => wrapper.find(`[data-testid="${name}"]`);
const findTitle = () => wrapper.find('[data-testid="title"]'); const findTitle = () => wrapper.find('[data-testid="title"]');
const findAvatar = () => wrapper.find(GlAvatar); const findAvatar = () => wrapper.find(GlAvatar);
const findInfoMessages = () => wrapper.findAll('[data-testid="info-message"]');
const mountComponent = ({ propsData = { title: 'foo' }, slots } = {}) => { const mountComponent = ({ propsData = { title: 'foo' }, slots } = {}) => {
wrapper = shallowMount(component, { wrapper = shallowMount(component, {
propsData, propsData,
stubs: { GlSprintf },
slots: { slots: {
'sub-header': '<div data-testid="sub-header" />', 'sub-header': '<div data-testid="sub-header" />',
'right-actions': '<div data-testid="right-actions" />', 'right-actions': '<div data-testid="right-actions" />',
...@@ -95,4 +97,33 @@ describe('title area', () => { ...@@ -95,4 +97,33 @@ describe('title area', () => {
}); });
}); });
}); });
describe('info-messages', () => {
it('shows a message when the props contains one', () => {
mountComponent({ propsData: { infoMessages: [{ text: 'foo foo bar bar' }] } });
const messages = findInfoMessages();
expect(messages).toHaveLength(1);
expect(messages.at(0).text()).toBe('foo foo bar bar');
});
it('shows a link when the props contains one', () => {
mountComponent({
propsData: {
infoMessages: [{ text: 'foo %{docLinkStart}link%{docLinkEnd}', link: 'bar' }],
},
});
const message = findInfoMessages().at(0);
expect(message.find(GlLink).attributes('href')).toBe('bar');
expect(message.text()).toBe('foo link');
});
it('multiple messages generates multiple spans', () => {
mountComponent({ propsData: { infoMessages: [{ text: 'foo' }, { text: 'bar' }] } });
expect(findInfoMessages()).toHaveLength(2);
});
});
}); });
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment