73 lines
1.7 KiB
Vue
73 lines
1.7 KiB
Vue
<script>
|
|
import { linkRegex } from '../../utils';
|
|
|
|
import LineNumber from './line_number.vue';
|
|
|
|
export default {
|
|
functional: true,
|
|
props: {
|
|
line: {
|
|
type: Object,
|
|
required: true,
|
|
},
|
|
path: {
|
|
type: String,
|
|
required: true,
|
|
},
|
|
},
|
|
render(h, { props }) {
|
|
const { line, path } = props;
|
|
|
|
let chars;
|
|
if (gon?.features?.ciJobLineLinks) {
|
|
chars = line.content.map(content => {
|
|
return h(
|
|
'span',
|
|
{
|
|
class: ['gl-white-space-pre-wrap', content.style],
|
|
},
|
|
// Simple "tokenization": Split text in chunks of text
|
|
// which alternate between text and urls.
|
|
content.text.split(linkRegex).map(chunk => {
|
|
// Return normal string for non-links
|
|
if (!chunk.match(linkRegex)) {
|
|
return chunk;
|
|
}
|
|
return h(
|
|
'a',
|
|
{
|
|
attrs: {
|
|
href: chunk,
|
|
class: 'gl-reset-color! gl-text-decoration-underline',
|
|
rel: 'nofollow noopener noreferrer', // eslint-disable-line @gitlab/require-i18n-strings
|
|
},
|
|
},
|
|
chunk,
|
|
);
|
|
}),
|
|
);
|
|
});
|
|
} else {
|
|
chars = line.content.map(content => {
|
|
return h(
|
|
'span',
|
|
{
|
|
class: ['gl-white-space-pre-wrap', content.style],
|
|
},
|
|
content.text,
|
|
);
|
|
});
|
|
}
|
|
|
|
return h('div', { class: 'js-line log-line' }, [
|
|
h(LineNumber, {
|
|
props: {
|
|
lineNumber: line.lineNumber,
|
|
path,
|
|
},
|
|
}),
|
|
...chars,
|
|
]);
|
|
},
|
|
};
|
|
</script>
|