119 lines
3.4 KiB
Vue
119 lines
3.4 KiB
Vue
<script>
|
|
import { GlIcon, GlTooltipDirective } from '@gitlab/ui';
|
|
import { __, sprintf } from '~/locale';
|
|
import relatedIssuableMixin from '~/issuable/mixins/related_issuable_mixin';
|
|
|
|
export default {
|
|
name: 'IssueToken',
|
|
components: {
|
|
GlIcon,
|
|
},
|
|
directives: {
|
|
GlTooltip: GlTooltipDirective,
|
|
},
|
|
mixins: [relatedIssuableMixin],
|
|
props: {
|
|
isCondensed: {
|
|
type: Boolean,
|
|
required: false,
|
|
default: false,
|
|
},
|
|
},
|
|
computed: {
|
|
removeButtonLabel() {
|
|
const { displayReference } = this;
|
|
/*
|
|
* Giving false as third argument to prevent unescaping of ampersand in
|
|
* epic reference. Eg. &42 will remain &42 instead of &42
|
|
*
|
|
* https://docs.gitlab.com/ee/development/i18n/externalization.html#interpolation
|
|
*/
|
|
return sprintf(__('Remove %{displayReference}'), { displayReference }, false);
|
|
},
|
|
stateTitle() {
|
|
if (this.isCondensed) return '';
|
|
|
|
return this.isOpen ? __('Open') : __('Closed');
|
|
},
|
|
innerComponentType() {
|
|
return this.isCondensed ? 'span' : 'div';
|
|
},
|
|
issueTitle() {
|
|
return this.isCondensed ? this.title : '';
|
|
},
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div
|
|
:class="{
|
|
'issue-token gl-display-inline-flex gl-align-items-stretch gl-max-w-full gl-line-height-24 gl-white-space-nowrap': isCondensed,
|
|
'flex-row issuable-info-container': !isCondensed,
|
|
}"
|
|
>
|
|
<component
|
|
:is="computedLinkElementType"
|
|
ref="link"
|
|
v-gl-tooltip
|
|
:class="{
|
|
'issue-token-link gl-display-inline-flex gl-min-w-0 gl-text-gray-500': isCondensed,
|
|
'issuable-main-info': !isCondensed,
|
|
}"
|
|
:href="computedPath"
|
|
:title="issueTitle"
|
|
data-placement="top"
|
|
>
|
|
<component
|
|
:is="innerComponentType"
|
|
v-if="hasTitle"
|
|
ref="title"
|
|
:class="{
|
|
'issue-token-title issue-token-end gl-overflow-hidden gl-display-flex gl-align-items-baseline gl-text-gray-500 gl-pl-3': isCondensed,
|
|
'issue-title block-truncated': !isCondensed,
|
|
'gl-rounded-top-right-small gl-rounded-bottom-right-small gl-pr-3': !canRemove,
|
|
}"
|
|
class="js-issue-token-title"
|
|
>
|
|
<span class="gl-text-truncate">{{ title }}</span>
|
|
</component>
|
|
<component
|
|
:is="innerComponentType"
|
|
ref="reference"
|
|
:class="{
|
|
'issue-token-reference gl-display-flex gl-align-items-center gl-rounded-top-left-small gl-rounded-bottom-left-small gl-px-3': isCondensed,
|
|
'issuable-info': !isCondensed,
|
|
}"
|
|
>
|
|
<gl-icon
|
|
v-if="hasState"
|
|
v-gl-tooltip
|
|
:class="iconClass"
|
|
:name="iconName"
|
|
:size="12"
|
|
:title="stateTitle"
|
|
:aria-label="state"
|
|
data-testid="referenceIcon"
|
|
/>
|
|
{{ displayReference }}
|
|
</component>
|
|
</component>
|
|
<button
|
|
v-if="canRemove"
|
|
ref="removeButton"
|
|
v-gl-tooltip
|
|
:class="{
|
|
'issue-token-remove-button gl-display-flex gl-align-items-center gl-px-3 gl-border-0 gl-rounded-top-right-small gl-rounded-bottom-right-small gl-text-gray-500': isCondensed,
|
|
'btn btn-default': !isCondensed,
|
|
}"
|
|
:title="removeButtonLabel"
|
|
:aria-label="removeButtonLabel"
|
|
:disabled="removeDisabled"
|
|
data-testid="removeBtn"
|
|
type="button"
|
|
@click="onRemoveRequest"
|
|
>
|
|
<gl-icon name="close" />
|
|
</button>
|
|
</div>
|
|
</template>
|