pleroma-fe/test/unit/specs/components/rich_content.spec.js

547 lines
13 KiB
JavaScript
Raw Normal View History

2022-03-22 16:40:45 +02:00
import { mount, shallowMount } from '@vue/test-utils'
2026-01-08 17:26:52 +02:00
import RichContent from 'src/components/rich_content/rich_content.jsx'
2021-06-22 21:09:29 +03:00
const attentions = []
2022-03-22 16:40:45 +02:00
const global = {
mocks: {
2022-07-31 12:35:48 +03:00
$store: {
2022-06-13 13:19:54 +03:00
state: {},
getters: {
mergedConfig: () => ({
2026-01-06 16:22:52 +02:00
mentionLinkShowTooltip: true,
2022-06-13 13:19:54 +03:00
}),
2026-01-06 16:22:52 +02:00
findUserByUrl: () => null,
},
},
2022-03-22 16:40:45 +02:00
},
stubs: {
2026-01-06 16:22:52 +02:00
FAIcon: true,
},
2022-03-22 16:40:45 +02:00
}
2022-11-27 00:11:54 +02:00
const makeMention = (who, noClass) => {
2021-06-22 21:09:29 +03:00
attentions.push({ statusnet_profile_url: `https://fake.tld/@${who}` })
2022-11-27 00:11:54 +02:00
return noClass
? `<span><a href="https://fake.tld/@${who}">@<span>${who}</span></a></span>`
: `<span class="h-card"><a class="u-url mention" href="https://fake.tld/@${who}">@<span>${who}</span></a></span>`
2021-06-22 21:09:29 +03:00
}
const p = (...data) => `<p>${data.join('')}</p>`
2026-01-06 16:22:52 +02:00
const compwrap = (...data) =>
`<span class="RichContent">${data.join('')}</span>`
const mentionsLine = (times) =>
[
'<mentions-line-stub mentions="',
new Array(times).fill('[object Object]').join(','),
'"></mentions-line-stub>',
].join('')
describe('RichContent', () => {
it('renders simple post without exploding', () => {
const html = p('Hello world!')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
})
2022-03-22 16:40:45 +02:00
expect(wrapper.html().replace(/\n/g, '')).to.eql(compwrap(html))
})
2021-06-18 21:42:46 +03:00
it('unescapes everything as needed', () => {
2026-01-06 16:22:52 +02:00
const html = [p('Testing &#39;em all'), 'Testing &#39;em all'].join('')
const expected = [p("Testing 'em all"), "Testing 'em all"].join('')
2021-06-18 21:42:46 +03:00
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
2021-06-18 21:42:46 +03:00
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
2021-06-18 21:42:46 +03:00
})
2022-03-22 16:40:45 +02:00
expect(wrapper.html().replace(/\n/g, '')).to.eql(compwrap(expected))
2021-06-18 21:42:46 +03:00
})
it('replaces mention with mentionsline', () => {
2026-01-06 16:22:52 +02:00
const html = p(makeMention('John'), ' how are you doing today?')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
2021-06-14 10:30:08 +03:00
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
2021-06-14 10:30:08 +03:00
})
2026-01-06 16:22:52 +02:00
expect(wrapper.html().replace(/\n/g, '')).to.eql(
compwrap(p(mentionsLine(1), ' how are you doing today?')),
)
2021-06-14 10:30:08 +03:00
})
it('replaces mentions at the end of the hellpost', () => {
2021-06-14 10:30:08 +03:00
const html = [
p('How are you doing today, fine gentlemen?'),
2026-01-06 16:22:52 +02:00
p(makeMention('John'), makeMention('Josh'), makeMention('Jeremy')),
2021-06-14 10:30:08 +03:00
].join('')
const expected = [
2026-01-06 16:22:52 +02:00
p('How are you doing today, fine gentlemen?'),
2021-06-14 10:30:08 +03:00
// TODO fix this extra line somehow?
p(
2022-03-22 16:40:45 +02:00
'<mentions-line-stub mentions="',
2021-06-14 10:30:08 +03:00
'[object Object],',
'[object Object],',
'[object Object]',
2026-01-06 16:22:52 +02:00
'"></mentions-line-stub>',
),
2021-06-14 10:30:08 +03:00
].join('')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
})
2022-03-22 16:40:45 +02:00
expect(wrapper.html().replace(/\n/g, '')).to.eql(compwrap(expected))
})
it('Does not touch links if link handling is disabled', () => {
const html = [
2026-01-06 16:22:52 +02:00
[makeMention('Jack'), "let's meet up with ", makeMention('Janet')].join(
'',
),
[makeMention('John'), makeMention('Josh'), makeMention('Jeremy')].join(
'',
),
].join('\n')
2022-11-27 00:11:54 +02:00
const strippedHtml = [
[
makeMention('Jack', true),
2026-01-06 16:22:52 +02:00
"let's meet up with ",
makeMention('Janet', true),
2022-11-27 00:11:54 +02:00
].join(''),
[
makeMention('John', true),
2026-01-06 16:22:52 +02:00
makeMention('Josh', true),
makeMention('Jeremy', true),
].join(''),
2022-11-27 00:11:54 +02:00
].join('\n')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
handleLinks: false,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
})
2022-11-27 00:11:54 +02:00
expect(wrapper.html()).to.eql(compwrap(strippedHtml))
})
it('Adds greentext and cyantext to the post', () => {
2026-01-06 16:22:52 +02:00
const html = ['&gt;preordering videogames', '&gt;any year'].join('\n')
const expected = [
'<span class="greentext">&gt;preordering videogames</span>',
2026-01-06 16:22:52 +02:00
'<span class="greentext">&gt;any year</span>',
].join('\n')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
handleLinks: false,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
})
expect(wrapper.html()).to.eql(compwrap(expected))
})
it('Does not add greentext and cyantext if setting is set to false', () => {
2026-01-06 16:22:52 +02:00
const html = ['&gt;preordering videogames', '&gt;any year'].join('\n')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
handleLinks: false,
greentext: false,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
})
expect(wrapper.html()).to.eql(compwrap(html))
})
it('Adds emoji to post', () => {
const html = p('Ebin :DDDD :spurdo:')
const expected = p(
'Ebin :DDDD ',
2026-01-06 16:22:52 +02:00
'<anonymous-stub shortcode="spurdo" islocal="true" class="emoji img" src="about:blank" title=":spurdo:" alt=":spurdo:"></anonymous-stub>',
)
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
handleLinks: false,
greentext: false,
emoji: [{ url: 'about:blank', shortcode: 'spurdo' }],
2026-01-06 16:22:52 +02:00
html,
},
})
2022-03-22 16:40:45 +02:00
expect(wrapper.html().replace(/\n/g, '')).to.eql(compwrap(expected))
})
2026-01-06 16:22:52 +02:00
it("Doesn't add nonexistent emoji to post", () => {
const html = p('Lol :lol:')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
handleLinks: false,
greentext: false,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
})
2022-03-22 16:40:45 +02:00
expect(wrapper.html().replace(/\n/g, '')).to.eql(compwrap(html))
})
2021-06-12 20:42:17 +03:00
it('Greentext + last mentions', () => {
const html = [
'&gt;quote',
makeMention('lol'),
'&gt;quote',
2026-01-06 16:22:52 +02:00
'&gt;quote',
2021-06-12 20:42:17 +03:00
].join('\n')
const expected = [
'<span class="greentext">&gt;quote</span>',
mentionsLine(1),
2021-06-12 20:42:17 +03:00
'<span class="greentext">&gt;quote</span>',
2026-01-06 16:22:52 +02:00
'<span class="greentext">&gt;quote</span>',
2021-06-12 20:42:17 +03:00
].join('\n')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
2021-06-12 20:42:17 +03:00
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
2021-06-12 20:42:17 +03:00
})
expect(wrapper.html()).to.eql(compwrap(expected))
})
it('One buggy example', () => {
const html = [
'Bruh',
'Bruh',
2026-01-06 16:22:52 +02:00
[makeMention('foo'), makeMention('bar'), makeMention('baz')].join(''),
'Bruh',
].join('<br>')
2026-01-06 16:22:52 +02:00
const expected = ['Bruh', 'Bruh', mentionsLine(3), 'Bruh'].join('<br>')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
2021-06-13 22:22:59 +03:00
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
2021-06-13 22:22:59 +03:00
})
2022-03-22 16:40:45 +02:00
expect(wrapper.html().replace(/\n/g, '')).to.eql(compwrap(expected))
2021-06-13 22:22:59 +03:00
})
2021-06-15 14:43:44 +03:00
it('buggy example/hashtags', () => {
const html = [
'<p>',
'<a href="http://macrochan.org/images/N/H/NHCMDUXJPPZ6M3Z2CQ6D2EBRSWGE7MZY.jpg">',
'NHCMDUXJPPZ6M3Z2CQ6D2EBRSWGE7MZY.jpg</a>',
' <a class="hashtag" data-tag="nou" href="https://shitposter.club/tag/nou">',
'#nou</a>',
' <a class="hashtag" data-tag="screencap" href="https://shitposter.club/tag/screencap">',
'#screencap</a>',
2026-01-06 16:22:52 +02:00
' </p>',
2021-06-15 14:43:44 +03:00
].join('')
const expected = [
'<p>',
'<a href="http://macrochan.org/images/N/H/NHCMDUXJPPZ6M3Z2CQ6D2EBRSWGE7MZY.jpg" target="_blank">',
'NHCMDUXJPPZ6M3Z2CQ6D2EBRSWGE7MZY.jpg</a>',
2022-03-22 16:40:45 +02:00
' <hashtag-link-stub url="https://shitposter.club/tag/nou" content="#nou" tag="nou">',
'</hashtag-link-stub>',
' <hashtag-link-stub url="https://shitposter.club/tag/screencap" content="#screencap" tag="screencap">',
'</hashtag-link-stub>',
2026-01-06 16:22:52 +02:00
' </p>',
2021-06-15 14:43:44 +03:00
].join('')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
2021-06-15 14:43:44 +03:00
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
2021-06-15 14:43:44 +03:00
})
2022-03-22 16:40:45 +02:00
expect(wrapper.html().replace(/\n/g, '')).to.eql(compwrap(expected))
2021-06-15 14:43:44 +03:00
})
2021-06-16 01:20:20 +03:00
2021-06-16 01:44:29 +03:00
it('rich contents of a mention are handled properly', () => {
2021-06-22 21:09:29 +03:00
attentions.push({ statusnet_profile_url: 'lol' })
2021-06-16 01:44:29 +03:00
const html = [
p(
'<a href="lol" class="mention">',
'<span>',
'https://</span>',
'<span>',
'lol.tld/</span>',
'<span>',
'</span>',
2026-01-06 16:22:52 +02:00
'</a>',
2021-06-16 01:44:29 +03:00
),
2026-01-06 16:22:52 +02:00
p('Testing'),
2021-06-16 01:44:29 +03:00
].join('')
const expected = [
p(
'<span class="MentionsLine">',
'<span class="MentionLink mention-link">',
2022-03-22 16:40:45 +02:00
'<a href="lol" class="original" target="_blank">',
2021-06-16 01:44:29 +03:00
'<span>',
'https://</span>',
'<span>',
'lol.tld/</span>',
'<span>',
'</span>',
'</a>',
'</span>',
2026-01-06 16:22:52 +02:00
'</span>',
2021-06-16 01:44:29 +03:00
),
2026-01-06 16:22:52 +02:00
p('Testing'),
2021-06-16 01:44:29 +03:00
].join('')
const wrapper = mount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
2021-06-16 01:44:29 +03:00
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
2021-06-16 01:44:29 +03:00
})
2026-01-06 16:22:52 +02:00
expect(
wrapper
.html()
.replace(/\n/g, '')
.replace(/<!--.*?-->/g, ''),
).to.eql(compwrap(expected))
2021-06-16 01:44:29 +03:00
})
2022-02-19 23:04:51 +02:00
it('rich contents of nested mentions are handled properly', () => {
attentions.push({ statusnet_profile_url: 'lol' })
const html = [
2022-03-22 16:40:45 +02:00
'<span class="poast-style">',
'<a href="lol" class="mention">',
'<span>',
'https://</span>',
'<span>',
'lol.tld/</span>',
'<span>',
'</span>',
'</a>',
' ',
'<a href="lol" class="mention">',
'<span>',
'https://</span>',
'<span>',
'lol.tld/</span>',
'<span>',
'</span>',
'</a>',
' ',
'</span>',
2026-01-06 16:22:52 +02:00
'Testing',
2022-02-19 23:04:51 +02:00
].join('')
const expected = [
2022-11-27 00:11:54 +02:00
'<span>',
2022-03-22 16:40:45 +02:00
'<span class="MentionsLine">',
'<span class="MentionLink mention-link">',
'<a href="lol" class="original" target="_blank">',
'<span>',
'https://</span>',
'<span>',
'lol.tld/</span>',
'<span>',
'</span>',
'</a>',
'</span>',
'<span class="MentionLink mention-link">',
'<a href="lol" class="original" target="_blank">',
'<span>',
'https://</span>',
'<span>',
'lol.tld/</span>',
'<span>',
'</span>',
'</a>',
'</span>',
'</span>',
2022-02-19 23:04:51 +02:00
' ',
2022-03-22 16:40:45 +02:00
'</span>',
2026-01-06 16:22:52 +02:00
'Testing',
2022-02-19 23:04:51 +02:00
].join('')
const wrapper = mount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2022-02-19 23:04:51 +02:00
attentions,
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
2022-02-19 23:04:51 +02:00
})
2026-01-06 16:22:52 +02:00
expect(
wrapper
.html()
.replace(/\n/g, '')
.replace(/<!--.*?-->/g, ''),
).to.eql(compwrap(expected))
2022-02-19 23:04:51 +02:00
})
2021-06-16 01:44:29 +03:00
it('rich contents of a link are handled properly', () => {
2021-06-16 01:20:20 +03:00
const html = [
'<p>',
'Freenode is dead.</p>',
'<p>',
'<a href="https://isfreenodedeadyet.com/">',
'<span>',
'https://</span>',
'<span>',
'isfreenodedeadyet.com/</span>',
'<span>',
'</span>',
'</a>',
2026-01-06 16:22:52 +02:00
'</p>',
2021-06-16 01:20:20 +03:00
].join('')
const expected = [
'<p>',
'Freenode is dead.</p>',
'<p>',
'<a href="https://isfreenodedeadyet.com/" target="_blank">',
'<span>',
'https://</span>',
'<span>',
'isfreenodedeadyet.com/</span>',
'<span>',
'</span>',
'</a>',
2026-01-06 16:22:52 +02:00
'</p>',
2021-06-16 01:20:20 +03:00
].join('')
const wrapper = shallowMount(RichContent, {
2022-03-22 16:40:45 +02:00
global,
props: {
2021-06-22 21:09:29 +03:00
attentions,
2021-06-16 01:20:20 +03:00
handleLinks: true,
greentext: true,
emoji: [],
2026-01-06 16:22:52 +02:00
html,
},
2021-06-16 01:20:20 +03:00
})
2022-03-22 16:40:45 +02:00
expect(wrapper.html().replace(/\n/g, '')).to.eql(compwrap(expected))
2021-06-16 01:20:20 +03:00
})
it.skip('[INFORMATIVE] Performance testing, 10 000 simple posts', () => {
const amount = 20
const onePost = p(
makeMention('Lain'),
makeMention('Lain'),
makeMention('Lain'),
makeMention('Lain'),
makeMention('Lain'),
makeMention('Lain'),
makeMention('Lain'),
makeMention('Lain'),
makeMention('Lain'),
makeMention('Lain'),
2026-01-06 16:22:52 +02:00
' i just landed in l a where are you',
)
const TestComponent = {
template: `
<div v-if="!vhtml">
${new Array(amount).fill(`<RichContent html="${onePost}" :greentext="true" :handleLinks="handeLinks" :emoji="[]" :attentions="attentions"/>`)}
</div>
<div v-else="vhtml">
${new Array(amount).fill(`<div v-html="${onePost}"/>`)}
</div>
`,
2026-01-06 16:22:52 +02:00
props: ['handleLinks', 'attentions', 'vhtml'],
}
const ptest = (handleLinks, vhtml) => {
const t0 = performance.now()
const wrapper = mount(TestComponent, {
2022-03-22 16:40:45 +02:00
global,
props: {
attentions,
handleLinks,
2026-01-06 16:22:52 +02:00
vhtml,
},
})
const t1 = performance.now()
wrapper.destroy()
const t2 = performance.now()
return `Mount: ${t1 - t0}ms, destroy: ${t2 - t1}ms, avg ${(t1 - t0) / amount}ms - ${(t2 - t1) / amount}ms per item`
}
console.debug(`${amount} items with links handling:`)
console.debug(ptest(true))
console.debug(`${amount} items without links handling:`)
console.debug(ptest(false))
console.debug(`${amount} items plain v-html:`)
console.debug(ptest(false, true))
})
})